In case Flash no longer exists; a copy of this site is included in the Flashpoint archive's "ultimate" collection.

Dead Code Preservation :: Archived AS3 works from wonderfl.net

Alternativa3D text3D

------------------------------------------------------------------------------
Alternativa3D text3D
illustratorで作成した文字のパスデータをsvgで受け取って、押し出した3Dtextです。

PV3Dのtext3Dのようにフォントのパスを全部打ち込む事も考えましたが、あまりにもしんどいので、
イラレでパスを作成して、読み込むようにしました。
その他、文字をBitmapDataに転写して、そこから、pathを作成するライブラリも海外にあるのですが、
ライブラリ自体が大きくここでは使えませんし・・・

サンプルのデータはちょっと、面を細かくしすぎてしまったので、少々重いです。

2010/01/19 XMLパースをfrocessing利用に変更

------------------------------------------------------------------------------
Get Adobe Flash player
by narutohyper 19 Oct 2010
/**
 * Copyright narutohyper ( http://wonderfl.net/user/narutohyper )
 * MIT License ( http://www.opensource.org/licenses/mit-license.php )
 * Downloaded from: http://wonderfl.net/c/ii0R
 */

// forked from clockmaker's [Alternativa3D] Basic Template
package
{
	import alternativ5.engine3d.materials.FillMaterial;
	import alternativ5.types.Point3D;
	import alternativ5.engine3d.core.Object3D;
	import alternativ5.utils.*

	import flash.net.URLLoader;
	import flash.net.URLRequest;
	import flash.events.Event;
	import flash.display.Sprite;
	import flash.utils.Dictionary;
	import flash.geom.Point;
	import flash.display.GraphicsPathCommand;

	import frocessing.shape.*;

	/*------------------------------------------------------------------------------
	Alternativa3D text3D
		illustratorで作成した文字のパスデータをsvgで受け取って、押し出した3Dtextです。

		PV3Dのtext3Dのようにフォントのパスを全部打ち込む事も考えましたが、あまりにもしんどいので、
		イラレでパスを作成して、読み込むようにしました。
		その他、文字をBitmapDataに転写して、そこから、pathを作成するライブラリも海外にあるのですが、
		ライブラリ自体が大きくここでは使えませんし・・・
		
		サンプルのデータはちょっと、面を細かくしすぎてしまったので、少々重いです。

		2010/01/19 XMLパースをfrocessing利用に変更

	------------------------------------------------------------------------------*/
	[SWF(backgroundColor="0x000000",width = 465, height = 465,	frameRate="24")]	

	public class SimpleDemo extends Sprite {
		private var dic:Dictionary;
		private var template:BasicTemplate;

		private var loader:URLLoader;

		public function SimpleDemo():void
		{
			// テンプレートを作成します
			template = new BasicTemplate();
			addChild(template);
			template.camera.z = -2000;

			FPS.init(stage)


			//テキスト用パスのSVGファイルをLoadする。
			loader = new URLLoader();
			loader.addEventListener( Event.COMPLETE, onload );

			loader.load( new URLRequest("http://marubayashi.net/archive/sample/images/abc.svg") );

		}


		public function onload( e:Event ):void
		{
			//XML取得
			var svg:XML = XML(loader.data);


			//扱えるのは、一筆書き状態の、複合Pathのみです。
			var sp:FShape
			var temp:Array = [];
			for each( var child:XML in XML(loader.data).children() ) {
				sp = FShapeSVG.parsePath(child);
				sp.styleEnabled = false;
				temp.push({command:sp.commands.concat(), vertices:sp.vertices.concat(),roll:0});
			}

			//各文字に割り付ける
			dic = new Dictionary();
			var rol:String='0100000000000000000000000000000000000000000000000000010000000000000';
			var str:String='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789?!&-_';
			var i:uint=0
			for each (var item:Object in temp) {
				dic[str.charAt(i)]=item
				//イラレから吐き出した場合、右回り左回りが、まちまちなので0(左周り)or1(右回り)で調整する
				//Bと1だけどうしても、ひっくり返らない・・・法則がいまいちわからんw
				dic[str.charAt(i)].roll=uint(rol.charAt(i))
				i++;
			}


			var base:Object3D=new Object3D()
			template.scene.root.addChild(base);

			//実際に書いてみる
			//str='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789?!&-_';
			str="Welcome\nAlternativa3D\nWorld!!";
			var position:Point=new Point();
			var size:Point=new Point();
			var maxWArray:Array=[]
			var maxWCounter:uint=0
			var maxW:Number=0
			var maxH:Number=0
			var strArray:Array=[]

			maxWArray[maxWCounter]=0
			position.x=0
			position.y=-400

			for (i=0;i<str.length;i++) {
				if (str.charAt(i)==" ") {
					position.x+=50
				} else if (str.charAt(i)=="\n") {
					position.y+=300
					position.x=0
					maxWCounter++
					maxWArray[maxWCounter]=0

				} else {
					var temp3D:text3D=new text3D(str.charAt(i),100,dic);
					strArray.push(temp3D);

					temp3D.x=position.x
					temp3D.y=position.y
					position.x+=temp3D.width+10
					maxWArray[maxWCounter]+=temp3D.width+10

					base.addChild(temp3D);

					temp3D.cloneMaterialToAllSurfaces(new FillMaterial(0xFF0000));
					temp3D.setMaterialToSurface(new FillMaterial(0x660000),'bottom');
					temp3D.setMaterialToSurface(new FillMaterial(0x990000),'side');
				}

			}

			//幅の最大値を求める
			for (i=0;i<maxWArray.length;i++) {
				if (maxW<maxWArray[i]) maxW=maxWArray[i]
			}
			for (i=0;i<strArray.length;i++) {
				strArray[i].x-=maxW/2
			}


			// Event.ENTER_FRAME 時に実行されるレンダリングのイベントです。
			// レンダリング前に実行したい処理を記述します。
			template.onPreRender = function():void {

				// 立方体を回転させます (角度はラジアン)
				base.rotationY += 3 * Math.PI / 180;
			
				// マウスがステージの高さ何%の位置にあるか算出
				var rateY:Number = mouseY / stage.stageHeight;
				
				// カメラの高さの座標を調整
				// イージングの公式 対象の値 += (目標値 - 現在の値) * 減速率
				template.camera.y += ( - 1000 * rateY - template.camera.y) * 0.1;
				
				// カメラの座標を中央に向かせる
				template.cameraContoller.lookAt(new Point3D());

			}



		}

	}


}



