[mapguide-users] MG Fusion: Ruler Widget improvement - ruler file...

web web at infogeo.ch
Fri Aug 17 08:16:10 EDT 2007


/********************************************************************** * 
 * @project Fusion
 * @revision $Id: Ruler.js,v 1.37 2007/07/04 21:25:54 assefa Exp $
 * @purpose Ruler widget
 * @author pspencer at dmsolutions.ca
 * Copyright (c) 2007 DM Solutions Group Inc.
 *****************************************************************************
 * This code shall not be copied or used without the expressed written consent
 * of DM Solutions Group Inc.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 * 
 * The above copyright notice and this permission notice shall be included
 * in all copies or substantial portions of the Software.
 * ********************************************************************
 *
 * The Ruler widget allows the user to measure distances on the map in
 * one or more segments.
 *
 * Distances for the current segment and the total distance are advertised
 * through the Fusion.Event.RULER_DISTANCE_CHANGED event.
 * 
 * **********************************************************************/

 

Fusion.Event.RULER_DISTANCE_CHANGED = Fusion.Event.lastEventId++;

Fusion.Widget.Ruler = Class.create();
Fusion.Widget.Ruler.prototype = {
    isDigitizing: false,
    //distance of each segment
    aDistances: [],
    //cumulativeDistance distance
    cumulativeDistance: 0,
    lastDistance: 0,
    
    /* the units to display distances in */
    units: Fusion.UNKNOWN,

    /* Precision of the distance displayed */
    precision: null,
    
    /* an HTML container to put the current distance in */
    rulerTip: null,
    
    /* Static position of Tooltip Box TOP */
    rulerTipPositionTop: null,
   
    /* Static position of Tooltip Box LEFT */ 
    rulerTipPositionLeft: null,
    
    /* Tooltip appearance: static or dynamic */
    tooltipType: '',
    
    initialize : function(oCommand) {
        /*console.log('Ruler.initialize');*/
        Object.inheritFrom(this, Fusion.Widget.prototype, ['Ruler', true, oCommand]);
        Object.inheritFrom(this, Fusion.Tool.ButtonBase.prototype, []);
        Object.inheritFrom(this, Fusion.Tool.Canvas.prototype, []);
        this.setMap(oCommand.getMap());
        this.asCursor = ['crosshair'];
        var json = oCommand.jsonNode;
        this.units = (json.Units && (json.Units[0] != '')) ?
                      Fusion.unitFromName(json.Units[0]): this.units;      
        this.precision = json.Precision ? json.Precision[0] : 2;
        var container = json.RulerTooltipContainer ? json.RulerTooltipContainer[0] : '';
        //set default container if not specified
        
        var cont = document.getElementById(container);
        if (cont == null){
        cont = document.createElement('div');
        cont.id = container;
        }
       
        if (cont != '') {
            this.rulerTip = $(cont);
        }
        if (this.rulerTip) {
            this.tooltipType = json.RulerTooltipType ? json.RulerTooltipType[0] : 'dynamic';
            if (this.tooltipType == 'dynamic') {
                var oDomElem =  this.getMap().getDomObj();
                oDomElem.appendChild(this.rulerTip);
                this.rulerTip.style.position = 'absolute';
                this.rulerTip.style.display = 'none';
                this.rulerTip.style.top = '0px';
                this.rulerTip.style.left = '0px';
                this.rulerTip.style.zIndex = 101;
            }
            if (this.tooltipType == 'static') {
             this.rulerTipPositionTop = json.RulerTipPositionTop ? json.RulerTipPositionTop[0] : 15;
                this.rulerTipPositionLeft = json.RulerTipPositionLeft ? json.RulerTipPositionLeft[0] : 15;                                
                var oDomElem =  this.getMap().getDomObj();
                oDomElem.appendChild(this.rulerTip);
                this.rulerTip.style.position = 'absolute';
                this.rulerTip.style.display = 'none';
                this.rulerTip.style.top = this.rulerTipPositionTop;
                this.rulerTip.style.left = this.rulerTipPositionLeft;
                this.rulerTip.style.zIndex = 101;
            }
        }
        
        this.registerEventID(Fusion.Event.RULER_DISTANCE_CHANGED);
        this.getMap().registerForEvent(Fusion.Event.MAP_EXTENTS_CHANGED, this.resetCanvas.bind(this));
        this.keyHandler = this.onKeyPress.bind(this);
    },
    
    onKeyPress: function(e) {
        //console.log('Rule::onKeyPress');
        var charCode = (e.charCode ) ? e.charCode : ((e.keyCode) ? e.keyCode : e.which);
        //console.log(charCode);
        if (charCode == Event.KEY_ESC) {
            this.resetCanvas();
        } 
    },
    
    /**
     * (public) activate()
     *
     * activate the ruler tool
     */
    activateTool: function() {
        this.getMap().activateWidget(this);
        this._oButton.activateTool();
    },
    
    
    activate: function() {
        this.activateCanvas();
        this.aDistances = [];
        this.cumulativeDistance = 0;
        this.lastDistance = 0;
        this.triggerEvent(Fusion.Event.RULER_DISTANCE_CHANGED, this);
        Event.observe(document,"keypress",this.keyHandler);
    },
    
    /**
     * (public) deactivate()
     *
     * deactivate the ruler tool
     */
    deactivate: function() {
        //console.log('Ruler.deactivate');
        Event.stopObserving(document, 'keypress', this.keyHandler);           
        this._oButton.deactivateTool();
        this.deactivateCanvas();
        this.resetCanvas();
    },
    
    resetCanvas: function() {
        if (this.isDigitizing) {
            this.isDigitizing = false;
        }
        this.clearContext();
        this.aDistances = [];
        this.cumulativeDistance = 0;
        this.lastDistance = 0;
        this.triggerEvent(Fusion.Event.RULER_DISTANCE_CHANGED, this);
        if (this.rulerTip) {
            this.updateTip(null);
        }
    },
    
    /**
     * (public) mouseDown(e)
     *
     * handle the mouse down event
     *
     * @param e Event the event that happened on the mapObj
     */
    mouseDown: function(e) {
        if (Event.isLeftClick(e)) {
            var map = this.getMap();
            var p = map.getEventPosition(e);
            var gp = map.pixToGeo(p.x, p.y);

            if (!this.isDigitizing) {
                this.currentFeature = new Fusion.Tool.Canvas.Line(map);
                this.lastDistance = 0;
                this.cumulativeDistance = 0;
                this.aDistances = [];
                var from = new Fusion.Tool.Canvas.Node(gp.x,gp.y, map);
                var to = new Fusion.Tool.Canvas.Node(gp.x,gp.y, map);
                var seg = new Fusion.Tool.Canvas.Segment(from,to);
                seg.setEditing(true);
                this.currentFeature.addSegment(seg);
                this.clearContext();
                this.currentFeature.draw(this.context);     
                this.isDigitizing = true;
            } else {
                var seg = this.currentFeature.lastSegment();
                var d = this.measureSegment(seg);
                this.aDistances.push(d);
                this.cumulativeDistance += d;
                this.lastDistance = 0;
                this.triggerEvent(Fusion.Event.RULER_DISTANCE_CHANGED, this, this.getDistance());
                if (this.rulerTip) {
                    this.updateTip(e);
                }
                seg.setEditing(false);
                seg = this.currentFeature.extendLine();
                seg.setEditing(true);
                this.clearContext();
                this.currentFeature.draw(this.context);
            }
        }
    },

    /**
     * (public) mouseMove(e)
     *
     * handle the mouse move event
     *
     * @param e Event the event that happened on the mapObj
     */
    mouseMove: function(e) {
        if (!this.isDigitizing) {
            return;
        }
        
        var oElement = this.getMap().getDomObj();
        var target = e.target || e.srcElement;
        if (target.id != '_oEventDiv_'+oElement.id) {
            return;
        }

        var p = this.getMap().getEventPosition(e);
        var gp = this.getMap().pixToGeo(p.x, p.y);
        var seg = this.currentFeature.lastSegment();
        seg.to.set(gp.x,gp.y);
        this.clearContext();
        this.currentFeature.draw(this.context);
        this.updateDistance(seg, e);
    },

    /**
     * (public) dblClick(e)
     *
     * handle the mouse dblclick event
     *
     * @param e Event the event that happened on the mapObj
     */
    dblClick: function(e) {
        //console.log('Digitizer.dblClick');
        if (!this.isDigitizing) return;
        var p = this.getMap().getEventPosition(e);
        var gp = this.getMap().pixToGeo(p.x, p.y);
        var seg = this.currentFeature.lastSegment();
        seg.setEditing(false);
        seg.to.set(gp.x,gp.y);
        this.clearContext();
        this.currentFeature.draw(this.context);
        this.isDigitizing = false;
    },
    
    measureSegment: function(seg) {
        var map = this.getMap();
        var dist = Math.sqrt(Math.pow(seg.to.x-seg.from.x,2) +
                         Math.pow(seg.to.y-seg.from.y,2));
        if (this.units != Fusion.PIXELS) {
            dist = dist * map._fMetersperunit;
        }
        /* magic number - this means the map units are meters! */
        if (map._fMetersperunit == 111319.4908) {
            var center = map.getCurrentCenter();
            dist = dist * Math.cos(2 * Math.PI * center.y / 360);
        }
        return dist;
    },
    
    updateDistance: function(seg, e) {
        this.lastDistance = this.measureSegment(seg);
        this.triggerEvent(Fusion.Event.RULER_DISTANCE_CHANGED, this, this.getDistance());
        if (this.rulerTip) {
            this.updateTip(e);
        }
    },
    
    getLastDistance: function() {
        var d = this.lastDistance;
        if (this.units != Fusion.PIXELS && this.units != Fusion.METERS) {
            d = Fusion.fromMeter(this.units,d);
        }
        return d;
    },
    
    getDistance: function() {
        var totalDistance = this.cumulativeDistance + this.lastDistance;
        if (this.units != Fusion.PIXELS && this.units != Fusion.METERS) {
            totalDistance = Fusion.fromMeter(this.units,totalDistance);
        }
        return totalDistance;
    },
    
    setParameter: function(param, value) {
      //console.log('setParameter: ' + param + ' = ' + value);
        if (param == 'Units') {
            this.units = Fusion.unitFromName(value);
        }
    },
    
    updateTip: function(e) {
        if (!this.rulerTip) {
            return;
        }
        var segDistance = this.getLastDistance();
        var totalDistance = this.getDistance();
        segDistance = parseInt(segDistance * this.precision)/this.precision;
        totalDistance = parseInt(totalDistance * this.precision)/this.precision;
        if (segDistance == 0 && totalDistance == 0) 
         {
            this.rulerTip.innerHTML = '';
            if(this.tooltipType == 'dynamic' ||this.tooltipType == 'static' ) 
            {
                this.rulerTip.style.display = 'none';
            }
            
         } 
         else 
         {  
            if(Fusion.unitName(this.units) == "Meters")
            var unitDisplay = " m";
            this.rulerTip.innerHTML = "<table><tr><td align='left' style='font-weight: bold'>Segment: </td><td>&nbsp</td><td>" + segDistance + unitDisplay + "</td>" + "<tr><td align='left' style='font-weight: bold'>Total: </td><td>&nbsp</td><td>" + totalDistance + unitDisplay +"</td></tr></table>";
            if(this.tooltipType == 'dynamic') 
            {
                this.rulerTip.style.display = 'block';
                this.rulerTip.style.visibility = 'visible';
                var p = this.getMap().getEventPosition(e);
                var size = Element.getDimensions(this.rulerTip);
                var t = (p.y - size.height * 1.5);
                if (t < 0) {
                    t = p.y + size.height * 0.5;
                }
                
                var pSize = Element.getDimensions(this.rulerTip.parentNode);
                var l = p.x;
                if (l+size.width > pSize.width) {
                    l = p.x - size.width;
                }
                
                this.rulerTip.style.top = t + 'px';
                this.rulerTip.style.left = l + 'px';
            }//end dynamic
            if(this.tooltipType == 'static') 
            {
                this.rulerTip.style.display = 'block';
                this.rulerTip.style.visibility = 'visible';
            }//end dynamic
          }//else
    }
};


-------------- next part --------------
An HTML attachment was scrubbed...
URL: http://lists.osgeo.org/pipermail/mapguide-users/attachments/20070817/d65a72cf/attachment.html


More information about the mapguide-users mailing list