查看: 3600|回复: 6
打印 上一主题 下一主题

[Flare3D] Flash制作3D类动画

[复制链接]

9210

主题

1

听众

5万

积分

内部人员

Rank: 7Rank: 7Rank: 7

纳金币
12582
精华
186

活跃会员

跳转到指定楼层
楼主
发表于 2013-9-26 16:38:44 |只看该作者 |倒序浏览

最近对Flash3D效果产生了兴趣,下决心学习一下Sandy。一下将学习的体会的感受写下来,一来帮助大家熟悉Sandy,二来加强自己的记忆。

先从基础开始吧


原理解释:

  • 窗口:
    用户观看的窗口,简单的可以想成就是flash里面的画布大小。窗口也可以理解成渲染的尺寸,否则画面就无限大了
  • 场景:
    场景是指整个三维的场景。
  • 摄像机:
    很多人要问,为什么有了摄像机还要窗口呢?摄像机是用来拍画面的,看画面还是得电视机/窗口不是吗,^_^
    渲染器:如果没有这个东西,所有以上的东西都只是数据,渲染器就是把所有数据变成图像的东西。

下面这幅图虽然并不算准确的表述,但希望能帮助我们理解:

然后开始写代码了(目前感觉Sandy的代码还是比较简洁的):

要先将Sandy的类库下载下来哦!!
官方网站:http://www.flashsandy.org
下载地址:http://sandy.googlecode.com/files/Sandy3-1-1_src_rev1008.zip

先尝试创建一个立方体。
package
{
    import flash.display.Sprite;
    import flash.events.Event;
    import sandy.core.Scene3D;
    import sandy.core.scenegraph.*;
    import sandy.primitive.*;
   
    /**
     * ...
     * @author ever5u
     */
    public class fuxi extends Sprite
    {
        private var scene:Scene3D;
        private var camera:Camera3D;
        public function fuxi() {
            //创建一个摄像机
            camera = new Camera3D(300, 300);
            camera.z = -300;
            //创建一个 Group
            var root:Group = createScene();
            //创建场景
            scene = new Scene3D( "scene", this, camera, root );
            //创建实时侦听
            addEventListener( Event.ENTER_FRAME, enterFrameHandler );
        }
        var box = new Box("box", 100, 100, 100);
        public function createScene() {
            var g:Group = new Group();
            g.addChild( box );
            return g;
        }
        public function enterFrameHandler(_evt:Event) {
            box.rotateX = mouseX;
            box.rotateY = mouseY;
            scene.render();
        }
    }
   
}
效果如下:


接下来尝试给这个立方体着色。

这里需要用到 sandy.materials.attributes 类。

其中为线着色的方法 LineAttributes 有三个属性:
LineAttributes(p_nThickness:uint = 1,  p_nColor:uint = 0,  p_nAlpha:Number = 1)

  • p_nThickness:uint (default = 1) — 线的粗细
  • p_nColor:uint (default = 0) — 线的颜色
  • p_nAlpha:Number (default = 1) —  线的透明度

这里设置是否使用光,需要先设置 lightingEnable = true 。
LightAttributes(p_bBright:Boolean = false, p_nAmbient:Number = 0.3)

  • p_bBright:Boolean (default = false) — 设置是否支持光
  • p_nAmbient:Number (default = 0.3) — 设置光的亮度(数值范围是 0 - 1)

为立方体渲染用
ColorMaterial(p_nColor:uint = 0x00, p_nAlpha:Number = 1, p_oAttr:MaterialAttributes = null)

  • p_nColor:uint (default = 0x00) — 颜色
  • p_nAlpha:Number (default = 1) — 透明度
  • p_oAttr:MaterialAttributes (default = null) — 线设置

