//////////////////////////////////////////////////////////////////////////////// // // 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.managers { import flash.accessibility.Accessibility; import flash.accessibility.AccessibilityProperties; import flash.display.DisplayObject; import flash.display.DisplayObjectContainer; import flash.display.Graphics; import flash.display.InteractiveObject; import flash.display.Sprite; import flash.display.Stage; import flash.events.Event; import flash.events.EventDispatcher; import flash.events.MouseEvent; import flash.geom.Point; import flash.geom.Rectangle; import flash.system.Capabilities; import mx.automation.IAutomationObject; import mx.core.FlexGlobals; import mx.core.FlexSprite; import mx.core.FlexVersion; import mx.core.IChildList; import mx.core.IFlexDisplayObject; import mx.core.IFlexModule; import mx.core.IFlexModuleFactory; import mx.core.IInvalidating; import mx.core.ILayoutDirectionElement; import mx.core.IUIComponent; import mx.core.LayoutDirection; import mx.core.UIComponent; import mx.core.UIComponentGlobals; import mx.core.mx_internal; import mx.effects.Blur; import mx.effects.Fade; import mx.effects.IEffect; import mx.events.DynamicEvent; import mx.events.EffectEvent; import mx.events.FlexEvent; import mx.events.FlexMouseEvent; import mx.events.Request; import mx.managers.systemClasses.ActiveWindowManager; import mx.styles.IStyleClient; use namespace mx_internal; [ExcludeClass] /** * @private * The PopUpManager singleton class creates new top-level windows and * places or removes those windows from the layer on top of all other * visible windows. See the SystemManager for a description of the layering. * It is used for popup dialogs, menus, and dropdowns in the ComboBox control * and in similar components. * *
The PopUpManager also provides modality, so that windows below the popup * cannot receive mouse events, and also provides an event if the user clicks * the mouse outside the window so the developer can choose to dismiss * the window or warn the user.
* * @see PopUpManagerChildList */ public class PopUpManagerImpl extends EventDispatcher implements IPopUpManager { include "../core/Version.as"; //-------------------------------------------------------------------------- // // Class variables // //-------------------------------------------------------------------------- /** * @private */ private static var instance:IPopUpManager; /** * @private * * Place to hook in additional classes */ public static var mixins:Array; mx_internal static var popUpInfoClass:Class; mx_internal static function createPopUpData():PopUpData { if (!popUpInfoClass) return new PopUpData(); return new popUpInfoClass() as PopUpData; } //-------------------------------------------------------------------------- // // Class methods // //-------------------------------------------------------------------------- /** * @private */ private static function weakDependency():void { ActiveWindowManager }; /** * @private */ public static function getInstance():IPopUpManager { if (!instance) instance = new PopUpManagerImpl(); return instance; } //-------------------------------------------------------------------------- // // Constructor // //-------------------------------------------------------------------------- /** * @private */ public function PopUpManagerImpl() { super(); if (mixins) { var n:int = mixins.length; for (var i:int = 0; i < n; i++) { new mixins[i](this); } } if (hasEventListener("initialize")) dispatchEvent(new Event("initialize")); } //-------------------------------------------------------------------------- // // Variables // //-------------------------------------------------------------------------- /** * @private * The class used to create the shield that makes a window appear modal. */ mx_internal var modalWindowClass:Class; /** * @private * An array of information about currently active popups */ mx_internal var popupInfo:Array = []; //-------------------------------------------------------------------------- // // Methods // //-------------------------------------------------------------------------- /** * Creates a top-level window and places it above other windows in the * z-order. * It is good practice to call theremovePopUp()
method
* to remove popups created by using the createPopUp()
method.
*
* If the class implements IFocusManagerContainer, the window will have its
* own FocusManager so that, if the user uses the TAB key to navigate between
* controls, only the controls in the window will be accessed.
*
* Example
* *pop = mx.managers.PopUpManager.createPopUp(pnl, TitleWindow, false);* *
Creates a popup window based on the TitleWindow class, using pnl
as the MovieClip
* for determining where to place the popup. It is defined to be a non-modal window
* meaning that other windows can receive mouse events
true
, the window is modal which means that
* the user will not be able to interact with other popups until the window
* is removed.
*
* @param childList The child list in which to add the popup.
* One of PopUpManagerChildList.APPLICATION
,
* PopUpManagerChildList.POPUP
,
* or PopUpManagerChildList.PARENT
(default).
*
* @param moduleFactory The moduleFactory where this pop-up should look for
* its embedded fonts and style manager.
*
* @return Reference to new top-level window.
*
* @see PopUpManagerChildList
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function createPopUp(parent:DisplayObject,
className:Class,
modal:Boolean = false,
childList:String = null,
moduleFactory:IFlexModuleFactory = null):IFlexDisplayObject
{
const window:IUIComponent = new className();
addPopUp(window, parent, modal, childList, moduleFactory);
return window;
}
/**
* Pops up a top-level window.
* It is good practice to call removePopUp()
to remove popups
* created by using the createPopUp()
method.
* If the class implements IFocusManagerContainer, the window will have its
* own FocusManager so that, if the user uses the TAB key to navigate between
* controls, only the controls in the window will be accessed.
*
* Example
* *var tw = new TitleWindow(); * tw.title = "My Title"; * mx.managers.PopUpManager.addPopUp(tw, pnl, false);* *
Creates a popup window using the tw
instance of the
* TitleWindow class and pnl
as the Sprite for determining
* where to place the popup.
* It is defined to be a non-modal window.
true
, the window is modal which means that
* the user will not be able to interact with other popups until the window
* is removed.
*
* @param childList The child list in which to add the pop-up.
* One of PopUpManagerChildList.APPLICATION
,
* PopUpManagerChildList.POPUP
,
* or PopUpManagerChildList.PARENT
(default).
*
* @param moduleFactory The moduleFactory where this pop-up should look for
* its embedded fonts and style manager.
*
* @see PopUpManagerChildList
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function addPopUp(window:IFlexDisplayObject,
parent:DisplayObject,
modal:Boolean = false,
childList:String = null,
moduleFactory:IFlexModuleFactory = null):void
{
// trace("POPUP: window is " + window);
// All popups go on the local root.
// trace("POPUP: root is " + parent.root);
// trace("POPUP: initial parent is " + parent);
const visibleFlag:Boolean = window.visible;
if (parent is IUIComponent && window is IUIComponent &&
IUIComponent(window).document == null)
IUIComponent(window).document = IUIComponent(parent).document;
if (window is IFlexModule && IFlexModule(window).moduleFactory == null)
{
if (moduleFactory)
IFlexModule(window).moduleFactory = moduleFactory;
else if (parent is IUIComponent && IUIComponent(parent).document is IFlexModule)
IFlexModule(window).moduleFactory = IFlexModule(IUIComponent(parent).document).moduleFactory;
}
var sm:ISystemManager = getTopLevelSystemManager(parent);
var children:IChildList;
var topMost:Boolean;
if (!sm)
{
// check if parent is our sandbox root
sm = ISystemManager(SystemManagerGlobals.topLevelSystemManagers[0]);
if (sm.getSandboxRoot() != parent)
{
//trace("error: popup root was not SystemManager");
return; // and maybe a nice error message
}
}
// smp is the actual systemManager that will parent the popup
// it might get changed by the request
var smp:ISystemManager = sm;
if (hasEventListener("addPopUp"))
{
var request:Request = new Request("addPopUp", false, true, { parent: parent, sm: sm, modal: modal, childList: childList} );
if (!dispatchEvent(request))
smp = request.value as ISystemManager;
}
if (window is IUIComponent)
IUIComponent(window).isPopUp = true;
if (!childList || childList == PopUpManagerChildList.PARENT)
topMost = smp.popUpChildren.contains(parent);
else
topMost = (childList == PopUpManagerChildList.POPUP);
children = topMost ? smp.popUpChildren : smp;
if (DisplayObject(window).parent != children)
children.addChild(DisplayObject(window));
window.visible = false;
const o:PopUpData = createPopUpData();
o.owner = DisplayObject(window);
o.topMost = topMost;
o.systemManager = smp;
popupInfo.push(o);
var awm:IActiveWindowManager =
IActiveWindowManager(smp.getImplementation("mx.managers::IActiveWindowManager"));
if (window is IFocusManagerContainer)
{
if (IFocusManagerContainer(window).focusManager)
{
awm.addFocusManager(IFocusManagerContainer(window));
}
else
// Popups get their own focus loop
IFocusManagerContainer(window).focusManager =
new FocusManager(IFocusManagerContainer(window), true);
}
if (hasEventListener("addPlaceHolder"))
{
var event:DynamicEvent = new DynamicEvent("addPlaceHolder");
event.sm = sm;
event.window = window;
dispatchEvent(event);
}
// force into automation hierarchy
if (window is IAutomationObject)
IAutomationObject(window).showInAutomationHierarchy = true;
if (window is ILayoutManagerClient )
UIComponentGlobals.layoutManager.validateClient(ILayoutManagerClient (window), true);
o.parent = parent;
if (window is IUIComponent)
{
IUIComponent(window).setActualSize(
IUIComponent(window).getExplicitOrMeasuredWidth(),
IUIComponent(window).getExplicitOrMeasuredHeight());
}
// The layout direction may have changed since last time the
// popup was opened so need to reinit mirroring fields.
if (FlexVersion.compatibilityVersion >= FlexVersion.VERSION_4_0)
{
if (window is ILayoutDirectionElement)
ILayoutDirectionElement(window).invalidateLayoutDirection();
}
if (modal)
{
// handles accessibility for modal popUps.
if(Capabilities.hasAccessibility && Accessibility.active)
window.addEventListener(FlexEvent.CREATION_COMPLETE, modalPopUpCreationCompleteHandler, false, 0, true);
// create a modal window shield which blocks input and sets up mouseDownOutside logic
createModalWindow(parent, o, children, visibleFlag, smp, smp.getSandboxRoot());
}
else
{
o._mouseDownOutsideHandler = nonmodalMouseDownOutsideHandler;
o._mouseWheelOutsideHandler = nonmodalMouseWheelOutsideHandler;
window.visible = visibleFlag;
}
// Add show/hide listener so mouse out listeners can be added when
// a pop up is shown because applications can be launched and
// terminated between the time a pop up is hidden to when it is
// shown again.
o.owner.addEventListener(FlexEvent.SHOW, showOwnerHandler);
o.owner.addEventListener(FlexEvent.HIDE, hideOwnerHandler);
addMouseOutEventListeners(o);
// Listen for unload so we know to kill the window (and the modalWindow if modal)
// this handles _all_ cleanup
window.addEventListener(Event.REMOVED, popupRemovedHandler);
if (window is IFocusManagerContainer && visibleFlag)
{
if (hasEventListener("addedPopUp"))
{
event = new DynamicEvent("addedPopUp", false, true);
event.window = window;
event.systemManager = smp;
dispatchEvent(event);
}
else
awm.activate(IFocusManagerContainer(window));
}
// trace("END POPUP: addPopUp" + parent);
}
mx_internal function getTopLevelSystemManager(parent:DisplayObject):ISystemManager
{
var localRoot:DisplayObjectContainer;
var sm:ISystemManager;
if (hasEventListener("topLevelSystemManager"))
{
var request:Request = new Request("topLevelSystemManager", false, true);
request.value = parent;
if (!dispatchEvent(request))
localRoot = request.value as DisplayObjectContainer;
}
if (!localRoot)
localRoot = DisplayObjectContainer(parent.root);
// If the parent isn't rooted yet,
// Or the root is the stage (which is the case in a second AIR window)
// use the global system manager instance.
if ((!localRoot || localRoot is Stage) && parent is IUIComponent)
localRoot = DisplayObjectContainer(IUIComponent(parent).systemManager);
if (localRoot is ISystemManager)
{
sm = ISystemManager(localRoot);
if (!sm.isTopLevel())
sm = sm.topLevelSystemManager;
}
return sm;
}
/**
* Centers a popup window over whatever window was used in the call
* to the createPopUp()
or addPopUp()
method.
*
* Note that the position of the popup window may not * change immediately after this call since Flex may wait to measure and layout the * popup window before centering it.
* * @param The IFlexDisplayObject representing the popup. * * @langversion 3.0 * @playerversion Flash 9 * @playerversion AIR 1.1 * @productversion Flex 3 */ public function centerPopUp(popUp:IFlexDisplayObject):void { if (popUp is IInvalidating) IInvalidating(popUp).validateNow(); const o:PopUpData = findPopupInfoByOwner(popUp); // If we don't find the pop owner or if the owner's parent is not specified or is not on the // stage, then center based on the popUp's current parent. var popUpParent:DisplayObject = (o && o.parent && o.parent.stage) ? o.parent : popUp.parent; if (popUpParent) { var systemManager:ISystemManager = o.systemManager; var x:Number; var y:Number; var appWidth:Number; var appHeight:Number; var parentWidth:Number; var parentHeight:Number; var rect:Rectangle; var clippingOffset:Point = new Point(); var pt:Point; var isTopLevelRoot:Boolean; var sbRoot:DisplayObject = systemManager.getSandboxRoot(); var request:Request; if (hasEventListener("isTopLevelRoot")) { request = new Request("isTopLevelRoot", false, true); } if (request && !dispatchEvent(request)) isTopLevelRoot = Boolean(request.value); else isTopLevelRoot = systemManager.isTopLevelRoot(); if (isTopLevelRoot && (FlexVersion.compatibilityVersion < FlexVersion.VERSION_4_6)) { // The sandbox root is the top level root. // The application width is just the screen width. var screen:Rectangle = systemManager.screen; appWidth = screen.width; appHeight = screen.height; } else { rect = systemManager.getVisibleApplicationRect(); rect.topLeft = DisplayObject(systemManager).globalToLocal(rect.topLeft); rect.bottomRight = DisplayObject(systemManager).globalToLocal(rect.bottomRight); // Offset the top, left of the window to bring it into view. clippingOffset = rect.topLeft.clone(); appWidth = rect.width; appHeight = rect.height; } // If parent is a UIComponent, check for clipping between // the object and its SystemManager if (popUpParent is UIComponent) { rect = UIComponent(popUpParent).getVisibleRect(); if (UIComponent(popUpParent).systemManager != sbRoot) rect = UIComponent(popUpParent).systemManager.getVisibleApplicationRect(rect); var offset:Point = popUpParent.globalToLocal(rect.topLeft); clippingOffset.x += offset.x; clippingOffset.y += offset.y; parentWidth = rect.width; parentHeight = rect.height; } else { parentWidth = popUpParent.width; parentHeight = popUpParent.height; } // The appWidth may smaller than parentWidth if the application is // clipped by the parent application. x = Math.max(0, (Math.min(appWidth, parentWidth) - popUp.width) / 2); y = Math.max(0, (Math.min(appHeight, parentHeight) - popUp.height) / 2); // If the layout has been mirrored, then 0,0 is the uppper // right corner; compensate here. if (FlexVersion.compatibilityVersion >= FlexVersion.VERSION_4_0) { // If popUp has layout direction different than the parent (or parent doesn't // have layout direction and popUp is RTL) flip it to the other side of the x axis. const popUpLDE:ILayoutDirectionElement = popUp as ILayoutDirectionElement; const parentLDE:ILayoutDirectionElement = popUpParent as ILayoutDirectionElement; if (popUpLDE && ((parentLDE && parentLDE.layoutDirection != popUpLDE.layoutDirection) || (!parentLDE && popUpLDE.layoutDirection == LayoutDirection.RTL))) { x = -x /* to flip it on the other side of the x axis*/ -popUp.width /* because 0 is the right edge */; } } pt = new Point(clippingOffset.x, clippingOffset.y); pt = popUpParent.localToGlobal(pt); pt = popUp.parent.globalToLocal(pt); popUp.move(Math.round(x) + pt.x, Math.round(y) + pt.y); } } /** * Removes a popup window popped up by * thecreatePopUp()
or addPopUp()
method.
*
* @param window The IFlexDisplayObject representing the popup window.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function removePopUp(popUp:IFlexDisplayObject):void
{
// all we want to do here is verify that this popup is one of ours
// and remove it from the display list; the REMOVED handler will do the rest
// (this is so that we never leak memory, popups will self-manage even if
// removePopUp is not called).
if (popUp && popUp.parent)
{
const o:PopUpData = findPopupInfoByOwner(popUp);
if (o)
{
var sm:ISystemManager = o.systemManager;
if (!sm)
{
var iui:IUIComponent = popUp as IUIComponent;
// cross-versioning error sometimes returns wrong parent
if (iui)
sm = ISystemManager(iui.systemManager);
else
return;
}
if (o.topMost)
sm.popUpChildren.removeChild(DisplayObject(popUp));
else
sm.removeChild(DisplayObject(popUp));
}
}
}
/**
* Makes sure a popup window is higher than other objects in its child list
* The SystemManager does this automatically if the popup is a top level window
* and is moused on,
* but otherwise you have to take care of this yourself.
*
* @param The IFlexDisplayObject representing the popup.
*
* @langversion 3.0
* @playerversion Flash 9
* @playerversion AIR 1.1
* @productversion Flex 3
*/
public function bringToFront(popUp:IFlexDisplayObject):void
{
if (popUp && popUp.parent)
{
const o:PopUpData = findPopupInfoByOwner(popUp);
if (o)
{
if (hasEventListener("bringToFront"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("bringToFront", false, true);
dynamicEvent.popUpData = o;
dynamicEvent.popUp = popUp;
if (!dispatchEvent(dynamicEvent))
return;
}
const sm:ISystemManager = ISystemManager(popUp.parent);
if (o.topMost)
sm.popUpChildren.setChildIndex(DisplayObject(popUp), sm.popUpChildren.numChildren - 1);
else
sm.setChildIndex(DisplayObject(popUp), sm.numChildren - 1);
}
}
}
/**
* @private
*
* Create the modal window.
* This is called in two different cases.
* 1. Create a modal window for a local pop up.
* 2. Create a modal window for a remote pop up. In this case o.owner will be null.
*/
mx_internal function createModalWindow(parentReference:DisplayObject,
o:PopUpData,
childrenList:IChildList,
visibleFlag:Boolean,
sm:ISystemManager,
sbRoot:DisplayObject):void
{
const popup:IFlexDisplayObject = IFlexDisplayObject(o.owner);
const popupStyleClient:IStyleClient = popup as IStyleClient;
var duration:Number = 0;
// Create a modalWindow the size of the stage
// that eats all mouse clicks.
var modalWindow:Sprite;
if (modalWindowClass)
{
modalWindow = new modalWindowClass();
}
else
{
modalWindow = new FlexSprite();
modalWindow.name = "modalWindow";
}
if (!sm && parentReference)
sm = IUIComponent(parentReference).systemManager;
var awm:IActiveWindowManager =
IActiveWindowManager(sm.getImplementation("mx.managers::IActiveWindowManager"));
awm.numModalWindows++;
// Add it to the collection just below the popup
if (popup)
childrenList.addChildAt(modalWindow,
childrenList.getChildIndex(DisplayObject(popup)));
else
childrenList.addChild(modalWindow);
// force into the automation hierarchy
if (popup is IAutomationObject)
IAutomationObject(popup).showInAutomationHierarchy = true;
o.modalWindow = modalWindow;
if (popupStyleClient)
modalWindow.alpha = popupStyleClient.getStyle("modalTransparency");
else
modalWindow.alpha = 0;
modalWindow.tabEnabled = false;
const screen:Rectangle = sm.screen;
const g:Graphics = modalWindow.graphics;
var c:Number = 0xFFFFFF;
if (popupStyleClient)
{
c = popupStyleClient.getStyle("modalTransparencyColor");
}
if (hasEventListener("createModalWindow"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("createModalWindow", false, true);
dynamicEvent.popUpData = o;
dynamicEvent.popUp = popup;
dynamicEvent.color = c;
dynamicEvent.visibleFlag = visibleFlag;
dynamicEvent.childrenList = childrenList;
if (!dispatchEvent(dynamicEvent))
c = dynamicEvent.color;
}
// trace("createModalWindow: drawing modal " + s);
g.clear();
g.beginFill(c, 100);
g.drawRect(screen.x-1, screen.y-1, screen.width+2, screen.height+2);
g.endFill();
if (hasEventListener("updateModalMask"))
{
dynamicEvent = new DynamicEvent("updateModalMask");
dynamicEvent.popUpData = o;
dynamicEvent.popUp = popup;
dynamicEvent.childrenList = childrenList;
dispatchEvent(dynamicEvent);
}
// a modal mousedownoutside handler just dispatches the event
o._mouseDownOutsideHandler = dispatchMouseDownOutsideEvent;
o._mouseWheelOutsideHandler = dispatchMouseWheelOutsideEvent;
// the following handlers all get removed in REMOVED on the popup
// Set the resize handler so the modal can stay the size of the screen
sm.addEventListener(Event.RESIZE, o.resizeHandler);
if (popup)
{
// Listen for show so we know to show the modal window
popup.addEventListener(FlexEvent.SHOW, popupShowHandler);
// Listen for hide so we know to hide the modal window
popup.addEventListener(FlexEvent.HIDE, popupHideHandler);
}
if (visibleFlag)
showModalWindow(o, sm, false);
else if (popup)
popup.visible = visibleFlag;
if (hasEventListener("createdModalWindow"))
{
dynamicEvent = new DynamicEvent("createdModalWindow");
dynamicEvent.popUpData = o;
dynamicEvent.popUp = popup;
dynamicEvent.visibleFlag = visibleFlag;
dynamicEvent.childrenList = childrenList;
dispatchEvent(dynamicEvent);
}
}
/**
* @private
*/
private function endEffects(o:PopUpData):void
{
if (o.fade)
{
o.fade.end();
o.fade = null;
}
if (o.blur)
{
o.blur.end();
o.blur = null;
}
}
mx_internal function showModalWindow(o:PopUpData, sm:ISystemManager, sendRequest:Boolean = true):void
{
const popUpStyleClient:IStyleClient = o.owner as IStyleClient;
var duration:Number = 0;
var alpha:Number = 0;
if (popUpStyleClient)
{
duration = popUpStyleClient.getStyle("modalTransparencyDuration");
}
if (popUpStyleClient)
{
alpha = popUpStyleClient.getStyle("modalTransparency");
}
var blurAmount:Number = 0;
if (popUpStyleClient)
{
blurAmount = popUpStyleClient.getStyle("modalTransparencyBlur");
}
var transparencyColor:Number = 0xFFFFFF;
if (popUpStyleClient)
{
transparencyColor = popUpStyleClient.getStyle("modalTransparencyColor");
}
var sbRoot:DisplayObject = sm.getSandboxRoot();
if (hasEventListener("showModalWindow"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("showModalWindow", false, true);
dynamicEvent.popUpData = o;
dynamicEvent.sendRequest = sendRequest;
dynamicEvent.alpha = alpha;
dynamicEvent.blurAmount = blurAmount;
dynamicEvent.duration = duration;
dynamicEvent.systemManager = sm;
dynamicEvent.transparencyColor = transparencyColor;
if (!dispatchEvent(dynamicEvent))
{
alpha = dynamicEvent.alpha;
blurAmount = dynamicEvent.blurAmount;
duration = dynamicEvent.duration;
transparencyColor = dynamicEvent.transparencyColor;
}
}
o.modalWindow.alpha = alpha;
showModalWindowInternal(o, duration, alpha, transparencyColor, blurAmount, sm, sbRoot);
}
private function setModalPopupVisible(popup:DisplayObject, value:Boolean):void
{
popup.removeEventListener(FlexEvent.SHOW, popupShowHandler);
popup.removeEventListener(FlexEvent.HIDE, popupHideHandler);
popup.visible = value;
popup.addEventListener(FlexEvent.SHOW, popupShowHandler);
popup.addEventListener(FlexEvent.HIDE, popupHideHandler);
}
/**
* @private
* Show the modal transparency blocker, playing effects if needed.
*/
private function showModalWindowInternal(o:PopUpData,
transparencyDuration:Number,
transparency:Number,
transparencyColor:Number,
transparencyBlur:Number,
sm:ISystemManager,
sbRoot:DisplayObject):void
{
// NO POPUP Data
// End any effects that are currently playing for this popup.
endEffects(o);
if (transparencyDuration)
{
// Fade effect on the modal transparency blocker
const fade:Fade = new Fade(o.modalWindow);
fade.alphaFrom = 0;
fade.alphaTo = transparency;
fade.duration = transparencyDuration;
fade.addEventListener(EffectEvent.EFFECT_END, fadeInEffectEndHandler);
o.modalWindow.alpha = 0;
o.modalWindow.visible = true;
o.fade = fade;
if (o.owner)
setModalPopupVisible(o.owner, false);
fade.play();
// Blur effect on the application
var blurAmount:Number = transparencyBlur;
if (blurAmount)
{
// Ensure we blur the appropriate top level document.
if (DisplayObject(sm).parent is Stage)
{
// Checking this case first allows WindowedSystemManagers be the blur target.
o.blurTarget = sm.document;
}
else if (sm != sbRoot)
{
// Get the application document of the sandbox root.
// Use a request to get the document so APIs may change
// between Flex versions.
var sbRootApp:Object; // sbRoot.application;
if (hasEventListener("blurTarget"))
{
var request:Request = new Request("blurTarget", false, true, { popUpData: o });
if (!dispatchEvent(request))
{
o.blurTarget = request.value;
}
}
}
else
o.blurTarget = FlexGlobals.topLevelApplication;
const blur:Blur = new Blur(o.blurTarget);
blur.blurXFrom = blur.blurYFrom = 0;
blur.blurXTo = blur.blurYTo = blurAmount;
blur.duration = transparencyDuration;
blur.addEventListener(EffectEvent.EFFECT_END, effectEndHandler);
o.blur = blur;
blur.play();
}
}
else
{
if (o.owner)
setModalPopupVisible(o.owner, true);
o.modalWindow.visible = true;
}
}
/**
* @private
* Hide the modal transparency blocker, playing effects if needed.
*
*/
mx_internal function hideModalWindow(o:PopUpData, destroy:Boolean = false):void
{
const popUpStyleClient:IStyleClient = o.owner as IStyleClient;
var duration:Number = 0;
if (popUpStyleClient)
duration = popUpStyleClient.getStyle("modalTransparencyDuration");
// end any effects that are current playing for this popup
endEffects(o);
if (duration)
{
// Fade effect on the modal transparency blocker
const fade:Fade = new Fade(o.modalWindow);
fade.alphaFrom = o.modalWindow.alpha;
fade.alphaTo = 0;
fade.duration = duration;
fade.addEventListener(EffectEvent.EFFECT_END,
destroy ? fadeOutDestroyEffectEndHandler : fadeOutCloseEffectEndHandler);
o.modalWindow.visible = true;
o.fade = fade;
fade.play();
// Blur effect on the application
const blurAmount:Number = popUpStyleClient.getStyle("modalTransparencyBlur");
if (blurAmount)
{
const blur:Blur = new Blur(o.blurTarget);
blur.blurXFrom = blur.blurYFrom = blurAmount;
blur.blurXTo = blur.blurYTo = 0;
blur.duration = duration;
blur.addEventListener(EffectEvent.EFFECT_END, effectEndHandler);
o.blur = blur;
blur.play();
}
}
else
{
o.modalWindow.visible = false;
}
if (hasEventListener("hideModalWindow"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("hideModalWindow", false, false);
dynamicEvent.popUpData = o;
dynamicEvent.destroy = destroy;
dispatchEvent(dynamicEvent);
}
}
/**
* @private
* Returns the index position of the PopUpData in the popupInfo array (or -1)
* for a given popupInfo.owner
*/
private function findPopupInfoIndexByOwner(owner:Object):int
{
const n:int = popupInfo.length;
for (var i:int = 0; i < n; i++)
{
var o:PopUpData = popupInfo[i];
if (o.owner == owner)
return i;
}
return -1;
}
/**
* @private
* Returns the PopUpData (or null) for a given popupInfo.owner
*/
private function findPopupInfoByOwner(owner:Object):PopUpData
{
var index:int = findPopupInfoIndexByOwner(owner);
return index > -1 ? popupInfo[index] : null;
}
/**
* @private
* Add mouse out listeners for modal and non-modal windows.
*/
private function addMouseOutEventListeners(o:PopUpData):void
{
var sbRoot:DisplayObject = o.systemManager.getSandboxRoot();
if (o.modalWindow)
{
o.modalWindow.addEventListener(MouseEvent.MOUSE_DOWN, o.mouseDownOutsideHandler);
o.modalWindow.addEventListener(MouseEvent.MOUSE_WHEEL, o.mouseWheelOutsideHandler, true);
}
else
{
sbRoot.addEventListener(MouseEvent.MOUSE_DOWN, o.mouseDownOutsideHandler);
sbRoot.addEventListener(MouseEvent.MOUSE_WHEEL, o.mouseWheelOutsideHandler, true);
}
if (hasEventListener("addMouseOutEventListeners"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("addMouseOutEventListeners", false, false);
dynamicEvent.popUpData = o;
dispatchEvent(dynamicEvent);
}
}
/**
* @private
* Remove mouse out listeners for modal and non-modal windows.
*/
private function removeMouseOutEventListeners(o:PopUpData):void
{
var sbRoot:DisplayObject = o.systemManager.getSandboxRoot();
if (o.modalWindow)
{
o.modalWindow.removeEventListener(MouseEvent.MOUSE_DOWN, o.mouseDownOutsideHandler);
o.modalWindow.removeEventListener(MouseEvent.MOUSE_WHEEL, o.mouseWheelOutsideHandler, true);
}
else
{
sbRoot.removeEventListener(MouseEvent.MOUSE_DOWN, o.mouseDownOutsideHandler);
sbRoot.removeEventListener(MouseEvent.MOUSE_WHEEL, o.mouseWheelOutsideHandler, true);
}
if (hasEventListener("removeMouseOutEventListeners"))
{
var dynamicEvent:DynamicEvent = new DynamicEvent("removeMouseOutEventListeners", false, false);
dynamicEvent.popUpData = o;
dispatchEvent(dynamicEvent);
}
}
//--------------------------------------------------------------------------
//
// Event handlers
//
//--------------------------------------------------------------------------
/**
* @private
* Set by PopUpManager on modal windows so they show when the parent shows.
*/
private function popupShowHandler(event:FlexEvent):void
{
const o:PopUpData = findPopupInfoByOwner(event.target);
if (o)
showModalWindow(o, getTopLevelSystemManager(o.parent));
}
/**
* @private
* Set by PopUpManager on modal windows so they hide when the parent hide
*/
private function popupHideHandler(event:FlexEvent):void
{
const o:PopUpData = findPopupInfoByOwner(event.target);
if (o)
hideModalWindow(o);
}
/**
* @private
*/
private function showOwnerHandler(event:FlexEvent):void
{
const o:PopUpData = findPopupInfoByOwner(event.target);
if (o)
{
// add mouse out listeners.
addMouseOutEventListeners(o);
}
}
/**
* @private
*/
private function hideOwnerHandler(event:FlexEvent):void
{
const o:PopUpData = findPopupInfoByOwner(event.target);
if (o)
{
// remove mouse out listeners
removeMouseOutEventListeners(o);
}
}
/**
* @private
* Set by PopUpManager on modal windows to monitor when the parent window gets killed.
* PopUps self-manage their memory -- when they are removed using removePopUp OR
* manually removed with removeChild, they will clean themselves up when they leave the
* display list (including all references to PopUpManager).
*/
private function popupRemovedHandler(event:Event):void
{
const n:int = popupInfo.length;
for (var i:int = 0; i < n; i++)
{
var o:PopUpData = popupInfo[i],
popUp:DisplayObject = o.owner;
if (popUp == event.target)
{
var popUpParent:DisplayObject = o.parent,
modalWindow:DisplayObject = o.modalWindow,
sm:ISystemManager = o.systemManager;
if (!sm.isTopLevel())
sm = sm.topLevelSystemManager;
if (popUp is IUIComponent)
IUIComponent(popUp).isPopUp = false;
var awm:IActiveWindowManager =
IActiveWindowManager(sm.getImplementation("mx.managers::IActiveWindowManager"));
if (popUp is IFocusManagerContainer)
{
awm.removeFocusManager(IFocusManagerContainer(popUp));
}
popUp.removeEventListener(Event.REMOVED, popupRemovedHandler);
if (hasEventListener("removeMouseOutEventListeners"))
{
var event2:DynamicEvent = new DynamicEvent("popUpRemoved");
event2.popUpData = o;
dispatchEvent(event2);
}
if (o.owner)
{
o.owner.removeEventListener(FlexEvent.SHOW, showOwnerHandler);
o.owner.removeEventListener(FlexEvent.HIDE, hideOwnerHandler);
}
removeMouseOutEventListeners(o);
// modal
if (modalWindow)
{
// restore accessibility to document
removeModalPopUpAccessibility(popUp);
// clean up all handlers
sm.removeEventListener(Event.RESIZE, o.resizeHandler);
popUp.removeEventListener(FlexEvent.SHOW, popupShowHandler);
popUp.removeEventListener(FlexEvent.HIDE, popupHideHandler);
hideModalWindow(o, true);
awm.numModalWindows--;
}
popupInfo.splice(i, 1);
break;
}
}
}
/**
* @private
* Show the modal window after the fade effect finishes
*/
private function fadeInEffectEndHandler(event:EffectEvent):void
{
effectEndHandler(event);
const n:int = popupInfo.length;
for (var i:int = 0; i < n; i++)
{
var o:PopUpData = popupInfo[i];
if (o.owner && o.modalWindow == event.effectInstance.target)
{
setModalPopupVisible(o.owner, true);
break;
}
}
}
/**
* @private
* Remove the modal window after the fade effect finishes
*/
private function fadeOutDestroyEffectEndHandler(event:EffectEvent):void
{
effectEndHandler(event);
const obj:DisplayObject = DisplayObject(event.effectInstance.target);
var modalMask:DisplayObject = obj.mask;
if (modalMask)
{
// modal mask is always added to the popupChildren list.
obj.mask = null;
sm.popUpChildren.removeChild(modalMask);
}
if (obj.parent is ISystemManager)
{
const sm:ISystemManager = ISystemManager(obj.parent)
if (sm.popUpChildren.contains(obj))
sm.popUpChildren.removeChild(obj);
else
sm.removeChild(obj);
}
else
{
if (obj.parent) // Mustella can already take you off stage
obj.parent.removeChild(obj);
}
}
/**
* @private
* Remove the modal window after the fade effect finishes
*/
private function fadeOutCloseEffectEndHandler(event:EffectEvent):void
{
effectEndHandler(event);
DisplayObject(event.effectInstance.target).visible = false;
}
/**
* @private
*/
private function effectEndHandler(event:EffectEvent):void
{
const n:int = popupInfo.length;
for (var i:int = 0; i < n; i++)
{
var o:PopUpData = popupInfo[i];
var e:IEffect = event.effectInstance.effect;
if (e == o.fade)
o.fade = null;
else if (e == o.blur)
o.blur = null;
}
}
/**
* @private
* If not modal, use this kind of mouseDownOutside logic
*/
private static function nonmodalMouseDownOutsideHandler(owner:DisplayObject, evt:MouseEvent):void
{
// shapeFlag is false here for performance reasons
if (owner.hitTestPoint(evt.stageX, evt.stageY, true))
{
}
else
{
if (owner is IUIComponent)
if (IUIComponent(owner).owns(DisplayObject(evt.target)))
return;
dispatchMouseDownOutsideEvent(owner, evt);
}
}
/**
* @private
* If not modal, use this kind of mouseWheelOutside logic
*/
private static function nonmodalMouseWheelOutsideHandler(owner:DisplayObject, evt:MouseEvent):void
{
// shapeFlag is false here for performance reasons
if (owner.hitTestPoint(evt.stageX, evt.stageY, true))
{
}
else
{
if (owner is IUIComponent)
if (IUIComponent(owner).owns(DisplayObject(evt.target)))
return;
dispatchMouseWheelOutsideEvent(owner, evt);
}
}
/**
* @private
* This mouseWheelOutside handler just dispatches the event.
*/
private static function dispatchMouseWheelOutsideEvent(owner:DisplayObject, evt:MouseEvent):void
{
// this is a modal window without a popup owner to dispatch the message to.
if (!owner)
return;
const event:MouseEvent = new FlexMouseEvent(FlexMouseEvent.MOUSE_WHEEL_OUTSIDE);
const pt:Point = owner.globalToLocal(new Point(evt.stageX, evt.stageY));
event.localX = pt.x;
event.localY = pt.y;
event.buttonDown = evt.buttonDown;
event.shiftKey = evt.shiftKey;
event.altKey = evt.altKey;
event.ctrlKey = evt.ctrlKey;
event.delta = evt.delta;
event.relatedObject = InteractiveObject(evt.target);
owner.dispatchEvent(event);
}
/**
* @private
* This mouseDownOutside handler just dispatches the event.
*/
private static function dispatchMouseDownOutsideEvent(owner:DisplayObject, evt:MouseEvent):void
{
// this is a modal window without a popup owner to dispatch the message to.
if (!owner)
return;
const event:MouseEvent = new FlexMouseEvent(FlexMouseEvent.MOUSE_DOWN_OUTSIDE);
const pt:Point = owner.globalToLocal(new Point(evt.stageX, evt.stageY));
event.localX = pt.x;
event.localY = pt.y;
event.buttonDown = evt.buttonDown;
event.shiftKey = evt.shiftKey;
event.altKey = evt.altKey;
event.ctrlKey = evt.ctrlKey;
event.delta = evt.delta;
event.relatedObject = InteractiveObject(evt.target);
owner.dispatchEvent(event);
}
/**
* @private
* This method handles the creation of a modal popUp
*/
private function modalPopUpCreationCompleteHandler(event:FlexEvent):void
{
event.target.removeEventListener(FlexEvent.CREATION_COMPLETE, modalPopUpCreationCompleteHandler);
addModalPopUpAccessibility(event.currentTarget as DisplayObject);
}
/**
* @private
* This method handles the creation of a modal popUp when assistive
* technology is active, by silencing the content of the top-level document.
*/
private function addModalPopUpAccessibility(popUp:DisplayObject):Boolean
{
if (Capabilities.hasAccessibility && Accessibility.active)
{
const p:PopUpData = findPopupInfoByOwner(popUp);
if (p)
{
const n:int = popupInfo.length;
for (var i:int = 0; i < n; i++)
{
var o:PopUpData = popupInfo[i];
if (o && o != p && o.owner.accessibilityProperties)
{
o.owner.accessibilityProperties.silent = true;
}
}
var sbRoot:Object = p.systemManager.getSandboxRoot(); // getTopLevelSystemManager(p.parent);
if (!sbRoot.document.accessibilityProperties)
sbRoot.document.accessibilityProperties = new AccessibilityProperties();
// This hides top-level document content from assistive technology.
sbRoot.document.accessibilityProperties.silent = true;
Accessibility.updateProperties();
}
return true;
}
return false;
}
/**
* @private
* This method handles the removal of a modal popUp when assistive technology is active,
* by exposing the content of the top-level document.
*
* @return
*/
private function removeModalPopUpAccessibility(popUp:DisplayObject):Boolean
{
if (Capabilities.hasAccessibility && Accessibility.active)
{
const p:PopUpData = findPopupInfoByOwner(popUp);
if (p)
{
handleAccessibilityForNestedPopups(popUp);
if (popupInfo.length<=1)
{
var sbRoot:Object = p.systemManager.getSandboxRoot();
sbRoot.document.accessibilityProperties.silent = false;
}
Accessibility.updateProperties();
}
return true;
}
return false;
}
/**
* @private
* This method handles accessibility for nested popUps.
* If this method is called from outside removeModalPopUpAccessibility,
* it is mandatory to invoke Accessibility.updateProperties() to communicate
* the individual popup's accessibilityProperties change to the screen reader.
*
* This method should only come into play with modal popUps, in which case it will * either expose accessibility of a modal popUp underneath the popUp being closed * without exposing the underlying content, or continue exposing accessibility * of popUps until there are no more to expose.
*/ private function handleAccessibilityForNestedPopups(popUpBeingVisited:DisplayObject):void { if (!popUpBeingVisited) return; var index:int = findPopupInfoIndexByOwner(popUpBeingVisited); var popupData:PopUpData = index > -1 ? popupInfo[index] : null; var underneathPopUpData:PopUpData; if (index == 0) { var sm:ISystemManager = getTopLevelSystemManager(popupData.parent); if (sm) { // If this is the only popUp, we should expose accessibility // of the top-level system manager's document sm.document.accessibilityProperties.silent = false; // We should also expose accessibility // of the sandbox root's document. var sbRoot:Object = popupData.systemManager.getSandboxRoot(); sbRoot.document.accessibilityProperties.silent = false; } } else if (index > 0) { // If more than one popUp is open, we should expose accessibility // of the underlying popUp underneathPopUpData = popupInfo[index - 1]; underneathPopUpData.owner.accessibilityProperties.silent = false; // All the nested popUp's AccessibilityProperties // changes should be handled by a single // Accessibility.updateProperties // call in removeModalPopUpAccessibility. // If the underlying popUp is modal, we should stop recursing. if (underneathPopUpData.modalWindow) return; handleAccessibilityForNestedPopups(underneathPopUpData.owner); } } } }