mirror of
https://github.com/zhigang1992/react-native.git
synced 2026-04-24 04:16:00 +08:00
SectionList
Summary: Simple API takes structured `sections` prop instead of `data` array. `sections` is an array of `Section` objects, each of which has a `key` and an `itemData` array which is analogous to a `FlatList` `data` prop, plus optional props like `ItemComponent` that can be overridden on a per-section level, allowing heterogeneous section item rendering via clean composition. Flattens the sections data and renders with VirtualizedList under the hood. Doesn't support sticky headers yet. Reviewed By: yungsters Differential Revision: D4519354 fbshipit-source-id: 58de959dadb6f55f681245ecd99a5dc356a48f36
This commit is contained in:
committed by
Facebook Github Bot
parent
7baecca9a4
commit
73e0b01b06
@@ -94,6 +94,7 @@ class FlatListExample extends React.PureComponent {
|
||||
{renderSmallSwitchOption(this, 'debug')}
|
||||
</View>
|
||||
</View>
|
||||
<SeparatorComponent />
|
||||
<FlatList
|
||||
HeaderComponent={HeaderComponent}
|
||||
FooterComponent={FooterComponent}
|
||||
@@ -164,8 +165,7 @@ const styles = StyleSheet.create({
|
||||
alignItems: 'center',
|
||||
},
|
||||
searchRow: {
|
||||
backgroundColor: '#eeeeee',
|
||||
padding: 10,
|
||||
paddingHorizontal: 10,
|
||||
},
|
||||
});
|
||||
|
||||
|
||||
@@ -85,6 +85,23 @@ class ItemComponent extends React.PureComponent {
|
||||
}
|
||||
}
|
||||
|
||||
class StackedItemComponent extends React.PureComponent {
|
||||
props: {
|
||||
item: Item,
|
||||
};
|
||||
render() {
|
||||
const {item} = this.props;
|
||||
const itemHash = Math.abs(hashCode(item.title));
|
||||
const imgSource = THUMB_URLS[itemHash % THUMB_URLS.length];
|
||||
return (
|
||||
<View style={styles.stacked}>
|
||||
<Text style={styles.stackedText}>{item.title} - {item.text}</Text>
|
||||
<Image style={styles.thumb} source={imgSource} />
|
||||
</View>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FooterComponent extends React.PureComponent {
|
||||
render() {
|
||||
return (
|
||||
@@ -245,10 +262,19 @@ const styles = StyleSheet.create({
|
||||
transform: [{scale: 0.5}],
|
||||
},
|
||||
}),
|
||||
stacked: {
|
||||
alignItems: 'center',
|
||||
backgroundColor: '#F6F6F6',
|
||||
padding: 10,
|
||||
},
|
||||
thumb: {
|
||||
width: 64,
|
||||
height: 64,
|
||||
},
|
||||
stackedText: {
|
||||
padding: 4,
|
||||
fontSize: 18,
|
||||
},
|
||||
text: {
|
||||
flex: 1,
|
||||
},
|
||||
@@ -260,6 +286,7 @@ module.exports = {
|
||||
ItemComponent,
|
||||
PlainInput,
|
||||
SeparatorComponent,
|
||||
StackedItemComponent,
|
||||
genItemData,
|
||||
getItemLayout,
|
||||
pressItem,
|
||||
|
||||
139
Examples/UIExplorer/js/SectionListExample.js
Normal file
139
Examples/UIExplorer/js/SectionListExample.js
Normal file
@@ -0,0 +1,139 @@
|
||||
/**
|
||||
* Copyright (c) 2013-present, Facebook, Inc.
|
||||
* All rights reserved.
|
||||
*
|
||||
* This source code is licensed under the BSD-style license found in the
|
||||
* LICENSE file in the root directory of this source tree. An additional grant
|
||||
* of patent rights can be found in the PATENTS file in the same directory.
|
||||
*
|
||||
* The examples provided by Facebook are for non-commercial testing and
|
||||
* evaluation purposes only.
|
||||
*
|
||||
* Facebook reserves all rights not expressly granted.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||||
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL
|
||||
* FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
||||
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*
|
||||
* @flow
|
||||
*/
|
||||
'use strict';
|
||||
|
||||
const React = require('react');
|
||||
const ReactNative = require('react-native');
|
||||
const {
|
||||
StyleSheet,
|
||||
Text,
|
||||
View,
|
||||
} = ReactNative;
|
||||
|
||||
const SectionList = require('SectionList');
|
||||
const UIExplorerPage = require('./UIExplorerPage');
|
||||
|
||||
const infoLog = require('infoLog');
|
||||
|
||||
const {
|
||||
FooterComponent,
|
||||
ItemComponent,
|
||||
PlainInput,
|
||||
SeparatorComponent,
|
||||
StackedItemComponent,
|
||||
genItemData,
|
||||
pressItem,
|
||||
renderSmallSwitchOption,
|
||||
} = require('./ListExampleShared');
|
||||
|
||||
const SectionHeaderComponent = ({section}) =>
|
||||
<View>
|
||||
<Text style={styles.headerText}>SECTION HEADER: {section.key}</Text>
|
||||
<SeparatorComponent />
|
||||
</View>;
|
||||
|
||||
class SectionListExample extends React.PureComponent {
|
||||
static title = '<SectionList>';
|
||||
static description = 'Performant, scrollable list of data.';
|
||||
|
||||
state = {
|
||||
data: genItemData(1000),
|
||||
filterText: '',
|
||||
logViewable: false,
|
||||
virtualized: true,
|
||||
};
|
||||
render() {
|
||||
const filterRegex = new RegExp(String(this.state.filterText), 'i');
|
||||
const filter = (item) => (filterRegex.test(item.text) || filterRegex.test(item.title));
|
||||
const filteredData = this.state.data.filter(filter);
|
||||
return (
|
||||
<UIExplorerPage
|
||||
noSpacer={true}
|
||||
noScroll={true}>
|
||||
<View style={styles.searchRow}>
|
||||
<PlainInput
|
||||
onChangeText={filterText => {
|
||||
this.setState(() => ({filterText}));
|
||||
}}
|
||||
placeholder="Search..."
|
||||
value={this.state.filterText}
|
||||
/>
|
||||
<View style={styles.optionSection}>
|
||||
{renderSmallSwitchOption(this, 'virtualized')}
|
||||
{renderSmallSwitchOption(this, 'logViewable')}
|
||||
</View>
|
||||
</View>
|
||||
<SeparatorComponent />
|
||||
<SectionList
|
||||
FooterComponent={FooterComponent}
|
||||
ItemComponent={this._renderItemComponent}
|
||||
SectionHeaderComponent={SectionHeaderComponent}
|
||||
SeparatorComponent={SeparatorComponent}
|
||||
enableVirtualization={this.state.virtualized}
|
||||
onRefresh={() => alert('onRefresh: nothing to refresh :P')}
|
||||
onViewableItemsChanged={this._onViewableItemsChanged}
|
||||
refreshing={false}
|
||||
shouldItemUpdate={(prev, next) => prev.item !== next.item}
|
||||
sections={[
|
||||
{ItemComponent: StackedItemComponent, key: 's1', data: [
|
||||
{title: 'Item In Header Section', text: 's1', key: '0'}
|
||||
]},
|
||||
{key: 's2', data: filteredData},
|
||||
]}
|
||||
viewablePercentThreshold={100}
|
||||
/>
|
||||
</UIExplorerPage>
|
||||
);
|
||||
}
|
||||
_renderItemComponent = ({item}) => <ItemComponent item={item} onPress={this._pressItem} />;
|
||||
// This is called when items change viewability by scrolling into our out of the viewable area.
|
||||
_onViewableItemsChanged = (info: {
|
||||
changed: Array<{
|
||||
key: string, isViewable: boolean, item: {columns: Array<*>}, index: ?number, section?: any
|
||||
}>},
|
||||
) => {
|
||||
// Impressions can be logged here
|
||||
if (this.state.logViewable) {
|
||||
infoLog('onViewableItemsChanged: ', info.changed.map((v: Object) => (
|
||||
{...v, item: '...', section: v.section.key}
|
||||
)));
|
||||
}
|
||||
};
|
||||
_pressItem = (index: number) => {
|
||||
pressItem(this, index);
|
||||
};
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
headerText: {
|
||||
padding: 4,
|
||||
},
|
||||
optionSection: {
|
||||
flexDirection: 'row',
|
||||
},
|
||||
searchRow: {
|
||||
paddingHorizontal: 10,
|
||||
},
|
||||
});
|
||||
|
||||
module.exports = SectionListExample;
|
||||
Reference in New Issue
Block a user