forked from: MenuDock

by bradsedito forked from MenuDock (diff: 2)
[AS3.0] MenuDockクラスに挑戦! (2)
http://www.project-nya.jp/modules/weblog/details.php?blog_id=1143

参考資料 : http://www.egashira.jp/as/dock1.html
♥1 | Line 333 | Modified 2011-04-23 16:58:46 | MIT License
play

ActionScript3 source code

/**
 * Copyright bradsedito ( http://wonderfl.net/user/bradsedito )
 * MIT License ( http://www.opensource.org/licenses/mit-license.php )
 * Downloaded from: http://wonderfl.net/c/cm6L
 */

// forked from ProjectNya's MenuDock
////////////////////////////////////////////////////////////////////////////////
// [AS3.0] MenuDockクラスに挑戦! (2)
// http://www.project-nya.jp/modules/weblog/details.php?blog_id=1143
//
// 参考資料 : http://www.egashira.jp/as/dock1.html
////////////////////////////////////////////////////////////////////////////////

package {

    import flash.display.Sprite;

    [SWF(backgroundColor="#FFFFFF", width="465", height="465", frameRate="30")]

    public class Main extends Sprite {
        private var iconList:Array;
        private var menuDock:MenuDock;

        public function Main() {
            init();
        }

        private function init():void {
            var background:Background = new Background(465, 465);
            addChild(background);
            iconList = new Array();
            var icon1:Icon = new Icon(112);
            icon1.init([0x3366CC, 0x333366], "Ps", 64, 14);
            icon1.y = - 16;
            icon1.scaleX = icon1.scaleY = 0.25;
            var icon2:Icon = new Icon(112);
            icon2.init([0xFFCC00, 0xFF6600], "Ai", 64, 14);
            icon2.y = - 16;
            icon2.scaleX = icon2.scaleY = 0.25;
            var icon3:Icon = new Icon(112);
            icon3.init([0xCC0000, 0x660000], "Fl", 64, 14);
            icon3.y = - 16;
            icon3.scaleX = icon3.scaleY = 0.25;
            var icon4:Icon = new Icon(112);
            icon4.init([0x999999, 0x333333], "Fx", 64, 14);
            icon4.y = - 16;
            icon4.scaleX = icon4.scaleY = 0.25;
            var icon5:Icon = new Icon(112);
            icon5.init([0xFFCC00, 0x996600], "Fw", 64, 14);
            icon5.y = - 16;
            icon5.scaleX = icon5.scaleY = 0.25;
            var icon6:Icon = new Icon(112);
            icon6.init([0x99CC33, 0x666633], "Dw", 64, 14);
            icon6.y = - 16;
            icon6.scaleX = icon6.scaleY = 0.25;
            iconList.push({icon: icon1, label: "Photoshop"});
            iconList.push({icon: icon2, label: "Illustrator"});
            iconList.push({icon: icon3, label: "Flash"});
            iconList.push({icon: icon4, label: "Flex"});
            iconList.push({icon: icon5, label: "Fireworks"});
            iconList.push({icon: icon6, label: "Dreamweaver"});
            menuDock = new MenuDock(iconList);
            addChild(menuDock);
            menuDock.x = 232;
            menuDock.y = 465;
        }

    }

}


//////////////////////////////////////////////////
// MenuDockクラス
//////////////////////////////////////////////////

import flash.display.Sprite;
import flash.display.Shape;
import flash.events.Event;
import flash.events.MouseEvent;

class MenuDock extends Sprite {
    private var base:Shape;
    private var iconList:Array;
    private var max:uint;
    private var bWidth:uint;
    private static var iWidth:uint = 32;
    private var limit:uint;
    private var icons:Array;
    private static var scale:uint = 4;
    private static var dockBase:uint = 32;
    private static var dockTarget:uint = dockBase*scale;
    private var dockTop:uint = dockBase;
    private var limitTarget:Number = dockTarget/(scale - 1)*Math.PI/2;
    private static var offset:uint = 5;
    private static var deceleration:Number = 0.5;
    private var active:Boolean = false;

    public function MenuDock(list:Array) {
        iconList = list;
        max = iconList.length;
        limit = iWidth*max*0.5 + offset;
        icons = new Array();
        if (stage) init();
        else addEventListener(Event.ADDED_TO_STAGE, init, false, 0, true);
    }