import alternativ5.types.Point3D;
import alternativ5.engine3d.core.Mesh;
import flash.utils.Dictionary;
import flash.geom.Point;
import flash.display.GraphicsPathCommand;


//---------------------------------------------------------------------------
//与えられたArrayのPoint(2D)情報で面を作成し、3Dに押し出すクラス
//---------------------------------------------------------------------------

class pressMesh extends Mesh{

	public var _width:Number			//x
	public var _length:Number		//y
	public var _height:Number		//z
	public var roll:uint

	public function pressMesh(pt:Array,h:Number,_roll:uint=0,separate:Boolean=false) {
		//pt[faceId:uint][verticesId:uint]:Pointの構造を持つArray
		//separate:Boolean =表面(top || bottom)のface仕様、trueでfaceIdで分割、falseは一枚のfaceにまとめる
		//roll:uint				=Pointの右回り、左回りを指定

		_height=h;
		roll=_roll;

		if (separate) {
			drawPathToPlane(pt,0,'top',false,true)
			drawPathToPlane(pt,height,'bottom',true,true)
		} else {
			drawPathToPlane(pt,0,'top',false,false)
			drawPathToPlane(pt,height,'bottom',true,false)
		}

		drawPathToSide(pt,'side')

		var wMax:Number=0
		var wMin:Number=0
		var hMax:Number=0
		var hMin:Number=0

		for (var n:uint=0; n<pt.length; n++) {
			for (var i:uint=0; i<pt[n].length; i++) {
			if (wMax<pt[n][i].x){
				wMax=pt[n][i].x;
			} else if (wMin>pt[n][i].x){
				wMin=pt[n][i].x;
			}
			if (hMax<pt[n][i].y) {
				hMax=pt[n][i].y;
			} else if (hMin>pt[n][i].y){
				hMin=pt[n][i].y;
			}
			}
		}

		_width=wMax-wMin
		_length=hMax-hMin

		//出来たMeshの中心を0,0,0にする
		for each(var item:* in this.vertices) {
		//	item.x-=width/2
			item.y-=300/2
			item.z-=height/2
		}
		
	}


	//-----------------------------------------------------------
	//sideの形成
	//-----------------------------------------------------------
	private function drawPathToSide(va:Array,id:String):void {
		var delimiter:String='_';
		var i:int;
		var n:int;

		var tId:String;
		var faceArray:Array;
		var surfaceArray:Array;
		surfaceArray=[];
		for (n=va.length-1;n>=0;n--) {
			for (i=1;i< va[n].length;i++) {
				faceArray=[]

				tId=id+delimiter+String(n)+delimiter+String(i)

				if (roll) {
					faceArray.push('bottom'+delimiter+String(n)+delimiter+String(i));

					if (i==va[n].length-1) {
						faceArray.push('bottom'+delimiter+String(n)+delimiter+String(1));
						faceArray.push('top'+delimiter+String(n)+delimiter+String(1));
					} else {
						faceArray.push('bottom'+delimiter+String(n)+delimiter+String(i+1));
						faceArray.push('top'+delimiter+String(n)+delimiter+String(i+1));
					}

					faceArray.push('top'+delimiter+String(n)+delimiter+String(i));
				
				} else {

					faceArray.push('top'+delimiter+String(n)+delimiter+String(i));

					if (i==va[n].length-1) {
						faceArray.push('top'+delimiter+String(n)+delimiter+String(1));
						faceArray.push('bottom'+delimiter+String(n)+delimiter+String(1));
					} else {
						faceArray.push('top'+delimiter+String(n)+delimiter+String(i+1));
						faceArray.push('bottom'+delimiter+String(n)+delimiter+String(i+1));
					}

					faceArray.push('bottom'+delimiter+String(n)+delimiter+String(i));
				
				}

				this.createFace(faceArray, tId);
				this.setUVsToFace(new Point(0, 0), new Point(1, 0), new Point(1,1), tId);
				surfaceArray.push(tId)

			}



		}
		this.createSurface(surfaceArray,id)



		
	}






