| //////////////////////////////////////////////////////////////////////////////// |
| // |
| // 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.skins.halo |
| { |
| |
| import flash.display.DisplayObject; |
| import flash.display.GradientType; |
| import mx.core.IFlexDisplayObject; |
| import mx.core.UIComponent; |
| import mx.core.mx_internal; |
| import mx.styles.StyleManager; |
| import mx.utils.ColorUtil; |
| import mx.core.IProgrammaticSkin; |
| |
| use namespace mx_internal; |
| |
| /** |
| * The skin for all the states of a PopUpButton. |
| * |
| * @langversion 3.0 |
| * @playerversion Flash 9 |
| * @playerversion AIR 1.1 |
| * @productversion Flex 3 |
| */ |
| public class PopUpButtonSkin extends UIComponent implements IProgrammaticSkin |
| { |
| include "../../core/Version.as"; |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Class variables |
| // |
| //-------------------------------------------------------------------------- |
| |
| /** |
| * @private |
| */ |
| private static var cache:Object = {}; |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Class methods |
| // |
| //-------------------------------------------------------------------------- |
| |
| /** |
| * @private |
| * Several colors used for drawing are calculated from the base colors |
| * of the component (themeColor, borderColor and fillColors). |
| * Since these calculations can be a bit expensive, |
| * we calculate once per color set and cache the results. |
| */ |
| private static function calcDerivedStyles(themeColor:uint, |
| fillColor0:uint, |
| fillColor1:uint):Object |
| { |
| var key:String = HaloColors.getCacheKey(themeColor, |
| fillColor0, fillColor1); |
| |
| if (!cache[key]) |
| { |
| var o:Object = cache[key] = {}; |
| |
| // Cross-component styles. |
| HaloColors.addHaloColors(o, themeColor, fillColor0, fillColor1); |
| } |
| |
| return cache[key]; |
| } |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Constructor |
| // |
| //-------------------------------------------------------------------------- |
| |
| /** |
| * Constructor. |
| * |
| * @langversion 3.0 |
| * @playerversion Flash 9 |
| * @playerversion AIR 1.1 |
| * @productversion Flex 3 |
| */ |
| public function PopUpButtonSkin() |
| { |
| super(); |
| |
| mouseEnabled = false; |
| } |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Overridden properties |
| // |
| //-------------------------------------------------------------------------- |
| |
| |
| //---------------------------------- |
| // measuredWidth |
| //---------------------------------- |
| |
| /** |
| * @private |
| */ |
| override public function get measuredWidth():Number |
| { |
| return DEFAULT_MEASURED_MIN_WIDTH; |
| } |
| |
| //---------------------------------- |
| // measuredHeight |
| //---------------------------------- |
| |
| /** |
| * @private |
| */ |
| override public function get measuredHeight():Number |
| { |
| return DEFAULT_MEASURED_MIN_HEIGHT; |
| } |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Overridden methods |
| // |
| //-------------------------------------------------------------------------- |
| |
| /** |
| * @private |
| */ |
| override protected function updateDisplayList(w:Number, h:Number):void |
| { |
| super.updateDisplayList(w, h); |
| |
| // User-defined styles. |
| var arrowColor:uint = getStyle("iconColor"); |
| var borderColor:uint = getStyle("borderColor"); |
| var cornerRadius:Number = getStyle("cornerRadius"); |
| var fillAlphas:Array = getStyle("fillAlphas"); |
| var fillColors:Array = getStyle("fillColors"); |
| StyleManager.getStyleManager(moduleFactory).getColorNames(fillColors); |
| var highlightAlphas:Array = getStyle("highlightAlphas"); |
| var themeColor:uint = getStyle("themeColor"); |
| |
| // Derivative styles. |
| var derStyles:Object = calcDerivedStyles(themeColor, fillColors[0], |
| fillColors[1]); |
| |
| var borderColorDrk1:Number = |
| ColorUtil.adjustBrightness2(borderColor, -50); |
| |
| var themeColorDrk1:Number = |
| ColorUtil.adjustBrightness2(themeColor, -25); |
| |
| var popUpIcon:IFlexDisplayObject = |
| IFlexDisplayObject(getChildByName("popUpIcon")); |
| |
| if (!popUpIcon) |
| { |
| var popUpIconClass:Class = Class(getStyle("popUpIcon")); |
| popUpIcon = new popUpIconClass(); |
| DisplayObject(popUpIcon).name = "popUpIcon"; |
| addChild(DisplayObject(popUpIcon)); |
| DisplayObject(popUpIcon).visible = true; |
| } |
| |
| var arrowButtonWidth:Number = Math.max(getStyle("arrowButtonWidth"), |
| popUpIcon.width + 3 + 1); |
| |
| var dividerPosX:Number = w - arrowButtonWidth; |
| |
| popUpIcon.move(w - (arrowButtonWidth + popUpIcon.width) / 2, |
| (h - popUpIcon.height) / 2); |
| |
| var cr:Number = Math.max(0, cornerRadius); |
| var cr1:Number = Math.max(0, cornerRadius - 1); |
| |
| var upFillColors:Array; |
| var upFillAlphas:Array; |
| |
| var overFillColors:Array; |
| var overFillAlphas:Array; |
| |
| graphics.clear(); |
| |
| switch (name) |
| { |
| case "upSkin": |
| { |
| upFillColors = [ fillColors[0], fillColors[1] ]; |
| upFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| // button border/edge |
| drawRoundRect( |
| 0, 0, w, h, cr, |
| [ borderColor, borderColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ borderColor, borderColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h)); |
| |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, cr1, |
| upFillColors, upFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: h - 2, r: 0 }); |
| |
| // top highlight |
| drawRoundRect( |
| 1, 1, w - 2, (h - 2) / 2, |
| { tl: cr1, tr: cr1, bl: 0, br: 0 }, |
| [ 0xFFFFFF, 0xFFFFFF ], highlightAlphas, |
| verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: (h - 2) / 2, r: 0 }); |
| |
| // side bevel highlight edges |
| drawRoundRect( |
| dividerPosX - 1, 1, 1, h - 2, 0, |
| borderColor, 0); |
| drawRoundRect( |
| dividerPosX + 1, 1, 1, h - 2, 0, |
| borderColor, 0); |
| |
| break; |
| } |
| |
| case "overSkin": // for hover on the main button (left) side |
| { |
| upFillColors = [ fillColors[0], fillColors[1] ]; |
| upFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| if (fillColors.length > 2) |
| overFillColors = [ fillColors[2], fillColors[3] ]; |
| else |
| overFillColors = [ fillColors[0], fillColors[1] ]; |
| |
| if (fillAlphas.length > 2) |
| overFillAlphas = [ fillAlphas[2], fillAlphas[3] ]; |
| else |
| overFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| // button border/edge |
| drawRoundRect( |
| 0, 0, w, h, cr, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h)); |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, cr1, |
| upFillColors, upFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: h - 2, |
| r: getRadius(cr1, true) }); |
| |
| // left/main button fill |
| drawRoundRect( |
| 1, 1, w - arrowButtonWidth - 2, h - 2, |
| getRadius(cr1, true), |
| overFillColors, overFillAlphas, |
| verticalGradientMatrix(1, 1, dividerPosX - 2, h - 2)); |
| |
| // top highlight |
| drawRoundRect( |
| 1, 1, w - 2, (h -2) / 2, |
| { tl: cr1, tr: cr1, bl: 0, br: 0 }, |
| [ 0xFFFFFF, 0xFFFFFF ], highlightAlphas, |
| verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: (h - 2) / 2, r: 0 }); |
| |
| // side bevel highlight edges |
| drawRoundRect( |
| dividerPosX - 1, 1, 1, h - 2, 0, |
| themeColor, 0.35); |
| |
| break; |
| } |
| |
| case "popUpOverSkin": // for hover on the arrow-button (right) side |
| { |
| upFillColors = [ fillColors[0], fillColors[1] ]; |
| upFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| if (fillColors.length > 2) |
| overFillColors = [ fillColors[2], fillColors[3] ]; |
| else |
| overFillColors = [ fillColors[0], fillColors[1] ]; |
| |
| if (fillAlphas.length > 2) |
| overFillAlphas = [ fillAlphas[2], fillAlphas[3] ]; |
| else |
| overFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| // button border/edge |
| drawRoundRect( |
| 0, 0, w, h, cr, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h)); |
| |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, getRadius(cr1, true), |
| upFillColors, upFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: arrowButtonWidth - 1, h: h - 2, |
| r: getRadius(cr1, true) }); |
| |
| // right button fill |
| drawRoundRect( |
| dividerPosX + 1, 1, arrowButtonWidth - 2, h - 2, |
| getRadius(cr1, false), |
| overFillColors, overFillAlphas, |
| verticalGradientMatrix(dividerPosX, 0, |
| arrowButtonWidth - 1, h - 2)); |
| |
| // top highlight |
| drawRoundRect( |
| 1, 1, w - 2, (h -2) / 2, |
| { tl: cr1, tr: cr1, bl: 0, br: 0 }, |
| [ 0xFFFFFF, 0xFFFFFF ], highlightAlphas, |
| verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: (h - 2) / 2, r: 0 }); |
| |
| // side bevel highlight edges |
| drawRoundRect( |
| dividerPosX + 1, 1, 1, h - 2, 0, |
| themeColor, 0.35); |
| |
| break; |
| } |
| |
| case "downSkin": // for press on the main button (left) side |
| { |
| upFillColors = [ fillColors[0], fillColors[1] ]; |
| upFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| // button border/ddge |
| drawRoundRect( |
| 0, 0, w, h, cr, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h ), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h)); |
| |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, cr1, |
| upFillColors, upFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: h - 2, r: 0 }); |
| |
| // left/main button fill |
| drawRoundRect( |
| 1, 1, w - arrowButtonWidth - 2, h - 2, |
| getRadius(cr1, true), |
| [ derStyles.fillColorPress1, derStyles.fillColorPress2], 1, |
| verticalGradientMatrix(1, 1, dividerPosX - 2, h - 2)); |
| |
| // top highlight (checked, works) |
| drawRoundRect( |
| 1, 1, w - 2, (h -2) / 2, |
| { tl: cr1, tr: cr1, bl: 0, br: 0 }, |
| [ 0xFFFFFF, 0xFFFFFF ], highlightAlphas, |
| verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: (h -2) / 2, r: 0 }); |
| |
| // side bevel highlight edges |
| drawRoundRect( |
| dividerPosX - 1, 1, 1, h - 2, 0, |
| themeColorDrk1, 0.3); |
| drawRoundRect( |
| dividerPosX + 1, 1, 1, h - 2, 0, |
| borderColor, 0); |
| |
| break; |
| } |
| |
| case "popUpDownSkin": // for press on the arrow-button (right) side |
| { |
| upFillColors = [ fillColors[0], fillColors[1] ]; |
| upFillAlphas = [ fillAlphas[0], fillAlphas[1] ]; |
| |
| // button border/edge |
| drawRoundRect( |
| 0, 0, w, h, cr, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h ), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ themeColor, themeColorDrk1 ], 1, |
| verticalGradientMatrix(0, 0, w, h)); |
| |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, cr1, |
| upFillColors, upFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: h - 2, r: 0 }); |
| |
| // right button fill |
| drawRoundRect( |
| dividerPosX + 1, 1, arrowButtonWidth - 2, h - 2, |
| getRadius(cr1, false), |
| [ derStyles.fillColorPress1, |
| derStyles.fillColorPress2], 1, |
| verticalGradientMatrix(dividerPosX, 0, |
| arrowButtonWidth - 1, h - 2)); |
| |
| // top highlight |
| drawRoundRect( |
| 1, 1, w - 2, (h -2) / 2, |
| { tl: cr1, tr: cr1, bl: 0, br: 0 }, |
| [ 0xFFFFFF, 0xFFFFFF ], highlightAlphas, |
| verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2), |
| GradientType.LINEAR, null, |
| { x: dividerPosX, y: 1, w: 1, h: (h - 2) / 2, r: 0 }); |
| |
| // side bevel highlight edges |
| drawRoundRect( |
| dividerPosX - 1, 1, 1, h - 2, 0, |
| borderColor, 0); |
| drawRoundRect( |
| dividerPosX + 1, 1, 1, h - 2, 0, |
| themeColorDrk1, 0.3); |
| |
| break; |
| } |
| |
| case "disabledSkin": |
| { |
| arrowColor = getStyle("disabledIconColor"); |
| |
| var disFillColors:Array = [ fillColors[0], fillColors[1] ]; |
| |
| var disFillAlphas:Array = |
| [ Math.max(0, fillAlphas[0] - 0.15), |
| Math.max(0, fillAlphas[1] - 0.15) ]; |
| |
| // outer edge |
| drawRoundRect( |
| 0, 0, w, h, cornerRadius, |
| [ borderColor, borderColorDrk1 ], 0.5, |
| verticalGradientMatrix(0, 0, w, h ), |
| GradientType.LINEAR, null, |
| { x: 1, y: 1, w: w - 2, h: h - 2, r: cornerRadius - 1 }); |
| |
| drawRoundRect( |
| dividerPosX, 1, 1, h - 2, 0, |
| [ borderColor, borderColorDrk1 ], 0.5); |
| |
| // button fill |
| drawRoundRect( |
| 1, 1, w - 2, h - 2, cr1, |
| disFillColors, disFillAlphas, |
| verticalGradientMatrix(1, 1, w - 2, h - 2), |
| null, null, |
| { x: dividerPosX, y: 1, w: 1, h: h - 2, r: 0 }); |
| |
| break; |
| } |
| } |
| |
| if (popUpIcon is PopUpIcon) |
| PopUpIcon(popUpIcon).arrowColor = arrowColor; |
| } |
| |
| //-------------------------------------------------------------------------- |
| // |
| // Methods |
| // |
| //-------------------------------------------------------------------------- |
| |
| /** |
| * @private |
| */ |
| private function getRadius(r:Number, left:Boolean):Object |
| { |
| return left ? |
| { br: 0, bl: r, tr: 0, tl: r } : |
| { br: r, bl: 0, tr: r, tl: 0 }; |
| } |
| } |
| |
| } |