Welcome!

Adobe Flex Authors: Matthew Lobas, PR.com Newswire, Shelly Palmer, Kevin Benedict

Related Topics: Adobe Flex, FlexSeminar.com

Adobe Flex: Article

Freely Available "SizeableTitleWindow" With Adobe's ColdFusion/Flex Connectivity

I found that Adobe includes a really nice resizeable closeable window

In playing around with the ColdFusion/Flex wizards, I found that Adobe includes a really nice resizeable closeable window.  When I get more time, I'll integrate this with my MaximizeRestorePanel class.


package com.adobe.ColdFusion.components
{
 import mx.containers.TitleWindow;
 import mx.managers.CursorManager;
 import mx.managers.ISystemManager;
 import mx.controls.Alert;
 import mx.logging.Log;
 import mx.utils.ObjectUtil;
 import mx.controls.Button;
 import flash.events.MouseEvent;
 import flash.events.Event;
 import flash.system.System;
 import flash.geom.Rectangle;
 
 public class SizeableTitleWindow extends TitleWindow
 {
   private const dragThreshold:int = 2;
   // sanity constraints. 
   private const minSizeWidth:int = 180; 
   private const minSizeHeight:int = 220;
   
   public const cursorSizeNone:int = -1;
   public const cursorSizeNE:int   = 0;
   public const cursorSizeN:int    = 1;
   public const cursorSizeNW:int   = 2;
   public const cursorSizeW:int    = 3;
   public const cursorSizeSW:int   = 4;
   public const cursorSizeS:int    = 5;
   public const cursorSizeSE:int   = 6;
   public const cursorSizeE:int    = 7;
   public const cursorSizeAll:int  = 8;
   
   [Embed(source="cursorImages/sizeNS.gif")]
   public static var sizeNSCursorSymbol:Class;

   [Embed(source="cursorImages/sizeNESW.gif")]
   public static var sizeNESWCursorSymbol:Class;
   
   [Embed(source="cursorImages/sizeWE.gif")]
   public static var sizeWECursorSymbol:Class;

   [Embed(source="cursorImages/sizeNWSE.gif")]
   public static var sizeNWSECursorSymbol:Class;
         
   [Embed(source="cursorImages/sizeAll.gif")]
   public static var sizeAllCursorSymbol:Class;

   private var downX:int;
   private var downY:int;
   private var startLeft:int;
   private var startTop:int;
   private var startHeight:int;
   private var startWidth:int;
   
   private var resizeCursor:int;
   private var currentCursorID:int;
   private var prevCursor:int;
   private var isResizing:Boolean;
   
   public function SizeableTitleWindow()
   {
    super();
    isResizing = false;
    currentCursorID = CursorManager.NO_CURSOR;
    prevCursor = cursorSizeNone;
   }
   
   override protected function createChildren():void
   {
    super.createChildren();

    // make the cursor change to the resize cursor
    this.titleBar.addEventListener(MouseEvent.MOUSE_MOVE, titleBar_resizeMoveListener);
    this.addEventListener(MouseEvent.MOUSE_MOVE, resizeMoveListener);
    
    this.addEventListener(MouseEvent.MOUSE_OUT, cursorMouseOutListener);
    
    // since the titlebar mousedown listener calls startDragging, but the listener is private
    // we will do our checking in the overridden startDragging event for the titlebar
    this.addEventListener(MouseEvent.MOUSE_DOWN, resizeDownListener);

   }
   
   protected function getCursorStyle(x:int, y:int, isTitleBar:Boolean):int
   {
    if (isResizing)
     return resizeCursor;
    
    // the NW corner has to be done in a seperate section because this
    // corner is twitchy and we add a 1 pix buffer
    if (x >= 0 && x <= dragThreshold + 1 && y >= 0 && y <= dragThreshold + 1)
    {
     return cursorSizeNW;
    } else if (x >= 0 && x <= dragThreshold)
    {
     if (y >= this.height - dragThreshold)
     {
      return cursorSizeSW;
     } else {
      return cursorSizeW;
     }
    } else if (x >= this.width - dragThreshold)
    {
     if (y >= 0 && y <= dragThreshold)
     {
      return cursorSizeNE;
     } else if (y >= this.height - dragThreshold)
     {
      return cursorSizeSE;
     } else {
      return cursorSizeE;
     }     
    } else if (y >= 0 && y <= dragThreshold)
    {
     return cursorSizeN;
    } else if (y >= this.height - dragThreshold)
    {
     return cursorSizeS;
// if you want to have the "move" style cursor when over the title bar, uncomment the next three lines     
//    } else if (isTitleBar)
//    {
//     return cursorSizeAll;
    }
    return cursorSizeNone;
   }
   
   protected function clearCursor():void
   {
    if (currentCursorID != CursorManager.NO_CURSOR)
    {
     CursorManager.removeCursor(currentCursorID);
     currentCursorID = CursorManager.NO_CURSOR;
    }
    prevCursor = cursorSizeNone;    
   }
   
   /**
    *  @protected
    *  Returns the height of the header.
    */
   protected function getHeaderHeight():Number
   {
    var headerHeight:Number = getStyle("headerHeight");
    
    if (isNaN(headerHeight))
     headerHeight = measureHeaderText().height + HEADER_PADDING;
    
    return headerHeight;
   }
   
   /**
    *  @protected. Returns a Rectangle containing the largest piece of header
    *  text (can be either the title or status, whichever is bigger).
    */
   protected function measureHeaderText():Rectangle
   {
    var textWidth:Number = 20;
    var textHeight:Number = 14;
    
    if (titleTextField && titleTextField.text)
    {
     titleTextField.validateNow();
     textWidth = titleTextField.textWidth;
     textHeight = titleTextField.textHeight;
    }
    
    if (statusTextField)
    {
     statusTextField.validateNow();
     textWidth = Math.max(textWidth, statusTextField.textWidth);
     textHeight = Math.max(textHeight, statusTextField.textHeight);
    }
    
    return new Rectangle(0, 0, textWidth, textHeight);
   }
         
   protected function adjustCursor(event:MouseEvent, isTitleBar:Boolean):void
   {
    var c:int;
    
    // we only want the move event from the title bar itself, not from it's children
    // otherwise you get weird cursor behavior in the middle of the titlebar
    if (isTitleBar && event.target != titleBar)
    {
     c = cursorSizeAll;
    } else {
     c = getCursorStyle(event.localX, event.localY, isTitleBar);
    }
    
    // don't switch stuff around if we don't have to
    if (c == prevCursor)
    {
     return;
    }
    
    prevCursor = c;
        
    clearCursor();
    
    switch (c)
    {
// if you want to have the "move" style cursor when over the title bar, uncomment the next three lines     
//     case cursorSizeAll:
//      currentCursorID = CursorManager.setCursor(sizeAllCursorSymbol, 2, -10, -10); 
//      break;
     case cursorSizeE:
     case cursorSizeW:
      currentCursorID = CursorManager.setCursor(sizeWECursorSymbol, 2, -10, -11);    
      break;
     case cursorSizeNW:
     case cursorSizeSE:
      currentCursorID = CursorManager.setCursor(sizeNWSECursorSymbol, 2, -11, -11);
      break;
     case cursorSizeNE:
     case cursorSizeSW:
      currentCursorID = CursorManager.setCursor(sizeNESWCursorSymbol, 2, -11, -10);
      break;
     case cursorSizeN:
     case cursorSizeS:
      currentCursorID = CursorManager.setCursor(sizeNSCursorSymbol, 2, -10, -10);
      break;
    }
   }
      
   protected function titleBar_resizeMoveListener(event:MouseEvent):void
   {
    if (event.target is Button)
    {
     //the base class doesn't give me access to "closeButton", so this
     //is the only way to check if we are over the button
     clearCursor();
     return;
    }

    adjustCursor(event, true);
   }
   
   protected function resizeMoveListener(event:MouseEvent):void
   {
    if (event.localY > getHeaderHeight())  //don't do it twice, the title bar takes care of it
    {
     adjustCursor(event, false);
    }
   }
   
   override protected function startDragging(event:MouseEvent):void
   {
    // check for the threshholds first, 
    // if we are within the threshold do our stuff, else call super
    var cursorStyle:int = getCursorStyle(event.localX, event.localY, true);
    if (cursorStyle != cursorSizeNone && cursorStyle != cursorSizeAll )
    {
     startSizing(cursorStyle, event.stageX, event.stageY);
    } else {
     super.startDragging(event);
    }
   }
   
   protected function resizeDownListener(event:MouseEvent):void
   {
    // check for the threshholds first,
    // if we are within the threshold do our stuff, else call super
    var cursorStyle:int = getCursorStyle(event.localX, event.localY, false);
    if (cursorStyle != cursorSizeNone && cursorStyle != cursorSizeAll)
    {
     startSizing(cursorStyle, event.stageX, event.stageY);
    }     
   }
   
   protected function startSizing(cursor:int, x:int, y:int):void
   {
    downX = x;
    downY = y;
    startHeight = this.height;
    startWidth = this.width;
    startLeft = this.x;
    startTop = this.y;
    resizeCursor = cursor;
    isResizing = true;        
        
    systemManager.addEventListener(
     MouseEvent.MOUSE_MOVE, systemManager_resizeMouseMoveHandler, true);
  
    systemManager.addEventListener(
     MouseEvent.MOUSE_UP, systemManager_resizeMouseUpHandler, true);
  
    stage.addEventListener(
     Event.MOUSE_LEAVE, stage_resizeMouseLeaveHandler);
   }   
 
   protected function stopSizing():void
   {
    isResizing = false;
    
    systemManager.removeEventListener(
     MouseEvent.MOUSE_MOVE, systemManager_resizeMouseMoveHandler, true);
  
    systemManager.removeEventListener(
     MouseEvent.MOUSE_UP, systemManager_resizeMouseUpHandler, true);
  
    stage.removeEventListener(
     Event.MOUSE_LEAVE, stage_resizeMouseLeaveHandler);

    clearCursor();
   }
   
   private function sizeWidth(event:MouseEvent):void
   {
    var tmp:int;
    tmp = startWidth + event.stageX - downX;
    if (tmp >= minSizeWidth)  
    {
     this.width = tmp;
    }    
   }

   private function sizeHeight(event:MouseEvent):void
   {
    var tmp:int;
    tmp = startHeight + event.stageY - downY;
    if (tmp >= minSizeHeight)  
    {
     this.height = tmp;
    }
   }
      
   private function sizeTop(event:MouseEvent):int
   {
    var tmp:int;
    var delta:int = downY - event.stageY;
    tmp = startHeight + delta;
    if (tmp < minSizeHeight)  
    {     
     delta = minSizeHeight - startHeight;
     tmp = startHeight + delta;     
    }
    
    this.height = tmp;
    return delta;
//    return 0;
   }

   private function sizeLeft(event:MouseEvent):int
   {
    var tmp:int;
    var delta:int = downX - event.stageX;
    tmp = startWidth + delta;
    if (tmp < minSizeWidth)  
    {
     delta = minSizeWidth - startWidth;
     tmp = startWidth + delta;
    }
    
    this.width = tmp;
    return delta;
//    return 0;
   }

   /**
    *  @private
    */
   private function systemManager_resizeMouseMoveHandler(event:MouseEvent):void
   {
    var leftDelta:int=0;
    var topDelta:int=0;    
    switch (resizeCursor)
    {
     case cursorSizeE:
      sizeWidth(event);
      break;
     case cursorSizeSE:
      sizeWidth(event);
      sizeHeight(event);
      break;      
     case cursorSizeS:
      sizeHeight(event);
      break;
     case cursorSizeSW:
      leftDelta = sizeLeft(event);
      sizeHeight(event);
      break;
     case cursorSizeW:
      leftDelta = sizeLeft(event);
      break;
     case cursorSizeNW:
      topDelta = sizeTop(event);
      leftDelta = sizeLeft(event);
      break;
     case cursorSizeN:
      topDelta = sizeTop(event);
      break;
     case cursorSizeNE:
      topDelta = sizeTop(event);
      sizeWidth(event);
      break;
    }
    
    // when sizing, we only want to do the move once (multiple moves cause ugly refresh problems)
    // a move happens when dragging involves the left or top side    
    if (leftDelta != 0 || topDelta != 0)
    {
     move(startLeft - leftDelta, startTop - topDelta);
    }
   }
       
   /**
    *  @private
    */
   private function systemManager_resizeMouseUpHandler(event:MouseEvent):void
   {
    stopSizing();
   }
  
   /**
    *  @private
    */
   private function stage_resizeMouseLeaveHandler(event:Event):void
   {
    stopSizing();
   }   
   
   private function cursorMouseOutListener(event:MouseEvent):void
   {
    if (!isResizing)
    {
     clearCursor();
    }
   }
  
 }
}

