react-native/Libraries/Experimental/IncrementalExample.js
Spencer Ahrens f21da3aa31 <Incremental> for incremental rendering
Summary:Everything wrapped in `<Incremental>` is rendered sequentially via `InteractionManager`.
The `onDone` callback is called when all descendent incremental components have
finished rendering, used by `<IncrementalPresenter>` to make the story visible all at once
instead of the parts popping in randomly.

This includes an example that demonstrates streaming rendering and the use of
`<IncrementalPresenter>`.  Pressing down pauses rendering and you can see the
`TouchableOpacity` animation runs smoothly.  Video:

https://youtu.be/4UNf4-8orQ4

Ideally this will be baked into React Core at some point, but according to jordwalke that's
going to require a major refactoring and take a long time, so going with this for now.

Reviewed By: ericvicenti

Differential Revision: D2506522

fb-gh-sync-id: 5969bf248de10d38b0ac22f34d7d49bf1b3ac4b6
shipit-source-id: 5969bf248de10d38b0ac22f34d7d49bf1b3ac4b6
2016-03-10 08:14:23 -08:00

198 lines
5.2 KiB
JavaScript

/**
* 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.
*
* @providesModule IncrementalExample
* @flow
*/
'use strict';
const React = require('react-native');
const {
InteractionManager,
ScrollView,
StyleSheet,
Text,
TouchableOpacity,
View,
} = React;
const Incremental = require('Incremental');
const IncrementalGroup = require('IncrementalGroup');
const IncrementalPresenter = require('IncrementalPresenter');
const JSEventLoopWatchdog = require('JSEventLoopWatchdog');
const StaticContainer = require('StaticContainer.react');
const performanceNow = require('performanceNow');
InteractionManager.setDeadline(1000);
JSEventLoopWatchdog.install({thresholdMS: 200});
const NUM_ITEMS = 20;
let totalWidgets = 0;
class SlowWidget extends React.Component {
state: {ctorTimestamp: number, timeToMount: number};
constructor(props, context) {
super(props, context);
this.state = {
ctorTimestamp: performanceNow(),
timeToMount: 0,
};
}
render() {
this.state.timeToMount === 0 && burnCPU(20);
return (
<View style={styles.widgetContainer}>
<Text style={styles.widgetText}>
{`${this.state.timeToMount || '?'} ms`}
</Text>
</View>
);
}
componentDidMount() {
const timeToMount = performanceNow() - this.state.ctorTimestamp;
this.setState({timeToMount});
totalWidgets++;
}
}
let imHandle;
function startInteraction() {
imHandle = InteractionManager.createInteractionHandle();
}
function stopInteraction() {
InteractionManager.clearInteractionHandle(imHandle);
}
function Block(props: Object) {
const IncrementalContainer = props.stream ? IncrementalGroup : IncrementalPresenter;
return (
<IncrementalContainer name={'b_' + props.idx}>
<TouchableOpacity
onPressIn={startInteraction}
onPressOut={stopInteraction}>
<View style={styles.block}>
<Text>
{props.idx + ': ' + (props.stream ? 'Streaming' : 'Presented')}
</Text>
{props.children}
</View>
</TouchableOpacity>
</IncrementalContainer>
);
}
const Row = (props: Object) => <View style={styles.row} {...props} />;
class IncrementalExample extends React.Component {
static title = '<Incremental*>';
static description = 'Enables incremental rendering of complex components.';
start: number;
state: {stats: ?Object};
constructor(props: mixed, context: mixed) {
super(props, context);
this.start = performanceNow();
this.state = {
stats: null,
};
(this: any)._onDone = this._onDone.bind(this);
}
_onDone() {
const onDoneElapsed = performanceNow() - this.start;
setTimeout(() => {
const stats = {
onDoneElapsed,
totalWidgets,
...JSEventLoopWatchdog.getStats(),
setTimeoutElapsed: performanceNow() - this.start,
};
stats.avgStall = stats.totalStallTime / stats.stallCount;
this.setState({stats});
console.log('onDone:', stats);
}, 0);
}
render(): ReactElement {
return (
<IncrementalGroup
disabled={false}
name="root"
onDone={this._onDone}>
<ScrollView style={styles.scrollView}>
<Text style={styles.headerText}>
Press and hold on a row to pause rendering.
</Text>
{this.state.stats && <Text>
Finished: {JSON.stringify(this.state.stats, null, 2)}
</Text>}
{Array(8).fill().map((_, blockIdx) => {
return (
<Block key={blockIdx} idx={blockIdx} stream={blockIdx < 2}>
{Array(4).fill().map((_b, rowIdx) => (
<Row key={rowIdx}>
{Array(14).fill().map((_c, widgetIdx) => (
<Incremental key={widgetIdx} name={'w_' + widgetIdx}>
<SlowWidget idx={widgetIdx} />
</Incremental>
))}
</Row>
))}
</Block>
);
})}
</ScrollView>
</IncrementalGroup>
);
}
}
function burnCPU(milliseconds) {
const start = performanceNow();
while (performanceNow() < (start + milliseconds)) {}
}
var styles = StyleSheet.create({
scrollView: {
margin: 10,
backgroundColor: 'white',
flex: 1,
},
headerText: {
fontSize: 20,
margin: 10,
},
block: {
borderRadius: 6,
borderWidth: 2,
borderColor: '#a52a2a',
padding: 14,
margin: 5,
backgroundColor: 'white',
},
row: {
flexDirection: 'row',
},
widgetContainer: {
backgroundColor: '#dddddd',
padding: 2,
margin: 2,
},
widgetText: {
color: 'black',
fontSize: 4,
},
});
module.exports = IncrementalExample;