	//-----------------------------------------------------------
	//上、下面の形成
	//-----------------------------------------------------------
	private function drawPathToPlane(va:Array,_h:Number,id:String,reverse:Boolean=false,separate:Boolean=false):void {
		
		var delimiter:String='_';
		var faceArray:Array=[];
		var i:int;
		var n:int;
		var tId:String;

		if (roll==1 && reverse) {
			reverse=false
		} else if (roll==1 && !reverse) {
			reverse=true
		}

		if (separate) {
			//すべて、Faceを分割
			if (reverse) {
				for (n=va.length-1;n>=0;n--) {
					faceArray[n]=[]
					for (i=1;i< va[n].length;i++) {

						tId=id+delimiter+String(n)+delimiter+String(i)
						this.createVertex(va[n][i].x, va[n][i].y, _h, tId);
						faceArray[n].push(tId)
					}

				}
			} else {
				for (n=0;n< va.length;n++) {
					faceArray[n]=[]
					for (i=va[n].length-1;i>0 ;i--) {

						tId=id+delimiter+String(n)+delimiter+String(i)
						this.createVertex(va[n][i].x, va[n][i].y, _h, tId);
						faceArray[n].push(tId)
					}
				}
			}
		} else {
			//すべて、一枚のFaceに
			faceArray[0]=[]
			if (reverse) {
				for (n=va.length-1;n>=0;n--) {
					for (i=1;i< va[n].length;i++) {
						tId=id+delimiter+String(n)+delimiter+String(i)
						this.createVertex(va[n][i].x, va[n][i].y, _h, tId);
						faceArray[0].push(tId)
					}

				}
			} else {
				for (n=0;n< va.length;n++) {
					for (i=va[n].length-1;i>0 ;i--) {
						tId=id+delimiter+String(n)+delimiter+String(i)
						this.createVertex(va[n][i].x, va[n][i].y, _h, tId);
						faceArray[0].push(tId)
					}
				}
			}
		}


		var surfaceArray:Array=[]

		for (i=0;i< faceArray.length;i++) {
			this.createFace(faceArray[i], id+delimiter+String(i));
			surfaceArray.push(id+delimiter+String(i))
			if (reverse) {
				this.setUVsToFace(new Point(1, -1), new Point(0, -1), new Point(0,0), id+delimiter+String(i));
			} else {
				this.setUVsToFace(new Point(0, 0), new Point(1, 0), new Point(1,1), id+delimiter+String(i));
			}
		}
		this.createSurface(surfaceArray,id)

	}



	public function get width():Number {
		return _width
	}

	public function get height():Number {
		return _height
	}

	public function get length():Number {
		return _length

	}


}





//3Dテキストを形成するクラス。
class text3D extends pressMesh{

	public function text3D(str:String,_height:Number,fontSet:Dictionary) {

		var n:uint=0;
		var m:int=-1;

		var varArray:Array=[]

		for (var i:uint=0; i<fontSet[str].command.length; i++) {
			if (fontSet[str].command[i]!=GraphicsPathCommand.NO_OP && fontSet[str].command[i]!=100) {
			if (fontSet[str].command[i]==GraphicsPathCommand.MOVE_TO) {
				//moveTo
				m++;
				varArray[m]=[];
			}
			varArray[m].push(new Point(fontSet[str].vertices[n],fontSet[str].vertices[n+1]));
			n+=2;
			} else {
			//End
			}
		}

		var separat:Boolean
		if (str=='i' || str=='j' || str=='?' || str=='!') {
			//ij?!は、特殊 2枚faceを作る
			separat=true
		} else {
			separat=false
		}

		super(varArray,_height,fontSet[str].roll,separat)


	}

}







import alternativ5.engine3d.controllers.CameraController;
import alternativ5.engine3d.core.Camera3D;
import alternativ5.engine3d.core.Object3D;
import alternativ5.engine3d.core.Scene3D;
import alternativ5.engine3d.display.View;
import flash.display.Sprite;
import flash.display.StageAlign;
import flash.display.StageQuality;
import flash.display.StageScaleMode;
import flash.events.Event;


