Versions Compared

Key

  • This line was added.
  • This line was removed.
  • Formatting was changed.

...

With the donation of FalconJS, the Apache Flex community has an important tool in the pursuit of getting Flex to run without Flash/AIR. There are multiple possible approaches to achieving this goal. I have chosen one approach. The basic premise is that Apache Flex can provide parallel frameworks, one written in AS and another written in JS (and potentially others written in other languages) that abstract the differences between the platform into components in the framework. MXML is used to declaratively assemble components, and AS is used as the glue code and business logic. FalconJS converts the MXML and AS code to JS and the JS framework is swapped in instead of the AS framework and suddenly your Flex app runs in the browser.

...

In the example, the source in the FlexJSUI project is the AS framework and has been compiled into a SWC and used by the FlexJSTest app. Don't worry about the sub-folders in the FlexJSUI project just yet. The key point is that there is a lightweight Button and Label component in FlexJSUI.

...

Code Block
<basic:Application xmlns:fx="http://ns.adobe.com/mxml/2009"
				   xmlns:local="*"
				   xmlns:basic="http://ns.apache.org/flex/basic" 
				   xmlns:html="http://ns.apache.org/flex/html" 
				   xmlns:models="models.*" 
				   xmlns:controllers="controllers.*">
        ...
	<basic:initialView>
		<local:MyInitialView />
	</basic:initialView>
	<basic:model>
		<models:MyModel />
	</basic:model>
	<basic:controller>
		<controllers:MyController />
	</basic:controller>
</basic:Application>

The app consists of an initial view, a model, and a controller. The MXML for the initial view might look like it does in MyInitialViewMXML.mxml:

...

In the subfolders of FlexJSUI, you will see a pattern for the framework. The pattern is that components are composited from smaller pieces. I call them "beads". There They are placed on a "strand". This is also an extensibility pattern as new beads can be placed on the strand and modify the behavior of the component. Example Examples of such beads would be accessibility beads or test automation beads. There also a potential for an IDE to place beads on component instances only the equivalent of the Adobe Flash Builder design view in order to get the components to cooperate in a visual designer tool.

...

Another feature of this framework is a central "ValuesImpl". Flex today has several ways of getting a configurable default value for certain properties and styles like StyleManager and ResourceManager. I want to create a single place you get your default values whether it is some dependency you need or some default value for a property or style. And the implementation behind the values can be swapped out for things really simple like a flat list for really small apps, or full CSS and locale order support for more sophisticated apps.

Pros and Cons

Pros

  • SWFs will be smaller (the example SWF is 11K) and should run faster mainly because there is less code running just-in-case.
  • Beads make it easier to unit test pieces
  • Beads encapsulate platform-specific code from platform-independent code
  • Composition encourages more code re-use for smaller and faster apps

Cons

Components Proxy Beads

In theory, you could write an app by declaring all base classes and setting the right beads on them. That of course, wouldn't be very usable, so Components are created that subclass the base classes and proxy a set of bead APIs out to the API surface of the component.

Folks who used Flex 3 and then Flex 4 may recall the complaints when Flex 3's layout selection went from:

Code Block

<mx:Application layout="vertical" ...>

to

Code Block

<s:Application >
    <s:layout>
       <s:VerticalLayout />
    </s:layout>

Essentially, Flex 3's API is a shorthand for the more verbose but more flexible Flex 4 pattern.

Pros and Cons

Pros

  • SWFs will be smaller (the example SWF is 11K) and should run faster mainly because there is less code running just-in-case.
  • Beads make it easier to unit test pieces
  • Beads encapsulate platform-specific code from platform-independent code
  • Composition encourages more code re-use for smaller and faster apps

Cons

  • Subclassing is no longer the way to customize components. This is because the the behavior you may want to change is in a bead that doesn't proxy its APIs to the component hosting it. This will curtail the use of MXML components.
  • Interstitial costs. Everything is now one more level of indirection away. Will performance suffer?
  • More typing. See the "Components Proxy Beads" above.
  • Number of choices. Because everything is pay as you go and common patterns will be encapsulated into components, there are likely to be several flavors of each common component. In the prototype, there is both a Button and TextButton where the TextButton carries the extra overhead of being able to have its label changed dynamically. If you don't need that, you might just want to create a bitmap of the entire button with the text burned in. Similarly, the basic List won't handle change events or editing or even variable row heights or virtualization. More and more complex beads replace simpler beads to create these more advanced Lists. This is a radical departure from the current Flex trying to jam every possible feature into a single Button class. The number of choices might be overwhelming, but folks can always just choose the fattest component in early prototyping and slim down if they need to optimize.

Other Notes

MXML generates data not code.

If you look at MyInitialView.as and compare it to MyInitialViewMXML.mxml, you will see that the generated code isn't really code like it is in Flex 4, but rather, an array of values. This provides an abstraction where the AS or JS framework can figure out the optimal way of creating the right widgets for the view. Early prototypes of this kind of data generation showed that performs as fast as the generated functions in Flex 4.

