blob: c7e07520f8e2c5355a73ce7bdaee83ba249a5831 [file] [log] [blame]
////////////////////////////////////////////////////////////////////////////////
//
// Licensed to the Apache Software Foundation (ASF) under one or more
// contributor license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright ownership.
// The ASF licenses this file to You under the Apache License, Version 2.0
// (the "License"); you may not use this file except in compliance with
// the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
////////////////////////////////////////////////////////////////////////////////
package mx.charts.chartClasses
{
import flash.display.BitmapData;
import flash.display.DisplayObject;
import flash.display.Graphics;
import flash.events.Event;
import flash.geom.Matrix;
import flash.geom.Point;
import flash.utils.Dictionary;
import mx.core.mx_internal;
import mx.charts.CategoryAxis;
import mx.charts.PieChart;
import mx.core.IFlexDisplayObject;
import mx.core.IUIComponent;
use namespace mx_internal;
[DefaultProperty("dataChildren")]
/**
* PolarDataCanvas class enables user to use graphics API
* with respect to data coordinates instead of screen coordinates.
*
* <p>The drawing region for canvas is determined by <code>radialAxis</code>
* and <code>angularAxis</code> if they are specified. Otherwise,
* default axes of chart will be considered to compute canvas drawing region.</p>
*
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public class PolarDataCanvas extends ChartElement implements IDataCanvas
{
include "../../core/Version.as";
//----------------------------------------------
//
// Constructor
//
//----------------------------------------------
/**
* Constructor.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function PolarDataCanvas()
{
super();
dataTransform = new PolarTransform();
}
//----------------------------------------------
//
// Variables
//
//----------------------------------------------
private var _xMap:Dictionary;
private var _yMap:Dictionary;
private var _hDataDesc:DataDescription = new DataDescription();
private var _vDataDesc:DataDescription = new DataDescription();
private var _dataCache:PolarDataCache;
private var _dataCacheDirty:Boolean = true;
private var _filterDirty:Boolean = true;
private var _mappingDirty:Boolean = true;
private var _transformDirty:Boolean = true;
private var _oldUW:Number;
private var _oldUH:Number;
private var borderWidth:Number = 0;
private var _bAxesDirty:Boolean = false;
private var _childMap:Dictionary = new Dictionary(true);
//----------------------------------------------
//
// Properties
//
//----------------------------------------------
//----------------------------------
// angularAxis
//----------------------------------
/**
* @private
* Storage for the angularAxis property.
*/
private var _angularAxis:IAxis;
[Inspectable(category="Data")]
/**
* Defines the labels, tick marks, and data position
* for items on the x-axis.
* Use either the LinearAxis class or the CategoryAxis class
* to set the properties of the angularAxis as a child tag in MXML
* or create a LinearAxis or CategoryAxis object in ActionScript.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function get angularAxis():IAxis
{
return _angularAxis;
}
/**
* @private
*/
public function set angularAxis(value:IAxis):void
{
_angularAxis = value;
_bAxesDirty = true;
invalidateData();
invalidateProperties();
}
//-----------------------------------------
// dataChildren
//-----------------------------------------
/**
* @private
* Storage for dataChildren property
*/
private var _dataChildren:Array /* of DisplayObject */ = [];
[Inspectable(category="General")]
/**
* @private
* Array of child objects
*/
public function get dataChildren():Array /* of DisplayObject */
{
return _dataChildren;
}
/**
* @private
*/
public function set dataChildren(value:Array /* of DisplayObject */):void
{
for (var p:* in _childMap)
{
removeChild(_childMap[p].child);
}
_childMap = new Dictionary(true);
_dataChildren = value;
var n:int = value.length;
for (var i:int =0; i < n; i++)
{
var dc:PolarDataChild;
if (value[i] is PolarDataChild)
dc = value[i];
else
dc = new PolarDataChild(value[i]);
_childMap[dc.child] = dc;
dc.left = 0;
dc.top = 0;
dc.addEventListener("change",dataChildChangeHandler,false,0,true);
super.addChild(dc.child);
}
invalidateOpCodes();
}
//-----------------------------------------
// includeInRanges
//-----------------------------------------
/**
* @private
* Storage for includeInRanges property
*/
private var _includeInRanges:Boolean = false;
[Inspectable(category="General")]
/**
* If <code>true</code>, the computed range of the chart is affected by this
* canvas.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function get includeInRanges():Boolean
{
return _includeInRanges;
}
/**
* @private
*/
public function set includeInRanges(value:Boolean):void
{
if (_includeInRanges == value)
return;
_includeInRanges = value;
dataChanged();
}
//----------------------------------
// radialAxis
//----------------------------------
/**
* @private
* Storage for the radialAxis property.
*/
private var _radialAxis:IAxis;
[Inspectable(category="Data")]
/**
* Defines the labels, tick marks, and data position
* for items on the y-axis.
* Use either the LinearAxis class or the CategoryAxis class
* to set the properties of the angular axis as a child tag in MXML
* or to create a LinearAxis or CategoryAxis object in ActionScript.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function get radialAxis():IAxis
{
return _radialAxis;
}
/**
* @private
*/
public function set radialAxis(value:IAxis):void
{
_radialAxis = value;
_bAxesDirty = true;
invalidateData();
invalidateProperties();
}
//--------------------------------------------
// opCodes
//--------------------------------------------
/**
* @private
* Storage for opCodes property
*/
private var _opCodes:Array /* of PolarOpCode */ = [];
[ArrayElementType("PolarOpCode")]
/**
* @private
*/
private function get opCodes():Array /* of PolarOpCode */
{
return _opCodes;
}
/**
* @private
*/
private function set opCodes(value:Array /* of PolarOpCode */):void
{
_opCodes = value;
invalidateOpCodes();
}
//----------------------------------------
// totalValue
//----------------------------------------
/**
* @private
* Storage for totalValue porperty
*/
private var _totalValue:Number = 100;
/**
* @private
* Value used to calcualte angle of the data value.
* This will be used only if angularAxis is not CategoryAxis
*/
public function get totalValue():Number
{
return _totalValue;
}
/**
* @private
*/
public function set totalValue(value:Number):void
{
_totalValue = value;
invalidateData();
invalidateProperties();
}
//----------------------------------------------
//
// Overridden Methods
//
//----------------------------------------------
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function addChild(child:DisplayObject):DisplayObject
{
var dc:PolarDataChild = new PolarDataChild(child);
dc.left = 0;
dc.top = 0;
_childMap[child] = dc;
_dataChildren.push(dc);
dc.addEventListener("change",dataChildChangeHandler,false,0,true);
invalidateOpCodes();
return super.addChild(child);
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function addChildAt(child:DisplayObject,index:int):DisplayObject
{
var dc:PolarDataChild = new PolarDataChild(child);
_childMap[child] = dc;
_dataChildren.push(dc);
dc.addEventListener("change",dataChildChangeHandler,false,0,true);
invalidateOpCodes();
return super.addChildAt(child,index);
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function removeChild(child:DisplayObject):DisplayObject
{
super.removeChild(child);
if (child in _childMap)
delete _childMap[child];
_dataChildren.splice(_dataChildren.indexOf(child),1);
return child;
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function removeChildAt(index:int):DisplayObject
{
var child:DisplayObject = super.removeChildAt(index);
if (child in _childMap)
delete _childMap[child];
_dataChildren.splice(index,1);
return child;
}
/**
* @private
*/
override protected function commitProperties():void
{
super.commitProperties();
if (_bAxesDirty)
{
if (dataTransform)
{
if (_angularAxis)
dataTransform.setAxis(PolarTransform.ANGULAR_AXIS, _angularAxis);
if (_radialAxis)
dataTransform.setAxis(PolarTransform.RADIAL_AXIS, _radialAxis);
}
_bAxesDirty = false;
}
var c:PolarChart = PolarChart(chart);
if (c)
{
if (!_angularAxis)
PolarTransform(dataTransform).setAxis(PolarTransform.ANGULAR_AXIS, c.angularAxis);
if (!_radialAxis)
PolarTransform(dataTransform).setAxis(PolarTransform.RADIAL_AXIS, c.radialAxis);
}
dataTransform.elements = [this];
invalidateOpCodes();
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override protected function updateDisplayList(unscaledWidth:Number,
unscaledHeight:Number):void
{
super.updateDisplayList(unscaledWidth, unscaledHeight);
validateData();
mapChildren();
var updated:Boolean = validateTransform();
if (updated)
{
var g:Graphics = graphics;
g.clear();
var n:int = _opCodes.length;
for (var i:int = 0; i < n; i++)
{
_opCodes[i].render(this,_dataCache);
}
positionChildren();
}
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function invalidateDisplayList():void
{
_dataCacheDirty = true;
_mappingDirty = true;
_filterDirty = true;
_transformDirty = true;
super.invalidateDisplayList();
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function set dataTransform(value:DataTransform):void
{
if (value)
{
super.dataTransform = value;
}
else
{
var p:String;
for (p in dataTransform.axes)
dataTransform.getAxis(p).unregisterDataTransform(dataTransform);
}
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
override public function describeData(dimension:String,
requiredFields:uint):Array /* of DataDescription */
{
updateMapping();
var result:Array /* of DataDescription */ = [];
if (_includeInRanges)
{
if (dimension == PolarTransform.RADIAL_AXIS)
{
if (_dataCache.xCache.length)
result.push(_vDataDesc);
}
else if (dimension == PolarTransform.ANGULAR_AXIS)
{
if (_dataCache.yCache.length)
result.push(_hDataDesc);
}
}
return result;
}
/**
* @inheritDoc
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
// this function is called by the charting package when the axes that affect this element change their mapping some how.
// that means we need to call the mapCache function again to get new mappings.
override public function mappingChanged():void
{
invalidateDisplayList();
}
//----------------------------------------------
//
// Methods
//
//----------------------------------------------
mx_internal function dataChildChangeHandler(event:Event):void
{
dataChildren = dataChildren;
}
/**
* This adds any <code>DisplayObject</code> as child to current canvas
*
* @param child A DisplayObject instance that is to be added as child to the current canvas.
* @param angleLeft Left angular coordinate of the child, in data coordinates.
* @param radialTop Top radial coordinate of the child, in data coordinates.
* @param angleRight Right angular coordinate of the child, in data coordinates.
* @param radialBottom Bottom radial coordinate of the child, in data coordinates.
* @param angleCenter Middle angular coordinate of the child, in data coordinates.
* @param radialCenter Middle radial coordinate of the child, in data coordinates.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function addDataChild(child:DisplayObject,angleLeft:* = undefined, radialTop:* = undefined, angleRight:* = undefined,
radialBottom:* = undefined , angleCenter:* = undefined, radialCenter:* = undefined):void
{
var dc:PolarDataChild = new PolarDataChild(child,angleLeft,radialTop,angleRight,radialBottom);
dc.addEventListener("change",dataChildChangeHandler,false,0,true);
addChild(child);
updateDataChild(child,angleLeft,radialTop,angleRight,radialBottom,angleCenter,radialCenter);
invalidateOpCodes();
}
/**
* Removes all data children (DisplayObject instances) of the canvas.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function removeAllChildren():void
{
var n:int = _dataChildren.length;
for (var i:int = n - 1; i >= 0; i--)
{
removeChildAt(i);
}
}
/**
* Updates the position of any child to the current canvas.
*
* @param child A DisplayObject instance that is to be added as a child of the current canvas.
* @param angleLeft Left angular coordinate of the child, in data coordinates.
* @param radialTop Top radial coordinate of the child, in data coordinates.
* @param angleRight Right angular coordinate of the child, in data coordinates.
* @param radialBottom Bottom radial coordinate of the child, in data coordinates.
* @param angleCenter Middle angular coordinate of the child, in data coordinates.
* @param radialCenter Middle radial coordinate of the child, in data coordinates.
*
* <p>For example:
* <pre>
* var lbl:Label = new Label();
* lbl.text = "Last Month";
* canvas.addChild(lbl);
* canvas.updateDataChild(lbl,200,20);
* </pre>
* </p>
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function updateDataChild(child:DisplayObject,angleLeft:* = undefined, radialTop:* = undefined, angleRight:* = undefined,
radialBottom:* = undefined, angleCenter:* = undefined, radialCenter:* = undefined):void
{
var dc:PolarDataChild = _childMap[child];
dc.left = angleLeft;
dc.top = radialTop;
dc.right = angleRight;
dc.bottom = radialBottom;
dc.aCenter = angleCenter;
dc.rCenter = radialCenter;
invalidateOpCodes();
}
/**
* Clears the canvas.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function clear():void
{
_opCodes = [];
invalidateOpCodes();
}
/**
* @copy flash.display.Graphics#beginFill()
* @see flash.display.Graphics
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function beginFill(color:uint , alpha:Number = 1):void
{
pushOp(PolarOpCode.BEGIN_FILL, { color: color, alpha: alpha });
}
/**
* Fills a drawing area with a bitmap image. Coordinate are in terms of the <code>angularAxis</code> or
* <code>radialAxis</code> properties of the canvas.
*
* <p>The usage and parameters of this method are identical to the <code>beginBitmapFill()</code> method of the
* flash.display.Graphics class.</p>
*
* @param bitmap A transparent or opaque bitmap image that contains the bits to be displayed.
*
* @param x Angle specifying the location of the fill.
*
* @param y The distance, in pixels, from the center of the chart to the fill.
*
* @param matrix A matrix object (of the flash.geom.Matrix class),
* which you can use to define transformations on the bitmap.
*
* @param repeat If <code>true</code>, the bitmap image repeats in a tiled pattern.
* If <code>false</code>, the bitmap image does not repeat, and the edges of
* the bitmap are used for any fill area that extends beyond the bitmap.
*
* @param smooth If <code>false</code>, upscaled bitmap images are rendered
* by using a nearest-neighbor algorithm and look pixelated.
* If <code>true</code>, upscaled bitmap images are rendered by using a bilinear algorithm.
* Rendering by using the nearest neighbor algorithm is faster.
*
* @see flash.display.Graphics#beginBitmapFill()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function beginBitmapFill(bitmap:BitmapData, x:* = undefined,
y:* = undefined, matrix:Matrix = null,
repeat:Boolean = true, smooth:Boolean = true):void
{
pushOp(PolarOpCode.BEGIN_BITMAP_FILL, { bitmap:bitmap, x:x, y:y, repeat:repeat, smooth:smooth, matrix:matrix });
}
/**
* Draws a curve using the current line style from the current drawing position to (anchorX, anchorY) and using the
* control point that (controlX, controlY) specifies. The coordinates that you pass to this method are in terms of
* chart data rather than screen coordinates.
*
* <p>The usage and parameters of this method are identical to the <code>curveTo()</code> method of the
* flash.display.Graphics class.</p>
*
* @param controlAngle The angle that specifies the position of the
* control point.
*
* @param controlRadial The distance, in pixels, from the center of the chart to
* the control point.
*
* @param anchorAngle The angle that specifies the position of the
* anchor point.
*
* @param anchorRadial The distance, in pixels, from the center of the chart to
* the anchor point.
*
* @see flash.display.Graphics#curveTo()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function curveTo(controlAngle:*, controlRadial:*, anchorAngle:*, anchorRadial:*):void
{
pushOp(PolarOpCode.CURVE_TO, { controlAngle: controlAngle, controlRadial:controlRadial, anchorAngle:anchorAngle, anchorRadial:anchorRadial, borderWidth: borderWidth });
}
/**
* Draws a circle.
* Set the line style, fill, or both before you call the <code>drawCircle()</code> method,
* by calling the <code>linestyle()</code>, <code>lineGradientStyle()</code>,
* <code>beginFill()</code>, <code>beginGradientFill()</code>, or <code>beginBitmapFill()</code> method.
* Coordinates are in terms of data rather than screen coordinates.
*
* @param angle The angle that specifies the position of the center of the circle.
*
* @param radial The distance, in pixels, from the center of the chart to
* the center of the circle.
*
* @param radius The radius of the circle (in pixels).
*
* @see flash.display.Graphics#drawCircle()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function drawCircle(angle:*, radial:*, radius:Number):void
{
pushOp(PolarOpCode.DRAW_CIRCLE, { angle: angle, radial: radial, radius: radius, borderWidth: borderWidth });
}
/**
* Draws an ellipse.
* Set the line style, fill, or both before you call the <code>drawEllipse()</code> method,
* by calling the <code>linestyle()</code>, <code>lineGradientStyle()</code>,
* <code>beginFill()</code>, <code>beginGradientFill()</code>, or <code>beginBitmapFill()</code> method.
* Coordinates are in terms of data rather than screen coordinates.
*
* @param angleLeft The angle specifying the location of the top-left corner of the bounding-box of the ellipse.
*
* @param radialTop The distance, in pixels, from the center of the chart to the location of the
* top-left corner of the bounding-box of the ellipse.
*
* @param angleRight The angle specifying the location of the bottom-right corner of the bounding-box of the ellipse.
*
* @param radialBottom The distance, in pixels, from the center of the chart to the location of the
* bottom-right corner of the bounding-box of the ellipse.
*
* @see flash.display.Graphics#drawEllipse()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function drawEllipse(angleLeft:*, radialTop:*, angleRight:*, radialBottom:*):void
{
pushOp(PolarOpCode.DRAW_ELLIPSE, { angleLeft: angleLeft, radialTop: radialTop, angleRight: angleRight,
radialBottom: radialBottom, borderWidth: borderWidth });
}
/**
* Draws a rectangle.
* Set the line style, fill, or both before you call the <code>drawRect()</code> method,
* by calling the <code>linestyle()</code>, <code>lineGradientStyle()</code>,
* <code>beginFill()</code>, <code>beginGradientFill()</code>, or <code>beginBitmapFill()</code> method.
* Coordinates are in terms of data rather than screen coordinates.
*
* @param angleLeft The angle specifying the location of the top-left corner of the rectangle.
*
* @param radialTop The distance, in pixels, from the center of the chart to the location of the
* top-left corner of the rectangle.
*
* @param angleRight The angle specifying the location of the bottom-right corner of the rectangle.
*
* @param radialBottom The distance, in pixels, from the center of the chart to the location of the
* bottom-right corner of the rectangle.
*
* @see flash.display.Graphics#drawRect()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function drawRect(angleLeft:*, radialTop:*, angleRight:*, radialBottom:*):void
{
pushOp(PolarOpCode.DRAW_RECT, { angleLeft: angleLeft, radialTop: radialTop, angleRight: angleRight,
radialBottom: radialBottom, borderWidth: borderWidth });
}
/**
* Draws a rounded rectangle.
* Set the line style, fill, or both before you call the <code>drawRoundRect()</code> method,
* by calling the <code>linestyle()</code>, <code>lineGradientStyle()</code>,
* <code>beginFill()</code>, <code>beginGradientFill()</code>, or <code>beginBitmapFill()</code> method.
* Coordinates are in terms of data rather than screen coordinates.
*
* @param angleLeft The angle specifying the location of the top-left corner of the rectangle.
*
* @param radialTop The distance, in pixels, from the center of the chart to the location of the
* top-left corner of the rectangle.
*
* @param angleRight The angle specifying the location of the bottom-right corner of the rectangle.
*
* @param radialBottom The distance, in pixels, from the center of the chart to the location of the
* bottom-right corner of the rectangle.
*
* @param cornerRadius The radius of the corners, in pixels.
*
* @see flash.display.Graphics#drawRoundedRect()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function drawRoundedRect(angleLeft:*, radialTop:*, angleRight:*, radialBottom:*, cornerRadius:Number):void
{
pushOp(PolarOpCode.DRAW_ROUNDRECT, { angleLeft: angleLeft, radialTop: radialTop, angleRight: angleRight,
radialBottom: radialBottom, borderWidth: borderWidth,
cornerRadius: cornerRadius });
}
/**
* @copy flash.display.Graphics#endFill()
* @see flash.display.Graphics
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function endFill():void
{
pushOp(PolarOpCode.END_FILL);
}
/**
* Specifies a line style that Flash uses for subsequent calls to other Graphics methods (such as <code>lineTo()</code>
* or <code>drawCircle()</code>) for the object.
*
* <p>The usage and parameters of this method are identical to the <code>lineStyle()</code> method of the
* flash.display.Graphics class.</p>
*
* @param thickness An integer that indicates the thickness of the line in points; valid values are 0-255.
* If a number is not specified, or if the parameter is undefined, a line is not drawn.
* If a value of less than 0 is passed, the default is 0.
* The value 0 indicates hairline thickness; the maximum thickness is 255.
* If a value greater than 255 is passed, the default is 255.
*
* @param color A hexadecimal color value of the line; for example, red is 0xFF0000, blue is 0x0000FF, and so on.
* If a value is not indicated, the default is 0x000000 (black).
*
* @param alpha A number that indicates the alpha value of the color of the line; valid values are 0 to 1.
* If a value is not indicated, the default is 1 (solid).
* If the value is less than 0, the default is 0.
* If the value is greater than 1, the default is 1.
*
* @param pixelHinting A Boolean value that specifies whether to hint strokes to full pixels.
* This affects both the position of anchors of a curve and the line stroke size itself.
* With <code>pixelHinting</code> set to true, line widths are adjusted to full pixel widths.
* With <code>pixelHinting</code> set to false, disjoints can appear for curves and straight lines.
*
* @param scaleMode A value from the flash.display.LineScaleMode class that specifies which scale mode to use:
*
* <ul>
* <li><code>LineScaleMode.NORMAL</code>: Always scale the line thickness when the object is scaled (the default).</li>
* <li><code>LineScaleMode.NONE</code>: Never scale the line thickness.</li>
* <li><code>LineScaleMode.VERTICAL</code>: Do not scale the line thickness if the object is scaled vertically only. </li>
* <li><code>LineScaleMode.HORIZONTAL</code>: Do not scale the line thickness if the object is scaled horizontally only. </li>
* </ul>
*
* @param caps A value from the flash.display.CapsStyle class that specifies the type of
* caps at the end of lines.
* Valid values are: <code>CapsStyle.NONE</code>, <code>CapsStyle.ROUND</code>,
* and <code>CapsStyle.SQUARE</code>. If a value is not indicated, use round caps.
*
* @param joints A value from the flash.display.JointStyle class that specifies the type of
* joint appearance used at angles.
* Valid values are: <code>JointStyle.BEVEL</code>, <code>JointStyle.MITER</code>, and <code>JointStyle.ROUND</code>.
* If a value is not indicated, use round joints.
*
* @param miterLimit A number that indicates the limit at which a miter is cut off.
* Valid values range from 1 to 255 (and values outside that range are rounded to 1 or 255).
* This value is only used if the jointStyle is set to "miter".
* The <code>miterLimit</code> value represents the length that a miter can extend beyond the point at which
* the lines meet to form a joint.
* The value expresses a factor of the line thickness.
* For example, with a <code>miterLimit</code> factor of 2.5 and a thickness of 10 pixels, the miter is cut off at 25 pixels.
*
* @see flash.display.Graphics#lineStyle()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function lineStyle(thickness:Number, color:uint = 0, alpha:Number = 1.0,
pixelHinting:Boolean = false, scaleMode:String = "normal",
caps:String = null, joints:String = null, miterLimit:Number = 3):void
{
borderWidth = thickness;
pushOp(PolarOpCode.LINE_STYLE, { thickness: thickness, color: color, alpha: alpha, pixelHinting: pixelHinting, scaleMode: scaleMode,
caps: caps, joints: joints, miterLimit: miterLimit });
}
/**
* Draws a line using the current line style from the current drawing position to (angle, radial);
* the current drawing position is then set to (angle, radial).
* If the display object in which you are drawing contains content that was created
* with the Flash drawing tools, calls to the <code>lineTo()</code> method are drawn underneath the content.
* If you call <code>lineTo()</code> before any calls to the moveTo() method, the default position
* for the current drawing is (0, 0).
* If any of the parameters are missing, this method fails and the current drawing
* position is not changed. Coordinates are in terms of data rather than screen coordinates.
*
* @param angle The angle of the drawing position.
*
* @param radial The distance, in pixels, from the center of the chart to the drawing position.
*
* @see flash.display.Graphics#lineTo()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function lineTo(angle:*, radial:*):void
{
pushOp(PolarOpCode.LINE_TO, { angle: angle, radial:radial, borderWidth: borderWidth });
}
/**
* Moves the current drawing position to (angle, radial).
* If any of the parameters are missing, this method fails and the current
* drawing position is not changed.
* Coordinates are in terms of data rather than screen coordinates.
*
* @param angle The angle of the drawing position.
*
* @param radial The distance, in pixels, from the center of the chart to the drawing position.
*
* @see flash.display.Graphics#moveTo()
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function moveTo(angle:*, radial:*):void
{
pushOp(PolarOpCode.MOVE_TO, { angle: angle, radial:radial, borderWidth: borderWidth});
}
/**
* Calls the <code>updateTransform()</code> method of the canvas, if necessary.
* This method is called automatically by the canvas
* during the <code>commitProperties()</code> method, as necessary,
* but a derived canvas might call it explicitly
* if the generated values are needed at an explicit time.
* Filtering and transforming of data relies on specific values
* being calculated by the axes, which can in turn
* depend on the data that is displayed in the chart.
* Calling this function at the wrong time might result
* in extra work being done, if those values are updated.
*
* @return <code>true</code> if the underlying data of the data provider has changed; otherwise, <code>false</code>.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
protected function validateTransform():Boolean
{
var updated:Boolean = false;
if (dataTransform && _transformDirty)
{
updated = updateTransform();
}
return updated;
}
/**
* Calls the <code>updateMapping()</code>
* and <code>updateFilter()</code> methods of the canvas, if necessary.
* This method is called automatically by the canvas
* from the <code>commitProperties()</code> method, as necessary,
* but a derived canvas might call it explicitly
* if the generated values are needed at an explicit time.
*
* <p>Loading and mapping data against the axes is designed
* to be acceptable by the axes at any time.
* It is safe to call this method explicitly at any point.</p>
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
protected function validateData():void
{
if (dataTransform)
{
if (_mappingDirty)
{
updateMapping();
}
if (_filterDirty)
{
updateFilter();
}
}
}
/**
* Called when the underlying data the canvas represents
* needs to be filtered against the ranges represented by the axes
* of the associated data transform.
* This can happen either because the underlying data has changed
* or because the range of the associated axes has changed.
*
* <p>If you implement a custom canvas type, you should override this method
* and filter out any outlying data using the <code>filterCache()</code>
* method of the axes managed by its associated data transform.
* The <code>filterCache()</code> method converts any values
* that are out of range to <code>NaN</code>.
* You must be sure to call the <code>super.updateFilter()</code> method
* in your subclass.</p>
*
* <p>You should not generally call this method directly.
* Instead, if you need to guarantee that your data has been filtered
* at a given point, call the <code>validateTransform()</code> method
* of the PolarDataCanvas class.</p>
*
* <p>You can generally assume that your <code>updateData()</code>
* and <code>updateMapping()</code> methods have been called
* prior to this method, if necessary.</p>
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
protected function updateFilter():void
{
var n:int = _dataCache.xCache.length;
for (var i:int = 0; i < n; i++)
{
if (isNaN(_dataCache.xCache[i].mappedValue))
delete _dataCache.xMap[_dataCache.xCache[i].value];
}
if (!(chart is PieChart))
{
n = _dataCache.yCache.length;
for (i = 0; i < n; i++)
{
if (isNaN(_dataCache.yCache[i].mappedValue))
delete _dataCache.yMap[_dataCache.yCache[i].value];
}
}
stripNaNs(_dataCache.xCache,"mappedValue");
if (!(chart is PieChart))
stripNaNs(_dataCache.yCache,"mappedValue");
_filterDirty = false;
}
/**
* @private
*/
private function mapChildren():void
{
var width:Number;
var height:Number;
for (var p:* in _childMap)
{
var dc:PolarDataChild = _childMap[p];
var pt:Point;
var dataTransform:PolarTransform = PolarTransform(dataTransform);
var data:Object = {};
var da:Array /* of Object */ = [data];
if (dc.aCenter != undefined)
{
data["d0"] = dc.aCenter;
dataTransform.getAxis(PolarTransform.ANGULAR_AXIS).
mapCache(da, "d0", "v0");
if (chart is PieChart && !(angularAxis is CategoryAxis))
data["v0"] = data["v0"]* 100 / totalValue;
}
else if (dc.right != undefined)
{
data["d0"] = dc.right;
dataTransform.getAxis(PolarTransform.ANGULAR_AXIS).
mapCache(da, "d0", "v0");
if (chart is PieChart && !(angularAxis is CategoryAxis))
data["v0"] = data["v0"] * 100 / totalValue;
}
else
{
data["d0"] = dc.left;
dataTransform.getAxis(PolarTransform.ANGULAR_AXIS).
mapCache(da, "d0", "v0");
if (chart is PieChart && !(angularAxis is CategoryAxis))
data["v0"] = data["v0"]* 100 / totalValue;
}
if (!(chart is PieChart))
{
if (dc.rCenter != undefined)
{
data["d1"] = dc.rCenter;
dataTransform.getAxis(PolarTransform.RADIAL_AXIS).
mapCache(da, "d1", "v1");
dataTransform.transformCache(da,"v0","s0","v1","s1");
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0)*data.s1 - widthFor(dc.child) / 2,
dataTransform.origin.y -
Math.sin(data.s0)*data.s1 - heightFor(dc.child) / 2);
}
else if (dc.bottom != undefined)
{
data["d1"] = dc.bottom;
dataTransform.getAxis(PolarTransform.RADIAL_AXIS).mapCache(da, "d1", "v1");
dataTransform.transformCache(da,"v0","s0","v1","s1");
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0) * data.s1 - widthFor(dc.child),
dataTransform.origin.y -
Math.sin(data.s0) * data.s1 - heightFor(dc.child));
}
else if (dc.top != undefined)
{
data["d1"] = dc.top;
dataTransform.getAxis(PolarTransform.RADIAL_AXIS).
mapCache(da, "d1", "v1");
dataTransform.transformCache(da,"v0","s0","v1","s1");
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0)*data.s1,
dataTransform.origin.y -
Math.sin(data.s0)*data.s1);
}
}
else
{
dataTransform.transformCache(da,"v0","s0",null,null);
if (dc.aCenter != undefined)
{
if (dc.rCenter == undefined)
dc.rCenter = 1;
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0) * dc.rCenter - widthFor(dc.child) / 2,
dataTransform.origin.y -
Math.sin(data.s0) * dc.rCenter - heightFor(dc.child) / 2);
}
if (dc.right != undefined)
{
if (dc.bottom == undefined)
dc.bottom = 1;
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0) * dc.bottom - widthFor(dc.child),
dataTransform.origin.y -
Math.sin(data.s0) * dc.bottom - heightFor(dc.child));
}
else
{
if (dc.top == undefined)
dc.top = 1;
pt = new Point(dataTransform.origin.x +
Math.cos(data.s0) * dc.top,
dataTransform.origin.y -
Math.sin(data.s0) * dc.top);
}
}
dc.renderLeft = pt.x;
dc.renderTop = pt.y;
}
}
/**
* Removes any item from the provided cache whose <code>field</code>
* property is <code>NaN</code>.
* Derived classes can call this method from their <code>updateFilter()</code>
* implementation to remove any ChartItem objects that were filtered out by the axes.
*
* @param cache The data cache for the PolarDataCanvas object.
*
* @param field The value of the item's <code>field</code> property.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
protected function stripNaNs(cache:Array /* of Object */, field:String):void
{
var len:int = cache.length;
var start:int = -1;
var end:int = -1;
var i:int;
var n:int = cache.length;
if (field == "")
{
for (i = n - 1; i >= 0; i--)
{
if (isNaN(cache[i]))
{
if (start < 0)
{
start = end = i;
}
else if (end - 1 == i)
{
end = i;
}
else
{
cache.splice(end, start - end + 1);
start = end = i;
}
}
}
}
else
{
for (i = n - 1; i >= 0; i--)
{
if (isNaN(cache[i][field]))
{
if (start < 0)
{
start = end = i;
}
else if (end - 1 == i)
{
end = i;
}
else
{
cache.splice(end, start - end + 1);
start = end = i;
}
}
}
}
if (start >= 0)
cache.splice(end, start - end + 1);
}
/**
* Informs the canvas that the underlying data
* in the dataProvider has changed.
* This function triggers calls to the <code>updateMapping()</code>
* and <code>updateTransform()</code> methods on the next call
* to the <code>commitProperties()</code> method.
*
* @param invalid <code>true</code> if the data has changed.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
protected function invalidateData(invalid:Boolean = true):void
{
if (invalid)
{
invalidateDisplayList();
}
}
/**
* @private
* Takes our data values and converts them into pixel values.
*/
protected function updateMapping():void
{
if (_dataCacheDirty)
{
_dataCache = new PolarDataCache();
var i:int;
var p:*;
var record:*;
var value:*;
var boundedValue:BoundedValue;
var n:int = _opCodes.length;
for (i = 0; i < n; i++)
{
_opCodes[i].collectValues(_dataCache);
}
_dataCache.xCache = [];
_dataCache.yCache = [];
_hDataDesc.min = Number.MAX_VALUE;
_hDataDesc.max = Number.MIN_VALUE;
_vDataDesc.min = Number.MAX_VALUE;
_vDataDesc.max = Number.MIN_VALUE;
for (p in _dataCache.xMap)
{
value = _dataCache.xMap[p];
_dataCache.xCache.push({ value: value });
}
for (p in _dataCache.yMap)
{
value = _dataCache.yMap[p];
_dataCache.yCache.push({ value: value });
}
_mappingDirty = true;
_dataCacheDirty = false;
}
if (_mappingDirty)
{
dataTransform.getAxis(PolarTransform.ANGULAR_AXIS).mapCache(_dataCache.xCache,"value","mappedValue");
if (!(angularAxis is CategoryAxis))
{
n = _dataCache.xCache.length;
for (i = 0; i < n; i++)
{
_dataCache.xCache[i].mappedValue = _dataCache.xCache[i].mappedValue * 100 / totalValue;
}
}
if (!(chart is PieChart))
dataTransform.getAxis(PolarTransform.RADIAL_AXIS).mapCache(_dataCache.yCache,"value","mappedValue");
_transformDirty = true;
_mappingDirty = false;
_filterDirty = true;
var boundedValues:Array /* of BoundedValue */ = [];
n = _dataCache.xCache.length;
for (i = 0; i < n; i++)
{
value = _dataCache.xCache[i];
boundedValue = _dataCache.xBoundedValues[value.value];
if (boundedValue)
{
boundedValue.value = value.mappedValue;
boundedValues.push(boundedValue);
}
}
if (boundedValues.length > 0)
{
_hDataDesc.boundedValues = boundedValues;
boundedValues = [];
}
n = _dataCache.yCache.length;
for (i = 0; i < n; i++)
{
value = _dataCache.yCache[i];
boundedValue = _dataCache.yBoundedValues[value.value];
if (boundedValue)
{
boundedValue.value = value.mappedValue;
boundedValues.push(boundedValue);
}
}
if (boundedValues.length > 0)
{
_vDataDesc.boundedValues = boundedValues;
}
n = _dataCache.yCache.length;
for (i = 0; i < n; i++)
{
record = _dataCache.yCache[i];
_vDataDesc.min = Math.min(_vDataDesc.min, record.mappedValue);
_vDataDesc.max = Math.max(_vDataDesc.max, record.mappedValue);
}
n = _dataCache.xCache.length;
for (i = 0; i < n; i++)
{
record = _dataCache.xCache[i];
_hDataDesc.min = Math.min(_hDataDesc.min, record.mappedValue);
_hDataDesc.max = Math.max(_hDataDesc.max, record.mappedValue);
}
}
}
/**
* @private
*/
protected function updateTransform():Boolean
{
var i:int;
var record:Object;
var dataTransform:PolarTransform = PolarTransform(dataTransform);
var updated:Boolean = false;
if (_transformDirty == false)
{
if (unscaledHeight != _oldUW || unscaledWidth != _oldUW)
{
_transformDirty = true;
_oldUW = unscaledWidth;
_oldUH = unscaledHeight;
}
}
if (_transformDirty)
{
var pt:Point;
updated = true;
_transformDirty = false;
dataTransform.transformCache(_dataCache.xCache,"mappedValue","pixelValue",null,null);
var n:int = _dataCache.xCache.length;
for (i = 0; i < n; i++)
{
_dataCache.xMap[_dataCache.xCache[i].value] = _dataCache.xCache[i].pixelValue;
}
if (!(chart is PieChart))
{
dataTransform.transformCache(_dataCache.yCache,null,null,"mappedValue","pixelValue");
n = _dataCache.yCache.length;
for (i = 0; i < n; i++)
{
_dataCache.yMap[_dataCache.yCache[i].value] = _dataCache.yCache[i].pixelValue;
}
}
else
{
n = _dataCache.yCache.length;
for (i = 0; i < n; i++)
{
_dataCache.yMap[_dataCache.yCache[i].value] = _dataCache.yCache[i].value;
}
}
}
return updated;
}
/**
* @private
*/
private function widthFor(child:DisplayObject):Number
{
return (child is IUIComponent)? IUIComponent(child).getExplicitOrMeasuredWidth() + 2:
(child is IFlexDisplayObject)? IFlexDisplayObject(child).measuredWidth + 2:
child.width;
}
/**
* @private
*/
private function heightFor(child:DisplayObject):Number
{
return (child is IUIComponent)? IUIComponent(child).getExplicitOrMeasuredHeight() + 2:
(child is IFlexDisplayObject)? IFlexDisplayObject(child).measuredHeight + 2:
child.height;
}
/**
* private
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
private function positionChildren():void
{
var dataTransform:PolarTransform = PolarTransform(dataTransform);
for (var p:* in _childMap)
{
var dc:PolarDataChild = _childMap[p];
var left:Number;
var right:Number;
var top:Number;
var bottom:Number;
var width:Number;
var height:Number;
left = dc.renderLeft;
right = left + widthFor(dc.child);
top = dc.renderTop;
bottom = top + heightFor(dc.child);
if (isNaN(left) || isNaN(right) || isNaN(top) || isNaN(bottom))
{
removeChild(p);
continue;
}
if (dc.child is IFlexDisplayObject)
{
IFlexDisplayObject(dc.child).setActualSize(right-left,bottom-top);
IFlexDisplayObject(dc.child).move(left, top);
}
else
{
dc.child.width = right - left;
dc.child.height = bottom - top;
dc.child.x = left;
dc.child.y = top;
}
}
}
/**
* @private
*/
private function invalidateOpCodes():void
{
dataChanged();
invalidateDisplayList();
}
/**
* @private
*/
private function pushOp(code:int, params:Object = null):PolarOpCode
{
var op:PolarOpCode = new PolarOpCode(this,code,params);
_opCodes.push(op);
invalidateOpCodes();
return op;
}
}
}
import mx.charts.chartClasses.PolarDataCanvas;
import flash.utils.Dictionary;
import mx.charts.chartClasses.BoundedValue;
import flash.geom.Matrix;
import flash.display.Graphics;
import mx.core.mx_internal;
import flash.geom.Point;
import mx.charts.chartClasses.PolarTransform;
use namespace mx_internal;
class PolarOpCode
{
public var canvas:PolarDataCanvas;
public var code:int;
public var params:Object;
public function PolarOpCode(canvas:PolarDataCanvas,code:int, params:Object = null):void
{
this.canvas = canvas;
this.code = code;
this.params = params == null ? {} : params;
}
public static const BEGIN_BITMAP_FILL:int = 0;
public static const BEGIN_FILL:int = 1;
public static const CURVE_TO:int = 2;
public static const DRAW_CIRCLE:int = 3;
public static const DRAW_ELLIPSE:int = 4;
public static const DRAW_RECT:int = 5;
public static const DRAW_ROUNDRECT:int = 6;
public static const END_FILL:int = 7;
public static const LINE_STYLE:int = 8;
public static const LINE_TO:int = 9;
public static const MOVE_TO:int = 10;
mx_internal function collectValues(cache:PolarDataCache):void
{
switch (code)
{
case BEGIN_BITMAP_FILL:
cache.storeX(params.x)
cache.storeY(params.y);
break;
case CURVE_TO:
cache.storeX(params.anchorAngle);
cache.storeY(params.anchorRadial);
cache.storeX(params.controlAngle);
cache.storeY(params.controlRadial);
break;
case DRAW_CIRCLE:
case MOVE_TO:
case LINE_TO:
cache.storeX(params.angle);
cache.storeY(params.radial);
break;
case DRAW_ELLIPSE:
cache.storeX(params.angleLeft);
cache.storeY(params.radialTop);
cache.storeX(params.angleRight);
cache.storeY(params.radialBottom);
break;
case DRAW_RECT:
cache.storeX(params.angleLeft);
cache.storeY(params.radialTop);
cache.storeX(params.angleRight);
cache.storeY(params.radialBottom);
break;
case DRAW_ROUNDRECT:
cache.storeX(params.angleLeft);
cache.storeY(params.radialTop);
cache.storeX(params.angleRight);
cache.storeY(params.radialBottom);
break;
}
}
mx_internal function render(target:PolarDataCanvas,cache:PolarDataCache):void
{
var left:Number;
var top:Number;
var right:Number;
var bottom:Number;
var controlX:Number;
var controlY:Number;
var anchorX:Number;
var anchorY:Number;
var x:Number;
var y:Number;
var pt:Point;
var dataTransform:PolarTransform = PolarTransform(dataTransform);
var data:Object = {};
var da:Array /* of Object */ = [data];
var g:Graphics = target.graphics;
switch (code)
{
case BEGIN_BITMAP_FILL:
var m:Matrix;
if (!(params.matrix))
m = new Matrix();
else
m = params.matrix.clone();
var d:* = params.x;
if (d != undefined)
m.tx = cache.x(d);
d = params.y;
if (d != undefined)
m.ty = cache.y(d);
g.beginBitmapFill(params.bitmap,m,params.repeat,params.smooth);
break;
case BEGIN_FILL:
g.beginFill(params.color,params.alpha);
break;
case CURVE_TO:
controlX = cache.x(params.controlAngle);
controlY = cache.y(params.controlRadial);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(controlX) * controlY,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(controlX) * controlY);
controlX = pt.x;
controlY = pt.y;
anchorX = cache.x(params.anchorAngle);
anchorY = cache.y(params.anchorRadial);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(anchorX) * anchorY,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(anchorX) * anchorY);
anchorX = pt.x;
anchorY = pt.y;
if (isNaN(controlX) || isNaN(controlY) || isNaN(anchorX) || isNaN(anchorY))
return;
g.curveTo(controlX,controlY,anchorX,anchorY);
break;
case DRAW_CIRCLE:
x = cache.x(params.angle);
y = cache.y(params.radial);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(x) * y,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(x) * y);
x = pt.x;
y = pt.y;
if (isNaN(x) || isNaN(y))
return;
g.drawCircle(x, y, params.radius);
break;
case DRAW_ELLIPSE:
left = cache.x(params.angleLeft);
top = cache.y(params.radialTop);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(left) * top,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(left) * top);
left = pt.x;
top = pt.y;
right = cache.x(params.angleRight);
bottom = cache.y(params.radialBottom);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(right) * bottom,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(right) * bottom);
right = pt.x;
bottom = pt.y;
if (isNaN(left) || isNaN(top) || isNaN(right) || isNaN(bottom))
return;
g.drawEllipse(left, top,
right - left,bottom - top);
break;
case DRAW_RECT:
left = cache.x(params.angleLeft);
top = cache.y(params.radialTop);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(left) * top,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(left) * top);
left = pt.x;
top = pt.y;
right = cache.x(params.angleRight);
bottom = cache.y(params.radialBottom);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(right) * bottom,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(right) * bottom);
right = pt.x;
bottom = pt.y;
if (isNaN(left) || isNaN(top) || isNaN(right) || isNaN(bottom))
return;
g.drawRect(left, top,
right - left,bottom - top);
break;
case DRAW_ROUNDRECT:
left = cache.x(params.angleLeft);
top = cache.y(params.radialTop);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(left) * top,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(left) * top);
left = pt.x;
top = pt.y;
right = cache.x(params.angleRight);
bottom = cache.y(params.radialBottom);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(right) * bottom,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(right) * bottom);
right = pt.x;
bottom = pt.y;
if (isNaN(left) || isNaN(top) || isNaN(right) || isNaN(bottom))
return;
g.drawRoundRect(left, top,
right - left,bottom - top,params.cornerRadius,params.cornerRadius);
break;
case END_FILL:
g.endFill();
break;
case LINE_STYLE:
g.lineStyle(params.thickness,params.color,params.alpha,params.pixleHinting,params.scaleMode,params.caps,params.joints,params.miterLimit);
break;
case MOVE_TO:
x = cache.x(params.angle);
y = cache.y(params.radial);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(x) * y,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(x) * y);
x = pt.x;
y = pt.y;
if (isNaN(x) || isNaN(y))
return;
g.moveTo(x, y);
break;
case LINE_TO:
x = cache.x(params.angle);
y = cache.y(params.radial);
pt = new Point(PolarTransform(canvas.dataTransform).origin.x + Math.cos(x) * y,
PolarTransform(canvas.dataTransform).origin.y - Math.sin(x) * y);
x = pt.x;
y = pt.y;
if (isNaN(x) || isNaN(y))
return;
g.lineTo(x, y);
break;
}
}
}
import mx.core.mx_internal;
use namespace mx_internal;
class PolarDataCache
{
public var xCache:Array /* of Object */;
public var yCache:Array; /* of Object */
public var xBoundedValues:Dictionary;
public var yBoundedValues:Dictionary;
public var xMap:Dictionary;
public var yMap:Dictionary;
public function PolarDataCache():void
{
xMap = new Dictionary(true);
yMap = new Dictionary(true);
xCache = [];
yCache = [];
xBoundedValues = new Dictionary(true);
yBoundedValues = new Dictionary(true);
}
mx_internal function storeX(value:*, leftMargin:Number = 0,rightMargin:Number = 0):void
{
var bounds:BoundedValue;
xMap[value] = value;
if (leftMargin != 0 || rightMargin != 0)
bounds = xBoundedValues[value];
if (leftMargin > 0)
leftMargin += 2;
if (rightMargin > 0)
rightMargin += 2;
if (!bounds)
{
xBoundedValues[value] = bounds = new BoundedValue(0,leftMargin,rightMargin);
}
else
{
bounds.lowerMargin = Math.max(bounds.lowerMargin,leftMargin);
bounds.upperMargin = Math.max(bounds.upperMargin,rightMargin);
}
}
mx_internal function storeY(value:*,topMargin:Number = 0,bottomMargin:Number = 0):void
{
var bounds:BoundedValue;
yMap[value] = value;
if (topMargin != 0 || bottomMargin != 0)
{
bounds = yBoundedValues[value];
if (!bounds)
{
yBoundedValues[value] = bounds = new BoundedValue(0,bottomMargin,topMargin);
}
else
{
bounds.lowerMargin = Math.max(bounds.lowerMargin,bottomMargin);
bounds.upperMargin = Math.max(bounds.upperMargin,topMargin);
}
}
}
mx_internal function x(value:*):Number
{
return Number(xMap[value]);
}
mx_internal function y(value:*):Number
{
return Number(yMap[value]);
}
}
import flash.display.DisplayObject;
import flash.events.EventDispatcher;
import flash.events.Event;
use namespace mx_internal;
[DefaultProperty("content")]
[Event("change")]
class PolarDataChild extends EventDispatcher
{
public function PolarDataChild(child:DisplayObject = null,left:* = undefined, top:* = undefined, right:* = undefined, bottom:* = undefined,
aCenter:* = undefined, rCenter:* = undefined):void
{
this.child = child;
this.left = left;
this.top = top;
this.bottom = bottom;
this.right = right;
this.aCenter = aCenter;
this.rCenter = rCenter;
}
public var child:DisplayObject;
public var left:*;
public var right:*;
public var top:*;
public var bottom:*;
public var renderLeft:*;
public var renderTop:*;
public var rCenter:*;
public var aCenter:*;
public function set content(value:*):void
{
if (value is DisplayObject)
child = value;
else if (value is Class)
child = new value();
dispatchEvent(new Event("change"));
}
}