You can find the associated images for this class in the zip file here.

posted Friday, 9 June 2006

More Stories By Flex News Desk

Flex News Desk provides the very latest news on the cross-platform Flex development framework for creating rich Internet applications, and on Adobe's AIR/Flex/Flash product combination.

Comments (3) View Comments

Share your thoughts on this story.

Add your comment
You must be signed in to add a comment. Sign-in | Register

In accordance with our Comment Policy, we encourage comments that are on topic, relevant and to-the-point. We will remove comments that include profanity, personal attacks, racial slurs, threats of violence, or other inappropriate material that violates our Terms and Conditions, and will block users who make repeated violations. We ask all readers to expect diversity of opinion and to treat one another with dignity and respect.


Most Recent Comments
Todd Cullen 05/22/07 01:19:48 AM EDT

I realize that your trying to make one class to encompass the entire window's functionality but wouldn't you want to use the Decorator design pattern to seperate some of the functionality from the core class? Improving its reuseability in the long-run. Solid code, just a question regarding your design pattern.

SYS-CON Italy News Desk 06/09/06 02:12:43 PM EDT

In playing around with the ColdFusion/Flex wizards, I found that Adobe includes a really nice resizeable closeable window. When I get more time, I'll integrate this with my MaximizeRestorePanel class.