代码如下:
package
{
    import flash.display.Sprite;
    import flash.events.Event;
    import sandy.core.Scene3D;
    import sandy.core.scenegraph.*;
    import sandy.primitive.*;
    import sandy.materials.*;
    import sandy.materials.attributes.*;
   
    /**
     * ...
     * @author ever5u
     */
    public class fuxi extends Sprite
    {
        private var scene:Scene3D;
        private var camera:Camera3D;
        public function fuxi() {
            //创建一个摄像机
            camera = new Camera3D(300, 300);
            camera.z = -300;
            //创建一个 Group
            var root:Group = createScene();
            //创建场景
            scene = new Scene3D( "scene", this, camera, root );
            //创建实时侦听
            addEventListener( Event.ENTER_FRAME, enterFrameHandler );
        }
        var box = new Box("box", 100, 100, 100);
        public function createScene() {
            var g:Group = new Group();
            //设置立方体的颜色、线条色和环境光
            material.lightingEnable = true;
            var materialAttr:MaterialAttributes = new MaterialAttributes(
               new LineAttributes( 0.5, 0x000000, 0.4 ),
               new LightAttributes( true, 0.2)
            );
            var material:Material = new ColorMaterial( 0xCC3300, 1, materialAttr );
            var app:Appearance = new Appearance( material );
            
            box.appearance = app;
            
            g.addChild( box );
            return g;
        }
        public function enterFrameHandler(_evt:Event) {
            box.rotateX = mouseX;
            box.rotateY = mouseY;
            scene.render();
        }
    }
   
}
这是效果:


继续试试看用图片为立方体贴图。

首先导入一张位图到库里,并声明类名为 MyPalm
package
{
    import flash.display.Sprite;
    import flash.events.Event;
    import flash.display.Bitmap;
    import flash.display.BitmapData;
    import flash.display.Loader;
    import flash.net.URLRequest;
    import sandy.core.Scene3D;
    import sandy.core.scenegraph.*;
    import sandy.primitive.*;
    import sandy.materials.*;
    import sandy.materials.attributes.*;
   
    /**
     * ...
     * @author ever5u
     */
    public class fuxi extends Sprite
    {
        private var scene:Scene3D;
        private var camera:Camera3D;
        public function fuxi() {
            //创建一个摄像机
            camera = new Camera3D(300, 300);
            camera.z = -300;
            //创建一个 Group
            var root:Group = createScene();
            //创建场景
            scene = new Scene3D( "scene", this, camera, root );
            //创建实时侦听
            addEventListener( Event.ENTER_FRAME, enterFrameHandler );
        }
        var box = new Box("box", 100, 100, 100);
        public function createScene() {
            var g:Group = new Group();
            //设置立方体的贴图
            var bitmap:BitmapData = new MyPalm(0, 0);
            var material:Material = new BitmapMaterial( bitmap );
            var app:Appearance = new Appearance( material );
            
            box.appearance = app;
            
            g.addChild( box );
            return g;
        }
        public function enterFrameHandler(_evt:Event) {
            box.rotateX = mouseX;
            box.rotateY = mouseY;
            scene.render();
        }
    }
   
}

效果:


继续上次的学习,接下来要试试摄像机的移动了。
这里需要涉及几个概念,摄像机的坐标(x, y, z)和视觉角度(LookAt)
摄像机的位置可以使用(x, y, z)来定位;
视觉角度可以定义如何通过窗口来看场景。
lookAt(p_nX:Number, p_nY:Number, p_nZ:Number)
如:lookAt(0,0,0);//可以理解为通过摄像机的位置看场景。

摄像机的移动方式比较有意思,如果直接修改x、y、z坐标,视觉效果会与现实看到的情况相同;
而tilt、pan则是与场景平行移动;
roll是以z轴移动,通过摄像机视野看上去是摄像机的旋转效果。

额外说一下 Line3D 是在场景中绘制了线段,这里用做参考线,代码不难理解我就不过多解释了。