/**
 * BasicTemplate for Alternativa3D
 * Alternativa3Dを扱いやすくするためのテンプレートです
 * @author Yasu
 */
class BasicTemplate extends Sprite{
	/**
	 * シーンインスタンスです。
	 */
	public var scene:Scene3D;
	/**
	 * ビューインスタンスです。
	 */
	public var view:View;
	/**
	 * カメラインスタンスです。
	 */
	public var camera:Camera3D;
	/**
	 * カメラコントローラーです。
	 */
	public var cameraContoller:CameraController;
	
	private var _viewWidth:int;
	private var _viewHeight:int;
	private var _scaleToStage:Boolean;

	/**
	 * 新しい BasicTemplate インスタンスを作成します。
	 * @param	viewWidth
	 * @param	viewHeight
	 * @param	scaleToStage
	 */
	public function BasicTemplate(viewWidth:int=640, viewHeight:int=480, scaleToStage:Boolean = true) {
		_viewWidth = viewWidth;
		_viewHeight = viewHeight;
		_scaleToStage = scaleToStage;
		
		// Creating scene
		scene = new Scene3D();
		scene.splitAnalysis = false; // not analysis for performance
		scene.root = new Object3D();
		
		// Adding camera
		camera = new Camera3D();
		camera.z = -1000;
		scene.root.addChild(camera);
		
		// camera contoller
		cameraContoller = new CameraController(this);
		cameraContoller.camera = camera;
		
		// set view
		view = new View();
		view.camera = camera;
		addChild(view);
		
		// stage
		if (stage) init();
		else addEventListener(Event.ADDED_TO_STAGE, init);
	}
	
	/**
	 * 初期化されたときに実行されるイベントです。
	 * 初期化時に実行したい処理をオーバーライドして記述します。
	 */
	protected function atInit():void {}
	
	/**
	 * 初期化されたときに実行されるイベントです。
	 * 初期化時に実行したい処理を記述します。
	 */
	private var _onInit:Function = function():void { };
	public function get onInit():Function { return _onInit; }
	public function set onInit(value:Function):void {
		_onInit = value;
	}
	
	/**
	 * Event.ENTER_FRAME 時に実行されるレンダリングのイベントです。
	 * レンダリング前に実行したい処理をオーバーライドして記述します。
	 */
	protected function atPreRender():void {}
	
	/**
	 * Event.ENTER_FRAME 時に実行されるレンダリングのイベントです。
	 * レンダリング前に実行したい処理を記述します。
	 */
	private var _onPreRender:Function = function():void{};
	public function get onPreRender():Function { return _onPreRender; }
	public function set onPreRender(value:Function):void {
		_onPreRender = value;
	}
	
	/**
	 * Event.ENTER_FRAME 時に実行されるレンダリングのイベントです。
	 * レンダリング後に実行したい処理をオーバーライドして記述します。
	 */
	protected function atPostRender():void {
	}
	
	/**
	 * Event.ENTER_FRAME 時に実行されるレンダリングのイベントです。
	 * レンダリング後に実行したい処理を記述します。
	 */
	protected var _onPostRender:Function = function():void{};
	public function get onPostRender():Function { return _onPostRender; }
	public function set onPostRender(value:Function):void {
		_onPostRender = value;
	}
	
	/**
	 * レンダリングを開始します。
	 */
	public function startRendering():void {
		addEventListener(Event.ENTER_FRAME, onRenderTick);
	}
	/**
	 * レンダリングを停止します。
	 */
	public function stopRendering():void {
		removeEventListener(Event.ENTER_FRAME, onRenderTick);
	}
	
	/**
	 * シングルレンダリング(レンダリングを一回だけ)を実行します。
	 */
	public function singleRender():void {
		onRenderTick();
	}
	
	/**
	 * @private
	 */
	private function init(e:Event = null):void {
		stage.scaleMode = StageScaleMode.NO_SCALE;
		stage.align = StageAlign.TOP_LEFT;
		stage.quality = StageQuality.HIGH;

		// resize
		stage.addEventListener(Event.RESIZE, onResize);
		onResize(null);
		
		// render
		startRendering();
		
		atInit();
		_onInit();
		
	}
	
	/**
	 * @private
	 */
	private function onRenderTick(e:Event = null):void {
		atPostRender();
		_onPostRender();
		scene.calculate();
		atPreRender();
		_onPreRender();
	}
	
	/**
	 * @private
	 */
	private function onResize(event:Event = null):void {
		if (_scaleToStage) {
			view.width = stage.stageWidth;
			view.height = stage.stageHeight;
		}else {
			view.width = _viewWidth;
			view.height = _viewHeight;
		}
	}
}