intersectRay(Mouse位置を取得する) [Alternativa3D 7.6 TIPS]
7.5.1で実装された、Object3D.intersectRayを使うことで、Mouse位置のObject3D情報(RayIntersectionData)を取得する事が出来るようになりました。
これを使用して、Object3D上にMouseでお絵描き(絵といってもモノクロラインのみ)するサンプルです。
なお、7.5.0では使用できませんので、ローカルでテストする際は、最新のSWCを入手してください。
Object外(黒Back)部分をDragで、白いBOXを回転。Object上で、絵がかけます。
(角の処理がちょっと甘いのはご愛嬌)
2011/01/12 7.5から7.6仕様に変更
/**
* Copyright narutohyper ( http://wonderfl.net/user/narutohyper )
* MIT License ( http://www.opensource.org/licenses/mit-license.php )
* Downloaded from: http://wonderfl.net/c/hVFp
*/
// forked from narutohyper's Alternativa3D 7.5 Template
package
{
import alternativ7.engine3d.controllers.SimpleObjectController;
import alternativ7.engine3d.core.MouseEvent3D;
import alternativ7.engine3d.core.Object3D;
import alternativ7.engine3d.core.RayIntersectionData;
import alternativ7.engine3d.materials.TextureMaterial;
import alternativ7.engine3d.primitives.Box;
import alternativ7.engine3d.primitives.Plane;
import flash.display.BitmapData;
import flash.display.Shape;
import flash.display.Sprite;
import flash.events.Event;
import flash.filters.*;
import flash.geom.Point;
/**
* Alternativa3D 7.5.1
*
* interactive
* Object3DにMouseでお絵かき
*
* @author narutohyper
*/
[SWF(backgroundColor="#000000", frameRate="100", width="800", height="600")]
public class Main extends Sprite
{
private var box:Box;
private var objectController:SimpleObjectController;
public function Main():void {
if (stage) init();
else addEventListener(Event.ADDED_TO_STAGE, init);
}
private function init(e:Event=null):void {
removeEventListener(Event.ADDED_TO_STAGE, init);
//AlternativaTemplate作成
var scene:AlternativaTemplate = new AlternativaTemplate(this);
//
var materials:Vector.<TextureMaterial> = new Vector.<TextureMaterial>(6);
for (var i:int = 0; i < 7; i++)
{
var color:uint = (i == 6) ? 0xCCFFCC:0xFFFFFF;
var bmd:BitmapData = new BitmapData(500, 500, false, color);
materials[i] = new TextureMaterial(bmd);
}
box = new Box(500,500,500,1,1,1,false,false,materials[0],materials[1],materials[2],materials[3],materials[4],materials[5]);
scene.container.addChild(box);
box.addEventListener(MouseEvent3D.MOUSE_DOWN, onMouseDown3D);
scene.cameraController.setObjectPosXYZ(0, -1000,0);
scene.cameraController.lookAtXYZ(0, 0, 0);
var plane:Plane = new Plane(2000, 2000);
plane.setMaterialToAllFaces(materials[6]);
scene.container.addChild(plane);
plane.addEventListener(MouseEvent3D.MOUSE_DOWN, onMouseDown3D);
plane.z = -600;
//ObjectControllerの作成
//MouseDragで、Objectを回転させる為のController
objectController = new SimpleObjectController(stage, box, 100);
objectController.mouseSensitivity = 0.5;
objectController.unbindAll();
var RADIAN:Number = Math.PI/180;
scene.onPreRender = function():void {
objectController.update();
}
//描画開始
scene.startRendering();
}
private var counter:uint =0;
private function onMouseDown3D(e:MouseEvent3D):void
{
objectController.mouseSensitivity = 0;
var object:Object3D = e.target as Object3D;
object.removeEventListener(MouseEvent3D.MOUSE_DOWN, onMouseDown3D);
object.addEventListener(MouseEvent3D.MOUSE_MOVE, onMouseMove3D);
object.addEventListener(MouseEvent3D.MOUSE_OUT,onMouseOut3D);
object.addEventListener(MouseEvent3D.MOUSE_UP, onMouseUp);
}
private function onMouseUp(e:MouseEvent3D):void
{
objectController.mouseSensitivity = 0.5;
var object:Object3D = e.target as Object3D;
object.addEventListener(MouseEvent3D.MOUSE_DOWN, onMouseDown3D);
object.removeEventListener(MouseEvent3D.MOUSE_MOVE, onMouseMove3D);
object.removeEventListener(MouseEvent3D.MOUSE_OUT,onMouseOut3D);
object.removeEventListener(MouseEvent3D.MOUSE_UP, onMouseUp);
oldX=-1;
oldY=-1;
}
private var oldX:Number=-1;
private var oldY:Number=-1;
private var oldTexture:BitmapData;
private function onMouseMove3D(e:MouseEvent3D):void
{
var object:Object3D = e.target as Object3D;
var data:RayIntersectionData = object.intersectRay(e.localOrigin, e.localDirection);
if (data != null) {
var texture:BitmapData = (data.face.material as TextureMaterial).texture;
if (oldTexture==texture) {
var uv:Point = data.face.getUV(data.point);
var spot:Shape = new Shape();
spot.graphics.lineStyle(10, 0x000000, 1);
if (oldX>=0 && oldY>=0) {
spot.graphics.moveTo(oldX, oldY);
} else {
spot.graphics.moveTo(uv.x * texture.width, uv.y * texture.height);
}
spot.graphics.lineTo(uv.x * texture.width, uv.y * texture.height);
texture.draw(spot);
oldX = uv.x * texture.width;
oldY = uv.y * texture.height;
} else {
oldTexture = texture;
oldX=-1;
oldY=-1;
}
}
}
private function onMouseOut3D(e:MouseEvent3D):void
{
oldTexture = null;
oldX=-1;
oldY=-1;
}
}
}
/**
* BasicTemplate for Alternativa3D 7.5
* Alternativa3D 7.5を扱いやすくするためのテンプレートです
* @author narutohyper & clockmaker
*
*/
import alternativ7.engine3d.containers.BSPContainer;
import alternativ7.engine3d.containers.ConflictContainer;
import alternativ7.engine3d.containers.DistanceSortContainer;
import alternativ7.engine3d.containers.KDContainer;
import alternativ7.engine3d.containers.LODContainer;
import alternativ7.engine3d.controllers.SimpleObjectController;
import alternativ7.engine3d.core.Camera3D;
import alternativ7.engine3d.core.Object3DContainer;
import alternativ7.engine3d.core.View;
import flash.display.DisplayObject;
import flash.display.DisplayObjectContainer;
import flash.display.Sprite;
import flash.display.StageAlign;
import flash.display.StageQuality;
import flash.display.StageScaleMode;
import flash.events.Event;
class AlternativaTemplate extends Sprite
{
/**
* 子オブジェクトを最適な方法でソートするコンテナ
* (ConflictContainer)
*/
public static const CONFLICT:String = 'conflict';
/**
* 子オブジェクトをBSP(バイナリ空間分割法)によってソートするコンテナ
* (BSPContainer)
*/
public static const BSP:String = 'bsp';
/**
* 子オブジェクトをカメラからのZ値でソートするコンテナ
* (DistanceSortContainer)
*/
public static const ZSORT:String = 'zsort';
/**
* KDツリー(http://ja.wikipedia.org/wiki/Kd%E6%9C%A8)によってソートするコンテナ
* (KDContainer)
*/
public static const KD:String = 'kd';
/**
* detalizationと子オブジェクトの距離でソートするコンテナ(詳細は調査中)
* (LODContainer)
*/
public static const LOD:String = 'lod';
/**
* 3dオブジェクト格納するコンテナインスタンス。
*/
public var container:Object3DContainer;
/**
* ビューインスタンスです。
*/
public var view:View;
/**
* カメラインスタンスです。
*/
public var camera:Camera3D;
/**
* カメラコントローラーです。
*/
public var cameraController:SimpleObjectController;
private var _mc:DisplayObjectContainer;
private var _viewWidth:int;
private var _viewHeight:int;
private var _scaleToStage:Boolean;
private var _containerType:String;
/**
* 新しい Alternativa3DTemplate インスタンスを作成します。
* @param mc
* @param containerType
* @param viewWidth
* @param viewHeight
* @param scaleToStage
*/
public function AlternativaTemplate(mc:DisplayObjectContainer,containerType:String=CONFLICT,viewWidth:int=640, viewHeight:int=480, scaleToStage:Boolean = true)
{
_mc = mc;
_mc.addChild(this);
_containerType = containerType;
_viewWidth = viewWidth;
_viewHeight = viewHeight;
_scaleToStage = scaleToStage;
if (stage) init();
else addEventListener(Event.ADDED_TO_STAGE, init);
}
/**
* 初期化されたときに実行されるイベントです。
* 初期化時に実行したい処理をオーバーライドして記述します。
*/
protected function atInit():void {}
/**
* 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 時に実行されるレンダリングのイベントです。
* レンダリング後に実行したい処理を記述します。
*/
private 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
{
removeEventListener(Event.ADDED_TO_STAGE, init);
// entry point
stage.scaleMode = StageScaleMode.NO_SCALE;
stage.align = StageAlign.TOP_LEFT;
stage.quality = StageQuality.HIGH;
//Root objectの作成
if (_containerType == CONFLICT) {
container = new ConflictContainer();
} else if (_containerType == BSP) {
container = new BSPContainer();
} else if (_containerType == ZSORT) {
container = new DistanceSortContainer();
} else if (_containerType == KD) {
container = new KDContainer();
} else if (_containerType == LOD) {
container = new LODContainer();
}
//Viewの作成
view = new View(stage.stageWidth, stage.stageHeight);
_mc.addChild(view);
//cameraの作成
camera = new Camera3D();
camera.view = view;
camera.x = 0;
camera.y = -500;
camera.z = 0;
container.addChild(camera);
// Camera controller
cameraController = new SimpleObjectController(stage, camera, 10);
cameraController.mouseSensitivity = 0;
cameraController.unbindAll();
cameraController.lookAtXYZ(0, 0, 0);
onResize();
stage.addEventListener(Event.RESIZE, onResize);
atInit();
}
/**
* @private
*/
private function onResize(e:Event = null):void
{
if (_scaleToStage)
{
view.width = stage.stageWidth;
view.height = stage.stageHeight;
}
else
{
view.width = _viewWidth;
view.height = _viewHeight;
}
}
/**
* @private
*/
private function onRenderTick(e:Event = null):void
{
atPreRender();
_onPreRender();
cameraController.update();
camera.render();
atPostRender();
_onPostRender();
}
}