代码如下:
package
{
    import flash.display.Sprite;
    import flash.events.*;
    import flash.ui.*;
    import flash.display.Bitmap;
    import flash.display.BitmapData;
    import flash.display.Loader;
    import flash.net.URLRequest;
    import sandy.core.Scene3D;
    import sandy.core.scenegraph.*;
    import sandy.primitive.*;
    import sandy.materials.*;
    import sandy.materials.attributes.*;
    import sandy.core.data.*;
   
    /**
     * ...
     * @author ever5u
     */
    public class fuxi extends Sprite
    {
        private var scene:Scene3D;
        private var camera:Camera3D;
        public function fuxi() {
            //创建一个摄像机
            camera = new Camera3D(300, 300);
            camera.x = 100;
            camera.y = 100;
            camera.z = -300;
            camera.lookAt(0,0,0);
            //创建一个 Group
            var root:Group = createScene();
            //创建场景
            scene = new Scene3D( "scene", this, camera, root );
            //创建实时侦听
            addEventListener( Event.ENTER_FRAME, enterFrameHandler );
            stage.addEventListener(KeyboardEvent.KEY_DOWN, keyPressed);
        }
        var box = new Box("box", 100, 100, 100);
        var Planelane3D;
        public function createScene() {
            var g:Group = new Group();
            
            //在场景画一个坐标定位点
            Plane = new Plane3D("Texture", 300, 300);
            var myXLineine3D = new Line3D( "x-coord", new Point3D( -20, 0, 0), new Point3D( 20, 0, 0 ));
            var myYLineine3D = new Line3D( "y-coord", new Point3D(0, -20, 0), new Point3D( 0, 20, 0 ));
            var myZLineine3D = new Line3D( "z-coord", new Point3D(0, 0, -20), new Point3D( 0, 0, 20 ));
            
            g.addChild(myXLine);
            g.addChild(myYLine);
            g.addChild(myZLine);
            g.addChild( box );
            return g;
        }
        public function enterFrameHandler(_evt:Event) {
            box.rotateX = mouseX;
            box.rotateY = mouseY;
            scene.render();
        }
        public function keyPressed(_evt:KeyboardEvent):void {
            switch(_evt.keyCode) {
                case Keyboard.UP:
                    camera.tilt += 2;
                    //camera.y -= 2;
                    break;
                case Keyboard.DOWN:   
                    camera.tilt -= 2;
                    //camera.y += 2;
                    break;
                case Keyboard.RIGHT:
                    camera.pan -= 2;
                    //camera.x += 2;
                    break;
                case Keyboard.LEFT:
                    camera.pan += 2;
                    //camera.x -= 2;
                    break;
                case Keyboard.CONTROL:
                    camera.roll += 2;
                    break;   
                case Keyboard.PAGE_DOWN:
                    camera.z -= 5;
                    break;
                case Keyboard.PAGE_UP:
                    camera.z += 5;
                    break;   
            }
        }
    }
   
}

效果:


还有场景自适应的问题,需要说明一下。

摄像机的镜头可视角度使用 fov(vertical field of view angle)来定义。

换算公式:

换算公式:

var fl:Numer = (viewport.height / 2) / Math.tan (camera.fov / 2 * (Math.PI / 180));

代码:

