Commit df6bf5a4efd67d0903c2cc3605e3b3d6fc22a9e3
1 parent
9f277f10
update
Showing
20 changed files
with
3189 additions
and
53 deletions
src/main/java/com/bsth/controller/realcontrol/RealMapController.java
| @@ -23,29 +23,40 @@ public class RealMapController { | @@ -23,29 +23,40 @@ public class RealMapController { | ||
| 23 | * 根据线路获取站点路由及空间数据 | 23 | * 根据线路获取站点路由及空间数据 |
| 24 | */ | 24 | */ |
| 25 | @RequestMapping(value = "/stationSpatialData") | 25 | @RequestMapping(value = "/stationSpatialData") |
| 26 | - public Map<String, Object> stationSpatialData(@RequestParam String idx){ | 26 | + public Map<String, Object> stationSpatialData(@RequestParam String idx) { |
| 27 | return realMapService.stationSpatialData(idx); | 27 | return realMapService.stationSpatialData(idx); |
| 28 | } | 28 | } |
| 29 | 29 | ||
| 30 | /** | 30 | /** |
| 31 | * 停车场 | 31 | * 停车场 |
| 32 | + * | ||
| 32 | * @return | 33 | * @return |
| 33 | */ | 34 | */ |
| 34 | @RequestMapping(value = "/carParkSpatialData") | 35 | @RequestMapping(value = "/carParkSpatialData") |
| 35 | - public Map<String, Object> carParkSpatialData(){ | 36 | + public Map<String, Object> carParkSpatialData() { |
| 36 | return realMapService.carParkSpatialData(); | 37 | return realMapService.carParkSpatialData(); |
| 37 | } | 38 | } |
| 38 | 39 | ||
| 39 | 40 | ||
| 40 | /** | 41 | /** |
| 41 | - * | ||
| 42 | - * @Title: findRouteByLine | ||
| 43 | - * @Description: TODO(获取线路的站点,路段路由) | ||
| 44 | * @param @param lineCode | 42 | * @param @param lineCode |
| 45 | * @throws | 43 | * @throws |
| 44 | + * @Title: findRouteByLine | ||
| 45 | + * @Description: TODO(获取线路的站点,路段路由) | ||
| 46 | */ | 46 | */ |
| 47 | - @RequestMapping(value = "/findRouteByLine") | ||
| 48 | - public Map<String, Object> findRouteByLine(@RequestParam String lineCode){ | ||
| 49 | - return realMapService.findRouteByLine(lineCode); | ||
| 50 | - } | 47 | + @RequestMapping(value = "/findRouteByLine") |
| 48 | + public Map<String, Object> findRouteByLine(@RequestParam String lineCode) { | ||
| 49 | + return realMapService.findRouteByLine(lineCode); | ||
| 50 | + } | ||
| 51 | + | ||
| 52 | + | ||
| 53 | + /** | ||
| 54 | + * 获取线路的路段路由 和 站点路由信息 (为前端站间距计算提供数据源) | ||
| 55 | + * @param lineCode | ||
| 56 | + * @return | ||
| 57 | + */ | ||
| 58 | + @RequestMapping(value = "/findRouteAndStationByLine") | ||
| 59 | + public Map<String, Object> findRouteAndStationByLine(@RequestParam String lineCode){ | ||
| 60 | + return realMapService.findRouteAndStationByLine(lineCode); | ||
| 61 | + } | ||
| 51 | } | 62 | } |
src/main/java/com/bsth/data/schedule/DayOfSchedule.java
| @@ -117,7 +117,7 @@ public class DayOfSchedule implements CommandLineRunner { | @@ -117,7 +117,7 @@ public class DayOfSchedule implements CommandLineRunner { | ||
| 117 | @Override | 117 | @Override |
| 118 | public void run(String... arg0) throws Exception { | 118 | public void run(String... arg0) throws Exception { |
| 119 | //翻班线程 | 119 | //翻班线程 |
| 120 | -// Application.mainServices.scheduleWithFixedDelay(scheduleRefreshThread, 15, 240, TimeUnit.SECONDS); | 120 | + Application.mainServices.scheduleWithFixedDelay(scheduleRefreshThread, 15, 240, TimeUnit.SECONDS); |
| 121 | //入库 | 121 | //入库 |
| 122 | // Application.mainServices.scheduleWithFixedDelay(schedulePstThread, 60, 60, TimeUnit.SECONDS); | 122 | // Application.mainServices.scheduleWithFixedDelay(schedulePstThread, 60, 60, TimeUnit.SECONDS); |
| 123 | //首班出场指令补发器 | 123 | //首班出场指令补发器 |
src/main/java/com/bsth/service/realcontrol/RealMapService.java
| @@ -11,4 +11,6 @@ public interface RealMapService { | @@ -11,4 +11,6 @@ public interface RealMapService { | ||
| 11 | Map<String,Object> carParkSpatialData(); | 11 | Map<String,Object> carParkSpatialData(); |
| 12 | 12 | ||
| 13 | Map<String,Object> findRouteByLine(String lineCode); | 13 | Map<String,Object> findRouteByLine(String lineCode); |
| 14 | + | ||
| 15 | + Map<String,Object> findRouteAndStationByLine(String lineCode); | ||
| 14 | } | 16 | } |
src/main/java/com/bsth/service/realcontrol/impl/RealChartsServiceImpl.java
| @@ -218,7 +218,7 @@ public class RealChartsServiceImpl implements RealChartsService { | @@ -218,7 +218,7 @@ public class RealChartsServiceImpl implements RealChartsService { | ||
| 218 | } | 218 | } |
| 219 | inStr = " (" + inStr.substring(1) + ")"; | 219 | inStr = " (" + inStr.substring(1) + ")"; |
| 220 | 220 | ||
| 221 | - String sql = "select min(DFSJ) as STARTDATE,max(DFSJ) as ENDDATE,SCHEDULE_DATE_STR,XL_BM,XL_DIR,REAL_EXEC_DATE from (SELECT SCHEDULE_DATE_STR,concat_ws('_',concat_ws('/', dfsj, fcsj_actual),fcno) AS DFSJ,XL_BM,XL_DIR,REAL_EXEC_DATE FROM bsth_c_s_sp_info_real WHERE schedule_date_str LIKE :month AND bc_type = 'normal' AND dfsj IS NOT NULL AND xl_bm IN "+inStr+") t group by SCHEDULE_DATE_STR,XL_BM,XL_DIR,REAL_EXEC_DATE"; | 221 | + String sql = "select min(DFSJ) as STARTDATE,max(DFSJ) as ENDDATE,SCHEDULE_DATE_STR,XL_BM,XL_DIR,REAL_EXEC_DATE from (SELECT SCHEDULE_DATE_STR,concat_ws('_',concat_ws('/', dfsj, fcsj_actual),fcno) AS DFSJ,XL_BM,XL_DIR,REAL_EXEC_DATE FROM bsth_c_s_sp_info_real WHERE schedule_date_str LIKE :month AND bc_type = 'normal' AND dfsj IS NOT NULL AND xl_bm IN "+inStr+" AND status !=-1) t group by SCHEDULE_DATE_STR,XL_BM,XL_DIR,REAL_EXEC_DATE"; |
| 222 | MapSqlParameterSource parameters = new MapSqlParameterSource(); | 222 | MapSqlParameterSource parameters = new MapSqlParameterSource(); |
| 223 | parameters.addValue("month", month+"-%"); | 223 | parameters.addValue("month", month+"-%"); |
| 224 | 224 |
src/main/java/com/bsth/service/realcontrol/impl/RealMapServiceImpl.java
| @@ -10,6 +10,7 @@ import com.google.common.base.Splitter; | @@ -10,6 +10,7 @@ import com.google.common.base.Splitter; | ||
| 10 | import org.slf4j.Logger; | 10 | import org.slf4j.Logger; |
| 11 | import org.slf4j.LoggerFactory; | 11 | import org.slf4j.LoggerFactory; |
| 12 | import org.springframework.beans.factory.annotation.Autowired; | 12 | import org.springframework.beans.factory.annotation.Autowired; |
| 13 | +import org.springframework.dao.DataAccessException; | ||
| 13 | import org.springframework.jdbc.core.BeanPropertyRowMapper; | 14 | import org.springframework.jdbc.core.BeanPropertyRowMapper; |
| 14 | import org.springframework.jdbc.core.JdbcTemplate; | 15 | import org.springframework.jdbc.core.JdbcTemplate; |
| 15 | import org.springframework.stereotype.Service; | 16 | import org.springframework.stereotype.Service; |
| @@ -111,8 +112,31 @@ public class RealMapServiceImpl implements RealMapService { | @@ -111,8 +112,31 @@ public class RealMapServiceImpl implements RealMapService { | ||
| 111 | return rs; | 112 | return rs; |
| 112 | } | 113 | } |
| 113 | 114 | ||
| 115 | + @Override | ||
| 116 | + public Map<String, Object> findRouteAndStationByLine(String lineCode) { | ||
| 117 | + Map<String, Object> rs = new HashMap<>(); | ||
| 118 | + try { | ||
| 119 | + //查询路段信息 | ||
| 120 | + String sql = "select r.LINE_CODE,r.SECTION_CODE,r.SECTIONROUTE_CODE,s.SECTION_NAME,ST_AsText(s.GSECTION_VECTOR) as GSECTION_VECTOR, r.DIRECTIONS from bsth_c_sectionroute r INNER JOIN bsth_c_section s on r.section=s.id where r.line_code=? and r.destroy=0"; | ||
| 121 | + List<Map<String, Object>> secList = jdbcTemplate.queryForList(sql, lineCode); | ||
| 122 | + rs.put("section", secList); | ||
| 123 | + | ||
| 124 | + //查询站点信息 | ||
| 125 | + sql="select r.STATION_NAME,r.STATION_ROUTE_CODE,r.LINE_CODE,r.STATION_CODE,r.STATION_MARK,s.G_LONX,s.G_LATY, r.DIRECTIONS from bsth_c_stationroute r INNER JOIN bsth_c_station s on r.station=s.id and r.line_code=? and r.destroy=0"; | ||
| 126 | + List<Map<String, Object>> stationList = jdbcTemplate.queryForList(sql, lineCode); | ||
| 127 | + rs.put("station", stationList); | ||
| 128 | + | ||
| 129 | + rs.put("status", ResponseCode.SUCCESS); | ||
| 130 | + } catch (DataAccessException e) { | ||
| 131 | + logger.error("", e); | ||
| 132 | + rs.put("status", ResponseCode.ERROR); | ||
| 133 | + } | ||
| 134 | + return rs; | ||
| 135 | + } | ||
| 136 | + | ||
| 114 | /** | 137 | /** |
| 115 | * wgs 坐标数组转 百度 | 138 | * wgs 坐标数组转 百度 |
| 139 | + * | ||
| 116 | * @param list | 140 | * @param list |
| 117 | * @return | 141 | * @return |
| 118 | */ | 142 | */ |
| @@ -122,11 +146,11 @@ public class RealMapServiceImpl implements RealMapService { | @@ -122,11 +146,11 @@ public class RealMapServiceImpl implements RealMapService { | ||
| 122 | StringBuilder itemStr; | 146 | StringBuilder itemStr; |
| 123 | String[] subArr, cds; | 147 | String[] subArr, cds; |
| 124 | TransGPS.Location location; | 148 | TransGPS.Location location; |
| 125 | - for(String item : list){ | 149 | + for (String item : list) { |
| 126 | subArr = item.split(","); | 150 | subArr = item.split(","); |
| 127 | 151 | ||
| 128 | itemStr = new StringBuilder(); | 152 | itemStr = new StringBuilder(); |
| 129 | - for(String coord : subArr){ | 153 | + for (String coord : subArr) { |
| 130 | cds = coord.split(" "); | 154 | cds = coord.split(" "); |
| 131 | //城建转经纬度 | 155 | //城建转经纬度 |
| 132 | //Map<String, Double> map = JWDUtil.ConvertSHToJW(Double.parseDouble(cds[0]), Double.parseDouble(cds[1])); | 156 | //Map<String, Double> map = JWDUtil.ConvertSHToJW(Double.parseDouble(cds[0]), Double.parseDouble(cds[1])); |
| @@ -143,6 +167,7 @@ public class RealMapServiceImpl implements RealMapService { | @@ -143,6 +167,7 @@ public class RealMapServiceImpl implements RealMapService { | ||
| 143 | 167 | ||
| 144 | /** | 168 | /** |
| 145 | * wgs 坐标数组转 Gcj | 169 | * wgs 坐标数组转 Gcj |
| 170 | + * | ||
| 146 | * @param list | 171 | * @param list |
| 147 | * @return | 172 | * @return |
| 148 | */ | 173 | */ |
| @@ -152,11 +177,11 @@ public class RealMapServiceImpl implements RealMapService { | @@ -152,11 +177,11 @@ public class RealMapServiceImpl implements RealMapService { | ||
| 152 | StringBuilder itemStr; | 177 | StringBuilder itemStr; |
| 153 | String[] subArr, cds; | 178 | String[] subArr, cds; |
| 154 | TransGPS.Location location; | 179 | TransGPS.Location location; |
| 155 | - for(String item : list){ | 180 | + for (String item : list) { |
| 156 | subArr = item.split(","); | 181 | subArr = item.split(","); |
| 157 | 182 | ||
| 158 | itemStr = new StringBuilder(); | 183 | itemStr = new StringBuilder(); |
| 159 | - for(String coord : subArr){ | 184 | + for (String coord : subArr) { |
| 160 | cds = coord.split(" "); | 185 | cds = coord.split(" "); |
| 161 | location = TransGPS.transformFromWGSToGCJ(TransGPS.LocationMake(Double.parseDouble(cds[0]), Double.parseDouble(cds[1]))); | 186 | location = TransGPS.transformFromWGSToGCJ(TransGPS.LocationMake(Double.parseDouble(cds[0]), Double.parseDouble(cds[1]))); |
| 162 | itemStr.append(location.getLng() + " " + location.getLat() + ","); | 187 | itemStr.append(location.getLng() + " " + location.getLat() + ","); |
src/main/resources/static/pages/control/lineallot/allot.html
| @@ -403,7 +403,7 @@ $(function(){ | @@ -403,7 +403,7 @@ $(function(){ | ||
| 403 | } | 403 | } |
| 404 | var f = arguments.callee | 404 | var f = arguments.callee |
| 405 | ,item = lsData[i]; | 405 | ,item = lsData[i]; |
| 406 | - | 406 | + |
| 407 | //$.get('/realMap/findRouteByLine', {lineCode: item.lineCode}, function(rs){ | 407 | //$.get('/realMap/findRouteByLine', {lineCode: item.lineCode}, function(rs){ |
| 408 | $.get('/realSchedule/findRouteByLine', {lineCode: item.lineCode}, function(rs){ | 408 | $.get('/realSchedule/findRouteByLine', {lineCode: item.lineCode}, function(rs){ |
| 409 | if(rs && rs.lineId){ | 409 | if(rs && rs.lineId){ |
src/main/resources/static/real_control_v2/css/line_schedule.css
| @@ -859,10 +859,46 @@ input.i-cbox[type=checkbox]{ | @@ -859,10 +859,46 @@ input.i-cbox[type=checkbox]{ | ||
| 859 | #add-sub-task-inpark-modal.uk-modal .uk-form.fv-form{ | 859 | #add-sub-task-inpark-modal.uk-modal .uk-form.fv-form{ |
| 860 | margin-bottom: 0px !important; | 860 | margin-bottom: 0px !important; |
| 861 | } | 861 | } |
| 862 | -#add-sub-task-inpark-modal .uk-form select{ | 862 | +#add-sub-task-inpark-modal .uk-form select, |
| 863 | +#add-sub-task-inpark-modal .uk-form input[type=text], | ||
| 864 | +#add-sub-task-inpark-modal .uk-form input[type=time]{ | ||
| 863 | width: 100% | 865 | width: 100% |
| 864 | } | 866 | } |
| 865 | 867 | ||
| 866 | #add-sub-task-inpark-modal .uk-grid{ | 868 | #add-sub-task-inpark-modal .uk-grid{ |
| 867 | margin-top: 10px; | 869 | margin-top: 10px; |
| 870 | +} | ||
| 871 | + | ||
| 872 | +.sub-task-card{ | ||
| 873 | + display: inline-block;vertical-align: top;margin-right: 1%; | ||
| 874 | + width: 48.5%; | ||
| 875 | +} | ||
| 876 | + | ||
| 877 | +.sub-task-card.destroy-card{ | ||
| 878 | + vertical-align: top; | ||
| 879 | + margin-right: 0; | ||
| 880 | + display: none; | ||
| 881 | +} | ||
| 882 | + | ||
| 883 | +#add-sub-task-inpark-modal .uk-modal-dialog{ | ||
| 884 | + width: 755px; | ||
| 885 | + transition: all .3s ease; | ||
| 886 | +} | ||
| 887 | + | ||
| 888 | +#add-sub-task-inpark-modal .uk-modal-dialog.three-children{ | ||
| 889 | + width: 955px; | ||
| 890 | +} | ||
| 891 | + | ||
| 892 | +#add-sub-task-inpark-modal .uk-modal-dialog.three-children .sub-task-card{ | ||
| 893 | + width: 32%; | ||
| 894 | +} | ||
| 895 | + | ||
| 896 | +.uk-modal-footer .ct-footer-left{ | ||
| 897 | + float: left; | ||
| 898 | + font-size: 12px; | ||
| 899 | +} | ||
| 900 | + | ||
| 901 | +.uk-modal-footer .ct-footer-left a{ | ||
| 902 | + color: #6e6b6b; | ||
| 903 | + margin-right: 5px; | ||
| 868 | } | 904 | } |
| 869 | \ No newline at end of file | 905 | \ No newline at end of file |
src/main/resources/static/real_control_v2/css/main.css
| @@ -330,4 +330,43 @@ li.map-panel{ | @@ -330,4 +330,43 @@ li.map-panel{ | ||
| 330 | background-color: #a09b9c; | 330 | background-color: #a09b9c; |
| 331 | background-image: -webkit-linear-gradient(top,#9a9898,#8c8687); | 331 | background-image: -webkit-linear-gradient(top,#9a9898,#8c8687); |
| 332 | background-image: linear-gradient(to bottom,#9a9898,#8c8687); | 332 | background-image: linear-gradient(to bottom,#9a9898,#8c8687); |
| 333 | +} | ||
| 334 | + | ||
| 335 | +.ct-bottom-drawer{ | ||
| 336 | + width: 100%; | ||
| 337 | + background: rgb(255, 255, 255); | ||
| 338 | + position: fixed; | ||
| 339 | + bottom: 0; | ||
| 340 | + box-shadow: 0 0 10px rgba(0,0,0,.3); | ||
| 341 | + transition: all .3s ease; | ||
| 342 | + height: 0; | ||
| 343 | +} | ||
| 344 | + | ||
| 345 | +.ct-bottom-drawer.open{ | ||
| 346 | + height: 320px; | ||
| 347 | +} | ||
| 348 | + | ||
| 349 | +.ct-bottom-drawer a.ct-bottom-drawer-close{ | ||
| 350 | + position: absolute; | ||
| 351 | + right: 10px; | ||
| 352 | + top: 10px; | ||
| 353 | + color: #c3c3c3; | ||
| 354 | + width: 20px; | ||
| 355 | + line-height: 20px; | ||
| 356 | +} | ||
| 357 | + | ||
| 358 | +.ct-bottom-drawer a.ct-bottom-drawer-close:hover{ | ||
| 359 | + text-decoration: none; | ||
| 360 | + color: grey; | ||
| 361 | +} | ||
| 362 | + | ||
| 363 | +.ct-bottom-drawer a.ct-bottom-drawer-close:after { | ||
| 364 | + display: block; | ||
| 365 | + content: "\f00d"; | ||
| 366 | + font-family: FontAwesome; | ||
| 367 | +} | ||
| 368 | + | ||
| 369 | +.ct-bottom-drawer-body{ | ||
| 370 | + width: 100%; | ||
| 371 | + height: 100%; | ||
| 333 | } | 372 | } |
| 334 | \ No newline at end of file | 373 | \ No newline at end of file |
src/main/resources/static/real_control_v2/css/pace.css
0 → 100644
| 1 | +.pace { | ||
| 2 | + width: 140px; | ||
| 3 | + height: 300px; | ||
| 4 | + margin-left: calc(50% - 70px); | ||
| 5 | + | ||
| 6 | + z-index: 2000; | ||
| 7 | + -webkit-transform: scale(0); | ||
| 8 | + -moz-transform: scale(0); | ||
| 9 | + -ms-transform: scale(0); | ||
| 10 | + -o-transform: scale(0); | ||
| 11 | + transform: scale(0); | ||
| 12 | + opacity: 0; | ||
| 13 | + -webkit-transition: all 2s linear 0s; | ||
| 14 | + -moz-transition: all 2s linear 0s; | ||
| 15 | + transition: all 2s linear 0s; | ||
| 16 | +} | ||
| 17 | + | ||
| 18 | +.pace .text{ | ||
| 19 | + position: absolute; | ||
| 20 | + bottom: -50px; | ||
| 21 | + font-size: 12px; | ||
| 22 | + transform: scale(4); | ||
| 23 | + color: grey; | ||
| 24 | +} | ||
| 25 | + | ||
| 26 | +.pace.pace-active { | ||
| 27 | + -webkit-transform: scale(.25); | ||
| 28 | + -moz-transform: scale(.25); | ||
| 29 | + -ms-transform: scale(.25); | ||
| 30 | + -o-transform: scale(.25); | ||
| 31 | + transform: scale(.25); | ||
| 32 | + opacity: 1; | ||
| 33 | +} | ||
| 34 | + | ||
| 35 | +.pace .pace-activity { | ||
| 36 | + width: 140px; | ||
| 37 | + height: 140px; | ||
| 38 | + border-radius: 70px; | ||
| 39 | + background: #29d; | ||
| 40 | + position: absolute; | ||
| 41 | + top: 0; | ||
| 42 | + z-index: 1911; | ||
| 43 | + -webkit-animation: pace-bounce 1s infinite; | ||
| 44 | + -moz-animation: pace-bounce 1s infinite; | ||
| 45 | + -o-animation: pace-bounce 1s infinite; | ||
| 46 | + -ms-animation: pace-bounce 1s infinite; | ||
| 47 | + animation: pace-bounce 1s infinite; | ||
| 48 | +} | ||
| 49 | + | ||
| 50 | +.pace .pace-progress { | ||
| 51 | + position: absolute; | ||
| 52 | + display: block; | ||
| 53 | + left: 50%; | ||
| 54 | + bottom: 0; | ||
| 55 | + z-index: 1910; | ||
| 56 | + margin-left: -30px; | ||
| 57 | + width: 60px; | ||
| 58 | + height: 75px; | ||
| 59 | + background: rgba(20, 20, 20, .1); | ||
| 60 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .1); | ||
| 61 | + border-radius: 30px / 40px; | ||
| 62 | + -webkit-transform: scaleY(.3) !important; | ||
| 63 | + -moz-transform: scaleY(.3) !important; | ||
| 64 | + -ms-transform: scaleY(.3) !important; | ||
| 65 | + -o-transform: scaleY(.3) !important; | ||
| 66 | + transform: scaleY(.3) !important; | ||
| 67 | + -webkit-animation: pace-compress .5s infinite alternate; | ||
| 68 | + -moz-animation: pace-compress .5s infinite alternate; | ||
| 69 | + -o-animation: pace-compress .5s infinite alternate; | ||
| 70 | + -ms-animation: pace-compress .5s infinite alternate; | ||
| 71 | + animation: pace-compress .5s infinite alternate; | ||
| 72 | +} | ||
| 73 | + | ||
| 74 | +@-webkit-keyframes pace-bounce { | ||
| 75 | + 0% { | ||
| 76 | + top: 0; | ||
| 77 | + -webkit-animation-timing-function: ease-in; | ||
| 78 | + } | ||
| 79 | + 40% {} | ||
| 80 | + 50% { | ||
| 81 | + top: 140px; | ||
| 82 | + height: 140px; | ||
| 83 | + -webkit-animation-timing-function: ease-out; | ||
| 84 | + } | ||
| 85 | + 55% { | ||
| 86 | + top: 160px; | ||
| 87 | + height: 120px; | ||
| 88 | + border-radius: 70px / 60px; | ||
| 89 | + -webkit-animation-timing-function: ease-in; | ||
| 90 | + } | ||
| 91 | + 65% { | ||
| 92 | + top: 120px; | ||
| 93 | + height: 140px; | ||
| 94 | + border-radius: 70px; | ||
| 95 | + -webkit-animation-timing-function: ease-out; | ||
| 96 | + } | ||
| 97 | + 95% { | ||
| 98 | + top: 0; | ||
| 99 | + -webkit-animation-timing-function: ease-in; | ||
| 100 | + } | ||
| 101 | + 100% { | ||
| 102 | + top: 0; | ||
| 103 | + -webkit-animation-timing-function: ease-in; | ||
| 104 | + } | ||
| 105 | +} | ||
| 106 | + | ||
| 107 | +@-moz-keyframes pace-bounce { | ||
| 108 | + 0% { | ||
| 109 | + top: 0; | ||
| 110 | + -moz-animation-timing-function: ease-in; | ||
| 111 | + } | ||
| 112 | + 40% {} | ||
| 113 | + 50% { | ||
| 114 | + top: 140px; | ||
| 115 | + height: 140px; | ||
| 116 | + -moz-animation-timing-function: ease-out; | ||
| 117 | + } | ||
| 118 | + 55% { | ||
| 119 | + top: 160px; | ||
| 120 | + height: 120px; | ||
| 121 | + border-radius: 70px / 60px; | ||
| 122 | + -moz-animation-timing-function: ease-in; | ||
| 123 | + } | ||
| 124 | + 65% { | ||
| 125 | + top: 120px; | ||
| 126 | + height: 140px; | ||
| 127 | + border-radius: 70px; | ||
| 128 | + -moz-animation-timing-function: ease-out;} | ||
| 129 | + 95% { | ||
| 130 | + top: 0; | ||
| 131 | + -moz-animation-timing-function: ease-in; | ||
| 132 | + } | ||
| 133 | + 100% {top: 0; | ||
| 134 | + -moz-animation-timing-function: ease-in; | ||
| 135 | + } | ||
| 136 | +} | ||
| 137 | + | ||
| 138 | +@keyframes pace-bounce { | ||
| 139 | + 0% { | ||
| 140 | + top: 0; | ||
| 141 | + animation-timing-function: ease-in; | ||
| 142 | + } | ||
| 143 | + 50% { | ||
| 144 | + top: 140px; | ||
| 145 | + height: 140px; | ||
| 146 | + animation-timing-function: ease-out; | ||
| 147 | + } | ||
| 148 | + 55% { | ||
| 149 | + top: 160px; | ||
| 150 | + height: 120px; | ||
| 151 | + border-radius: 70px / 60px; | ||
| 152 | + animation-timing-function: ease-in; | ||
| 153 | + } | ||
| 154 | + 65% { | ||
| 155 | + top: 120px; | ||
| 156 | + height: 140px; | ||
| 157 | + border-radius: 70px; | ||
| 158 | + animation-timing-function: ease-out; | ||
| 159 | + } | ||
| 160 | + 95% { | ||
| 161 | + top: 0; | ||
| 162 | + animation-timing-function: ease-in; | ||
| 163 | + } | ||
| 164 | + 100% { | ||
| 165 | + top: 0; | ||
| 166 | + animation-timing-function: ease-in; | ||
| 167 | + } | ||
| 168 | +} | ||
| 169 | + | ||
| 170 | +@-webkit-keyframes pace-compress { | ||
| 171 | + 0% { | ||
| 172 | + bottom: 0; | ||
| 173 | + margin-left: -30px; | ||
| 174 | + width: 60px; | ||
| 175 | + height: 75px; | ||
| 176 | + background: rgba(20, 20, 20, .1); | ||
| 177 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .1); | ||
| 178 | + border-radius: 30px / 40px; | ||
| 179 | + -webkit-animation-timing-function: ease-in; | ||
| 180 | + } | ||
| 181 | + 100% { | ||
| 182 | + bottom: 30px; | ||
| 183 | + margin-left: -10px; | ||
| 184 | + width: 20px; | ||
| 185 | + height: 5px; | ||
| 186 | + background: rgba(20, 20, 20, .3); | ||
| 187 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .3); | ||
| 188 | + border-radius: 20px / 20px; | ||
| 189 | + -webkit-animation-timing-function: ease-out; | ||
| 190 | + } | ||
| 191 | +} | ||
| 192 | + | ||
| 193 | +@-moz-keyframes pace-compress { | ||
| 194 | + 0% { | ||
| 195 | + bottom: 0; | ||
| 196 | + margin-left: -30px; | ||
| 197 | + width: 60px; | ||
| 198 | + height: 75px; | ||
| 199 | + background: rgba(20, 20, 20, .1); | ||
| 200 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .1); | ||
| 201 | + border-radius: 30px / 40px; | ||
| 202 | + -moz-animation-timing-function: ease-in; | ||
| 203 | + } | ||
| 204 | + 100% { | ||
| 205 | + bottom: 30px; | ||
| 206 | + margin-left: -10px; | ||
| 207 | + width: 20px; | ||
| 208 | + height: 5px; | ||
| 209 | + background: rgba(20, 20, 20, .3); | ||
| 210 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .3); | ||
| 211 | + border-radius: 20px / 20px; | ||
| 212 | + -moz-animation-timing-function: ease-out; | ||
| 213 | + } | ||
| 214 | +} | ||
| 215 | + | ||
| 216 | +@keyframes pace-compress { | ||
| 217 | + 0% { | ||
| 218 | + bottom: 0; | ||
| 219 | + margin-left: -30px; | ||
| 220 | + width: 60px; | ||
| 221 | + height: 75px; | ||
| 222 | + background: rgba(20, 20, 20, .1); | ||
| 223 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .1); | ||
| 224 | + border-radius: 30px / 40px; | ||
| 225 | + animation-timing-function: ease-in; | ||
| 226 | + } | ||
| 227 | + 100% { | ||
| 228 | + bottom: 30px; | ||
| 229 | + margin-left: -10px; | ||
| 230 | + width: 20px; | ||
| 231 | + height: 5px; | ||
| 232 | + background: rgba(20, 20, 20, .3); | ||
| 233 | + box-shadow: 0 0 20px 35px rgba(20, 20, 20, .3); | ||
| 234 | + border-radius: 20px / 20px; | ||
| 235 | + animation-timing-function: ease-out; | ||
| 236 | + } | ||
| 237 | +} | ||
| 0 | \ No newline at end of file | 238 | \ No newline at end of file |
src/main/resources/static/real_control_v2/fragments/geo/calc_station_space.html
0 → 100644
| 1 | +<!-- 嵌入下抽屉 --> | ||
| 2 | +<div id="cal_station_space_drawer" style="width: 100%;height: 100%;"> | ||
| 3 | + | ||
| 4 | + <div style="text-align: center;padding-top: 30px;" class="calcBeforePanel"> | ||
| 5 | + <p style="margin: 9px 0 9px 0;">当前站点间里程数据取自 <a target="_blank" id="stationRouteLink">基础信息->线路信息->站点详细</a> | ||
| 6 | + 你可以手动编辑。</p> | ||
| 7 | + <p style="margin: 9px 0 9px 0;">此外,我们提供另一种选择,根据路段和站点空间坐标计算站间距。计算后的数据将另存一份副本,不影响原数据。 | ||
| 8 | + </p> | ||
| 9 | + <p style="margin: 0px 0 9px 0;"> | ||
| 10 | + <span style="color: grey;font-size: 12px;">该操作会借用本地计算能力进行,如果你同意,请点击下方按钮!</span> | ||
| 11 | + </p> | ||
| 12 | + <p style="margin: 9px 0 9px 0;"> | ||
| 13 | + <button class="uk-button uk-button-large uk-button-primary" type="button" id="startCalcBtn">开始计算</button> | ||
| 14 | + </p> | ||
| 15 | + </div> | ||
| 16 | + | ||
| 17 | + <div class="pace pace-active" style="display: none;"> | ||
| 18 | + <div class="pace-progress" data-progress="50" data-progress-text="50%" | ||
| 19 | + style="-webkit-transform: translate3d(50%, 0px, 0px); -ms-transform: translate3d(50%, 0px, 0px); transform: translate3d(50%, 0px, 0px);"> | ||
| 20 | + <div class="pace-progress-inner"></div> | ||
| 21 | + </div> | ||
| 22 | + <div class="pace-activity"></div> | ||
| 23 | + <span class="text"></span> | ||
| 24 | + </div> | ||
| 25 | + | ||
| 26 | + <script> | ||
| 27 | + | ||
| 28 | + (function () { | ||
| 29 | + var drawer = '#cal_station_space_drawer' | ||
| 30 | + , sch, line; | ||
| 31 | + | ||
| 32 | + $(drawer).on('drawer-init', function (e, data) { | ||
| 33 | + sch = data.sch; | ||
| 34 | + line = gb_data_basic.codeToLine[sch.xlBm]; | ||
| 35 | + //线路路由信息调整链接 | ||
| 36 | + $('#stationRouteLink', drawer).attr('href', '/pages/base/stationroute/list.html?no=' + line.id); | ||
| 37 | + | ||
| 38 | + $('#startCalcBtn', drawer).on('click', startCalc); | ||
| 39 | + }); | ||
| 40 | + | ||
| 41 | + | ||
| 42 | + function startCalc() { | ||
| 43 | + $(this).attr('disabled', 'disabled'); | ||
| 44 | + | ||
| 45 | + gb_common.$get('/realMap/findRouteAndStationByLine', {lineCode: line.lineCode}, function (spatialData) { | ||
| 46 | + $('.calcBeforePanel', drawer).hide(); | ||
| 47 | + $('.pace', drawer).show(); | ||
| 48 | + | ||
| 49 | + startGeoCalc(spatialData); | ||
| 50 | + }); | ||
| 51 | + } | ||
| 52 | + | ||
| 53 | + function startGeoCalc(spatialData) { | ||
| 54 | + setCalcText('初始化数据'); | ||
| 55 | + //console.log('spatialData', spatialData); | ||
| 56 | + //路段上下行分组排序 | ||
| 57 | + var sections = gb_common.groupBy(spatialData.section, 'DIRECTIONS'); | ||
| 58 | + sections[0].sort(sectionArrSort); | ||
| 59 | + sections[1].sort(sectionArrSort); | ||
| 60 | + | ||
| 61 | + //站点上下行分组排序 | ||
| 62 | + var stations = gb_common.groupBy(spatialData.station, 'DIRECTIONS'); | ||
| 63 | + stations[0].sort(stationArrSort); | ||
| 64 | + stations[1].sort(stationArrSort); | ||
| 65 | + | ||
| 66 | + setCalcText('拼接路段,计算重叠区域'); | ||
| 67 | + console.log('路段', sections); | ||
| 68 | + console.log('站点', stations); | ||
| 69 | + | ||
| 70 | + //完整的线路路段(去掉重叠区域) | ||
| 71 | + var fullSections = [ | ||
| 72 | + connectSections(sections[0]), | ||
| 73 | + connectSections(sections[1]) | ||
| 74 | + ]; | ||
| 75 | + | ||
| 76 | + console.log('fullSections', fullSections); | ||
| 77 | + setCalcText('计算上行站距'); | ||
| 78 | + calcStationSpace(stations[0], fullSections[0]); | ||
| 79 | + /*$.each(stations[0], function (i) { | ||
| 80 | + if(i==0) | ||
| 81 | + return true; | ||
| 82 | + | ||
| 83 | + calcStationSpace(fullSections[0], this); | ||
| 84 | + });*/ | ||
| 85 | + //fullSections[0] | ||
| 86 | + } | ||
| 87 | + | ||
| 88 | + function sectionArrSort(s1, s2) { | ||
| 89 | + return s1.SECTIONROUTE_CODE - s2.SECTIONROUTE_CODE; | ||
| 90 | + } | ||
| 91 | + | ||
| 92 | + function stationArrSort(s1, s2) { | ||
| 93 | + return s1.STATION_ROUTE_CODE - s2.STATION_ROUTE_CODE; | ||
| 94 | + } | ||
| 95 | + | ||
| 96 | + function setCalcText(t) { | ||
| 97 | + $('.pace .text', drawer).text(t); | ||
| 98 | + } | ||
| 99 | + | ||
| 100 | + //连接路段数组 | ||
| 101 | + function connectSections(sectionArray) { | ||
| 102 | + var coords = []; | ||
| 103 | + | ||
| 104 | + var vs, temps; | ||
| 105 | + for (var i = 0, section; section = sectionArray[i++];) { | ||
| 106 | + vs = section.GSECTION_VECTOR; | ||
| 107 | + temps = vs.substr(11, vs.length - 2).split(','); | ||
| 108 | + | ||
| 109 | + connectCoords(coords, temps); | ||
| 110 | + } | ||
| 111 | + return coords; | ||
| 112 | + } | ||
| 113 | + | ||
| 114 | + function connectCoords(all, subArr) { | ||
| 115 | + | ||
| 116 | + var point,ts,len=all.length, inLine, sIndex=0; | ||
| 117 | + if(len > 0){ | ||
| 118 | + $.each(subArr, function (i, p) { | ||
| 119 | + ts = p.split(' '); | ||
| 120 | + point={latitude: parseFloat(ts[1]), longitude: parseFloat(ts[0])}; | ||
| 121 | + //找到起始点 | ||
| 122 | + sIndex=0; | ||
| 123 | + for(var j = 1; j < len; j++){ | ||
| 124 | + inLine=geolib.isPointInLine(point, all[j - 1], all[j]); | ||
| 125 | + | ||
| 126 | + if(inLine) | ||
| 127 | + break; | ||
| 128 | + else | ||
| 129 | + sIndex = i; | ||
| 130 | + } | ||
| 131 | + }); | ||
| 132 | + } | ||
| 133 | + | ||
| 134 | + //拼接 | ||
| 135 | + for(var i=sIndex; i<subArr.length; i++){ | ||
| 136 | + ts = subArr[i].split(' '); | ||
| 137 | + all.push({ | ||
| 138 | + latitude: parseFloat(ts[1]), longitude: parseFloat(ts[0]) | ||
| 139 | + }); | ||
| 140 | + } | ||
| 141 | + } | ||
| 142 | + | ||
| 143 | + //计算站点间距 | ||
| 144 | + function calcStationSpace(stations, coords) { | ||
| 145 | + | ||
| 146 | + | ||
| 147 | + var point,pSection; | ||
| 148 | + for(var i=1; i<stations.length; i++){ | ||
| 149 | + | ||
| 150 | + point = {latitude: stations[i].G_LATY, longitude: stations[i].G_LONX}; | ||
| 151 | + //最近的路段点 | ||
| 152 | + //near=calcPointToArrayNear(point, coords); | ||
| 153 | + //计算前后点位,得到站点在路段中的位置 | ||
| 154 | + //console.log('站点' + i + '最近索引', near, stations[i]); | ||
| 155 | + //pSection = calcPointInSection(point, coords); | ||
| 156 | + //console.log('站点' + i + '包含在路段'+pSection, stations[i]); | ||
| 157 | + } | ||
| 158 | + } | ||
| 159 | + | ||
| 160 | +/* //点在路段上, 返回路段 | ||
| 161 | + function calcPointInSection(point, array) { | ||
| 162 | + /!*$.each(array, function (i) { | ||
| 163 | + | ||
| 164 | + });*!/ | ||
| 165 | + var index=-1; | ||
| 166 | + for(var i=1; i<array.length; i++){ | ||
| 167 | + console.log(point, array[i - 1], array[i], geolib.getDistanceSimple(point, array[i - 1])); | ||
| 168 | + if(geolib.isPointInLine(point, array[i-1], array[i])){ | ||
| 169 | + index=i; | ||
| 170 | + break; | ||
| 171 | + } | ||
| 172 | + } | ||
| 173 | + return index; | ||
| 174 | + }*/ | ||
| 175 | + | ||
| 176 | + //计算点 与 点数组中最近的索引 | ||
| 177 | + /*function calcPointToArrayNear(point, array) { | ||
| 178 | + //var rs, distance, temp; | ||
| 179 | + var distances=[]; | ||
| 180 | + $.each(array, function (i) { | ||
| 181 | + distances.push({ | ||
| 182 | + index: i, | ||
| 183 | + d: geolib.getDistanceSimple(this, point) | ||
| 184 | + }); | ||
| 185 | + }); | ||
| 186 | + | ||
| 187 | + var rs=distances[0]; | ||
| 188 | + for(var i=1;i<distances.length;i++){ | ||
| 189 | + if(rs.d > distances[i].d) | ||
| 190 | + rs=distances[i]; | ||
| 191 | + } | ||
| 192 | + | ||
| 193 | + return rs; | ||
| 194 | + }*/ | ||
| 195 | + })(); | ||
| 196 | + </script> | ||
| 197 | +</div> | ||
| 0 | \ No newline at end of file | 198 | \ No newline at end of file |
src/main/resources/static/real_control_v2/fragments/line_schedule/context_menu/lj_zrw.html
| @@ -75,7 +75,7 @@ | @@ -75,7 +75,7 @@ | ||
| 75 | 75 | ||
| 76 | <script id="schedule-main-table-temp" type="text/html"> | 76 | <script id="schedule-main-table-temp" type="text/html"> |
| 77 | {{each list as sch i}} | 77 | {{each list as sch i}} |
| 78 | - <dl data-id={{sch.id}}> | 78 | + <dl data-id={{sch.id}} data-line-code="{{sch.xlBm}}"> |
| 79 | <dd>{{i+1}}</dd> | 79 | <dd>{{i+1}}</dd> |
| 80 | <dd> | 80 | <dd> |
| 81 | {{if sch.fcsjActual==null}} | 81 | {{if sch.fcsjActual==null}} |
| @@ -200,7 +200,7 @@ | @@ -200,7 +200,7 @@ | ||
| 200 | 200 | ||
| 201 | function getActiveSch() { | 201 | function getActiveSch() { |
| 202 | var dl = $('.main-schedule-table dl.active', modal) | 202 | var dl = $('.main-schedule-table dl.active', modal) |
| 203 | - , sch, lineCode = $('[name=lineSelect]', f).val(); | 203 | + , sch, lineCode = dl.data('line-code');//$('[name=lineSelect]', f).val(); |
| 204 | if (dl.length > 0) | 204 | if (dl.length > 0) |
| 205 | sch = gb_schedule_table.findScheduleByLine(lineCode)[dl.data('id')]; | 205 | sch = gb_schedule_table.findScheduleByLine(lineCode)[dl.data('id')]; |
| 206 | 206 | ||
| @@ -285,7 +285,6 @@ | @@ -285,7 +285,6 @@ | ||
| 285 | if (activeDl.length == 0) | 285 | if (activeDl.length == 0) |
| 286 | return; | 286 | return; |
| 287 | var dds = activeDl.find('dd'); | 287 | var dds = activeDl.find('dd'); |
| 288 | - console.log(dds); | ||
| 289 | var id = activeDl.data('id') | 288 | var id = activeDl.data('id') |
| 290 | , str = '<h3>确定要删除子任务<span style="color:red;margin: 0 5px;">' + $(dds[4]).text() + ',' + $(dds[6]).text() + ' </span>?</h3>'; | 289 | , str = '<h3>确定要删除子任务<span style="color:red;margin: 0 5px;">' + $(dds[4]).text() + ',' + $(dds[6]).text() + ' </span>?</h3>'; |
| 291 | alt_confirm(str, function () { | 290 | alt_confirm(str, function () { |
src/main/resources/static/real_control_v2/fragments/line_schedule/context_menu/sub_task/add_sub_task_inpark.html
| 1 | <div class="uk-modal" id="add-sub-task-inpark-modal"> | 1 | <div class="uk-modal" id="add-sub-task-inpark-modal"> |
| 2 | - <div class="uk-modal-dialog" style="width: 935px;"> | 2 | + <div class="uk-modal-dialog"> |
| 3 | <a href="" class="uk-modal-close uk-close"></a> | 3 | <a href="" class="uk-modal-close uk-close"></a> |
| 4 | <div class="uk-modal-header"> | 4 | <div class="uk-modal-header"> |
| 5 | <h2>回场子任务</h2></div> | 5 | <h2>回场子任务</h2></div> |
| 6 | 6 | ||
| 7 | - <div class="uk-grid tm-grid-truncate" data-uk-grid-margin=""> | ||
| 8 | - <div class="uk-width-medium-1-3 uk-row-first"> | 7 | + <div style="width: 100%;padding-left: 1%;"> |
| 8 | + <div class="sub-task-card"> | ||
| 9 | <div class="uk-panel uk-panel-box uk-panel-box-primary"> | 9 | <div class="uk-panel uk-panel-box uk-panel-box-primary"> |
| 10 | <div class="uk-panel-badge uk-badge">营运</div> | 10 | <div class="uk-panel-badge uk-badge">营运</div> |
| 11 | <h3 class="uk-panel-title">线路上站点间</h3> | 11 | <h3 class="uk-panel-title">线路上站点间</h3> |
| 12 | - <form class="uk-form uk-form-horizontal"> | 12 | + <form class="uk-form uk-form-horizontal" service_form> |
| 13 | <div class="uk-grid"> | 13 | <div class="uk-grid"> |
| 14 | <div class="uk-width-3-10"> | 14 | <div class="uk-width-3-10"> |
| 15 | <div class="uk-form-row"> | 15 | <div class="uk-form-row"> |
| @@ -17,7 +17,7 @@ | @@ -17,7 +17,7 @@ | ||
| 17 | </div> | 17 | </div> |
| 18 | </div> | 18 | </div> |
| 19 | <div class="uk-width-7-10 pl5"> | 19 | <div class="uk-width-7-10 pl5"> |
| 20 | - <select name="startStation" required></select> | 20 | + <select name="startStation" disabled></select> |
| 21 | </div> | 21 | </div> |
| 22 | </div> | 22 | </div> |
| 23 | <div class="uk-grid"> | 23 | <div class="uk-grid"> |
| @@ -27,7 +27,7 @@ | @@ -27,7 +27,7 @@ | ||
| 27 | </div> | 27 | </div> |
| 28 | </div> | 28 | </div> |
| 29 | <div class="uk-width-7-10 pl5"> | 29 | <div class="uk-width-7-10 pl5"> |
| 30 | - <select name="endStation" required></select> | 30 | + <select name="endStation" class="ct_focus"></select> |
| 31 | </div> | 31 | </div> |
| 32 | </div> | 32 | </div> |
| 33 | <div class="uk-grid"> | 33 | <div class="uk-grid"> |
| @@ -37,38 +37,183 @@ | @@ -37,38 +37,183 @@ | ||
| 37 | </div> | 37 | </div> |
| 38 | </div> | 38 | </div> |
| 39 | <div class="uk-width-7-10 pl5"> | 39 | <div class="uk-width-7-10 pl5"> |
| 40 | - <select name="endStation" required></select> | 40 | + <input type="text" name="mileageType" required> |
| 41 | + </div> | ||
| 42 | + </div> | ||
| 43 | + <div class="uk-grid"> | ||
| 44 | + <div class="uk-width-3-10"> | ||
| 45 | + <div class="uk-form-row"> | ||
| 46 | + <label class="uk-form-label">开始时间</label> | ||
| 47 | + </div> | ||
| 48 | + </div> | ||
| 49 | + <div class="uk-width-7-10 pl5"> | ||
| 50 | + <input type="time" name="startDate" required> | ||
| 51 | + </div> | ||
| 52 | + </div> | ||
| 53 | + <div class="uk-grid"> | ||
| 54 | + <div class="uk-width-3-10"> | ||
| 55 | + <div class="uk-form-row"> | ||
| 56 | + <label class="uk-form-label">结束时间</label> | ||
| 57 | + </div> | ||
| 58 | + </div> | ||
| 59 | + <div class="uk-width-7-10 pl5"> | ||
| 60 | + <input type="time" name="endDate" required> | ||
| 41 | </div> | 61 | </div> |
| 42 | </div> | 62 | </div> |
| 43 | </form> | 63 | </form> |
| 44 | </div> | 64 | </div> |
| 45 | </div> | 65 | </div> |
| 46 | - <div class="uk-width-medium-1-3"> | 66 | + <div class="sub-task-card"> |
| 47 | <div class="uk-panel uk-panel-box uk-panel-box-secondary"> | 67 | <div class="uk-panel uk-panel-box uk-panel-box-secondary"> |
| 48 | <div class="uk-panel-badge uk-badge uk-badge-default">空驶</div> | 68 | <div class="uk-panel-badge uk-badge uk-badge-default">空驶</div> |
| 49 | <h3 class="uk-panel-title">进场</h3> | 69 | <h3 class="uk-panel-title">进场</h3> |
| 50 | - Lorem ipsum dolor sit amet, consectetur adipisicing elit. <code>.uk-panel-box-primary</code> | 70 | + <form class="uk-form uk-form-horizontal" empty_form> |
| 71 | + <div class="uk-grid"> | ||
| 72 | + <div class="uk-width-3-10"> | ||
| 73 | + <div class="uk-form-row"> | ||
| 74 | + <label class="uk-form-label">起点 </label> | ||
| 75 | + </div> | ||
| 76 | + </div> | ||
| 77 | + <div class="uk-width-7-10 pl5"> | ||
| 78 | + <select name="startStation" disabled></select> | ||
| 79 | + </div> | ||
| 80 | + </div> | ||
| 81 | + <div class="uk-grid"> | ||
| 82 | + <div class="uk-width-3-10"> | ||
| 83 | + <div class="uk-form-row"> | ||
| 84 | + <label class="uk-form-label">终点 </label> | ||
| 85 | + </div> | ||
| 86 | + </div> | ||
| 87 | + <div class="uk-width-7-10 pl5"> | ||
| 88 | + <select name="endStation" class="ct_focus"></select> | ||
| 89 | + </div> | ||
| 90 | + </div> | ||
| 91 | + <div class="uk-grid"> | ||
| 92 | + <div class="uk-width-3-10"> | ||
| 93 | + <div class="uk-form-row"> | ||
| 94 | + <label class="uk-form-label">空驶里程</label> | ||
| 95 | + </div> | ||
| 96 | + </div> | ||
| 97 | + <div class="uk-width-7-10 pl5"> | ||
| 98 | + <input type="text" name="mileageType" required style="width: calc(100% - 62px);"> | ||
| 99 | + <a style="font-size: 12px;">路径(3)</a> | ||
| 100 | + </div> | ||
| 101 | + </div> | ||
| 102 | + <div class="uk-grid"> | ||
| 103 | + <div class="uk-width-3-10"> | ||
| 104 | + <div class="uk-form-row"> | ||
| 105 | + <label class="uk-form-label">开始时间</label> | ||
| 106 | + </div> | ||
| 107 | + </div> | ||
| 108 | + <div class="uk-width-7-10 pl5"> | ||
| 109 | + <input type="time" name="startDate" required> | ||
| 110 | + </div> | ||
| 111 | + </div> | ||
| 112 | + <div class="uk-grid"> | ||
| 113 | + <div class="uk-width-3-10"> | ||
| 114 | + <div class="uk-form-row"> | ||
| 115 | + <label class="uk-form-label">结束时间</label> | ||
| 116 | + </div> | ||
| 117 | + </div> | ||
| 118 | + <div class="uk-width-7-10 pl5"> | ||
| 119 | + <input type="time" name="endDate" required> | ||
| 120 | + </div> | ||
| 121 | + </div> | ||
| 122 | + </form> | ||
| 51 | </div> | 123 | </div> |
| 52 | </div> | 124 | </div> |
| 53 | - <div class="uk-width-medium-1-3"> | 125 | + <div class="sub-task-card destroy-card uk-animation-scale"> |
| 54 | <div class="uk-panel uk-panel-box uk-panel-box-danger"> | 126 | <div class="uk-panel uk-panel-box uk-panel-box-danger"> |
| 55 | <div class="uk-panel-badge uk-badge uk-badge-danger">烂班</div> | 127 | <div class="uk-panel-badge uk-badge uk-badge-danger">烂班</div> |
| 56 | <h3 class="uk-panel-title">线路上站点间</h3> | 128 | <h3 class="uk-panel-title">线路上站点间</h3> |
| 57 | - Lorem ipsum dolor sit amet, consectetur adipisicing elit. <code>.uk-panel-box-primary</code> | 129 | + <form class="uk-form uk-form-horizontal" destroy_form> |
| 130 | + <div class="uk-grid"> | ||
| 131 | + <div class="uk-width-3-10"> | ||
| 132 | + <div class="uk-form-row"> | ||
| 133 | + <label class="uk-form-label">起点 </label> | ||
| 134 | + </div> | ||
| 135 | + </div> | ||
| 136 | + <div class="uk-width-7-10 pl5"> | ||
| 137 | + <select name="startStation" disabled></select> | ||
| 138 | + </div> | ||
| 139 | + </div> | ||
| 140 | + <div class="uk-grid"> | ||
| 141 | + <div class="uk-width-3-10"> | ||
| 142 | + <div class="uk-form-row"> | ||
| 143 | + <label class="uk-form-label">终点 </label> | ||
| 144 | + </div> | ||
| 145 | + </div> | ||
| 146 | + <div class="uk-width-7-10 pl5"> | ||
| 147 | + <select name="endStation" class="ct_focus" disabled></select> | ||
| 148 | + </div> | ||
| 149 | + </div> | ||
| 150 | + <div class="uk-grid"> | ||
| 151 | + <div class="uk-width-3-10"> | ||
| 152 | + <div class="uk-form-row"> | ||
| 153 | + <label class="uk-form-label">烂班里程</label> | ||
| 154 | + </div> | ||
| 155 | + </div> | ||
| 156 | + <div class="uk-width-7-10 pl5"> | ||
| 157 | + <input type="text" name="mileageType" required> | ||
| 158 | + </div> | ||
| 159 | + </div> | ||
| 160 | + <div class="uk-grid"> | ||
| 161 | + <div class="uk-width-3-10"> | ||
| 162 | + <div class="uk-form-row"> | ||
| 163 | + <label class="uk-form-label">开始时间</label> | ||
| 164 | + </div> | ||
| 165 | + </div> | ||
| 166 | + <div class="uk-width-7-10 pl5"> | ||
| 167 | + <input type="time" name="startDate" required> | ||
| 168 | + </div> | ||
| 169 | + </div> | ||
| 170 | + <div class="uk-grid"> | ||
| 171 | + <div class="uk-width-3-10"> | ||
| 172 | + <div class="uk-form-row"> | ||
| 173 | + <label class="uk-form-label">烂班原因</label> | ||
| 174 | + </div> | ||
| 175 | + </div> | ||
| 176 | + <div class="uk-width-7-10 pl5"> | ||
| 177 | + <select name="destroyReason" required></select> | ||
| 178 | + </div> | ||
| 179 | + </div> | ||
| 180 | + </form> | ||
| 58 | </div> | 181 | </div> |
| 59 | </div> | 182 | </div> |
| 183 | + | ||
| 184 | + <form class="uk-form" style="margin-top: 15px; padding: 0 10px 0 0;"> | ||
| 185 | + <textarea placeholder="备注" style="width: 100%;height: 70px;"></textarea> | ||
| 186 | + </form> | ||
| 187 | + </div> | ||
| 188 | + | ||
| 189 | + <div class="uk-modal-footer uk-text-right"> | ||
| 190 | + <button type="button" class="uk-button uk-modal-close">取消</button> | ||
| 191 | + <button type="button" class="uk-button uk-button-primary">提交子任务</button> | ||
| 192 | + | ||
| 193 | + <div class="ct-footer-left"> | ||
| 194 | + <a id="betweenStationRangeCalc" data-drawer-id="cal_station_space_drawer">站点间公里不准?</a> | ||
| 195 | + </div> | ||
| 60 | </div> | 196 | </div> |
| 61 | </div> | 197 | </div> |
| 62 | 198 | ||
| 199 | + <div class="ct-bottom-drawer"> | ||
| 200 | + <a class="ct-bottom-drawer-close"></a> | ||
| 201 | + | ||
| 202 | + <div class="ct-bottom-drawer-body"></div> | ||
| 203 | + </div> | ||
| 204 | + | ||
| 63 | <script id="sub-task-inpark-form-temp" type="text/html"> | 205 | <script id="sub-task-inpark-form-temp" type="text/html"> |
| 64 | </script> | 206 | </script> |
| 65 | 207 | ||
| 66 | <script> | 208 | <script> |
| 67 | - (function() { | 209 | + (function () { |
| 68 | var modal = '#add-sub-task-inpark-modal', | 210 | var modal = '#add-sub-task-inpark-modal', |
| 69 | - sch, stationRoutes, parks, information; | 211 | + serviceForm = $('form[service_form]', modal), |
| 212 | + emptyForm = $('form[empty_form]', modal), | ||
| 213 | + destroyForm = $('form[destroy_form]', modal), | ||
| 214 | + sch, stationRoutes, parks, information, esCode; | ||
| 70 | var adjustExps = ['配车', '保养', '故障', '肇事', '路阻', '纠纷', '缺人', '客稀', '缺车', '气候', '援外', '吊慢', '抽减', '其他']; | 215 | var adjustExps = ['配车', '保养', '故障', '肇事', '路阻', '纠纷', '缺人', '客稀', '缺车', '气候', '援外', '吊慢', '抽减', '其他']; |
| 71 | - $(modal).on('init', function(e, data) { | 216 | + $(modal).on('init', function (e, data) { |
| 72 | sch = data.sch; | 217 | sch = data.sch; |
| 73 | //var formHtml = template('sub-task-inpark-form-temp', {sch:sch, adjustExps: adjustExps}); | 218 | //var formHtml = template('sub-task-inpark-form-temp', {sch:sch, adjustExps: adjustExps}); |
| 74 | //$('form', modal).html(formHtml); | 219 | //$('form', modal).html(formHtml); |
| @@ -76,31 +221,112 @@ | @@ -76,31 +221,112 @@ | ||
| 76 | //dictionaryUtils.transformDom($('.nt-dictionary', modal)); | 221 | //dictionaryUtils.transformDom($('.nt-dictionary', modal)); |
| 77 | 222 | ||
| 78 | //站点路由 | 223 | //站点路由 |
| 79 | - stationRoutes = gb_common.groupBy(gb_data_basic.stationRoutes(sch.xlBm).sort(function(a, b){ | ||
| 80 | - return a.stationRouteCode-b.stationRouteCode; | ||
| 81 | - }), 'directions'); | ||
| 82 | - //停车场 | ||
| 83 | - $.get('/basic/parks', function(rs){ | ||
| 84 | - parks=rs; | 224 | + stationRoutes = gb_common.groupBy(gb_data_basic.stationRoutes(sch.xlBm).sort(function (a, b) { |
| 225 | + return a.stationRouteCode - b.stationRouteCode; | ||
| 226 | + }), 'directions')[sch.xlDir]; | ||
| 227 | + //空驶终点(停车场) | ||
| 228 | + $.get('/basic/parks', function (rs) { | ||
| 229 | + parks = rs; | ||
| 230 | + var opts = ''; | ||
| 231 | + for (var code in parks) { | ||
| 232 | + opts += '<option value="' + code + '">' + parks[code] + '</option>'; | ||
| 233 | + } | ||
| 234 | + $('select[name=endStation]', emptyForm).html(opts).val(information.carPark); | ||
| 85 | }); | 235 | }); |
| 86 | //线路标准 | 236 | //线路标准 |
| 87 | - information=gb_data_basic.getLineInformation(sch.xlBm); | 237 | + information = gb_data_basic.getLineInformation(sch.xlBm); |
| 238 | + | ||
| 239 | + var routeOpts = ''; | ||
| 240 | + $.each(stationRoutes, function () { | ||
| 241 | + routeOpts += '<option value="' + this.stationCode + '">' + this.stationName + '</option>'; | ||
| 242 | + }); | ||
| 243 | + esCode = stationRoutes[stationRoutes.length - 1].stationCode; | ||
| 244 | + | ||
| 245 | + //营运起终点 | ||
| 246 | + $('select[name=startStation],select[name=endStation]', serviceForm).html(routeOpts); | ||
| 247 | + //空驶起点 | ||
| 248 | + $('select[name=startStation]', emptyForm).html(routeOpts).val(esCode); | ||
| 249 | + //烂班起终点 | ||
| 250 | + $('select[name=startStation],select[name=endStation]', destroyForm).html(routeOpts).val(esCode); | ||
| 251 | + | ||
| 252 | + //烂班原因 | ||
| 253 | + var adjustExpsOpts = '<option value="">请选择...</option>'; | ||
| 254 | + $.each(adjustExps, function (i, str) { | ||
| 255 | + adjustExpsOpts += '<option value="' + str + '">' + str + '</option>'; | ||
| 256 | + }); | ||
| 257 | + $('select[name=destroyReason]', destroyForm).html(adjustExpsOpts); | ||
| 258 | + | ||
| 259 | + //切换营运终点 | ||
| 260 | + $('select[name=endStation]', serviceForm).val(esCode).on('change', function () { | ||
| 261 | + var val = $(this).val(); | ||
| 262 | + if (val == esCode) { | ||
| 263 | + $('.uk-modal-dialog', modal).removeClass('three-children'); | ||
| 264 | + $('.sub-task-card.destroy-card').css("display", "none"); | ||
| 265 | + } | ||
| 266 | + else { | ||
| 267 | + $('.uk-modal-dialog', modal).addClass('three-children'); | ||
| 268 | + $('.sub-task-card.destroy-card').css("display", "inline-block"); | ||
| 269 | + } | ||
| 270 | + | ||
| 271 | + reClac(); | ||
| 272 | + }); | ||
| 273 | + | ||
| 88 | 274 | ||
| 89 | //submit | 275 | //submit |
| 90 | - var f = $('form', modal).formValidation(gb_form_validation_opts); | ||
| 91 | - f.on('success.form.fv', function(e) { | ||
| 92 | - disabled_submit_btn(this); | ||
| 93 | - e.preventDefault(); | ||
| 94 | - var data = $(this).serializeJSON(); | ||
| 95 | -/* | ||
| 96 | - gb_common.$post('/childTask', data, function(rs){ | ||
| 97 | - notify_succ('子任务添加成功'); | ||
| 98 | - gb_schedule_table.updateSchedule(rs.t); | ||
| 99 | - $('#schedule-lj_zrw-modal .main-schedule-table').trigger('refresh', {sch: rs.t}); | ||
| 100 | - UIkit.modal(modal).hide(); | ||
| 101 | - });*/ | 276 | + /*var f = $('form', modal).formValidation({ |
| 277 | + framework: 'uikit', | ||
| 278 | + locale: 'zh_CN', | ||
| 279 | + }); | ||
| 280 | + f.on('success.form.fv', function(e) { | ||
| 281 | + disabled_submit_btn(this); | ||
| 282 | + e.preventDefault(); | ||
| 283 | + var data = $(this).serializeJSON(); | ||
| 284 | + | ||
| 285 | + });*/ | ||
| 286 | + }); | ||
| 287 | + | ||
| 288 | + function reClac() { | ||
| 289 | + var serviceEnd = $('select[name=endStation]', serviceForm).val(); | ||
| 290 | + //空驶起点 == 营运终点 | ||
| 291 | + $('select[name=startStation]', emptyForm).val(serviceEnd); | ||
| 292 | + | ||
| 293 | + if (serviceEnd != esCode) { | ||
| 294 | + //烂班起点 == 营运终点 | ||
| 295 | + $('select[name=startStation]', destroyForm).val(serviceEnd); | ||
| 296 | + } | ||
| 297 | + } | ||
| 298 | + | ||
| 299 | + | ||
| 300 | + //------ 下抽屉 ------ | ||
| 301 | + $('#betweenStationRangeCalc', modal).on('click', function () { | ||
| 302 | + var id = $(this).data('drawer-id'); | ||
| 303 | + switchBtmDrawer(id, '/real_control_v2/fragments/geo/calc_station_space.html'); | ||
| 304 | + }); | ||
| 305 | + | ||
| 306 | + var btmDrawer = $('.ct-bottom-drawer', modal); | ||
| 307 | + //打开事件 | ||
| 308 | + btmDrawer.on('drawer_show', function () { | ||
| 309 | + var url = $(this).data('url'); | ||
| 310 | + if(!url){ | ||
| 311 | + alert('无效的地址'); | ||
| 312 | + } | ||
| 313 | + | ||
| 314 | + var drawerPanel=$(this).data('name'); | ||
| 315 | + $('.ct-bottom-drawer-body', btmDrawer).load(url, function () { | ||
| 316 | + $('#'+drawerPanel).trigger('drawer-init', {sch: sch}); | ||
| 102 | }); | 317 | }); |
| 103 | }); | 318 | }); |
| 319 | + | ||
| 320 | + function switchBtmDrawer(id, url) { | ||
| 321 | + if (btmDrawer.hasClass('open') && btmDrawer.data('name') == id) { | ||
| 322 | + btmDrawer.removeClass('open'); | ||
| 323 | + btmDrawer.removeData('name').removeData('url'); | ||
| 324 | + } | ||
| 325 | + else { | ||
| 326 | + btmDrawer.addClass('open'); | ||
| 327 | + btmDrawer.data('name', id).data('url', url).trigger('drawer_show'); | ||
| 328 | + } | ||
| 329 | + } | ||
| 104 | })(); | 330 | })(); |
| 105 | </script> | 331 | </script> |
| 106 | </div> | 332 | </div> |
src/main/resources/static/real_control_v2/geolib/geolib.elevation.js
0 → 100644
| 1 | +/*! geolib.elevation 2.0.21 by Manuel Bieh | ||
| 2 | +* | ||
| 3 | +* Elevation Addon for Geolib.js | ||
| 4 | +* | ||
| 5 | +* @author Manuel Bieh | ||
| 6 | +* @url http://www.manuelbieh.com/ | ||
| 7 | +* @version 2.0.21 | ||
| 8 | +* @license MIT | ||
| 9 | +*/ | ||
| 10 | +;(function(global, geolib, undefined) { | ||
| 11 | + | ||
| 12 | + var elevation = { | ||
| 13 | + | ||
| 14 | + /*global google:true geolib:true require:true module:true elevationResult:true */ | ||
| 15 | + | ||
| 16 | + /** | ||
| 17 | + * @param Array Collection of coords [{latitude: 51.510, longitude: 7.1321}, {latitude: 49.1238, longitude: "8° 30' W"}, ...] | ||
| 18 | + * @return Array [{lat:#lat, lng:#lng, elev:#elev},....]} | ||
| 19 | + */ | ||
| 20 | + getElevation: function() { | ||
| 21 | + if (typeof global.navigator !== 'undefined') { | ||
| 22 | + this.getElevationClient.apply(this, arguments); | ||
| 23 | + } else { | ||
| 24 | + this.getElevationServer.apply(this, arguments); | ||
| 25 | + } | ||
| 26 | + }, | ||
| 27 | + | ||
| 28 | + | ||
| 29 | + /* Optional elevation addon requires Googlemaps API JS */ | ||
| 30 | + getElevationClient: function(coords, cb) { | ||
| 31 | + | ||
| 32 | + if (!global.google) { | ||
| 33 | + throw new Error("Google maps api not loaded"); | ||
| 34 | + } | ||
| 35 | + | ||
| 36 | + if (coords.length === 0) { | ||
| 37 | + return cb(null, null); | ||
| 38 | + } | ||
| 39 | + | ||
| 40 | + if (coords.length === 1) { | ||
| 41 | + return cb(new Error("getElevation requires at least 2 points.")); | ||
| 42 | + } | ||
| 43 | + | ||
| 44 | + var path = []; | ||
| 45 | + | ||
| 46 | + for(var i = 0; i < coords.length; i++) { | ||
| 47 | + path.push(new google.maps.LatLng( | ||
| 48 | + this.latitude(coords[i]), | ||
| 49 | + this.longitude(coords[i]) | ||
| 50 | + )); | ||
| 51 | + } | ||
| 52 | + | ||
| 53 | + var positionalRequest = { | ||
| 54 | + 'path': path, | ||
| 55 | + 'samples': path.length | ||
| 56 | + }; | ||
| 57 | + | ||
| 58 | + var elevationService = new google.maps.ElevationService(); | ||
| 59 | + var geolib = this; | ||
| 60 | + | ||
| 61 | + elevationService.getElevationAlongPath(positionalRequest, function (results, status) { | ||
| 62 | + geolib.elevationHandler(results, status, coords, cb); | ||
| 63 | + }); | ||
| 64 | + | ||
| 65 | + }, | ||
| 66 | + | ||
| 67 | + | ||
| 68 | + getElevationServer: function(coords, cb) { | ||
| 69 | + | ||
| 70 | + if (coords.length === 0) { | ||
| 71 | + return cb(null, null); | ||
| 72 | + } | ||
| 73 | + | ||
| 74 | + if (coords.length === 1) { | ||
| 75 | + return cb(new Error("getElevation requires at least 2 points.")); | ||
| 76 | + } | ||
| 77 | + | ||
| 78 | + var gm = require('googlemaps'); | ||
| 79 | + var path = []; | ||
| 80 | + | ||
| 81 | + for(var i = 0; i < coords.length; i++) { | ||
| 82 | + path.push( | ||
| 83 | + this.latitude(coords[i]) + ',' + this.longitude(coords[i]) | ||
| 84 | + ); | ||
| 85 | + } | ||
| 86 | + | ||
| 87 | + var geolib = this; | ||
| 88 | + | ||
| 89 | + gm.elevationFromPath(path.join('|'), path.length, function(err, results) { | ||
| 90 | + geolib.elevationHandler(results.results, results.status, coords, cb); | ||
| 91 | + }); | ||
| 92 | + | ||
| 93 | + }, | ||
| 94 | + | ||
| 95 | + | ||
| 96 | + elevationHandler: function(results, status, coords, cb) { | ||
| 97 | + | ||
| 98 | + var latsLngsElevs = []; | ||
| 99 | + | ||
| 100 | + if (status == "OK" ) { | ||
| 101 | + | ||
| 102 | + for (var i = 0; i < results.length; i++) { | ||
| 103 | + latsLngsElevs.push({ | ||
| 104 | + "lat": this.latitude(coords[i]), | ||
| 105 | + "lng": this.longitude(coords[i]), | ||
| 106 | + "elev":results[i].elevation | ||
| 107 | + }); | ||
| 108 | + } | ||
| 109 | + | ||
| 110 | + cb(null, latsLngsElevs); | ||
| 111 | + | ||
| 112 | + } else { | ||
| 113 | + | ||
| 114 | + cb(new Error("Could not get elevation using Google's API"), elevationResult.status); | ||
| 115 | + | ||
| 116 | + } | ||
| 117 | + | ||
| 118 | + }, | ||
| 119 | + | ||
| 120 | + | ||
| 121 | + /** | ||
| 122 | + * @param Array [{lat:#lat, lng:#lng, elev:#elev},....]} | ||
| 123 | + * @return Number % grade | ||
| 124 | + */ | ||
| 125 | + getGrade: function(coords) { | ||
| 126 | + | ||
| 127 | + var rise = Math.abs( | ||
| 128 | + this.elevation(coords[coords.length-1]) - this.elevation(coords[0]) | ||
| 129 | + ); | ||
| 130 | + | ||
| 131 | + var run = this.getPathLength(coords); | ||
| 132 | + | ||
| 133 | + return Math.floor((rise/run)*100); | ||
| 134 | + | ||
| 135 | + }, | ||
| 136 | + | ||
| 137 | + | ||
| 138 | + /** | ||
| 139 | + * @param Array [{lat:#lat, lng:#lng, elev:#elev},....]} | ||
| 140 | + * @return Object {gain:#gain, loss:#loss} | ||
| 141 | + */ | ||
| 142 | + getTotalElevationGainAndLoss: function(coords) { | ||
| 143 | + | ||
| 144 | + var gain = 0; | ||
| 145 | + var loss = 0; | ||
| 146 | + | ||
| 147 | + for(var i = 0; i < coords.length - 1; i++) { | ||
| 148 | + | ||
| 149 | + var deltaElev = this.elevation(coords[i]) - this.elevation(coords[i + 1]); | ||
| 150 | + | ||
| 151 | + if (deltaElev > 0) { | ||
| 152 | + loss += deltaElev; | ||
| 153 | + } else { | ||
| 154 | + gain += Math.abs(deltaElev); | ||
| 155 | + } | ||
| 156 | + | ||
| 157 | + } | ||
| 158 | + | ||
| 159 | + return { | ||
| 160 | + "gain": gain, | ||
| 161 | + "loss": loss | ||
| 162 | + }; | ||
| 163 | + | ||
| 164 | + } | ||
| 165 | + | ||
| 166 | + }; | ||
| 167 | + | ||
| 168 | + // Node module | ||
| 169 | + if (typeof module !== 'undefined' && | ||
| 170 | + typeof module.exports !== 'undefined') { | ||
| 171 | + | ||
| 172 | + geolib = require('geolib'); | ||
| 173 | + geolib.extend(elevation); | ||
| 174 | + | ||
| 175 | + // AMD module | ||
| 176 | + } else if (typeof define === "function" && define.amd) { | ||
| 177 | + | ||
| 178 | + define(["geolib"], function (geolib) { | ||
| 179 | + geolib.extend(elevation); | ||
| 180 | + return geolib; | ||
| 181 | + }); | ||
| 182 | + | ||
| 183 | + // we're in a browser | ||
| 184 | + } else { | ||
| 185 | + | ||
| 186 | + geolib.extend(elevation); | ||
| 187 | + | ||
| 188 | + } | ||
| 189 | + | ||
| 190 | +}(this, this.geolib)); | ||
| 0 | \ No newline at end of file | 191 | \ No newline at end of file |
src/main/resources/static/real_control_v2/geolib/geolib.elevation.min.js
0 → 100644
| 1 | +/*! geolib.elevation 2.0.21 by Manuel Bieh | ||
| 2 | +* | ||
| 3 | +* Elevation Addon for Geolib.js | ||
| 4 | +* | ||
| 5 | +* @author Manuel Bieh | ||
| 6 | +* @url http://www.manuelbieh.com/ | ||
| 7 | +* @version 2.0.21 | ||
| 8 | +* @license MIT | ||
| 9 | +*/ | ||
| 10 | +!function(a,b,c){var d={getElevation:function(){"undefined"!=typeof a.navigator?this.getElevationClient.apply(this,arguments):this.getElevationServer.apply(this,arguments)},getElevationClient:function(b,c){if(!a.google)throw new Error("Google maps api not loaded");if(0===b.length)return c(null,null);if(1===b.length)return c(new Error("getElevation requires at least 2 points."));for(var d=[],e=0;e<b.length;e++)d.push(new google.maps.LatLng(this.latitude(b[e]),this.longitude(b[e])));var f={path:d,samples:d.length},g=new google.maps.ElevationService,h=this;g.getElevationAlongPath(f,function(a,d){h.elevationHandler(a,d,b,c)})},getElevationServer:function(a,b){if(0===a.length)return b(null,null);if(1===a.length)return b(new Error("getElevation requires at least 2 points."));for(var c=require("googlemaps"),d=[],e=0;e<a.length;e++)d.push(this.latitude(a[e])+","+this.longitude(a[e]));var f=this;c.elevationFromPath(d.join("|"),d.length,function(c,d){f.elevationHandler(d.results,d.status,a,b)})},elevationHandler:function(a,b,c,d){var e=[];if("OK"==b){for(var f=0;f<a.length;f++)e.push({lat:this.latitude(c[f]),lng:this.longitude(c[f]),elev:a[f].elevation});d(null,e)}else d(new Error("Could not get elevation using Google's API"),elevationResult.status)},getGrade:function(a){var b=Math.abs(this.elevation(a[a.length-1])-this.elevation(a[0])),c=this.getPathLength(a);return Math.floor(b/c*100)},getTotalElevationGainAndLoss:function(a){for(var b=0,c=0,d=0;d<a.length-1;d++){var e=this.elevation(a[d])-this.elevation(a[d+1]);e>0?c+=e:b+=Math.abs(e)}return{gain:b,loss:c}}};"undefined"!=typeof module&&"undefined"!=typeof module.exports?(b=require("geolib"),b.extend(d)):"function"==typeof define&&define.amd?define(["geolib"],function(a){return a.extend(d),a}):b.extend(d)}(this,this.geolib); | ||
| 0 | \ No newline at end of file | 11 | \ No newline at end of file |
src/main/resources/static/real_control_v2/geolib/geolib.isPointInsideRobust.js
0 → 100644
| 1 | +/*! geolib.isPointInsideRobust 2.0.21 | ||
| 2 | +* !!EXPERIMENTAL!! | ||
| 3 | +* | ||
| 4 | +* Robust version of isPointInside for Geolib.js | ||
| 5 | +* | ||
| 6 | +* Based on https://github.com/mikolalysenko/robust-point-in-polygon | ||
| 7 | +* by Mikola Lysenko, licensed under MIT | ||
| 8 | +* | ||
| 9 | +* @author Manuel Bieh | ||
| 10 | +* @url http://www.manuelbieh.com/ | ||
| 11 | +* @version 2.0.21 | ||
| 12 | +* @license MIT | ||
| 13 | +* | ||
| 14 | +*/ | ||
| 15 | +;(function(global, geolib, undefined) { | ||
| 16 | + | ||
| 17 | + var addOn = function(geolib) { | ||
| 18 | + | ||
| 19 | + var SPLITTER = +(Math.pow(2, 27) + 1.0); | ||
| 20 | + | ||
| 21 | + var NUM_EXPAND = 5; | ||
| 22 | + var EPSILON = 1.1102230246251565e-16; | ||
| 23 | + var ERRBOUND3 = (3.0 + 16.0 * EPSILON) * EPSILON; | ||
| 24 | + var ERRBOUND4 = (7.0 + 56.0 * EPSILON) * EPSILON; | ||
| 25 | + | ||
| 26 | + var twoProduct = function(a, b, result) { | ||
| 27 | + var x = a * b; | ||
| 28 | + var c = SPLITTER * a; | ||
| 29 | + var abig = c - a; | ||
| 30 | + var ahi = c - abig; | ||
| 31 | + var alo = a - ahi; | ||
| 32 | + var d = SPLITTER * b; | ||
| 33 | + var bbig = d - b; | ||
| 34 | + var bhi = d - bbig; | ||
| 35 | + var blo = b - bhi; | ||
| 36 | + var err1 = x - (ahi * bhi); | ||
| 37 | + var err2 = err1 - (alo * bhi); | ||
| 38 | + var err3 = err2 - (ahi * blo); | ||
| 39 | + var y = alo * blo - err3; | ||
| 40 | + if(result) { | ||
| 41 | + result[0] = y; | ||
| 42 | + result[1] = x; | ||
| 43 | + return result; | ||
| 44 | + } | ||
| 45 | + return [ y, x ]; | ||
| 46 | + }; | ||
| 47 | + | ||
| 48 | + var fastTwoSum = function(a, b, result) { | ||
| 49 | + var x = a + b; | ||
| 50 | + var bv = x - a; | ||
| 51 | + var av = x - bv; | ||
| 52 | + var br = b - bv; | ||
| 53 | + var ar = a - av; | ||
| 54 | + if(result) { | ||
| 55 | + result[0] = ar + br; | ||
| 56 | + result[1] = x; | ||
| 57 | + return result; | ||
| 58 | + } | ||
| 59 | + return [ar+br, x]; | ||
| 60 | + }; | ||
| 61 | + | ||
| 62 | + var twoSum = fastTwoSum; | ||
| 63 | + | ||
| 64 | + var linearExpansionSum = function(e, f) { | ||
| 65 | + var ne = e.length|0; | ||
| 66 | + var nf = f.length|0; | ||
| 67 | + if(ne === 1 && nf === 1) { | ||
| 68 | + return scalarScalar(e[0], f[0]); | ||
| 69 | + } | ||
| 70 | + var n = ne + nf; | ||
| 71 | + var g = new Array(n); | ||
| 72 | + var count = 0; | ||
| 73 | + var eptr = 0; | ||
| 74 | + var fptr = 0; | ||
| 75 | + var abs = Math.abs; | ||
| 76 | + var ei = e[eptr]; | ||
| 77 | + var ea = abs(ei); | ||
| 78 | + var fi = f[fptr]; | ||
| 79 | + var fa = abs(fi); | ||
| 80 | + var a, b; | ||
| 81 | + if(ea < fa) { | ||
| 82 | + b = ei; | ||
| 83 | + eptr += 1; | ||
| 84 | + if(eptr < ne) { | ||
| 85 | + ei = e[eptr]; | ||
| 86 | + ea = abs(ei); | ||
| 87 | + } | ||
| 88 | + } else { | ||
| 89 | + b = fi; | ||
| 90 | + fptr += 1; | ||
| 91 | + if(fptr < nf) { | ||
| 92 | + fi = f[fptr]; | ||
| 93 | + fa = abs(fi); | ||
| 94 | + } | ||
| 95 | + } | ||
| 96 | + if((eptr < ne && ea < fa) || (fptr >= nf)) { | ||
| 97 | + a = ei; | ||
| 98 | + eptr += 1; | ||
| 99 | + if(eptr < ne) { | ||
| 100 | + ei = e[eptr]; | ||
| 101 | + ea = abs(ei); | ||
| 102 | + } | ||
| 103 | + } else { | ||
| 104 | + a = fi; | ||
| 105 | + fptr += 1; | ||
| 106 | + if(fptr < nf) { | ||
| 107 | + fi = f[fptr]; | ||
| 108 | + fa = abs(fi); | ||
| 109 | + } | ||
| 110 | + } | ||
| 111 | + var x = a + b; | ||
| 112 | + var bv = x - a; | ||
| 113 | + var y = b - bv; | ||
| 114 | + var q0 = y; | ||
| 115 | + var q1 = x; | ||
| 116 | + var _x, _bv, _av, _br, _ar; | ||
| 117 | + while(eptr < ne && fptr < nf) { | ||
| 118 | + if(ea < fa) { | ||
| 119 | + a = ei; | ||
| 120 | + eptr += 1; | ||
| 121 | + if(eptr < ne) { | ||
| 122 | + ei = e[eptr]; | ||
| 123 | + ea = abs(ei); | ||
| 124 | + } | ||
| 125 | + } else { | ||
| 126 | + a = fi; | ||
| 127 | + fptr += 1; | ||
| 128 | + if(fptr < nf) { | ||
| 129 | + fi = f[fptr]; | ||
| 130 | + fa = abs(fi); | ||
| 131 | + } | ||
| 132 | + } | ||
| 133 | + b = q0; | ||
| 134 | + x = a + b; | ||
| 135 | + bv = x - a; | ||
| 136 | + y = b - bv; | ||
| 137 | + if(y) { | ||
| 138 | + g[count++] = y; | ||
| 139 | + } | ||
| 140 | + _x = q1 + x; | ||
| 141 | + _bv = _x - q1; | ||
| 142 | + _av = _x - _bv; | ||
| 143 | + _br = x - _bv; | ||
| 144 | + _ar = q1 - _av; | ||
| 145 | + q0 = _ar + _br; | ||
| 146 | + q1 = _x; | ||
| 147 | + } | ||
| 148 | + while(eptr < ne) { | ||
| 149 | + a = ei; | ||
| 150 | + b = q0; | ||
| 151 | + x = a + b; | ||
| 152 | + bv = x - a; | ||
| 153 | + y = b - bv; | ||
| 154 | + if(y) { | ||
| 155 | + g[count++] = y; | ||
| 156 | + } | ||
| 157 | + _x = q1 + x; | ||
| 158 | + _bv = _x - q1; | ||
| 159 | + _av = _x - _bv; | ||
| 160 | + _br = x - _bv; | ||
| 161 | + _ar = q1 - _av; | ||
| 162 | + q0 = _ar + _br; | ||
| 163 | + q1 = _x; | ||
| 164 | + eptr += 1; | ||
| 165 | + if(eptr < ne) { | ||
| 166 | + ei = e[eptr]; | ||
| 167 | + } | ||
| 168 | + } | ||
| 169 | + while(fptr < nf) { | ||
| 170 | + a = fi; | ||
| 171 | + b = q0; | ||
| 172 | + x = a + b; | ||
| 173 | + bv = x - a; | ||
| 174 | + y = b - bv; | ||
| 175 | + if(y) { | ||
| 176 | + g[count++] = y; | ||
| 177 | + } | ||
| 178 | + _x = q1 + x; | ||
| 179 | + _bv = _x - q1; | ||
| 180 | + _av = _x - _bv; | ||
| 181 | + _br = x - _bv; | ||
| 182 | + _ar = q1 - _av; | ||
| 183 | + q0 = _ar + _br; | ||
| 184 | + q1 = _x; | ||
| 185 | + fptr += 1; | ||
| 186 | + if(fptr < nf) { | ||
| 187 | + fi = f[fptr]; | ||
| 188 | + } | ||
| 189 | + } | ||
| 190 | + if(q0) { | ||
| 191 | + g[count++] = q0; | ||
| 192 | + } | ||
| 193 | + if(q1) { | ||
| 194 | + g[count++] = q1; | ||
| 195 | + } | ||
| 196 | + if(!count) { | ||
| 197 | + g[count++] = 0.0; | ||
| 198 | + } | ||
| 199 | + g.length = count; | ||
| 200 | + return g; | ||
| 201 | + }; | ||
| 202 | + | ||
| 203 | + var robustSum = linearExpansionSum; | ||
| 204 | + | ||
| 205 | + var scaleLinearExpansion = function(e, scale) { | ||
| 206 | + var n = e.length; | ||
| 207 | + if(n === 1) { | ||
| 208 | + var ts = twoProduct(e[0], scale); | ||
| 209 | + if(ts[0]) { | ||
| 210 | + return ts; | ||
| 211 | + } | ||
| 212 | + return [ ts[1] ]; | ||
| 213 | + } | ||
| 214 | + var g = new Array(2 * n); | ||
| 215 | + var q = [0.1, 0.1]; | ||
| 216 | + var t = [0.1, 0.1]; | ||
| 217 | + var count = 0; | ||
| 218 | + twoProduct(e[0], scale, q); | ||
| 219 | + if(q[0]) { | ||
| 220 | + g[count++] = q[0]; | ||
| 221 | + } | ||
| 222 | + for(var i=1; i<n; ++i) { | ||
| 223 | + twoProduct(e[i], scale, t); | ||
| 224 | + var pq = q[1]; | ||
| 225 | + twoSum(pq, t[0], q); | ||
| 226 | + if(q[0]) { | ||
| 227 | + g[count++] = q[0]; | ||
| 228 | + } | ||
| 229 | + var a = t[1]; | ||
| 230 | + var b = q[1]; | ||
| 231 | + var x = a + b; | ||
| 232 | + var bv = x - a; | ||
| 233 | + var y = b - bv; | ||
| 234 | + q[1] = x; | ||
| 235 | + if(y) { | ||
| 236 | + g[count++] = y; | ||
| 237 | + } | ||
| 238 | + } | ||
| 239 | + if(q[1]) { | ||
| 240 | + g[count++] = q[1]; | ||
| 241 | + } | ||
| 242 | + if(count === 0) { | ||
| 243 | + g[count++] = 0.0; | ||
| 244 | + } | ||
| 245 | + g.length = count; | ||
| 246 | + return g; | ||
| 247 | + }; | ||
| 248 | + | ||
| 249 | + var robustScale = scaleLinearExpansion; | ||
| 250 | + | ||
| 251 | + var scalarScalar = function(a, b) { | ||
| 252 | + var x = a + b; | ||
| 253 | + var bv = x - a; | ||
| 254 | + var av = x - bv; | ||
| 255 | + var br = b - bv; | ||
| 256 | + var ar = a - av; | ||
| 257 | + var y = ar + br; | ||
| 258 | + if(y) { | ||
| 259 | + return [y, x]; | ||
| 260 | + } | ||
| 261 | + return [x]; | ||
| 262 | + }; | ||
| 263 | + | ||
| 264 | + var robustSubtract = function(e, f) { | ||
| 265 | + var ne = e.length|0; | ||
| 266 | + var nf = f.length|0; | ||
| 267 | + if(ne === 1 && nf === 1) { | ||
| 268 | + return scalarScalar(e[0], -f[0]); | ||
| 269 | + } | ||
| 270 | + var n = ne + nf; | ||
| 271 | + var g = new Array(n); | ||
| 272 | + var count = 0; | ||
| 273 | + var eptr = 0; | ||
| 274 | + var fptr = 0; | ||
| 275 | + var abs = Math.abs; | ||
| 276 | + var ei = e[eptr]; | ||
| 277 | + var ea = abs(ei); | ||
| 278 | + var fi = -f[fptr]; | ||
| 279 | + var fa = abs(fi); | ||
| 280 | + var a, b; | ||
| 281 | + if(ea < fa) { | ||
| 282 | + b = ei; | ||
| 283 | + eptr += 1; | ||
| 284 | + if(eptr < ne) { | ||
| 285 | + ei = e[eptr]; | ||
| 286 | + ea = abs(ei); | ||
| 287 | + } | ||
| 288 | + } else { | ||
| 289 | + b = fi; | ||
| 290 | + fptr += 1; | ||
| 291 | + if(fptr < nf) { | ||
| 292 | + fi = -f[fptr]; | ||
| 293 | + fa = abs(fi); | ||
| 294 | + } | ||
| 295 | + } | ||
| 296 | + if((eptr < ne && ea < fa) || (fptr >= nf)) { | ||
| 297 | + a = ei; | ||
| 298 | + eptr += 1; | ||
| 299 | + if(eptr < ne) { | ||
| 300 | + ei = e[eptr]; | ||
| 301 | + ea = abs(ei); | ||
| 302 | + } | ||
| 303 | + } else { | ||
| 304 | + a = fi; | ||
| 305 | + fptr += 1; | ||
| 306 | + if(fptr < nf) { | ||
| 307 | + fi = -f[fptr]; | ||
| 308 | + fa = abs(fi); | ||
| 309 | + } | ||
| 310 | + } | ||
| 311 | + var x = a + b; | ||
| 312 | + var bv = x - a; | ||
| 313 | + var y = b - bv; | ||
| 314 | + var q0 = y; | ||
| 315 | + var q1 = x; | ||
| 316 | + var _x, _bv, _av, _br, _ar; | ||
| 317 | + while(eptr < ne && fptr < nf) { | ||
| 318 | + if(ea < fa) { | ||
| 319 | + a = ei; | ||
| 320 | + eptr += 1; | ||
| 321 | + if(eptr < ne) { | ||
| 322 | + ei = e[eptr]; | ||
| 323 | + ea = abs(ei); | ||
| 324 | + } | ||
| 325 | + } else { | ||
| 326 | + a = fi; | ||
| 327 | + fptr += 1; | ||
| 328 | + if(fptr < nf) { | ||
| 329 | + fi = -f[fptr]; | ||
| 330 | + fa = abs(fi); | ||
| 331 | + } | ||
| 332 | + } | ||
| 333 | + b = q0; | ||
| 334 | + x = a + b; | ||
| 335 | + bv = x - a; | ||
| 336 | + y = b - bv; | ||
| 337 | + if(y) { | ||
| 338 | + g[count++] = y; | ||
| 339 | + } | ||
| 340 | + _x = q1 + x; | ||
| 341 | + _bv = _x - q1; | ||
| 342 | + _av = _x - _bv; | ||
| 343 | + _br = x - _bv; | ||
| 344 | + _ar = q1 - _av; | ||
| 345 | + q0 = _ar + _br; | ||
| 346 | + q1 = _x; | ||
| 347 | + } | ||
| 348 | + while(eptr < ne) { | ||
| 349 | + a = ei; | ||
| 350 | + b = q0; | ||
| 351 | + x = a + b; | ||
| 352 | + bv = x - a; | ||
| 353 | + y = b - bv; | ||
| 354 | + if(y) { | ||
| 355 | + g[count++] = y; | ||
| 356 | + } | ||
| 357 | + _x = q1 + x; | ||
| 358 | + _bv = _x - q1; | ||
| 359 | + _av = _x - _bv; | ||
| 360 | + _br = x - _bv; | ||
| 361 | + _ar = q1 - _av; | ||
| 362 | + q0 = _ar + _br; | ||
| 363 | + q1 = _x; | ||
| 364 | + eptr += 1; | ||
| 365 | + if(eptr < ne) { | ||
| 366 | + ei = e[eptr]; | ||
| 367 | + } | ||
| 368 | + } | ||
| 369 | + while(fptr < nf) { | ||
| 370 | + a = fi; | ||
| 371 | + b = q0; | ||
| 372 | + x = a + b; | ||
| 373 | + bv = x - a; | ||
| 374 | + y = b - bv; | ||
| 375 | + if(y) { | ||
| 376 | + g[count++] = y; | ||
| 377 | + } | ||
| 378 | + _x = q1 + x; | ||
| 379 | + _bv = _x - q1; | ||
| 380 | + _av = _x - _bv; | ||
| 381 | + _br = x - _bv; | ||
| 382 | + _ar = q1 - _av; | ||
| 383 | + q0 = _ar + _br; | ||
| 384 | + q1 = _x; | ||
| 385 | + fptr += 1; | ||
| 386 | + if(fptr < nf) { | ||
| 387 | + fi = -f[fptr]; | ||
| 388 | + } | ||
| 389 | + } | ||
| 390 | + if(q0) { | ||
| 391 | + g[count++] = q0; | ||
| 392 | + } | ||
| 393 | + if(q1) { | ||
| 394 | + g[count++] = q1; | ||
| 395 | + } | ||
| 396 | + if(!count) { | ||
| 397 | + g[count++] = 0.0; | ||
| 398 | + } | ||
| 399 | + g.length = count; | ||
| 400 | + return g; | ||
| 401 | + }; | ||
| 402 | + | ||
| 403 | + var cofactor = function(m, c) { | ||
| 404 | + var result = new Array(m.length-1); | ||
| 405 | + for(var i=1; i<m.length; ++i) { | ||
| 406 | + var r = result[i-1] = new Array(m.length-1); | ||
| 407 | + for(var j=0,k=0; j<m.length; ++j) { | ||
| 408 | + if(j === c) { | ||
| 409 | + continue; | ||
| 410 | + } | ||
| 411 | + r[k++] = m[i][j]; | ||
| 412 | + } | ||
| 413 | + } | ||
| 414 | + return result; | ||
| 415 | + }; | ||
| 416 | + | ||
| 417 | + var matrix = function(n) { | ||
| 418 | + var result = new Array(n); | ||
| 419 | + for(var i=0; i<n; ++i) { | ||
| 420 | + result[i] = new Array(n); | ||
| 421 | + for(var j=0; j<n; ++j) { | ||
| 422 | + result[i][j] = ["m", j, "[", (n-i-1), "]"].join(""); | ||
| 423 | + } | ||
| 424 | + } | ||
| 425 | + return result; | ||
| 426 | + }; | ||
| 427 | + | ||
| 428 | + var sign = function(n) { | ||
| 429 | + if(n & 1) { | ||
| 430 | + return "-"; | ||
| 431 | + } | ||
| 432 | + return ""; | ||
| 433 | + }; | ||
| 434 | + | ||
| 435 | + var generateSum = function(expr) { | ||
| 436 | + if(expr.length === 1) { | ||
| 437 | + return expr[0]; | ||
| 438 | + } else if(expr.length === 2) { | ||
| 439 | + return ["sum(", expr[0], ",", expr[1], ")"].join(""); | ||
| 440 | + } else { | ||
| 441 | + var m = expr.length>>1; | ||
| 442 | + return ["sum(", generateSum(expr.slice(0, m)), ",", generateSum(expr.slice(m)), ")"].join(""); | ||
| 443 | + } | ||
| 444 | + }; | ||
| 445 | + | ||
| 446 | + var determinant = function(m) { | ||
| 447 | + if(m.length === 2) { | ||
| 448 | + return [["sum(prod(", m[0][0], ",", m[1][1], "),prod(-", m[0][1], ",", m[1][0], "))"].join("")]; | ||
| 449 | + } else { | ||
| 450 | + var expr = []; | ||
| 451 | + for(var i=0; i<m.length; ++i) { | ||
| 452 | + expr.push(["scale(", generateSum(determinant(cofactor(m, i))), ",", sign(i), m[0][i], ")"].join("")); | ||
| 453 | + } | ||
| 454 | + return expr; | ||
| 455 | + } | ||
| 456 | + }; | ||
| 457 | + | ||
| 458 | + var orientation = function(n) { | ||
| 459 | + var pos = []; | ||
| 460 | + var neg = []; | ||
| 461 | + var m = matrix(n); | ||
| 462 | + var args = []; | ||
| 463 | + for(var i=0; i<n; ++i) { | ||
| 464 | + if((i&1)===0) { | ||
| 465 | + pos.push.apply(pos, determinant(cofactor(m, i))); | ||
| 466 | + } else { | ||
| 467 | + neg.push.apply(neg, determinant(cofactor(m, i))); | ||
| 468 | + } | ||
| 469 | + args.push("m" + i); | ||
| 470 | + } | ||
| 471 | + var posExpr = generateSum(pos); | ||
| 472 | + var negExpr = generateSum(neg); | ||
| 473 | + var funcName = "orientation" + n + "Exact"; | ||
| 474 | + var code = [ | ||
| 475 | + "function ", | ||
| 476 | + funcName, | ||
| 477 | + "(", args.join(), "){var p=", | ||
| 478 | + posExpr, | ||
| 479 | + ",n=", | ||
| 480 | + negExpr, | ||
| 481 | + ",d=sub(p,n);return d[d.length-1];};return ", | ||
| 482 | + funcName | ||
| 483 | + ].join(""); | ||
| 484 | + var proc = new Function("sum", "prod", "scale", "sub", code); | ||
| 485 | + return proc(robustSum, twoProduct, robustScale, robustSubtract); | ||
| 486 | + }; | ||
| 487 | + | ||
| 488 | + var orient; | ||
| 489 | + var orientation3Exact = orientation(3); | ||
| 490 | + var orientation4Exact = orientation(4); | ||
| 491 | + | ||
| 492 | + var CACHED = [ | ||
| 493 | + function orientation0() { return 0; }, | ||
| 494 | + function orientation1() { return 0; }, | ||
| 495 | + function orientation2(a, b) { | ||
| 496 | + return b[0] - a[0]; | ||
| 497 | + }, | ||
| 498 | + function orientation3(a, b, c) { | ||
| 499 | + var l = (a[1] - c[1]) * (b[0] - c[0]); | ||
| 500 | + var r = (a[0] - c[0]) * (b[1] - c[1]); | ||
| 501 | + var det = l - r; | ||
| 502 | + var s; | ||
| 503 | + if(l > 0) { | ||
| 504 | + if(r <= 0) { | ||
| 505 | + return det; | ||
| 506 | + } else { | ||
| 507 | + s = l + r; | ||
| 508 | + } | ||
| 509 | + } else if(l < 0) { | ||
| 510 | + if(r >= 0) { | ||
| 511 | + return det; | ||
| 512 | + } else { | ||
| 513 | + s = -(l + r); | ||
| 514 | + } | ||
| 515 | + } else { | ||
| 516 | + return det; | ||
| 517 | + } | ||
| 518 | + var tol = ERRBOUND3 * s; | ||
| 519 | + if(det >= tol || det <= -tol) { | ||
| 520 | + return det; | ||
| 521 | + } | ||
| 522 | + return orientation3Exact(a, b, c); | ||
| 523 | + }, | ||
| 524 | + function orientation4(a,b,c,d) { | ||
| 525 | + var adx = a[0] - d[0]; | ||
| 526 | + var bdx = b[0] - d[0]; | ||
| 527 | + var cdx = c[0] - d[0]; | ||
| 528 | + var ady = a[1] - d[1]; | ||
| 529 | + var bdy = b[1] - d[1]; | ||
| 530 | + var cdy = c[1] - d[1]; | ||
| 531 | + var adz = a[2] - d[2]; | ||
| 532 | + var bdz = b[2] - d[2]; | ||
| 533 | + var cdz = c[2] - d[2]; | ||
| 534 | + var bdxcdy = bdx * cdy; | ||
| 535 | + var cdxbdy = cdx * bdy; | ||
| 536 | + var cdxady = cdx * ady; | ||
| 537 | + var adxcdy = adx * cdy; | ||
| 538 | + var adxbdy = adx * bdy; | ||
| 539 | + var bdxady = bdx * ady; | ||
| 540 | + var det = adz * (bdxcdy - cdxbdy) + | ||
| 541 | + bdz * (cdxady - adxcdy) + | ||
| 542 | + cdz * (adxbdy - bdxady); | ||
| 543 | + var permanent = (Math.abs(bdxcdy) + Math.abs(cdxbdy)) * Math.abs(adz) + | ||
| 544 | + (Math.abs(cdxady) + Math.abs(adxcdy)) * Math.abs(bdz) + | ||
| 545 | + (Math.abs(adxbdy) + Math.abs(bdxady)) * Math.abs(cdz); | ||
| 546 | + var tol = ERRBOUND4 * permanent; | ||
| 547 | + if ((det > tol) || (-det > tol)) { | ||
| 548 | + return det; | ||
| 549 | + } | ||
| 550 | + return orientation4Exact(a,b,c,d); | ||
| 551 | + } | ||
| 552 | + ]; | ||
| 553 | + | ||
| 554 | + var slowOrient = function(args) { | ||
| 555 | + var proc = CACHED[args.length]; | ||
| 556 | + if(!proc) { | ||
| 557 | + proc = CACHED[args.length] = orientation(args.length); | ||
| 558 | + } | ||
| 559 | + return proc.apply(undefined, args); | ||
| 560 | + }; | ||
| 561 | + | ||
| 562 | + var generateOrientationProc = function() { | ||
| 563 | + while(CACHED.length <= NUM_EXPAND) { | ||
| 564 | + CACHED.push(orientation(CACHED.length)); | ||
| 565 | + } | ||
| 566 | + var args = []; | ||
| 567 | + var procArgs = ["slow"]; | ||
| 568 | + for(var i=0; i<=NUM_EXPAND; ++i) { | ||
| 569 | + args.push("a" + i); | ||
| 570 | + procArgs.push("o" + i); | ||
| 571 | + } | ||
| 572 | + var code = [ | ||
| 573 | + "function getOrientation(", | ||
| 574 | + args.join(), | ||
| 575 | + "){switch(arguments.length){case 0:case 1:return 0;" | ||
| 576 | + ]; | ||
| 577 | + for(i=2; i<=NUM_EXPAND; ++i) { | ||
| 578 | + code.push("case ", i, ":return o", i, "(", args.slice(0, i).join(), ");"); | ||
| 579 | + } | ||
| 580 | + code.push("}var s=new Array(arguments.length);for(var i=0;i<arguments.length;++i){s[i]=arguments[i]};return slow(s);}return getOrientation"); | ||
| 581 | + procArgs.push(code.join("")); | ||
| 582 | + | ||
| 583 | + var proc = Function.apply(undefined, procArgs); | ||
| 584 | + orient = proc.apply(undefined, [slowOrient].concat(CACHED)); | ||
| 585 | + for(i=0; i<=NUM_EXPAND; ++i) { | ||
| 586 | + orient[i] = CACHED[i]; | ||
| 587 | + } | ||
| 588 | + }; | ||
| 589 | + | ||
| 590 | + generateOrientationProc(); | ||
| 591 | + | ||
| 592 | + var robustPointInPolygon = function(vs, point) { | ||
| 593 | + // transform from geolib format to array syntax | ||
| 594 | + var x = geolib.longitude(point); | ||
| 595 | + var y = geolib.latitude(point); | ||
| 596 | + var coords = vs.map(function(coords) { | ||
| 597 | + return [geolib.longitude(coords), geolib.latitude(coords)]; | ||
| 598 | + }); | ||
| 599 | + | ||
| 600 | + vs = coords; | ||
| 601 | + point = [x,y]; | ||
| 602 | + | ||
| 603 | + var n = vs.length; | ||
| 604 | + var inside = 1; | ||
| 605 | + var lim = n; | ||
| 606 | + | ||
| 607 | + var s, c, yk, px, p; | ||
| 608 | + | ||
| 609 | + for(var i = 0, j = n-1; i<lim; j=i++) { | ||
| 610 | + var a = vs[i]; | ||
| 611 | + var b = vs[j]; | ||
| 612 | + var yi = a[1]; | ||
| 613 | + var yj = b[1]; | ||
| 614 | + if(yj < yi) { | ||
| 615 | + if(yj < y && y < yi) { | ||
| 616 | + s = orient(a, b, point); | ||
| 617 | + if(s === 0) { | ||
| 618 | + return 0; | ||
| 619 | + } else { | ||
| 620 | + inside ^= (0 < s)|0; | ||
| 621 | + } | ||
| 622 | + } else if(y === yi) { | ||
| 623 | + c = vs[(i+1)%n]; | ||
| 624 | + yk = c[1]; | ||
| 625 | + if(yi < yk) { | ||
| 626 | + s = orient(a, b, point); | ||
| 627 | + if(s === 0) { | ||
| 628 | + return 0; | ||
| 629 | + } else { | ||
| 630 | + inside ^= (0 < s)|0; | ||
| 631 | + } | ||
| 632 | + } | ||
| 633 | + } | ||
| 634 | + } else if(yi < yj) { | ||
| 635 | + if(yi < y && y < yj) { | ||
| 636 | + s = orient(a, b, point); | ||
| 637 | + if(s === 0) { | ||
| 638 | + return 0; | ||
| 639 | + } else { | ||
| 640 | + inside ^= (s < 0)|0; | ||
| 641 | + } | ||
| 642 | + } else if(y === yi) { | ||
| 643 | + c = vs[(i+1)%n]; | ||
| 644 | + yk = c[1]; | ||
| 645 | + if(yk < yi) { | ||
| 646 | + s = orient(a, b, point); | ||
| 647 | + if(s === 0) { | ||
| 648 | + return 0; | ||
| 649 | + } else { | ||
| 650 | + inside ^= (s < 0)|0; | ||
| 651 | + } | ||
| 652 | + } | ||
| 653 | + } | ||
| 654 | + } else if(y === yi) { | ||
| 655 | + var x0 = Math.min(a[0], b[0]); | ||
| 656 | + var x1 = Math.max(a[0], b[0]); | ||
| 657 | + if(i === 0) { | ||
| 658 | + while(j>0) { | ||
| 659 | + var k = (j+n-1)%n; | ||
| 660 | + p = vs[k]; | ||
| 661 | + if(p[1] !== y) { | ||
| 662 | + break; | ||
| 663 | + } | ||
| 664 | + px = p[0]; | ||
| 665 | + x0 = Math.min(x0, px); | ||
| 666 | + x1 = Math.max(x1, px); | ||
| 667 | + j = k; | ||
| 668 | + } | ||
| 669 | + if(j === 0) { | ||
| 670 | + if(x0 <= x && x <= x1) { | ||
| 671 | + return 0; | ||
| 672 | + } | ||
| 673 | + return 1; | ||
| 674 | + } | ||
| 675 | + lim = j+1; | ||
| 676 | + } | ||
| 677 | + var y0 = vs[(j+n-1)%n][1]; | ||
| 678 | + while(i+1<lim) { | ||
| 679 | + p = vs[i+1]; | ||
| 680 | + if(p[1] !== y) { | ||
| 681 | + break; | ||
| 682 | + } | ||
| 683 | + px = p[0]; | ||
| 684 | + x0 = Math.min(x0, px); | ||
| 685 | + x1 = Math.max(x1, px); | ||
| 686 | + i += 1; | ||
| 687 | + } | ||
| 688 | + if(x0 <= x && x <= x1) { | ||
| 689 | + return 0; | ||
| 690 | + } | ||
| 691 | + var y1 = vs[(i+1)%n][1]; | ||
| 692 | + if(x < x0 && (y0 < y !== y1 < y)) { | ||
| 693 | + inside ^= 1; | ||
| 694 | + } | ||
| 695 | + } | ||
| 696 | + } | ||
| 697 | + return 2 * inside - 1; | ||
| 698 | + }; | ||
| 699 | + | ||
| 700 | + return { | ||
| 701 | + | ||
| 702 | + /** | ||
| 703 | + * @param object coordinate to check e.g. {latitude: 51.5023, longitude: 7.3815} | ||
| 704 | + * @param array array with coords e.g. [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 705 | + * @return integer -1 if point is inside, 0 if point is on boundaries, 1 if point is outside | ||
| 706 | + */ | ||
| 707 | + isPointInsideRobust: function(latlng, coords) { | ||
| 708 | + return robustPointInPolygon(coords, latlng); | ||
| 709 | + }, | ||
| 710 | + | ||
| 711 | + isInside: function() { | ||
| 712 | + return this.isPointInsideRobust.apply(this, arguments); | ||
| 713 | + } | ||
| 714 | + | ||
| 715 | + }; | ||
| 716 | + | ||
| 717 | + }; | ||
| 718 | + | ||
| 719 | + | ||
| 720 | + // Node module | ||
| 721 | + if (typeof module !== 'undefined' && | ||
| 722 | + typeof module.exports !== 'undefined') { | ||
| 723 | + | ||
| 724 | + module.exports = function(geolib) { | ||
| 725 | + geolib.extend(addOn(geolib), true); | ||
| 726 | + return geolib; | ||
| 727 | + }; | ||
| 728 | + | ||
| 729 | + // AMD module | ||
| 730 | + } else if (typeof define === "function" && define.amd) { | ||
| 731 | + | ||
| 732 | + define(["geolib"], function (geolib) { | ||
| 733 | + geolib.extend(addOn(geolib), true); | ||
| 734 | + return geolib; | ||
| 735 | + }); | ||
| 736 | + | ||
| 737 | + // we're in a browser | ||
| 738 | + } else { | ||
| 739 | + | ||
| 740 | + geolib.extend(addOn(geolib), true); | ||
| 741 | + | ||
| 742 | + } | ||
| 743 | + | ||
| 744 | +}(this, this.geolib)); |
src/main/resources/static/real_control_v2/geolib/geolib.isPointInsideRobust.min.js
0 → 100644
| 1 | +/*! geolib.isPointInsideRobust 2.0.21 | ||
| 2 | +* !!EXPERIMENTAL!! | ||
| 3 | +* | ||
| 4 | +* Robust version of isPointInside for Geolib.js | ||
| 5 | +* | ||
| 6 | +* Based on https://github.com/mikolalysenko/robust-point-in-polygon | ||
| 7 | +* by Mikola Lysenko, licensed under MIT | ||
| 8 | +* | ||
| 9 | +* @author Manuel Bieh | ||
| 10 | +* @url http://www.manuelbieh.com/ | ||
| 11 | +* @version 2.0.21 | ||
| 12 | +* @license MIT | ||
| 13 | +* | ||
| 14 | +*/ | ||
| 15 | +!function(a,b,c){var d=function(a){var b,d=+(Math.pow(2,27)+1),e=5,f=1.1102230246251565e-16,g=(3+16*f)*f,h=(7+56*f)*f,i=function(a,b,c){var e=a*b,f=d*a,g=f-a,h=f-g,i=a-h,j=d*b,k=j-b,l=j-k,m=b-l,n=e-h*l,o=n-i*l,p=o-h*m,q=i*m-p;return c?(c[0]=q,c[1]=e,c):[q,e]},j=function(a,b,c){var d=a+b,e=d-a,f=d-e,g=b-e,h=a-f;return c?(c[0]=h+g,c[1]=d,c):[h+g,d]},k=j,l=function(a,b){var c=0|a.length,d=0|b.length;if(1===c&&1===d)return p(a[0],b[0]);var e,f,g=c+d,h=new Array(g),i=0,j=0,k=0,l=Math.abs,m=a[j],n=l(m),o=b[k],q=l(o);q>n?(f=m,j+=1,c>j&&(m=a[j],n=l(m))):(f=o,k+=1,d>k&&(o=b[k],q=l(o))),c>j&&q>n||k>=d?(e=m,j+=1,c>j&&(m=a[j],n=l(m))):(e=o,k+=1,d>k&&(o=b[k],q=l(o)));for(var r,s,t,u,v,w=e+f,x=w-e,y=f-x,z=y,A=w;c>j&&d>k;)q>n?(e=m,j+=1,c>j&&(m=a[j],n=l(m))):(e=o,k+=1,d>k&&(o=b[k],q=l(o))),f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r;for(;c>j;)e=m,f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r,j+=1,c>j&&(m=a[j]);for(;d>k;)e=o,f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r,k+=1,d>k&&(o=b[k]);return z&&(h[i++]=z),A&&(h[i++]=A),i||(h[i++]=0),h.length=i,h},m=l,n=function(a,b){var c=a.length;if(1===c){var d=i(a[0],b);return d[0]?d:[d[1]]}var e=new Array(2*c),f=[.1,.1],g=[.1,.1],h=0;i(a[0],b,f),f[0]&&(e[h++]=f[0]);for(var j=1;c>j;++j){i(a[j],b,g);var l=f[1];k(l,g[0],f),f[0]&&(e[h++]=f[0]);var m=g[1],n=f[1],o=m+n,p=o-m,q=n-p;f[1]=o,q&&(e[h++]=q)}return f[1]&&(e[h++]=f[1]),0===h&&(e[h++]=0),e.length=h,e},o=n,p=function(a,b){var c=a+b,d=c-a,e=c-d,f=b-d,g=a-e,h=g+f;return h?[h,c]:[c]},q=function(a,b){var c=0|a.length,d=0|b.length;if(1===c&&1===d)return p(a[0],-b[0]);var e,f,g=c+d,h=new Array(g),i=0,j=0,k=0,l=Math.abs,m=a[j],n=l(m),o=-b[k],q=l(o);q>n?(f=m,j+=1,c>j&&(m=a[j],n=l(m))):(f=o,k+=1,d>k&&(o=-b[k],q=l(o))),c>j&&q>n||k>=d?(e=m,j+=1,c>j&&(m=a[j],n=l(m))):(e=o,k+=1,d>k&&(o=-b[k],q=l(o)));for(var r,s,t,u,v,w=e+f,x=w-e,y=f-x,z=y,A=w;c>j&&d>k;)q>n?(e=m,j+=1,c>j&&(m=a[j],n=l(m))):(e=o,k+=1,d>k&&(o=-b[k],q=l(o))),f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r;for(;c>j;)e=m,f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r,j+=1,c>j&&(m=a[j]);for(;d>k;)e=o,f=z,w=e+f,x=w-e,y=f-x,y&&(h[i++]=y),r=A+w,s=r-A,t=r-s,u=w-s,v=A-t,z=v+u,A=r,k+=1,d>k&&(o=-b[k]);return z&&(h[i++]=z),A&&(h[i++]=A),i||(h[i++]=0),h.length=i,h},r=function(a,b){for(var c=new Array(a.length-1),d=1;d<a.length;++d)for(var e=c[d-1]=new Array(a.length-1),f=0,g=0;f<a.length;++f)f!==b&&(e[g++]=a[d][f]);return c},s=function(a){for(var b=new Array(a),c=0;a>c;++c){b[c]=new Array(a);for(var d=0;a>d;++d)b[c][d]=["m",d,"[",a-c-1,"]"].join("")}return b},t=function(a){return 1&a?"-":""},u=function(a){if(1===a.length)return a[0];if(2===a.length)return["sum(",a[0],",",a[1],")"].join("");var b=a.length>>1;return["sum(",u(a.slice(0,b)),",",u(a.slice(b)),")"].join("")},v=function(a){if(2===a.length)return[["sum(prod(",a[0][0],",",a[1][1],"),prod(-",a[0][1],",",a[1][0],"))"].join("")];for(var b=[],c=0;c<a.length;++c)b.push(["scale(",u(v(r(a,c))),",",t(c),a[0][c],")"].join(""));return b},w=function(a){for(var b=[],c=[],d=s(a),e=[],f=0;a>f;++f)0===(1&f)?b.push.apply(b,v(r(d,f))):c.push.apply(c,v(r(d,f))),e.push("m"+f);var g=u(b),h=u(c),j="orientation"+a+"Exact",k=["function ",j,"(",e.join(),"){var p=",g,",n=",h,",d=sub(p,n);return d[d.length-1];};return ",j].join(""),l=new Function("sum","prod","scale","sub",k);return l(m,i,o,q)},x=w(3),y=w(4),z=[function(){return 0},function(){return 0},function(a,b){return b[0]-a[0]},function(a,b,c){var d,e=(a[1]-c[1])*(b[0]-c[0]),f=(a[0]-c[0])*(b[1]-c[1]),h=e-f;if(e>0){if(0>=f)return h;d=e+f}else{if(!(0>e))return h;if(f>=0)return h;d=-(e+f)}var i=g*d;return h>=i||-i>=h?h:x(a,b,c)},function(a,b,c,d){var e=a[0]-d[0],f=b[0]-d[0],g=c[0]-d[0],i=a[1]-d[1],j=b[1]-d[1],k=c[1]-d[1],l=a[2]-d[2],m=b[2]-d[2],n=c[2]-d[2],o=f*k,p=g*j,q=g*i,r=e*k,s=e*j,t=f*i,u=l*(o-p)+m*(q-r)+n*(s-t),v=(Math.abs(o)+Math.abs(p))*Math.abs(l)+(Math.abs(q)+Math.abs(r))*Math.abs(m)+(Math.abs(s)+Math.abs(t))*Math.abs(n),w=h*v;return u>w||-u>w?u:y(a,b,c,d)}],A=function(a){var b=z[a.length];return b||(b=z[a.length]=w(a.length)),b.apply(c,a)},B=function(){for(;z.length<=e;)z.push(w(z.length));for(var a=[],d=["slow"],f=0;e>=f;++f)a.push("a"+f),d.push("o"+f);var g=["function getOrientation(",a.join(),"){switch(arguments.length){case 0:case 1:return 0;"];for(f=2;e>=f;++f)g.push("case ",f,":return o",f,"(",a.slice(0,f).join(),");");g.push("}var s=new Array(arguments.length);for(var i=0;i<arguments.length;++i){s[i]=arguments[i]};return slow(s);}return getOrientation"),d.push(g.join(""));var h=Function.apply(c,d);for(b=h.apply(c,[A].concat(z)),f=0;e>=f;++f)b[f]=z[f]};B();var C=function(c,d){var e=a.longitude(d),f=a.latitude(d),g=c.map(function(b){return[a.longitude(b),a.latitude(b)]});c=g,d=[e,f];for(var h,i,j,k,l,m=c.length,n=1,o=m,p=0,q=m-1;o>p;q=p++){var r=c[p],s=c[q],t=r[1],u=s[1];if(t>u){if(f>u&&t>f){if(h=b(r,s,d),0===h)return 0;n^=h>0|0}else if(f===t&&(i=c[(p+1)%m],j=i[1],j>t)){if(h=b(r,s,d),0===h)return 0;n^=h>0|0}}else if(u>t){if(f>t&&u>f){if(h=b(r,s,d),0===h)return 0;n^=0>h|0}else if(f===t&&(i=c[(p+1)%m],j=i[1],t>j)){if(h=b(r,s,d),0===h)return 0;n^=0>h|0}}else if(f===t){var v=Math.min(r[0],s[0]),w=Math.max(r[0],s[0]);if(0===p){for(;q>0;){var x=(q+m-1)%m;if(l=c[x],l[1]!==f)break;k=l[0],v=Math.min(v,k),w=Math.max(w,k),q=x}if(0===q)return e>=v&&w>=e?0:1;o=q+1}for(var y=c[(q+m-1)%m][1];o>p+1&&(l=c[p+1],l[1]===f);)k=l[0],v=Math.min(v,k),w=Math.max(w,k),p+=1;if(e>=v&&w>=e)return 0;var z=c[(p+1)%m][1];v>e&&f>y!=f>z&&(n^=1)}}return 2*n-1};return{isPointInsideRobust:function(a,b){return C(b,a)},isInside:function(){return this.isPointInsideRobust.apply(this,arguments)}}};"undefined"!=typeof module&&"undefined"!=typeof module.exports?module.exports=function(a){return a.extend(d(a),!0),a}:"function"==typeof define&&define.amd?define(["geolib"],function(a){return a.extend(d(a),!0),a}):b.extend(d(b),!0)}(this,this.geolib); | ||
| 0 | \ No newline at end of file | 16 | \ No newline at end of file |
src/main/resources/static/real_control_v2/geolib/geolib.js
0 → 100644
| 1 | +/*! geolib 2.0.21 by Manuel Bieh | ||
| 2 | +* Library to provide geo functions like distance calculation, | ||
| 3 | +* conversion of decimal coordinates to sexagesimal and vice versa, etc. | ||
| 4 | +* WGS 84 (World Geodetic System 1984) | ||
| 5 | +* | ||
| 6 | +* @author Manuel Bieh | ||
| 7 | +* @url http://www.manuelbieh.com/ | ||
| 8 | +* @version 2.0.21 | ||
| 9 | +* @license MIT | ||
| 10 | +**/;(function(global, undefined) { | ||
| 11 | + | ||
| 12 | + "use strict"; | ||
| 13 | + | ||
| 14 | + function Geolib() {} | ||
| 15 | + | ||
| 16 | + // Constants | ||
| 17 | + Geolib.TO_RAD = Math.PI / 180; | ||
| 18 | + Geolib.TO_DEG = 180 / Math.PI; | ||
| 19 | + Geolib.PI_X2 = Math.PI * 2; | ||
| 20 | + Geolib.PI_DIV4 = Math.PI / 4; | ||
| 21 | + | ||
| 22 | + // Setting readonly defaults | ||
| 23 | + var geolib = Object.create(Geolib.prototype, { | ||
| 24 | + version: { | ||
| 25 | + value: "2.0.21" | ||
| 26 | + }, | ||
| 27 | + radius: { | ||
| 28 | + value: 6378137 | ||
| 29 | + }, | ||
| 30 | + minLat: { | ||
| 31 | + value: -90 | ||
| 32 | + }, | ||
| 33 | + maxLat: { | ||
| 34 | + value: 90 | ||
| 35 | + }, | ||
| 36 | + minLon: { | ||
| 37 | + value: -180 | ||
| 38 | + }, | ||
| 39 | + maxLon: { | ||
| 40 | + value: 180 | ||
| 41 | + }, | ||
| 42 | + sexagesimalPattern: { | ||
| 43 | + value: /^([0-9]{1,3})°\s*([0-9]{1,3}(?:\.(?:[0-9]{1,2}))?)'\s*(([0-9]{1,3}(\.([0-9]{1,4}))?)"\s*)?([NEOSW]?)$/ | ||
| 44 | + }, | ||
| 45 | + measures: { | ||
| 46 | + value: Object.create(Object.prototype, { | ||
| 47 | + "m" : {value: 1}, | ||
| 48 | + "km": {value: 0.001}, | ||
| 49 | + "cm": {value: 100}, | ||
| 50 | + "mm": {value: 1000}, | ||
| 51 | + "mi": {value: (1 / 1609.344)}, | ||
| 52 | + "sm": {value: (1 / 1852.216)}, | ||
| 53 | + "ft": {value: (100 / 30.48)}, | ||
| 54 | + "in": {value: (100 / 2.54)}, | ||
| 55 | + "yd": {value: (1 / 0.9144)} | ||
| 56 | + }) | ||
| 57 | + }, | ||
| 58 | + prototype: { | ||
| 59 | + value: Geolib.prototype | ||
| 60 | + }, | ||
| 61 | + extend: { | ||
| 62 | + value: function(methods, overwrite) { | ||
| 63 | + for(var prop in methods) { | ||
| 64 | + if(typeof geolib.prototype[prop] === 'undefined' || overwrite === true) { | ||
| 65 | + if(typeof methods[prop] === 'function' && typeof methods[prop].bind === 'function') { | ||
| 66 | + geolib.prototype[prop] = methods[prop].bind(geolib); | ||
| 67 | + } else { | ||
| 68 | + geolib.prototype[prop] = methods[prop]; | ||
| 69 | + } | ||
| 70 | + } | ||
| 71 | + } | ||
| 72 | + } | ||
| 73 | + } | ||
| 74 | + }); | ||
| 75 | + | ||
| 76 | + if (typeof(Number.prototype.toRad) === 'undefined') { | ||
| 77 | + Number.prototype.toRad = function() { | ||
| 78 | + return this * Geolib.TO_RAD; | ||
| 79 | + }; | ||
| 80 | + } | ||
| 81 | + | ||
| 82 | + if (typeof(Number.prototype.toDeg) === 'undefined') { | ||
| 83 | + Number.prototype.toDeg = function() { | ||
| 84 | + return this * Geolib.TO_DEG; | ||
| 85 | + }; | ||
| 86 | + } | ||
| 87 | + | ||
| 88 | + // Here comes the magic | ||
| 89 | + geolib.extend({ | ||
| 90 | + | ||
| 91 | + decimal: {}, | ||
| 92 | + | ||
| 93 | + sexagesimal: {}, | ||
| 94 | + | ||
| 95 | + distance: null, | ||
| 96 | + | ||
| 97 | + getKeys: function(point) { | ||
| 98 | + | ||
| 99 | + // GeoJSON Array [longitude, latitude(, elevation)] | ||
| 100 | + if(Object.prototype.toString.call(point) == '[object Array]') { | ||
| 101 | + | ||
| 102 | + return { | ||
| 103 | + longitude: point.length >= 1 ? 0 : undefined, | ||
| 104 | + latitude: point.length >= 2 ? 1 : undefined, | ||
| 105 | + elevation: point.length >= 3 ? 2 : undefined | ||
| 106 | + }; | ||
| 107 | + | ||
| 108 | + } | ||
| 109 | + | ||
| 110 | + var getKey = function(possibleValues) { | ||
| 111 | + | ||
| 112 | + var key; | ||
| 113 | + | ||
| 114 | + possibleValues.every(function(val) { | ||
| 115 | + // TODO: check if point is an object | ||
| 116 | + if(typeof point != 'object') { | ||
| 117 | + return true; | ||
| 118 | + } | ||
| 119 | + return point.hasOwnProperty(val) ? (function() { key = val; return false; }()) : true; | ||
| 120 | + }); | ||
| 121 | + | ||
| 122 | + return key; | ||
| 123 | + | ||
| 124 | + }; | ||
| 125 | + | ||
| 126 | + var longitude = getKey(['lng', 'lon', 'longitude']); | ||
| 127 | + var latitude = getKey(['lat', 'latitude']); | ||
| 128 | + var elevation = getKey(['alt', 'altitude', 'elevation', 'elev']); | ||
| 129 | + | ||
| 130 | + // return undefined if not at least one valid property was found | ||
| 131 | + if(typeof latitude == 'undefined' && | ||
| 132 | + typeof longitude == 'undefined' && | ||
| 133 | + typeof elevation == 'undefined') { | ||
| 134 | + return undefined; | ||
| 135 | + } | ||
| 136 | + | ||
| 137 | + return { | ||
| 138 | + latitude: latitude, | ||
| 139 | + longitude: longitude, | ||
| 140 | + elevation: elevation | ||
| 141 | + }; | ||
| 142 | + | ||
| 143 | + }, | ||
| 144 | + | ||
| 145 | + // returns latitude of a given point, converted to decimal | ||
| 146 | + // set raw to true to avoid conversion | ||
| 147 | + getLat: function(point, raw) { | ||
| 148 | + return raw === true ? point[this.getKeys(point).latitude] : this.useDecimal(point[this.getKeys(point).latitude]); | ||
| 149 | + }, | ||
| 150 | + | ||
| 151 | + // Alias for getLat | ||
| 152 | + latitude: function(point) { | ||
| 153 | + return this.getLat.call(this, point); | ||
| 154 | + }, | ||
| 155 | + | ||
| 156 | + // returns longitude of a given point, converted to decimal | ||
| 157 | + // set raw to true to avoid conversion | ||
| 158 | + getLon: function(point, raw) { | ||
| 159 | + return raw === true ? point[this.getKeys(point).longitude] : this.useDecimal(point[this.getKeys(point).longitude]); | ||
| 160 | + }, | ||
| 161 | + | ||
| 162 | + // Alias for getLon | ||
| 163 | + longitude: function(point) { | ||
| 164 | + return this.getLon.call(this, point); | ||
| 165 | + }, | ||
| 166 | + | ||
| 167 | + getElev: function(point) { | ||
| 168 | + return point[this.getKeys(point).elevation]; | ||
| 169 | + }, | ||
| 170 | + | ||
| 171 | + // Alias for getElev | ||
| 172 | + elevation: function(point) { | ||
| 173 | + return this.getElev.call(this, point); | ||
| 174 | + }, | ||
| 175 | + | ||
| 176 | + coords: function(point, raw) { | ||
| 177 | + | ||
| 178 | + var retval = { | ||
| 179 | + latitude: raw === true ? point[this.getKeys(point).latitude] : this.useDecimal(point[this.getKeys(point).latitude]), | ||
| 180 | + longitude: raw === true ? point[this.getKeys(point).longitude] : this.useDecimal(point[this.getKeys(point).longitude]) | ||
| 181 | + }; | ||
| 182 | + | ||
| 183 | + var elev = point[this.getKeys(point).elevation]; | ||
| 184 | + | ||
| 185 | + if(typeof elev !== 'undefined') { | ||
| 186 | + retval['elevation'] = elev; | ||
| 187 | + } | ||
| 188 | + | ||
| 189 | + return retval; | ||
| 190 | + | ||
| 191 | + }, | ||
| 192 | + | ||
| 193 | + // Alias for coords | ||
| 194 | + ll: function(point, raw) { | ||
| 195 | + return this.coords.call(this, point, raw); | ||
| 196 | + }, | ||
| 197 | + | ||
| 198 | + | ||
| 199 | + // checks if a variable contains a valid latlong object | ||
| 200 | + validate: function(point) { | ||
| 201 | + | ||
| 202 | + var keys = this.getKeys(point); | ||
| 203 | + | ||
| 204 | + if(typeof keys === 'undefined' || typeof keys.latitude === 'undefined' || keys.longitude === 'undefined') { | ||
| 205 | + return false; | ||
| 206 | + } | ||
| 207 | + | ||
| 208 | + var lat = point[keys.latitude]; | ||
| 209 | + var lng = point[keys.longitude]; | ||
| 210 | + | ||
| 211 | + if(typeof lat === 'undefined' || !this.isDecimal(lat) && !this.isSexagesimal(lat)) { | ||
| 212 | + return false; | ||
| 213 | + } | ||
| 214 | + | ||
| 215 | + if(typeof lng === 'undefined' || !this.isDecimal(lng) && !this.isSexagesimal(lng)) { | ||
| 216 | + return false; | ||
| 217 | + } | ||
| 218 | + | ||
| 219 | + lat = this.useDecimal(lat); | ||
| 220 | + lng = this.useDecimal(lng); | ||
| 221 | + | ||
| 222 | + if(lat < this.minLat || lat > this.maxLat || lng < this.minLon || lng > this.maxLon) { | ||
| 223 | + return false; | ||
| 224 | + } | ||
| 225 | + | ||
| 226 | + return true; | ||
| 227 | + | ||
| 228 | + }, | ||
| 229 | + | ||
| 230 | + /** | ||
| 231 | + * Calculates geodetic distance between two points specified by latitude/longitude using | ||
| 232 | + * Vincenty inverse formula for ellipsoids | ||
| 233 | + * Vincenty Inverse Solution of Geodesics on the Ellipsoid (c) Chris Veness 2002-2010 | ||
| 234 | + * (Licensed under CC BY 3.0) | ||
| 235 | + * | ||
| 236 | + * @param object Start position {latitude: 123, longitude: 123} | ||
| 237 | + * @param object End position {latitude: 123, longitude: 123} | ||
| 238 | + * @param integer Accuracy (in meters) | ||
| 239 | + * @param integer Precision (in decimal cases) | ||
| 240 | + * @return integer Distance (in meters) | ||
| 241 | + */ | ||
| 242 | + getDistance: function(start, end, accuracy, precision) { | ||
| 243 | + | ||
| 244 | + accuracy = Math.floor(accuracy) || 1; | ||
| 245 | + precision = Math.floor(precision) || 0; | ||
| 246 | + | ||
| 247 | + var s = this.coords(start); | ||
| 248 | + var e = this.coords(end); | ||
| 249 | + | ||
| 250 | + var a = 6378137, b = 6356752.314245, f = 1/298.257223563; // WGS-84 ellipsoid params | ||
| 251 | + var L = (e['longitude']-s['longitude']).toRad(); | ||
| 252 | + | ||
| 253 | + var cosSigma, sigma, sinAlpha, cosSqAlpha, cos2SigmaM, sinSigma; | ||
| 254 | + | ||
| 255 | + var U1 = Math.atan((1-f) * Math.tan(parseFloat(s['latitude']).toRad())); | ||
| 256 | + var U2 = Math.atan((1-f) * Math.tan(parseFloat(e['latitude']).toRad())); | ||
| 257 | + var sinU1 = Math.sin(U1), cosU1 = Math.cos(U1); | ||
| 258 | + var sinU2 = Math.sin(U2), cosU2 = Math.cos(U2); | ||
| 259 | + | ||
| 260 | + var lambda = L, lambdaP, iterLimit = 100; | ||
| 261 | + do { | ||
| 262 | + var sinLambda = Math.sin(lambda), cosLambda = Math.cos(lambda); | ||
| 263 | + sinSigma = ( | ||
| 264 | + Math.sqrt( | ||
| 265 | + ( | ||
| 266 | + cosU2 * sinLambda | ||
| 267 | + ) * ( | ||
| 268 | + cosU2 * sinLambda | ||
| 269 | + ) + ( | ||
| 270 | + cosU1 * sinU2 - sinU1 * cosU2 * cosLambda | ||
| 271 | + ) * ( | ||
| 272 | + cosU1 * sinU2 - sinU1 * cosU2 * cosLambda | ||
| 273 | + ) | ||
| 274 | + ) | ||
| 275 | + ); | ||
| 276 | + if (sinSigma === 0) { | ||
| 277 | + return geolib.distance = 0; // co-incident points | ||
| 278 | + } | ||
| 279 | + | ||
| 280 | + cosSigma = sinU1 * sinU2 + cosU1 * cosU2 * cosLambda; | ||
| 281 | + sigma = Math.atan2(sinSigma, cosSigma); | ||
| 282 | + sinAlpha = cosU1 * cosU2 * sinLambda / sinSigma; | ||
| 283 | + cosSqAlpha = 1 - sinAlpha * sinAlpha; | ||
| 284 | + cos2SigmaM = cosSigma - 2 * sinU1 * sinU2 / cosSqAlpha; | ||
| 285 | + | ||
| 286 | + if (isNaN(cos2SigmaM)) { | ||
| 287 | + cos2SigmaM = 0; // equatorial line: cosSqAlpha=0 (§6) | ||
| 288 | + } | ||
| 289 | + var C = ( | ||
| 290 | + f / 16 * cosSqAlpha * ( | ||
| 291 | + 4 + f * ( | ||
| 292 | + 4 - 3 * cosSqAlpha | ||
| 293 | + ) | ||
| 294 | + ) | ||
| 295 | + ); | ||
| 296 | + lambdaP = lambda; | ||
| 297 | + lambda = ( | ||
| 298 | + L + ( | ||
| 299 | + 1 - C | ||
| 300 | + ) * f * sinAlpha * ( | ||
| 301 | + sigma + C * sinSigma * ( | ||
| 302 | + cos2SigmaM + C * cosSigma * ( | ||
| 303 | + -1 + 2 * cos2SigmaM * cos2SigmaM | ||
| 304 | + ) | ||
| 305 | + ) | ||
| 306 | + ) | ||
| 307 | + ); | ||
| 308 | + | ||
| 309 | + } while (Math.abs(lambda-lambdaP) > 1e-12 && --iterLimit>0); | ||
| 310 | + | ||
| 311 | + if (iterLimit === 0) { | ||
| 312 | + return NaN; // formula failed to converge | ||
| 313 | + } | ||
| 314 | + | ||
| 315 | + var uSq = ( | ||
| 316 | + cosSqAlpha * ( | ||
| 317 | + a * a - b * b | ||
| 318 | + ) / ( | ||
| 319 | + b*b | ||
| 320 | + ) | ||
| 321 | + ); | ||
| 322 | + | ||
| 323 | + var A = ( | ||
| 324 | + 1 + uSq / 16384 * ( | ||
| 325 | + 4096 + uSq * ( | ||
| 326 | + -768 + uSq * ( | ||
| 327 | + 320 - 175 * uSq | ||
| 328 | + ) | ||
| 329 | + ) | ||
| 330 | + ) | ||
| 331 | + ); | ||
| 332 | + | ||
| 333 | + var B = ( | ||
| 334 | + uSq / 1024 * ( | ||
| 335 | + 256 + uSq * ( | ||
| 336 | + -128 + uSq * ( | ||
| 337 | + 74-47 * uSq | ||
| 338 | + ) | ||
| 339 | + ) | ||
| 340 | + ) | ||
| 341 | + ); | ||
| 342 | + | ||
| 343 | + var deltaSigma = ( | ||
| 344 | + B * sinSigma * ( | ||
| 345 | + cos2SigmaM + B / 4 * ( | ||
| 346 | + cosSigma * ( | ||
| 347 | + -1 + 2 * cos2SigmaM * cos2SigmaM | ||
| 348 | + ) -B / 6 * cos2SigmaM * ( | ||
| 349 | + -3 + 4 * sinSigma * sinSigma | ||
| 350 | + ) * ( | ||
| 351 | + -3 + 4 * cos2SigmaM * cos2SigmaM | ||
| 352 | + ) | ||
| 353 | + ) | ||
| 354 | + ) | ||
| 355 | + ); | ||
| 356 | + | ||
| 357 | + var distance = b * A * (sigma - deltaSigma); | ||
| 358 | + | ||
| 359 | + distance = distance.toFixed(precision); // round to 1mm precision | ||
| 360 | + | ||
| 361 | + //if (start.hasOwnProperty(elevation) && end.hasOwnProperty(elevation)) { | ||
| 362 | + if (typeof this.elevation(start) !== 'undefined' && typeof this.elevation(end) !== 'undefined') { | ||
| 363 | + var climb = Math.abs(this.elevation(start) - this.elevation(end)); | ||
| 364 | + distance = Math.sqrt(distance * distance + climb * climb); | ||
| 365 | + } | ||
| 366 | + | ||
| 367 | + return this.distance = Math.round(distance * Math.pow(10, precision) / accuracy) * accuracy / Math.pow(10, precision); | ||
| 368 | + | ||
| 369 | + /* | ||
| 370 | + // note: to return initial/final bearings in addition to distance, use something like: | ||
| 371 | + var fwdAz = Math.atan2(cosU2*sinLambda, cosU1*sinU2-sinU1*cosU2*cosLambda); | ||
| 372 | + var revAz = Math.atan2(cosU1*sinLambda, -sinU1*cosU2+cosU1*sinU2*cosLambda); | ||
| 373 | + | ||
| 374 | + return { distance: s, initialBearing: fwdAz.toDeg(), finalBearing: revAz.toDeg() }; | ||
| 375 | + */ | ||
| 376 | + | ||
| 377 | + }, | ||
| 378 | + | ||
| 379 | + | ||
| 380 | + /** | ||
| 381 | + * Calculates the distance between two spots. | ||
| 382 | + * This method is more simple but also far more inaccurate | ||
| 383 | + * | ||
| 384 | + * @param object Start position {latitude: 123, longitude: 123} | ||
| 385 | + * @param object End position {latitude: 123, longitude: 123} | ||
| 386 | + * @param integer Accuracy (in meters) | ||
| 387 | + * @return integer Distance (in meters) | ||
| 388 | + */ | ||
| 389 | + getDistanceSimple: function(start, end, accuracy) { | ||
| 390 | + | ||
| 391 | + accuracy = Math.floor(accuracy) || 1; | ||
| 392 | + | ||
| 393 | + var distance = | ||
| 394 | + Math.round( | ||
| 395 | + Math.acos( | ||
| 396 | + Math.sin( | ||
| 397 | + this.latitude(end).toRad() | ||
| 398 | + ) * | ||
| 399 | + Math.sin( | ||
| 400 | + this.latitude(start).toRad() | ||
| 401 | + ) + | ||
| 402 | + Math.cos( | ||
| 403 | + this.latitude(end).toRad() | ||
| 404 | + ) * | ||
| 405 | + Math.cos( | ||
| 406 | + this.latitude(start).toRad() | ||
| 407 | + ) * | ||
| 408 | + Math.cos( | ||
| 409 | + this.longitude(start).toRad() - this.longitude(end).toRad() | ||
| 410 | + ) | ||
| 411 | + ) * this.radius | ||
| 412 | + ); | ||
| 413 | + | ||
| 414 | + return geolib.distance = Math.floor(Math.round(distance/accuracy)*accuracy); | ||
| 415 | + | ||
| 416 | + }, | ||
| 417 | + | ||
| 418 | + | ||
| 419 | + /** | ||
| 420 | + * Calculates the center of a collection of geo coordinates | ||
| 421 | + * | ||
| 422 | + * @param array Collection of coords [{latitude: 51.510, longitude: 7.1321}, {latitude: 49.1238, longitude: "8° 30' W"}, ...] | ||
| 423 | + * @return object {latitude: centerLat, longitude: centerLng} | ||
| 424 | + */ | ||
| 425 | + getCenter: function(coords) { | ||
| 426 | + | ||
| 427 | + var coordsArray = coords; | ||
| 428 | + if(typeof coords === 'object' && !(coords instanceof Array)) { | ||
| 429 | + | ||
| 430 | + coordsArray = []; | ||
| 431 | + | ||
| 432 | + for(var key in coords) { | ||
| 433 | + coordsArray.push( | ||
| 434 | + this.coords(coords[key]) | ||
| 435 | + ); | ||
| 436 | + } | ||
| 437 | + | ||
| 438 | + } | ||
| 439 | + | ||
| 440 | + if(!coordsArray.length) { | ||
| 441 | + return false; | ||
| 442 | + } | ||
| 443 | + | ||
| 444 | + var X = 0.0; | ||
| 445 | + var Y = 0.0; | ||
| 446 | + var Z = 0.0; | ||
| 447 | + var lat, lon, hyp; | ||
| 448 | + | ||
| 449 | + coordsArray.forEach(function(coord) { | ||
| 450 | + | ||
| 451 | + lat = this.latitude(coord).toRad(); | ||
| 452 | + lon = this.longitude(coord).toRad(); | ||
| 453 | + | ||
| 454 | + X += Math.cos(lat) * Math.cos(lon); | ||
| 455 | + Y += Math.cos(lat) * Math.sin(lon); | ||
| 456 | + Z += Math.sin(lat); | ||
| 457 | + | ||
| 458 | + }, this); | ||
| 459 | + | ||
| 460 | + var nb_coords = coordsArray.length; | ||
| 461 | + X = X / nb_coords; | ||
| 462 | + Y = Y / nb_coords; | ||
| 463 | + Z = Z / nb_coords; | ||
| 464 | + | ||
| 465 | + lon = Math.atan2(Y, X); | ||
| 466 | + hyp = Math.sqrt(X * X + Y * Y); | ||
| 467 | + lat = Math.atan2(Z, hyp); | ||
| 468 | + | ||
| 469 | + return { | ||
| 470 | + latitude: (lat * Geolib.TO_DEG).toFixed(6), | ||
| 471 | + longitude: (lon * Geolib.TO_DEG).toFixed(6) | ||
| 472 | + }; | ||
| 473 | + | ||
| 474 | + }, | ||
| 475 | + | ||
| 476 | + | ||
| 477 | + /** | ||
| 478 | + * Gets the max and min, latitude, longitude, and elevation (if provided). | ||
| 479 | + * @param array array with coords e.g. [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 480 | + * @return object {maxLat: maxLat, | ||
| 481 | + * minLat: minLat | ||
| 482 | + * maxLng: maxLng, | ||
| 483 | + * minLng: minLng, | ||
| 484 | + * maxElev: maxElev, | ||
| 485 | + * minElev: minElev} | ||
| 486 | + */ | ||
| 487 | + getBounds: function(coords) { | ||
| 488 | + | ||
| 489 | + if (!coords.length) { | ||
| 490 | + return false; | ||
| 491 | + } | ||
| 492 | + | ||
| 493 | + var useElevation = this.elevation(coords[0]); | ||
| 494 | + | ||
| 495 | + var stats = { | ||
| 496 | + maxLat: -Infinity, | ||
| 497 | + minLat: Infinity, | ||
| 498 | + maxLng: -Infinity, | ||
| 499 | + minLng: Infinity | ||
| 500 | + }; | ||
| 501 | + | ||
| 502 | + if (typeof useElevation != 'undefined') { | ||
| 503 | + stats.maxElev = 0; | ||
| 504 | + stats.minElev = Infinity; | ||
| 505 | + } | ||
| 506 | + | ||
| 507 | + for (var i = 0, l = coords.length; i < l; ++i) { | ||
| 508 | + | ||
| 509 | + stats.maxLat = Math.max(this.latitude(coords[i]), stats.maxLat); | ||
| 510 | + stats.minLat = Math.min(this.latitude(coords[i]), stats.minLat); | ||
| 511 | + stats.maxLng = Math.max(this.longitude(coords[i]), stats.maxLng); | ||
| 512 | + stats.minLng = Math.min(this.longitude(coords[i]), stats.minLng); | ||
| 513 | + | ||
| 514 | + if (useElevation) { | ||
| 515 | + stats.maxElev = Math.max(this.elevation(coords[i]), stats.maxElev); | ||
| 516 | + stats.minElev = Math.min(this.elevation(coords[i]), stats.minElev); | ||
| 517 | + } | ||
| 518 | + | ||
| 519 | + } | ||
| 520 | + | ||
| 521 | + return stats; | ||
| 522 | + | ||
| 523 | + }, | ||
| 524 | + | ||
| 525 | + /** | ||
| 526 | + * Calculates the center of the bounds of geo coordinates. | ||
| 527 | + * | ||
| 528 | + * On polygons like political borders (eg. states) | ||
| 529 | + * this may gives a closer result to human expectation, than `getCenter`, | ||
| 530 | + * because that function can be disturbed by uneven distribution of | ||
| 531 | + * point in different sides. | ||
| 532 | + * Imagine the US state Oklahoma: `getCenter` on that gives a southern | ||
| 533 | + * point, because the southern border contains a lot more nodes, | ||
| 534 | + * than the others. | ||
| 535 | + * | ||
| 536 | + * @param array Collection of coords [{latitude: 51.510, longitude: 7.1321}, {latitude: 49.1238, longitude: "8° 30' W"}, ...] | ||
| 537 | + * @return object {latitude: centerLat, longitude: centerLng} | ||
| 538 | + */ | ||
| 539 | + getCenterOfBounds: function(coords) { | ||
| 540 | + var b = this.getBounds(coords); | ||
| 541 | + var latitude = b.minLat + ((b.maxLat - b.minLat) / 2); | ||
| 542 | + var longitude = b.minLng + ((b.maxLng - b.minLng) / 2); | ||
| 543 | + return { | ||
| 544 | + latitude: parseFloat(latitude.toFixed(6)), | ||
| 545 | + longitude: parseFloat(longitude.toFixed(6)) | ||
| 546 | + }; | ||
| 547 | + }, | ||
| 548 | + | ||
| 549 | + | ||
| 550 | + /** | ||
| 551 | + * Computes the bounding coordinates of all points on the surface | ||
| 552 | + * of the earth less than or equal to the specified great circle | ||
| 553 | + * distance. | ||
| 554 | + * | ||
| 555 | + * @param object Point position {latitude: 123, longitude: 123} | ||
| 556 | + * @param number Distance (in meters). | ||
| 557 | + * @return array Collection of two points defining the SW and NE corners. | ||
| 558 | + */ | ||
| 559 | + getBoundsOfDistance: function(point, distance) { | ||
| 560 | + | ||
| 561 | + var latitude = this.latitude(point); | ||
| 562 | + var longitude = this.longitude(point); | ||
| 563 | + | ||
| 564 | + var radLat = latitude.toRad(); | ||
| 565 | + var radLon = longitude.toRad(); | ||
| 566 | + | ||
| 567 | + var radDist = distance / this.radius; | ||
| 568 | + var minLat = radLat - radDist; | ||
| 569 | + var maxLat = radLat + radDist; | ||
| 570 | + | ||
| 571 | + var MAX_LAT_RAD = this.maxLat.toRad(); | ||
| 572 | + var MIN_LAT_RAD = this.minLat.toRad(); | ||
| 573 | + var MAX_LON_RAD = this.maxLon.toRad(); | ||
| 574 | + var MIN_LON_RAD = this.minLon.toRad(); | ||
| 575 | + | ||
| 576 | + var minLon; | ||
| 577 | + var maxLon; | ||
| 578 | + | ||
| 579 | + if (minLat > MIN_LAT_RAD && maxLat < MAX_LAT_RAD) { | ||
| 580 | + | ||
| 581 | + var deltaLon = Math.asin(Math.sin(radDist) / Math.cos(radLat)); | ||
| 582 | + minLon = radLon - deltaLon; | ||
| 583 | + | ||
| 584 | + if (minLon < MIN_LON_RAD) { | ||
| 585 | + minLon += Geolib.PI_X2; | ||
| 586 | + } | ||
| 587 | + | ||
| 588 | + maxLon = radLon + deltaLon; | ||
| 589 | + | ||
| 590 | + if (maxLon > MAX_LON_RAD) { | ||
| 591 | + maxLon -= Geolib.PI_X2; | ||
| 592 | + } | ||
| 593 | + | ||
| 594 | + } else { | ||
| 595 | + // A pole is within the distance. | ||
| 596 | + minLat = Math.max(minLat, MIN_LAT_RAD); | ||
| 597 | + maxLat = Math.min(maxLat, MAX_LAT_RAD); | ||
| 598 | + minLon = MIN_LON_RAD; | ||
| 599 | + maxLon = MAX_LON_RAD; | ||
| 600 | + } | ||
| 601 | + | ||
| 602 | + return [ | ||
| 603 | + // Southwest | ||
| 604 | + { | ||
| 605 | + latitude: minLat.toDeg(), | ||
| 606 | + longitude: minLon.toDeg() | ||
| 607 | + }, | ||
| 608 | + // Northeast | ||
| 609 | + { | ||
| 610 | + latitude: maxLat.toDeg(), | ||
| 611 | + longitude: maxLon.toDeg() | ||
| 612 | + } | ||
| 613 | + ]; | ||
| 614 | + | ||
| 615 | + }, | ||
| 616 | + | ||
| 617 | + | ||
| 618 | + /** | ||
| 619 | + * Checks whether a point is inside of a polygon or not. | ||
| 620 | + * Note that the polygon coords must be in correct order! | ||
| 621 | + * | ||
| 622 | + * @param object coordinate to check e.g. {latitude: 51.5023, longitude: 7.3815} | ||
| 623 | + * @param array array with coords e.g. [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 624 | + * @return bool true if the coordinate is inside the given polygon | ||
| 625 | + */ | ||
| 626 | + isPointInside: function(latlng, coords) { | ||
| 627 | + | ||
| 628 | + for(var c = false, i = -1, l = coords.length, j = l - 1; ++i < l; j = i) { | ||
| 629 | + | ||
| 630 | + if( | ||
| 631 | + ( | ||
| 632 | + (this.longitude(coords[i]) <= this.longitude(latlng) && this.longitude(latlng) < this.longitude(coords[j])) || | ||
| 633 | + (this.longitude(coords[j]) <= this.longitude(latlng) && this.longitude(latlng) < this.longitude(coords[i])) | ||
| 634 | + ) && | ||
| 635 | + ( | ||
| 636 | + this.latitude(latlng) < (this.latitude(coords[j]) - this.latitude(coords[i])) * | ||
| 637 | + (this.longitude(latlng) - this.longitude(coords[i])) / | ||
| 638 | + (this.longitude(coords[j]) - this.longitude(coords[i])) + | ||
| 639 | + this.latitude(coords[i]) | ||
| 640 | + ) | ||
| 641 | + ) { | ||
| 642 | + c = !c; | ||
| 643 | + } | ||
| 644 | + | ||
| 645 | + } | ||
| 646 | + | ||
| 647 | + return c; | ||
| 648 | + | ||
| 649 | + }, | ||
| 650 | + | ||
| 651 | + | ||
| 652 | + /** | ||
| 653 | + * Pre calculate the polygon coords, to speed up the point inside check. | ||
| 654 | + * Use this function before calling isPointInsideWithPreparedPolygon() | ||
| 655 | + * @see Algorythm from http://alienryderflex.com/polygon/ | ||
| 656 | + * @param array array with coords e.g. [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 657 | + */ | ||
| 658 | + preparePolygonForIsPointInsideOptimized: function(coords) { | ||
| 659 | + | ||
| 660 | + for(var i = 0, j = coords.length-1; i < coords.length; i++) { | ||
| 661 | + | ||
| 662 | + if(this.longitude(coords[j]) === this.longitude(coords[i])) { | ||
| 663 | + | ||
| 664 | + coords[i].constant = this.latitude(coords[i]); | ||
| 665 | + coords[i].multiple = 0; | ||
| 666 | + | ||
| 667 | + } else { | ||
| 668 | + | ||
| 669 | + coords[i].constant = this.latitude(coords[i]) - ( | ||
| 670 | + this.longitude(coords[i]) * this.latitude(coords[j]) | ||
| 671 | + ) / ( | ||
| 672 | + this.longitude(coords[j]) - this.longitude(coords[i]) | ||
| 673 | + ) + ( | ||
| 674 | + this.longitude(coords[i])*this.latitude(coords[i]) | ||
| 675 | + ) / ( | ||
| 676 | + this.longitude(coords[j])-this.longitude(coords[i]) | ||
| 677 | + ); | ||
| 678 | + | ||
| 679 | + coords[i].multiple = ( | ||
| 680 | + this.latitude(coords[j])-this.latitude(coords[i]) | ||
| 681 | + ) / ( | ||
| 682 | + this.longitude(coords[j])-this.longitude(coords[i]) | ||
| 683 | + ); | ||
| 684 | + | ||
| 685 | + } | ||
| 686 | + | ||
| 687 | + j=i; | ||
| 688 | + | ||
| 689 | + } | ||
| 690 | + | ||
| 691 | + }, | ||
| 692 | + | ||
| 693 | + /** | ||
| 694 | + * Checks whether a point is inside of a polygon or not. | ||
| 695 | + * "This is useful if you have many points that need to be tested against the same (static) polygon." | ||
| 696 | + * Please call the function preparePolygonForIsPointInsideOptimized() with the same coords object before using this function. | ||
| 697 | + * Note that the polygon coords must be in correct order! | ||
| 698 | + * | ||
| 699 | + * @see Algorythm from http://alienryderflex.com/polygon/ | ||
| 700 | + * | ||
| 701 | + * @param object coordinate to check e.g. {latitude: 51.5023, longitude: 7.3815} | ||
| 702 | + * @param array array with coords e.g. [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 703 | + * @return bool true if the coordinate is inside the given polygon | ||
| 704 | + */ | ||
| 705 | + isPointInsideWithPreparedPolygon: function(point, coords) { | ||
| 706 | + | ||
| 707 | + var flgPointInside = false, | ||
| 708 | + y = this.longitude(point), | ||
| 709 | + x = this.latitude(point); | ||
| 710 | + | ||
| 711 | + for(var i = 0, j = coords.length-1; i < coords.length; i++) { | ||
| 712 | + | ||
| 713 | + if ((this.longitude(coords[i]) < y && this.longitude(coords[j]) >=y || | ||
| 714 | + this.longitude(coords[j]) < y && this.longitude(coords[i]) >= y)) { | ||
| 715 | + | ||
| 716 | + flgPointInside^=(y*coords[i].multiple+coords[i].constant < x); | ||
| 717 | + | ||
| 718 | + } | ||
| 719 | + | ||
| 720 | + j=i; | ||
| 721 | + | ||
| 722 | + } | ||
| 723 | + | ||
| 724 | + return flgPointInside; | ||
| 725 | + | ||
| 726 | + }, | ||
| 727 | + | ||
| 728 | + | ||
| 729 | + /** | ||
| 730 | + * Shortcut for geolib.isPointInside() | ||
| 731 | + */ | ||
| 732 | + isInside: function() { | ||
| 733 | + return this.isPointInside.apply(this, arguments); | ||
| 734 | + }, | ||
| 735 | + | ||
| 736 | + | ||
| 737 | + /** | ||
| 738 | + * Checks whether a point is inside of a circle or not. | ||
| 739 | + * | ||
| 740 | + * @param object coordinate to check (e.g. {latitude: 51.5023, longitude: 7.3815}) | ||
| 741 | + * @param object coordinate of the circle's center (e.g. {latitude: 51.4812, longitude: 7.4025}) | ||
| 742 | + * @param integer maximum radius in meters | ||
| 743 | + * @return bool true if the coordinate is within the given radius | ||
| 744 | + */ | ||
| 745 | + isPointInCircle: function(latlng, center, radius) { | ||
| 746 | + return this.getDistance(latlng, center) < radius; | ||
| 747 | + }, | ||
| 748 | + | ||
| 749 | + | ||
| 750 | + /** | ||
| 751 | + * Shortcut for geolib.isPointInCircle() | ||
| 752 | + */ | ||
| 753 | + withinRadius: function() { | ||
| 754 | + return this.isPointInCircle.apply(this, arguments); | ||
| 755 | + }, | ||
| 756 | + | ||
| 757 | + | ||
| 758 | + /** | ||
| 759 | + * Gets rhumb line bearing of two points. Find out about the difference between rhumb line and | ||
| 760 | + * great circle bearing on Wikipedia. It's quite complicated. Rhumb line should be fine in most cases: | ||
| 761 | + * | ||
| 762 | + * http://en.wikipedia.org/wiki/Rhumb_line#General_and_mathematical_description | ||
| 763 | + * | ||
| 764 | + * Function heavily based on Doug Vanderweide's great PHP version (licensed under GPL 3.0) | ||
| 765 | + * http://www.dougv.com/2009/07/13/calculating-the-bearing-and-compass-rose-direction-between-two-latitude-longitude-coordinates-in-php/ | ||
| 766 | + * | ||
| 767 | + * @param object origin coordinate (e.g. {latitude: 51.5023, longitude: 7.3815}) | ||
| 768 | + * @param object destination coordinate | ||
| 769 | + * @return integer calculated bearing | ||
| 770 | + */ | ||
| 771 | + getRhumbLineBearing: function(originLL, destLL) { | ||
| 772 | + | ||
| 773 | + // difference of longitude coords | ||
| 774 | + var diffLon = this.longitude(destLL).toRad() - this.longitude(originLL).toRad(); | ||
| 775 | + | ||
| 776 | + // difference latitude coords phi | ||
| 777 | + var diffPhi = Math.log( | ||
| 778 | + Math.tan( | ||
| 779 | + this.latitude(destLL).toRad() / 2 + Geolib.PI_DIV4 | ||
| 780 | + ) / | ||
| 781 | + Math.tan( | ||
| 782 | + this.latitude(originLL).toRad() / 2 + Geolib.PI_DIV4 | ||
| 783 | + ) | ||
| 784 | + ); | ||
| 785 | + | ||
| 786 | + // recalculate diffLon if it is greater than pi | ||
| 787 | + if(Math.abs(diffLon) > Math.PI) { | ||
| 788 | + if(diffLon > 0) { | ||
| 789 | + diffLon = (Geolib.PI_X2 - diffLon) * -1; | ||
| 790 | + } | ||
| 791 | + else { | ||
| 792 | + diffLon = Geolib.PI_X2 + diffLon; | ||
| 793 | + } | ||
| 794 | + } | ||
| 795 | + | ||
| 796 | + //return the angle, normalized | ||
| 797 | + return (Math.atan2(diffLon, diffPhi).toDeg() + 360) % 360; | ||
| 798 | + | ||
| 799 | + }, | ||
| 800 | + | ||
| 801 | + | ||
| 802 | + /** | ||
| 803 | + * Gets great circle bearing of two points. See description of getRhumbLineBearing for more information | ||
| 804 | + * | ||
| 805 | + * @param object origin coordinate (e.g. {latitude: 51.5023, longitude: 7.3815}) | ||
| 806 | + * @param object destination coordinate | ||
| 807 | + * @return integer calculated bearing | ||
| 808 | + */ | ||
| 809 | + getBearing: function(originLL, destLL) { | ||
| 810 | + | ||
| 811 | + destLL['latitude'] = this.latitude(destLL); | ||
| 812 | + destLL['longitude'] = this.longitude(destLL); | ||
| 813 | + originLL['latitude'] = this.latitude(originLL); | ||
| 814 | + originLL['longitude'] = this.longitude(originLL); | ||
| 815 | + | ||
| 816 | + var bearing = ( | ||
| 817 | + ( | ||
| 818 | + Math.atan2( | ||
| 819 | + Math.sin( | ||
| 820 | + destLL['longitude'].toRad() - | ||
| 821 | + originLL['longitude'].toRad() | ||
| 822 | + ) * | ||
| 823 | + Math.cos( | ||
| 824 | + destLL['latitude'].toRad() | ||
| 825 | + ), | ||
| 826 | + Math.cos( | ||
| 827 | + originLL['latitude'].toRad() | ||
| 828 | + ) * | ||
| 829 | + Math.sin( | ||
| 830 | + destLL['latitude'].toRad() | ||
| 831 | + ) - | ||
| 832 | + Math.sin( | ||
| 833 | + originLL['latitude'].toRad() | ||
| 834 | + ) * | ||
| 835 | + Math.cos( | ||
| 836 | + destLL['latitude'].toRad() | ||
| 837 | + ) * | ||
| 838 | + Math.cos( | ||
| 839 | + destLL['longitude'].toRad() - originLL['longitude'].toRad() | ||
| 840 | + ) | ||
| 841 | + ) | ||
| 842 | + ).toDeg() + 360 | ||
| 843 | + ) % 360; | ||
| 844 | + | ||
| 845 | + return bearing; | ||
| 846 | + | ||
| 847 | + }, | ||
| 848 | + | ||
| 849 | + | ||
| 850 | + /** | ||
| 851 | + * Gets the compass direction from an origin coordinate to a destination coordinate. | ||
| 852 | + * | ||
| 853 | + * @param object origin coordinate (e.g. {latitude: 51.5023, longitude: 7.3815}) | ||
| 854 | + * @param object destination coordinate | ||
| 855 | + * @param string Bearing mode. Can be either circle or rhumbline | ||
| 856 | + * @return object Returns an object with a rough (NESW) and an exact direction (NNE, NE, ENE, E, ESE, etc). | ||
| 857 | + */ | ||
| 858 | + getCompassDirection: function(originLL, destLL, bearingMode) { | ||
| 859 | + | ||
| 860 | + var direction; | ||
| 861 | + var bearing; | ||
| 862 | + | ||
| 863 | + if(bearingMode == 'circle') { | ||
| 864 | + // use great circle bearing | ||
| 865 | + bearing = this.getBearing(originLL, destLL); | ||
| 866 | + } else { | ||
| 867 | + // default is rhumb line bearing | ||
| 868 | + bearing = this.getRhumbLineBearing(originLL, destLL); | ||
| 869 | + } | ||
| 870 | + | ||
| 871 | + switch(Math.round(bearing/22.5)) { | ||
| 872 | + case 1: | ||
| 873 | + direction = {exact: "NNE", rough: "N"}; | ||
| 874 | + break; | ||
| 875 | + case 2: | ||
| 876 | + direction = {exact: "NE", rough: "N"}; | ||
| 877 | + break; | ||
| 878 | + case 3: | ||
| 879 | + direction = {exact: "ENE", rough: "E"}; | ||
| 880 | + break; | ||
| 881 | + case 4: | ||
| 882 | + direction = {exact: "E", rough: "E"}; | ||
| 883 | + break; | ||
| 884 | + case 5: | ||
| 885 | + direction = {exact: "ESE", rough: "E"}; | ||
| 886 | + break; | ||
| 887 | + case 6: | ||
| 888 | + direction = {exact: "SE", rough: "E"}; | ||
| 889 | + break; | ||
| 890 | + case 7: | ||
| 891 | + direction = {exact: "SSE", rough: "S"}; | ||
| 892 | + break; | ||
| 893 | + case 8: | ||
| 894 | + direction = {exact: "S", rough: "S"}; | ||
| 895 | + break; | ||
| 896 | + case 9: | ||
| 897 | + direction = {exact: "SSW", rough: "S"}; | ||
| 898 | + break; | ||
| 899 | + case 10: | ||
| 900 | + direction = {exact: "SW", rough: "S"}; | ||
| 901 | + break; | ||
| 902 | + case 11: | ||
| 903 | + direction = {exact: "WSW", rough: "W"}; | ||
| 904 | + break; | ||
| 905 | + case 12: | ||
| 906 | + direction = {exact: "W", rough: "W"}; | ||
| 907 | + break; | ||
| 908 | + case 13: | ||
| 909 | + direction = {exact: "WNW", rough: "W"}; | ||
| 910 | + break; | ||
| 911 | + case 14: | ||
| 912 | + direction = {exact: "NW", rough: "W"}; | ||
| 913 | + break; | ||
| 914 | + case 15: | ||
| 915 | + direction = {exact: "NNW", rough: "N"}; | ||
| 916 | + break; | ||
| 917 | + default: | ||
| 918 | + direction = {exact: "N", rough: "N"}; | ||
| 919 | + } | ||
| 920 | + | ||
| 921 | + direction['bearing'] = bearing; | ||
| 922 | + return direction; | ||
| 923 | + | ||
| 924 | + }, | ||
| 925 | + | ||
| 926 | + | ||
| 927 | + /** | ||
| 928 | + * Shortcut for getCompassDirection | ||
| 929 | + */ | ||
| 930 | + getDirection: function(originLL, destLL, bearingMode) { | ||
| 931 | + return this.getCompassDirection.apply(this, arguments); | ||
| 932 | + }, | ||
| 933 | + | ||
| 934 | + | ||
| 935 | + /** | ||
| 936 | + * Sorts an array of coords by distance from a reference coordinate | ||
| 937 | + * | ||
| 938 | + * @param object reference coordinate e.g. {latitude: 51.5023, longitude: 7.3815} | ||
| 939 | + * @param mixed array or object with coords [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 940 | + * @return array ordered array | ||
| 941 | + */ | ||
| 942 | + orderByDistance: function(latlng, coords) { | ||
| 943 | + | ||
| 944 | + var coordsArray = []; | ||
| 945 | + | ||
| 946 | + for(var coord in coords) { | ||
| 947 | + | ||
| 948 | + var distance = this.getDistance(latlng, coords[coord]); | ||
| 949 | + var augmentedCoord = Object.create(coords[coord]); | ||
| 950 | + augmentedCoord.distance = distance; | ||
| 951 | + augmentedCoord.key = coord; | ||
| 952 | + | ||
| 953 | + coordsArray.push(augmentedCoord); | ||
| 954 | + | ||
| 955 | + } | ||
| 956 | + | ||
| 957 | + return coordsArray.sort(function(a, b) { | ||
| 958 | + return a.distance - b.distance; | ||
| 959 | + }); | ||
| 960 | + | ||
| 961 | + }, | ||
| 962 | + | ||
| 963 | + /** | ||
| 964 | + * Check if a point lies in line created by two other points | ||
| 965 | + * | ||
| 966 | + * @param object Point to check: {latitude: 123, longitude: 123} | ||
| 967 | + * @param object Start of line {latitude: 123, longitude: 123} | ||
| 968 | + * @param object End of line {latitude: 123, longitude: 123} | ||
| 969 | + * @return boolean | ||
| 970 | + */ | ||
| 971 | + isPointInLine: function(point, start, end) { | ||
| 972 | + | ||
| 973 | + return (this.getDistance(start, point, 1, 3)+this.getDistance(point, end, 1, 3)).toFixed(3)==this.getDistance(start, end, 1, 3); | ||
| 974 | + }, | ||
| 975 | + | ||
| 976 | + /** | ||
| 977 | + * Check if a point lies within a given distance from a line created by two other points | ||
| 978 | + * | ||
| 979 | + * @param object Point to check: {latitude: 123, longitude: 123} | ||
| 980 | + * @param object Start of line {latitude: 123, longitude: 123} | ||
| 981 | + * @param object End of line {latitude: 123, longitude: 123} | ||
| 982 | + * @pararm float maximum distance from line | ||
| 983 | + * @return boolean | ||
| 984 | + */ | ||
| 985 | + isPointNearLine: function(point, start, end, distance) { | ||
| 986 | + return this.getDistanceFromLine(point, start, end) < distance; | ||
| 987 | + }, | ||
| 988 | + | ||
| 989 | + /** | ||
| 990 | + * return the minimum distance from a point to a line | ||
| 991 | + * | ||
| 992 | + * @param object Point away from line | ||
| 993 | + * @param object Start of line {latitude: 123, longitude: 123} | ||
| 994 | + * @param object End of line {latitude: 123, longitude: 123} | ||
| 995 | + * @return float distance from point to line | ||
| 996 | + */ | ||
| 997 | + getDistanceFromLine: function(point, start, end) { | ||
| 998 | + var d1 = this.getDistance(start, point, 1, 3); | ||
| 999 | + var d2 = this.getDistance(point, end, 1, 3); | ||
| 1000 | + var d3 = this.getDistance(start, end, 1, 3); | ||
| 1001 | + var distance = 0; | ||
| 1002 | + | ||
| 1003 | + // alpha is the angle between the line from start to point, and from start to end // | ||
| 1004 | + var alpha = Math.acos((d1*d1 + d3*d3 - d2*d2)/(2*d1*d3)); | ||
| 1005 | + // beta is the angle between the line from end to point and from end to start // | ||
| 1006 | + var beta = Math.acos((d2*d2 + d3*d3 - d1*d1)/(2*d2*d3)); | ||
| 1007 | + | ||
| 1008 | + // if the angle is greater than 90 degrees, then the minimum distance is the | ||
| 1009 | + // line from the start to the point // | ||
| 1010 | + if(alpha>Math.PI/2) { | ||
| 1011 | + distance = d1; | ||
| 1012 | + } | ||
| 1013 | + // same for the beta // | ||
| 1014 | + else if(beta > Math.PI/2) { | ||
| 1015 | + distance = d2; | ||
| 1016 | + } | ||
| 1017 | + // otherwise the minimum distance is achieved through a line perpendular to the start-end line, | ||
| 1018 | + // which goes from the start-end line to the point // | ||
| 1019 | + else { | ||
| 1020 | + distance = Math.sin(alpha) * d1; | ||
| 1021 | + } | ||
| 1022 | + | ||
| 1023 | + return distance; | ||
| 1024 | + }, | ||
| 1025 | + | ||
| 1026 | + /** | ||
| 1027 | + * Finds the nearest coordinate to a reference coordinate | ||
| 1028 | + * | ||
| 1029 | + * @param object reference coordinate e.g. {latitude: 51.5023, longitude: 7.3815} | ||
| 1030 | + * @param mixed array or object with coords [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 1031 | + * @return array ordered array | ||
| 1032 | + */ | ||
| 1033 | + findNearest: function(latlng, coords, offset, limit) { | ||
| 1034 | + | ||
| 1035 | + offset = offset || 0; | ||
| 1036 | + limit = limit || 1; | ||
| 1037 | + var ordered = this.orderByDistance(latlng, coords); | ||
| 1038 | + | ||
| 1039 | + if(limit === 1) { | ||
| 1040 | + return ordered[offset]; | ||
| 1041 | + } else { | ||
| 1042 | + return ordered.splice(offset, limit); | ||
| 1043 | + } | ||
| 1044 | + | ||
| 1045 | + }, | ||
| 1046 | + | ||
| 1047 | + | ||
| 1048 | + /** | ||
| 1049 | + * Calculates the length of a given path | ||
| 1050 | + * | ||
| 1051 | + * @param mixed array or object with coords [{latitude: 51.5143, longitude: 7.4138}, {latitude: 123, longitude: 123}, ...] | ||
| 1052 | + * @return integer length of the path (in meters) | ||
| 1053 | + */ | ||
| 1054 | + getPathLength: function(coords) { | ||
| 1055 | + | ||
| 1056 | + var dist = 0; | ||
| 1057 | + var last; | ||
| 1058 | + | ||
| 1059 | + for (var i = 0, l = coords.length; i < l; ++i) { | ||
| 1060 | + if(last) { | ||
| 1061 | + //console.log(coords[i], last, this.getDistance(coords[i], last)); | ||
| 1062 | + dist += this.getDistance(this.coords(coords[i]), last); | ||
| 1063 | + } | ||
| 1064 | + last = this.coords(coords[i]); | ||
| 1065 | + } | ||
| 1066 | + | ||
| 1067 | + return dist; | ||
| 1068 | + | ||
| 1069 | + }, | ||
| 1070 | + | ||
| 1071 | + | ||
| 1072 | + /** | ||
| 1073 | + * Calculates the speed between to points within a given time span. | ||
| 1074 | + * | ||
| 1075 | + * @param object coords with javascript timestamp {latitude: 51.5143, longitude: 7.4138, time: 1360231200880} | ||
| 1076 | + * @param object coords with javascript timestamp {latitude: 51.5502, longitude: 7.4323, time: 1360245600460} | ||
| 1077 | + * @param object options (currently "unit" is the only option. Default: km(h)); | ||
| 1078 | + * @return float speed in unit per hour | ||
| 1079 | + */ | ||
| 1080 | + getSpeed: function(start, end, options) { | ||
| 1081 | + | ||
| 1082 | + var unit = options && options.unit || 'km'; | ||
| 1083 | + | ||
| 1084 | + if(unit == 'mph') { | ||
| 1085 | + unit = 'mi'; | ||
| 1086 | + } else if(unit == 'kmh') { | ||
| 1087 | + unit = 'km'; | ||
| 1088 | + } | ||
| 1089 | + | ||
| 1090 | + var distance = geolib.getDistance(start, end); | ||
| 1091 | + var time = ((end.time*1)/1000) - ((start.time*1)/1000); | ||
| 1092 | + var mPerHr = (distance/time)*3600; | ||
| 1093 | + var speed = Math.round(mPerHr * this.measures[unit] * 10000)/10000; | ||
| 1094 | + return speed; | ||
| 1095 | + | ||
| 1096 | + }, | ||
| 1097 | + | ||
| 1098 | + | ||
| 1099 | + /** | ||
| 1100 | + * Computes the destination point given an initial point, a distance | ||
| 1101 | + * and a bearing | ||
| 1102 | + * | ||
| 1103 | + * see http://www.movable-type.co.uk/scripts/latlong.html for the original code | ||
| 1104 | + * | ||
| 1105 | + * @param object start coordinate (e.g. {latitude: 51.5023, longitude: 7.3815}) | ||
| 1106 | + * @param float longitude of the inital point in degree | ||
| 1107 | + * @param float distance to go from the inital point in meter | ||
| 1108 | + * @param float bearing in degree of the direction to go, e.g. 0 = north, 180 = south | ||
| 1109 | + * @param float optional (in meter), defaults to mean radius of the earth | ||
| 1110 | + * @return object {latitude: destLat (in degree), longitude: destLng (in degree)} | ||
| 1111 | + */ | ||
| 1112 | + computeDestinationPoint: function(start, distance, bearing, radius) { | ||
| 1113 | + | ||
| 1114 | + var lat = this.latitude(start); | ||
| 1115 | + var lng = this.longitude(start); | ||
| 1116 | + | ||
| 1117 | + radius = (typeof radius === 'undefined') ? this.radius : Number(radius); | ||
| 1118 | + | ||
| 1119 | + var δ = Number(distance) / radius; // angular distance in radians | ||
| 1120 | + var θ = Number(bearing).toRad(); | ||
| 1121 | + | ||
| 1122 | + var φ1 = Number(lat).toRad(); | ||
| 1123 | + var λ1 = Number(lng).toRad(); | ||
| 1124 | + | ||
| 1125 | + var φ2 = Math.asin( Math.sin(φ1)*Math.cos(δ) + | ||
| 1126 | + Math.cos(φ1)*Math.sin(δ)*Math.cos(θ) ); | ||
| 1127 | + var λ2 = λ1 + Math.atan2(Math.sin(θ)*Math.sin(δ)*Math.cos(φ1), | ||
| 1128 | + Math.cos(δ)-Math.sin(φ1)*Math.sin(φ2)); | ||
| 1129 | + λ2 = (λ2+3*Math.PI) % (2*Math.PI) - Math.PI; // normalise to -180..+180° | ||
| 1130 | + | ||
| 1131 | + return { | ||
| 1132 | + latitude: φ2.toDeg(), | ||
| 1133 | + longitude: λ2.toDeg() | ||
| 1134 | + }; | ||
| 1135 | + | ||
| 1136 | + }, | ||
| 1137 | + | ||
| 1138 | + | ||
| 1139 | + /** | ||
| 1140 | + * Converts a distance from meters to km, mm, cm, mi, ft, in or yd | ||
| 1141 | + * | ||
| 1142 | + * @param string Format to be converted in | ||
| 1143 | + * @param float Distance in meters | ||
| 1144 | + * @param float Decimal places for rounding (default: 4) | ||
| 1145 | + * @return float Converted distance | ||
| 1146 | + */ | ||
| 1147 | + convertUnit: function(unit, distance, round) { | ||
| 1148 | + | ||
| 1149 | + if(distance === 0) { | ||
| 1150 | + | ||
| 1151 | + return 0; | ||
| 1152 | + | ||
| 1153 | + } else if(typeof distance === 'undefined') { | ||
| 1154 | + | ||
| 1155 | + if(this.distance === null) { | ||
| 1156 | + throw new Error('No distance was given'); | ||
| 1157 | + } else if(this.distance === 0) { | ||
| 1158 | + return 0; | ||
| 1159 | + } else { | ||
| 1160 | + distance = this.distance; | ||
| 1161 | + } | ||
| 1162 | + | ||
| 1163 | + } | ||
| 1164 | + | ||
| 1165 | + unit = unit || 'm'; | ||
| 1166 | + round = (null == round ? 4 : round); | ||
| 1167 | + | ||
| 1168 | + if(typeof this.measures[unit] !== 'undefined') { | ||
| 1169 | + return this.round(distance * this.measures[unit], round); | ||
| 1170 | + } else { | ||
| 1171 | + throw new Error('Unknown unit for conversion.'); | ||
| 1172 | + } | ||
| 1173 | + | ||
| 1174 | + }, | ||
| 1175 | + | ||
| 1176 | + | ||
| 1177 | + /** | ||
| 1178 | + * Checks if a value is in decimal format or, if neccessary, converts to decimal | ||
| 1179 | + * | ||
| 1180 | + * @param mixed Value(s) to be checked/converted (array of latlng objects, latlng object, sexagesimal string, float) | ||
| 1181 | + * @return float Input data in decimal format | ||
| 1182 | + */ | ||
| 1183 | + useDecimal: function(value) { | ||
| 1184 | + | ||
| 1185 | + if(Object.prototype.toString.call(value) === '[object Array]') { | ||
| 1186 | + | ||
| 1187 | + var geolib = this; | ||
| 1188 | + | ||
| 1189 | + value = value.map(function(val) { | ||
| 1190 | + | ||
| 1191 | + //if(!isNaN(parseFloat(val))) { | ||
| 1192 | + if(geolib.isDecimal(val)) { | ||
| 1193 | + | ||
| 1194 | + return geolib.useDecimal(val); | ||
| 1195 | + | ||
| 1196 | + } else if(typeof val == 'object') { | ||
| 1197 | + | ||
| 1198 | + if(geolib.validate(val)) { | ||
| 1199 | + | ||
| 1200 | + return geolib.coords(val); | ||
| 1201 | + | ||
| 1202 | + } else { | ||
| 1203 | + | ||
| 1204 | + for(var prop in val) { | ||
| 1205 | + val[prop] = geolib.useDecimal(val[prop]); | ||
| 1206 | + } | ||
| 1207 | + | ||
| 1208 | + return val; | ||
| 1209 | + | ||
| 1210 | + } | ||
| 1211 | + | ||
| 1212 | + } else if(geolib.isSexagesimal(val)) { | ||
| 1213 | + | ||
| 1214 | + return geolib.sexagesimal2decimal(val); | ||
| 1215 | + | ||
| 1216 | + } else { | ||
| 1217 | + | ||
| 1218 | + return val; | ||
| 1219 | + | ||
| 1220 | + } | ||
| 1221 | + | ||
| 1222 | + }); | ||
| 1223 | + | ||
| 1224 | + return value; | ||
| 1225 | + | ||
| 1226 | + } else if(typeof value === 'object' && this.validate(value)) { | ||
| 1227 | + | ||
| 1228 | + return this.coords(value); | ||
| 1229 | + | ||
| 1230 | + } else if(typeof value === 'object') { | ||
| 1231 | + | ||
| 1232 | + for(var prop in value) { | ||
| 1233 | + value[prop] = this.useDecimal(value[prop]); | ||
| 1234 | + } | ||
| 1235 | + | ||
| 1236 | + return value; | ||
| 1237 | + | ||
| 1238 | + } | ||
| 1239 | + | ||
| 1240 | + | ||
| 1241 | + if (this.isDecimal(value)) { | ||
| 1242 | + | ||
| 1243 | + return parseFloat(value); | ||
| 1244 | + | ||
| 1245 | + } else if(this.isSexagesimal(value) === true) { | ||
| 1246 | + | ||
| 1247 | + return parseFloat(this.sexagesimal2decimal(value)); | ||
| 1248 | + | ||
| 1249 | + } | ||
| 1250 | + | ||
| 1251 | + throw new Error('Unknown format.'); | ||
| 1252 | + | ||
| 1253 | + }, | ||
| 1254 | + | ||
| 1255 | + /** | ||
| 1256 | + * Converts a decimal coordinate value to sexagesimal format | ||
| 1257 | + * | ||
| 1258 | + * @param float decimal | ||
| 1259 | + * @return string Sexagesimal value (XX° YY' ZZ") | ||
| 1260 | + */ | ||
| 1261 | + decimal2sexagesimal: function(dec) { | ||
| 1262 | + | ||
| 1263 | + if (dec in this.sexagesimal) { | ||
| 1264 | + return this.sexagesimal[dec]; | ||
| 1265 | + } | ||
| 1266 | + | ||
| 1267 | + var tmp = dec.toString().split('.'); | ||
| 1268 | + | ||
| 1269 | + var deg = Math.abs(tmp[0]); | ||
| 1270 | + var min = ('0.' + (tmp[1] || 0))*60; | ||
| 1271 | + var sec = min.toString().split('.'); | ||
| 1272 | + | ||
| 1273 | + min = Math.floor(min); | ||
| 1274 | + sec = (('0.' + (sec[1] || 0)) * 60).toFixed(2); | ||
| 1275 | + | ||
| 1276 | + this.sexagesimal[dec] = (deg + '° ' + min + "' " + sec + '"'); | ||
| 1277 | + | ||
| 1278 | + return this.sexagesimal[dec]; | ||
| 1279 | + | ||
| 1280 | + }, | ||
| 1281 | + | ||
| 1282 | + | ||
| 1283 | + /** | ||
| 1284 | + * Converts a sexagesimal coordinate to decimal format | ||
| 1285 | + * | ||
| 1286 | + * @param float Sexagesimal coordinate | ||
| 1287 | + * @return string Decimal value (XX.XXXXXXXX) | ||
| 1288 | + */ | ||
| 1289 | + sexagesimal2decimal: function(sexagesimal) { | ||
| 1290 | + | ||
| 1291 | + if (sexagesimal in this.decimal) { | ||
| 1292 | + return this.decimal[sexagesimal]; | ||
| 1293 | + } | ||
| 1294 | + | ||
| 1295 | + var regEx = new RegExp(this.sexagesimalPattern); | ||
| 1296 | + var data = regEx.exec(sexagesimal); | ||
| 1297 | + var min = 0, sec = 0; | ||
| 1298 | + | ||
| 1299 | + if(data) { | ||
| 1300 | + min = parseFloat(data[2]/60); | ||
| 1301 | + sec = parseFloat(data[4]/3600) || 0; | ||
| 1302 | + } | ||
| 1303 | + | ||
| 1304 | + var dec = ((parseFloat(data[1]) + min + sec)).toFixed(8); | ||
| 1305 | + //var dec = ((parseFloat(data[1]) + min + sec)); | ||
| 1306 | + | ||
| 1307 | + // South and West are negative decimals | ||
| 1308 | + dec = (data[7] == 'S' || data[7] == 'W') ? parseFloat(-dec) : parseFloat(dec); | ||
| 1309 | + //dec = (data[7] == 'S' || data[7] == 'W') ? -dec : dec; | ||
| 1310 | + | ||
| 1311 | + this.decimal[sexagesimal] = dec; | ||
| 1312 | + | ||
| 1313 | + return dec; | ||
| 1314 | + | ||
| 1315 | + }, | ||
| 1316 | + | ||
| 1317 | + | ||
| 1318 | + /** | ||
| 1319 | + * Checks if a value is in decimal format | ||
| 1320 | + * | ||
| 1321 | + * @param string Value to be checked | ||
| 1322 | + * @return bool True if in sexagesimal format | ||
| 1323 | + */ | ||
| 1324 | + isDecimal: function(value) { | ||
| 1325 | + | ||
| 1326 | + value = value.toString().replace(/\s*/, ''); | ||
| 1327 | + | ||
| 1328 | + // looks silly but works as expected | ||
| 1329 | + // checks if value is in decimal format | ||
| 1330 | + return (!isNaN(parseFloat(value)) && parseFloat(value) == value); | ||
| 1331 | + | ||
| 1332 | + }, | ||
| 1333 | + | ||
| 1334 | + | ||
| 1335 | + /** | ||
| 1336 | + * Checks if a value is in sexagesimal format | ||
| 1337 | + * | ||
| 1338 | + * @param string Value to be checked | ||
| 1339 | + * @return bool True if in sexagesimal format | ||
| 1340 | + */ | ||
| 1341 | + isSexagesimal: function(value) { | ||
| 1342 | + | ||
| 1343 | + value = value.toString().replace(/\s*/, ''); | ||
| 1344 | + | ||
| 1345 | + return this.sexagesimalPattern.test(value); | ||
| 1346 | + | ||
| 1347 | + }, | ||
| 1348 | + | ||
| 1349 | + round: function(value, n) { | ||
| 1350 | + var decPlace = Math.pow(10, n); | ||
| 1351 | + return Math.round(value * decPlace)/decPlace; | ||
| 1352 | + } | ||
| 1353 | + | ||
| 1354 | + }); | ||
| 1355 | + | ||
| 1356 | + // Node module | ||
| 1357 | + if (typeof module !== 'undefined' && typeof module.exports !== 'undefined') { | ||
| 1358 | + | ||
| 1359 | + module.exports = geolib; | ||
| 1360 | + | ||
| 1361 | + // react native | ||
| 1362 | + if (typeof global === 'object') { | ||
| 1363 | + global.geolib = geolib; | ||
| 1364 | + } | ||
| 1365 | + | ||
| 1366 | + // AMD module | ||
| 1367 | + } else if (typeof define === "function" && define.amd) { | ||
| 1368 | + | ||
| 1369 | + define("geolib", [], function () { | ||
| 1370 | + return geolib; | ||
| 1371 | + }); | ||
| 1372 | + | ||
| 1373 | + // we're in a browser | ||
| 1374 | + } else { | ||
| 1375 | + | ||
| 1376 | + global.geolib = geolib; | ||
| 1377 | + | ||
| 1378 | + } | ||
| 1379 | + | ||
| 1380 | +}(this)); |
src/main/resources/static/real_control_v2/geolib/geolib.min.js
0 → 100644
| 1 | +/*! geolib 2.0.21 by Manuel Bieh | ||
| 2 | +* Library to provide geo functions like distance calculation, | ||
| 3 | +* conversion of decimal coordinates to sexagesimal and vice versa, etc. | ||
| 4 | +* WGS 84 (World Geodetic System 1984) | ||
| 5 | +* | ||
| 6 | +* @author Manuel Bieh | ||
| 7 | +* @url http://www.manuelbieh.com/ | ||
| 8 | +* @version 2.0.21 | ||
| 9 | +* @license MIT | ||
| 10 | +**/ | ||
| 11 | +!function(a,b){"use strict";function c(){}c.TO_RAD=Math.PI/180,c.TO_DEG=180/Math.PI,c.PI_X2=2*Math.PI,c.PI_DIV4=Math.PI/4;var d=Object.create(c.prototype,{version:{value:"2.0.21"},radius:{value:6378137},minLat:{value:-90},maxLat:{value:90},minLon:{value:-180},maxLon:{value:180},sexagesimalPattern:{value:/^([0-9]{1,3})°\s*([0-9]{1,3}(?:\.(?:[0-9]{1,2}))?)'\s*(([0-9]{1,3}(\.([0-9]{1,4}))?)"\s*)?([NEOSW]?)$/},measures:{value:Object.create(Object.prototype,{m:{value:1},km:{value:.001},cm:{value:100},mm:{value:1e3},mi:{value:1/1609.344},sm:{value:1/1852.216},ft:{value:100/30.48},"in":{value:100/2.54},yd:{value:1/.9144}})},prototype:{value:c.prototype},extend:{value:function(a,b){for(var c in a)("undefined"==typeof d.prototype[c]||b===!0)&&("function"==typeof a[c]&&"function"==typeof a[c].bind?d.prototype[c]=a[c].bind(d):d.prototype[c]=a[c])}}});"undefined"==typeof Number.prototype.toRad&&(Number.prototype.toRad=function(){return this*c.TO_RAD}),"undefined"==typeof Number.prototype.toDeg&&(Number.prototype.toDeg=function(){return this*c.TO_DEG}),d.extend({decimal:{},sexagesimal:{},distance:null,getKeys:function(a){if("[object Array]"==Object.prototype.toString.call(a))return{longitude:a.length>=1?0:b,latitude:a.length>=2?1:b,elevation:a.length>=3?2:b};var c=function(b){var c;return b.every(function(b){return"object"!=typeof a?!0:a.hasOwnProperty(b)?function(){return c=b,!1}():!0}),c},d=c(["lng","lon","longitude"]),e=c(["lat","latitude"]),f=c(["alt","altitude","elevation","elev"]);return"undefined"==typeof e&&"undefined"==typeof d&&"undefined"==typeof f?b:{latitude:e,longitude:d,elevation:f}},getLat:function(a,b){return b===!0?a[this.getKeys(a).latitude]:this.useDecimal(a[this.getKeys(a).latitude])},latitude:function(a){return this.getLat.call(this,a)},getLon:function(a,b){return b===!0?a[this.getKeys(a).longitude]:this.useDecimal(a[this.getKeys(a).longitude])},longitude:function(a){return this.getLon.call(this,a)},getElev:function(a){return a[this.getKeys(a).elevation]},elevation:function(a){return this.getElev.call(this,a)},coords:function(a,b){var c={latitude:b===!0?a[this.getKeys(a).latitude]:this.useDecimal(a[this.getKeys(a).latitude]),longitude:b===!0?a[this.getKeys(a).longitude]:this.useDecimal(a[this.getKeys(a).longitude])},d=a[this.getKeys(a).elevation];return"undefined"!=typeof d&&(c.elevation=d),c},ll:function(a,b){return this.coords.call(this,a,b)},validate:function(a){var b=this.getKeys(a);if("undefined"==typeof b||"undefined"==typeof b.latitude||"undefined"===b.longitude)return!1;var c=a[b.latitude],d=a[b.longitude];return"undefined"==typeof c||!this.isDecimal(c)&&!this.isSexagesimal(c)?!1:"undefined"==typeof d||!this.isDecimal(d)&&!this.isSexagesimal(d)?!1:(c=this.useDecimal(c),d=this.useDecimal(d),c<this.minLat||c>this.maxLat||d<this.minLon||d>this.maxLon?!1:!0)},getDistance:function(a,b,c,e){c=Math.floor(c)||1,e=Math.floor(e)||0;var f,g,h,i,j,k,l,m=this.coords(a),n=this.coords(b),o=6378137,p=6356752.314245,q=1/298.257223563,r=(n.longitude-m.longitude).toRad(),s=Math.atan((1-q)*Math.tan(parseFloat(m.latitude).toRad())),t=Math.atan((1-q)*Math.tan(parseFloat(n.latitude).toRad())),u=Math.sin(s),v=Math.cos(s),w=Math.sin(t),x=Math.cos(t),y=r,z=100;do{var A=Math.sin(y),B=Math.cos(y);if(k=Math.sqrt(x*A*(x*A)+(v*w-u*x*B)*(v*w-u*x*B)),0===k)return d.distance=0;f=u*w+v*x*B,g=Math.atan2(k,f),h=v*x*A/k,i=1-h*h,j=f-2*u*w/i,isNaN(j)&&(j=0);var C=q/16*i*(4+q*(4-3*i));l=y,y=r+(1-C)*q*h*(g+C*k*(j+C*f*(-1+2*j*j)))}while(Math.abs(y-l)>1e-12&&--z>0);if(0===z)return NaN;var D=i*(o*o-p*p)/(p*p),E=1+D/16384*(4096+D*(-768+D*(320-175*D))),F=D/1024*(256+D*(-128+D*(74-47*D))),G=F*k*(j+F/4*(f*(-1+2*j*j)-F/6*j*(-3+4*k*k)*(-3+4*j*j))),H=p*E*(g-G);if(H=H.toFixed(e),"undefined"!=typeof this.elevation(a)&&"undefined"!=typeof this.elevation(b)){var I=Math.abs(this.elevation(a)-this.elevation(b));H=Math.sqrt(H*H+I*I)}return this.distance=Math.round(H*Math.pow(10,e)/c)*c/Math.pow(10,e)},getDistanceSimple:function(a,b,c){c=Math.floor(c)||1;var e=Math.round(Math.acos(Math.sin(this.latitude(b).toRad())*Math.sin(this.latitude(a).toRad())+Math.cos(this.latitude(b).toRad())*Math.cos(this.latitude(a).toRad())*Math.cos(this.longitude(a).toRad()-this.longitude(b).toRad()))*this.radius);return d.distance=Math.floor(Math.round(e/c)*c)},getCenter:function(a){var b=a;if("object"==typeof a&&!(a instanceof Array)){b=[];for(var d in a)b.push(this.coords(a[d]))}if(!b.length)return!1;var e,f,g,h=0,i=0,j=0;b.forEach(function(a){e=this.latitude(a).toRad(),f=this.longitude(a).toRad(),h+=Math.cos(e)*Math.cos(f),i+=Math.cos(e)*Math.sin(f),j+=Math.sin(e)},this);var k=b.length;return h/=k,i/=k,j/=k,f=Math.atan2(i,h),g=Math.sqrt(h*h+i*i),e=Math.atan2(j,g),{latitude:(e*c.TO_DEG).toFixed(6),longitude:(f*c.TO_DEG).toFixed(6)}},getBounds:function(a){if(!a.length)return!1;var b=this.elevation(a[0]),c={maxLat:-(1/0),minLat:1/0,maxLng:-(1/0),minLng:1/0};"undefined"!=typeof b&&(c.maxElev=0,c.minElev=1/0);for(var d=0,e=a.length;e>d;++d)c.maxLat=Math.max(this.latitude(a[d]),c.maxLat),c.minLat=Math.min(this.latitude(a[d]),c.minLat),c.maxLng=Math.max(this.longitude(a[d]),c.maxLng),c.minLng=Math.min(this.longitude(a[d]),c.minLng),b&&(c.maxElev=Math.max(this.elevation(a[d]),c.maxElev),c.minElev=Math.min(this.elevation(a[d]),c.minElev));return c},getCenterOfBounds:function(a){var b=this.getBounds(a),c=b.minLat+(b.maxLat-b.minLat)/2,d=b.minLng+(b.maxLng-b.minLng)/2;return{latitude:parseFloat(c.toFixed(6)),longitude:parseFloat(d.toFixed(6))}},getBoundsOfDistance:function(a,b){var d,e,f=this.latitude(a),g=this.longitude(a),h=f.toRad(),i=g.toRad(),j=b/this.radius,k=h-j,l=h+j,m=this.maxLat.toRad(),n=this.minLat.toRad(),o=this.maxLon.toRad(),p=this.minLon.toRad();if(k>n&&m>l){var q=Math.asin(Math.sin(j)/Math.cos(h));d=i-q,p>d&&(d+=c.PI_X2),e=i+q,e>o&&(e-=c.PI_X2)}else k=Math.max(k,n),l=Math.min(l,m),d=p,e=o;return[{latitude:k.toDeg(),longitude:d.toDeg()},{latitude:l.toDeg(),longitude:e.toDeg()}]},isPointInside:function(a,b){for(var c=!1,d=-1,e=b.length,f=e-1;++d<e;f=d)(this.longitude(b[d])<=this.longitude(a)&&this.longitude(a)<this.longitude(b[f])||this.longitude(b[f])<=this.longitude(a)&&this.longitude(a)<this.longitude(b[d]))&&this.latitude(a)<(this.latitude(b[f])-this.latitude(b[d]))*(this.longitude(a)-this.longitude(b[d]))/(this.longitude(b[f])-this.longitude(b[d]))+this.latitude(b[d])&&(c=!c);return c},preparePolygonForIsPointInsideOptimized:function(a){for(var b=0,c=a.length-1;b<a.length;b++)this.longitude(a[c])===this.longitude(a[b])?(a[b].constant=this.latitude(a[b]),a[b].multiple=0):(a[b].constant=this.latitude(a[b])-this.longitude(a[b])*this.latitude(a[c])/(this.longitude(a[c])-this.longitude(a[b]))+this.longitude(a[b])*this.latitude(a[b])/(this.longitude(a[c])-this.longitude(a[b])),a[b].multiple=(this.latitude(a[c])-this.latitude(a[b]))/(this.longitude(a[c])-this.longitude(a[b]))),c=b},isPointInsideWithPreparedPolygon:function(a,b){for(var c=!1,d=this.longitude(a),e=this.latitude(a),f=0,g=b.length-1;f<b.length;f++)(this.longitude(b[f])<d&&this.longitude(b[g])>=d||this.longitude(b[g])<d&&this.longitude(b[f])>=d)&&(c^=d*b[f].multiple+b[f].constant<e),g=f;return c},isInside:function(){return this.isPointInside.apply(this,arguments)},isPointInCircle:function(a,b,c){return this.getDistance(a,b)<c},withinRadius:function(){return this.isPointInCircle.apply(this,arguments)},getRhumbLineBearing:function(a,b){var d=this.longitude(b).toRad()-this.longitude(a).toRad(),e=Math.log(Math.tan(this.latitude(b).toRad()/2+c.PI_DIV4)/Math.tan(this.latitude(a).toRad()/2+c.PI_DIV4));return Math.abs(d)>Math.PI&&(d=d>0?-1*(c.PI_X2-d):c.PI_X2+d),(Math.atan2(d,e).toDeg()+360)%360},getBearing:function(a,b){b.latitude=this.latitude(b),b.longitude=this.longitude(b),a.latitude=this.latitude(a),a.longitude=this.longitude(a);var c=(Math.atan2(Math.sin(b.longitude.toRad()-a.longitude.toRad())*Math.cos(b.latitude.toRad()),Math.cos(a.latitude.toRad())*Math.sin(b.latitude.toRad())-Math.sin(a.latitude.toRad())*Math.cos(b.latitude.toRad())*Math.cos(b.longitude.toRad()-a.longitude.toRad())).toDeg()+360)%360;return c},getCompassDirection:function(a,b,c){var d,e;switch(e="circle"==c?this.getBearing(a,b):this.getRhumbLineBearing(a,b),Math.round(e/22.5)){case 1:d={exact:"NNE",rough:"N"};break;case 2:d={exact:"NE",rough:"N"};break;case 3:d={exact:"ENE",rough:"E"};break;case 4:d={exact:"E",rough:"E"};break;case 5:d={exact:"ESE",rough:"E"};break;case 6:d={exact:"SE",rough:"E"};break;case 7:d={exact:"SSE",rough:"S"};break;case 8:d={exact:"S",rough:"S"};break;case 9:d={exact:"SSW",rough:"S"};break;case 10:d={exact:"SW",rough:"S"};break;case 11:d={exact:"WSW",rough:"W"};break;case 12:d={exact:"W",rough:"W"};break;case 13:d={exact:"WNW",rough:"W"};break;case 14:d={exact:"NW",rough:"W"};break;case 15:d={exact:"NNW",rough:"N"};break;default:d={exact:"N",rough:"N"}}return d.bearing=e,d},getDirection:function(a,b,c){return this.getCompassDirection.apply(this,arguments)},orderByDistance:function(a,b){var c=[];for(var d in b){var e=this.getDistance(a,b[d]),f=Object.create(b[d]);f.distance=e,f.key=d,c.push(f)}return c.sort(function(a,b){return a.distance-b.distance})},isPointInLine:function(a,b,c){return(this.getDistance(b,a,1,3)+this.getDistance(a,c,1,3)).toFixed(3)==this.getDistance(b,c,1,3)},isPointNearLine:function(a,b,c,d){return this.getDistanceFromLine(a,b,c)<d},getDistanceFromLine:function(a,b,c){var d=this.getDistance(b,a,1,3),e=this.getDistance(a,c,1,3),f=this.getDistance(b,c,1,3),g=0,h=Math.acos((d*d+f*f-e*e)/(2*d*f)),i=Math.acos((e*e+f*f-d*d)/(2*e*f));return g=h>Math.PI/2?d:i>Math.PI/2?e:Math.sin(h)*d},findNearest:function(a,b,c,d){c=c||0,d=d||1;var e=this.orderByDistance(a,b);return 1===d?e[c]:e.splice(c,d)},getPathLength:function(a){for(var b,c=0,d=0,e=a.length;e>d;++d)b&&(c+=this.getDistance(this.coords(a[d]),b)),b=this.coords(a[d]);return c},getSpeed:function(a,b,c){var e=c&&c.unit||"km";"mph"==e?e="mi":"kmh"==e&&(e="km");var f=d.getDistance(a,b),g=1*b.time/1e3-1*a.time/1e3,h=f/g*3600,i=Math.round(h*this.measures[e]*1e4)/1e4;return i},computeDestinationPoint:function(a,b,c,d){var e=this.latitude(a),f=this.longitude(a);d="undefined"==typeof d?this.radius:Number(d);var g=Number(b)/d,h=Number(c).toRad(),i=Number(e).toRad(),j=Number(f).toRad(),k=Math.asin(Math.sin(i)*Math.cos(g)+Math.cos(i)*Math.sin(g)*Math.cos(h)),l=j+Math.atan2(Math.sin(h)*Math.sin(g)*Math.cos(i),Math.cos(g)-Math.sin(i)*Math.sin(k));return l=(l+3*Math.PI)%(2*Math.PI)-Math.PI,{latitude:k.toDeg(),longitude:l.toDeg()}},convertUnit:function(a,b,c){if(0===b)return 0;if("undefined"==typeof b){if(null===this.distance)throw new Error("No distance was given");if(0===this.distance)return 0;b=this.distance}if(a=a||"m",c=null==c?4:c,"undefined"!=typeof this.measures[a])return this.round(b*this.measures[a],c);throw new Error("Unknown unit for conversion.")},useDecimal:function(a){if("[object Array]"===Object.prototype.toString.call(a)){var b=this;return a=a.map(function(a){if(b.isDecimal(a))return b.useDecimal(a);if("object"==typeof a){if(b.validate(a))return b.coords(a);for(var c in a)a[c]=b.useDecimal(a[c]);return a}return b.isSexagesimal(a)?b.sexagesimal2decimal(a):a})}if("object"==typeof a&&this.validate(a))return this.coords(a);if("object"==typeof a){for(var c in a)a[c]=this.useDecimal(a[c]);return a}if(this.isDecimal(a))return parseFloat(a);if(this.isSexagesimal(a)===!0)return parseFloat(this.sexagesimal2decimal(a));throw new Error("Unknown format.")},decimal2sexagesimal:function(a){if(a in this.sexagesimal)return this.sexagesimal[a];var b=a.toString().split("."),c=Math.abs(b[0]),d=60*("0."+(b[1]||0)),e=d.toString().split(".");return d=Math.floor(d),e=(60*("0."+(e[1]||0))).toFixed(2),this.sexagesimal[a]=c+"° "+d+"' "+e+'"',this.sexagesimal[a]},sexagesimal2decimal:function(a){if(a in this.decimal)return this.decimal[a];var b=new RegExp(this.sexagesimalPattern),c=b.exec(a),d=0,e=0;c&&(d=parseFloat(c[2]/60),e=parseFloat(c[4]/3600)||0);var f=(parseFloat(c[1])+d+e).toFixed(8);return f="S"==c[7]||"W"==c[7]?parseFloat(-f):parseFloat(f),this.decimal[a]=f,f},isDecimal:function(a){return a=a.toString().replace(/\s*/,""),!isNaN(parseFloat(a))&&parseFloat(a)==a},isSexagesimal:function(a){return a=a.toString().replace(/\s*/,""),this.sexagesimalPattern.test(a)},round:function(a,b){var c=Math.pow(10,b);return Math.round(a*c)/c}}),"undefined"!=typeof module&&"undefined"!=typeof module.exports?(module.exports=d,"object"==typeof a&&(a.geolib=d)):"function"==typeof define&&define.amd?define("geolib",[],function(){return d}):a.geolib=d}(this); | ||
| 0 | \ No newline at end of file | 12 | \ No newline at end of file |
src/main/resources/static/real_control_v2/js/main.js
| @@ -73,6 +73,17 @@ var gb_main_ep = new EventProxy(), | @@ -73,6 +73,17 @@ var gb_main_ep = new EventProxy(), | ||
| 73 | //modal hide remove dom | 73 | //modal hide remove dom |
| 74 | $(document).on('hide.uk.modal', '.uk-modal', function() { | 74 | $(document).on('hide.uk.modal', '.uk-modal', function() { |
| 75 | $(this).remove(); | 75 | $(this).remove(); |
| 76 | +/* //如果modal内部存在context-menu,则注销 | ||
| 77 | + var ctx = $('.context-menu-item', this); | ||
| 78 | + debugger | ||
| 79 | + if(ctx.length > 0){ | ||
| 80 | + //ctx.contextMenu('destroy'); | ||
| 81 | + $.contextMenu('destroy', ctx); | ||
| 82 | + }*/ | ||
| 83 | +}); | ||
| 84 | + | ||
| 85 | +$(document).on('click', '.ct-bottom-drawer-close', function () { | ||
| 86 | + $(this).parent().removeClass('open'); | ||
| 76 | }); | 87 | }); |
| 77 | 88 | ||
| 78 | function connectArr(arr, separator, transFun) { | 89 | function connectArr(arr, separator, transFun) { |
src/main/resources/static/real_control_v2/main.html
| @@ -31,6 +31,7 @@ | @@ -31,6 +31,7 @@ | ||
| 31 | <!-- tooltip css--> | 31 | <!-- tooltip css--> |
| 32 | <link rel="stylesheet" href="/real_control_v2/assets/plugins/qtip/jquery.qtip.min.css" /> | 32 | <link rel="stylesheet" href="/real_control_v2/assets/plugins/qtip/jquery.qtip.min.css" /> |
| 33 | 33 | ||
| 34 | + <link rel="stylesheet" href="/real_control_v2/css/pace.css" /> | ||
| 34 | </head> | 35 | </head> |
| 35 | 36 | ||
| 36 | <body> | 37 | <body> |
| @@ -138,6 +139,8 @@ | @@ -138,6 +139,8 @@ | ||
| 138 | 139 | ||
| 139 | <!-- echart --> | 140 | <!-- echart --> |
| 140 | <script src="/real_control_v2/assets/echarts-3/echarts.js"></script> | 141 | <script src="/real_control_v2/assets/echarts-3/echarts.js"></script> |
| 142 | + <!-- Geolib --> | ||
| 143 | + <script src="/real_control_v2/geolib/geolib.js"></script> | ||
| 141 | </body> | 144 | </body> |
| 142 | 145 | ||
| 143 | </html> | 146 | </html> |