-
Notifications
You must be signed in to change notification settings - Fork 58
/
Copy pathindex.html
823 lines (502 loc) · 52.4 KB
/
index.html
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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8">
<title>White Paper: HTML5 Banner Ads With CreateJS</title>
<link rel="stylesheet" href="assets/styles.css"/>
<script src="assets/TOC.js"></script>
</head>
<body>
<article>
<h1>HTML5 Banner Ads With CreateJS</h1>
<div id="toc"></div>
<h1>Preamble</h1>
<p>This is a living document. It will be updated as standards and best practices evolve. See <strong>Appendix: Document History</strong> for a list of changes. The source, along with supporting materials, examples, and helper classes is available at <a href="http://github.com/createjs/ads/">github.com/createjs/html5ads/</a>, and contributions are welcome.</p>
<p>The content of this document is copyrighted, and is licensed under the <a href="http://creativecommons.org/licenses/by/4.0/">Creative Commons Attribution license</a>.</p>
<h1>Introduction</h1>
<p>The online advertising industry has been subject to numerous disruptive changes recently. Traditionally, most ads were built with Flash Pro and dependent on the Flash plugin. The rise of mobile browsing on platforms that do not support plugins has provided an increasingly compelling reason to find alternative workflows, but Google's recent move to block Flash ads in Chrome, and Amazon no longer accepting these ads make finding new solutions imperative.</p>
<p>This article explores how the CreateJS suite of libraries allows ad creators to transition easily to HTML5/JS, examines how the libraries fit into emerging IAB (Interactive Advertising Bureau) standards, and provides guidance for creators making the transition. Workflows using the new HTML5 Canvas publish target for Flash Pro will also be discussed, including how it boosts productivity and allows ad creators to leverage prior experience or even convert existing ads.</p>
<p><strong>Appendixes provide more in-depth information on technical topics.</strong></p>
<h1>Authors</h1>
<p>This document was authored by:</p>
<h2>Grant Skinner, CEO gskinner, inc.</h2>
<p>Grant and his team at gskinner conceptualize, design, and build world-class interactive experiences for smart clients such as Microsoft, Google, Mozilla, and EA. He has spoken at hundreds of events on five continents about interactive design and development. He heads the development of CreateJS, and partnered closely with Adobe to lead the development of the HTML5 Canvas publishing for Flash Professional.</p>
<h2>Cory Hudson, Sr. Director, Creative Technology, AOL</h2>
<p>Cory leads a large team of interactive designers and front-end developers who are entirely focused on building feature-rich, cross-screen advertising experiences. His team is directly responsible for developing highly engaging creative solutions and driving technical R&D initiatives in order to effectively evaluate and leverage emerging technologies within AOL's digital advertising formats and platforms. Cory also serves as Chair of the IAB's HTML5 Working Group, where he leads an industry-wide group of experts who are collaboratively working to establish standardized best practices and guidelines for HTML5 based digital advertising.</p>
<h1>Emerging IAB Standards</h1>
<p>The Interactive Advertising Bureau (IAB) recognizes that the industry's long established digital advertising specs tailored to the SWF file format are no longer viable for HTML5-based advertising. With HTML5 there is no longer a convenient dependency upon a single, optimized creative asset package, but rather a compilation of separate assets that must be created, optimized, measured and evaluated individually, while collectively adhering to standardized specs.</p>
<p>This introduces several new obstacles that must be adequately accounted for in order to successfully deliver a compliant advertising experience. The challenges that HTML5 faces when compared to Flash are mainly the required overall file size and number of HTTP server requests associated with an ad unit.</p>
<h2>File Size</h2>
<p>Client bandwidth has increased significantly since the 40kB file size limitation was first introduced. Additionally, the realities of building HTML5-based ad units make it more challenging to adhere to this limit.</p>
<p>Based on these factors, a desire to enable richer ad experiences, and in accordance with the results of recent IAB sponsored industry-wide performance testing, the IAB is proposing an increase in the file size standard for most HTML5 ad formats to 200kB.</p>
<p>Because most ad networks and browsers automatically gzip files crossing the network, file size will be calculated by combining the total gzipped, wire-weights of all utilized assets that are initially required in order for the ad to display in a local, standalone environment. This includes all local images, CSS, JS, HTML, etc.</p>
<h2>Shared Libraries Exception</h2>
<p>The sole exception to the file-size policy is CDN hosted JavaScript libraries that are granted an exception by the publisher or ad server, because these common libraries are leveraged across multiple campaigns and can be cached by users' browsers. A number of ad networks are already supporting this exception for CreateJS (ex. CoFactor/Pointroll, AOL).</p>
<p>CreateJS is on a shortlist of suitable libraries (widely used and distributed, professionally maintained, well documented, frequently updated, backwards compatible) that the IAB will publicly recommend through its soon-to-be-updated and republished HTML5 Best Practices documentation, located at <a href="http://www.iab.net/html5">http://www.iab.net/html5</a>. This would ultimately allow ad creators to exclude the file size of the library from the overall 200kB file size calculation.</p>
<h1>CreateJS</h1>
<p>Developed by widely-recognized interactive shop gskinner to facilitate the creation of Flash-like experiences using open web standards (ex. HTML5), CreateJS is a suite of four libraries that accelerate the production of rich interactive experiences such as games, ads, data visualizations, and micro-sites.</p>
<p>The four libraries in the suite can be used together or completely independently, and are focused on specific areas of functionality.</p>
<h3>EaselJS</h3>
<p>High performance 2D graphics and interaction, with a Flash-inspired display list and feature-rich API.</p>
<h3>SoundJS</h3>
<p>Audio playback and management, with transparent support for WebAudio, Audio elements, and optional Flash plugin fallback.</p>
<h3>PreloadJS</h3>
<p>Robust asset loading and management, with granular progress reports, and nested dependencies.</p>
<h3>TweenJS</h3>
<p>Simple but powerful animation engine with multi-step tweens, synced tweens, non-numeric property support, and motion paths.</p>
<p>CreateJS has no external dependencies, imposes no specific architectural requirements on your code, and is designed to work seamlessly with almost every other JS library, including favorites like GSAP (aka TweenLite), Three.js, AngularJS, and jQuery.</p>
<p>It is free and open source, licensed under the permissive MIT license. This means it can be easily incorporated into commercial projects, and the project welcomes contributions and feedback from the community.</p>
<p>CreateJS is robust and mature. The suite was created and is maintained by a highly experienced team of interactive developers and designers, who have been building rich content for over 15 years using Flash and HTML5. It is over 5 years old, has grown through 8 major releases, and has been sponsored by Adobe, Microsoft, Mozilla, and AOL. CreateJS is used by tens of thousands of developers, with over one third of <a href="http://www.thefwa.com/">FWA Site of the Day</a> winners using it, and content built with CreateJS is viewed by hundreds of millions of users every day.</p>
<h1>Flash Professional: HTML5 Canvas</h1>
<p>Great experiences require both code and design. While it's entirely possible to code amazing content with CreateJS, Flash Pro CC empowers designers to engage fully, and boosts productivity by accelerating the creation of art and animation.</p>
<p>With Flash Pro, you can create an "HTML5 Canvas" document, which will be output as human-readable JavaScript code leveraging the CreateJS libraries when published.</p>
<p>As is the case for traditional Flash development, it is possible to create interactive banners entirely in Flash with timeline scripting. For more complex projects though, you may realize considerable benefits from using Flash Pro to prepare an asset library of art and animation, then write business logic in your code editor of choice that utilizes those assets.</p>
<p>This approach provides a clear delineation of roles and asset ownership between developer and designer. Designers own the FLA, and hand off the published assets to a developer, who can integrate them into a larger project, and manage them via version control. The designer never has to worry about the code, and the developer never has to open the FLA or modify the published assets.</p>
<p><strong>See "Appendix: Designer / Developer Workflow" for more information on this topic.</strong></p>
<h1>Considerations</h1>
<p>While CreateJS and Flash Pro provide a highly productive workflow, and allow you to leverage prior experience with Flash, there are important considerations to keep in mind when getting started.</p>
<h2>Code & API Differences</h2>
<p>CreateJS content is developed using JavaScript, which is very similar to ActionScript, however there are some notable differences that you should be aware of when converting common functionality within Flash Pro CC.</p>
<p>The most immediate difference is scope, which must be defined explicitly. For example, on the timeline, rather than calling <code>stop()</code>, you must use <code>this.stop()</code>. </p>
<p>While the CreateJS API will be very familiar to ActionScript developers, there are significant differences. As examples, only <code>Container</code> and <code>MovieClip</code> can have child objects, and only <code>Shape</code> can draw vector graphics. Timelines are zero based: <code>this.gotoAndStop(0)</code> will take you to the first frame.</p>
<p>More detail on this topic can be found in <strong>"Appendix: Code & API Differences"</strong>.</p>
<h2>Browser Compatibility</h2>
<p>CreateJS is reliably supported by all modern browsers. The only unsupported browser with notable market share is IE8 (<2% market share and dropping) due to its lack of Canvas or Audio support. We recommend providing a fallback static image with a link for unsupported browsers. This can be done with code similar to the following, which prevents the fallback image from loading unless it is needed.</p>
<pre>
<canvas><script>
if (!window.CanvasRenderingContext2D) {
document.write("<a href='url'><img src='image.jpg' alt='text'></a>");
}
</script></canvas>
</pre>
<p>The CreateJS AdHelper class described in <strong>"Appendix: AdHelper"</strong> simplifies setting up fallback content, and makes the code above unnecessary.</p>
<h1>Load Optimization</h1>
<p>There are a number of strategies and techniques you can employ to reduce the size of the assets used in your ad and ensure it adheres to the 200kB limit specified by the IAB. Minimizing the number of HTTP requests required to display your ad is also important to decrease load times.</p>
<h2>Reducing HTTP Requests</h2>
<p>Recent IAB-sponsored performance testing has indicated that the number of HTTP requests can have a significant impact on the perceived loading performance of the visual ad unit, especially on mobile devices. Minimizing the number of requests allows the browser to optimize and parallelize loads, and reduces the overall time cost for establishing http connections.</p>
<p>The emerging HTTP2 standard will mitigate this concern by allowing browsers and servers to transfer multiple assets over a single connection.</p>
<p>Ideally, the number of initial HTTP requests should be kept below 8. This would include all creative assets, CDN hosted and cached support files, and necessary 3rd party impression or click tracking calls.</p>
<p>Using sprite sheets, code minification / concatenation, and data URIs (for embedded web fonts or SVGs) can all help reduce HTTP connections.</p>
<p>CreateJS's PreloadJS library provides capabilities for loading assets, including progress events, maintaining dependencies, loading subsequent assets on interaction, and specifying a maximum number of parallel http connections (via <code>queue.maxConnections</code>). EaselJS supports sprite sheets and audio sprites, and Flash Pro allows for the automatic creation of sprite sheets during publishing.</p>
<h2>Graphics</h2>
<p>The bulk of file size for most ads is in graphical assets. There are a number of techniques you can use to reduce this cost.</p>
<h3>Vector Graphics</h3>
<p>Vector graphics represent art as a collection of resolution-independent drawing commands. They are especially suited to illustrations and other non-photographic art. Unlike bitmap images, they can be scaled indefinitely and retain full fidelity, which makes them ideal for responsive designs targeting multiple display sizes or densities. When used with appropriate content, vectors can also be much smaller than the equivalent bitmap image.</p>
<p>The primary disadvantage to using vectors is that rendering performance scales directly with the complexity (ex. number of curves and points) in the graphic. This is especially true on mobile devices where CPUs are slower and less powerful.</p>
<p>Coding vector graphics manually is possible, but not pragmatic. Luckily, there are several tools that can make working with vector graphics very intuitive. You can create graphics within Flash Pro, or use other software programs such as Adobe Illustrator and then import the result directly into Flash Pro.</p>
<p>See "Bitmap Caching" below for a technique to reduce the ongoing cost of drawing complex vector graphics.</p>
<h3>Bitmaps</h3>
<p>Bitmaps graphics define the color of every pixel in an image. It is very inexpensive for the GPU to manipulate bitmap images, so leveraging bitmaps within your animations rather than vectors can often provide tremendous performance gains.</p>
<p>Bitmaps are ideal for photographic images as well as imagery that requires soft-edges and transparency. Using vectors for these types of graphics would require a gratuitous amount of complexity to represent the image detail, resulting in increased file size as well as visual results that don't appear as natural.</p>
<p>The main disadvantages to using bitmaps is that they come at the expense of increased memory overhead and lack of scalability when compared to vectors. Bitmaps can consume a great deal of both RAM and graphics memory and overuse can cause performance problems on mobile devices. This is especially true with the need to support high DPI screens, which can require you to double the size of images, effectively quadrupling their memory footprint.</p>
<h3>Bitmap Caching</h3>
<p>Similar to Flash Player, EaselJS supports bitmap caching, which allows you to pre-render complex graphics to an off-screen bitmap, so it does not have to be rendered each frame. This can provide significant performance benefits when used appropriately with static art.</p>
<p>The rules for using bitmap caching are similar to using it with Flash Player: use it only on static content, and only when the complexity of the graphics are sufficient to warrant its use. This is because bitmap caching creates new bitmaps, which use both RAM and graphics memory. The latter is limited on mobile devices and overuse can cause performance problems.</p>
<p>Bitmap caching can be applied from within Flash Pro via the "cache as bitmap" option, or with code using the <code>cache()</code> method of display objects.</p>
<h3>High DPI Screen Support </h3>
<p>In order to ensure that bitmaps are displayed crisply on high DPI screens, they should be created at double the display size and then scaled down accordingly within Flash Pro. For example an image that is intended to display at 300x250, should actually be created at 600x500 and then scaled down to 300x250 within Flash Pro. Alternatively, you could author your FLA at double resolution, and scale it to the appropriate size at runtime.</p>
<p>This should be approached in a strategic manner in order to avoid the unnecessary bloating of the overall file size associated with your image assets. You can sometimes avoid having to double the dimensions of photographic images that are not the focal point of the ad experience (e.g. photographic background image).</p>
<p>Conversely, photographic imagery that is the focal point of the ad experience or graphical elements that have sharp, crisp edges such as logos, line artwork and text will most likely always need to be doubled in dimensions because the difference in visual quality is very apparent and noticeable.</p>
<p>Testing your ad on high DPI devices will allow you to determine which graphics will need to be double-sized or not. With experience you will be able to identify these scenarios readily.</p>
<p>The CreateJS AdHelper class described in <strong>"Appendix: AdHelper"</strong> makes it easier to work with high DPI screens, including support for content authored at double scale.</p>
<h3>Compression</h3>
<p>Unfortunately, Flash Pro's default bitmap compression is not very aggressive. You should leverage other image optimization tools such as ImageOptim or ImageAlpha to more fully compress your images. In some cases, these tools can reduce image size by 50-75% without any visible loss in quality. These tools can also be integrated into automated build processes.</p>
<h3>Sprite Sheets</h3>
<p>Sprite sheets combine multiple images or frames of animation onto one larger image, and can dramatically reduce your overall file size and number of server requests.</p>
<p>It is suggested that you combine similar images into sprite sheets, a single JPG sprite sheet for photographic images with opaque backgrounds and another separate PNG sprite sheet for crisp graphics and imagery that requires transparency. There should generally be no need for more than 2 sprite sheets.</p>
<p>You can create sprite sheets manually in Photoshop or using standalone tools such as Zoë or TexturePacker. However the easiest way to create sprite sheets when creating ads with Flash Pro is to use its built-in features:</p>
<ul><li><strong>Generate Sprite Sheet.</strong> Available via the Flash Pro library panel and requires manual placement and masking for each displayed image.</li>
<li><strong>Export All Bitmaps As Sprite Sheets.</strong> Available via the Flash Pro publish settings which automatically handles sprite sheet creation for you however currently outputs a single sprite sheet for all asset types. <strong>"Appendix: Code & API Differences"</strong> has additional information on leveraging this new feature.</li></ul>
<h3>Audio</h3>
<p>When used, audio can also contribute significantly to file size. As with images, you should consider compressing your audio using a dedicated tool that will provide better results than Flash Pro. This is particularly true because as of the time of writing, Flash Pro does not compress audio below 128kbps.</p>
<p>When compressing play, with settings and get a feel for what provides adequate quality for your needs. For example, you might be able to use 48kbps mono for voice or some sound effects, but may not want to drop below 128 kbps for music.</p>
<h3>Code</h3>
<p>Traveling over the wire, your code will be gzipped, which will vastly reduce its file size automatically. However, you can further decrease the file size via minification, and reduce the number of HTTP requests via concatenation.</p>
<p>Concatenation involves combining multiple source files into a single file. Ideally, all of your source files (HTML, CSS, JS) are combined into a single HTML file prior to deployment.</p>
<p>Minification is the process of removing all unnecessary characters from the source code, such as whitespace and comments. More aggressive minification will also refactor your code to use shorter variable names and more compact syntax.</p>
<p>These processes make your code vastly harder to work with, and as such should be used to build a copy of the code for deployment. They are easily integrated into an automated build process using Grunt or Gulp. The functionality of the code should remain intact, but it's important to always test after concatenation and minification.</p>
<h2>Performance</h2>
<p>Building your ads with performance in mind is critical, especially when they will be viewed on mobile devices. Not only do you want your ads to play back smoothly, but you also want to be respectful of users' experience of the site they are visiting, and the battery life of their device.</p>
<p>There are a number of factors that can have a significant impact on the performance of CreateJS content. It's good to be aware of them, and avoid or use them sparingly.</p>
<ul><li>drop shadows and glows can be expensive to render, especially on mobile devices</li>
<li>filters are very performance intensive, and it is recommended to consider alternatives to animating them directly. For example, rather than animate a ColorMatrixFilter to fade a Bitmap from grayscale to color, consider desaturating a copy of the Bitmap by applying the filter once, and fade it over top of the colored version. Similar approaches can be used for blurs and color transforms (which are filters in EaselJS).</li>
<li>complex vector graphics with lots of curves or points (or large numbers of simpler vectors graphics). Consider using bitmap caching to pre-render complex vectors.</li>
<li>large or complex gradients (especially circular gradients).</li>
<li>vector masks are costly in some browsers, consider using <code>compositeOperation</code> in combination with bitmap caching to attain similar results.</li></ul>
<p>You should always profile your content's performance using browser developer tools prior to deployment. Be sure to consider the relative performance of the device you use to profile versus the devices your users may be using. Testing on actual devices is always ideal when possible.</p>
<h3>Performance monitoring</h3>
<p>Some devices are just slow. It doesn't matter how well you author your ad, it will likely never run well on a 5 year old phone intended for a third-world market. Given that, it's a good idea to monitor the performance of your ad, and fall back to alternate content (ex. an image) if it's performing poorly.</p>
<p>The EaselJS Ticker can help with this via its <code>getMeasuredFPS()</code> and <code>getMeasuredTickTime()</code> methods. The former returns the actual framerate that the Ticker is running at, and the latter reports the time actually spent executing each tick. For example, if an ad that was authored to play at 20fps reported 19.8fps, that would mean it's playing right around where we'd expect. However, if it was using 48ms each tick on a desktop PC, that would indicate that it was using almost all of the time available per frame (48ms of 50ms), which could lead to problems when running on less powerful devices.</p>
<p>The CreateJS AdHelper class described in <strong>"Appendix: AdHelper"</strong> provides a configurable method for monitoring performance and showing fallback content or providing actionable events. It also correctly avoids false positives when running in a background tab.</p>
<h2>Duration</h2>
<p>Ads are often required to run for a set amount of time before sleeping (pausing). It's reasonably easy to set a timer to do this in JavaScript, however it's important to consider the elastic nature of frame timing in EaselJS.</p>
<p>When you set a framerate, this doesn't guarantee that your content will play back at that speed. If the device becomes overburdened, frames may take longer to render, slowing the playback of your content. This can cause problems when your ad fails to reach the expected keyframe in the allotted time.</p>
<p>EaselJS allows you to set a target <code>framerate</code> on MovieClip and Sprite instances, placing them in a time-synced mode. If the real-world framerate drops, the MovieClip will adjust its playback rate to accommodate it. For example, if you set a target framerate of 20 on a MovieClip, and the real-world framerate drops to 10fps, it will play 2 frames per second to remain synchronized.</p>
<pre>
myMovieClip.framerate = 20;
</pre>
<p>Note that the timing may still vary slightly, but this is easily resolved by shortening your animation by a small amount. For example, if you have a 15s maximum duration, you may want to author your ad to be 14.8s long to ensure it reaches the last frame before sleeping.</p>
<p>The CreateJS AdHelper class described in <strong>"Appendix: AdHelper"</strong> provides tools for scheduling sleep and wake for ads, as well as propagating a target framerate to all MovieClip instances in the display list.</p>
<h1>Summary</h1>
<p>Authoring ads using CreateJS and Flash Pro allows you to leverage your existing skills and workflows to develop rich interactive content built on open web standards, that can play on any device, and adhere fully to the emerging IAB standards.</p>
<p>There are a number of differences from SWF and AS3 production, and as with any new platform, it will take a bit of adjustment to fully develop your expertise. However, a lot of effort has been put into both CreateJS and Flash Pro's HTML5 Canvas publishing to make it comfortable and familiar to developers coming from a SWF background.</p>
<p>To get started, check out the demos and documentation on the <a href="http://www.createjs.com/">CreateJS site</a> or download the latest source and examples from the <a href="https://github.com/CreateJS">CreateJS GitHub repo</a>. Then, install the latest version of <a href="http://www.adobe.com/products/flash.html">Flash Professional</a>, and check out the <a href="https://github.com/createjs/sandbox">AdHelper example</a>.</p>
<p><hr></p>
<h1>Appendix: Links</h1>
<h3>HTML5 Ads</h3>
<ul><li><a href="http://www.iab.net/html5">http://www.iab.net/html5</a> IAB HTML5 specs</li>
<li><a href="http://github.com/createjs/html5ads/">http://github.com/createjs/html5ads/</a> Repository for this document, supporting materials, and helper classes.</li></ul>
<h3>CreateJS</h3>
<ul><li><a href="http://createjs.com/">http://createjs.com/</a> CreateJS site</li>
<li><a href="http://github.com/createjs/">http://github.com/createjs/</a> CreateJS GitHub rep</li>
<li><a href="http://createjs.com/docs/">http://createjs.com/docs/</a> CreateJS API documentation</li>
<li><a href="http://createjs.com/demos/easeljs/">http://createjs.com/demos/easeljs/</a> CreateJS demos</li></ul>
<p>You can access tutorials and examples for each CreateJS library by cloning its GitHub repository (or simply downloading the latest as a zip file), and looking in the <code>/tutorials</code> and <code>/examples</code> directories.</p>
<h3>Tools</h3>
<ul><li><a href="http://www.adobe.com/products/flash.html">http://www.adobe.com/products/flash.html</a> Flash Professional</li>
<li><a href="http://www.createjs.com/zoe">http://www.createjs.com/zoe</a> Zoë</li>
<li><a href="https://www.codeandweb.com/texturepacker">https://www.codeandweb.com/texturepacker</a> TexturePacker</li>
<li><a href="https://imageoptim.com">https://imageoptim.com</a> ImageOptim</li>
<li><a href="http://pngmini.com">http://pngmini.com</a> ImageAlpha</li></ul>
<h3>JavaScript / Web</h3>
<ul><li><a href="https://developer.mozilla.org/docs/Web/">https://developer.mozilla.org/docs/Web/</a> MDN HTML & JS reference</li>
<li><a href="http://caniuse.com/">http://caniuse.com/</a> Browser feature support reference</li>
<li><a href="https://dev.modern.ie/platform/documentation/f12-devtools-guide/performance/">https://dev.modern.ie/platform/documentation/f12-devtools-guide/performance/</a> Performance profiling in MS Edge</li>
<li><a href="https://developer.mozilla.org/en-US/docs/Tools/Performance">https://developer.mozilla.org/en-US/docs/Tools/Performance</a> Performance profiling in Firefox</li>
<li><a href="https://developer.chrome.com/devtools/docs/timeline">https://developer.chrome.com/devtools/docs/timeline</a> Performance profiling in Chrome</li></ul>
<p><hr></p>
<h1>Appendix: AdHelper</h1>
<p>AdHelper is a helper class that makes it easier to build and deploy ads using CreateJS. It serves a number of functions, and is very easy to set up.</p>
<pre>
var ad = new createjs.AdHelper(myStage);
</pre>
<p>AdHelper can be found in the CreateJS Sandbox repository on GitHub:</p>
<p><a href="http://github.com/createjs/sandbox/">http://github.com/createjs/sandbox/</a></p>
<h2>Method Chaining</h2>
<p>Most methods can be chained for simpler setup. For example, the following code creates a new AdHelper instance and enables time sync, performance monitoring, and high dpi support using default settings.</p>
<pre>
new createjs.AdHelper(myStage).timeSync().watchFPS().highDPI();
</pre>
<h2>Sleep / Wake</h2>
<p>Puts the ad to sleep according to a configurable schedule, and wakes it for the specified amount of time when the mouse is within the ad, or the user clicks in the ad. When an ad is put to sleep, its Ticker is paused, its tick listener is unsubscribed, any GreenSock tweens are paused, and a <code>sleep</code> event is dispatched to allow custom handling. On wake, this is reversed, and a <code>wake</code> event is generated.</p>
<pre>
ad.setSleep(start, mousedown, inBounds, useTicks, listener);
</pre>
<p>Ads can also be put to sleep directly via the "sleep" and wake" methods:</p>
<pre>
ad.wake(time);
ad.sleep(delay);
</pre>
<h2>Visibility</h2>
<p>Detects when the ad is not visible (ex. in a background browser tab), and sleeps the ad and mutes all sound. Resumes / unmutes the ad when it is visible. The "hidden" property is set accordingly and the <code>sleep</code> and <code>wake</code> events are dispatched as normal.</p>
<p>This feature is enabled automatically when using AdHelper.</p>
<h2>CreateJS Support</h2>
<p>Test to see if CreateJS is supported on the current browser. For example, this can be used to display alternative content.</p>
<pre>
if (!createjs.AdHelper.isSupported()) {
createjs.AdHelper.showAltImage(myCanvas, "myImage.jpg", "url");
}
</pre>
<h2>Alternative Content</h2>
<p>Static methods to replace your ad canvas with alternative content. This can be useful when CreateJS is not supported, or if performance monitoring determines the ad is running too slowly. There are three methods, that give different levels of control.</p>
<pre>
createjs.AdHelper.showAltImage(canvas, imgSrc, href, alt, target);
createjs.AdHelper.showAltHTML(canvas, html);
createjs.AdHelper.showAlt(canvas, element);
</pre>
<h2>Time Sync</h2>
<p>Force all MovieClips in the ad to operate in time-synced mode (ie. by setting <code>framerate</code> on all MovieClip instances). This allows the ad to play back with a predictable duration even if the real framerate fluctuates.</p>
<pre>
ad.timeSync(framerate);
</pre>
<h2>High DPI Support</h2>
<p>Enable full resolution rendering on high DPI screens. Can also be used to disable full resolution rendering on high DPI screens to improve performance. Also allows for authoring the ad at a different scale.</p>
<pre>
ad.highDPI(enabled, nominalScale);
</pre>
<h2>Performance Monitoring</h2>
<p>Monitors playback framerate, and uses a simple heuristic to determine when the ad fails to satisfy configurable performance minimums. Dispatches a <code>slow</code> event and sleeps the ad by default. Works with visibility features to avoid failing while in a background tab due to throttling.</p>
<pre>
ad.watchFPS(minFPS, tolerance);
</pre>
<p><hr></p>
<h1>Appendix: Designer / Developer Workflow</h1>
<p>As with traditional SWF development, there are many valid workflows for developing HTML5 Canvas content using Flash Pro.</p>
<p>For simpler projects owned by a single designer, building everything on the timeline, including simple frame scripts is often perfectly acceptable.</p>
<p>As projects grow in complexity, it is generally wise to create a separation between the creation of visual assets (presentation), and the code that drives functionality (logic). In many cases these areas are divided between a designer(s) and developer(s) respectively. Even when a single creator is producing the full project there are usually significant benefits to productivity, quality control, reuse, and robustness that can be realized by separating presentation and logic.</p>
<p>One common approach to this separation when building CreateJS content is to have the FLA owned entirely by the designer, who passes the published JS library to the developer (via email, version control, etc), who consumes it into a larger project with additional HTML and JS files. It winds up looking something like this:</p>
<img src="assets/workflow.png">
<p>Using this approach, the developer should almost never need to touch the published JS. They can simply treat it as an asset library, instantiating elements from it using <code>new</code>. For example, if the FLA has a symbol named (or with the linkage) "FunnyCat", a developer could add an instance to the stage, position it, and play its "run" animation using:</p>
<pre>
var myCat = new lib.FunnyCat();
stage.addChild(myCat);
myCat.x = 100;
myCat.gotoAndPlay("run");
</pre>
<p>The JS file also includes a <code>properties</code> object that includes information on the original FLA (dimensions, background color, framerate), and a manifest of external media assets that can be passed directly to a PreloadJS <code>LoadQueue</code>.</p>
<pre>
var props = lib.properties;
createjs.Ticker.setFPS(props.fps);
var queue = new createjs.LoadQueue();
queue.on("complete", handleComplete, this);
queue.loadManifest(props.manifest);
</pre>
<p>It's often critical for a project's business logic to know when an animation finishes or reaches a particular point. For example, you may want to change the position of the cat in the example above each time it finishes the "run" animation. This is easily achieved. In addition to the presentation code to loop the animation, the designer could also add an event trigger at the end of the animation:</p>
<pre>
// timeline code on the last frame of the run animation:
this.gotoAndPlay("run"); // loop
this.dispatchEvent("runend"); // the name can be anything
</pre>
<p>Once this is in place, the developer can easily subscribe and react to this event:</p>
<pre>
myCat.on("runend", function(evt) {
// with on() "this" defaults to the dispatcher.
this.x += 100;
});
</pre>
<p>This supports a loosely coupled interaction between the timeline and the business logic. The designer only needs to worry about a single simple command to communicate events, and can move the event freely as the animation is revised. The developer can subscribe to events as needed. Events dispatched without subscribers do not generate errors, and have very low cost (no event object is created).</p>
<p><hr></p>
<h1>Appendix: Code & API Differences</h1>
<p>The following are common tasks that you are likely to encounter when building HTML5 ads with Flash Pro.</p>
<h2>Text </h2>
<p>Flash Pro currently only supports the usage of dynamic text fields with a single style which can be very limiting when it comes to controlling the appearance and presentation of type. When Flash Pro converts existing static text fields to dynamic text fields the text is converted to a single style and overrides any specific styling that was previously applied, such as leading, weight, etc., and also visually removes line-wrapping even though the hidden text actually exists and will properly wrap to a new line when published. </p>
<p>Most advertisers have very specific requirements when it comes to font styling and branding, so these limitations can be challenging and make the design process unnecessarily cumbersome when working within the Flash Pro Canvas project environment. In order to maintain the visual integrity of your styled text you may want to take one of the following approaches:</p>
<ul><li>Start your banner ad project as a Flash ActionScript 3 document type (not an HTML5 Canvas FLA) and then break apart all static text before converting the file over to HTML5 Canvas. This will convert all text into vector shapes, maintaining both appearance and scalability.</li>
<li>Maintain a separate AS3 FLA for the styling of all text and then paste the text into your working HTML5 Canvas document after it has been converted to vector outlines.</li>
<li>Design all of your text in Photoshop or Flash Pro and then integrate the text as bitmaps. This approach is not recommended as it removes the benefits of vector based text which would be small file size and scalability.</li></ul>
<p>Regardless of which approach you prefer it is suggested that you adhere to the following best practices when working with vector text inside of Flash Pro CC: </p>
<p> </p>
<ul><li>Create a movieclip of your broken apart text and set it to cache as bitmap. This will improve performance as the vector drawing calculations won't have to be repeatedly processed on every stage tick.</li>
<li>Be sure to maintain an editable version of your text by creating a "guided out" backup layer containing a static text field. This ensures that edits can be easily made in the future. </li></ul>
<p> </p>
<h2>Filters and Color Effects</h2>
<p>Flash Pro CC supports most filters and color effects. These effects fall into one of two categories: shadow effects, and filter effects. All effects are expensive to render, particularly on mobile devices, and should be tested thoroughly.</p>
<h3>Shadow Effects</h3>
<p>Drop shadows and glows (which are essentially shadows with no offset) are rendered in EaselJS using canvas's built in shadow features. They render reasonably fast on desktop, but can cause performance issues on mobile, and can be applied to animated content.</p>
<p>At this time, shadow effects cannot be animated on the timeline, though this is primarily an authoring limitation that may be removed in the future.</p>
<h3>Filter Effects</h3>
<p>Blurs, color transforms (excluding alpha), and color adjustments are treated as filters in EaselJS. These effects work by manipulating pixel data, can be quite expensive to render, and require bitmap caching to be enabled to work (this is done automatically when you apply these filters in Flash Pro).</p>
<p>This bitmap caching "freezes" the movieclip on its first frame, which stops it from animating. Similarly, the filter cannot be animated on the timeline. This is an intentional limitation, because updating filters is very expensive, and it would be very easy to unintentionally create content with very poor performance.</p>
<p>It is possible to animate content with a filter by re-rendering the filter each tick (or frame) via <code>myDisplayObject.updateCache()</code>, but in almost all cases it is better to consider alternative approaches. For example, to transition an image from grayscale to color, you could simply duplicate the image, apply a grayscale color adjustment to the top copy, and fade down its alpha over the color copy. Similar approaches can be used for blur and color transforms.</p>
<p>Another option is to create a sprite sheet animation of the filter effect and use that in your ad instead.</p>
<h2>Logging </h2>
<p>It's likely that you've previously used ActionScript's <code>trace()</code>statement to debug your code. In JavaScript, you can use <code>console.log()</code> instead:</p>
<p> </p>
<pre>
console.log("This is the same as a trace statement." );
</pre>
<p>To view <code>console.log()</code>statements when previewing your HTML file, you will need to open up the JavaScript Console in Chrome Dev Tools, or the Console tab in Firebug if you are testing using Firefox. Be aware that in IE9 the console must be open to function correctly or it will generate errors. Make sure you remove any <code>console.log()</code> calls prior to deployment.</p>
<p> </p>
<h2>Scope </h2>
<p>JavaScript does not use <code> this </code> as an implicit scope as is the case with ActionScript 3. You must explicitly specify scope in timeline scripts. For example:</p>
<p> </p>
<pre>
// "this" in a timeline script refers to the MovieClip or stage that it's defined in.
this.stop();
// access a child MovieClip instance:
this.myChildMC.gotoAndPlay("end");
</pre>
<p> </p>
<h3><strong>Defining Global Variables and Functions </strong> </h3>
<p>Variables are defined with the scope of their frame code, so if you define a variable in the first frame, you will not be able to access that variable in the final frame. For example:</p>
<pre>
// first frame
var foo = 20;
// second frame
console.log(foo); // undefined
</pre>
<p>Scope your variables using <code>this</code> to make them accessible across all frames. Please note that variables are not strictly typed in JavaScript as they were previously in AS3.
</p>
<pre>
// first frame
this.foo = 20;
// second frame
console.log(this.foo); // 20
</pre>
<p>The same approach should be taken for defining any functions on your timeline that will need to be called later in the animation or by a parent MovieClip:</p>
<pre>
this.startOver = function(){
this.score = 0;
this.gotoAndPlay("start");
};
</pre>
<h3>Stage, Root, and exportRoot</h3>
<p>When published from Flash Pro, the main timeline of the FLA is exported as a MovieClip and assigned a global reference named <code>exportRoot</code>, which can be used to access it from nested MovieClips. By default <code>exportRoot</code> is the only child of the EaselJS <code>Stage</code>.</p>
<p>EaselJS also exposes a <code>stage</code> property on all DisplayObject instances, which points to the <code>Stage</code> that contains the instance. As such, by default you can access your FLA's root timeline from a nested MovieClip using <code>this.stage.getChildAt(0)</code>.</p>
<p>You can also append global variables to the <code>window</code> object to make them accessible anywhere. This is generally frowned on in JS for the same reasons as using <code>global</code> or <code>root</code> in AS3 (scalability and polluting the global space), but should be fine for most ad development scenarios.</p>
<pre>
// on the root timeline:
window.root = this; // create a global reference
// in another MovieClip's frame action:
root.doSomething(); // can use the global variable without a scope
</pre>
<h2>Events </h2>
<p>The event model in CreateJS is similar to AS3's but has a few variations you should be aware of. Firstly, there are no <code>onevent</code> handlers. For example you can't use:</p>
<pre>
myBtn.onClick = function(){...};
</pre>
<p>Instead you can use <code>addEventListener()</code> or <code>on()</code>. The former is very similar to <code>addEventListener()</code> in AS3, and nearly identical to its namesake in JS.</p>
<pre>
// named function:
myBtn.addEventListener("click", handleClick);
// anonymous function:
myBtn.addEventListener("click", function() { … });
</pre>
<p>As in vanilla JS, these callbacks are not scoped, so in the examples above, "this" will not point to the local scope. You must either use <code>bind()</code> or create a scope in the activation object.</p>
<pre>
// bind:
myBtn.addEventListener("click", handleClick.bind(this));
// activation object:
var _this = this;
myBtn.addEventListener("click", function() { _this.doStuff(); });
</pre>
<p>The <code>on()</code> method provides an easy way to scope your methods (and offers some other handy features). By default, <code>on()</code> sets the scope to the dispatching object, but you can include a third parameter to specify your own scope.</p>
<pre>
// will execute in myBtn:
myBtn.on("click", function() {...});
// will execute in the local scope:
myBtn.on("click", function() {...}), this);
// also works with named functions:
myBtn.on("click", handleClick, this);
</pre>
<p>It's important to note that this scoping works by wrapping your handler in an anonymous function and subscribing it as the listener. This listener is returned by <code>on()</code> and must be used if you want to remove the listener later. There is also an <code>off()</code> method which is just a shortcut to <code>removeEventListener()</code>.</p>
<pre>
var listener = myBtn.on("click", handleClick);
// this won't work:
myBtn.off("click", handleClick);
// because the actual listener, is the one returned by on():
myBtn.off("click", listener);
</pre>
<p>The <code>on()</code> method also provides the ability to create single use event listeners, and to pass data to the listener.</p>
<pre>
// this event listener will automatically unsubscribe after it runs:
myBtn.on("click", handleClick, this, true, "some data");
function handleClick(evt, data) {
console.log(data); // "some data"
}
</pre>
<h2>Mouse Interactions</h2>
<p>EaselJS display objects dispatch most of the mouse events you would expect coming from AS3, including <code>click</code>, <code>dblclick</code>, and <code>mousedown</code>. To save costs when they are unneeded, you must explicitly enable <code>mouseover</code>, <code>mouseout</code>, <code>rollover</code>, and <code>rollout</code> events, via:</p>
<pre>
myStage.enableMouseOver()
</pre>
<p>This also enables support for the <code>cursor</code> property, and is included automatically in the HTML generated by Flash Pro if a button is included.</p>
<p>Similarly, untargeted mouse events such as <code>mousemove</code> and <code>mouseup</code> are not dispatched to every display object on stage to reduce unnecessary overhead. Instead, you can subscribe to stage level events <code>stagemousemove</code> and <code>stagemouseup</code>. The following shows a simple mouse follow behavior:
</p>
<pre>
this.stage.on("stagemousemove", moveTarget, this);
function moveTarget(evt){
var t = this.myTarget;
t.x = evt.stageX;
t.y = evt.stageY;
};
</pre>
<p>EaselJS also dispatches two events that make implementing drag interactions really simple. The <code>pressmove</code> event fires whenever the mouse moves after a press occurs on the target object. The <code>pressup</code> event is fired when the mouse is released after a press on the target, even if the mouse is no longer over the target (and in most browsers, even if it's outside the window). A drag interaction looks like this:</p>
<pre>
this.foo.on("pressmove", function(evt) {
this.foo.x = evt.localX;
this.foo.y = evt.localY;
}, this);
this.foo.on("pressup", function(evt) {
console.log("final position:", this.foo.x, this.foo.y);
}, this);
</pre>
<p>When targeting touch-based devices, you should generally enable touch interactions. EaselJS will listen for touch events from both the <code>touch</code> and <code>pointer</code> JS APIs, and translate them into mouse events. This has no effect on devices without touch.</p>
<pre>
createjs.Touch.enable(myStage);
</pre>
<h3>Cursor</h3>
<p>It is an ad-creation best practice to display a pointer icon anytime the user has moused over a clickable design element. This functionality is enabled by default on Button symbols (or when using <code>ButtonHelper</code>), but must be manually enabled on other clickable elements as follows:</p>
<pre>
this.childMC.cursor = "pointer";
// and mouseover must be enabled for your stage (just once):
myStage.enableMouseOver();
</pre>
<h2>DisplayObject Properties</h2>
<p>Published movieclip symbols expose methods to control their timelines and properties to adjust their settings, similarly to AS3. Here are some common examples:</p>
<p><code>--- code
mc.gotoAndPlay("yourFrameLabelName");
mc.gotoAndStop(10);
mc.stop();
mc.play();
mc.scaleX = 1;
mc.scaleY = .5;
mc.x = 0;
mc.y = 150;
mc.alpha = .75;
mc.visible = false;</code></p>
<h2>Programmatic Animations</h2>
<p>You can use tweening engines such as TweenJS, or GSAP TweenLite to create new animations (TweenJS powers the timeline animations published from Flash Pro), but it's also easy to write custom animation using the <code>tick</code> event, which is analogous to <code>enterframe</code> in AS3, and dispatched from all display objects within the display list each time the stage is updated, immediately before it redraws.</p>
<pre>
this.on("tick", function() {
this.x ++; // move right each tick
});
</pre>
<p>You can also subscribe to the global tick event dispatched from Ticker.</p>
<pre>
createjs.Ticker.on("tick", function() {
console.log("tick!");
});
</pre>
<p>The stage must have <code>update()</code> called each tick in order to propagate the <code>tick</code> event to child display objects and to redraw. By default, Flash Pro subscribes the stage as a listener to Ticker's <code>tick</code> event, which covers this requirement, but you may want to remove this and call your own tick handler in order to run logic before the update. It's very important to call <code>stage.update()</code> and pass the tick event object to it (this is needed for time syncing in EaselJS).</p>
<pre>
createjs.Ticker.on("tick", handleTick);
function handleTick(evt) {
// do stuff
myStage.update(evt);
}
</pre>
<p>Setting the frequency of these events can be done as follows, which effectively sets the framerate of your ad.
</p>
<pre>
//the following are equivalent, 1000ms / 40fps = 25ms
createjs.Ticker.interval = 25;
createjs.Ticker.framerate = 40;
</pre>
<h2>Frame Numbers & Labels</h2>
<p>EaselJS uses zero-based frame numbering rather than Flash's (unusual) one-based indexing. This can cause confusion as it (currently) requires you to subtract 1 from the indexes displayed in Flash Pro.</p>
<p>To avoid this confusion, it is suggested that you label your animation frames with frame labels, and reference those in your code rather than numbers.</p>
<pre>
this.childMC.gotoAndPlay(0); // first frame, may be confusing
this.childMC.gotoAndPlay("start"); // less ambiguous, and easier to update
</pre>
<p> </p>
<h2>Links</h2>
<p>Successfully navigating to a clickthrough URL can be achieved in a number of ways, however the following method is the closest to what was previously done in ActionScript and should function properly in all browser scenarios:</p>
<pre>
this.myBtn.on("click", function(evt){
window.open("URL", "_blank");
});
</pre>
<p>Note that you need to be careful where this code is placed. If it is on a frame in a looping MovieClip, it will be called repeatedly, and set up multiple listeners for the click. Ensure it only runs once, either by placing the code on a frame that won't loop, or by setting an initialization variable.</p>
<pre>
if (!this.inited) {
this.myBtn.on("click", ... );
this.inited = true;
}
</pre>
<h2>Flash Pro Publish Settings</h2>
<p>There are several Publish Settings available within Flash Pro that are very important to understand and configure properly. </p>
<h3>Overwrite Images</h3>
<p>Under the "Asset Export Options" publish setting in Flash Pro CC, you can uncheck "Images", so that Flash Pro will no longer replace your images with each republish. This allows you to replace images with optimized versions. Just remember that if you add new images to the FLA, they will not be exported until you re-enable this option.</p>
<h3>Overwrite HTML</h3>
<p>Under the "Output File" options you can uncheck this setting, and the contents of the HTML file will not be overwritten on subsequent publishes. This can be useful since you'll need to customize the HTML file for tracking implementation, meta tags, CSS, backup image logic, linking to externally referenced JavaScript helper files and libraries, etc.</p>
<p>Alternatively, you can duplicate the HTML file and modify the copy. You can then inject a line of code into the first frame of your main timeline to load the copy automatically. This is ideal when working with a developer, since it allows the developer to provide a deploy HTML file to test with. Ensure you remove the redirect for deployment.</p>
<pre>
// redirect to deploy.html if we aren't already there:
if (window.location.pathname.indexOf("deploy.html") == -1) {
window.location = "./deploy.html";
}
</pre>
<h3>Export All Bitmaps As Sprite Sheets</h3>
<p>By checking this setting under "Asset Export Options/Images" Flash Pro CC will automatically combine all of the utilized images within your Flash library into a single, PNG32 sprite sheet along with a JSON file defining all of the coordinates of the separate image regions that comprise the sprite sheet.</p>
<p>This is a significant time-saver, versus the manual process previously required. It is important to test and evaluate the outputted sprite sheet when using this new and evolving feature. Because Flash is outputting the sprite sheet to a single PNG32 sprite sheet, server requests are successfully reduced, however the overall file size is increased when compared to leveraging separate sprite sheets for PNG32 and JPEG image assets. Because of this, you should definitely compress the sprite sheet with ImageOptim or a comparable image optimization tool in order to maximize the file size compression.</p>
<p><hr></p>
<h1>Appendix: Document History</h1>
<h3>October 5, 2015</h3>
<p>Initial release.</p>
<h3>October 26, 2015</h3>
<p>Minor edits, and changes to reflect AdHelper updates.</p>
</article>
</body>
</html>