package
{
    import flash.display.Sprite;
    import flash.events.*;
    import flash.ui.*;
    import flash.display.Bitmap;
    import flash.display.BitmapData;
    import flash.display.Loader;
    import flash.net.URLRequest;
    import flash.display.Stage;
    import flash.display.StageAlign;
    import flash.display.StageScaleMode;
    import sandy.core.Scene3D;
    import sandy.core.scenegraph.*;
    import sandy.primitive.*;
    import sandy.materials.*;
    import sandy.materials.attributes.*;
    import sandy.core.data.*;
   
    /**
     * ...
     * @author ever5u
     */
    public class fuxi extends Sprite
    {
        private var scene:Scene3D;
        private var camera:Camera3D;
        public function fuxi() {
            stage.scaleMode = StageScaleMode.NO_SCALE;
            stage.align = StageAlign.TOP_LEFT;
            //创建一个摄像机
            camera = new Camera3D(300, 300);
            camera.x = 100;
            camera.y = 100;
            camera.z = -300;
            camera.lookAt(0,0,0);
            //创建一个 Group
            var root:Group = createScene();
            //创建场景
            scene = new Scene3D( "scene", this, camera, root );
            //创建实时侦听
            addEventListener( Event.ENTER_FRAME, enterFrameHandler );
            stage.addEventListener (Event.RESIZE, onResize);
        }
        var box = new Box("box", 100, 100, 100);
        var Planelane3D;
        public function createScene() {
            var g:Group = new Group();
            
            //在场景画一个坐标定位点
            Plane = new Plane3D("Texture", 300, 300);
            var myXLineine3D = new Line3D( "x-coord", new Point3D( -20, 0, 0), new Point3D( 20, 0, 0 ));
            var myYLineine3D = new Line3D( "y-coord", new Point3D(0, -20, 0), new Point3D( 0, 20, 0 ));
            var myZLineine3D = new Line3D( "z-coord", new Point3D(0, 0, -20), new Point3D( 0, 0, 20 ));
            
            g.addChild(myXLine);
            g.addChild(myYLine);
            g.addChild(myZLine);
            g.addChild( box );
            return g;
        }
        public function enterFrameHandler(_evt:Event) {
            box.rotateX = mouseX;
            box.rotateY = mouseY;
            scene.render();
        }
        function onResize (e:Event):void{
            // 获取场景宽高
            var w:Number = stage.stageWidth;
            var h:Number = stage.stageHeight;
            // 设置视野宽高
            scene.camera.viewport.width = w;
            scene.camera.viewport.height = h;
            // 获取物体box与摄像机间的距离
            var d:Number = box.getPosition ("camera").getNorm ();
            // 保持摄像机观看比例
            scene.camera.fov = 2 * Math.atan2 (h / 2, d) * (180 / Math.PI);
            // 执行渲染
            scene.render();
        }
    }
   
}



分享到: QQ好友和群QQ好友和群 腾讯微博腾讯微博 腾讯朋友腾讯朋友 微信微信
转播转播0 分享淘帖0 收藏收藏0 支持支持0 反对反对0
回复

使用道具 举报

0

主题

1

听众

2286

积分

中级设计师

Rank: 5Rank: 5

纳金币
0
精华
0

最佳新人 活跃会员 热心会员 灌水之王 突出贡献

沙发
发表于 2013-10-28 23:35:19 |只看该作者
thanks for sharing
回复

使用道具 举报

1

主题

1

听众

108

积分

设计实习生

Rank: 2

纳金币
46
精华
0

最佳新人

板凳
发表于 2014-1-25 10:43:38 |只看该作者
进来,学习下!
回复

使用道具 举报

1

主题

1

听众

108

积分

设计实习生

Rank: 2

纳金币
46
精华
0

最佳新人

地板
发表于 2014-1-25 10:43:56 |只看该作者
谢谢,分享!
回复

使用道具 举报

1

主题

1

听众

108

积分

设计实习生

Rank: 2

纳金币
46
精华
0

最佳新人

5#
发表于 2014-1-25 10:44:27 |只看该作者
强力的支持下!
回复

使用道具 举报

0

主题

1

听众

2458

积分

中级设计师

Rank: 5Rank: 5

纳金币
0
精华
0

最佳新人 活跃会员 热心会员 灌水之王 突出贡献

6#
发表于 2014-2-22 22:15:20 |只看该作者

谢谢分享
回复

使用道具 举报

0

主题

2

听众

3238

积分

中级设计师

Rank: 5Rank: 5

纳金币
0
精华
0

最佳新人 活跃会员 热心会员 灌水之王 突出贡献

7#
发表于 2014-2-27 14:46:34 |只看该作者
感谢分享
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

手机版|纳金网 ( 闽ICP备2021016425号-2/3

GMT+8, 2025-1-11 10:16 , Processed in 0.070904 second(s), 30 queries .

Powered by Discuz!-创意设计 X2.5

© 2008-2019 Narkii Inc.

回顶部