Source: util/editor/ShapeEditor.js

/*
 * Copyright 2003-2006, 2009, 2017, United States Government, as represented by the Administrator of the
 * National Aeronautics and Space Administration. All rights reserved.
 *
 * The NASAWorldWind/WebWorldWind platform is licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
/**
 * @exports ShapeEditor
 */
define([
        '../../shapes/Annotation',
        '../../shapes/AnnotationAttributes',
        '../../error/ArgumentError',
        '../Color',
        '../Font',
        '../Insets',
        '../../geom/Location',
        '../Logger',
        '../../shapes/Placemark',
        '../../shapes/PlacemarkAttributes',
        './PlacemarkEditorFragment',
        '../../geom/Position',
        '../../layer/RenderableLayer',
        '../../shapes/ShapeAttributes',
        './ShapeEditorConstants',
        './SurfaceEllipseEditorFragment',
        './SurfaceCircleEditorFragment',
        '../../shapes/SurfacePolygon',
        '../../shapes/SurfacePolyline',
        './SurfacePolygonEditorFragment',
        './SurfacePolylineEditorFragment',
        './SurfaceRectangleEditorFragment',
        './SurfaceSectorEditorFragment',
        '../../geom/Vec2',
        '../../geom/Vec3'
    ],
    function (Annotation,
              AnnotationAttributes,
              ArgumentError,
              Color,
              Font,
              Insets,
              Location,
              Logger,
              Placemark,
              PlacemarkAttributes,
              PlacemarkEditorFragment,
              Position,
              RenderableLayer,
              ShapeAttributes,
              ShapeEditorConstants,
              SurfaceEllipseEditorFragment,
              SurfaceCircleEditorFragment,
              SurfacePolygon,
              SurfacePolyline,
              SurfacePolygonEditorFragment,
              SurfacePolylineEditorFragment,
              SurfaceRectangleEditorFragment,
              SurfaceSectorEditorFragment,
              Vec2,
              Vec3) {
        "use strict";

        /**
         * Constructs a new shape editor attached to the specified World Window.
         * @alias ShapeEditor
         * @classdesc Provides a controller for editing shapes. Depending on the type of shape, the following actions
         * are available:
         * <ul>
         *     <li>Edit the location and size of its vertexes using control points;</li>
         *     <li>Rotate the shape using a handle;</li>
         *     <li>Drag the shape on the surface of the globe.</li>
         * </ul>
         * <p>
         * To start editing a shape, pass it to the {@link ShapeEditor#edit} method. To end the edition, call the
         * {@link ShapeEditor#stop} method.
         * <p>
         * Dragging the body of the shape moves the whole shape. Dragging a control point performs the action associated
         * with that control point. The editor provides vertex insertion and removal for SurfacePolygon and
         * SurfacePolyline. Shift-clicking when the cursor is over the shape inserts a control point near the position
         * of the cursor. Ctrl-clicking when the cursor is over a control point removes that particular control point.
         * <p>
         * This editor currently supports all surface shapes except SurfaceImage.
         * @param {WorldWindow} worldWindow The World Window to associate this shape editor controller with.
         * @throws {ArgumentError} If the specified World Window is <code>null</code> or <code>undefined</code>.
         * @constructor
         */
        var ShapeEditor = function (worldWindow) {
            if (!worldWindow) {
                throw new ArgumentError(Logger.logMessage(Logger.LEVEL_SEVERE, "ShapeEditor", "constructor",
                    "missingWorldWindow"));
            }

            // Documented in defineProperties below.
            this._worldWindow = worldWindow;

            // Documented in defineProperties below.
            this._shape = null;

            // Internal use only.
            // Flags indicating whether the specific action is allowed or not.
            this._allowMove = true;
            this._allowReshape = true;
            this._allowRotate = true;
            this._allowManageControlPoint = true;

            // Internal use only
            // List of highlighted control points - on mouse over
            this._highlightedItems = [];

            // Documented in defineProperties below (blue dot image).
            this._moveControlPointAttributes = new PlacemarkAttributes(null);
            this._moveControlPointAttributes.imageSource = "";
            this._moveControlPointAttributes.imageScale = 0.15;

            // Documented in defineProperties below (grey dot image).
            this._shadowControlPointAttributes = new PlacemarkAttributes(null);
            this._shadowControlPointAttributes.imageSource = "";
            this._shadowControlPointAttributes.imageScale = 0.15;

            // Documented in defineProperties below (yellow dot image).
            this._resizeControlPointAttributes = new PlacemarkAttributes(null);
            this._resizeControlPointAttributes.imageSource = "";
            this._resizeControlPointAttributes.imageScale = 0.15;

            // Documented in defineProperties below (green dot image).
            this._rotateControlPointAttributes = new PlacemarkAttributes(null);
            this._rotateControlPointAttributes.imageColor = WorldWind.Color.GREEN;
            this._rotateControlPointAttributes.imageSource = "";
            this._rotateControlPointAttributes.imageScale = 0.15;

            // Documented in defineProperties below.
            this._annotationAttributes = new AnnotationAttributes(null);
            this._annotationAttributes.altitudeMode = WorldWind.CLAMP_TO_GROUND;
            this._annotationAttributes.cornerRadius = 5;
            this._annotationAttributes.backgroundColor = new Color(0.67, 0.67, 0.67, 0.8);
            this._annotationAttributes.leaderGapHeight = 0;
            this._annotationAttributes.drawLeader = false;
            this._annotationAttributes.scale = 1;
            this._annotationAttributes.textAttributes.color = Color.BLACK;
            this._annotationAttributes.textAttributes.font = new Font(10);
            this._annotationAttributes.insets = new Insets(5, 5, 5, 5);

            // Internal use only.
            // The annotation that displays hints during the actions on the shape.
            this.annotation = new WorldWind.Annotation(new WorldWind.Position(0, 0, 0), this._annotationAttributes);

            //Internal use only. Intentionally not documented.
            this.editorFragments = [
                new PlacemarkEditorFragment(),
                new SurfaceCircleEditorFragment(),
                new SurfaceEllipseEditorFragment(),
                new SurfacePolygonEditorFragment(),
                new SurfacePolylineEditorFragment(),
                new SurfaceRectangleEditorFragment(),
                new SurfaceSectorEditorFragment()
            ];

            // Internal use only.
            // The layer that holds the control points created by the editor fragment.
            this.controlPointsLayer = new RenderableLayer("Shape Editor Control Points");

            // Internal use only.
            // The layer that holds the shadow control points created by the editor fragment.
            this.shadowControlPointsLayer = new RenderableLayer("Shape Editor Shadow Control Points");

            // Internal use only.
            // The layers that holds the additional accessories created by the editor fragment.
            this.accessoriesLayer = new RenderableLayer("Shape Editor Accessories");
            this.accessoriesLayer.pickEnabled = false;

            // Internal use only.
            // The layer that holds the above-mentioned annotation.
            this.annotationLayer = new RenderableLayer("Shape Editor Annotation");
            this.annotationLayer.pickEnabled = false;
            this.annotationLayer.enabled = false;
            this.annotationLayer.addRenderable(this.annotation);

            // Internal use only.
            // The layer that holds the shadow of the shape during the actions.
            this.shadowShapeLayer = new RenderableLayer("Shape Editor Shadow Shape");
            this.shadowShapeLayer.pickEnabled = false;

            // Internal use only.
            // The layer that holds the initial shapes created at shape creation by the create method.
            this.creatorShapeLayer = new RenderableLayer("Shape Creator Initial Shape");
            this.creatorShapeLayer.pickEnabled = false;

            // Internal use only.
            // The editor fragment selected for the shape being edited or null.
            this.activeEditorFragment = null;

            // Internal use only.
            // The type of action being conducted or null.
            this.actionType = null;

            // Internal use only.
            // The control point that triggered the current action or null.
            this.actionControlPoint = null;

            // Internal use only.
            // The lat/lon/alt position that is currently involved with the action or null.
            this.actionControlPosition = null;

            // Internal use only.
            // Flag indicating whether the action should trigger the secondary behavior in the editor fragment.
            this.actionSecondaryBehavior = false;

            // Internal use only.
            // The current client X position for the action.
            this.actionCurrentX = null;

            // Internal use only.
            // The current client Y position for the action.
            this.actionCurrentY = null;

            // Internal use only.
            // The original highlight attributes of the shape in order to restore them after the action.
            this.originalHighlightAttributes = new ShapeAttributes(null);
            this.originalPlacemarkHighlightAttributes = new PlacemarkAttributes(null);

            // Internal use only.
            // counters used to detect double click (time measured in ms)
            this._clicked0X = null;
            this._clicked0Y = null;
            this._clicked1X = null;
            this._clicked1Y = null;
            this._click0Time = 0;
            this._click1Time = 0;
            this._dbclickTimeout = 0;
            this._clickDelay = 200;

            // Internal use only.
            // Used for shape creation
            this.creatorEnabled = false;
            this.creatorShapeProperties = null;
            this.promisedShape = null;

            this._worldWindow.worldWindowController.addGestureListener(this);
        };

        Object.defineProperties(ShapeEditor.prototype, {
            /**
             * The World Window associated with this shape editor.
             * @memberof ShapeEditor.prototype
             * @type {WorldWindow}
             * @readonly
             */
            worldWindow: {
                get: function () {
                    return this._worldWindow;
                }
            },

            /**
             * The shape currently being edited.
             * @memberof ShapeEditor.prototype
             * @type {Object}
             * @readonly
             */
            shape: {
                get: function () {
                    return this._shape;
                }
            },

            /**
             * Attributes used for the control points that move the boundaries of the shape.
             * @memberof ShapeEditor.prototype
             * @type {PlacemarkAttributes}
             */
            moveControlPointAttributes: {
                get: function () {
                    return this._moveControlPointAttributes;
                },
                set: function (value) {
                    this._moveControlPointAttributes = value;
                }
            },

            /**
             * Attributes used for the shadow control points used to mask the middle of a segment.
             * @memberof ShapeEditor.prototype
             * @type {PlacemarkAttributes}
             */
            shadowControlPointAttributes: {
                get: function () {
                    return this._shadowControlPointAttributes;
                },
                set: function (value) {
                    this._shadowControlPointAttributes = value;
                }
            },

            /**
             * Attributes used for the control points that resize the shape.
             * @memberof ShapeEditor.prototype
             * @type {PlacemarkAttributes}
             */
            resizeControlPointAttributes: {
                get: function () {
                    return this._resizeControlPointAttributes;
                },
                set: function (value) {
                    this._resizeControlPointAttributes = value;
                }
            },

            /**
             * Attributes used for the control points that rotate the shape.
             * @memberof ShapeEditor.prototype
             * @type {PlacemarkAttributes}
             */
            rotateControlPointAttributes: {
                get: function () {
                    return this._rotateControlPointAttributes;
                },
                set: function (value) {
                    this._rotateControlPointAttributes = value;
                }
            },

            /**
             * Attributes used for the annotation that displays hints during the actions on the shape.
             * @memberof ShapeEditor.prototype
             * @type {AnnotationAttributes}
             */
            annotationAttributes: {
                get: function () {
                    return this._annotationAttributes;
                },
                set: function (value) {
                    this._annotationAttributes = value;
                    this.annotation.attributes = value;
                }
            }
        });

        /**
         * Creates the specified shape. Currently, only surface shapes are supported.
         * @param {SurfaceShape} shape The shape to edit.
         * @param {{}} properties Configuration properties for the shape:
         * <ul>
         *     <li>TODO: describe properties fro each shape</li>
         *     <li>attributes: {ShapeAttributes} attributes of the shape.</li>
         * <ul>
         * @return {Promise} <code>shape</code> if the creator can create the specified shape; otherwise
         * <code>null</code>.
         */
        ShapeEditor.prototype.create = function (shape, properties) {
            var res, rej;

            this.stop();
            this.setCreatorEnabled(true);

            for (var i = 0, len = this.editorFragments.length; i < len; i++) {
                var editorFragment = this.editorFragments[i];
                if (editorFragment.canHandle(shape)) {
                    this.activeEditorFragment = editorFragment;
                    this.creatorShapeProperties = properties;
                }
            }

            if (this.activeEditorFragment != null) {
                var promise = new Promise(function (resolve, reject) {
                    res = resolve;
                    rej = reject;
                });

                promise.resolve = res;
                promise.reject = rej;

                this.promisedShape = promise;

                return promise;
            } else {
                return null;
            }
        };

        /**
         * Identifies whether the shape editor create mode is armed.
         * @return true if armed, false if not armed.
         */
        ShapeEditor.prototype.isCreatorEnabled = function() {
            return this.creatorEnabled;
        };

        /**
         * Arms and disarms the shape editor create mode. When armed, editor monitors user input and builds the
         * shape in response to user actions. When disarmed, the shape editor ignores all user input for creation of a
         * new shape.
         *
         * @param armed true to arm the shape editor create mode, false to disarm it.
         */
        ShapeEditor.prototype.setCreatorEnabled = function(creatorEnabled) {
            if (this.creatorEnabled != creatorEnabled) {
                this.creatorEnabled = creatorEnabled;
            }
        };

        /**
         * Edits the specified shape. Currently, only surface shapes are supported.
         * @param {SurfaceShape} shape The shape to edit.
         * @param {{}} config Configuration properties for the ShapeEditor:
         * <ul>
         *     <li>move: {Boolean} move true to enable move action on shape, false to disable move action on shape.</li>
         *     <li>reshape: {Boolean} reshape true to enable reshape action on shape, false to disable reshape action on shape.</li>
         *     <li>rotate: {Boolean} rotate true to enable rotate action on shape, false to disable rotate action on shape.</li>
         *     <li>manageControlPoint: {Boolean} manageControlPoint true to enable the action to manage the control points of the shape, false to disable it.</li>
         * <ul>
         * @return {Boolean} <code>true</code> if the editor could start the edition of the specified shape; otherwise
         * <code>false</code>.
         */
        ShapeEditor.prototype.edit = function (shape, config) {
            this.stop();

            this._allowMove = Object.prototype.hasOwnProperty.call(config, 'move') ? config.move : this._allowMove;
            this._allowReshape = Object.prototype.hasOwnProperty.call(config, 'reshape') ? config.reshape : this._allowReshape;
            this._allowRotate = Object.prototype.hasOwnProperty.call(config, 'rotate') ?  config.rotate : this._allowRotate;
            this._allowManageControlPoint = Object.prototype.hasOwnProperty.call(config, 'manageControlPoint') ?  config.manageControlPoint : this._allowManageControlPoint;

            if (!this._allowReshape) {
                this._allowManageControlPoint = false;
            }

            // Look for a fragment that can handle the specified shape
            for (var i = 0, len = this.editorFragments.length; i < len; i++) {
                var editorFragment = this.editorFragments[i];
                if (editorFragment.canHandle(shape)) {
                    this.activeEditorFragment = editorFragment;
                }
            }

            // If we have a fragment for this shape, accept the shape and start the edition
            if (this.activeEditorFragment != null) {
                this._shape = shape;
                this._shape.highlighted = true;
                this.initializeControlElements();
                return true;
            }

            return false;
        };

        /**
         * Stops the current edition activity if any.
         * @return {SurfaceShape} The shape being edited if any; otherwise <code>null</code>.
         */
        ShapeEditor.prototype.stop = function () {
            this.removeControlElements();

            this.activeEditorFragment = null;

            this._allowMove = true;
            this._allowReshape = true;
            this._allowRotate = true;
            this._allowManageControlPoint = true;

            var currentShape = this._shape;
            this._shape = null;

            if (currentShape !== null) {
                currentShape.highlighted = false;
            }

            return currentShape;
        };

        // Internal use only.
        // Called by {@link ShapeEditor#edit} to initialize the control elements used for editing.
        ShapeEditor.prototype.initializeControlElements = function () {
            var moveControlAttributes = this._moveControlPointAttributes;
            var resizeControlAttributes = this._resizeControlPointAttributes;
            var rotateControlAttributes = this._rotateControlPointAttributes;
            var shadowControlAttributes = this._shadowControlPointAttributes;

            if (!this._allowMove) {
                moveControlAttributes = null;
            }

            if (!this._allowReshape) {
                resizeControlAttributes = null;
            }

            if (!this._allowRotate) {
                rotateControlAttributes = null;
            }

            if (!this._allowManageControlPoint) {
                shadowControlAttributes = null;
            }

            if (this._worldWindow.indexOfLayer(this.shadowShapeLayer) == -1) {
                this._worldWindow.insertLayer(0, this.shadowShapeLayer);
            }

            if (this._worldWindow.indexOfLayer(this.creatorShapeLayer) == -1) {
                this._worldWindow.addLayer(this.creatorShapeLayer);
            }

            if (this._worldWindow.indexOfLayer(this.controlPointsLayer) == -1) {
                this._worldWindow.addLayer(this.controlPointsLayer);
            }

            if (this._worldWindow.indexOfLayer(this.shadowControlPointsLayer) == -1) {
                this._worldWindow.addLayer(this.shadowControlPointsLayer);
            }


            if (this._worldWindow.indexOfLayer(this.accessoriesLayer) == -1) {
                this._worldWindow.addLayer(this.accessoriesLayer);
            }

            if (this._worldWindow.indexOfLayer(this.annotationLayer) == -1) {
                this._worldWindow.addLayer(this.annotationLayer);
            }

            if (this.isCreatorEnabled() &&
                (this.activeEditorFragment instanceof SurfacePolylineEditorFragment ||
                    this.activeEditorFragment instanceof SurfacePolygonEditorFragment)) {

                this.activeEditorFragment.initializeCreationControlElements(
                    this._shape,
                    this.controlPointsLayer.renderables,
                    moveControlAttributes
                );
            } else {
                this.activeEditorFragment.initializeControlElements(
                    this._shape,
                    this.controlPointsLayer.renderables,
                    this.shadowControlPointsLayer.renderables,
                    this.accessoriesLayer.renderables,
                    resizeControlAttributes,
                    rotateControlAttributes,
                    moveControlAttributes,
                    shadowControlAttributes
                );
            }

            this.updateControlElements();
        };

        // Internal use only.
        // Called by {@link ShapeEditor#stop} to remove the control elements used for editing.
        ShapeEditor.prototype.removeControlElements = function () {
            this._worldWindow.removeLayer(this.controlPointsLayer);
            this.controlPointsLayer.removeAllRenderables();

            this._worldWindow.removeLayer(this.creatorShapeLayer);
            this.creatorShapeLayer.removeAllRenderables();

            this._worldWindow.removeLayer(this.shadowControlPointsLayer);
            this.shadowControlPointsLayer.removeAllRenderables();

            this._worldWindow.removeLayer(this.accessoriesLayer);
            this.accessoriesLayer.removeAllRenderables();

            this._worldWindow.removeLayer(this.shadowShapeLayer);
            this.shadowShapeLayer.removeAllRenderables();

            this._worldWindow.removeLayer(this.annotationLayer);
        };

        // Internal use only.
        // Updates the position of the control elements.
        ShapeEditor.prototype.updateControlElements = function () {
            if (this.isCreatorEnabled() &&
                (this.activeEditorFragment instanceof SurfacePolylineEditorFragment ||
                    this.activeEditorFragment instanceof SurfacePolygonEditorFragment)) {
                this.activeEditorFragment.updateCreationControlElements(
                    this._shape,
                    this._worldWindow.globe,
                    this.controlPointsLayer.renderables
                );
            } else {
                this.activeEditorFragment.updateControlElements(
                    this._shape,
                    this._worldWindow.globe,
                    this.controlPointsLayer.renderables,
                    this.shadowControlPointsLayer.renderables,
                    this.accessoriesLayer.renderables
                );
            }
        };

        // Internal use only.
        // Dispatches the events relevant to the shape editor.
        ShapeEditor.prototype.onGestureEvent = function (event) {
            if(this._shape === null && !this.isCreatorEnabled()) {
                return;
            }

            // TODO Add support for touch devices

            if (event.type === "pointerup" || event.type === "mouseup") {
                this.handleMouseUp(event);
            } else if (event.type === "pointerdown" || event.type === "mousedown") {
                this.handleMouseDown(event);
            } else if (event.type === "pointermove" || event.type === "mousemove") {
                this.handleMouseMove(event);
            }
        };

        // Internal use only.
        // Triggers an action if the shape below the mouse is the shape being edited or a control point.
        ShapeEditor.prototype.handleMouseDown = function (event) {
            var x = event.clientX,
                y = event.clientY;

            this.actionCurrentX = x;
            this.actionCurrentY = y;

            var mousePoint = this._worldWindow.canvasCoordinates(x, y);
            var tmpOutlineWidth = 0;

            if (this._shape !== null && !this.isCreatorEnabled()) {
                tmpOutlineWidth = this._shape.highlightAttributes.outlineWidth;
                this._shape.highlightAttributes.outlineWidth = 5;
            }

            var pickList = this._worldWindow.pick(mousePoint);

            if (tmpOutlineWidth !== 0) {
                this._shape.highlightAttributes.outlineWidth = tmpOutlineWidth;
            }

            var terrainObject = pickList.terrainObject();

            if (this._click0Time && !this._click1Time) {
                this._clicked1X = x;
                this._clicked1Y = y;
                this._click1Time = Date.now() - this._click0Time;
            } else {
                this._clicked0X = x;
                this._clicked0Y = y;
                this._click0Time = Date.now();
                this._click1Time = 0;
                clearTimeout(this._dbclickTimeout);
                this._dbclickTimeout = setTimeout(function () {
                        this._click0Time = 0;
                    }, this._clickDelay
                );
            }

            for (var p = 0, len = pickList.objects.length; p < len; p++) {
                var object = pickList.objects[p];

                if (!object.isTerrain && !this.isCreatorEnabled()) {
                    var userObject = object.userObject;

                    if (userObject === this._shape) {
                        this.beginAction(terrainObject.position, this._allowManageControlPoint);
                        event.preventDefault();
                        break;

                    } else if (this.controlPointsLayer.renderables.indexOf(userObject) !== -1) {
                        this.beginAction(terrainObject.position, this._allowManageControlPoint, userObject);
                        event.preventDefault();
                        break;
                    } else if (this.shadowControlPointsLayer.renderables.indexOf(userObject) !== -1) {
                        this.beginAction(terrainObject.position, this._allowManageControlPoint, userObject);

                        if (this.actionType == 'shadow' && this._allowManageControlPoint) {
                            this.activeEditorFragment.convertShadowControlPoint(
                                this._shape,
                                this._worldWindow.globe,
                                userObject.userProperties.index,
                                terrainObject.position
                            );
                            this.updateControlElements();
                        }

                        event.preventDefault();
                        break;
                    }
                } else if (this.isCreatorEnabled() && this.activeEditorFragment !== null && this._shape === null) {
                    // set default shape attributes and highlight attributes
                    var attributes = new ShapeAttributes(null);
                    attributes.outlineColor = Color.BLACK;
                    attributes.interiorColor = new Color(0.8, 0.9, 0.9, 1.0);
                    attributes.outlineWidth = 5;

                    var highlightAttributes = new WorldWind.ShapeAttributes(attributes);
                    highlightAttributes.outlineColor = WorldWind.Color.RED;
                    highlightAttributes.outlineWidth = 5;

                    if (this.activeEditorFragment.isRegularShape()) {
                        if (this.activeEditorFragment instanceof PlacemarkEditorFragment) {
                            this.creatorShapeProperties.position = terrainObject.position;
                        } else {
                            this.creatorShapeProperties.center = terrainObject.position;
                        }

                        this.creatorShapeProperties.radius = 3;
                        this.creatorShapeProperties._boundaries = [
                            {
                                latitude: terrainObject.position.latitude - 0.5,
                                longitude: terrainObject.position.longitude - 0.5
                            },
                            {
                                latitude: terrainObject.position.latitude + 0.5,
                                longitude: terrainObject.position.longitude - 0.5
                            },
                            {
                                latitude: terrainObject.position.latitude + 0.5,
                                longitude: terrainObject.position.longitude + 0.5
                            }
                        ];

                        this._shape = this.activeEditorFragment.createShadowShape(this.creatorShapeProperties);

                        if (this.activeEditorFragment instanceof PlacemarkEditorFragment) {
                            var placemarkAttributes = new PlacemarkAttributes(null);
                            placemarkAttributes.imageSource = "";
                            placemarkAttributes.imageScale = 1;
                            placemarkAttributes.imageColor = Color.WHITE;
                            placemarkAttributes.drawLeaderLine = true;
                            placemarkAttributes.leaderLineAttributes.outlineColor = Color.RED;

                            var highlightPlacemarkAttributes = new PlacemarkAttributes(null);
                            highlightPlacemarkAttributes.imageSource = "";
                            highlightPlacemarkAttributes.imageScale = 1;
                            highlightPlacemarkAttributes.imageColor = Color.RED;

                            this._shape.attributes =  placemarkAttributes;
                            this._shape.highlightAttributes = highlightPlacemarkAttributes;
                        } else {

                            this._shape.attributes =  attributes;
                            this._shape.highlightAttributes = highlightAttributes;
                        }
                    } else {
                        // Polyline and Polygon are not regular shapes
                        if (this.creatorShapeProperties.boundaries == null) {
                            this.creatorShapeProperties.boundaries = [];
                        }

                        if (this.creatorShapeProperties.boundaries.length < 2) {
                            this.creatorShapeProperties.boundaries.push(new Location(
                                terrainObject.position.latitude,
                                terrainObject.position.longitude
                            ));

                            this.creatorShapeProperties.boundaries.push(new Location(
                                terrainObject.position.latitude,
                                terrainObject.position.longitude
                            ));

                            // return a polyline as shape
                            this._shape = this.editorFragments[4].createShadowShape(this.creatorShapeProperties);
                            this._shape.attributes =  attributes;
                            this._shape.highlightAttributes = highlightAttributes;
                        }
                    }

                    this._shape.highlighted = true;
                    this.initializeControlElements();
                    this.beginAction(terrainObject.position, this._allowManageControlPoint, this.controlPointsLayer.renderables[0]);

                    event.preventDefault();
                }
            }
        };

        // Internal use only.
        // Updates the current action if any.
        ShapeEditor.prototype.handleMouseMove = function (event) {
            var redrawRequired = this._highlightedItems.length > 0; // must redraw if we de-highlight previous shapes

            var mousePoint = this._worldWindow.canvasCoordinates(event.clientX, event.clientY);

            if (this._click0Time && !this._click1Time) {
                this._clicked1X = event.clientX;
                this._clicked1Y = event.clientY;
            }

            if (!(this._clicked0X === this._clicked1X
                    && this._clicked0Y === this._clicked1Y)) {
                clearTimeout(this._dbclickTimeout);
                this._click0Time = 0;
                this._click1Time = 0;
            }

            // De-highlight any previously highlighted shapes.
            for (var h = 0; h < this._highlightedItems.length; h++) {
                this._highlightedItems[h].highlighted = false;
            }
            this._highlightedItems = [];

            // Perform the pick. Must first convert from window coordinates to canvas coordinates, which are
            // relative to the upper left corner of the canvas rather than the upper left corner of the page.
            var pickList = this._worldWindow.pick(this._worldWindow.canvasCoordinates(event.clientX, event.clientY));
            if (pickList.objects.length > 0) {
                redrawRequired = true;
            }

            // Highlight the items picked by simply setting their highlight flag to true.
            if (pickList.objects.length > 0) {
                for (var p = 0; p < pickList.objects.length; p++) {
                    if (!pickList.objects[p].isTerrain && pickList.objects[p].userObject.userProperties.purpose) {
                        pickList.objects[p].userObject.highlighted = true;

                        // Keep track of highlighted items in order to de-highlight them later.
                        this._highlightedItems.push(pickList.objects[p].userObject);
                    }
                }
            }

            // Update the window if we changed anything.
            if (redrawRequired) {
                this._worldWindow.redraw(); // redraw to make the highlighting changes take effect on the screen
            }

            if (this.actionType && this._shape !== null) {

                var terrainObject = this._worldWindow.pickTerrain(mousePoint).terrainObject();

                if (terrainObject) {
                    if (this.actionType === ShapeEditorConstants.DRAG) {
                        if (this._allowMove) {
                            this.drag(event.clientX, event.clientY);
                        } else {
                            Logger.logMessage(Logger.LEVEL_INFO, "ShapeEditor", "handleMouseMove",
                                "Disabled action for selected shape.");
                        }
                    } else {
                        if (this._allowReshape || this._allowRotate) {
                            this.actionSecondaryBehavior = false;
                            this.actionControlPoint.highlighted = true;
                            this.reshape(terrainObject.position);
                        } else {
                            Logger.logMessage(Logger.LEVEL_INFO, "ShapeEditor", "handleMouseMove",
                                "Disabled action for selected shape.");
                        }
                    }

                    event.preventDefault();
                }
            }
        };

        // Internal use only.
        // Terminates the current action if any; otherwise handles other click responses.
        ShapeEditor.prototype.handleMouseUp = function (event) {
            var mousePoint = this._worldWindow.canvasCoordinates(event.clientX, event.clientY);
            var terrainObject = this._worldWindow.pickTerrain(mousePoint).terrainObject();

            if (this.isCreatorEnabled() && this.activeEditorFragment !== null && this._shape !== null) {
                if (this.activeEditorFragment.isRegularShape()) {
                    this.setCreatorEnabled(false);
                    this.promisedShape.resolve(this._shape);
                } else {
                    // add new vertex to shape
                    this.activeEditorFragment.createNewVertex(this._shape, this._worldWindow.globe, terrainObject.position);

                    if (this.activeEditorFragment instanceof SurfacePolylineEditorFragment) {
                        this.updateControlElements();
                        var latestIndex = this.controlPointsLayer.renderables.length - 1;
                        this.actionControlPoint = this.controlPointsLayer.renderables[latestIndex];
                    }

                    if (this.activeEditorFragment instanceof SurfacePolygonEditorFragment &&
                        this._shape instanceof SurfacePolyline &&
                        this._shape.boundaries.length === 3) {

                        var attributes = new ShapeAttributes(null);
                        attributes.outlineColor = Color.BLACK;
                        attributes.interiorColor = new Color(0.8, 0.9, 0.9, 1.0);
                        attributes.outlineWidth = 5;

                        var highlightAttributes = new ShapeAttributes(attributes);
                        highlightAttributes.outlineColor = Color.RED;
                        highlightAttributes.outlineWidth = 5;

                        this._shape = this.activeEditorFragment.createShadowShape(this._shape);
                        this._shape.attributes =  attributes;
                        this._shape.highlightAttributes = highlightAttributes;
                        this._shape.highlighted = true;
                        this.initializeControlElements();
                        this.beginAction(terrainObject.position, this._allowManageControlPoint, this.controlPointsLayer.renderables[0]);
                    }

                    this.updateControlElements();
                    this.updateAnnotation(this.actionControlPoint);

                    this._worldWindow.redraw();
                }
            }

            // The editor provides vertex insertion and removal for SurfacePolygon and SurfacePolyline.
            // Double click when the cursor is over a control point will remove it.
            // Single click when the cursor is over a shadow control point will add it.
            if (this.actionType) {
                if (this._click0Time && this._click1Time) {
                    if (this._click1Time <= this._clickDelay) {
                        if (this.actionControlPoint
                            && this.actionType == 'location'
                            && terrainObject
                            && this._allowManageControlPoint) {
                            if (this.isCreatorEnabled() && this.activeEditorFragment !== null) {
                                // if we are in the creation process and a double click is detected
                                // we verify the minimum conditions for the shape to be created
                                // and resolve the promise (only for polylines and polygons we need
                                // the double click as the final action)
                                if (this.activeEditorFragment instanceof SurfacePolylineEditorFragment) {
                                    if (this._shape.boundaries < 2) {
                                        this._shape = null;
                                    } else {
                                        this._shape.boundaries.splice(-2,2);
                                        this._shape.boundaries.shift();
                                        this.controlPointsLayer.renderables.pop();
                                        this.updateControlElements();
                                        this.updateAnnotation(this.actionControlPoint);

                                        this._worldWindow.redraw();
                                    }
                                } else if (this.activeEditorFragment instanceof SurfacePolygonEditorFragment) {
                                    if (this._shape.boundaries < 3) {
                                        this._shape = null;
                                    }  else {
                                        this._shape.boundaries.pop();
                                        this._shape.boundaries.shift();
                                        this.controlPointsLayer.renderables.pop();
                                        this.updateControlElements();
                                        this.updateAnnotation(this.actionControlPoint);

                                        this._worldWindow.redraw();
                                    }
                                } else {
                                    this._shape = null;
                                }

                                var scope = this;
                                this.promisedShape.resolve(this._shape);
                                this.promisedShape.then(function(shape) {
                                    scope.setCreatorEnabled(false);
                                    scope.endAction();
                                }).finally(function () {
                                    scope.stop();
                                });
                            }
                            this.actionSecondaryBehavior = true;
                            this.reshape(terrainObject.position);
                        }
                    }
                    clearTimeout(this._dbclickTimeout);
                    this._click0Time = 0;
                    this._click1Time = 0;

                }

                if (!this.isCreatorEnabled() && this.activeEditorFragment !== null) {
                    this.endAction();
                }
            }
        };

        // Internal use only.
        ShapeEditor.prototype.beginAction = function (initialPosition, alternateAction, controlPoint) {
            // Define the active transformation
            if (controlPoint) {
                this.actionType = controlPoint.userProperties.purpose;
            } else {
                this.actionType = ShapeEditorConstants.DRAG;
            }

            this.actionControlPoint = controlPoint;
            this.actionControlPosition = initialPosition;
            this.actionSecondaryBehavior = alternateAction;

            var editingAttributes = null;

            if (this.isCreatorEnabled()) {
                this.creatorShapeLayer.addRenderable(this._shape);
            }

            // Place a shadow shape at the original location of the shape
            if (this.activeEditorFragment instanceof PlacemarkEditorFragment) {
                this.originalHighlightAttributes = null;
                this.originalPlacemarkHighlightAttributes = this._shape.highlightAttributes;

                editingAttributes = new PlacemarkAttributes(this.originalPlacemarkHighlightAttributes);
                editingAttributes.imageColor.alpha = editingAttributes.imageColor.alpha * 0.7;
            } else {
                this.originalHighlightAttributes = this._shape.highlightAttributes;
                this.originalPlacemarkHighlightAttributes = null;

                editingAttributes = new ShapeAttributes(this.originalHighlightAttributes);
                editingAttributes.interiorColor.alpha = editingAttributes.interiorColor.alpha * 0.7;
                editingAttributes.outlineColor.alpha = editingAttributes.outlineColor.alpha * 0.7;
            }

            this._shape.highlightAttributes = editingAttributes;

            var shadowShape = this.activeEditorFragment.createShadowShape(this._shape);

            if (this.activeEditorFragment instanceof PlacemarkEditorFragment) {
                shadowShape.altitudeMode = WorldWind.CLAMP_TO_GROUND;
                shadowShape.highlightAttributes = new PlacemarkAttributes(this.originalPlacemarkHighlightAttributes);
            } else {
                shadowShape.highlightAttributes = new ShapeAttributes(this.originalHighlightAttributes);
            }
            shadowShape.highlighted = true;

            this.shadowShapeLayer.addRenderable(shadowShape);

            this._worldWindow.redraw();
        };

        // Internal use only.
        ShapeEditor.prototype.endAction = function () {
            this.creatorShapeLayer.removeAllRenderables();
            this.shadowShapeLayer.removeAllRenderables();

            if (this.activeEditorFragment instanceof PlacemarkEditorFragment) {
                this._shape.highlightAttributes = this.originalPlacemarkHighlightAttributes;
            } else {
                this._shape.highlightAttributes = this.originalHighlightAttributes;
            }

            this.hideAnnotation();

            if (this.actionControlPoint) {
                this.actionControlPoint.highlighted = false;
            }

            this.actionControlPoint = null;
            this.actionType = null;
            this.actionControlPosition = null;

            this._worldWindow.redraw();
        };

        // Internal use only.
        ShapeEditor.prototype.reshape = function (newPosition) {
            var purpose = this.actionControlPoint.userProperties.purpose;

            if ((purpose === ShapeEditorConstants.ROTATION && this._allowRotate) ||
                (purpose !== ShapeEditorConstants.ROTATION && this._allowReshape) ||
                (purpose === ShapeEditorConstants.LOCATION && this._allowManageControlPoint && this.actionSecondaryBehavior) ||
                (purpose === ShapeEditorConstants.SHADOW && this._allowManageControlPoint && this.actionSecondaryBehavior)) {

                this.activeEditorFragment.reshape(
                    this._shape,
                    this._worldWindow.globe,
                    this.actionControlPoint,
                    newPosition,
                    this.actionControlPosition,
                    this.actionSecondaryBehavior
                );

                this.actionControlPosition = newPosition;

                this.updateControlElements();
                this.updateAnnotation(this.actionControlPoint);

                this._worldWindow.redraw();
            }
        };

        // Internal use only.
        ShapeEditor.prototype.drag = function (clientX, clientY) {
            // Get reference position for the shape that is dragged
            var refPos = this._shape.getReferencePosition();

            // Get point for referenced position
            var refPoint = this._worldWindow.globe.computePointFromPosition(
                refPos.latitude,
                refPos.longitude,
                0,
                new Vec3(0, 0, 0)
            );

            var screenRefPoint = new Vec3(0, 0, 0);
            this._worldWindow.drawContext.project(refPoint, screenRefPoint);

            // Check drag distance
            var dx = clientX - this.actionCurrentX;
            var dy = clientY - this.actionCurrentY;

            // Get the latest position of mouse to calculate drag distance
            this.actionCurrentX = clientX;
            this.actionCurrentY = clientY;

            // Find intersection of the screen coordinates ref-point with globe
            var x = screenRefPoint[0] + dx;
            var y = this._worldWindow.canvas.height - screenRefPoint[1] + dy;

            var ray = this._worldWindow.rayThroughScreenPoint(new Vec2(x, y));

            // Check if the mouse is over the globe and move shape
            var intersection = new Vec3(0, 0, 0);
            if (this._worldWindow.globe.intersectsLine(ray, intersection)) {
                var p = new Position(0, 0, 0);
                this._worldWindow.globe.computePositionFromPoint(intersection[0], intersection[1], intersection[2], p);
                this._shape.moveTo(this._worldWindow.globe, new Location(p.latitude, p.longitude));
            }

            // Update control points and shape annotation
            this.updateControlElements();
            this.updateShapeAnnotation();

            this._worldWindow.redraw();
        };

        // Internal use only.
        ShapeEditor.prototype.updateAnnotation = function (controlPoint) {
            this.annotationLayer.enabled = true;

            this.annotation.position = new Position(
                controlPoint.position.latitude,
                controlPoint.position.longitude,
                0
            );

            var annotationText;
            if (controlPoint.userProperties.size !== undefined) {
                annotationText = this.formatLength(controlPoint.userProperties.size);
            } else if (controlPoint.userProperties.rotation !== undefined) {
                annotationText = this.formatRotation(controlPoint.userProperties.rotation);
            } else {
                annotationText = this.formatLatitude(controlPoint.position.latitude)
                    + " "
                    + this.formatLongitude(controlPoint.position.longitude);
            }
            this.annotation.text = annotationText;
        };

        // Internal use only.
        ShapeEditor.prototype.hideAnnotation = function (controlPoint) {
            this.annotationLayer.enabled = false;
        };

        // Internal use only.
        ShapeEditor.prototype.updateShapeAnnotation = function () {
            var center = this.activeEditorFragment.getShapeCenter(this._shape, this._worldWindow.globe);

            var temporaryMarker = new Placemark(
                new Position(center.latitude, center.longitude, 0),
                null
            );

            this.updateAnnotation(temporaryMarker);
        };

        // Internal use only.
        ShapeEditor.prototype.formatLatitude = function (number) {
            var suffix = number < 0 ? "\u00b0S" : "\u00b0N";
            return Math.abs(number).toFixed(4) + suffix;
        };

        // Internal use only.
        ShapeEditor.prototype.formatLongitude = function (number) {
            var suffix = number < 0 ? "\u00b0W" : "\u00b0E";
            return Math.abs(number).toFixed(4) + suffix;
        };

        // Internal use only.
        ShapeEditor.prototype.formatLength = function (number) {
            var suffix = " km";
            return Math.abs(number / 1000.0).toFixed(3) + suffix;
        };

        // Internal use only.
        ShapeEditor.prototype.formatRotation = function (rotation) {
            return rotation.toFixed(4) + "°";
        };

        return ShapeEditor;
    }
);