    private function init(evt:Event = null):void {
        removeEventListener(Event.ADDED_TO_STAGE, init);
        drawBase();
        drawIcons();
        active = true;
        addEventListener(Event.ENTER_FRAME, motion, false, 0, true);
        stage.addEventListener(Event.MOUSE_LEAVE, leave, false, 0, true);
        stage.addEventListener(MouseEvent.MOUSE_OVER, over, false, 0, true);
    }
    private function motion(evt:Event):void {
        if (active) {
            if (Math.abs(mouseX) <= limit && mouseY <= 0 && mouseY >= -dockTop) {
                moveTarget();
            } else {
                moveBase();
            }
        } else {
            moveBase();
        }
        resizeBase();
    }
    private function leave(evt:Event):void {
        active = false;
    }
    private function over(evt:MouseEvent):void {
        active = true;
    }
    private function moveTarget():void {
        dockTop = dockTarget;
        for (var n:uint = 0; n < max; n++) {
        var icon:MenuDockIcon = icons[n];
        var basePos:int = iWidth*(n - (max - 1)*0.5);
        var distance:int = mouseX - basePos;
        var radian:Number = distance*(scale - 1)/dockTarget;
        if (Math.abs(distance) < limitTarget) {
            var targetScale:Number = 1 + (scale - 1)*Math.cos(radian);
            var targetPos:Number = basePos - dockTarget*Math.sin(radian);
            icon.scale += (targetScale - icon.scale)*deceleration;
            icon.x += (targetPos - icon.x)*deceleration;
        } else {
            icon.scale += (1 - icon.scale)*deceleration;
            if (distance < -limitTarget) {
                icon.x += (basePos + dockTop + 10 - icon.x)*deceleration;
            } else {
                icon.x += (basePos - dockTop - 10 - icon.x)*deceleration;
            }
        }
        }
    }
    private function moveBase():void {
        dockTop = dockBase;
        for (var n:uint = 0; n < max; n++) {
            var icon:MenuDockIcon = icons[n];
            var basePos:int = iWidth*(n - (max - 1)*0.5);
            icon.scale += (1 - icon.scale)*deceleration;
            icon.x += (basePos - icon.x)*deceleration;
        }
    }
    private function resizeBase():void {
        var first:Number = icons[0].x - icons[0].base.width*0.5;
        var last:Number = icons[max-1].x + icons[max-1].base.width*0.5;
        base.width = last - first + offset*2;
        base.x = first + base.width*0.5 - offset;
    }
    private function drawBase():void {
        base = new Shape();
        addChild(base);
        var bWidth:uint = iWidth*max;
        base.graphics.beginFill(0xFFFFFF, 0.5);
        base.graphics.drawRect(-bWidth*0.5, 0, bWidth, -iWidth);
        base.graphics.endFill();
    }
    private function drawIcons():void {
        for (var n:uint = 0; n < max; n++) {
            var icon:MenuDockIcon = new MenuDockIcon(n, iWidth, iconList[n].icon);
            addChild(icon);
            icon.x = iWidth*(n - (max - 1)*0.5);
            icon.y = -1;
            icon.text = iconList[n].label;
            icons.push(icon);
        }
    }

}


//////////////////////////////////////////////////
// MenuDockIconクラス
//////////////////////////////////////////////////

import flash.display.Sprite;
import flash.display.DisplayObject;
import flash.events.MouseEvent;

class MenuDockIcon extends Sprite {
    public var id:uint;
    public var base:Sprite;
    private var _width:uint;
    private var _scale:Number = 1;
    private var icon:DisplayObject;
    private var label:MenuDockLabel;

    public function MenuDockIcon(n:uint, w:uint, i:DisplayObject) {
        id = n;
        _width = w;
        icon = i;
        draw();
    }

    private function draw():void {
        base = new Sprite();
        addChild(base);
        base.graphics.beginFill(0xFFFFFF, 0);
        base.graphics.drawRect(-_width*0.5, -_width, _width, _width);
        base.graphics.endFill();
        base.addChild(icon);
        label = new MenuDockLabel();
        label.y = - base.height;
        buttonMode = true;
        mouseChildren = false;
        addEventListener(MouseEvent.MOUSE_OVER, rollOver, false, 0, true);
        addEventListener(MouseEvent.MOUSE_OUT, rollOut, false, 0, true);
    }
    private function rollOver(evt:MouseEvent):void {
        addChild(label);
    }
    private function rollOut(evt:MouseEvent):void {
        if (contains(label)) removeChild(label);
    }
    public function get scale():Number {
        return _scale;
    }
    public function set scale(param:Number):void {
        _scale = param;
        base.scaleX = base.scaleY = _scale;
        label.y = - base.height;
    }
    public function set text(param:String):void {
        label.text = param;
    }

}


//////////////////////////////////////////////////
// MenuDockLabelクラス
//////////////////////////////////////////////////

import flash.display.Sprite;
import flash.text.TextField;
import flash.text.TextFieldType;
import flash.text.TextFieldAutoSize;
import flash.text.AntiAliasType;
import flash.text.TextFormat;
import flash.text.TextFormatAlign;
import flash.filters.DropShadowFilter;

