Project

General

Profile

1
/*
2
Author: imke doerge AT geodan.nl
3
License: LGPL as per: http://www.gnu.org/copyleft/lesser.html
4
$Id: Measurement.js 3821 2008-02-01 13:58:30Z ahocevar $
5
*/
6

    
7
// Ensure this object's dependencies are loaded.
8
mapbuilder.loadScript(baseDir+"/widget/EditButtonBase.js");
9

    
10

    
11
/**
12
 * When this button is selected, clicks on the MapPane will add a
13
 * new point to a line and the total distance of the line will be calculated
14
 * @constructor
15
 * @base EditButtonBase
16
 * @author Imke Doerge AT geodan.nl
17
 * @param widgetNode The widget from the Config XML file.
18
 * @param model  The parent model for this  widget.
19
 */
20
function Measurement(widgetNode, model) {
21
  EditButtonBase.apply(this, new Array(widgetNode, model));
22

    
23
  /**
24
   * Interactive EditLine control.
25
   * @param objRef reference to this object.
26
   * @return {OpenLayers.Control} class of the OL control.
27
   */
28
  this.createControl = function(objRef) {
29
    var Control = OpenLayers.Class(OpenLayers.Control.DrawFeature, {
30
      // this is needed because all editing tools are of type
31
      // OpenLayers.Control.DrawFeature
32
      CLASS_NAME: 'mbMeasurement'
33
    });
34
    return Control;
35
  }
36
  
37
  this.instantiateControl = function(objRef, Control) {
38
     var control = new Control(objRef.featureLayer, OpenLayers.Handler.Path,
39
          {callbacks: {point: objRef.startAction}});
40
    control.objRef = objRef;
41
    control.activate = function() {
42
      Control.prototype.activate.apply(this, arguments);
43
      objRef.targetModel.setParam('showDistance', 0);
44
    }
45
    control.deactivate = function() {
46
      Control.prototype.deactivate.apply(this, arguments);
47
      objRef.targetModel.setParam('showDistance', null);
48
    }
49
    return control;
50
  }
51
  
52
  // override default cursor by user
53
  // cursor can be changed by specifying a new cursor in config file
54
  this.cursor = "crosshair";	
55
        
56
  var totalDistance=0;
57
  var distance = 0;
58
  var state = false;
59
  var restart = false;
60
  
61
  this.startAction = function(pointGeometry) {
62
    var objRef = this.objRef;
63
    objRef.pointGeometry = pointGeometry;
64
    if (!objRef.targetModel.doc) {
65
      objRef.targetModel.addListener("loadModel", objRef.doAction, objRef);
66
      config.loadModel(objRef.targetModel.id, objRef.defaultModelUrl);
67
    } else {
68
      objRef.doAction(objRef);
69
    }
70
  }
71
  /**
72
   * Append a point to a line and calculate the distance between all
73
   * points on the line. This will be called by the OpenLayers control
74
   * for this widget in the context of the control.
75
   * @param pointGeometry OpenLayers Geometry of the point to add
76
   */
77
  this.doAction = function(objRef) {
78
    pointGeometry = objRef.pointGeometry;
79
    objRef.targetModel.removeListener("loadModel", objRef.doAction, objRef);
80
    if (objRef.enabled) {
81
		  if(objRef.restart) {
82
        objRef.model.setParam("clearMeasurementLine");
83
			  objRef.restart= false;
84
			}
85
      var point=[pointGeometry.x, pointGeometry.y];
86
      var old=objRef.targetModel.getXpathValue(objRef.targetModel,objRef.featureXpath);
87
      if(!old) old="";
88
        sucess=objRef.targetModel.setXpathValue(objRef.targetModel,objRef.featureXpath,old+" "+point[0]+","+point[1]);
89
      if(!sucess) {
90
        alert(mbGetMessage("invalidFeatureXpathMeasurement", objRef.featureXpath));
91
      }
92
			
93
      var lineCoords = objRef.targetModel.getXpathValue(objRef.targetModel, objRef.featureXpath);
94
      var coordArray = lineCoords.split(" ");
95
      if (coordArray.length >= 3) {
96
        var point_P = coordArray[coordArray.length-2];
97
        var point_Q = coordArray[coordArray.length-1];
98
            
99
        //Split point in x and y coordinate
100
        var P =point_P.split(",");
101
        var Q =point_Q.split(",");
102
              
103
        //transform coordinates from lat/lon to x/y in meter 
104
        objRef.srs = srs.toUpperCase();
105
        objRef.proj = new OpenLayers.Projection(objRef.srs);
106
        
107
        if (!P || !Q  ){
108
          alert(mbGetMessage("projectionNotSupported"));
109
        }
110
        else {
111
          //If projection is in meters use simple pythagoras
112
          if(objRef.proj.getUnits() == "meters" || objRef.proj.getUnits() == "m") {
113
            Xp=parseFloat(P[0]);
114
            Yp=parseFloat(P[1]);
115
            Xq=parseFloat(Q[0]);
116
            Yq=parseFloat(Q[1]);
117
            // calculate the distance between these two points via Pythagoras' theorem  
118
            distance=Math.sqrt(((Xp-Xq)*(Xp-Xq))+((Yp-Yq)*(Yp-Yq)))
119
			      if(distance==0) {
120
              objRef.restart = true;
121
              objRef.model.setParam("clearMouseLine");objRef.targetModel.setParam("mouseRenderer", false);
122
		    		return;
123
				  	}
124
            totalDistance = Math.round(totalDistance + distance);
125
          }
126
          //If projection is in degrees use great circle formulae 
127
          //http://williams.best.vwh.net/avform.htm#GCF
128
          else if(objRef.proj.getUnits() == "degrees" || objRef.proj.getUnits() == null) {
129
            var deg2rad = Math.PI / 180.0
130
            var centerOfEarth=new Array(0,0);
131
            LonpRad=parseFloat(P[0]) * deg2rad;
132
            LatpRad=parseFloat(P[1]) * deg2rad;
133
            LonqRad=parseFloat(Q[0]) * deg2rad;
134
            LatqRad=parseFloat(Q[1]) * deg2rad;
135
            
136
            if((LonpRad>0 && LonqRad<0) || (LonpRad<0 && LonqRad>0)){
137
	            if(LonpRad<0){
138
	           	 	radDistance=Math.acos(Math.sin(LatpRad)*Math.sin(LatqRad)+Math.cos(LatpRad)*Math.cos(LatqRad)*Math.cos(LonpRad));
139
	            	radDistance+=Math.acos(Math.sin(LatpRad)*Math.sin(LatqRad)+Math.cos(LatpRad)*Math.cos(LatqRad)*Math.cos(LonqRad));
140
	            }
141
	            if(LonqRad<0){
142
	            	radDistance=Math.acos(Math.sin(LatpRad)*Math.sin(LatqRad)+Math.cos(LatpRad)*Math.cos(LatqRad)*Math.cos(LonqRad));
143
	            	radDistance+=Math.acos(Math.sin(LatpRad)*Math.sin(LatqRad)+Math.cos(LatpRad)*Math.cos(LatqRad)*Math.cos(LonpRad));
144
	            }
145
             }
146
             else{
147
             
148
            	radDistance=Math.acos(Math.sin(LatpRad)*Math.sin(LatqRad)+Math.cos(LatpRad)*Math.cos(LatqRad)*Math.cos(LonpRad-LonqRad));
149
            
150
            }
151
            //radDistance=Math.acos(Math.sin(Latp)*Math.sin(Latq)+Math.cos(Latp)*Math.cos(Latq)*Math.cos(Lonp-Lonq));
152
            distance =radDistance * 6378137;
153
            if(distance==0) {
154
              objRef.restart = true;
155
              objRef.model.setParam("clearMouseLine");objRef.targetModel.setParam("mouseRenderer", false);
156
            return;
157
            }
158
            totalDistance = Math.round(totalDistance + distance);
159
          }
160
          else alert(mbGetMessage("cantCalculateDistance"));
161
        }
162
      }    
163
      objRef.targetModel.setParam("showDistance", totalDistance); // set parameter for the distance output
164
    }
165
  }
166
  
167
  /**
168
   * This will be called as defined in EditButtonBase.js. It is called
169
   * when measurement is finished (ie. when user double-clicks on the map)
170
   * @param objRef reference to this widget
171
   * @param feature complete measurement path - currently unused.
172
   */
173
  this.setFeature = function(objRef, feature) {
174
    objRef.restart = true;
175
  }
176
    
177
  this.clearMeasurementLine = function(objRef){
178
    if (totalDistance !=0) {
179
      totalDistance=0;
180
      sucess=objRef.targetModel.setXpathValue(objRef.targetModel,objRef.featureXpath,"");
181
      if(!sucess){
182
        alert(mbGetMessage("invalidFeatureXpathMeasurement", objRef.featureXpath));
183
      }
184
      objRef.targetModel.setParam("refresh");
185
    }
186
  }
187
  //add a Listener to the model
188
	this.model.addListener("clearMeasurementLine", this.clearMeasurementLine, this);
189

    
190
}
(84-84/145)