UNPKG

citong-react-web

Version:

A framework for building web apps with React

612 lines (564 loc) 22 kB
/** * Copyright (c) 2015-present, Alibaba Group Holding Limited. * All rights reserved. * * Copyright (c) 2015, Facebook, Inc. All rights reserved. * * @providesModule ReactListView */ 'use strict'; import React, { Component, PropTypes } from 'react'; import ReactDOM from 'react-dom'; import ListViewDataSource from 'ReactListViewDataSource'; import ScrollView from 'ReactScrollView'; import ScrollResponder from 'ReactScrollResponder'; import StaticRenderer from 'ReactStaticRenderer'; import TimerMixin from 'react-timer-mixin'; import mixin from 'react-mixin'; import assign from 'object-assign'; import autobind from 'autobind-decorator'; const DEFAULT_PAGE_SIZE = 1; const DEFAULT_INITIAL_ROWS = 10; const DEFAULT_SCROLL_RENDER_AHEAD = 1000; const DEFAULT_END_REACHED_THRESHOLD = 1000; const DEFAULT_SCROLL_CALLBACK_THROTTLE = 50; const SCROLLVIEW_REF = 'listviewscroll'; /** * ListView - A core component designed for efficient display of vertically * scrolling lists of changing data. The minimal API is to create a * `ListView.DataSource`, populate it with a simple array of data blobs, and * instantiate a `ListView` component with that data source and a `renderRow` * callback which takes a blob from the data array and returns a renderable * component. * * Minimal example: * * ``` * getInitialState: function() { * let ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2}); * return { * dataSource: ds.cloneWithRows(['row 1', 'row 2']), * }; * }, * * render: function() { * return ( * <ListView * dataSource={this.state.dataSource} * renderRow={(rowData) => <Text>{rowData}</Text>} * /> * ); * }, * ``` * * ListView also supports more advanced features, including sections with sticky * section headers, header and footer support, callbacks on reaching the end of * the available data (`onEndReached`) and on the set of rows that are visible * in the device viewport change (`onChangeVisibleRows`), and several * performance optimizations. * * There are a few performance operations designed to make ListView scroll * smoothly while dynamically loading potentially very large (or conceptually * infinite) data sets: * * * Only re-render changed rows - the rowHasChanged function provided to the * data source tells the ListView if it needs to re-render a row because the * source data has changed - see ListViewDataSource for more details. * * * Rate-limited row rendering - By default, only one row is rendered per * event-loop (customizable with the `pageSize` prop). This breaks up the * work into smaller chunks to reduce the chance of dropping frames while * rendering rows. */ class ListView extends Component { static DataSource = ListViewDataSource /** * You must provide a renderRow function. If you omit any of the other render * functions, ListView will simply skip rendering them. * * - renderRow(rowData, sectionID, rowID, highlightRow); * - renderSectionHeader(sectionData, sectionID); */ static propTypes = { ...ScrollView.propTypes, dataSource: PropTypes.instanceOf(ListViewDataSource).isRequired, /** * (sectionID, rowID, adjacentRowHighlighted) => renderable * * If provided, a renderable component to be rendered as the separator * below each row but not the last row if there is a section header below. * Take a sectionID and rowID of the row above and whether its adjacent row * is highlighted. */ renderSeparator: PropTypes.func, /** * (rowData, sectionID, rowID, highlightRow) => renderable * * Takes a data entry from the data source and its ids and should return * a renderable component to be rendered as the row. By default the data * is exactly what was put into the data source, but it's also possible to * provide custom extractors. ListView can be notified when a row is * being highlighted by calling highlightRow function. The separators above and * below will be hidden when a row is highlighted. The highlighted state of * a row can be reset by calling highlightRow(null). */ renderRow: PropTypes.func.isRequired, /** * How many rows to render on initial component mount. Use this to make * it so that the first screen worth of data appears at one time instead of * over the course of multiple frames. */ initialListSize: PropTypes.number, /** * Called when all rows have been rendered and the list has been scrolled * to within onEndReachedThreshold of the bottom. The native scroll * event is provided. */ onEndReached: PropTypes.func, /** * Threshold in pixels (virtual, not physical) for calling onEndReached. */ onEndReachedThreshold: PropTypes.number, /** * Number of rows to render per event loop. Note: if your 'rows' are actually * cells, i.e. they don't span the full width of your view (as in the * ListViewGridLayoutExample), you should set the pageSize to be a multiple * of the number of cells per row, otherwise you're likely to see gaps at * the edge of the ListView as new pages are loaded. */ pageSize: PropTypes.number, /** * () => renderable * * The header and footer are always rendered (if these props are provided) * on every render pass. If they are expensive to re-render, wrap them * in StaticContainer or other mechanism as appropriate. Footer is always * at the bottom of the list, and header at the top, on every render pass. */ renderFooter: PropTypes.func, renderHeader: PropTypes.func, /** * (sectionData, sectionID) => renderable * * If provided, a sticky header is rendered for this section. The sticky * behavior means that it will scroll with the content at the top of the * section until it reaches the top of the screen, at which point it will * stick to the top until it is pushed off the screen by the next section * header. */ renderSectionHeader: PropTypes.func, /** * (props) => renderable * * A function that returns the scrollable component in which the list rows * are rendered. Defaults to returning a ScrollView with the given props. */ renderScrollComponent: React.PropTypes.func.isRequired, /** * How early to start rendering rows before they come on screen, in * pixels. */ scrollRenderAheadDistance: React.PropTypes.number, /** * (visibleRows, changedRows) => void * * Called when the set of visible rows changes. `visibleRows` maps * { sectionID: { rowID: true }} for all the visible rows, and * `changedRows` maps { sectionID: { rowID: true | false }} for the rows * that have changed their visibility, with true indicating visible, and * false indicating the view has moved out of view. */ onChangeVisibleRows: React.PropTypes.func, /** * A performance optimization for improving scroll perf of * large lists, used in conjunction with overflow: 'hidden' on the row * containers. This is enabled by default. */ removeClippedSubviews: React.PropTypes.bool, /** * An array of child indices determining which children get docked to the * top of the screen when scrolling. For example, passing * `stickyHeaderIndices={[0]}` will cause the first child to be fixed to the * top of the scroll view. This property is not supported in conjunction * with `horizontal={true}`. * @platform ios */ stickyHeaderIndices: PropTypes.arrayOf(PropTypes.number), } /** * React life cycle hooks. */ static defaultProps = { initialListSize: DEFAULT_INITIAL_ROWS, pageSize: DEFAULT_PAGE_SIZE, renderScrollComponent: props => <ScrollView {...props} />, scrollRenderAheadDistance: DEFAULT_SCROLL_RENDER_AHEAD, onEndReachedThreshold: DEFAULT_END_REACHED_THRESHOLD, stickyHeaderIndices: [], } state = { curRenderedRowsCount: this.props.initialListSize, highlightedRow: {}, } /** * Exports some data, e.g. for perf investigations or analytics. */ getMetrics() { return { contentLength: this.scrollProperties.contentLength, totalRows: this.props.dataSource.getRowCount(), renderedRows: this.state.curRenderedRowsCount, visibleRows: Object.keys(this._visibleRows).length, }; } /** * Provides a handle to the underlying scroll responder. * Note that the view in `SCROLLVIEW_REF` may not be a `ScrollView`, so we * need to check that it responds to `getScrollResponder` before calling it. */ getScrollResponder() { return this.refs[SCROLLVIEW_REF] && this.refs[SCROLLVIEW_REF].getScrollResponder && this.refs[SCROLLVIEW_REF].getScrollResponder(); } scrollTo(...args) { this.refs[SCROLLVIEW_REF] && this.refs[SCROLLVIEW_REF].scrollTo && this.refs[SCROLLVIEW_REF].scrollTo(...args); } setNativeProps(props) { this.refs[SCROLLVIEW_REF] && this.refs[SCROLLVIEW_REF].setNativeProps(props); } getInnerViewNode() { return this.refs[SCROLLVIEW_REF].getInnerViewNode(); } componentWillMount() { // this data should never trigger a render pass, so don't put in state this.scrollProperties = { visibleLength: null, contentLength: null, offset: 0 }; this._childFrames = []; this._visibleRows = {}; this._prevRenderedRowsCount = 0; this._sentEndForContentLength = null; } componentDidMount() { // do this in animation frame until componentDidMount actually runs after // the component is laid out this.requestAnimationFrame(() => { this._measureAndUpdateScrollProps(); }); } componentWillReceiveProps(nextProps) { if (this.props.dataSource !== nextProps.dataSource || this.props.initialListSize !== nextProps.initialListSize) { this.setState((state, props) => { this._prevRenderedRowsCount = 0; return { curRenderedRowsCount: Math.min( Math.max( state.curRenderedRowsCount, props.initialListSize ), props.dataSource.getRowCount() ), }; }, () => this._renderMoreRowsIfNeeded()); } } componentDidUpdate() { this.requestAnimationFrame(() => { this._measureAndUpdateScrollProps(); }); } onRowHighlighted(sectionID, rowID) { this.setState({highlightedRow: {sectionID, rowID}}); } render() { let bodyComponents = []; let dataSource = this.props.dataSource; let allRowIDs = dataSource.rowIdentities; let rowCount = 0; let sectionHeaderIndices = []; let header = this.props.renderHeader && this.props.renderHeader(); let footer = this.props.renderFooter && this.props.renderFooter(); let totalIndex = header ? 1 : 0; for (let sectionIdx = 0; sectionIdx < allRowIDs.length; sectionIdx++) { let sectionID = dataSource.sectionIdentities[sectionIdx]; let rowIDs = allRowIDs[sectionIdx]; if (rowIDs.length === 0) { continue; } if (this.props.renderSectionHeader) { let shouldUpdateHeader = rowCount >= this._prevRenderedRowsCount && dataSource.sectionHeaderShouldUpdate(sectionIdx); bodyComponents.push( <StaticRenderer key={'s_' + sectionID} shouldUpdate={!!shouldUpdateHeader} render={this.props.renderSectionHeader.bind( null, dataSource.getSectionHeaderData(sectionIdx), sectionID )} /> ); sectionHeaderIndices.push(totalIndex++); } for (let rowIdx = 0; rowIdx < rowIDs.length; rowIdx++) { let rowID = rowIDs[rowIdx]; let comboID = sectionID + '_' + rowID; let shouldUpdateRow = rowCount >= this._prevRenderedRowsCount && dataSource.rowShouldUpdate(sectionIdx, rowIdx); let row = <StaticRenderer key={'r_' + comboID} shouldUpdate={!!shouldUpdateRow} render={this.props.renderRow.bind( null, dataSource.getRowData(sectionIdx, rowIdx), sectionID, rowID, this.onRowHighlighted )} />; bodyComponents.push(row); totalIndex++; if (this.props.renderSeparator && (rowIdx !== rowIDs.length - 1 || sectionIdx === allRowIDs.length - 1)) { let adjacentRowHighlighted = this.state.highlightedRow.sectionID === sectionID && ( this.state.highlightedRow.rowID === rowID || this.state.highlightedRow.rowID === rowIDs[rowIdx + 1] ); let separator = this.props.renderSeparator( sectionID, rowID, adjacentRowHighlighted ); if (separator) { bodyComponents.push(separator); totalIndex++; } } if (++rowCount === this.state.curRenderedRowsCount) { break; } } if (rowCount >= this.state.curRenderedRowsCount) { break; } } let { renderScrollComponent, ...props, } = this.props; if (!props.scrollEventThrottle) { props.scrollEventThrottle = DEFAULT_SCROLL_CALLBACK_THROTTLE; } if (props.removeClippedSubviews === undefined) { props.removeClippedSubviews = true; } assign(props, { onScroll: this._onScroll, stickyHeaderIndices: this.props.stickyHeaderIndices.concat(sectionHeaderIndices), // Do not pass these events downstream to ScrollView since they will be // registered in ListView's own ScrollResponder.Mixin onKeyboardWillShow: undefined, onKeyboardWillHide: undefined, onKeyboardDidShow: undefined, onKeyboardDidHide: undefined, }); // TODO(ide): Use function refs so we can compose with the scroll // component's original ref instead of clobbering it return React.cloneElement(renderScrollComponent(props), { ref: SCROLLVIEW_REF, onContentSizeChange: this._onContentSizeChange, onLayout: this._onLayout, }, header, bodyComponents, footer); } /** * Private methods */ _measureAndUpdateScrollProps() { let scrollComponent = this.getScrollResponder(); if (!scrollComponent || !scrollComponent.getInnerViewNode) { return; } // RCTScrollViewManager.calculateChildFrames is not available on // every platform // RCTScrollViewManager && RCTScrollViewManager.calculateChildFrames && // RCTScrollViewManager.calculateChildFrames( // React.findNodeHandle(scrollComponent), // this._updateVisibleRows, // ); } _onContentSizeChange(width, height) { let contentLength = !this.props.horizontal ? height : width; if (contentLength !== this.scrollProperties.contentLength) { this.scrollProperties.contentLength = contentLength; this._updateVisibleRows(); this._renderMoreRowsIfNeeded(); } this.props.onContentSizeChange && this.props.onContentSizeChange(width, height); } _onLayout(event) { let {width, height} = event.nativeEvent.layout; let visibleLength = !this.props.horizontal ? height : width; if (visibleLength !== this.scrollProperties.visibleLength) { this.scrollProperties.visibleLength = visibleLength; this._updateVisibleRows(); this._renderMoreRowsIfNeeded(); } this.props.onLayout && this.props.onLayout(event); } _maybeCallOnEndReached(event) { if (this.props.onEndReached && this.scrollProperties.contentLength !== this._sentEndForContentLength && this._getDistanceFromEnd(this.scrollProperties) < this.props.onEndReachedThreshold && this.state.curRenderedRowsCount === this.props.dataSource.getRowCount()) { this._sentEndForContentLength = this.scrollProperties.contentLength; this.props.onEndReached(event); return true; } return false; } _renderMoreRowsIfNeeded() { if (this.scrollProperties.contentLength === null || this.scrollProperties.visibleLength === null || this.state.curRenderedRowsCount === this.props.dataSource.getRowCount()) { this._maybeCallOnEndReached(); return; } let distanceFromEnd = this._getDistanceFromEnd(this.scrollProperties); if (distanceFromEnd < this.props.scrollRenderAheadDistance) { this._pageInNewRows(); } } _pageInNewRows() { this.setState((state, props) => { let rowsToRender = Math.min( state.curRenderedRowsCount + props.pageSize, props.dataSource.getRowCount() ); this._prevRenderedRowsCount = state.curRenderedRowsCount; return { curRenderedRowsCount: rowsToRender }; }, () => { this._measureAndUpdateScrollProps(); this._prevRenderedRowsCount = this.state.curRenderedRowsCount; }); } _getDistanceFromEnd(scrollProperties) { return scrollProperties.contentLength - scrollProperties.visibleLength - scrollProperties.offset; } _updateVisibleRows(updatedFrames) { // if (!this.props.onChangeVisibleRows) { // return; // No need to compute visible rows if there is no callback // } // if (updatedFrames) { // updatedFrames.forEach((newFrame) => { // this._childFrames[newFrame.index] = merge(newFrame); // }); // } // let isVertical = !this.props.horizontal; // let dataSource = this.props.dataSource; // let visibleMin = this.scrollProperties.offset; // let visibleMax = visibleMin + this.scrollProperties.visibleLength; // let allRowIDs = dataSource.rowIdentities; // // let header = this.props.renderHeader && this.props.renderHeader(); // let totalIndex = header ? 1 : 0; // let visibilityChanged = false; // let changedRows = {}; // for (let sectionIdx = 0; sectionIdx < allRowIDs.length; sectionIdx++) { // let rowIDs = allRowIDs[sectionIdx]; // if (rowIDs.length === 0) { // continue; // } // let sectionID = dataSource.sectionIdentities[sectionIdx]; // if (this.props.renderSectionHeader) { // totalIndex++; // } // let visibleSection = this._visibleRows[sectionID]; // if (!visibleSection) { // visibleSection = {}; // } // for (let rowIdx = 0; rowIdx < rowIDs.length; rowIdx++) { // let rowID = rowIDs[rowIdx]; // let frame = this._childFrames[totalIndex]; // totalIndex++; // if (!frame) { // break; // } // let rowVisible = visibleSection[rowID]; // let min = isVertical ? frame.y : frame.x; // let max = min + (isVertical ? frame.height : frame.width); // if (min > visibleMax || max < visibleMin) { // if (rowVisible) { // visibilityChanged = true; // delete visibleSection[rowID]; // if (!changedRows[sectionID]) { // changedRows[sectionID] = {}; // } // changedRows[sectionID][rowID] = false; // } // } else if (!rowVisible) { // visibilityChanged = true; // visibleSection[rowID] = true; // if (!changedRows[sectionID]) { // changedRows[sectionID] = {}; // } // changedRows[sectionID][rowID] = true; // } // } // if (!isEmpty(visibleSection)) { // this._visibleRows[sectionID] = visibleSection; // } else if (this._visibleRows[sectionID]) { // delete this._visibleRows[sectionID]; // } // } // visibilityChanged && this.props.onChangeVisibleRows(this._visibleRows, changedRows); } _onScroll(e) { let isVertical = !this.props.horizontal; // this.scrollProperties.visibleLength = e.nativeEvent.layoutMeasurement[ // isVertical ? 'height' : 'width' // ]; // this.scrollProperties.contentLength = e.nativeEvent.contentSize[ // isVertical ? 'height' : 'width' // ]; // this.scrollProperties.offset = e.nativeEvent.contentOffset[ // isVertical ? 'y' : 'x' // ]; let target = ReactDOM.findDOMNode(this.refs[SCROLLVIEW_REF]); this.scrollProperties.visibleLength = target[ isVertical ? 'offsetHeight' : 'offsetWidth' ]; this.scrollProperties.contentLength = target[ isVertical ? 'scrollHeight' : 'scrollWidth' ]; this.scrollProperties.offset = target[ isVertical ? 'scrollTop' : 'scrollLeft' ]; // this._updateVisibleRows(e.nativeEvent.updatedChildFrames); if (!this._maybeCallOnEndReached(e)) { this._renderMoreRowsIfNeeded(); } if (this.props.onEndReached && this._getDistanceFromEnd(this.scrollProperties) > this.props.onEndReachedThreshold) { // Scrolled out of the end zone, so it should be able to trigger again. this._sentEndForContentLength = null; } this.props.onScroll && this.props.onScroll(e); } } mixin.onClass(ListView, ScrollResponder.Mixin); mixin.onClass(ListView, TimerMixin); autobind(ListView); ListView.isReactNativeComponent = true; export default ListView;