hello friends,
I have created the application in which I have to use 3D map Action Script class in mxml file to display a map in form. that is in tab navigator last tab.
My ActionScript 3D map class is(FlyingDirections):-
package src.SBTSCoreObject
{
import src.SBTSCoreObject.JSONDecoder;
import com.google.maps.InfoWindowOptions;
import com.google.maps.LatLng;
import com.google.maps.LatLngBounds;
import com.google.maps.Map3D;
import com.google.maps.MapEvent;
import com.google.maps.MapOptions;
import com.google.maps.MapType;
import com.google.maps.MapUtil;
import com.google.maps.View;
import com.google.maps.controls.NavigationControl;
import com.google.maps.geom.Attitude;
import com.google.maps.interfaces.IPolyline;
import com.google.maps.overlays.Marker;
import com.google.maps.overlays.MarkerOptions;
import com.google.maps.services.Directions;
import com.google.maps.services.DirectionsEvent;
import com.google.maps.services.Route;
import flash.display.Bitmap;
import flash.display.DisplayObject;
import flash.display.DisplayObjectContainer;
import flash.display.Loader;
import flash.display.LoaderInfo;
import flash.display.Sprite;
import flash.events.Event;
import flash.events.IOErrorEvent;
import flash.events.MouseEvent;
import flash.events.TimerEvent;
import flash.filters.DropShadowFilter;
import flash.geom.Point;
import flash.net.URLLoader;
import flash.net.URLRequest;
import flash.net.navigateToURL;
import flash.text.TextField;
import flash.text.TextFieldAutoSize;
import flash.text.TextFormat;
import flash.utils.Timer;
import flash.utils.getTimer;
public class FlyingDirections extends Map3D
{
/**
* Panoramio home page.
*/
private static const PANORAMIO_HOME:String = "http://www.panoramio.com/";
/**
* The icon for the car.
*/
[Embed("assets/car-icon-24px.png")]
private static const Car:Class;
/**
* The Panoramio icon.
*/
[Embed("assets/iw_panoramio.png")]
private static const PanoramioIcon:Class;
/**
* We animate a zoom in to the start the route before the car starts
* to move. This constant sets the time in seconds over which this
* zoom occurs.
*/
private static const LEAD_IN_DURATION:Number = 3;
/**
* Duration of the trip in seconds.
*/
private static const TRIP_DURATION:Number = 40;
/**
* Constants that define the geometry of the Panoramio image markers.
*/
private static const BORDER_T:Number = 3;
private static const BORDER_L:Number = 10;
private static const BORDER_R:Number = 10;
private static const BORDER_B:Number = 3;
private static const GAP_T:Number = 2;
private static const GAP_B:Number = 1;
private static const IMAGE_SCALE:Number = 1;
/**
* Trajectory that the camera follows over time. Each element is an object
* containing properties used to generate parameter values for flyTo(..).
* fraction = 0 corresponds to the start of the trip; fraction = 1
* correspondsto the end of the trip.
*/
private var FLY_TRAJECTORY:Array = [
{ fraction: 0, zoom: 6, attitude: new Attitude(0, 0, 0) },
{ fraction: 0.2, zoom: 8.5, attitude: new Attitude(30, 30, 0) },
{ fraction: 0.5, zoom: 9, attitude: new Attitude(30, 40, 0) },
{ fraction: 1, zoom: 8, attitude: new Attitude(50, 50, 0) },
{ fraction: 1.1, zoom: 8, attitude: new Attitude(130, 50, 0) },
{ fraction: 1.2, zoom: 8, attitude: new Attitude(220, 50, 0) },
];
/**
* Number of panaramio photos for which we load data. We'll select a
* subset of these approximately evenly spaced along the route.
*/
private static const NUM_GEOTAGGED_PHOTOS:int = 50;
/**
* Number of panaramio photos that we actually show.
*/
private static const NUM_SHOWN_PHOTOS:int = 7;
/**
* Scaling between real trip time and animation time.
*/
private static const SCALE_TIME:Number = 0.001;
/**
* getTimer() value at the instant that we start the trip. If this is 0 then
* we have not yet started the car moving.
*/
private var startTimer:int = 0;
/**
* The current route.
*/
private var route:Route;
/**
* The polyline for the route.
*/
private var polyline:IPolyline;
/**
* The car marker.
*/
private var marker:Marker;
/**
* The cumulative duration in seconds over each step in the route.
* cumulativeStepDuration[0] is 0; cumulativeStepDuration[1] adds the
* duration of step 0; cumulativeStepDuration[2] adds the duration
* of step 1; etc.
*/
private var cumulativeStepDuration:/*Number*/Array = [];
/**
* The cumulative distance in metres over each vertex in the route polyline.
* cumulativeVertexDistance[0] is 0; cumulativeVertexDistance[1] adds the
* distance to vertex 1; cumulativeVertexDistance[2] adds the distance to
* vertex 2; etc.
*/
private var cumulativeVertexDistance:Array;
/**
* Array of photos loaded from Panoramio. This array has the same format as
* the 'photos' property within the JSON returned by the Panoramio API
* (see http://www.panoramio.com/api/), with additional properties added to
* individual photo elements to hold the loader structures that fetch
* the actual images.
*/
private var photos:Array = [];
/**
* Array of polyline vertices, where each element is in world coordinates.
* Several computations can be faster if we can use world coordinates
* instead of LatLng coordinates.
*/
private var worldPoly:/*Point*/Array;
/**
* Whether the start button has been pressed.
*/
private var startButtonPressed:Boolean = false;
/**
* Saved event from onDirectionsSuccess call.
*/
private var directionsSuccessEvent:DirectionsEvent = null;
/**
* Start button.
*/
private var startButton:Sprite;
/**
* Alpha value used for the Panoramio image markers.
*/
private var markerAlpha:Number = 0;
/**
* Index of the current driving direction step. Used to update the
* info window content each time we progress to a new step.
*/
private var currentStepIndex:int = -1;
/**
* The fly directions map constructor.
*
* @constructor
*/
public function FlyingDirections() {
key="ABQIAAAA7QUChpcnvnmXxsjC7s1fCxQGj0PqsCtxKvarsoS-iqLdqZSKfxTd7Xf-2rEc_PC9o8IsJde80Wnj4g";
super();
addEventListener(MapEvent.MAP_PREINITIALIZE, onMapPreinitialize);
addEventListener(MapEvent.MAP_READY, onMapReady);
}
/**
* Handles map preintialize. Initializes the map center and zoom level.
*
* @param event The map event.
*/
private function onMapPreinitialize(event:MapEvent):void {
setInitOptions(new MapOptions({
center: new LatLng(-26.1, 135.1),
zoom: 4,
viewMode: View.VIEWMODE_PERSPECTIVE,
mapType:MapType.PHYSICAL_MAP_TYPE
}));
}
/**
* Handles map ready and looks up directions.
*
* @param event The map event.
*/
private function onMapReady(event:MapEvent):void {
enableScrollWheelZoom();
enableContinuousZoom();
addControl(new NavigationControl());
// The driving animation will be updated on every frame.
addEventListener(Event.ENTER_FRAME, enterFrame);
addStartButton();
// We start the directions loading now, so that we're ready to go when
// the user hits the start button.
var directions:Directions = new Directions();
directions.addEventListener(
DirectionsEvent.DIRECTIONS_SUCCESS, onDirectionsSuccess);
directions.addEventListener(
DirectionsEvent.DIRECTIONS_FAILURE, onDirectionsFailure);
directions.load("48 Pirrama Rd, Pyrmont, NSW to Byron Bay, NSW");
}
/**
* Adds a big blue start button.
*/
private function addStartButton():void {
startButton = new Sprite();
startButton.buttonMode = true;
startButton.addEventListener(MouseEvent.CLICK, onStartClick);
startButton.graphics.beginFill(0x1871ce);
startButton.graphics.drawRoundRect(0, 0, 150, 100, 10, 10);
startButton.graphics.endFill();
var startField:TextField = new TextField();
startField.autoSize = TextFieldAutoSize.LEFT;
startField.defaultTextFormat =
new TextFormat("_sans", 20, 0xffffff, true);
startField.text = "Start!";
startButton.addChild(startField);
startField.x = 0.5 * (startButton.width - startField.width);
startField.y = 0.5 * (startButton.height - startField.height);
startButton.filters = [ new DropShadowFilter() ];
var container:DisplayObjectContainer =
getDisplayObject() as DisplayObjectContainer;
container.addChild(startButton);
startButton.x = 0.5 * (container.width - startButton.width);
startButton.y = 0.5 * (container.height - startButton.height);
var panoField:TextField = new TextField();
panoField.autoSize = TextFieldAutoSize.LEFT;
panoField.defaultTextFormat =
new TextFormat("_sans", 11, 0x000000, true);
panoField.text = "Photos provided by Panoramio are under the copyright of their owners.";
container.addChild(panoField);
panoField.x = container.width - panoField.width - 5;
panoField.y = 5;
}
/**
* Handles directions success. Starts flying the route if everything
* is ready.
*
* @param event The directions event.
*/
private function onDirectionsSuccess(event:DirectionsEvent):void {
directionsSuccessEvent = event;
flyRouteIfReady();
}
/**
* Handles click on the start button. Starts flying the route if everything
* is ready.
*/
private function onStartClick(event:MouseEvent):void {
startButton.removeEventListener(MouseEvent.CLICK, onStartClick);
var container:DisplayObjectContainer =
getDisplayObject() as DisplayObjectContainer;
container.removeChild(startButton);
startButtonPressed = true;
flyRouteIfReady();
}
/**
* If we have loaded the directions and the start button has been pressed
* start flying the directions route.
*/
private function flyRouteIfReady():void {
if (!directionsSuccessEvent || !startButtonPressed) {
return;
}
var directions:Directions = directionsSuccessEvent.directions;
// Extract the route.
route = directions.getRoute(0);
// Draws the polyline showing the route.
polyline = directions.createPolyline();
addOverlay(directions.createPolyline());
// Creates a car marker that is moved along the route.
var car:DisplayObject = new Car();
marker = new Marker(route.startGeocode.point, new MarkerOptions({
icon: car,
iconOffset: new Point(-car.width / 2, -car.height)
}));
addOverlay(marker);
transformPolyToWorld();
createCumulativeArrays();
// Load Panoramio data for the region covered by the route.
loadPanoramioData(directions.bounds);
var duration:Number = route.duration;
// Start a timer that will trigger the car moving after the lead in time.
var leadInTimer:Timer = new Timer(LEAD_IN_DURATION * 1000, 1);
leadInTimer.addEventListener(TimerEvent.TIMER, onLeadInDone);
leadInTimer.start();
var flyTime:Number = -LEAD_IN_DURATION;
// Set up the camera flight trajectory.
for each (var flyStep:Object in FLY_TRAJECTORY) {
var time:Number = flyStep.fraction * duration;
var center:LatLng = latLngAt(time);
var scaledTime:Number = time * SCALE_TIME;
var zoom:Number = flyStep.zoom;
var attitude:Attitude = flyStep.attitude;
var elapsed:Number = scaledTime - flyTime;
flyTime = scaledTime;
flyTo(center, zoom, attitude, elapsed);
}
}
/**
* Loads Panoramio data for the route bounds. We load data about more photos
* than we need, then select a subset lying along the route.
* @param bounds Bounds within which to fetch images.
*/
private function loadPanoramioData(bounds:LatLngBounds):void {
var params:Object = {
order: "popularity",
set: "full",
from: "0",
to: NUM_GEOTAGGED_PHOTOS.toString(10),
size: "small",
minx: bounds.getWest(),
miny: bounds.getSouth(),
maxx: bounds.getEast(),
maxy: bounds.getNorth()
};
var loader:URLLoader = new URLLoader();
var request:URLRequest = new URLRequest(
"http://www.panoramio.com/map/get_panoramas.php?" +
paramsToString(params));
loader.addEventListener(Event.COMPLETE, onPanoramioDataLoaded);
loader.addEventListener(IOErrorEvent.IO_ERROR, onPanoramioDataFailed);
loader.load(request);
}
/**
* Transforms the route polyline to world coordinates.
*/
private function transformPolyToWorld():void {
var numVertices:int = polyline.getVertexCount();
worldPoly = new Array(numVertices);
for (var i:int = 0; i < numVertices; ++i) {
var vertex:LatLng = polyline.getVertex(i);
worldPoly[i] = fromLatLngToPoint(vertex, 0);
}
}
/**
* Returns the time at which the route approaches closest to the
* given point.
* @param world Point in world coordinates.
* @return Route time at which the closest approach occurs.
*/
private function getTimeOfClosestApproach(world:Point):Number {
var minDistSqr:Number = Number.MAX_VALUE;
var numVertices:int = worldPoly.length;
var x:Number = world.x;
var y:Number = world.y;
var minVertex:int = 0;
for (var i:int = 0; i < numVertices; ++i) {
var dx:Number = worldPoly[i].x - x;
var dy:Number = worldPoly[i].y - y;
var distSqr:Number = dx * dx + dy * dy;
if (distSqr < minDistSqr) {
minDistSqr = distSqr;
minVertex = i;
}
}
return cumulativeVertexDistance[minVertex];
}
/**
* Returns the array index of the first element that compares greater than
* the given value.
* @param ordered Ordered array of elements.
* @param value Value to use for comparison.
* @return Array index of the first element that compares greater than
* the given value.
*/
private function upperBound(ordered:Array, value:Number,
first:int=0, last:int=-1):int {
if (last < 0) {
last = ordered.length;
}
var count:int = last - first;
var index:int;
while (count > 0) {
var step:int = count >> 1;
index = first + step;
if (value >= ordered[index]) {
first = index + 1;
count -= step - 1;
} else {
count = step;
}
}
return first;
}
/**
* Selects up to a given number of photos approximately evenly spaced along
* the route.
* @param ordered Array of photos, each of which is an object with
* a property 'closestTime'.
* @param number Number of photos to select.
*/
private function selectEvenlySpacedPhotos(ordered:Array, number:int):Array {
var start:Number = cumulativeVertexDistance[0];
var end:Number =
cumulativeVertexDistance[cumulativeVertexDistance.length - 2];
var closestTimes:Array = [];
for each (var photo:Object in ordered) {
closestTimes.push(photo.closestTime);
}
var selectedPhotos:Array = [];
for (var i:int = 0; i < number; ++i) {
var idealTime:Number = start + ((end - start) * (i + 0.5) / number);
var index:int = upperBound(closestTimes, idealTime);
if (index < 1) {
index = 0;
} else if (index >= ordered.length) {
index = ordered.length - 1;
} else {
var errorToPrev:Number =
Math.abs(idealTime - closestTimes[index - 1]);
var errorToNext:Number = Math.abs(idealTime - closestTimes[index]);
if (errorToPrev < errorToNext) {
--index;
}
}
selectedPhotos.push(ordered[index]);
}
return selectedPhotos;
}
/**
* Handles completion of loading the Panoramio index data. Selects from the
* returned photo indices a subset of those that lie along the route and
* initiates load of each of these.
* @param event Load completion event.
*/
private function onPanoramioDataLoaded(event:Event):void {
var loader:URLLoader = event.target as URLLoader;
var decoder:JSONDecoder = new JSONDecoder(loader.data as String);
var allPhotos:Array = decoder.getValue().photos;
for each (var photo:Object in allPhotos) {
var latLng:LatLng = new LatLng(photo.latitude, photo.longitude);
photo.closestTime =
getTimeOfClosestApproach(fromLatLngToPoint(latLng, 0));
}
allPhotos.sortOn("closestTime", Array.NUMERIC);
photos = selectEvenlySpacedPhotos(allPhotos, NUM_SHOWN_PHOTOS);
for each (photo in photos) {
var photoLoader:Loader = new Loader();
// The images aren't on panoramio.com: we can't acquire pixel access
// using "new LoaderContext(true)".
photoLoader.load(
new URLRequest(photo.photo_file_url));
photo.loader = photoLoader;
// Save the loader info: we use this to find the original element when
// the load completes.
photo.loaderInfo = photoLoader.contentLoaderInfo;
photoLoader.contentLoaderInfo.addEventListener(
Event.COMPLETE, onPhotoLoaded);
}
}
/**
* Creates a MouseEvent listener function that will navigate to the given
* URL in a new window.
* @param url URL to which to navigate.
*/
private function createOnClickUrlOpener(url:String):Function {
return function(event:MouseEvent):void {
navigateToURL(new URLRequest(url));
};
}
/**
* Handles completion of loading an individual Panoramio image.
* Adds a custom marker that displays the image. Initially this is made
* invisible so that it can be faded in as needed.
* @param event Load completion event.
*/
private function onPhotoLoaded(event:Event):void {
var loaderInfo:LoaderInfo = event.target as LoaderInfo;
// We need to find which photo element this image corresponds to.
for each (var photo:Object in photos) {
if (loaderInfo == photo.loaderInfo) {
var imageMarker:Sprite = createImageMarker(photo.loader,
photo.owner_name,
photo.owner_url);
var options:MarkerOptions = new MarkerOptions({
icon: imageMarker,
hasShadow: true,
iconAlignment: MarkerOptions.ALIGN_BOTTOM | MarkerOptions.ALIGN_LEFT
});
var latLng:LatLng = new LatLng(photo.latitude, photo.longitude);
var marker:Marker = new Marker(latLng, options);
photo.marker = marker;
addOverlay(marker);
// A hack: we add the actual image after the overlay has been added,
// which creates the shadow, so that the shadow is valid even if we
// don't have security privileges to generate the shadow from the
// image.
marker.foreground.visible = false;
marker.shadow.alpha = 0;
var imageHolder:Sprite = new Sprite();
imageHolder.addChild(photo.loader);
imageHolder.buttonMode = true;
imageHolder.addEventListener(
MouseEvent.CLICK, createOnClickUrlOpener(photo.photo_url));
imageMarker.addChild(imageHolder);
return;
}
}
trace("An image was loaded which could not be found in the photo array.");
}
/**
* Creates a custom marker showing an image.
*/
private function createImageMarker(child:DisplayObject,
ownerName:String,
ownerUrl:String):Sprite {
var content:Sprite = new Sprite();
var panoramioIcon:Bitmap = new PanoramioIcon();
var iconHolder:Sprite = new Sprite();
iconHolder.addChild(panoramioIcon);
iconHolder.buttonMode = true;
iconHolder.addEventListener(MouseEvent.CLICK, onPanoramioIconClick);
panoramioIcon.x = BORDER_L;
panoramioIcon.y = BORDER_T;
content.addChild(iconHolder);
// NOTE: we add the image as a child only after we've added the marker
// to the map. Currently the API requires this if it's to generate the
// shadow for unprivileged content.
// Shrink the image, so that it doesn't obcure too much screen space.
// Ideally, we'd subsample, but we don't have pixel level access.
child.scaleX = IMAGE_SCALE;
child.scaleY = IMAGE_SCALE;
var imageW:Number = child.width;
var imageH:Number = child.height;
child.x = BORDER_L + 30;
child.y = BORDER_T + iconHolder.height + GAP_T;
var authorField:TextField = new TextField();
authorField.autoSize = TextFieldAutoSize.LEFT;
authorField.defaultTextFormat = new TextFormat("_sans", 12);
authorField.text = "author:";
content.addChild(authorField);
authorField.x = BORDER_L;
authorField.y = BORDER_T + iconHolder.height + GAP_T + imageH + GAP_B;
var ownerField:TextField = new TextField();
ownerField.autoSize = TextFieldAutoSize.LEFT;
var textFormat:TextFormat = new TextFormat("_sans", 14, 0x0e5f9a);
ownerField.defaultTextFormat = textFormat;
ownerField.htmlText =
"<a href=\"" + ownerUrl + "\" target=\"_blank\">" +
ownerName + "</a>";
content.addChild(ownerField);
ownerField.x = BORDER_L + authorField.width;
ownerField.y = BORDER_T + iconHolder.height + GAP_T + imageH + GAP_B;
var totalW:Number =
BORDER_L + Math.max(imageW, ownerField.width + authorField.width) +
BORDER_R;
var totalH:Number =
BORDER_T + iconHolder.height +
GAP_T + imageH + GAP_B +
ownerField.height +
BORDER_B;
content.graphics.beginFill(0xffffff);
content.graphics.drawRoundRect(0, 0, totalW, totalH, 10, 10);
content.graphics.endFill();
var marker:Sprite = new Sprite();
marker.addChild(content);
content.x = 30;
content.y = 0;
marker.graphics.lineStyle();
marker.graphics.beginFill(0xff0000);
marker.graphics.drawCircle(0, totalH + 30, 3);
marker.graphics.endFill();
marker.graphics.lineStyle(2, 0xffffff);
marker.graphics.moveTo(30 + 10, totalH - 10);
marker.graphics.lineTo(0, totalH + 30);
return marker;
}
/**
* Handles click on the Panoramio icon.
*/
private function onPanoramioIconClick(event:MouseEvent):void {
navigateToURL(new URLRequest(PANORAMIO_HOME));
}
/**
* Handles failure of a Panoramio image load.
*/
private function onPanoramioDataFailed(event:IOErrorEvent):void {
trace("Load of image failed: " + event);
}
/**
* Returns a string containing cgi query parameters.
* @param Associative array mapping query parameter key to value.
* @return String containing cgi query parameters.
*/
private static function paramsToString(params:Object):String {
var result:String = "";
var separator:String = "";
for (var key:String in params) {
result += separator +
encodeURIComponent(key) + "=" + encodeURIComponent(params[key]);
separator = "&";
}
return result;
}
/**
* Called once the lead-in flight is done. Starts the car driving along
* the route and starts a timer to begin fade in of the Panoramio
* images in 1.5 seconds.
*/
private function onLeadInDone(event:Event):void {
// Set startTimer non-zero so that the car starts to move.
startTimer = getTimer();
// Start a timer that will fade in the Panoramio images.
var fadeInTimer:Timer = new Timer(1500, 1);
fadeInTimer.addEventListener(TimerEvent.TIMER, onFadeInTimer);
fadeInTimer.start();
}
/**
* Handles the fade in timer's TIMER event. Sets markerAlpha above zero
* which causes the frame enter handler to fade in the markers.
*/
private function onFadeInTimer(event:Event):void {
markerAlpha = 0.01;
}
/**
* The end time of the flight.
*/
private function get endTime():Number {
if (!cumulativeStepDuration || cumulativeStepDuration.length == 0) {
return startTimer;
}
return startTimer +
cumulativeStepDuration[cumulativeStepDuration.length - 1];
}
/**
* Creates the cumulative arrays, cumulativeStepDuration and
* cumulativeVertexDistance.
*/
private function createCumulativeArrays():void {
cumulativeStepDuration = new Array(route.numSteps + 1);
cumulativeVertexDistance = new Array(polyline.getVertexCount() + 1);
var polylineTotal:Number = 0;
var total:Number = 0;
var numVertices:int = polyline.getVertexCount();
for (var stepIndex:int = 0; stepIndex < route.numSteps; ++stepIndex) {
cumulativeStepDuration[stepIndex] = total;
total += route.getStep(stepIndex).duration;
var startVertex:int = stepIndex >= 0 ?
route.getStep(stepIndex).polylineIndex : 0;
var endVertex:int = stepIndex < (route.numSteps - 1) ?
route.getStep(stepIndex + 1).polylineIndex : numVertices;
var duration:Number = route.getStep(stepIndex).duration;
var stepVertices:int = endVertex - startVertex;
var latLng:LatLng = polyline.getVertex(startVertex);
for (var vertex:int = startVertex; vertex < endVertex; ++vertex) {
cumulativeVertexDistance[vertex] = polylineTotal;
if (vertex < numVertices - 1) {
var nextLatLng:LatLng = polyline.getVertex(vertex + 1);
polylineTotal += nextLatLng.distanceFrom(latLng);
}
latLng = nextLatLng;
}
}
cumulativeStepDuration[stepIndex] = total;
}
/**
* Opens the info window above the car icon that details the given
* step of the driving directions.
* @param stepIndex Index of the current step.
*/
private function openInfoForStep(stepIndex:int):void {
// Sets the content of the info window.
var content:String;
if (stepIndex >= route.numSteps) {
content = "<b>" + route.endGeocode.address + "</b>" +
"<br /><br />" +
route.summaryHtml;
} else {
content = "<b>" + stepIndex + ".</b> " +
route.getStep(stepIndex).descriptionHtml;
}
marker.openInfoWindow(new InfoWindowOptions({ contentHTML: content }));
}
/**
* Displays the driving directions step appropriate for the given time.
* Opens the info window showing the step instructions each time we
* progress to a new step.
* @param time Time for which to display the step.
*/
private function displayStepAt(time:Number):void {
var stepIndex:int = upperBound(cumulativeStepDuration, time) - 1;
var minStepIndex:int = 0;
var maxStepIndex:int = route.numSteps - 1;
if (stepIndex >= 0 &&
stepIndex <= maxStepIndex &&
currentStepIndex != stepIndex) {
openInfoForStep(stepIndex);
currentStepIndex = stepIndex;
}
}
/**
* Returns the LatLng at which the car should be positioned at the given
* time.
* @param time Time for which LatLng should be found.
* @return LatLng.
*/
private function latLngAt(time:Number):LatLng {
var stepIndex:int = upperBound(cumulativeStepDuration, time) - 1;
var minStepIndex:int = 0;
var maxStepIndex:int = route.numSteps - 1;
if (stepIndex < minStepIndex) {
return route.startGeocode.point;
} else if (stepIndex > maxStepIndex) {
return route.endGeocode.point;
}
var stepStart:Number = cumulativeStepDuration[stepIndex];
var stepEnd:Number = cumulativeStepDuration[stepIndex + 1];
var stepFraction:Number = (time - stepStart) / (stepEnd - stepStart);
var startVertex:int = route.getStep(stepIndex).polylineIndex;
var endVertex:int = (stepIndex + 1) < route.numSteps ?
route.getStep(stepIndex + 1).polylineIndex :
polyline.getVertexCount();
var stepVertices:int = endVertex - startVertex;
var stepLeng