blob: f8d51c918aa28da0d817fa2f983b907b592a99a4 [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 samples.flexstore
{
import flash.display.GradientType;
import mx.containers.BoxDirection;
import mx.controls.Button;
import mx.controls.ButtonBar;
import mx.skins.Border;
import mx.skins.halo.*;
import mx.styles.StyleManager;
import mx.utils.ColorUtil;
/**
* Adapted from mx.skins.halo.ButtonBarButtonSkin.
* This version of the ButtonBarButtonSkin is applied for the
* selectedOver, selectedUp, and over states to use the 2nd two
* values of the fillColors for the selected state of the
* button. The over state then uses a computed value from
* the themeColor to show emphasis. The border of the selected
* button also uses a computed value from the themeColor, but
* is partially transparent.
*/
public class ButtonBarButtonSkin extends Border
{
//--------------------------------------------------------------------------
//
// 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);
// Button-specific styles.
o.innerEdgeColor1 = ColorUtil.adjustBrightness2(fillColor0, -10);
o.innerEdgeColor2 = ColorUtil.adjustBrightness2(fillColor1, -25);
}
return cache[key];
}
//--------------------------------------------------------------------------
//
// Constructor
//
//--------------------------------------------------------------------------
/**
* @private
* Constructor.
*/
public function ButtonBarButtonSkin()
{
super();
}
//--------------------------------------------------------------------------
//
// Overridden properties
//
//--------------------------------------------------------------------------
//----------------------------------
// measuredWidth
//----------------------------------
/**
* @private
*/
override public function get measuredWidth():Number
{
return 50;
}
//----------------------------------
// measuredHeight
//----------------------------------
/**
* @private
*/
override public function get measuredHeight():Number
{
return 22;
}
//--------------------------------------------------------------------------
//
// Overridden methods
//
//--------------------------------------------------------------------------
/**
* @private
*/
override protected function updateDisplayList(w:Number, h:Number):void
{
super.updateDisplayList(w, h);
// User-defined styles.
var borderColor:uint = getStyle("borderColor");
var cornerRadius:Number = getStyle("cornerRadius");
var fillAlphas:Array = getStyle("fillAlphas");
var fillColors:Array = getStyle("fillColors");
StyleManager.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 emph:Boolean = false;
if (parent is Button)
emph = (parent as Button).emphasized;
var tmp:Number;
var bar:ButtonBar = parent ? ButtonBar(parent.parent) : null;
var horizontal:Boolean = true;
var pos:int = 0;
if (bar)
{
if (bar.direction == BoxDirection.VERTICAL)
horizontal = false;
// first: -1, middle: 0, last: 1
var index:int = bar.getChildIndex(parent);
pos = (index == 0 ? -1 : (index == bar.numChildren - 1 ? 1 : 0));
}
var radius:Object = getCornerRadius(pos, horizontal, cornerRadius);
var cr:Object = getCornerRadius(pos, horizontal, cornerRadius);
var cr1:Object = getCornerRadius(pos, horizontal, cornerRadius - 1);
var cr2:Object = getCornerRadius(pos, horizontal, cornerRadius - 2);
var cr3:Object = getCornerRadius(pos, horizontal, cornerRadius - 3);
graphics.clear();
switch (name)
{
case "selectedUpSkin":
case "selectedOverSkin":
{
var overFillColors:Array;
if (fillColors.length > 2)
overFillColors = [ fillColors[2], fillColors[3] ];
else
overFillColors = [ fillColors[0], fillColors[1] ];
var overFillAlphas:Array;
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, derStyles.themeColDrk1 ], 0.5,
verticalGradientMatrix(0, 0, w , h),
GradientType.LINEAR, null,
{ x: 1, y: 1, w: w - 2, h: h - 2, r: cr1 });
// button fill
drawRoundRect(
1, 1, w - 2, h - 2, cr1,
overFillColors, overFillAlphas,
verticalGradientMatrix(0, 0, w - 2, h - 2));
// top highlight
if (!(radius is Number))
{ radius.bl = radius.br = 0;}
drawRoundRect(
1, 1, w - 2, (h - 2) / 2, radius,
[ 0xFFFFFF, 0xFFFFFF ], highlightAlphas,
verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2));
break;
}
case "overSkin":
{
// button border/edge
drawRoundRect(
0, 0, w, h, cr,
[ themeColor, derStyles.themeColDrk1 ], 0.5,
verticalGradientMatrix(0, 0, w, h));
// button fill
drawRoundRect(
1, 1, w - 2, h - 2, cr1,
[ derStyles.fillColorPress1, derStyles.fillColorPress2 ], 1,
verticalGradientMatrix(0, 0, w - 2, h - 2));
// top highlight
if (!(radius is Number))
{ radius.bl = radius.br = 0;}
drawRoundRect(
1, 1, w - 2, (h - 2) / 2, radius,
[ 0xFFFFFF, 0xFFFFFF ], highlightAlphas,
verticalGradientMatrix(1, 1, w - 2, (h - 2) / 2));
break;
}
}
}
//--------------------------------------------------------------------------
//
// Methods
//
//--------------------------------------------------------------------------
/**
* @private
*/
private function getCornerRadius(pos:int, horizontal:Boolean,
radius:Number):Object
{
if (pos == 0)
return 0;
radius = Math.max(0, radius);
if (horizontal)
{
if (pos == -1)
return { tl: radius, tr: 0, bl: radius, br: 0 };
else // pos == 1
return { tl: 0, tr: radius, bl: 0, br: radius };
}
else
{
if (pos == -1)
return { tl: radius, tr: radius, bl: 0, br: 0 };
else // pos == 1
return { tl: 0, tr: 0, bl: radius, br: radius };
}
}
}
}