Basic components aren't skinnable

You will note that the components in the AS framework are not really skinnable. That's because the basic components in HTML aren't skinnable either. So that becomes the lowest-level starting point and more complex beads can build up from there.

JS components are not DisplayObjects

You will note that in framework.js, the components do not extend any sort of HTMLElement/DisplayObject. That is because it is best for the JS components to work on the DOM and proxy APIs to their surface. We could have done the same on the AS side, but there are certain efficiencies to having the components be in the DOM itself that I think is worth keeping, and the main Application class must be a Sprite or MovieClip in Flash.

Other Thoughts

States will be a degenerate case of a more complex "conditionals" capability.

States were used often but had a limitation once the set of states you wanted started having "dimensions". For example, the Flex default button has an emphasized state, which adds to the selected state to create a lot of potential skin states. And when we considered having this same button show a focused state, it got unwieldy so we didn't do it. We never got around to doing multi-dimensional states, but now it is time to try since we can break backward compatibility if we have to.

Early versions may not support {} data binding syntax.

The generated code for {} databinding syntax is awful, very fat and slow. Prototypes have shown that having special kinds of binding objects that know something about the kind of binding can make binding much more efficient. Eventually, I hope to make Falcon smart enough to generate different databinding objects for different databinding conditions, but for early versions, you might have to avoid using {} and declare binding objects directly in the app.

Properties vs Styles

It might be time to remove the distinction between properties and styles since that seems to be the case for CSS in HTML/JS.

Beads may not be instantiated all at once.

Or maybe not initialized all at once. There is no need to setup mouse and keyboard or even accessibility beads until very late in the component lifecycle (like after the screen is first displayed). So we will probably find a way to determine which beads get run in what order.

Flex things we will hopefully not need

  • Preloaders. Hopefully an application and its dependencies are so small you won't need a preloader. Now, you may need to use a sequence of states to stage how your application appears on screen so there isn't dead air, but you will be able to use the same set of components (although they should be the lighter ones) to create the initial visuals. No need to avoid the use of components in the preloaders like you have to do in Flex today.
  • Marshall Plan. Hopefully interfaces will not depend on complex classes, and as new APIs are needed, new interfaces will be created instead of changing existing ones. Then the Marshall Plan should not be needed, although you may need to create "smart" beads that can handle legacy vs current code paths.
  • Versioning API/Conditional Compilation By Version. If we decide to change the behavior of an API, that should be implemented via a new default bead. You can use an old bead to get old behavior. Or write a new bead that knows how to switch behavior, but keeping multiple code paths around will not be a recommended practice.

The Last Mile

As Mike Labriola claims in his Randori presentation, most of the work in finishing HTML apps is in tweaking the UI for the various browsers. This will likely be true for FlexJS apps as well, although my prototype is using absolute positioning as much as possible.

It might be that if we implement really good CSS Media Query support that folks will be able to tune their UIs via Media Query, but right now, the prototype doesn't produce HTML, the MXML generates a data structure and the components create the DOM elements dynamically. We might want to add a capability to dump out a screen as HTML and make the components smart enough to try to find themselves in the DOM at some point in their lifecycle. That way, folks who specialize in browser-specific UI tuning can tune the HTML and CSS in the language they know best.

To Do Items/Next Steps

  • Pick a place to work in SVN. I think I will leave the code untouched, branch it and start working in the branch so the demo continues to stay in working order. Not sure where to put that branch, but will decide soon.
  • Improve Falcon MXML handling. We need to get Falcon to handle more and more MXML scenarios
  • Get Falcon to generate data structures instead of code. Probably via some flag that leaves the old generated code around so we can double check if a bug is in the compiler or somewhere else.
  • Figure out how to generate the right JS from MXML. I'm not clear that Falcon goes through the AST reducer phase for MXML so we might need to add some hooks for FalconJS to get in there.
  • More components. Feel free to try to write new versions of existing Spark and MX components via beads. Remember, start simple and build up from there.
  • More CSS and Media Query support. We will need to have better CSS support than we do now, especially for media query so folks can try to tune their apps without needing to see the HTML. I've been wondering for some time whether the current Flex CSS implementation which uses the prototype chain continues to be the right approach. Several years back, there was an article by someone who wrote the CSS implementation for one of the browsers and he said he wasn't building up chains, he would just walk the HTML and see how the CSS applied.
    Several tests have shown that in Flash, the ability to use the prototype chain is about 10 times faster than equivalent AS code that tests of the object has a property then checks some other object higher in the chain. However, it isn't clear that in a full CSS implementation, especially one that honors the "inherit" value, that the overhead of creating and maintaining the chains as the UI morphs through various view states overwhelms the advantages of the prototype lookup. So some experimenting needs to be done in this area.
  • Subclassing is no longer the way to customize components. This is because the the behavior you may want to change is in a bead that doesn't proxy its APIs to the component hosting it. This will curtail the use of MXML components.
  • Interstitial costs. Everything is now one more level of indirection away. Will performance suffer?
    Number of choices