You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							155 lines
						
					
					
						
							4.0 KiB
						
					
					
				
			
		
		
	
	
							155 lines
						
					
					
						
							4.0 KiB
						
					
					
				| # raf-schd
 | |
| 
 | |
| A `throttle` function that uses [`requestAnimationFrame`](https://developer.mozilla.org/en-US/docs/Web/API/window/requestAnimationFrame) to limit the rate at which a function is called.
 | |
| 
 | |
| [](https://travis-ci.org/alexreardon/raf-schd) [](https://david-dm.org/alexreardon/raf-schd) [](https://www.npmjs.com/package/raf-schd) [](http://semver.org/spec/v2.0.0.html)
 | |
| 
 | |
| > For background information on rate limiting functions, see [Rate limiting functions from scratch](https://www.youtube.com/watch?v=g_3e_bNU2e0)
 | |
| 
 | |
| ```js
 | |
| import rafSchd from 'raf-schd';
 | |
| 
 | |
| const expensiveFn = arg => {
 | |
|   //...
 | |
|   console.log(arg);
 | |
| };
 | |
| 
 | |
| const schedule = rafSchd(expensiveFn);
 | |
| 
 | |
| schedule('foo');
 | |
| schedule('bar');
 | |
| schedule('baz');
 | |
| 
 | |
| // animation frame fires
 | |
| 
 | |
| // => 'baz'
 | |
| ```
 | |
| 
 | |
| ## Why?
 | |
| 
 | |
| `raf-schd` supports the use case where you want to limit the rate at which your functions are called based on `requestAnimationFrame`. Unlike a standard `throttle` function, `raf-schd` uses `requestAnimationFrame` to rate limit, rather than waiting a fixed amount of time. Using `requestAnimationFrame` for throttling is powerful because the browser will automatically reduce the amount of frames provided based on the available resources. So if the browser only provides 30fps then your throttled function will only be called 30 times.
 | |
| 
 | |
| `raf-schd` is an **extremely** useful performance utility.
 | |
| 
 | |
| ### Without `raf-schd`
 | |
| 
 | |
| > Optimised scroll listener example taken from [MDN](https://developer.mozilla.org/en-US/docs/Web/Events/scroll)
 | |
| 
 | |
| ```js
 | |
| var last_known_scroll_position = 0;
 | |
| var ticking = false;
 | |
| 
 | |
| function doSomething(scroll_pos) {
 | |
|   // do something with the scroll position
 | |
| }
 | |
| 
 | |
| window.addEventListener('scroll', function(e) {
 | |
|   last_known_scroll_position = window.scrollY;
 | |
|   if (!ticking) {
 | |
|     window.requestAnimationFrame(function() {
 | |
|       doSomething(last_known_scroll_position);
 | |
|       ticking = false;
 | |
|     });
 | |
|   }
 | |
|   ticking = true;
 | |
| });
 | |
| ```
 | |
| 
 | |
| ### With `raf-schd`
 | |
| 
 | |
| ```js
 | |
| import rafSchd from 'raf-schd';
 | |
| 
 | |
| function doSomething(scroll_pos) {
 | |
|   // do something with the scroll position
 | |
| }
 | |
| 
 | |
| const schedule = rafSchd(doSomething);
 | |
| 
 | |
| window.addEventListener('scroll', function() {
 | |
|   schedule(window.scrollY);
 | |
| });
 | |
| ```
 | |
| 
 | |
| At the top level `raf-schd` accepts any function a returns a new `ResultFn` (a function that wraps your original function).
 | |
| 
 | |
| The `ResultFn` will execute your function with the **latest arguments** provided to it on the next animation frame.
 | |
| 
 | |
| ### Throttled with latest argument
 | |
| 
 | |
| ```js
 | |
| import rafSchd from 'raf-schd';
 | |
| 
 | |
| const doSomething = () => {...};
 | |
| 
 | |
| const schedule = rafSchd(doSomething);
 | |
| 
 | |
| schedule(1, 2);
 | |
| schedule(3, 4);
 | |
| schedule(5, 6);
 | |
| 
 | |
| // animation frame fires
 | |
| 
 | |
| // do something called with => '5, 6'
 | |
| ```
 | |
| 
 | |
| ### Cancelling a frame with `.cancel`
 | |
| 
 | |
| `raf-schd` adds a `.cancel` property to the `ResultFn` so that it can be easily cancelled. The frame will only be cancelled if it has not yet executed.
 | |
| 
 | |
| ```js
 | |
| const schedule = rafSchd(doSomething);
 | |
| 
 | |
| schedule('foo');
 | |
| 
 | |
| schedule.cancel();
 | |
| 
 | |
| // now doSomething will not be executed in the next animation frame
 | |
| ```
 | |
| 
 | |
| ## Types
 | |
| 
 | |
| ### `rafSchedule`
 | |
| 
 | |
| ```js
 | |
| type rafSchedule = (fn: Function) => ResultFn;
 | |
| 
 | |
| // Adding a .cancel property to the WrapperFn
 | |
| 
 | |
| type WrapperFn = (...arg: mixed[]) => void;
 | |
| type CancelFn = {|
 | |
|   cancel: () => void,
 | |
| |};
 | |
| type ResultFn = WrapperFn & CancelFn;
 | |
| ```
 | |
| 
 | |
| ## Testing your code
 | |
| 
 | |
| If you want to really ensure that your code is working how you intend it to - use [`raf-stub`](https://github.com/alexreardon/raf-stub) to test your animation frame logic.
 | |
| 
 | |
| ## Installation
 | |
| 
 | |
| ```bash
 | |
| # yarn
 | |
| yarn add raf-schd
 | |
| 
 | |
| # npm
 | |
| npm install raf-schd --save
 | |
| ```
 | |
| 
 | |
| ## Module usage
 | |
| 
 | |
| ### ES6 module
 | |
| 
 | |
| ```js
 | |
| import rafSchd from 'raf-schd';
 | |
| ```
 | |
| 
 | |
| ### CommonJS
 | |
| 
 | |
| If you are in a CommonJS environment (eg [Node](https://nodejs.org)), then **you will need add `.default` to your import**:
 | |
| 
 | |
| ```js
 | |
| const rafSchd = require('raf-schd').default;
 | |
| ```
 |