1<!DOCTYPE html> 2<html> 3<head> 4 <title>Lottie Filmstrip Capture</title> 5 <meta charset="utf-8" /> 6 <meta http-equiv="X-UA-Compatible" content="IE=edge"> 7 <meta name="viewport" content="width=device-width, initial-scale=1.0"> 8 <script src="/lottie.js" type="text/javascript" charset="utf-8"></script> 9 <style type="text/css" media="screen"> 10 body, 11 main, 12 .anim { 13 margin: 0; 14 padding: 0; 15 } 16 17 main { 18 display: flex; 19 width: 1000px; 20 height: 1000px; 21 flex-flow: row wrap; 22 } 23 </style> 24</head> 25<body> 26 <main> 27 <div class=anim></div> 28 </main> 29 <script type="text/javascript" charset="utf-8"> 30 (function () { 31 const TILE_COUNT = 5; // Number of tiles in x or y direction. 32 const TARGET_SIZE = 1000; // Image size in pixels both x and y direction. 33 const PATH = '/lottie.json'; 34 35 let renderer = 'svg'; 36 let hash = window.location.hash; 37 if (hash) { 38 renderer = hash.slice(1); 39 } 40 41 // This global is used by puppeteer to determine if all tiles have finished drawing. 42 window._tileCount = 0; 43 44 // First load the animation for just a single tile 45 // so we can read out some values and calculate what 46 // the filmstrip should look like. 47 let anim = lottie.loadAnimation({ 48 container: document.querySelector('.anim'), 49 renderer: renderer, 50 loop: false, 51 autoplay: true, 52 path: PATH, 53 rendererSettings: { 54 preserveAspectRatio:'xMidYMid meet' 55 }, 56 }); 57 58 anim.addEventListener('data_ready', (e) => { 59 // Once the first tile is loaded, calculate what 60 // the filmstrip should look like. 61 let animationData = anim.animationData; 62 let totalFrames = anim.totalFrames; 63 // t_rate mimics DMSrcSink.cpp::SkottieSrc::draw 64 let t_rate = 1.0 / (TILE_COUNT * TILE_COUNT - 1); 65 66 let main = document.querySelector('main'); 67 68 // Clear out the first div now that our measurements are done. 69 main.firstElementChild.remove(); 70 71 // Add in all the tiles. 72 for (let i = 0; i < TILE_COUNT*TILE_COUNT; i++) { 73 let div = document.createElement('div'); 74 div.classList.add('anim'); 75 div.style.width = (TARGET_SIZE / TILE_COUNT) + 'px'; 76 div.style.height = (TARGET_SIZE / TILE_COUNT) + 'px'; 77 main.appendChild(div); 78 79 // create a new animation for each tile. It is tempting to try having 80 // one animation and "clone" each frame, but that doesn't work 81 // because of how bodymovin cleans up the URLObjects that are the path 82 // data for the svgs. 83 // We can re-use the animationData to avoid having to hit the 84 // (local) network a bunch of times. 85 let anim = lottie.loadAnimation({ 86 container: div, 87 renderer: renderer, 88 loop: false, 89 autoplay: false, 90 animationData: animationData, 91 rendererSettings: { 92 preserveAspectRatio:'xMidYMid meet' 93 }, 94 }); 95 96 let t = Math.max(Math.min(t_rate * i, 1.0), 0.0); 97 let seekToFrame = totalFrames * t; 98 if (seekToFrame >= totalFrames) { 99 // bodymovin player sometimes draws blank when requesting 100 // to draw the very last frame. Subtracting a small value 101 // seems to fix this and make it draw the last frame. 102 seekToFrame -= .001; 103 } 104 105 // don't need to wait for data_ready because it's instantly ready. 106 console.log(`t = ${t}, go to frame ${seekToFrame}`); 107 anim.goToAndStop(seekToFrame, true); 108 window._tileCount += 1; 109 } 110 }); 111 })(); 112 </script> 113</body> 114</html> 115