Managing assets in Feathers themes (AS3/Starling version)

There are two primary approaches to managing assets (such as textures or bitmap fonts) in Feathers themes. You can embed assets in the SWF or you can load assets at runtime.

Embedded Assets

The first approach packages everything up in one place by embedded the assets into the final SWF (or SWC) with the compiled code. The assets are preloaded and ready to instantiate and use as needed.

It's the easiest way for beginners to get started with the Feathers example themes. Just drop a SWC into your project, and instantiate the theme:

new MetalWorksMobileTheme();

However, this approach requires more memory. The Flash runtimes store the full running SWF in memory at all times. When an embedded asset is instantiated, the memory for that asset is duplicated from the SWF, essentially doubling the memory.

The example themes have small texture atlases, so this impact will be low, but more advanced custom themes may require a much larger amount of memory.

Loading Assets at Runtime

The second approach keeps the assets separate from the code by using the Starling AssetManager to load assets at rutnime. The assets can be loaded from a URL, and in AIR, the assets can also be loaded from the file system. The memory usage is lower because the assets will only appear in memory once, unlike when they are embedded. However, setting up the theme to use an asset manager is a little more complicated.

First, you need to specify the location of the assets. In an AIR app, you need to include the asset files when packaging your app. MetalWorksMobileThemeWithAssetManager requires two files:

  • images/metalworks.xml

  • images/metalworks.png

In the following example, we tell the theme that the images folder (which contains the two files) are placed in File.applicationDirectory:

var theme:MetalWorksMobileThemeWithAssetManager =
    new MetalWorksMobileThemeWithAssetManager( File.applicationDirectory.url );

You don't necessarily need to add the assets directly to File.applicationDirectory. You might want to add them to a subdirectory instead. Flash Builder and other IDEs should allow you to specify the location of included files, and you can also specify the included files when packaging with the ADT command line tool.

Next, let's add a listener to the theme for Event.COMPLETE:

theme.addEventListener( Event.COMPLETE, theme_completeHandler );

This event will be dispatched when the asset manager has finished loading all of the assets, and the theme is ready to start skinning Feathers components. In other words, you should always wait until the theme dispatches this event before you start showing any of your app's user interface. Otherwise, the theme won't have textures and things ready to provide skins.

The listener for this event might look something like this:

private function theme_completeHandler( event:Event ):void
{
    // the theme is ready!
 
    this.button = new Button();
    this.button.label = "Click Me";
    this.addChild( button );
}

Other themes will require their own set of files. The main thing to keep in mind is that you need to point to what the theme considers the root directory of its assets. In the case of MetalWorksMobileThemeWithAssetManager, that isn't actually the images directory, but its parent directory instead. In future versions of Feathers, this theme might require additional assets that are in a directory next to the images directory, so we want to keep it flexible. As an example, MinimalMobileThemeWithAssetManager requires these files:

  • images/minimal.xml

  • images/minimal.png

  • fonts/pf_ronda_seven.fnt

As you can see, the bitmap font file that MinimalMobileThemeWithAssetManager uses is in a separate directory from the image files. The parent directory of images and fonts is the real "root" directory for this theme's files.

Loading assets with a custom asset manager

By default, the theme will create its own AssetManager. If you would like to load extra assets that the theme isn't aware of, you may optionally pass in your own asset manager to the example theme's constructor:

var assets:AssetManager = new AssetManager();
assets.enqueue( File.applicationDirectory.resolvePath( "./images/custom-asset.png" ) );
 
var theme:MetalWorksMobileThemeWithAssetManager =
    new MetalWorksMobileThemeWithAssetManager( File.applicationDirectory.url, assets );
 
theme.addEventListener( Event.COMPLETE, theme_completeHandler );

Add all of extra assets that you need before passing the asset manager to the theme.

Never call the loadQueue() function on the AssetManager that you pass to a theme. The theme must call it for you because the asset manager does not dispatch any kind of complete event that the theme can listen to, and the theme needs to know when its assets are ready. If you need to know when the assets are fully loaded, listen for Event.COMPLETE on the theme.