summaryrefslogtreecommitdiffstats
path: root/examples/animation.html
blob: 3a094994b0f25bb69c86d64dc25b397c0fe6ec76 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
<!DOCTYPE html>
<html>
	<head>
		<title>FastDom: Animation Example</title>
		<script type="text/javascript" src="../fastdom.min.js"></script>
	</head>
	<body>

		<style>

			.mover {
				/* Stolen shamelessly from https://developers.google.com/chrome-developer-tools/docs/demos/too-much-layout/ */
				background: url("");
				height: 100px;
				width: 100px;
				position: absolute;
				/*20% fps boost as described here https://developers.google.com/web/fundamentals/performance/rendering/simplify-paint-complexity-and-reduce-paint-areas/ */
				will-change: transform;
			}

			button.active {
				color: red;
			}

			.intro {
				margin: 10px;
				background: rgb(214, 213, 213);
				border: 1px solid grey;
				padding: 10px;
			}

		</style>

		<div class="intro">An adaptation of the demo from the Google Developers article <a href="https://developers.google.com/chrome-developer-tools/docs/demos/too-much-layout/">Diagnosing forced synchronous layouts</a>. In the article, forced synchronous layouts are fixed by simply <em>not doing any reads at all</em>. Rather than this extreme solution, which will not be appropriate for many use cases, we instead use <strong>fastdom</strong> to intelligently defer and batch the DOM reads and writes, and get similar performance gains as a result.</div>
		<label>Number of elements <input id="count" type="text" value="400" /></label>
		<button id="sync" class="active">Forced synchronous layout</button>
		<button id="async">Run with FastDom</button>
		<button id="noread">No DOM reads</button>
		<button id="toggle">Start</button>

		<div id='test'></div>

		<script>

			var moveMethod = 'sync',
				count      = document.getElementById('count'),
				test       = document.getElementById('test'),
				timestamp, raf, movers;

			var mover = {
				sync: function(m) {

					// Read the top offset, and use that for the left position
					mover.setLeft(movers[m], movers[m].offsetTop);
				},
				async: function(m) {

					// Use fastdom to batch the reads
					// and writes with exactly the same
					// code as the 'sync' routine
					fastdom.measure(function() {
						var top = movers[m].offsetTop;
						fastdom.mutate(function() {
							mover.setLeft(movers[m], top);
						});
					});
				},
				noread: function(m) {

					// Simply use the array index
					// as the top value, so no DOM
					// read is required
					mover.setLeft(movers[m], m);
				},
				setLeft: function(mover, top) {
					mover.style.transform = 'translateX( ' +((Math.sin(top + timestamp/1000) + 1) * 500) + 'px)';
				}
			};

			function update(thisTimestamp) {
				timestamp = thisTimestamp;
				for (var m = 0; m < movers.length; m++) {
					mover[moveMethod](m);
				}
				raf = window.requestAnimationFrame(update);
			}

			function toggleAnim(e) {

				var html, num;

				if (raf) {

					window.cancelAnimationFrame(raf);
					raf = false;
					e.currentTarget.innerHTML = 'Start';
					count.disabled = false;

				} else {

					html = '';
					num = count.value;

					for (i = 0; i < num; i++) {
						html += '<div class="mover"></div>';
					}
					test.innerHTML = html;

					movers = test.querySelectorAll('.mover');
					movers[0].style.top = '150px';
					for (var m = 1; m < movers.length; m++) {
						movers[m].style.top = (m * 20) + 150 + 'px';
					}

					raf = window.requestAnimationFrame(update);
					e.currentTarget.innerHTML = 'Stop';
					count.disabled = true;
				}
			}

			function setMethod(method) {
				document.getElementById(moveMethod).classList.remove('active');
				document.getElementById(method).classList.add('active');
				moveMethod = method;
			}

			document.getElementById('toggle').addEventListener('click', toggleAnim);
			document.getElementById('sync').addEventListener('click', function() {
				setMethod('sync');
			});
			document.getElementById('async').addEventListener('click', function() {
				setMethod('async');
			});
			document.getElementById('noread').addEventListener('click', function() {
				setMethod('noread');
			});

		</script>


	</body>
</html>