So here is the solution that I have:
- Define a Palette Class based on Proxy that can translate names (String) to colors(uint)
- Have this Palette Class read in an embeded text file that defines the colors
- Pass the palettes to a stylesheetMixin class that parses the css to replace the names in the css with the hexdecimal values before applying the styles.
Palette.as
package com.squaredi.styles
{
 /**
  * 
  * @author Drew Shefman 
  * dshefman at squaredi dot com
  * 
  */
 import flash.utils.Dictionary;
 import flash.utils.Proxy;
 import flash.utils.flash_proxy;
 
 import mx.core.ByteArrayAsset;
 
 /**
  * Read name value pairs of color names from an "external file."
  * "external file" is quoted because to get it to sync with CSS we actually embed the external files.
  * Here is an example of the external file 
* 
  * _companyRed=0xFF0000
  * _companyGreen=0x00FF00
  * _companyBlue=0x0000FF
  * _columnGreen=_companyGreen
  * 
  * Note: Watch out for spaces in the file
*/
 dynamic public class Palette extends Proxy
 {
  public static var throwErrorsForMissingColors:Boolean = true;
  public static const MISSING_COLOR:int = -1
  
  private  var dict:Dictionary;
  public var paletteName:String;
  
  public function Palette(p_name:String)
  {
   dict = new Dictionary();
  }
  
  /**
   * Create an instance of the embedded text file, 
   * Process the file and split it into name value pairs
   * Save off the names into the palette
   **/ 
  public  function addColorsViaEmbededTextClass(theClass:Class):void
  {
   //Convert the embeded text file to a string
   var byteArray:ByteArrayAsset = ByteArrayAsset(new theClass());
   var str:String = byteArray.readUTFBytes(byteArray.length);
   
   //Clean up extra characters
   str = str.split(" ").join(""); //remove white spaces
   
   
   //Loop over name/value pairs
   var split:Array = str.split("\r\n");
   var obj:Object = this;
   var key:String;
   var value:String;
   var pair:Array;
   var pairStr:String;
   var len:int = split.length;
   for ( var i:int =0 ; i < len ; i++ )
   {
    pairStr = split[i];
    pair = pairStr.split("=");
    key = pair[0];
    value = pair[1];
    obj[key] = value;
   }
  }
  
  
  flash_proxy override function getProperty(name:*):*
  {
   if (isValid(name))
   {
    if (flash_proxy::hasProperty(name))
    {
     return Number(dict[name]);
    }
    else
    {
     if (throwErrorsForMissingColors)
     {
      throw new Error(name + " is an invalid color. Please look in the external colors file to figure out the name.");
     }
     else
     {
      return MISSING_COLOR;
     }
    }
   }
   return null;
  }
  
  flash_proxy override function hasProperty(name:*):Boolean
  {
   return dict[name] != null;
  }
  
  flash_proxy override function setProperty(name:*, value:*):void
  {
   
   //Make sure the name/values are not blank
   if (name=="" || value == "") {return;}
   
   //name comes in as a QName, so we need to getString to get the value out of it.
   name = name.toString();
   
   //Allow for colors to reference previously defined colors
   if (isValid(value)) { value = flash_proxy::getProperty(value)};
   
   //Verify that the name is correct and save it 
   if (isValid(name))
   {
    dict[name] = value;
   }
   else
   {
    throw new Error("Palette colors should start with a '_' to allow for easy distinction from other css variables");
   }
  }
  
  /**
   * Force that color variables start with and "_" 
   * */
  private function isValid(value:String):Boolean
  {
   if (value.indexOf("_") == 0)
   {
    return true;
   }
   
   return false;
  }
  
 }
}
StyleSheetMixin.as:
StyleSheeMixin.as
usage: http://stackoverflow.com/questions/2292127/how-to-have-constants-in-flex-css-files
// ================================================================= /* * http://stackoverflow.com/questions/2292127/how-to-have-constants-in-flex-css-files** * Copyright (c) 2010 viatropos http://www.viatropos.com/ * Lance Pollard * lancejpollard at gmail dot com * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following * conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR * OTHER DEALINGS IN THE SOFTWARE. */ // ================================================================= package com.squaredi.styles { import flash.display.Sprite; import flash.events.Event; import flash.utils.getDefinitionByName; import mx.core.IMXMLObject; import mx.core.Singleton; import mx.styles.CSSStyleDeclaration; import mx.styles.IStyleManager2; import mx.styles.StyleManager; public class StylesheetMixin implements IMXMLObject { private var _palettes:Array; /** * Classes of static constants storing values for css */ public function get palettes():Array { return _palettes; } public function set palettes(value:Array):void { _palettes = value; } public function StylesheetMixin() { } public function setStyles():void { // get all selectors in the application var styleManager:IStyleManager2 = getStyleManager(); var selectors:Array = styleManager.selectors; var declaration:CSSStyleDeclaration; var i:int = 0; var n:int = selectors.length; for (i; i < n; i++) { declaration = styleManager.getStyleDeclaration(selectors[i]); // set palette properties to each declaration setProperties(declaration); } } protected function getStyleManager():IStyleManager2 { var application:*; try { application = flash.utils.getDefinitionByName("mx.core::FlexGlobals")["topLevelApplication"]; if (application) return application.styleManager as IStyleManager2; } catch (error:Error) { application = flash.utils.getDefinitionByName("mx.core::Application")["application"]; if (application) return IStyleManager2(Singleton.getInstance("mx.styles::IStyleManager2")); } return null; } protected function setProperties(declaration:CSSStyleDeclaration):void { var selector:Object = getDeclarationToken(declaration); var property:String; for (property in selector) { setProperty(declaration, property, selector[property]); } } public function getDeclarationToken(declaration:CSSStyleDeclaration):Object { var selector:Object = {factory:declaration.factory}; // maybe your selector has a "factory" property which we should avoid? if (!(typeof(selector.factory) == "function") || selector.factory == null) return null; selector.factory(); delete selector.factory; return selector; } public function setProperty(declaration:CSSStyleDeclaration, property:String, value:*):* { var paletteValue:*; var changed:Boolean = false; if (value is Array) { var i:int = 0; var n:int = value.length; for (i; i < n; i++) { paletteValue = getPaletteItem(value[i]); if (paletteValue) { changed = true; value[i] = paletteValue; } } } else if (value is String) { paletteValue = getPaletteItem(value); if (paletteValue) { value = paletteValue; changed = true; } } if (changed) { declaration.setStyle(property, value); } } public function getPaletteItem(targetId:String):* { var i:int = 0; var n:int = palettes.length; var PaletteClass:Object; for (i; i < n; i++) { PaletteClass = palettes[i]; if (PaletteClass[targetId]) return PaletteClass[targetId]; } return null; } private var timer:Sprite = new Sprite(); // have to wait a frame for styles to be initialized public function initialized(document:Object, id:String):void { var handler:Function = function(event:Event):void { timer.removeEventListener(Event.ENTER_FRAME, handler); timer = null; setStyles(); } timer.addEventListener(Event.ENTER_FRAME, handler); } } }So then in your Application, the way that you would use it would be:
Application.mxml
<?xml version="1.0" encoding="utf-8"?>
<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009" 
      xmlns:s="library://ns.adobe.com/flex/spark" 
      xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600"
      xmlns:styles="com.squaredi.styles.*"
      
      initialize="onInitialize(event)" 
      >
 <fx:Script>
  <![CDATA[
   import com.squaredi.styles.Palette;
   
   import mx.events.FlexEvent;
   [Embed(source="examples/styles/palette.txt", mimeType="application/octet-stream")]
   public  var paletteClass:Class
   
   [Bindable] private var allPalettes:Array
   
   protected function onInitialize(event:FlexEvent):void
   {
    var pal:Palette = new Palette("main");
    pal.addColorsViaEmbededTextClass(paletteClass);
    
    allPalettes = new Array(pal);
    
   }
  ]]>
 </fx:Script>
 
 <fx:Style>
  @namespace s "library://ns.adobe.com/flex/spark";
  @namespace mx "library://ns.adobe.com/flex/mx";
  
  
  
  .example
  {
   backgroundColor: _corpRed;
   backgroundAlpha:1.0;
   
  }
  </fx:Style>
 
 <fx:Declarations>
  <!-- Place non-visual elements (e.g., services, value objects) here -->
  <styles:StylesheetMixin palettes="{allPalettes}" />
 </fx:Declarations>
 <s:SkinnableContainer styleName="example" width="200" height="200" >
  
  <s:Label text="Hello world" />
 </s:SkinnableContainer>
</s:Application>
and your embeded text file would look like: _corpRed=0xFF0000
 
No comments:
Post a Comment