class MenuDockLabel extends Sprite {
    private var txt:TextField;
    private static var fontType:String = "_ゴシック";
    private static var _height:uint = 22;
    private static var yOffset:uint = 6;
    private static var bColor:uint = 0xFFFFFF;
    private static var sColor:uint = 0x000000;

    public function MenuDockLabel() {
        draw();
    }

    private function draw():void {
        txt = new TextField();
        addChild(txt);
        txt.height = _height - 1;
        txt.y = - _height + yOffset;
        txt.autoSize = TextFieldAutoSize.CENTER;
        txt.type = TextFieldType.DYNAMIC;
        txt.selectable = false;
        //txt.embedFonts = true;
        //txt.antiAliasType = AntiAliasType.ADVANCED;
        //txt.antiAliasType = AntiAliasType.NORMAL;
        var tf:TextFormat = new TextFormat();
        tf.font = fontType;
        tf.size = 12;
        tf.align = TextFormatAlign.CENTER;
        txt.defaultTextFormat = tf;
        txt.textColor = bColor;
        var shade:DropShadowFilter = new DropShadowFilter(1, 90, sColor, 1, 2, 2, 2, 3, false, false, false);
        txt.filters = [shade];
    }
    public function set text(param:String):void {
        txt.text = param;
        txt.x = - uint(txt.width*0.5);
    }

}


//////////////////////////////////////////////////
// Iconクラス
//////////////////////////////////////////////////

import flash.display.Sprite;
import flash.display.Shape;
import flash.geom.Matrix;
import flash.display.GradientType;
import flash.display.SpreadMethod;
import flash.display.InterpolationMethod;
import flash.filters.DropShadowFilter;
import flash.text.TextField;
import flash.text.TextFieldType;
import flash.text.AntiAliasType;
import flash.text.TextFormat;
import flash.text.TextFormatAlign;

class Icon extends Sprite {
    private var size:uint;
    private var colors:Array;
    private var txt:TextField;
    private static var fontType:String = "_ゴシック";
    private var fontSize:uint;
    private var yOffset:int;

    public function Icon(s:uint) {
        size = s;
    }

    public function init(list:Array, t:String, s:uint, y:int = 0):void {
        colors = list;
        fontSize = s;
        yOffset = y;
        draw();
        txt.text = t;
    }
    private function draw():void {
        var base:Shape = new Shape();
        addChild(base);
        var alphas:Array = [1, 1];
        var ratios:Array = [0, 255];
        var matrix:Matrix = new Matrix();
        var offset:Number = size*0.1;
        matrix.createGradientBox(size*2, size*2, 0, -size*1.5+offset, -size*1.5+offset);
        base.graphics.beginGradientFill(GradientType.RADIAL, colors, alphas, ratios, matrix, SpreadMethod.PAD, InterpolationMethod.RGB, 0);
        base.graphics.drawRect(-size*0.5, -size*0.5, size, size);
        base.graphics.endFill();
        var shade:DropShadowFilter = new DropShadowFilter(4, 90, 0x000000, 0.3, 8, 8, 2, 3, false, false);
        base.filters = [shade];
        txt = new TextField();
        addChild(txt);
        txt.x = -size*0.5;
        txt.y = -size*0.5 + yOffset;
        txt.width = size;
        txt.height = size;
        txt.type = TextFieldType.DYNAMIC;
        txt.selectable = false;
        //txt.embedFonts = true;
        //txt.antiAliasType = AntiAliasType.ADVANCED;
        //txt.antiAliasType = AntiAliasType.NORMAL;
        var tf:TextFormat = new TextFormat();
        tf.font = fontType;
        tf.size = fontSize;
        tf.align = TextFormatAlign.CENTER;
        txt.defaultTextFormat = tf;
        txt.textColor = 0xFFFFFF;
        var shadow:DropShadowFilter = new DropShadowFilter(0, 90, 0x000000, 0.5, 2, 2, 2, 3, false, false);
        txt.filters = [shadow];
    }

}


//////////////////////////////////////////////////
// Backgroundクラス
//////////////////////////////////////////////////

import flash.display.Shape;
import flash.geom.Matrix;
import flash.display.GradientType;

class Background extends Shape {
    private static var _width:uint;
    private static var _height:uint;
    private static var color1:uint = 0x77B2EE;
    private static var color2:uint = 0x3F68AB;

    public function Background(w:uint, h:uint) {
        _width = w;
        _height = h;
        draw();
    }

    private function draw():void {
        var colors:Array = [color1, color2];
        var alphas:Array = [1, 1];
        var ratios:Array = [0, 255];
        var matrix:Matrix = new Matrix();
        matrix.createGradientBox(_width, _height, 0.5*Math.PI, 0, 0);
        graphics.beginGradientFill(GradientType.LINEAR, colors, alphas, ratios, matrix);
        graphics.drawRect(0, 0, _width, _height);
        graphics.endFill();
    }

}