/* Copyright (c) 2007 FlexLib Contributors. See: http://code.google.com/p/flexlib/wiki/ProjectContributors 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 flexlib.containers { import flash.display.DisplayObject; import flash.events.Event; import flash.events.MouseEvent; import flexlib.controls.ScrollableArrowMenu; import flexlib.controls.SuperTabBar; import flexlib.controls.tabBarClasses.SuperTab; import flexlib.events.SuperTabEvent; import flexlib.events.TabReorderEvent; import flexlib.skins.TabPopUpButtonSkin; import mx.collections.ArrayCollection; import mx.collections.IList; import mx.containers.Box; import mx.containers.BoxDirection; import mx.containers.Canvas; import mx.containers.TabNavigator; import mx.containers.ViewStack; import mx.controls.Menu; import mx.controls.PopUpButton; import mx.core.Container; import mx.core.EdgeMetrics; import mx.core.ScrollPolicy; import mx.effects.Tween; import mx.events.ChildExistenceChangedEvent; import mx.events.IndexChangedEvent; import mx.events.MenuEvent; import mx.styles.CSSStyleDeclaration; import mx.styles.StyleManager; [IconFile("SuperTabNavigator.png")] /** * Fired when the close button of a tab is clicked. To stop the default action, which will remove the * child from the collection of children, call event.preventDefault() in your listener. */ [Event(name="tabClose", type="flexlib.events.SuperTabEvent")] [Event(name="tabsReordered", type="flexlib.events.TabReorderEvent")] /** * Name of CSS style declaration that specifies style for the Button that appears * on the right side of the tab bar. * * @default undefined */ [Style(name="popupButtonStyleName", type="String", inherit="no")] /** * Name of CSS style declaration that specifies style for button used to control the horizontal * scrolling of the tabs. This button appears on the right-most side of the tab bar. * * @default undefined */ [Style(name="leftButtonStyleName", type="String", inherit="no")] /** * Name of CSS style declaration that specifies style for button used to control the horizontal * scrolling of the tabs. This button appears on the left-most side of the tab bar. * * @default undefined */ [Style(name="rightButtonStyleName", type="String", inherit="no")] /** * The SuperTabNavigator is an extension of the TabNavigator navigation * container. * *
The SuperTabNavigator functions exactly like the TabNavigator, but * adds some functionality. Added functionality includes:
*These features make the SuperTabNavigator function much more like the * tabs in Firefox.
* * @mxml * *The <mx:SuperTabNavigator> tag inherits all of the
* tag attributes of the TabNavigator,
* and adds the following tag attributes:
* <mx:SuperTabNavigator * Styles * popupButtonStyleName="Value of the* * @see mx.containers.TabNavigator * @see flexlib.controls.SuperTabBar */ public class SuperTabNavigator extends TabNavigator { private var stopIndexChangeEvent:Boolean = false; /** * @private * * holder is a Box component that we are going to use to hold the scrolling * canvas and the Button. I've left this Box, instead of HBox * or VBox because I intend to extend this component to allow for vertical * tabs along the side. */ protected var holder:Box; /** * @private * * canvas is the Canvas component that will hold the TabBar. Tis is used * so we get access to the scrolling functionality of a container. To do * the scrolling we're going to use the horizontalScrollPosition of the * Canvas. */ protected var canvas:ButtonScrollingCanvas; /** * @private */ protected var popupButton:PopUpButton; /** * @private */ protected var menu:Menu; /** * @private */ private var forcedTabWidth:Number = -1; /** * @private */ private var originalTabWidthStyle:Number = -1; /** * @private */ private var _minTabWidth:Number = 60; /** * Static variable indicating the Button will be shown if there is more * than one tab in the SuperTabNavigator. * Used to set popUpButtonPolicy. */ public static var POPUPPOLICY_AUTO:String = "auto"; /** * Static variable indicating the Button will always be shown to the * right of the tab bar, no matter how many tabs are open. * Used to set popUpButtonPolicy. */ public static var POPUPPOLICY_ON:String = "on"; /** * Static variable indicating the Button will never be shown to the * right of the tab bar. * Used to set popUpButtonPolicy. */ public static var POPUPPOLICY_OFF:String = "off"; [Embed (source="../assets/assets.swf", symbol="indicator")] private static var DEFAULT_INDICATOR:Class; [Embed (source="../assets/assets.swf", symbol="firefox_close_up")] private static var DEFAULT_CLOSE_UP:Class; [Embed (source="../assets/assets.swf", symbol="firefox_close_over")] private static var DEFAULT_CLOSE_OVER:Class; [Embed (source="../assets/assets.swf", symbol="firefox_close_down")] private static var DEFAULT_CLOSE_DOWN:Class; [Embed (source="../assets/assets.swf", symbol="firefox_close_disabled")] private static var DEFAULT_CLOSE_DISABLED:Class; [Embed (source="../assets/assets.swf", symbol="left_arrow")] private static var DEFAULT_LEFT_BUTTON:Class; [Embed (source="../assets/assets.swf", symbol="right_arrow")] private static var DEFAULT_RIGHT_BUTTON:Class; /** * @private * Our internal variable to keep track of the policy to show the Button */ private var _popUpButtonPolicy:String; /** * Either POPUPPOLICY.AUTO, POPUPOLICY_ON, or POPUPPOLICY_OFF. *popupButtonStyleNameproperty" * leftButtonStyleName="Value of theleftButtonStyleNameproperty" * rightButtonStyleName="Value of therightButtonStyleNameproperty" * * Properties * popUpButtonPolicy="on|auto|off" * startScrollingEvent="MouseEvent.MOUSE_DOWN|MouseEvent.MOUSE_OVER" * stopScrollingEvent="MouseEvent.MOUSE_UP|MouseEvent.MOUSE_OUT" * scrollSpeed="100" * dragEnabled="true|false" * dropEnabled="true|false" * minTabWidth="60" * * > * ... * child tags * ... * </mx:SuperTabNavigator> *
Indicates how the Button to the right of the tabs should * be shown. AUTO means the button will be shown if there is more than * one tab. ON means it will always be shown, and OFF means it will never * be shown. */ public function get popUpButtonPolicy():String { return _popUpButtonPolicy; } /** * @private */ public function set popUpButtonPolicy(value:String):void { var old:String = this._popUpButtonPolicy; this._popUpButtonPolicy = value; if(old != value) { this.invalidateDisplayList(); } } /** * @private */ private var _startScrollingEvent:String = MouseEvent.MOUSE_DOWN; public function get startScrollingEvent():String { if(canvas) { return canvas.startScrollingEvent; } else { return _startScrollingEvent; } } /** * @private */ public function set startScrollingEvent(value:String):void { _startScrollingEvent = value; if(canvas) { canvas.startScrollingEvent = value; } } /** * @private */ private var _stopScrollingEvent:String = MouseEvent.MOUSE_UP; public function get stopScrollingEvent():String { if(canvas) { return canvas.stopScrollingEvent; } else { return _stopScrollingEvent; } } /** * @private */ public function set stopScrollingEvent(value:String):void { _stopScrollingEvent = value; canvas.stopScrollingEvent = value; } /** * @private */ private var _scrollSpeed:Number = 100; /** * The delay in milliseconds between scrolling the tabs. *
The smaller the number here the faster the scrolling speed.
*/ public function set scrollSpeed(value:Number):void { if(canvas) { canvas.scrollSpeed = value; } _scrollSpeed = value; } /** * @private */ public function get scrollSpeed():Number { if(canvas) return canvas.scrollSpeed; return _scrollSpeed; } /** * @private */ private var _dragEnabled:Boolean = true; /** * Boolean indicating whether or not this SuperTabNavigator allows * tabs to be dragged from the tab bar. *If both dragEnabled and dropEnabled are true then the SuperTabNavigator allows * reordering of tabs by drag and drop
* * @default true */ public function get dragEnabled():Boolean { if(tabBar) { return (tabBar as SuperTabBar).dragEnabled; } else { return _dragEnabled; } } /** * @private */ public function set dragEnabled(value:Boolean):void { _dragEnabled = value; if(tabBar) { (tabBar as SuperTabBar).dragEnabled = value; } } /** * @private */ private var _dropEnabled:Boolean = true; /** * Boolean indicating whether or not this SuperTabNavigator allows * tabs to be dropped on the tab bar. *If both dragEnabled and dropEnabled are true then the SuperTabNavigator allows * reordering of tabs by drag and drop
* * @default true */ public function get dropEnabled():Boolean { if(tabBar) { return (tabBar as SuperTabBar).dropEnabled; } else { return _dropEnabled; } } /** * @private */ public function set dropEnabled(value:Boolean):void { _dropEnabled = value; if(tabBar) { (tabBar as SuperTabBar).dropEnabled = value; } } /** * The minimum tab width allowed to display tabs. *If tabs cannot fit at their default size, then they are shrunk until they * reach minTabWidth. If they still cannot fit then they remain at minTabWidth and * the SuperTabBar scrolls the tabs.
* * @default 60 */ public function get minTabWidth():Number { return _minTabWidth; } /** * @private */ public function set minTabWidth(value:Number):void { this._minTabWidth = value; this.invalidateDisplayList(); } /** * @private */ private var _editableTabLabels:Boolean=false; /** * Boolean indicating if tab labels can be edited. If true, the user can double click on * a tab label and edit the label. */ public function get editableTabLabels():Boolean { return _editableTabLabels; } /** * @private */ public function set editableTabLabels(value:Boolean):void { this._editableTabLabels = value; if(tabBar) { (tabBar as SuperTabBar).editableTabLabels = value; } } /** * @private */ private var _allowTabSqueezing:Boolean = true; /** * Boolean indicating if tab width should be adjusted to try to squeeze all tabs so we don't need * scrolling. If true, tabs will be squeezed until they reach theminTabWidth value, at which
* point they scroll. If false, tabs will never be resized smaller than their default widths.
*
* @default true
*/
public function get allowTabSqueezing():Boolean {
return _allowTabSqueezing;
}
/**
* @private
*/
public function set allowTabSqueezing(value:Boolean):void {
_allowTabSqueezing = value;
invalidateDisplayList();
}
/**
* @private
*/
private static function initializeStyles():void
{
var selector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("SuperTabNavigator");
if(!selector)
{
selector = new CSSStyleDeclaration();
}
selector.defaultFactory = function():void
{
this.popupButtonStyleName = "popupButton";
this.tabStyleName = "superTab";
this.leftButtonStyleName = "leftButton";
this.rightButtonStyleName = "rightButton";
this.buttonWidth = 20;
}
// Style for the PopUpButton on the right
var popupButtonStyleName:String = selector.getStyle("popupButtonStyleName");
var pSelector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("." + popupButtonStyleName);
if(!pSelector)
{
pSelector = new CSSStyleDeclaration();
}
pSelector.defaultFactory = function():void
{
this.upSkin = TabPopUpButtonSkin;
this.popUpDownSkin = TabPopUpButtonSkin;
this.popUpOverSkin = TabPopUpButtonSkin;
this.disabledSkin = TabPopUpButtonSkin;
}
StyleManager.setStyleDeclaration("." + popupButtonStyleName, pSelector, false);
// Style for the SuperTabs
var tabStyleName:String = selector.getStyle("tabStyleName");
var tSelector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("." + tabStyleName);
if(!tSelector)
{
tSelector = new CSSStyleDeclaration();
}
tSelector.defaultFactory = function():void
{
this.textAlign = "left";
this.indicatorClass = DEFAULT_INDICATOR;
this.tabCloseButtonStyleName = "tabCloseButton";
}
StyleManager.setStyleDeclaration("." + tabStyleName, tSelector, false);
// Style for the close button on each tab
var tabCloseStyleName:String = tSelector.getStyle("tabCloseButtonStyleName");
var cSelector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("." + tabCloseStyleName);
if(!cSelector)
{
cSelector = new CSSStyleDeclaration();
}
cSelector.defaultFactory = function():void
{
this.upSkin = DEFAULT_CLOSE_UP;
this.overSkin = DEFAULT_CLOSE_OVER;
this.downSkin = DEFAULT_CLOSE_DOWN;
this.disabledSkin = DEFAULT_CLOSE_DISABLED;
}
StyleManager.setStyleDeclaration("." + tabCloseStyleName, cSelector, false);
// Style for the left arrow for tab scrolling
var leftStyleName:String = selector.getStyle("leftButtonStyleName");
var lSelector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("." + leftStyleName);
if(!lSelector)
{
lSelector = new CSSStyleDeclaration();
}
lSelector.defaultFactory = function():void
{
this.icon = DEFAULT_LEFT_BUTTON;
this.fillAlphas = [1,1,1,1];
this.cornerRadius = 0;
}
StyleManager.setStyleDeclaration("." + leftStyleName, lSelector, false);
// Style for the right arrow for tab scrolling
var rightStyleName:String = selector.getStyle("rightButtonStyleName");
var rSelector:CSSStyleDeclaration = StyleManager.getStyleDeclaration("." + rightStyleName);
if(!rSelector)
{
rSelector = new CSSStyleDeclaration();
}
rSelector.defaultFactory = function():void
{
this.icon = DEFAULT_RIGHT_BUTTON;
this.fillAlphas = [1,1,1,1];
this.cornerRadius = 0;
}
StyleManager.setStyleDeclaration("." + rightStyleName, rSelector, false);
StyleManager.setStyleDeclaration("SuperTabNavigator", selector, false);
}
initializeStyles();
override public function initialize():void {
super.initialize();
SuperTabNavigator.initializeStyles();
}
/**
* @private
*/
override protected function createChildren():void
{
if (!tabBar){
// We're using our custom SuperTabBar class instead of TabBar
tabBar = createTabBar();
tabBar.name = "tabBar";
tabBar.focusEnabled = false;
tabBar.styleName = this;
(tabBar as SuperTabBar).dragEnabled = this._dragEnabled;
(tabBar as SuperTabBar).dropEnabled = this._dropEnabled;
(tabBar as SuperTabBar).editableTabLabels = this._editableTabLabels;
(tabBar as SuperTabBar).closePolicy = _closePolicy;
tabBar.setStyle("borderStyle", "none");
tabBar.setStyle("paddingTop", 0);
tabBar.setStyle("paddingBottom", 0);
//this is a bit of a hack. We don't want tabBar to inherit the left, right, top, or bottom
//styles from SuperTabNavigator. So we disable the use of left, right, top, or bottom on SuperTabBar
//if it's used in SuperTabNavigator
//fixes issue 29: http://code.google.com/p/flexlib/issues/detail?id=29
tabBar.setStyle("left", NaN);
tabBar.setStyle("right", NaN);
tabBar.setStyle("top", NaN);
tabBar.setStyle("bottom", NaN);
tabBar.setStyle("horizontalCenter", NaN);
(tabBar as SuperTabBar).addEventListener(SuperTabEvent.TAB_CLOSE, handleTabClose);
}
// We need to create our tabBar above BEFORE calling createChildren
// because otherwise it would get created in the super class.
// Once we create it then the super class will skip it. It still hasn't
// been added as a child however (this gets done below).
super.createChildren();
if(!holder) {
// Why not just use HBox? Because in the future we might want
// to use a VBox for vertical tabs. This lets a subclass simply
// change the direction.
holder = new Box();
holder.direction = BoxDirection.HORIZONTAL;
holder.setStyle("horizontalGap", 0);
holder.setStyle("borderStyle", "none");
holder.setStyle("paddingTop", 0);
holder.setStyle("paddingBottom", 0);
holder.horizontalScrollPolicy = "off";
rawChildren.addChild(holder);
}
if(!canvas) {
canvas = new ButtonScrollingCanvas();
canvas.styleName = this;
canvas.setStyle("borderStyle", "none");
canvas.setStyle("backgroundAlpha", 0);
canvas.setStyle("paddingTop", 0);
canvas.setStyle("paddingBottom", 0);
canvas.startScrollingEvent = _startScrollingEvent;
canvas.stopScrollingEvent = _stopScrollingEvent;
canvas.scrollSpeed = _scrollSpeed;
// So we can see our child heirarchy:
// holder (Box) -> canvas (ButtonScrollingCanvas) -> tabBar (SuperTabBar)
canvas.addChild(tabBar);
holder.addChild(canvas);
}
// We create the menu once. This doesn't get shown until we click
// the Button. But it can get created here so we don't have
// to create it every time.
if(!menu) {
//menu = new Menu();
menu = new ScrollableArrowMenu();
menu.setStyle("textAlign", "left");
// If we wanted to change the scroll policy for the scrolling menu we
// could modify the following two lines. For example, turning
// verticalScrollPolicy to OFF will remove the side scrollbars and leave
// just the arrow buttons on top and bottom.
menu.verticalScrollPolicy = ScrollPolicy.AUTO;
(menu as ScrollableArrowMenu).arrowScrollPolicy = ScrollPolicy.AUTO;
menu.addEventListener(MenuEvent.ITEM_CLICK, changeTabs);
}
if(!popupButton) {
popupButton = new PopUpButton();
popupButton.popUp = menu;
popupButton.width = 18;
popupButton.styleName = getStyle("popupButtonStyleName");
// So now holder has 2 children: canvas and popupButton
holder.addChild(popupButton);
}
tabBar.addEventListener(ChildExistenceChangedEvent.CHILD_ADD, tabsChanged);
tabBar.addEventListener(ChildExistenceChangedEvent.CHILD_REMOVE, tabsChanged);
SuperTabBar(tabBar).addEventListener(SuperTabEvent.TAB_UPDATED, tabsChanged);
// This is a custom event that gets fired from SuperTabBar if the tabs are
// dragged and reordered.
tabBar.addEventListener(SuperTabBar.TABS_REORDERED, tabsReordered);
this.addEventListener(IndexChangedEvent.CHANGE,tabChangedEvent);
invalidateSize();
this.addEventListener(IndexChangedEvent.CHILD_INDEX_CHANGE,indexChangeListener);
}
/**
* For extensibility, if you want to use a custom extended version of SuperTabBar, you can do so by
* overriding the createTabBar method and returning an instance of any class that extends
* SuperTabBar.
*/
protected function createTabBar():SuperTabBar {
return new SuperTabBar();
}
private function indexChangeListener(event:IndexChangedEvent):void {
if(stopIndexChangeEvent) {
stopIndexChangeEvent = false;
event.stopImmediatePropagation();
}
}
private function handleTabClose(event:SuperTabEvent):void {
var clone:SuperTabEvent = event.clone() as SuperTabEvent;
dispatchEvent(clone);
if(clone.isDefaultPrevented()) {
event.preventDefault();
}
}
private var _closePolicy:String = SuperTab.CLOSE_ROLLOVER;
/**
* The close policy for tabs.
* @see flexlib.controls.SuperTabBar
*/
public function get closePolicy():String {
return _closePolicy;
}
/**
* @private
*/
public function set closePolicy(value:String):void {
if(_closePolicy != value) {
_closePolicy = value;
if(tabBar != null) {
(tabBar as SuperTabBar).closePolicy = value;
}
invalidateDisplayList();
}
}
public function setClosePolicyForTab(index:int, value:String):void {
(tabBar as SuperTabBar).setClosePolicyForTab(index, value);
}
/**
* @private
*/
private function tabsReordered(event:TabReorderEvent):void {
var clone:Event = event.clone();
dispatchEvent(clone);
if(clone.isDefaultPrevented()) {
return;
}
// The relatedObject of our custom event is the SuperTabBar component
// where the tab originated. This is so we can properly move tabs from
// one navigator to another.
var sourceBar:SuperTabBar = event.relatedObject as SuperTabBar;
var droppedTab:SuperTab = (sourceBar.getChildAt(event.oldIndex) as SuperTab);
// The oldIndex property of the event specifies the index of the tab
// in the original navigator. Note that the tab might not be a child of
// this current tab navigator that we're in (ie sourceNav might not == this).
var child:Object;
var curDataProvider:Object = sourceBar.dataProvider;
var lastClosePolicy:String = sourceBar.getClosePolicyForTab(event.oldIndex);
var lastStyleName:Object = droppedTab.styleName;
if(sourceBar != tabBar) {
if(curDataProvider is IList){
child = (curDataProvider as IList).getItemAt(event.oldIndex);
curDataProvider.removeItemAt(event.oldIndex);
var newChild:Canvas = new Canvas();
if(child.hasOwnProperty("label")) {
newChild.label = child.label;
}
else {
newChild.label = child.toString();
}
if(child.hasOwnProperty("icon")) {
newChild.icon = child.icon;
}
child = newChild;
}
else if(curDataProvider is ViewStack){
child = (curDataProvider as ViewStack).getChildAt(event.oldIndex);
curDataProvider.removeChildAt(event.oldIndex);
}
if(child is DisplayObject) {
// We add the tab to ourself at the new index position
this.addChildAt(child as DisplayObject, event.newIndex);
}
this.setClosePolicyForTab(event.newIndex, lastClosePolicy);
droppedTab = this.getTabAt(event.newIndex) as SuperTab;
droppedTab.styleName = lastStyleName;
}
else {
//If we're simply moving a tab within the same SuperTabNavigator
//then we'll just use the setChildIndex method. There's a sneaky thing
//though, maybe a bug, which dispatches the wrong indexes in the
//IndexChangeEvent. So we dispatch our own event with the correct indexes,
//and we catch the event that gets dispatched with setChildIndex and stop
//it from propogating. I don't know why there's a problem with the
//setChildAt method dispatching an event with an incorrect oldIndex, but
//god it was hard to figure this one out.
if(event.oldIndex < event.newIndex) {
event.newIndex--;
}
var devent:IndexChangedEvent =
new IndexChangedEvent(IndexChangedEvent.CHILD_INDEX_CHANGE);
devent.oldIndex = event.oldIndex;
devent.newIndex = event.newIndex;
devent.relatedObject = getChildAt(event.newIndex);
dispatchEvent(devent);
stopIndexChangeEvent = true;
this.setChildIndex(getChildAt(event.oldIndex), event.newIndex);
}
// Calling validateNow before calling selectedIndex makes sure we
// don't get a little display bug that tends to creep up
this.validateNow();
// If we just dropped a tab then we want to select it,
// that just seems like the intuitive thing to do
this.selectedIndex = event.newIndex;
// Now update the drop-down menu to show the newly ordered tabs
reorderTabList();
}
/**
* @private
*
* tabBarHeight is the same as the same funtion in TabNavigator, but the
* one in TabNavigator was private, so we had to reproduce it here.
*/
protected function get tabBarHeight():Number
{
var tabHeight:Number = getStyle("tabHeight");
if (isNaN(tabHeight))
tabHeight = tabBar.getExplicitOrMeasuredHeight();
return tabHeight - 1;
}
override protected function updateDisplayList(unscaledWidth:Number,
unscaledHeight:Number):void
{
//We need to calculate the tab widths first, so we call super.updateDisplayList later
// Are we supposed to be showing the Button?
if(_popUpButtonPolicy == SuperTabNavigator.POPUPPOLICY_AUTO) {
popupButton.includeInLayout = popupButton.visible = this.numChildren > 1;
}
else if(_popUpButtonPolicy == SuperTabNavigator.POPUPPOLICY_ON) {
popupButton.includeInLayout = popupButton.visible = true;
}
else if(_popUpButtonPolicy == SuperTabNavigator.POPUPPOLICY_OFF) {
popupButton.includeInLayout = popupButton.visible = false;
}
var vm:EdgeMetrics = viewMetrics;
var w:Number = unscaledWidth - vm.left - vm.right;
var th:Number = tabBarHeight + 1;
// tabBarSpace is used to try to figure out how much space we
// need for the tabs, to figure out if we need to scroll them
var tabBarSpace:Number = w;
if(popupButton.includeInLayout) {
tabBarSpace -= popupButton.width;
}
var tempWidth:Number = tabBar.getStyle("tabWidth");
tabBar.clearStyle("tabWidth");
tabBar.setStyle("tabWidth", NaN);
tabBar.validateNow();
var pw:Number = Math.max(tabBar.getExplicitOrMeasuredWidth(), tabBarSpace);
tabBar.setStyle("tabWidth", tempWidth);
var tabBarWidth:Number = tabBar.measuredWidth;
holder.move(0, 1);
holder.setActualSize(unscaledWidth, th);
var canvasWidth:Number = unscaledWidth;
if(popupButton.includeInLayout) {
canvasWidth -= popupButton.width;
}
canvas.width = canvasWidth;
canvas.height = th;
canvas.explicitButtonHeight = th - 1;
canvas.verticalScrollPolicy = ScrollPolicy.OFF;
if(pw <= canvasWidth) {
canvas.horizontalScrollPosition = 0;
}
//now we're good to go with the tab widths, so this should be OK
//if we called this first we would see some flickering of the tabs
//as they are resized quickly. No good.
super.updateDisplayList(unscaledWidth, unscaledHeight);
//OK, so one problem: in the call to super.updateDisplayList() the width of tabBar gets reset.
//that's crap, because we need to set it to what we set it to, damnit. But we need to call super.updateDisplayList
//or the component won't get drawn right at all. And we can't call super.super.updateDisplayList, so we
//have to reset the width of tabBar after our call to updateDisplayList
if(tabBarWidth >= canvasWidth) {
if(_allowTabSqueezing) {
var measuredTabBarWidth:Number = tabBar.numChildren * _minTabWidth + (tabBar.numChildren-1) * getStyle("horizontalGap");
tabBar.width = Math.max(canvasWidth, measuredTabBarWidth);
}
else {
tabBar.width = tabBarWidth;
}
}
else {
tabBar.width = tabBarWidth;
}
/* we only care about horizontalAlign if we're not taking up too
much space already */
if(tabBarWidth < canvasWidth) {
switch (getStyle("horizontalAlign")) {
case "left":
tabBar.move(0, tabBar.y);
break;
case "right":
tabBar.move(unscaledWidth - tabBar.width, tabBar.y);
break;
case "center":
tabBar.move((unscaledWidth - tabBar.width) / 2, tabBar.y);
}
}
}
override public function styleChanged(styleProp:String):void {
super.styleChanged(styleProp);
if(styleProp == "tabWidth" && forcedTabWidth == -1) {
originalTabWidthStyle = this.getStyle(styleProp);
}
}
/**
* @private
*
* This is the event handler for when the user clicks the drop-down menu.
* If we're selecting a new tab then we'll dispatch an IndexChangedEvent, which
* will trigger the call to ensureTabIsVisible(). If we are not switching tabs
* though, then the event wouldn't get dispatched, so we have to call ensureTabIsVisible
*/
private function changeTabs(event:MenuEvent):void {
if(this.selectedIndex == event.index) {
ensureTabIsVisible();
}
this.selectedIndex = event.index;
}
/**
* @private
*
* The tabs can be changed any number of ways (via drop-down menu, via AS, etc)
* so this listener function will make sure that the tab that gets selected is
* visible.
*/
private function tabChangedEvent(event:IndexChangedEvent):void {
callLater(ensureTabIsVisible);
}
/**
* @private
*
* Check to make sure that the currently selected tab is visible. This means
* that we might have to scroll the canvas component so the tab comes into view
*/
private function ensureTabIsVisible():void {
var tab:DisplayObject = this.tabBar.getChildAt(this.selectedIndex);
var newHorizontalPosition:Number;
if(tab.x + tab.width > this.canvas.horizontalScrollPosition + this.canvas.width) {
newHorizontalPosition = tab.x - canvas.width + tab.width + canvas.getStyle("buttonWidth");
}
else if(this.canvas.horizontalScrollPosition > tab.x) {
newHorizontalPosition = tab.x - canvas.getStyle("buttonWidth");
}
else {
newHorizontalPosition = canvas.horizontalScrollPosition;
}
if(newHorizontalPosition) {
// We tween the motion so it looks super sweet
var tween:Tween = new Tween(this, canvas.horizontalScrollPosition, newHorizontalPosition, 500);
// Alternatively if we didn't want to use the tweening we could just set the
// horizontalScrollPosition right away (this is what I first did)
//canvas.horizontalScrollPosition = newHorizontalPosition;
}
}
/**
* @private
*/
public function onTweenUpdate(val:Object):void {
canvas.horizontalScrollPosition = val as Number;
}
/**
* @private
*/
public function onTweenEnd(val:Object):void {
canvas.horizontalScrollPosition = val as Number;
}
/**
* @private
*
* Listener that gets caled when a tab is added or removed.
*/
private function tabsChanged(event:Event):void {
callLater(reorderTabList);
}
/**
* @private
*
* reorderTabList loops over all the tabs and makes sure that the drop-down
* list is correct. This should get called every time tabs are added, removed,
* or re-ordered. This is a public method so other SuperTabNavigators can call
* this method on each other if we're dragging tabs from one SuperTabNavigator
* to another.
*/
public function reorderTabList():void {
var popupMenuDP:ArrayCollection = new ArrayCollection();
for(var i:int=0; i