mirror of
https://github.com/status-im/react-native.git
synced 2025-01-27 01:40:08 +00:00
1154f2a371
Summary: <img width="601" alt="screen shot 2016-01-27 at 3 37 22 pm" src="https://cloud.githubusercontent.com/assets/197597/12632329/12084e98-c50c-11e5-96c4-4f56e8219b3b.png"> <img width="303" alt="screen shot 2016-01-27 at 3 37 16 pm" src="https://cloud.githubusercontent.com/assets/197597/12632328/12075e70-c50c-11e5-95a3-5f483b4dcb47.png"> <img width="135" alt="screen shot 2016-01-27 at 3 37 38 pm" src="https://cloud.githubusercontent.com/assets/197597/12632327/12067668-c50c-11e5-92f9-ee0c719302b5.png"> Closes https://github.com/facebook/react-native/pull/5587 Reviewed By: svcscm Differential Revision: D2878110 Pulled By: vjeux fb-gh-sync-id: 00330e2857b07aac2e3d3c24f1003cdd9a2516c6
78 lines
3.5 KiB
Markdown
78 lines
3.5 KiB
Markdown
---
|
|
id: timers
|
|
title: Timers
|
|
layout: docs
|
|
category: Polyfills
|
|
permalink: docs/timers.html
|
|
next: colors
|
|
---
|
|
|
|
Timers are an important part of an application and React Native implements the [browser timers](https://developer.mozilla.org/en-US/Add-ons/Code_snippets/Timers).
|
|
|
|
## Timers
|
|
|
|
- setTimeout, clearTimeout
|
|
- setInterval, clearInterval
|
|
- setImmediate, clearImmediate
|
|
- requestAnimationFrame, cancelAnimationFrame
|
|
|
|
`requestAnimationFrame(fn)` is not the same as `setTimeout(fn, 0)` - the former will fire after all the frame has flushed, whereas the latter will fire as quickly as possible (over 1000x per second on a iPhone 5S).
|
|
|
|
`setImmediate` is executed at the end of the current JavaScript execution block, right before sending the batched response back to native. Note that if you call `setImmediate` within a `setImmediate` callback, it will be executed right away, it won't yield back to native in between.
|
|
|
|
The `Promise` implementation uses `setImmediate` as its asynchronicity primitive.
|
|
|
|
|
|
## InteractionManager
|
|
|
|
One reason why well-built native apps feel so smooth is by avoiding expensive operations during interactions and animations. In React Native, we currently have a limitation that there is only a single JS execution thread, but you can use `InteractionManager` to make sure long-running work is scheduled to start after any interactions/animations have completed.
|
|
|
|
Applications can schedule tasks to run after interactions with the following:
|
|
|
|
```javascript
|
|
InteractionManager.runAfterInteractions(() => {
|
|
// ...long-running synchronous task...
|
|
});
|
|
```
|
|
|
|
Compare this to other scheduling alternatives:
|
|
|
|
- requestAnimationFrame(): for code that animates a view over time.
|
|
- setImmediate/setTimeout/setInterval(): run code later, note this may delay animations.
|
|
- runAfterInteractions(): run code later, without delaying active animations.
|
|
|
|
The touch handling system considers one or more active touches to be an 'interaction' and will delay `runAfterInteractions()` callbacks until all touches have ended or been cancelled.
|
|
|
|
InteractionManager also allows applications to register animations by creating an interaction 'handle' on animation start, and clearing it upon completion:
|
|
|
|
```javascript
|
|
var handle = InteractionManager.createInteractionHandle();
|
|
// run animation... (`runAfterInteractions` tasks are queued)
|
|
// later, on animation completion:
|
|
InteractionManager.clearInteractionHandle(handle);
|
|
// queued tasks run if all handles were cleared
|
|
```
|
|
|
|
|
|
## TimerMixin
|
|
|
|
We found out that the primary cause of fatals in apps created with React Native was due to timers firing after a component was unmounted. To solve this recurring issue, we introduced `TimerMixin`. If you include `TimerMixin`, then you can replace your calls to `setTimeout(fn, 500)` with `this.setTimeout(fn, 500)` (just prepend `this.`) and everything will be properly cleaned up for you when the component unmounts.
|
|
|
|
This library does not ship with React Native - in order to use it on your project, you will need to install it with `npm i react-timer-mixin --save` from your project directory.
|
|
|
|
```javascript
|
|
var TimerMixin = require('react-timer-mixin');
|
|
|
|
var Component = React.createClass({
|
|
mixins: [TimerMixin],
|
|
componentDidMount: function() {
|
|
this.setTimeout(
|
|
() => { console.log('I do not leak!'); },
|
|
500
|
|
);
|
|
}
|
|
});
|
|
```
|
|
|
|
We strongly discourage using the global `setTimeout(...)` and recommend instead that you use `this.setTimeout(...)` provided by react-timer-mixin. This will eliminate a lot of hard work tracking down bugs, such as crashes caused by timeouts firing after a component has been unmounted.
|