Web Developer's & Designer's Journal 06/09/06 01:32:26 PM EDT

In playing around with the ColdFusion/Flex wizards, I found that Adobe includes a really nice resizeable closeable window. When I get more time, I'll integrate this with my MaximizeRestorePanel class.

IoT & Smart Cities Stories
To Really Work for Enterprises, MultiCloud Adoption Requires Far Better and Inclusive Cloud Monitoring and Cost Management … But How? Overwhelmingly, even as enterprises have adopted cloud computing and are expanding to multi-cloud computing, IT leaders remain concerned about how to monitor, manage and control costs across hybrid and multi-cloud deployments. It’s clear that traditional IT monitoring and management approaches, designed after all for on-premises data centers, are falling short in ...
We are seeing a major migration of enterprises applications to the cloud. As cloud and business use of real time applications accelerate, legacy networks are no longer able to architecturally support cloud adoption and deliver the performance and security required by highly distributed enterprises. These outdated solutions have become more costly and complicated to implement, install, manage, and maintain.SD-WAN offers unlimited capabilities for accessing the benefits of the cloud and Internet. ...
The deluge of IoT sensor data collected from connected devices and the powerful AI required to make that data actionable are giving rise to a hybrid ecosystem in which cloud, on-prem and edge processes become interweaved. Attendees will learn how emerging composable infrastructure solutions deliver the adaptive architecture needed to manage this new data reality. Machine learning algorithms can better anticipate data storms and automate resources to support surges, including fully scalable GPU-c...
The Founder of NostaLab and a member of the Google Health Advisory Board, John is a unique combination of strategic thinker, marketer and entrepreneur. His career was built on the "science of advertising" combining strategy, creativity and marketing for industry-leading results. Combined with his ability to communicate complicated scientific concepts in a way that consumers and scientists alike can appreciate, John is a sought-after speaker for conferences on the forefront of healthcare science,...
Machine learning has taken residence at our cities' cores and now we can finally have "smart cities." Cities are a collection of buildings made to provide the structure and safety necessary for people to function, create and survive. Buildings are a pool of ever-changing performance data from large automated systems such as heating and cooling to the people that live and work within them. Through machine learning, buildings can optimize performance, reduce costs, and improve occupant comfort by ...
René Bostic is the Technical VP of the IBM Cloud Unit in North America. Enjoying her career with IBM during the modern millennial technological era, she is an expert in cloud computing, DevOps and emerging cloud technologies such as Blockchain. Her strengths and core competencies include a proven record of accomplishments in consensus building at all levels to assess, plan, and implement enterprise and cloud computing solutions. René is a member of the Society of Women Engineers (SWE) and a m...
Poor data quality and analytics drive down business value. In fact, Gartner estimated that the average financial impact of poor data quality on organizations is $9.7 million per year. But bad data is much more than a cost center. By eroding trust in information, analytics and the business decisions based on these, it is a serious impediment to digital transformation.
DXWorldEXPO LLC announced today that Ed Featherston has been named the "Tech Chair" of "FinTechEXPO - New York Blockchain Event" of CloudEXPO's 10-Year Anniversary Event which will take place on November 12-13, 2018 in New York City. CloudEXPO | DXWorldEXPO New York will present keynotes, general sessions, and more than 20 blockchain sessions by leading FinTech experts.
Apps and devices shouldn't stop working when there's limited or no network connectivity. Learn how to bring data stored in a cloud database to the edge of the network (and back again) whenever an Internet connection is available. In his session at 17th Cloud Expo, Ben Perlmutter, a Sales Engineer with IBM Cloudant, demonstrated techniques for replicating cloud databases with devices in order to build offline-first mobile or Internet of Things (IoT) apps that can provide a better, faster user e...
Bill Schmarzo, Tech Chair of "Big Data | Analytics" of upcoming CloudEXPO | DXWorldEXPO New York (November 12-13, 2018, New York City) today announced the outline and schedule of the track. "The track has been designed in experience/degree order," said Schmarzo. "So, that folks who attend the entire track can leave the conference with some of the skills necessary to get their work done when they get back to their offices. It actually ties back to some work that I'm doing at the University of ...