/* * PROJECT: FLARManager * http://transmote.com/flar * Copyright 2009, Eric Socolofsky * -------------------------------------------------------------------------------- * This work complements FLARToolkit, developed by Saqoosha as part of the Libspark project. * http://www.libspark.org/wiki/saqoosha/FLARToolKit * FLARToolkit is Copyright (C)2008 Saqoosha, * and is ported from NYARToolkit, which is ported from ARToolkit. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this framework; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * For further information please contact: * * http://transmote.com/flar * */ package com.transmote.flar.utils { import __AS3__.vec.Vector; import com.transmote.flar.flarManagerInternal; import com.transmote.flar.marker.FLARMarker; import com.transmote.flar.marker.FLARMarkerEvent; import com.transmote.flar.source.IFLARSource; import com.transmote.flar.utils.geom.FLARGeomUtils; import flash.display.BitmapData; import flash.display.Sprite; import flash.events.Event; import flash.events.KeyboardEvent; import flash.events.MouseEvent; import flash.geom.Rectangle; import jp.nyatla.nyartoolkit.as3.core.transmat.NyARTransMatResult; import org.libspark.flartoolkit.core.param.FLARParam; use namespace flarManagerInternal; /** * FLARProxy provides a way to test FLARToolkit applications with a mouse and keyboard. * to use, add useProxy='true' to the flarSourceSettings node in your configuration XML file. * since FLARProxy responds to mouse interaction, it must be added to the display list * in order to capture mouse interaction and dispatch FLARMarkerEvents. * * clicking the mouse sends a MARKER_ADDED event, * dragging the mouse sends a MARKER_UPDATED event, * and releasing the mouse sends a MARKER_REMOVED event. * these events can be handled exactly as if they were dispatched by FLARManager. * * press keys 0-9 to specify a patternId. * FLARProxy only supports patternIds 0 through 9. */ public class FLARProxy extends Sprite implements IFLARSource { private static const MAX_NUMBER_MARKERS:uint = 10; private static const DEFAULT_MARKER_Z:Number = 400; private var mouseArea:Sprite; private var mouseIsDown:Boolean; private var activePatternId:uint = 0; private var activeMarkers:Vector.; private var bLoadingCameraParams:Boolean = false; private var bActivatePending:Boolean = false; private var _cameraParams:FLARParam; /** * constructor. * @param displayWidth width of active area. * @param displayHeight height of active area. */ public function FLARProxy (displayWidth:Number, displayHeight:Number) { this.initMouseArea(displayWidth, displayHeight); this.activeMarkers = new Vector.(MAX_NUMBER_MARKERS, true); } /** * begin marker simulation. */ public function activate () :void { if (this.stage) { this.mouseArea.addEventListener(MouseEvent.MOUSE_DOWN, this.onMouseDown, false, 0, true); this.stage.addEventListener(KeyboardEvent.KEY_DOWN, this.onKeyDown, false, 0, true); this.mouseArea.addEventListener(MouseEvent.MOUSE_UP, this.onMouseUp, false, 0, true); } else { this.addEventListener(Event.ADDED_TO_STAGE, this.onAddedToStage, false, 0, true); } } /** * stop marker simulation. */ public function deactivate () :void { if (this.stage) { this.mouseArea.removeEventListener(MouseEvent.MOUSE_DOWN, this.onMouseDown); this.mouseArea.removeEventListener(MouseEvent.MOUSE_UP, this.onMouseUp); this.stage.removeEventListener(KeyboardEvent.KEY_DOWN, this.onKeyDown); } if (this.activeMarkers) { var i:int = this.activeMarkers.length; while (i--) { // remove all active markers this.activeMarkers[i].dispose(); } this.activeMarkers = null; } } /** * FLARProxy updates on mouse interaction; * method is here only for compliance with IFLARSource. */ public function update () :void {} /** * FLARProxy has no BitmapData source; * method is here only for compliance with IFLARSource. */ public function get source () :BitmapData { return null; } public function set source (val:BitmapData) :void { // } /** * size of BitmapData source used for analysis. */ public function get sourceSize () :Rectangle { return new Rectangle(0, 0, this.mouseArea.width, this.mouseArea.height); } /** * ratio of area of reported results to display size. * use to scale (multiply) results of FLARToolkit analysis to correctly fit display area. * FLARProxy always uses the whole screen size, so this method returns 1.0. */ public function get resultsToDisplayRatio () :Number { return 1.0; } /** * FLARProxy cannot be mirrored; * method is here only for compliance with IFLARSource. */ public function get mirrored () :Boolean { return false; } public function set mirrored (val:Boolean) :void {} /** * returns true if initialization is complete. * FLARProxy is inited automatically in constructor. */ public function get inited () :Boolean { return true; } /** * halts all processes and frees this instance for garbage collection. */ public function dispose () :void { this.deactivate(); this._cameraParams = null; } private function initMouseArea (displayWidth:Number, displayHeight:Number) :void { this.mouseArea = new Sprite(); this.mouseArea.graphics.beginFill(0, 0); this.mouseArea.graphics.drawRect(0, 0, displayWidth, displayHeight); this.mouseArea.graphics.endFill(); this.addChild(this.mouseArea); } private function onAddedToStage (evt:Event) :void { this.removeEventListener(Event.ADDED_TO_STAGE, this.onAddedToStage); this.addEventListener(Event.REMOVED_FROM_STAGE, this.onRemovedFromStage, false, 0, true); this.activate(); } private function onRemovedFromStage (evt:Event) :void { this.removeEventListener(Event.REMOVED_FROM_STAGE, this.onRemovedFromStage); this.deactivate(); } private function addMarker (patternId:uint, x:Number, y:Number) :void { var proxyResult:ProxyResult = this.createProxyResult(patternId, x, y); var proxyMarker:FLARMarker = new FLARMarker(this.createProxyMatrix(x, y), this, null, proxyResult.patternId, proxyResult.direction, proxyResult.square, proxyResult.confidence); proxyMarker.setSessionId(); this.activeMarkers[patternId] = proxyMarker; this.dispatchEvent(new FLARMarkerEvent(FLARMarkerEvent.MARKER_ADDED, proxyMarker)); } private function updateMarker (patternId:uint, x:Number, y:Number) :void { var proxyMarker:FLARMarker = this.activeMarkers[patternId]; if (!proxyMarker) { return; } var proxyResult:ProxyResult = this.createProxyResult(patternId, x, y); var updatedMarker:FLARMarker = new FLARMarker(this.createProxyMatrix(x, y), this, null, proxyResult.patternId, proxyResult.direction, proxyResult.square, proxyResult.confidence); proxyMarker.copy(updatedMarker); this.dispatchEvent(new FLARMarkerEvent(FLARMarkerEvent.MARKER_UPDATED, proxyMarker)); } private function removeMarker (patternId:uint, x:Number, y:Number) :void { var proxyMarker:FLARMarker = this.activeMarkers[patternId]; if (!proxyMarker) { return; } var proxyResult:ProxyResult = this.createProxyResult(patternId, x, y); var removedMarker:FLARMarker = new FLARMarker(this.createProxyMatrix(x, y), this, null, proxyResult.patternId, proxyResult.direction, proxyResult.square, proxyResult.confidence); proxyMarker.copy(removedMarker); this.activeMarkers[patternId] = null; this.dispatchEvent(new FLARMarkerEvent(FLARMarkerEvent.MARKER_REMOVED, proxyMarker)); } private function onMouseDown (evt:MouseEvent) :void { this.mouseArea.addEventListener(MouseEvent.MOUSE_MOVE, this.onMouseMove); this.mouseIsDown = true; this.addMarker(this.activePatternId, evt.localX, evt.localY); } private function onMouseMove (evt:MouseEvent) :void { this.updateMarker(this.activePatternId, evt.localX, evt.localY); } private function onMouseUp (evt:MouseEvent) :void { this.mouseArea.removeEventListener(MouseEvent.MOUSE_MOVE, this.onMouseMove); this.mouseIsDown = false; this.removeMarker(this.activePatternId, evt.localX, evt.localY); } private function onKeyDown (evt:KeyboardEvent) :void { var newPatternId:uint = evt.keyCode - 48; if (newPatternId == this.activePatternId || newPatternId < 0 || newPatternId > 9) { return; } trace("FLARProxy active patternId: "+ newPatternId); var lastPatternId:uint = this.activePatternId; this.activePatternId = newPatternId; if (this.mouseIsDown) { this.removeMarker(lastPatternId, this.mouseArea.mouseX, this.mouseArea.mouseY); this.addMarker(this.activePatternId, this.mouseArea.mouseX, this.mouseArea.mouseY); } } private function createProxyResult (patternId:int, x:Number, y:Number) :ProxyResult { return new ProxyResult(patternId, 0, 1.0, x, y); // return new FLARMultiMarkerDetectorResult(patternId, 0, 1.0, this.createProxySquare(x, y)); } /* private function createProxyResult (patternId:int, x:Number, y:Number) :FLARMultiMarkerDetectorResult { return new FLARMultiMarkerDetectorResult(patternId, 0, 1.0, this.createProxySquare(x, y)); } private function createProxySquare (x:Number, y:Number) :FLARSquare { var sqvertex:Array = new Array( new FLARDoublePoint2d(x-this.markerCornerVal, y-this.markerCornerVal), new FLARDoublePoint2d(x+this.markerCornerVal, y-this.markerCornerVal), new FLARDoublePoint2d(x+this.markerCornerVal, y+this.markerCornerVal), new FLARDoublePoint2d(x-this.markerCornerVal, y+this.markerCornerVal)); var flarSquare:FLARSquare = new FLARSquare(); flarSquare.sqvertex = sqvertex; return flarSquare; } */ private function createProxyMatrix (x:Number, y:Number) :NyARTransMatResult { var matrix:NyARTransMatResult = FLARGeomUtils.createFLARIdentityTransMat(); matrix.m03 = x - 0.5*this.mouseArea.width; matrix.m13 = y - 0.5*this.mouseArea.height; matrix.m23 = DEFAULT_MARKER_Z; return matrix; } } } import jp.nyatla.nyartoolkit.as3.core.squaredetect.NyARSquare; import jp.nyatla.nyartoolkit.as3.core.types.NyARDoublePoint2d; import com.transmote.flar.pattern.FLARPattern; import __AS3__.vec.Vector; class ProxyResult { private static const MARKER_CORNER_VAL:Number = 0.5 * FLARPattern.DEFAULT_UNSCALED_MARKER_WIDTH; public var patternId:int; public var direction:int; public var confidence:Number; public var square:NyARSquare; public function ProxyResult (patternId:int, direction:int, confidence:Number, x:Number, y:Number) { this.patternId = patternId; this.direction = direction; this.confidence = confidence; this.square = this.createProxySquare(x, y); } private function createProxySquare (x:Number, y:Number) :NyARSquare { var sqvertex:Vector. = new Vector.([ new NyARDoublePoint2d(x-MARKER_CORNER_VAL, y-MARKER_CORNER_VAL), new NyARDoublePoint2d(x+MARKER_CORNER_VAL, y-MARKER_CORNER_VAL), new NyARDoublePoint2d(x+MARKER_CORNER_VAL, y+MARKER_CORNER_VAL), new NyARDoublePoint2d(x-MARKER_CORNER_VAL, y+MARKER_CORNER_VAL)]); var flarSquare:NyARSquare = new NyARSquare(); flarSquare.sqvertex = sqvertex; return flarSquare; } }