WME GIS Layers

Adds GIS layers in WME

当前为 2022-07-21 提交的版本,查看 最新版本

  1. /* eslint-disable brace-style, curly, nonblock-statement-body-position, no-template-curly-in-string, func-names */
  2. // ==UserScript==
  3. // @name WME GIS Layers
  4. // @namespace https://greasyfork.org/users/45389
  5. // @version 2022.07.21.001
  6. // @description Adds GIS layers in WME
  7. // @author MapOMatic
  8. // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
  9. // @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
  10. // @grant GM_xmlhttpRequest
  11. // @grant GM_info
  12. // @license GNU GPLv3
  13. // @contributionURL https://github.com/WazeDev/Thank-The-Authors
  14. // @connect *
  15. // @connect tigerweb.geo.census.gov
  16. // @connect 131.156.137.22
  17. // @connect 136.234.13.165
  18. // @connect 184.12.255.122
  19. // @connect 206.74.124.99
  20. // @connect 216.167.160.20
  21. // @connect 23.96.59.134
  22. // @connect 35.172.145.31
  23. // @connect 52.37.30.30
  24. // @connect 54.213.14.253
  25. // @connect 65.183.210.212
  26. // @connect 72.10.206.73
  27. // @connect a2maps.a2gov.org
  28. // @connect adairgis.integritygis.com
  29. // @connect admin205.ispa.fsu.edu
  30. // @connect agis.charlottecountyfl.gov
  31. // @connect agis.wingis.org
  32. // @connect agomaps.larimer.org
  33. // @connect ags.agdmaps.com
  34. // @connect ags.bhamaps.com
  35. // @connect ags.kitsapgov.com
  36. // @connect ags.myokaloosa.com
  37. // @connect ags.roseville.ca.us
  38. // @connect ags10s1.dot.illinois.gov
  39. // @connect ags3.scgov.net
  40. // @connect aldotgis.dot.state.al.us
  41. // @connect alleganygis.allconet.org
  42. // @connect alphagis.alpharetta.ga.us
  43. // @connect andrewgis.integritygis.com
  44. // @connect ansoncountygis.com
  45. // @connect apnsgis1.apsu.edu
  46. // @connect app.mdt.mt.gov
  47. // @connect apps.alamance-nc.com
  48. // @connect apps.douglas.co.us
  49. // @connect apps.fs.usda.gov
  50. // @connect apps.lcounty.com
  51. // @connect apps.vernoncounty.org
  52. // @connect apps.wyoroad.info
  53. // @connect arc2000.florenceco.org
  54. // @connect arcgis-morrowarcgis-1015369042.us-east-1.elb.amazonaws.com
  55. // @connect arcgis-web.chinohills.org
  56. // @connect arcgis.ashevillenc.gov
  57. // @connect arcgis.atlantaregional.com
  58. // @connect arcgis.cityofcapegirardeau.org
  59. // @connect arcgis.co.beltrami.mn.us
  60. // @connect arcgis.forneytx.gov
  61. // @connect arcgis.gis.lacounty.gov
  62. // @connect arcgis.kingsporttn.gov
  63. // @connect arcgis.lakecountyohio.gov
  64. // @connect arcgis.leaguecitytx.gov
  65. // @connect arcgis.lewiscountywa.gov
  66. // @connect arcgis.mobile311.com
  67. // @connect arcgis.racinecounty.com
  68. // @connect arcgis.sd.gov
  69. // @connect arcgis.tampagov.net
  70. // @connect arcgis.tuscco.com
  71. // @connect arcgis.vgsi.com
  72. // @connect arcgis.waxahachie.com
  73. // @connect arcgis.yumacountyaz.gov
  74. // @connect arcgis2.williamsoncounty-tn.gov
  75. // @connect arcgis4.roktech.net
  76. // @connect arcgis5.roktech.net
  77. // @connect arcgisce.co.valencia.nm.us
  78. // @connect arcgisserver.lincolncounty.org
  79. // @connect arcgisserver.maine.gov
  80. // @connect arcgissrv.cityofbartlesville.org
  81. // @connect arcgiswap01.ci.temple.tx.us
  82. // @connect arcgisweb.carteretcountync.gov
  83. // @connect arcgisweb.countyofnewaygo.com
  84. // @connect arcgisweb.fortbendcountytx.gov
  85. // @connect arcgiswebadp3.morpc.org
  86. // @connect arcmobile.co.albany.wy.us
  87. // @connect arcserv.co.washington.ar.us
  88. // @connect arcserver.madisoncountyky.us
  89. // @connect arcserver2.oconeesc.com
  90. // @connect arcweb.hcad.org
  91. // @connect arcweb1.ycpc.org
  92. // @connect ardmoregis.ardmorecity.org
  93. // @connect atchisongis.integritygis.com
  94. // @connect atlas.co.chelan.wa.us
  95. // @connect atlas.geoportalmaps.com
  96. // @connect audraingis.integritygis.com
  97. // @connect batesgis.integritygis.com
  98. // @connect bcgis.baltimorecountymd.gov
  99. // @connect bcgis.broward.org
  100. // @connect bcmaps.bradfordco.org
  101. // @connect bentongis.integritygis.com
  102. // @connect bocagis.ci.boca-raton.fl.us
  103. // @connect bonneville.esriemcs.com
  104. // @connect bpagis.bossierparish.org
  105. // @connect bryangis.bryan-county.org
  106. // @connect bsm.sfdpw.org
  107. // @connect buchanangis.integritygis.com
  108. // @connect c39gisserver.co.richland.nd.us
  109. // @connect ca.dep.state.fl.us
  110. // @connect cagisonline.hamilton-co.org
  111. // @connect calmaps.co.calumet.wi.us
  112. // @connect caltrans-gis.dot.ca.gov
  113. // @connect calvertgis.co.cal.md.us
  114. // @connect camdengis.integritygis.com
  115. // @connect cassweb3.co.cass.mn.us
  116. // @connect ccapps.org
  117. // @connect ccgis.crawfordcountypa.net
  118. // @connect ccmap.cccounty.us
  119. // @connect cdsd.co.teller.co.us
  120. // @connect cecilmaps.org
  121. // @connect charitongis.integritygis.com
  122. // @connect christiangis.integritygis.com
  123. // @connect clearfieldco.org
  124. // @connect cloud.longviewtexas.gov
  125. // @connect cloud.sagis.org
  126. // @connect cloudgis.bonnercountyid.gov
  127. // @connect co.kent.de.us
  128. // @connect coagisweb.cabq.gov
  129. // @connect com.blountgis.org
  130. // @connect conservationgis.alabama.gov
  131. // @connect coopergis.integritygis.com
  132. // @connect crgis.cedar-rapids.org
  133. // @connect currituckncgov.com
  134. // @connect cw.townofclaytonnc.org
  135. // @connect dadegis.integritygis.com
  136. // @connect dallasgis.integritygis.com
  137. // @connect data.wsdot.wa.gov
  138. // @connect data1.digitaldataservices.com
  139. // @connect dc-web.co.douglas.mn.us
  140. // @connect dcgis.co.delaware.pa.us
  141. // @connect dcimapapps.countyofdane.com
  142. // @connect dekalbgis.integritygis.com
  143. // @connect delta.co.clatsop.or.us
  144. // @connect doniphangis.integritygis.com
  145. // @connect dotapp9.dot.state.mn.us
  146. // @connect douglasgis.integritygis.com
  147. // @connect douglasil.com
  148. // @connect dtdapps.coloradodot.info
  149. // @connect dungis.dunwoodyga.gov
  150. // @connect ecgis.co.ellis.tx.us
  151. // @connect egis.pinellas.gov
  152. // @connect elb.elevatemaps.io
  153. // @connect emapsplus.com
  154. // @connect engineer.gomvo.org
  155. // @connect enigma.accgov.com
  156. // @connect eoc.franklin-gov.com
  157. // @connect epv.ci.juneau.ak.us
  158. // @connect eservices.co.crook.or.us
  159. // @connect essex-gis.co.essex.ny.us
  160. // @connect esuite.concordnh.gov
  161. // @connect firstmap.gis.delaware.gov
  162. // @connect fremontgis.com
  163. // @connect gcgis.guilfordcountync.gov
  164. // @connect geaugarealink.co.geauga.oh.us
  165. // @connect geo.brunswickcountync.gov
  166. // @connect geo.co.dodge.wi.us
  167. // @connect geo.co.harrison.ms.us
  168. // @connect geo.dentoncad.com
  169. // @connect geo.forsythco.com
  170. // @connect geo.friscotexas.gov
  171. // @connect geo.oit.ohio.gov
  172. // @connect geo.sanjoseca.gov
  173. // @connect geo.tompkins-co.org
  174. // @connect geodata.hawaii.gov
  175. // @connect geodata.md.gov
  176. // @connect geodata.sarpy.com
  177. // @connect geodataportal.net
  178. // @connect geoint2.odessa-tx.gov
  179. // @connect geopower.jws.com
  180. // @connect geoweb.martin.fl.us
  181. // @connect gicwebsrv.csuchico.edu
  182. // @connect gis-2.warrencountyny.gov
  183. // @connect gis-server.co.becker.mn.us
  184. // @connect gis-server.co.montezuma.co.us
  185. // @connect gis-webpub.sonoma-county.org
  186. // @connect gis.aacounty.org
  187. // @connect gis.abilenetx.com
  188. // @connect gis.adamscounty.org
  189. // @connect gis.addisontx.gov
  190. // @connect gis.aecomonline.net
  191. // @connect gis.allegancounty.org
  192. // @connect gis.allencountyohio.com
  193. // @connect gis.apachejunctionaz.gov
  194. // @connect gis.apsu.edu
  195. // @connect gis.arapahoegov.com
  196. // @connect gis.arkansas.gov
  197. // @connect gis.arlingtonva.us
  198. // @connect gis.ashecountygov.com
  199. // @connect gis.atlantaga.gov
  200. // @connect gis.auburnalabama.org
  201. // @connect gis.auglaizecounty.org
  202. // @connect gis.azdot.gov
  203. // @connect gis.bakersfieldcity.us
  204. // @connect gis.baltimorecity.gov
  205. // @connect gis.baycountyfl.gov
  206. // @connect gis.beaufortcountysc.gov
  207. // @connect gis.beaumonttexas.gov
  208. // @connect gis.bentoncountyar.gov
  209. // @connect gis.berkeleycountysc.gov
  210. // @connect gis.bgky.org
  211. // @connect gis.bladenco.org
  212. // @connect gis.blairco.org
  213. // @connect gis.bransonmo.gov
  214. // @connect gis.brevardfl.gov
  215. // @connect gis.brookhavenga.gov
  216. // @connect gis.browncountywi.gov
  217. // @connect gis.buncombecounty.org
  218. // @connect gis.burkenc.org
  219. // @connect gis.burleighco.com
  220. // @connect gis.buttecounty.net
  221. // @connect gis.caldwellcountync.org
  222. // @connect gis.calhouncounty.org
  223. // @connect gis.carboncounty.com
  224. // @connect gis.cayugacounty.us
  225. // @connect gis.cccounty.us
  226. // @connect gis.ccgisonline.com
  227. // @connect gis.ccgov.net
  228. // @connect gis.ccpa.net
  229. // @connect gis.cedarhilltx.com
  230. // @connect gis.cherokeega.com
  231. // @connect gis.chippewa.mn
  232. // @connect gis.ci.janesville.wi.us
  233. // @connect gis.ci.mcminnville.or.us
  234. // @connect gis.ci.waco.tx.us
  235. // @connect gis.citruspa.org
  236. // @connect gis.cityofaikensc.gov
  237. // @connect gis.cityofberkeley.info
  238. // @connect gis.cityofboston.gov
  239. // @connect gis.cityofirvine.org
  240. // @connect gis.cityofmiddletown.com
  241. // @connect gis.cityofmoore.com
  242. // @connect gis.clark.wa.gov
  243. // @connect gis.clearwatercounty.org
  244. // @connect gis.clevelandtn.gov
  245. // @connect gis.co.benton.or.us
  246. // @connect gis.co.berks.pa.us
  247. // @connect gis.co.big-stone.mn.us
  248. // @connect gis.co.butler.pa.us
  249. // @connect gis.co.carlton.mn.us
  250. // @connect gis.co.carver.mn.us
  251. // @connect gis.co.clarion.pa.us
  252. // @connect gis.co.cumberland.nc.us
  253. // @connect gis.co.douglas.or.us
  254. // @connect gis.co.eau-claire.wi.us
  255. // @connect gis.co.fillmore.mn.us
  256. // @connect gis.co.grand.co.us
  257. // @connect gis.co.grant.mn.us
  258. // @connect gis.co.grant.wi.gov
  259. // @connect gis.co.green-lake.wi.us
  260. // @connect gis.co.holmes.oh.us
  261. // @connect gis.co.hubbard.mn.us
  262. // @connect gis.co.josephine.or.us
  263. // @connect gis.co.kandiyohi.mn.us
  264. // @connect gis.co.kittitas.wa.us
  265. // @connect gis.co.knox.il.us
  266. // @connect gis.co.lancaster.pa.us
  267. // @connect gis.co.linn.or.us
  268. // @connect gis.co.mason.wa.us
  269. // @connect gis.co.mille-lacs.mn.us
  270. // @connect gis.co.nezperce.id.us
  271. // @connect gis.co.oneida.wi.us
  272. // @connect gis.co.pepin.wi.us
  273. // @connect gis.co.polk.mn.us
  274. // @connect gis.co.pope.mn.us
  275. // @connect gis.co.richland.wi.us
  276. // @connect gis.co.roseau.mn.us
  277. // @connect gis.co.routt.co.us
  278. // @connect gis.co.sauk.wi.us
  279. // @connect gis.co.sherburne.mn.us
  280. // @connect gis.co.stearns.mn.us
  281. // @connect gis.co.tuscarawas.oh.us
  282. // @connect gis.co.wadena.mn.us
  283. // @connect gis.co.waseca.mn.us
  284. // @connect gis.co.washington.ny.us
  285. // @connect gis.co.waushara.wi.us
  286. // @connect gis.co.wood.wi.us
  287. // @connect gis.co.ym.mn.gov
  288. // @connect gis.collincountytx.gov
  289. // @connect gis.colorado.gov
  290. // @connect gis.coloradosprings.gov
  291. // @connect gis.columbiacountymaps.com
  292. // @connect gis.columbiasc.gov
  293. // @connect gis.columbusga.org
  294. // @connect gis.cookeville-tn.org
  295. // @connect gis.corvallisoregon.gov
  296. // @connect gis.cosb.us
  297. // @connect gis.countyofriverside.us
  298. // @connect gis.cowleycounty.org
  299. // @connect gis.cranstonri.org
  300. // @connect gis.cravencountync.gov
  301. // @connect gis.crookcounty.wy.gov
  302. // @connect gis.crowwing.us
  303. // @connect gis.dallascityhall.com
  304. // @connect gis.danville-va.gov
  305. // @connect gis.dauphincounty.org
  306. // @connect gis.dbqco.org
  307. // @connect gis.dcga.us
  308. // @connect gis.deerparktx.gov
  309. // @connect gis.dekalbcountyga.gov
  310. // @connect gis.dentoncounty.gov
  311. // @connect gis.dogis.org
  312. // @connect gis.donaanacounty.org
  313. // @connect gis.dot.nh.gov
  314. // @connect gis.dot.state.oh.us
  315. // @connect gis.douglascountyks.org
  316. // @connect gis.dupageco.org
  317. // @connect gis.duplincountync.com
  318. // @connect gis.eastgreenwichri.com
  319. // @connect gis.ebparks.org
  320. // @connect gis.edgecombecountync.gov
  321. // @connect gis.elpasotexas.gov
  322. // @connect gis.emmetcounty.org
  323. // @connect gis.fortlauderdale.gov
  324. // @connect gis.franklincountyohio.gov
  325. // @connect gis.franklincountypa.gov
  326. // @connect gis.fultoncountyoh.com
  327. // @connect gis.fwb.org
  328. // @connect gis.fwp.mt.gov
  329. // @connect gis.gallatin.mt.gov
  330. // @connect gis.gallupnm.us
  331. // @connect gis.garfield-county.com
  332. // @connect gis.gastongov.com
  333. // @connect gis.gilacountyaz.gov
  334. // @connect gis.gocolumbiamo.com
  335. // @connect gis.goshencounty.org
  336. // @connect gis.gptx.org
  337. // @connect gis.greenegovernment.com
  338. // @connect gis.greensboro-nc.gov
  339. // @connect gis.gscplanning.com
  340. // @connect gis.hardeecounty.net
  341. // @connect gis.harnett.org
  342. // @connect gis.hawaiicounty.gov
  343. // @connect gis.hcpafl.org
  344. // @connect gis.hennepin.us
  345. // @connect gis.in.gov
  346. // @connect gis.interdev.com
  347. // @connect gis.inyoco.com
  348. // @connect gis.iowadot.gov
  349. // @connect gis.itd.idaho.gov
  350. // @connect gis.jacksonnc.org
  351. // @connect gis.jccal.org
  352. // @connect gis.johnscreekga.gov
  353. // @connect gis.johnson-county.com
  354. // @connect gis.johnsoncitytn.org
  355. // @connect gis.kanawhacountyassessor.com
  356. // @connect gis.kaufmancounty.net
  357. // @connect gis.kcgov.us
  358. // @connect gis.kentcountymi.gov
  359. // @connect gis.kleinfelder.com
  360. // @connect gis.lacrossecounty.org
  361. // @connect gis.lafayettecountywi.org
  362. // @connect gis.lakecountyfl.gov
  363. // @connect gis.lapazcountyaz.org
  364. // @connect gis.latah.id.us
  365. // @connect gis.leecountyil.com
  366. // @connect gis.lehighcounty.org
  367. // @connect gis.leoc.net
  368. // @connect gis.linncounty.org
  369. // @connect gis.littleelm.org
  370. // @connect gis.livingstoncounty.us
  371. // @connect gis.lja.com
  372. // @connect gis.losalamosnm.us
  373. // @connect gis.luzernecounty.org
  374. // @connect gis.lyco.org
  375. // @connect gis.lyon-county.org
  376. // @connect gis.macombgov.org
  377. // @connect gis.maconnc.org
  378. // @connect gis.maderacounty.com
  379. // @connect gis.marinpublic.com
  380. // @connect gis.marionfl.org
  381. // @connect gis.massdot.state.ma.us
  382. // @connect gis.mbakerintl.com
  383. // @connect gis.mcassessor.maricopa.gov
  384. // @connect gis.mendocinocounty.org
  385. // @connect gis.mercercountyohio.org
  386. // @connect gis.minnehahacounty.org
  387. // @connect gis.missoulacounty.us
  388. // @connect gis.montgomeryal.gov
  389. // @connect gis.moorecountync.gov
  390. // @connect gis.mytoddcounty.com
  391. // @connect gis.napa.ca.gov
  392. // @connect gis.nassaucountyny.gov
  393. // @connect gis.ncboc.com
  394. // @connect gis.nccde.org
  395. // @connect gis.ne.gov
  396. // @connect gis.neccog.org
  397. // @connect gis.nevadadot.com
  398. // @connect gis.nevcounty.net
  399. // @connect gis.newedgeservices.com
  400. // @connect gis.nhcgov.com
  401. // @connect gis.niagaracounty.com
  402. // @connect gis.nola.gov
  403. // @connect gis.norrycopa.net
  404. // @connect gis.northamptoncounty.org
  405. // @connect gis.odot.state.or.us
  406. // @connect gis.ohiodnr.gov
  407. // @connect gis.okc.gov
  408. // @connect gis.orangecountygov.com
  409. // @connect gis.orangecountync.gov
  410. // @connect gis.orangecountyva.gov
  411. // @connect gis.outagamie.org
  412. // @connect gis.owensboro.org
  413. // @connect gis.pandai.com
  414. // @connect gis.pearlandtx.gov
  415. // @connect gis.pendercountync.gov
  416. // @connect gis.pendoreilleco.org
  417. // @connect gis.penndot.gov
  418. // @connect gis.peoriacounty.org
  419. // @connect gis.perryco.org
  420. // @connect gis.personcountync.gov
  421. // @connect gis.pgatlas.com
  422. // @connect gis.pikepa.org
  423. // @connect gis.pittcountync.gov
  424. // @connect gis.port-orange.org
  425. // @connect gis.pottcounty-ia.gov
  426. // @connect gis.qac.org
  427. // @connect gis.randolphcountync.gov
  428. // @connect gis.rapides911.org
  429. // @connect gis.rcgov.org
  430. // @connect gis.renvillecountymn.com
  431. // @connect gis.rileycountyks.gov
  432. // @connect gis.rowancountync.gov
  433. // @connect gis.rrnm.gov
  434. // @connect gis.rtcsnv.com
  435. // @connect gis.salkeiz.k12.or.us
  436. // @connect gis.sangis.org
  437. // @connect gis.sanjuanco.com
  438. // @connect gis.santa-clarita.com
  439. // @connect gis.santacruzcounty.us
  440. // @connect gis.sccwi.gov
  441. // @connect gis.sheboygancounty.com
  442. // @connect gis.shelbycountytn.gov
  443. // @connect gis.showmeboone.com
  444. // @connect gis.slocounty.ca.gov
  445. // @connect gis.snco.us
  446. // @connect gis.southkingstownri.com
  447. // @connect gis.spokanecounty.org
  448. // @connect gis.stlouiscountymn.gov
  449. // @connect gis.stlucieco.gov
  450. // @connect gis.summitcountyco.gov
  451. // @connect gis.sumtercountyfl.gov
  452. // @connect gis.surryinfo.net
  453. // @connect gis.tazewell.com
  454. // @connect gis.texoma.cog.tx.us
  455. // @connect gis.thecolonytx.gov
  456. // @connect gis.thomsonreuters.com
  457. // @connect gis.townoftruckee.com
  458. // @connect gis.transportation.wv.gov
  459. // @connect gis.traviscountytx.gov
  460. // @connect gis.tularecounty.ca.gov
  461. // @connect gis.ucdavis.edu
  462. // @connect gis.ulstercountyny.gov
  463. // @connect gis.vernon-ct.gov
  464. // @connect gis.victorvilleca.gov
  465. // @connect gis.weatherfordtx.gov
  466. // @connect gis.westplains.net
  467. // @connect gis.whitfieldcountyga.com
  468. // @connect gis.wilco.org
  469. // @connect gis.wilkescounty.net
  470. // @connect gis.willcountyillinois.com
  471. // @connect gis.wilson-co.com
  472. // @connect gis.wilsonnc.org
  473. // @connect gis.wiu.edu
  474. // @connect gis.worldviewsolutions.com
  475. // @connect gis.wyo.gov
  476. // @connect gis.yadkincountync.gov
  477. // @connect gis.yavapaiaz.gov
  478. // @connect gis.yolocounty.org
  479. // @connect gis01.cityofarcata.org
  480. // @connect gis1.georgetowncountysc.org
  481. // @connect gis1.hartford.gov
  482. // @connect gis11.services.ncdot.gov
  483. // @connect gis12.cookcountyil.gov
  484. // @connect gis2.arlingtontx.gov
  485. // @connect gis2.ashtabulacounty.us
  486. // @connect gis2.co.dakota.mn.us
  487. // @connect gis2.co.marathon.wi.us
  488. // @connect gis2.co.ozaukee.wi.us
  489. // @connect gis2.erie.gov
  490. // @connect gis2.gworks.com
  491. // @connect gis2.idaho.gov
  492. // @connect gis2.lawrenceks.org
  493. // @connect gis2.orangeburgcounty.org
  494. // @connect gis2.sandyspringsga.gov
  495. // @connect gis2.siouxfalls.org
  496. // @connect gis2.stancounty.com
  497. // @connect gis2.totaland.com
  498. // @connect gis2.waukcogeo.com
  499. // @connect gis3.cdmsmithgis.com
  500. // @connect gis3.cmpdd.org
  501. // @connect gis3.gwinnettcounty.com
  502. // @connect gis3.gworks.com
  503. // @connect gis3.montgomerycountymd.gov
  504. // @connect gis3.richmondnc.com
  505. // @connect gis4.montgomerycountymd.gov
  506. // @connect gis5.oit.ohio.gov
  507. // @connect gisago-qa.mcgi.state.mi.us
  508. // @connect gisago.mcgi.state.mi.us
  509. // @connect gisapp.adcogov.org
  510. // @connect gisapp.mahoningcountyoh.gov
  511. // @connect gisapps.rileycountyks.gov
  512. // @connect gisapps.wicomicocounty.org
  513. // @connect gisapps1.mapoakland.com
  514. // @connect gisccapps.charlestoncounty.org
  515. // @connect gisdata.alleghenycounty.us
  516. // @connect gisdata.dot.ca.gov
  517. // @connect gisdata.farrwestengineering.com
  518. // @connect gisdata.jeffersoncountyoh.com
  519. // @connect gisdata.kingcounty.gov
  520. // @connect gisdata.pandai.com
  521. // @connect gisdata.pima.gov
  522. // @connect gisdata.seattle.gov
  523. // @connect gisdemo1.cdmsmith.com
  524. // @connect gisdemo2.cdmsmith.com
  525. // @connect gisext.lincoln.ne.gov
  526. // @connect gisgate.co.clark.nv.us
  527. // @connect gisinfo.co.portage.wi.us
  528. // @connect gisinfo.co.walworth.wi.us
  529. // @connect gisinfo.lawrencevillega.org
  530. // @connect gismap.augustaga.gov
  531. // @connect gismap.cityofboise.org
  532. // @connect gismap.co.juneau.wi.us
  533. // @connect gismap.co.marshall.mn.us
  534. // @connect gismap.co.norman.mn.us
  535. // @connect gismapping.stafford.va.us
  536. // @connect gismaps.cityofboise.org
  537. // @connect gismaps.cityofgreer.org
  538. // @connect gismaps.co.cerro-gordo.ia.us
  539. // @connect gismaps.co.sangamon.il.us
  540. // @connect gismaps.coconino.az.gov
  541. // @connect gismaps.columbiapa.org
  542. // @connect gismaps.flower-mound.com
  543. // @connect gismaps.fultoncountyga.gov
  544. // @connect gismaps.hctra.org
  545. // @connect gismaps.kingcounty.gov
  546. // @connect gismaps.mckeancountypa.org
  547. // @connect gismaps.pinalcountyaz.gov
  548. // @connect gismaps.vdem.virginia.gov
  549. // @connect gismaps.wichita.gov
  550. // @connect gismapserver.leegov.com
  551. // @connect gisonline.greenvillenc.gov
  552. // @connect gisp.co.genesee.ny.us
  553. // @connect gisp1.polkcountyiowa.gov
  554. // @connect gisportal.co.calaveras.ca.us
  555. // @connect gisportal.co.madison.il.us
  556. // @connect gisportal.fnsb.us
  557. // @connect gisportal.ircgov.com
  558. // @connect gisprod10.co.fresno.ca.us
  559. // @connect gisprodops.chesco.org
  560. // @connect gisprpxy.itd.state.ma.us
  561. // @connect gispub.cityofaspen.com
  562. // @connect gispub.co.washington.or.us
  563. // @connect gispublic.co.lake.ca.us
  564. // @connect gispw.coloradosprings.gov
  565. // @connect gisrevprxy.seattle.gov
  566. // @connect giss3.cmpdd.org
  567. // @connect gissd.sandag.org
  568. // @connect gisserver.christiancountymo.gov
  569. // @connect gisserver18.co.teton.id.us
  570. // @connect gisservice.cityofmesquite.com
  571. // @connect gisservicemt.gov
  572. // @connect gisservices.chathamcountync.gov
  573. // @connect gisservices.chathamnc.org
  574. // @connect gisservices.co.anoka.mn.us
  575. // @connect gisservices.dorchestercounty.net
  576. // @connect gisservices.its.ny.gov
  577. // @connect gisservices.oakgov.com
  578. // @connect gisservices2.suffolkcountyny.gov
  579. // @connect gissites4.centrecountypa.gov
  580. // @connect gissvr.watgov.org
  581. // @connect gisweb-18.ci.killeen.tx.us
  582. // @connect gisweb.albemarle.org
  583. // @connect gisweb.birminghamal.gov
  584. // @connect gisweb.casscountynd.gov
  585. // @connect gisweb.champaignil.gov
  586. // @connect gisweb.co.aitkin.mn.us
  587. // @connect gisweb.co.hinds.ms.us
  588. // @connect gisweb.co.wilkin.mn.us
  589. // @connect gisweb.fdlco.wi.gov
  590. // @connect gisweb.jeffcowa.us
  591. // @connect gisweb.miamidade.gov
  592. // @connect gisweb.putnam-fl.com
  593. // @connect gisweb.wycokck.org
  594. // @connect gisweb1.unioncountync.gov
  595. // @connect gisweb102.highpointnc.gov
  596. // @connect gisweb2014.gordoncounty.org
  597. // @connect giswebservices.ci.salinas.ca.us
  598. // @connect giswww.westchestergov.com
  599. // @connect git.co.tioga.ny.us
  600. // @connect gmdnags.colliercountyfl.gov
  601. // @connect grant.co.jefferson.id.us
  602. // @connect gweb01.co.olmsted.mn.us
  603. // @connect harpergis.integritygis.com
  604. // @connect harrisonms.geopowered.com
  605. // @connect haslet.halff.com
  606. // @connect hazards.fema.gov
  607. // @connect hdgis.ingham.org
  608. // @connect helenamontanamaps.org
  609. // @connect henrygis.integritygis.com
  610. // @connect hgis.hialeahfl.gov
  611. // @connect holtgis.integritygis.com
  612. // @connect host.cdmsmithgis.com
  613. // @connect hostingdata2.tighebond.com
  614. // @connect hostingdata3.tighebond.com
  615. // @connect huntsvillegis.com
  616. // @connect idpgis.ncep.noaa.gov
  617. // @connect ifgis.idahofallsidaho.gov
  618. // @connect imap.klickitatcounty.org
  619. // @connect ims.districtiii.org
  620. // @connect intervector.leoncountyfl.gov
  621. // @connect iowagis.integritygis.com
  622. // @connect ira.property-appraiser.org
  623. // @connect jcgis.jacksongov.org
  624. // @connect jeffarcgis.jeffersoncountywi.gov
  625. // @connect joplingis.org
  626. // @connect k3gis.com
  627. // @connect kygisserver.ky.gov
  628. // @connect lacledegis.integritygis.com
  629. // @connect lafayettegis.integritygis.com
  630. // @connect landrecords.greencountywi.org
  631. // @connect lawrencegis.integritygis.com
  632. // @connect lcapps.co.lucas.oh.us
  633. // @connect lcmaps.lanecounty.org
  634. // @connect lee-arcgis.leecountync.gov
  635. // @connect lgmap.wdm.iowa.gov
  636. // @connect lincolngis.integritygis.com
  637. // @connect linngis.integritygis.com
  638. // @connect lio.milwaukeecountywi.gov
  639. // @connect livingstongis.integritygis.com
  640. // @connect location.cabarruscounty.us
  641. // @connect logis.loudoun.gov
  642. // @connect lpcgis.laplata.co.us
  643. // @connect lrs.co.columbia.wi.us
  644. // @connect lucity.sbpg.net
  645. // @connect macongis.integritygis.com
  646. // @connect madison.rexburg.org
  647. // @connect madisongis.cityofalbany.net
  648. // @connect manitowocmaps.info
  649. // @connect map.claycountymn.gov
  650. // @connect map.co.clearwater.mn.us
  651. // @connect map.co.merced.ca.us
  652. // @connect map.co.thurston.wa.us
  653. // @connect map.coppelltx.gov
  654. // @connect map.eaglecounty.us
  655. // @connect map.newberrycounty.net
  656. // @connect map.opkansas.org
  657. // @connect map.polkpa.org
  658. // @connect map.stclairco.com
  659. // @connect map.wyoroad.info
  660. // @connect map8.incog.org
  661. // @connect mapd.kcmo.org
  662. // @connect mapdata.baytown.org
  663. // @connect mapdata.lasvegasnevada.gov
  664. // @connect mapit.fortworthtexas.gov
  665. // @connect mapitwest.fortworthtexas.gov
  666. // @connect mapping.adamscounty.us
  667. // @connect mapping.huntingdoncounty.net
  668. // @connect mapping.kenoshacountywi.gov
  669. // @connect mapping.mitchellcounty.org
  670. // @connect mapping.modot.org
  671. // @connect maps.alexandercountync.gov
  672. // @connect maps.alexandriava.gov
  673. // @connect maps.austintexas.gov
  674. // @connect maps.bannockcounty.us
  675. // @connect maps.bayfieldcounty.org
  676. // @connect maps.bayfieldcounty.wi.gov
  677. // @connect maps.berkeleywv.org
  678. // @connect maps.boonecountyil.org
  679. // @connect maps.bouldercounty.org
  680. // @connect maps.brazoriacountytx.gov
  681. // @connect maps.bryantx.gov
  682. // @connect maps.burlesontx.com
  683. // @connect maps.butlercountyauditor.org
  684. // @connect maps.c3gov.com
  685. // @connect maps.canyonco.org
  686. // @connect maps.capturecama.com
  687. // @connect maps.casperwy.gov
  688. // @connect maps.chautauquacounty.com
  689. // @connect maps.cherokeecounty-nc.gov
  690. // @connect maps.ci.longmont.co.us
  691. // @connect maps.ci.nacogdoches.tx.us
  692. // @connect maps.cityhs.net
  693. // @connect maps.cityofconroe.org
  694. // @connect maps.cityofhenderson.com
  695. // @connect maps.cityofmadison.com
  696. // @connect maps.cityofmobile.org
  697. // @connect maps.cityofsherman.com
  698. // @connect maps.cityoftulsa.org
  699. // @connect maps.clarkcountynv.gov
  700. // @connect maps.claycountygov.com
  701. // @connect maps.clermontauditor.org
  702. // @connect maps.clintoncountypa.com
  703. // @connect maps.co.blaine.id.us
  704. // @connect maps.co.forsyth.nc.us
  705. // @connect maps.co.goodhue.mn.us
  706. // @connect maps.co.gov
  707. // @connect maps.co.grayson.tx.us
  708. // @connect maps.co.itasca.mn.us
  709. // @connect maps.co.kendall.il.us
  710. // @connect maps.co.kern.ca.us
  711. // @connect maps.co.lincoln.wi.us
  712. // @connect maps.co.monterey.ca.us
  713. // @connect maps.co.palm-beach.fl.us
  714. // @connect maps.co.polk.or.us
  715. // @connect maps.co.pueblo.co.us
  716. // @connect maps.co.ramsey.mn.us
  717. // @connect maps.co.shasta.ca.us
  718. // @connect maps.co.shawano.wi.us
  719. // @connect maps.co.warren.oh.us
  720. // @connect maps.co.washington.mn.us
  721. // @connect maps.co.yellowstone.mt.gov
  722. // @connect maps.coj.net
  723. // @connect maps.crc.ga.gov
  724. // @connect maps.cstx.gov
  725. // @connect maps.ctmetro.org
  726. // @connect maps.dancgis.org
  727. // @connect maps.dcad.org
  728. // @connect maps.delco-gis.org
  729. // @connect maps.deltacounty.com
  730. // @connect maps.deschutes.org
  731. // @connect maps.desotocountyms.gov
  732. // @connect maps.dmgov.org
  733. // @connect maps.dotd.la.gov
  734. // @connect maps.douglascountywa.net
  735. // @connect maps.escpa.org
  736. // @connect maps.fayetteville-ar.gov
  737. // @connect maps.flathead.mt.gov
  738. // @connect maps.floridadisaster.org
  739. // @connect maps.frederickcountymd.gov
  740. // @connect maps.fredericksburgva.gov
  741. // @connect maps.garfield-county.com
  742. // @connect maps.garlandtx.gov
  743. // @connect maps.garrettcounty.org
  744. // @connect maps.grcity.us
  745. // @connect maps.groton-ct.gov
  746. // @connect maps.grundyco.org
  747. // @connect maps.hayward-ca.gov
  748. // @connect maps.haywoodnc.net
  749. // @connect maps.highlandvillage.org
  750. // @connect maps.huntsvilleal.gov
  751. // @connect maps.indiana.edu
  752. // @connect maps.itos.uga.edu
  753. // @connect maps.jocogov.org
  754. // @connect maps.kpb.us
  755. // @connect maps.kytc.ky.gov
  756. // @connect maps.lacity.org
  757. // @connect maps.lagrange-ga.org
  758. // @connect maps.lakecountyil.gov
  759. // @connect maps.laramiecounty.com
  760. // @connect maps.lcwy.org
  761. // @connect maps.lex-co.com
  762. // @connect maps.lexingtonky.gov
  763. // @connect maps.libertymo.gov
  764. // @connect maps.lincolncountysd.org
  765. // @connect maps.linkgis.org
  766. // @connect maps.matsugov.us
  767. // @connect maps.mckinneytexas.org
  768. // @connect maps.meshekgis.com
  769. // @connect maps.miamigov.com
  770. // @connect maps.nashville.gov
  771. // @connect maps.nassauflpa.com
  772. // @connect maps.nj.gov
  773. // @connect maps.normanok.gov
  774. // @connect maps.ocgov.net
  775. // @connect maps.orcity.org
  776. // @connect maps.outdoornebraska.gov
  777. // @connect maps.palmcoastgov.com
  778. // @connect maps.parkco.us
  779. // @connect maps.phoenix.gov
  780. // @connect maps.pitkincounty.com
  781. // @connect maps.placer.ca.gov
  782. // @connect maps.planogis.org
  783. // @connect maps.raleighnc.gov
  784. // @connect maps.richlandcountyoh.us
  785. // @connect maps.rutherfordcountytn.gov
  786. // @connect maps.sanmiguelcountyco.gov
  787. // @connect maps.santa-clarita.com
  788. // @connect maps.santabarbaraca.gov
  789. // @connect maps.sccmo.org
  790. // @connect maps.sgcity.org
  791. // @connect maps.shelbyal.com
  792. // @connect maps.slocity.org
  793. // @connect maps.snoco.org
  794. // @connect maps.spartanburgcounty.org
  795. // @connect maps.springfieldmo.gov
  796. // @connect maps.stlouisco.com
  797. // @connect maps.sussexcountyde.gov
  798. // @connect maps.talbgov.org
  799. // @connect maps.townofcary.org
  800. // @connect maps.udot.utah.gov
  801. // @connect maps.vcgi.vermont.gov
  802. // @connect maps.ventura.org
  803. // @connect maps.vermont.gov
  804. // @connect maps.victoriatx.org
  805. // @connect maps.vilascountywi.gov
  806. // @connect maps.vtrans.vermont.gov
  807. // @connect maps.wakegov.com
  808. // @connect maps.washco-md.net
  809. // @connect maps.washcowisco.gov
  810. // @connect maps1.eriecounty.oh.gov
  811. // @connect maps1.larimer.org
  812. // @connect maps11.eriecounty.oh.gov
  813. // @connect maps2.bgadd.org
  814. // @connect maps2.cattco.org
  815. // @connect maps2.ci.euless.tx.us
  816. // @connect maps2.dcgis.dc.gov
  817. // @connect maps2.san-marcos.net
  818. // @connect maps2.timmons.com
  819. // @connect maps2.tucsonaz.gov
  820. // @connect maps2.vcgov.org
  821. // @connect maps5.kcmo.org
  822. // @connect maps6.stlouis-mo.gov
  823. // @connect maps7.eriecounty.oh.gov
  824. // @connect maps8.eriecounty.oh.gov
  825. // @connect mapsdev.hamiltontn.gov
  826. // @connect mapserv.mesquitenv.gov
  827. // @connect mapservices.gis.saccounty.net
  828. // @connect mapservices.legis.wisconsin.gov
  829. // @connect mapservices.pasda.psu.edu
  830. // @connect mapservices.santacruzcountyaz.gov
  831. // @connect mapservices1.jeffco.us
  832. // @connect mapservices2.jeffco.us
  833. // @connect mapsonline.columbiacountyga.gov
  834. // @connect mariongis.integritygis.com
  835. // @connect mcggis.mcgtn.org
  836. // @connect mcgis.mesacounty.us
  837. // @connect mcgis.mohave.gov
  838. // @connect mcgis4.monroecounty-fl.gov
  839. // @connect mcmap2.montrosecounty.net
  840. // @connect mcogis.co.marion.oh.us
  841. // @connect mgrcmaps.org
  842. // @connect midland.newedgeservices.com
  843. // @connect millergis.integritygis.com
  844. // @connect mms.hursttx.gov
  845. // @connect mndotgis.dot.state.mn.us
  846. // @connect moberlygis.integritygis.com
  847. // @connect moniteaugis.integritygis.com
  848. // @connect morgangis.integritygis.com
  849. // @connect msdisweb.missouri.edu
  850. // @connect mycity2.houstontx.gov
  851. // @connect nbgis.newportbeachca.gov
  852. // @connect ndgishub.nd.gov
  853. // @connect newtongis.integritygis.com
  854. // @connect nhgeodata.unh.edu
  855. // @connect northlake.halff.com
  856. // @connect oak.co.lake-of-the-woods.mn.us
  857. // @connect oc17maps.co.oconto.wi.us
  858. // @connect ocgis4.ocfl.net
  859. // @connect oklahomacounty.geocortex.com
  860. // @connect oldhamgis.org
  861. // @connect oncorng.co.ontario.ny.us
  862. // @connect operationserver.ci.henderson.nc.us
  863. // @connect orfmaps.norfolk.gov
  864. // @connect pagis.org
  865. // @connect parcels.rsdigital.com
  866. // @connect parcelviewer.geodecisions.com
  867. // @connect pascogis.pascocountyfl.net
  868. // @connect pgis.plantation.org
  869. // @connect phelpsgis.integritygis.com
  870. // @connect polaris3g.mecklenburgcountync.gov
  871. // @connect polkgis.integritygis.com
  872. // @connect portal.carolinabeach.org
  873. // @connect programs.iowadnr.gov
  874. // @connect propaccess.wadtx.com
  875. // @connect propertyviewer.andersoncountysc.org
  876. // @connect proxy2.roktech.net
  877. // @connect psportal.harrisoncountywv.com
  878. // @connect pubgis.ci.lubbock.tx.us
  879. // @connect public.co.wasco.or.us
  880. // @connect public1.co.waupaca.wi.us
  881. // @connect publicmap01.co.st-clair.il.us
  882. // @connect publicmaps.txkusa.org
  883. // @connect pulaskigis.integritygis.com
  884. // @connect pwmaps.reno.gov
  885. // @connect rallsgis.integritygis.com
  886. // @connect raygis.integritygis.com
  887. // @connect renogis3.renogov.org
  888. // @connect rockgis.co.rock.wi.us
  889. // @connect romefloyd.agdmaps.com
  890. // @connect rptsgisweb.oswegocounty.com
  891. // @connect salinegis.integritygis.com
  892. // @connect saludacountysc.net
  893. // @connect scgisa.starkcountyohio.gov
  894. // @connect secure.boonecountygis.com
  895. // @connect secure.scgnet.us
  896. // @connect sedaliagis.integritygis.com
  897. // @connect see-eldorado.edcgov.us
  898. // @connect seminolearcgis.seminolecountyfl.gov
  899. // @connect server1.mapxpress.net
  900. // @connect server2.mapxpress.net
  901. // @connect services.arcgis.com
  902. // @connect services.ccgisc.org
  903. // @connect services.geoportalmaps.com
  904. // @connect services.gis.ca.gov
  905. // @connect services.kansasgis.org
  906. // @connect services.mh-gis.com
  907. // @connect services.nconemap.gov
  908. // @connect services.putnamco.org
  909. // @connect services.wvgis.wvu.edu
  910. // @connect services1.arcgis.com
  911. // @connect services2.arcgis.com
  912. // @connect services2.bhamaps.com
  913. // @connect services2.integritygis.com
  914. // @connect services3.arcgis.com
  915. // @connect services5.arcgis.com
  916. // @connect services6.arcgis.com
  917. // @connect services7.arcgis.com
  918. // @connect services8.arcgis.com
  919. // @connect services9.arcgis.com
  920. // @connect showlowmaps.com
  921. // @connect skyview.hornershifrin.com
  922. // @connect slco.org
  923. // @connect smgis.sanmarcostx.gov
  924. // @connect smpesri.scdot.org
  925. // @connect socogis.sonomacounty.ca.gov
  926. // @connect solanocountygis.com
  927. // @connect spatial.gishost.com
  928. // @connect spatial.jacksoncounty.org
  929. // @connect spatialags.vhb.com
  930. // @connect springsgis.coralsprings.org
  931. // @connect stmgis.stmarysmd.com
  932. // @connect stokescountygis.com
  933. // @connect stonegis.integritygis.com
  934. // @connect summitmaps.summitoh.net
  935. // @connect svr4.sumtercountysc.org
  936. // @connect thamo2.thewoodlands-tx.gov
  937. // @connect tigerweb.geo.census.gov
  938. // @connect tiogagis.tiogacountypa.us
  939. // @connect tnmap.tn.gov
  940. // @connect tsc-gis-ags101a.schneidercorp.com
  941. // @connect twu.newedgeservices.com
  942. // @connect utility.arcgis.com
  943. // @connect vernongis.integritygis.com
  944. // @connect vtransmap01.aot.state.vt.us
  945. // @connect wallawallagis.com
  946. // @connect warrengis.integritygis.com
  947. // @connect warrensburggis.integritygis.com
  948. // @connect wcgis3.co.winnebago.wi.us
  949. // @connect wcgisweb.washoecounty.us
  950. // @connect wcoh.geopowered.com
  951. // @connect wcsvrgis.washcopa.org
  952. // @connect web.co.wright.mn.us
  953. // @connect web2.kcsgis.com
  954. // @connect web4.kcsgis.com
  955. // @connect web5.kcsgis.com
  956. // @connect weba.co.clayton.ga.us
  957. // @connect webgis.bedfordcountyva.gov
  958. // @connect webgis.co.davidson.nc.us
  959. // @connect webgis.co.humboldt.ca.us
  960. // @connect webgis.durhamnc.gov
  961. // @connect webgis.lafayetteassessor.com
  962. // @connect webgis.providenceri.gov
  963. // @connect webgis.sccgov.org
  964. // @connect webgis.waterburyct.org
  965. // @connect webmap.co.jackson.ms.us
  966. // @connect webmap.jeffparish.net
  967. // @connect webmap.trueautomation.com
  968. // @connect webmaps.elkgrovecity.org
  969. // @connect webmaps.sjcounty.net
  970. // @connect webserv.lincolnparish.org
  971. // @connect websrv19.clallam.net
  972. // @connect webstergis.integritygis.com
  973. // @connect wfs.ksdot.org
  974. // @connect wilsontn.geopowered.com
  975. // @connect wvsams.mapwv.org
  976. // @connect ww1.bucoks.com
  977. // @connect www.1stdistrict.org
  978. // @connect www.adacountyassessor.org
  979. // @connect www.adamscountyarcserver.com
  980. // @connect www.ancgis.com
  981. // @connect www.bartowgis.org
  982. // @connect www.bcad.org
  983. // @connect www.bcgis.com
  984. // @connect www.bcpao.us
  985. // @connect www.cameroncad.org
  986. // @connect www.centralilmaps.com
  987. // @connect www.cmbgis.com
  988. // @connect www.co.bingham.id.us
  989. // @connect www.co.coles.il.us
  990. // @connect www.co.pierce.wi.us
  991. // @connect www.co.steele.mn.us
  992. // @connect www.cobbgis.org
  993. // @connect www.ctgismaps2.ct.gov
  994. // @connect www.denvergov.org
  995. // @connect www.dmcwebgis.com
  996. // @connect www.efsedge.com
  997. // @connect www.finneycountygis.com
  998. // @connect www.franklinmo.net
  999. // @connect www.gcgis.org
  1000. // @connect www.gfgis.com
  1001. // @connect www.gis.bocc.co.st-johns.fl.us
  1002. // @connect www.gis.hctx.net
  1003. // @connect www.gisonline.ms.gov
  1004. // @connect www.greenwoodsc.gov
  1005. // @connect www.hernandocountygis-fl.us
  1006. // @connect www.hogarcmaps.org
  1007. // @connect www.horrycounty.org
  1008. // @connect www.iqmap.org
  1009. // @connect www.landmarkgeospatial.com
  1010. // @connect www.laurenscountygis.org
  1011. // @connect www.loraincountyauditor.com
  1012. // @connect www.mcc.co.mercer.pa.us
  1013. // @connect www.mcgisweb.org
  1014. // @connect www.mchenrycountygis.org
  1015. // @connect www.monroegis.org
  1016. // @connect www.mymanatee.org
  1017. // @connect www.ncpub.org
  1018. // @connect www.ocgis.com
  1019. // @connect www.ottertailcounty.net
  1020. // @connect www.sanduskycountygis.org
  1021. // @connect www.sciotocountyengineer.org
  1022. // @connect www.semogis.com
  1023. // @connect www.sjmap.org
  1024. // @connect www.skagitcounty.net
  1025. // @connect www.smithcountymapsite.org
  1026. // @connect www.tgisites.com
  1027. // @connect www.tremplocounty.com
  1028. // @connect www.unionco.org
  1029. // @connect www.valorgis.com
  1030. // @connect www.waynecounty.com
  1031. // @connect www.wcgis.us
  1032. // @connect www.webgis.net
  1033. // @connect www.wingis.org
  1034. // @connect www1.cityofwebster.com
  1035. // @connect www2.ci.lancaster.oh.us
  1036. // @connect www2.pottcounty.org
  1037. // @connect www3.multco.us
  1038. // @connect www7.co.union.oh.us
  1039. // ==/UserScript==
  1040.  
  1041. /* global OpenLayers */
  1042. /* global W */
  1043. /* global GM_info */
  1044. /* global WazeWrap */
  1045. /* global _ */
  1046. /* global $ */
  1047. /* global localStorage */
  1048. /* global GM_xmlhttpRequest */
  1049. /* global performance */
  1050. /* global atob */
  1051. /* global window */
  1052. /* global jQuery */
  1053.  
  1054. // **************************************************************************************************************
  1055. // IMPORTANT: Update this when releasing a new version of script that includes changes to the spreadsheet format
  1056. // that may cause old code to break. This # should match the version listed in the spreadsheet
  1057. // i.e. update them at the same time.
  1058.  
  1059. // const LAYER_DEF_VERSION = '2018.04.27.001'; // NOT ACTUALLY USED YET
  1060.  
  1061. // **************************************************************************************************************
  1062. // const UPDATE_MESSAGE = 'Bug fix due to WME update';
  1063. // const UPDATE_MESSAGE = `<ul>${[
  1064. // 'Added ability to shift layers. Right click a layer in the list to bring up the layer settings window.'
  1065. // ].map(item => `<li>${item}</li>`).join('')}</ul><br>`;
  1066. const GF_URL = 'https://greasyfork.org/scripts/369632-wme-gis-layers';
  1067. // Used in tooltips to tell people who to report issues to. Update if a new author takes ownership of this script.
  1068. const SCRIPT_AUTHOR = 'MapOMatic';
  1069. // const LAYER_INFO_URL = 'https://spreadsheets.google.com/feeds/list/1cEG3CvXSCI4TOZyMQTI50SQGbVhJ48Xip-jjWg4blWw/o7gusx3/public/values?alt=json';
  1070. const LAYER_DEF_SPREADSHEET_URL = 'https://sheets.googleapis.com/v4/spreadsheets/1cEG3CvXSCI4TOZyMQTI50SQGbVhJ48Xip-jjWg4blWw/values/layerDefs';
  1071. const API_KEY = 'YTJWNVBVRkplbUZUZVVGTlNXOWlVR1pWVjIxcE9VdHJNbVY0TTFoeWNrSlpXbFZuVmtWelRrMVVWUT09';
  1072. const REQUEST_FORM_URL = 'https://docs.google.com/forms/d/e/1FAIpQLSevPQLz2ohu_LTge9gJ9Nv6PURmCmaSSjq0ayOJpGdRr2xI0g/viewform?usp=pp_url&entry.2116052852={username}';
  1073. const DEC = s => atob(atob(s));
  1074. const PRIVATE_LAYERS = { 'nc-henderson-sl-signs': ['the_cre8r', 'mapomatic'] }; // case sensitive -- use all lower case
  1075. const COUNTRIES = {
  1076. 'United States': {
  1077. sheetId: '1cEG3CvXSCI4TOZyMQTI50SQGbVhJ48Xip-jjWg4blWw',
  1078. sheetLayerRange: 'layerDefs'
  1079. }
  1080. };
  1081. const DEFAULT_STYLE = {
  1082. fillColor: '#000',
  1083. pointRadius: 4,
  1084. label: '${label}',
  1085. strokeColor: '#ffa500',
  1086. strokeOpacity: '0.95',
  1087. strokeWidth: 1.5,
  1088. fontColor: '#ffc520',
  1089. fontSize: '13',
  1090. labelOutlineColor: 'black',
  1091. labelOutlineWidth: 3
  1092. };
  1093. const LAYER_STYLES = {
  1094. cities: {
  1095. fillOpacity: 0.3,
  1096. fillColor: '#f65',
  1097. strokeColor: '#f65',
  1098. fontColor: '#f62'
  1099. },
  1100. forests_parks: {
  1101. fillOpacity: 0.4,
  1102. fillColor: '#585',
  1103. strokeColor: '#484',
  1104. fontColor: '#8b8'
  1105. },
  1106. milemarkers: {
  1107. strokeColor: '#fff',
  1108. fontColor: '#fff',
  1109. fontWeight: 'bold',
  1110. fillOpacity: 0,
  1111. labelYOffset: 10,
  1112. pointRadius: 2,
  1113. fontSize: 12
  1114. },
  1115. parcels: {
  1116. fillOpacity: 0,
  1117. fillColor: '#ffa500'
  1118. },
  1119. points: {
  1120. strokeColor: '#000',
  1121. fontColor: '#0ff',
  1122. fillColor: '#0ff',
  1123. labelYOffset: -10,
  1124. labelAlign: 'ct'
  1125. },
  1126. post_offices: {
  1127. strokeColor: '#000',
  1128. fontColor: '#f84',
  1129. fillColor: '#f84',
  1130. fontWeight: 'bold',
  1131. labelYOffset: -10,
  1132. labelAlign: 'ct'
  1133. },
  1134. state_parcels: {
  1135. fillOpacity: 0,
  1136. strokeColor: '#e62',
  1137. fillColor: '#e62',
  1138. fontColor: '#e73'
  1139. },
  1140. state_points: {
  1141. strokeColor: '#000',
  1142. fontColor: '#3cf',
  1143. fillColor: '#3cf',
  1144. labelYOffset: -10,
  1145. labelAlign: 'ct'
  1146. },
  1147. road_labels: {
  1148. strokeOpacity: 0,
  1149. fillOpacity: 0,
  1150. fontColor: '#faf'
  1151. },
  1152. structures: {
  1153. fillOpacity: 0,
  1154. strokeColor: '#f7f',
  1155. fontColor: '#f7f'
  1156. }
  1157. };
  1158. const ROAD_STYLE = new OpenLayers.Style(
  1159. {
  1160. pointRadius: 12,
  1161. fillColor: '#369',
  1162. pathLabel: '${label}',
  1163. label: '',
  1164. fontColor: '#faf',
  1165. labelSelect: true,
  1166. pathLabelYOffset: '${getOffset}',
  1167. pathLabelCurve: '${getSmooth}',
  1168. pathLabelReadable: '${getReadable}',
  1169. labelAlign: '${getAlign}',
  1170. labelOutlineWidth: 3,
  1171. labelOutlineColor: '#000',
  1172. strokeWidth: 3,
  1173. stroke: true,
  1174. strokeColor: '#f0f',
  1175. strokeOpacity: 0.4,
  1176. fontWeight: 'bold',
  1177. fontSize: 11
  1178. }, {
  1179. context: {
  1180. getOffset() { return -(W.map.getZoom() + 5); },
  1181. getSmooth() { return ''; },
  1182. getReadable() { return '1'; },
  1183. getAlign() { return 'cb'; }
  1184. }
  1185. }
  1186. );
  1187. // eslint-disable-next-line no-unused-vars
  1188. const _regexReplace = {
  1189. // Strip leading zeros or blank full label for any label starting with a non-digit or
  1190. // is a Zero Address, use with '' as replace.
  1191. r0: /^(0+(\s.*)?|\D.*)/,
  1192. // Strip Everything After Street Type to end of the string by use $1 and $2 capture
  1193. // groups, use with replace '$1$2'
  1194. // eslint-disable-next-line max-len
  1195. r1: /^(.* )(Ave(nue)?|Dr(ive)?|St(reet)?|C(our)?t|Cir(cle)?|Blvd|Boulevard|Pl(ace)?|Ln|Lane|Fwy|Freeway|R(oa)?d|Ter(r|race)?|Tr(ai)?l|Way|Rte \d+|Route \d+)\b.*/gi,
  1196. // Strip SPACE 5 Digits from end of string, use with replace ''
  1197. r2: /\s\d{5}$/,
  1198. // Strip Everything after a "~", ",", ";" to the end of the string, use with replace ''
  1199. r3: /(~|,|;|\s?\r\n).*$/,
  1200. // Move the digits after the last space to before the rest of the string using, use with
  1201. // replace '$2 $1'
  1202. r4: /^(.*)\s(\d+).*/,
  1203. // Insert newline between digits (including "-") and everything after the digits,
  1204. // except(and before) a ",", use with replace '$1\n$2'
  1205. r5: /^([-\d]+)\s+([^,]+).*/,
  1206. // Insert newline between digits and everything after the digits, use with
  1207. // replace '$1\n$2'
  1208. r6: /^(\d+)\s+(.*)/
  1209. };
  1210.  
  1211. let _gisLayers = [];
  1212.  
  1213. const _layerRefinements = [
  1214. {
  1215. id: 'us-post-offices',
  1216. labelHeaderFields: ['LOCALE_NAME']
  1217. }
  1218. ];
  1219.  
  1220. const STATES = {
  1221. _states: [
  1222. ['US (Country)', 'US', -1], ['Alabama', 'AL', 1], ['Alaska', 'AK', 2],
  1223. ['American Samoa', 'AS', 60], ['Arizona', 'AZ', 4], ['Arkansas', 'AR', 5],
  1224. ['California', 'CA', 6], ['Colorado', 'CO', 8], ['Connecticut', 'CT', 9],
  1225. ['Delaware', 'DE', 10], ['District of Columbia', 'DC', 11], ['Florida', 'FL', 12],
  1226. ['Georgia', 'GA', 13], ['Guam', 'GU', 66], ['Hawaii', 'HI', 15], ['Idaho', 'ID', 16],
  1227. ['Illinois', 'IL', 17], ['Indiana', 'IN', 18], ['Iowa', 'IA', 19],
  1228. ['Kansas', 'KS', 20], ['Kentucky', 'KY', 21], ['Louisiana', 'LA', 22],
  1229. ['Maine', 'ME', 23], ['Maryland', 'MD', 24], ['Massachusetts', 'MA', 25],
  1230. ['Michigan', 'MI', 26], ['Minnesota', 'MN', 27], ['Mississippi', 'MS', 28],
  1231. ['Missouri', 'MO', 29], ['Montana', 'MT', 30], ['Nebraska', 'NE', 31],
  1232. ['Nevada', 'NV', 32], ['New Hampshire', 'NH', 33], ['New Jersey', 'NJ', 34],
  1233. ['New Mexico', 'NM', 35], ['New York', 'NY', 36], ['North Carolina', 'NC', 37],
  1234. ['North Dakota', 'ND', 38], ['Northern Mariana Islands', 'MP', 69], ['Ohio', 'OH', 39],
  1235. ['Oklahoma', 'OK', 40], ['Oregon', 'OR', 41], ['Pennsylvania', 'PA', 42],
  1236. ['Puerto Rico', 'PR', 72], ['Rhode Island', 'RI', 44], ['South Carolina', 'SC', 45],
  1237. ['South Dakota', 'SD', 46], ['Tennessee', 'TN', 47], ['Texas', 'TX', 48],
  1238. ['Utah', 'UT', 49], ['Vermont', 'VT', 50], ['Virgin Islands', 'VI', 78],
  1239. ['Virginia', 'VA', 51], ['Washington', 'WA', 53], ['West Virginia', 'WV', 54],
  1240. ['Wisconsin', 'WI', 55], ['Wyoming', 'WY', 56]
  1241. ],
  1242. toAbbr(fullName) { return this._states.find(a => a[0] === fullName)[1]; },
  1243. toFullName(abbr) { return this._states.find(a => a[1] === abbr)[0]; },
  1244. toFullNameArray() { return this._states.map(a => a[0]); },
  1245. toAbbrArray() { return this._states.map(a => a[1]); },
  1246. fromId(id) { return this._states.find(a => a[2] === id); }
  1247. };
  1248. const DEFAULT_VISIBLE_AT_ZOOM = 18;
  1249. const SETTINGS_STORE_NAME = 'wme_gis_layers_fl';
  1250. const COUNTIES_URL = 'https://tigerweb.geo.census.gov/arcgis/rest/services/Census2020/State_County/MapServer/1/';
  1251. const ALERT_UPDATE = false;
  1252. const SCRIPT_VERSION = GM_info.script.version;
  1253. const SCRIPT_VERSION_CHANGES = [];
  1254. let _mapLayer = null;
  1255. let _roadLayer = null;
  1256. let _settings = {};
  1257. let _ignoreFetch = false;
  1258. let _lastToken = {};
  1259.  
  1260. const DEBUG = true;
  1261. function log(message) { console.log('GIS Layers:', message); }
  1262. function logError(message) { console.error('GIS Layers:', message); }
  1263. function logDebug(message) { if (DEBUG) console.debug('GIS Layers:', message); }
  1264. // function logWarning(message) { console.warn('GIS Layers:', message); }
  1265.  
  1266. let _layerSettingsDialog;
  1267.  
  1268. class LayerSettingsDialog {
  1269. constructor() {
  1270. this._$titleText = $('<span>');
  1271. this._$closeButton = $('<span>', {
  1272. style: 'cursor:pointer;padding-left:4px;font-size:17px;color:#d6e6f3;float:right;',
  1273. class: 'fa fa-window-close'
  1274. }).click(() => this._onCloseButtonClick());
  1275. this._$shiftUpButton = LayerSettingsDialog._createShiftButton('fa-angle-up').click(() => this._onShiftButtonClick(0, 1));
  1276. this._$shiftLeftButton = LayerSettingsDialog._createShiftButton('fa-angle-left').click(() => this._onShiftButtonClick(-1, 0));
  1277. this._$shiftRightButton = LayerSettingsDialog._createShiftButton('fa-angle-right').click(() => this._onShiftButtonClick(1, 0));
  1278. this._$shiftDownButton = LayerSettingsDialog._createShiftButton('fa-angle-down').click(() => this._onShiftButtonClick(0, -1));
  1279. this._$resetButton = $('<button>', {
  1280. class: 'form-control',
  1281. style: 'height: 24px; width: auto; padding: 2px 6px 0px 6px; display: inline-block; float: right;'
  1282. }).text('Reset').click(() => this._onResetButtonClick());
  1283.  
  1284. this._dialogDiv = $('<div>', {
  1285. style: 'position: fixed; top: 15%; left: 400px; width: 200px; z-index: 100; background-color: #73a9bd; border-width: 1px; border-style: solid;'
  1286. + 'border-radius: 10px; box-shadow: 5px 5px 10px rgba(0, 0, 0, 0.7); border-color: #50667b; padding: 4px;'
  1287. }).append($('<div>').append( // The extra div is needed here. When the header text wraps, the main dialog div won't expand properly without it.
  1288. // HEADER
  1289. $('<div>', { style: 'border-radius:5px 5px 0px 0px; padding: 4px; color: #fff; font-weight: bold; text-align:left; cursor: default;' }).append(
  1290. this._$closeButton,
  1291. this._$titleText
  1292. ),
  1293. // BODY
  1294. $('<div>', { style: 'border-radius: 5px; width: 100%; padding: 4px; background-color:#d6e6f3; display:inline-block; margin-right:5px;' }).append(
  1295. this._$resetButton,
  1296. $('<input>', {
  1297. type: 'radio', id: 'gisLayerShiftAmt1', name: 'gisLayerShiftAmt', value: '1', checked: 'checked'
  1298. }),
  1299. $('<label>', { for: 'gisLayerShiftAmt1' }).text('1m'),
  1300. $('<input>', {
  1301. type: 'radio', id: 'gisLayerShiftAmt10', name: 'gisLayerShiftAmt', value: '10', style: 'margin-left: 6px'
  1302. }),
  1303. $('<label>', { for: 'gisLayerShiftAmt10' }).text('10m'),
  1304. $('<div>', { style: 'padding: 4px' }).append(
  1305. $('<table>', { style: 'table-layout:fixed; width:60px; height:84px; margin-left:auto;margin-right:auto;' }).append(
  1306. $('<tr>', { style: 'width: 20px; height: 28px;' }).append(
  1307. $('<td>', { align: 'center' }),
  1308. $('<td>', { align: 'center' }).append(this._$shiftUpButton),
  1309. $('<td>', { align: 'center' })
  1310. ),
  1311. $('<tr>', { style: 'width: 20px; height: 28px;' }).append(
  1312. $('<td>', { align: 'center' }).append(this._$shiftLeftButton),
  1313. $('<td>', { align: 'center' }),
  1314. $('<td>', { align: 'center' }).append(this._$shiftRightButton)
  1315. ),
  1316. $('<tr>', { style: 'width: 20px; height: 28px;' }).append(
  1317. $('<td>', { align: 'center' }),
  1318. $('<td>', { align: 'center' }).append(this._$shiftDownButton),
  1319. $('<td>', { align: 'center' })
  1320. )
  1321. )
  1322. )
  1323. )
  1324. ));
  1325.  
  1326. this.hide();
  1327. this._dialogDiv.appendTo('body');
  1328.  
  1329. if (typeof jQuery.ui !== 'undefined') {
  1330. const that = this;
  1331. this._dialogDiv.draggable({
  1332. // Gotta nuke the height setting the dragging inserts otherwise the panel cannot dynamically resize
  1333. stop() { that._dialogDiv.css('height', ''); }
  1334. });
  1335. }
  1336. }
  1337.  
  1338. get gisLayer() {
  1339. return this._gisLayer;
  1340. }
  1341.  
  1342. set gisLayer(value) {
  1343. if (value !== this._gisLayer) {
  1344. this._gisLayer = value;
  1345. this.title = value.name;
  1346. }
  1347. }
  1348.  
  1349. get title() {
  1350. return this._$titleText.text();
  1351. }
  1352.  
  1353. set title(value) {
  1354. this._$titleText.text(value);
  1355. }
  1356.  
  1357. // eslint-disable-next-line class-methods-use-this
  1358. getShiftAmount() {
  1359. return $('input[name=gisLayerShiftAmt]:checked').val();
  1360. }
  1361.  
  1362. show() {
  1363. this._dialogDiv.show();
  1364. }
  1365.  
  1366. hide() {
  1367. this._dialogDiv.hide();
  1368. }
  1369.  
  1370. _onCloseButtonClick() {
  1371. this.hide();
  1372. }
  1373.  
  1374. _onShiftButtonClick(x, y) {
  1375. const shiftAmount = this.getShiftAmount();
  1376. x *= shiftAmount;
  1377. y *= shiftAmount;
  1378. this._shiftLayerFeatures(x, y);
  1379. const { id } = this._gisLayer;
  1380. let offset = _settings.getLayerSetting(id, 'offset');
  1381. if (!offset) {
  1382. offset = { x: 0, y: 0 };
  1383. _settings.setLayerSetting(id, 'offset', offset);
  1384. }
  1385. offset.x += x;
  1386. offset.y += y;
  1387. saveSettingsToStorage();
  1388. }
  1389.  
  1390. _onResetButtonClick() {
  1391. const offset = _settings.getLayerSetting(this._gisLayer.id, 'offset');
  1392. if (offset) {
  1393. this._shiftLayerFeatures(offset.x * -1, offset.y * -1);
  1394. delete _settings.layers[this._gisLayer.id].offset;
  1395. saveSettingsToStorage();
  1396. }
  1397. }
  1398.  
  1399. _shiftLayerFeatures(x, y) {
  1400. const layer = this.gisLayer.isRoadLayer ? _roadLayer : _mapLayer;
  1401. layer.getFeaturesByAttribute('layerID', this.gisLayer.id).forEach(f => f.geometry.move(x, y));
  1402. layer.redraw();
  1403. }
  1404.  
  1405. static _createShiftButton(fontAwesomeClass) {
  1406. return $('<button>', {
  1407. class: 'form-control',
  1408. style: 'cursor:pointer;font-size:14px;padding: 3px;border-radius: 5px;width: 21px;height: 21px;'
  1409. }).append(
  1410. $('<i>', { class: 'fa', style: 'vertical-align: super' }).addClass(fontAwesomeClass)
  1411. );
  1412. }
  1413. }
  1414.  
  1415. function loadSettingsFromStorage() {
  1416. const loadedSettings = $.parseJSON(localStorage.getItem(SETTINGS_STORE_NAME));
  1417. const defaultSettings = {
  1418. lastVersion: null,
  1419. visibleLayers: [],
  1420. onlyShowApplicableLayers: false,
  1421. selectedStates: [],
  1422. enabled: true,
  1423. fillParcels: false,
  1424. toggleHnsOnlyShortcut: '',
  1425. oneTimeAlerts: {},
  1426. layers: {}
  1427. };
  1428. _settings = loadedSettings || defaultSettings;
  1429. Object.keys(defaultSettings).forEach(prop => {
  1430. if (!_settings.hasOwnProperty(prop)) {
  1431. _settings[prop] = defaultSettings[prop];
  1432. }
  1433. });
  1434.  
  1435. _settings.getLayerSetting = function getLayerSetting(layerID, settingName) {
  1436. const layerSettings = this.layers[layerID];
  1437. if (!layerSettings) {
  1438. return undefined;
  1439. }
  1440. return layerSettings[settingName];
  1441. };
  1442. _settings.setLayerSetting = function setLayerSetting(layerID, settingName, value) {
  1443. let layerSettings = this.layers[layerID];
  1444. if (!layerSettings) {
  1445. layerSettings = {};
  1446. this.layers[layerID] = layerSettings;
  1447. }
  1448. layerSettings[settingName] = value;
  1449. };
  1450. }
  1451.  
  1452. function saveSettingsToStorage() {
  1453. // Check for existance of action first, due to WME beta issue.
  1454. if (W.accelerators.Actions.GisLayersAddrDisplay) {
  1455. let keys = '';
  1456. const { shortcut } = W.accelerators.Actions.GisLayersAddrDisplay;
  1457. if (shortcut) {
  1458. if (shortcut.altKey) keys += 'A';
  1459. if (shortcut.shiftKey) keys += 'S';
  1460. if (shortcut.ctrlKey) keys += 'C';
  1461. if (keys.length) keys += '+';
  1462. if (shortcut.keyCode) keys += shortcut.keyCode;
  1463. }
  1464. _settings.toggleHnsOnlyShortcut = keys;
  1465. }
  1466. _settings.lastVersion = SCRIPT_VERSION;
  1467. localStorage.setItem(SETTINGS_STORE_NAME, JSON.stringify(_settings));
  1468. log('Settings saved');
  1469. }
  1470.  
  1471. function getUrl(extent, gisLayer) {
  1472. if (gisLayer.spatialReference) {
  1473. const proj = new OpenLayers.Projection(`EPSG:${gisLayer.spatialReference}`);
  1474. extent.transform(W.map.getProjectionObject(), proj);
  1475. }
  1476. let layerOffset = _settings.getLayerSetting(gisLayer.id, 'offset');
  1477. if (!layerOffset) {
  1478. layerOffset = { x: 0, y: 0 };
  1479. }
  1480. const geometry = {
  1481. xmin: extent.left - layerOffset.x,
  1482. ymin: extent.bottom - layerOffset.y,
  1483. xmax: extent.right - layerOffset.x,
  1484. ymax: extent.top - layerOffset.y,
  1485. spatialReference: {
  1486. wkid: gisLayer.spatialReference ? gisLayer.spatialReference : 102100,
  1487. latestWkid: gisLayer.spatialReference ? gisLayer.spatialReference : 3857
  1488. }
  1489. };
  1490. const geometryStr = JSON.stringify(geometry);
  1491. let fields = gisLayer.labelFields;
  1492. if (gisLayer.labelHeaderFields) {
  1493. fields = fields.concat(gisLayer.labelHeaderFields);
  1494. }
  1495. if (gisLayer.distinctFields) {
  1496. fields = fields.concat(gisLayer.distinctFields);
  1497. }
  1498. let url = `${gisLayer.url}/query?geometry=${encodeURIComponent(geometryStr)}`;
  1499. url += gisLayer.token ? `&token=${gisLayer.token}` : '';
  1500. url += `&outFields=${encodeURIComponent(fields.join(','))}`;
  1501. url += '&returnGeometry=true&spatialRel=esriSpatialRelIntersects&geometryType=esriGeometryEnvelope';
  1502. url += `&inSR=${gisLayer.spatialReference ? gisLayer.spatialReference : '102100'}`;
  1503. url += '&outSR=3857&f=json';
  1504. url += gisLayer.where ? `&where=${encodeURIComponent(gisLayer.where)}` : '';
  1505.  
  1506. logDebug(`Request URL: ${url}`);
  1507. return url;
  1508. }
  1509.  
  1510. function hashString(value) {
  1511. let hash = 0;
  1512. if (value.length === 0) return hash;
  1513. for (let i = 0; i < value.length; i++) {
  1514. const chr = value.charCodeAt(i);
  1515. // eslint-disable-next-line no-bitwise
  1516. hash = ((hash << 5) - hash) + chr;
  1517. // eslint-disable-next-line no-bitwise
  1518. hash |= 0; // Convert to 32bit integer
  1519. }
  1520. return hash;
  1521. }
  1522.  
  1523. function getCountiesUrl(extent) {
  1524. const geometry = {
  1525. xmin: extent.left,
  1526. ymin: extent.bottom,
  1527. xmax: extent.right,
  1528. ymax: extent.top,
  1529. spatialReference: { wkid: 102100, latestWkid: 3857 }
  1530. };
  1531. const url = `${COUNTIES_URL}/query?geometry=${encodeURIComponent(JSON.stringify(geometry))}`;
  1532. return `${url}&outFields=BASENAME%2CSTATE&returnGeometry=false&spatialRel=esriSpatialRelIntersects`
  1533. + '&geometryType=esriGeometryEnvelope&inSR=102100&outSR=3857&f=json';
  1534. }
  1535.  
  1536. let _countiesInExtent = [];
  1537. let _statesInExtent = [];
  1538.  
  1539. function getFetchableLayers(getInvisible) {
  1540. return _gisLayers.filter(gisLayer => {
  1541. const isValidUrl = gisLayer.url && gisLayer.url.trim().length > 0;
  1542. const isVisible = (getInvisible || _settings.visibleLayers.indexOf(gisLayer.id) > -1)
  1543. && _settings.selectedStates.indexOf(gisLayer.state) > -1;
  1544. const isInState = gisLayer.state === 'US' || _statesInExtent.indexOf(STATES.toFullName(gisLayer.state)) > -1;
  1545. // Be sure to use hasOwnProperty when checking this, since 0 is a valid value.
  1546. const isValidZoom = getInvisible || W.map.getZoom() >= (gisLayer.hasOwnProperty('visibleAtZoom')
  1547. ? gisLayer.visibleAtZoom : DEFAULT_VISIBLE_AT_ZOOM);
  1548. return isValidUrl && isInState && isVisible && isValidZoom;
  1549. });
  1550. }
  1551.  
  1552. function filterLayerCheckboxes() {
  1553. const applicableLayers = getFetchableLayers(true).filter(layer => {
  1554. const hasCounties = layer.hasOwnProperty('counties');
  1555. return (hasCounties && layer.counties.some(county => _countiesInExtent.indexOf(county.toLowerCase()) > -1))
  1556. || !hasCounties;
  1557. });
  1558. const statesToHide = STATES.toAbbrArray();
  1559.  
  1560. _gisLayers.forEach(gisLayer => {
  1561. const id = `#gis-layer-${gisLayer.id}-container`;
  1562. if (!_settings.onlyShowApplicableLayers || applicableLayers.indexOf(gisLayer) > -1) {
  1563. $(id).show();
  1564. $(`#gis-layers-for-${gisLayer.state}`).show();
  1565. const idx = statesToHide.indexOf(gisLayer.state);
  1566. if (idx > -1) statesToHide.splice(idx, 1);
  1567. } else {
  1568. $(id).hide();
  1569. }
  1570. });
  1571. if (_settings.onlyShowApplicableLayers) {
  1572. statesToHide.forEach(st => $(`#gis-layers-for-${st}`).hide());
  1573. }
  1574. }
  1575.  
  1576. const ROAD_ABBR = [
  1577. [/\bAVENUE$/, 'AVE'], [/\bCIRCLE$/, 'CIR'], [/\bCOURT$/, 'CT'], [/\bDRIVE$/, 'DR'],
  1578. [/\bLANE$/, 'LN'], [/\bPARK$/, 'PK'], [/\bPLACE$/, 'PL'], [/\bROAD$/, 'RD'], [/\bSTREET$/, 'ST'],
  1579. [/\bTERRACE$/, 'TER']
  1580. ];
  1581. function processFeatures(data, token, gisLayer) {
  1582. const features = [];
  1583. if (data.skipIt) {
  1584. // do nothing
  1585. } else if (data.error) {
  1586. logError(`Error in layer "${gisLayer.name}": ${data.error.message}`);
  1587. } else {
  1588. const items = data.features || [];
  1589. if (!token.cancel) {
  1590. let error = false;
  1591. const distinctValues = [];
  1592. items.forEach(item => {
  1593. let skipIt = false;
  1594. if (!token.cancel && !error) {
  1595. let feature;
  1596. let featureGeometry;
  1597. let area;
  1598. if (gisLayer.distinctFields) {
  1599. if (distinctValues.some(v => gisLayer.distinctFields.every(
  1600. fld => v[fld] === item.attributes[fld]
  1601. ))) {
  1602. skipIt = true;
  1603. } else {
  1604. const dist = {};
  1605. gisLayer.distinctFields.forEach(fld => (dist[fld] = item.attributes[fld]));
  1606. distinctValues.push(dist);
  1607. }
  1608. }
  1609. if (!skipIt) {
  1610. let layerOffset = _settings.getLayerSetting(gisLayer.id, 'offset');
  1611. if (!layerOffset) {
  1612. layerOffset = { x: 0, y: 0 };
  1613. }
  1614. // Special handling for this layer, because it doesn't have a geometry property.
  1615. // Coordinates are stored in the attributes.
  1616. if (gisLayer.id === 'nc-richmond-co-pts') {
  1617. const pt = new OpenLayers.Geometry.Point(item.attributes.XCOOR, item.attributes.YCOOR);
  1618. pt.transform(W.map.getOLMap().displayProjection, W.map.getProjectionObject());
  1619. item.geometry = pt;
  1620. }
  1621. if (item.geometry) {
  1622. if (item.geometry.x) {
  1623. featureGeometry = new OpenLayers.Geometry.Point(item.geometry.x + layerOffset.x,
  1624. item.geometry.y + layerOffset.y);
  1625. } else if (item.geometry.points) {
  1626. // @TODO Fix for multiple points instead of just grabbing first.
  1627. featureGeometry = new OpenLayers.Geometry.Point(item.geometry.points[0][0] + layerOffset.x,
  1628. item.geometry.points[0][1] + layerOffset.y);
  1629. } else if (item.geometry.rings) {
  1630. const rings = [];
  1631. item.geometry.rings.forEach(ringIn => {
  1632. const pnts = [];
  1633. for (let i = 0; i < ringIn.length; i++) {
  1634. pnts.push(new OpenLayers.Geometry.Point(ringIn[i][0] + layerOffset.x,
  1635. ringIn[i][1] + layerOffset.y));
  1636. }
  1637. rings.push(new OpenLayers.Geometry.LinearRing(pnts));
  1638. });
  1639. featureGeometry = new OpenLayers.Geometry.Polygon(rings);
  1640. if (gisLayer.areaToPoint) {
  1641. featureGeometry = featureGeometry.getCentroid();
  1642. } else {
  1643. area = featureGeometry.getArea();
  1644. }
  1645. } else if (data.geometryType === 'esriGeometryPolyline') {
  1646. const pointList = [];
  1647. item.geometry.paths.forEach(path => {
  1648. path.forEach(point => pointList.push(new OpenLayers.Geometry.Point(point[0] + layerOffset.x,
  1649. point[1] + layerOffset.y)));
  1650. });
  1651. featureGeometry = new OpenLayers.Geometry.LineString(pointList);
  1652. featureGeometry.skipDupeCheck = true;
  1653. } else {
  1654. logDebug(`Unexpected feature type in layer: ${JSON.stringify(item)}`);
  1655. logError(`Error: Unexpected feature type in layer "${gisLayer.name}"`);
  1656. error = true;
  1657. }
  1658. if (!error) {
  1659. const hasVisibleAtZoom = gisLayer.hasOwnProperty('visibleAtZoom');
  1660. const hasLabelsVisibleAtZoom = gisLayer.hasOwnProperty('labelsVisibleAtZoom');
  1661. const displayLabelsAtZoom = hasLabelsVisibleAtZoom ? gisLayer.labelsVisibleAtZoom
  1662. : (hasVisibleAtZoom ? gisLayer.visibleAtZoom : DEFAULT_VISIBLE_AT_ZOOM) + 1;
  1663. let label = '';
  1664. if (gisLayer.labelHeaderFields) {
  1665. label = `${gisLayer.labelHeaderFields.map(
  1666. fieldName => item.attributes[fieldName]
  1667. ).join(' ').trim()}\n`;
  1668. }
  1669. if (W.map.getZoom() >= displayLabelsAtZoom || area >= 5000) {
  1670. label += gisLayer.labelFields.map(
  1671. fieldName => item.attributes[fieldName]
  1672. ).join(' ').trim();
  1673. if (gisLayer.processLabel) {
  1674. label = gisLayer.processLabel(label, item.attributes);
  1675. label = label ? label.trim() : '';
  1676. }
  1677. }
  1678. if (label && [
  1679. LAYER_STYLES.points, LAYER_STYLES.parcels, LAYER_STYLES.state_points,
  1680. LAYER_STYLES.state_parcels
  1681. ].indexOf(gisLayer.style) > -1) {
  1682. if (_settings.addrLabelDisplay === 'hn') {
  1683. const m = label.match(/^\d+/);
  1684. label = m ? m[0] : '';
  1685. } else if (_settings.addrLabelDisplay === 'street') {
  1686. const m = label.match(/^(?:\d+\s)?(.*)/);
  1687. label = m ? m[1].trim() : '';
  1688. }
  1689. else if (_settings.addrLabelDisplay === 'none') {
  1690. label = '';
  1691. }
  1692. }
  1693. const attributes = {
  1694. layerID: gisLayer.id,
  1695. label
  1696. };
  1697. feature = new OpenLayers.Feature.Vector(featureGeometry, attributes);
  1698. features.push(feature);
  1699. }
  1700. }
  1701. }
  1702. }
  1703. });
  1704. }
  1705. }
  1706. if (!token.cancel) {
  1707. // Check for duplicate geometries.
  1708. for (let i = 0; i < features.length; i++) {
  1709. const f1 = features[i];
  1710. if (!f1.geometry.skipDupeCheck) {
  1711. const c1 = f1.geometry.getCentroid();
  1712. let labels = [f1.attributes.label];
  1713. for (let j = i + 1; j < features.length; j++) {
  1714. const f2 = features[j];
  1715. if (!f2.geometry.skipDupeCheck && f2.geometry.getCentroid().distanceTo(c1) < 1) {
  1716. features.splice(j, 1);
  1717. labels.push(f2.attributes.label);
  1718. j--;
  1719. }
  1720. }
  1721. labels = _.uniq(labels);
  1722. if (labels.length > 1) {
  1723. labels.forEach((label, idx) => {
  1724. label = label.replace(/\n/g, ' ').replace(/\s{2,}/, ' ').replace(/\bUNIT\s.{1,5}$/i, '').trim();
  1725. ROAD_ABBR.forEach(abbr => (label = label.replace(abbr[0], abbr[1])));
  1726. labels[idx] = label;
  1727. });
  1728. labels = _.uniq(labels);
  1729. labels.sort();
  1730. if (labels.length > 12) {
  1731. const len = labels.length;
  1732. labels = labels.slice(0, 10);
  1733. labels.push(`(${len - 10} more...)`);
  1734. }
  1735. f1.attributes.label = _.uniq(labels).join('\n');
  1736. } else {
  1737. let { label } = f1.attributes;
  1738. ROAD_ABBR.forEach(abbr => (label = label.replace(abbr[0], abbr[1])));
  1739. f1.attributes.label = label;
  1740. }
  1741. }
  1742. }
  1743.  
  1744. const layer = gisLayer.isRoadLayer ? _roadLayer : _mapLayer;
  1745. layer.removeFeatures(layer.getFeaturesByAttribute('layerID', gisLayer.id));
  1746. layer.addFeatures(features);
  1747.  
  1748. if (features.length) {
  1749. $(`label[for="gis-layer-${gisLayer.id}"]`).css({ color: '#00a009' });
  1750. }
  1751. }
  1752. } // END processFeatures()
  1753.  
  1754. function fetchFeatures() {
  1755. if (_ignoreFetch) return;
  1756. _lastToken.cancel = true;
  1757. _lastToken = { cancel: false, features: [], layersProcessed: 0 };
  1758. $('.gis-state-layer-label').css({ color: '#777' });
  1759.  
  1760. let _layersCleared = false;
  1761.  
  1762. // if (layersToFetch.length) {
  1763. const extent = W.map.getExtent();
  1764. GM_xmlhttpRequest({
  1765. url: getCountiesUrl(extent),
  1766. method: 'GET',
  1767. onload(res) {
  1768. if (res.status < 400) {
  1769. const data = $.parseJSON(res.responseText);
  1770. if (data.error) {
  1771. logError(`Error in US Census counties data: ${data.error.message}`);
  1772. } else {
  1773. _countiesInExtent = data.features.map(feature => feature.attributes.BASENAME.toLowerCase());
  1774. logDebug(`US Census counties: ${_countiesInExtent.join(', ')}`);
  1775. _statesInExtent = _.uniq(data.features.map(
  1776. // eslint-disable-next-line radix
  1777. feature => STATES.fromId(parseInt(feature.attributes.STATE, 10))[0]
  1778. ));
  1779.  
  1780. let layersToFetch;
  1781. if (!_layersCleared) {
  1782. _layersCleared = true;
  1783. layersToFetch = getFetchableLayers();
  1784.  
  1785. // Remove features of any layers that won't be mapped.
  1786. _gisLayers.forEach(gisLayer => {
  1787. if (layersToFetch.indexOf(gisLayer) === -1) {
  1788. _mapLayer.removeFeatures(_mapLayer.getFeaturesByAttribute('layerID', gisLayer.id));
  1789. _roadLayer.removeFeatures(_roadLayer.getFeaturesByAttribute('layerID', gisLayer.id));
  1790. }
  1791. });
  1792. }
  1793.  
  1794. layersToFetch = layersToFetch.filter(layer => !layer.hasOwnProperty('counties')
  1795. || layer.counties.some(county => _countiesInExtent.indexOf(county.toLowerCase()) > -1));
  1796. filterLayerCheckboxes();
  1797. logDebug(`Fetching ${layersToFetch.length} layers...`);
  1798. logDebug(layersToFetch);
  1799. layersToFetch.forEach(gisLayer => {
  1800. const url = getUrl(extent, gisLayer);
  1801. GM_xmlhttpRequest({
  1802. url,
  1803. context: _lastToken,
  1804. method: 'GET',
  1805. onload(res2) {
  1806. if (res2.status < 400) { // Handle stupid issue where http 4## is considered success
  1807. processFeatures($.parseJSON(res2.responseText), res2.context, gisLayer);
  1808. } else {
  1809. logDebug(`HTTP request error: ${JSON.stringify(res2)}`);
  1810. logError(`Could not fetch layer "${gisLayer.id}". Request returned ${res2.status}`);
  1811. }
  1812. },
  1813. onerror(res3) {
  1814. logDebug(`xmlhttpRequest error:${JSON.stringify(res3)}`);
  1815. logError(`Could not fetch layer "${gisLayer.id}". An error was thrown.`);
  1816. }
  1817. });
  1818. });
  1819. }
  1820. } else {
  1821. logDebug(`HTTP request error: ${JSON.stringify(res)}`);
  1822. logError(`Could not fetch counties from US Census site. Request returned ${res.status}`);
  1823. }
  1824. },
  1825. onerror(res) {
  1826. logDebug(`xmlhttpRequest error:${JSON.stringify(res)}`);
  1827. logError('Could not fetch counties from US Census site. An error was thrown.');
  1828. }
  1829. });
  1830. }
  1831.  
  1832. function showScriptInfoAlert() {
  1833. /* Check version and alert on update */
  1834. if (ALERT_UPDATE && SCRIPT_VERSION !== _settings.lastVersion) {
  1835. // alert(SCRIPT_VERSION_CHANGES);
  1836. let releaseNotes = '';
  1837. releaseNotes += '<p>What\'s New:</p>';
  1838. if (SCRIPT_VERSION_CHANGES.length > 0) {
  1839. releaseNotes += '<ul>';
  1840. for (let idx = 0; idx < SCRIPT_VERSION_CHANGES.length; idx++)
  1841. releaseNotes += `<li>${SCRIPT_VERSION_CHANGES[idx]}`;
  1842. releaseNotes += '</ul>';
  1843. }
  1844. else {
  1845. releaseNotes += '<ul><li>Nothing major.</ul>';
  1846. }
  1847. WazeWrap.Interface.ShowScriptUpdate(GM_info.script.name, SCRIPT_VERSION, releaseNotes, GF_URL);
  1848. }
  1849. }
  1850.  
  1851. function setEnabled(value) {
  1852. _settings.enabled = value;
  1853. saveSettingsToStorage();
  1854. _mapLayer.setVisibility(value);
  1855. _roadLayer.setVisibility(value);
  1856. const color = value ? '#00bd00' : '#ccc';
  1857. $('span#gis-layers-power-btn').css({ color });
  1858. if (value) fetchFeatures();
  1859. $('#layer-switcher-item_gis_layers').prop('checked', value);
  1860. }
  1861.  
  1862. function onGisLayerToggleChanged() {
  1863. const checked = $(this).is(':checked');
  1864. const layerId = $(this).data('layer-id');
  1865. const idx = _settings.visibleLayers.indexOf(layerId);
  1866. if (checked) {
  1867. const gisLayer = _gisLayers.find(l => l.id === layerId);
  1868. if (gisLayer.oneTimeAlert) {
  1869. const lastAlertHash = _settings.oneTimeAlerts[layerId];
  1870. const newAlertHash = hashString(gisLayer.oneTimeAlert);
  1871. if (lastAlertHash !== newAlertHash) {
  1872. // alert(`Layer: ${gisLayer.name}\n\nMessage:\n${gisLayer.oneTimeAlert}`);
  1873. WazeWrap.Alerts.info(GM_info.script.name, `Layer: ${gisLayer.name}<br><br>Message:<br>${gisLayer.oneTimeAlert}`);
  1874. _settings.oneTimeAlerts[layerId] = newAlertHash;
  1875. saveSettingsToStorage();
  1876. }
  1877. }
  1878. if (idx === -1) _settings.visibleLayers.push(layerId);
  1879. } else if (idx > -1) _settings.visibleLayers.splice(idx, 1);
  1880. if (!_ignoreFetch) {
  1881. saveSettingsToStorage();
  1882. fetchFeatures();
  1883. }
  1884. }
  1885.  
  1886. function onOnlyShowApplicableLayersChanged() {
  1887. _settings.onlyShowApplicableLayers = $(this).is(':checked');
  1888. saveSettingsToStorage();
  1889. fetchFeatures();
  1890. }
  1891.  
  1892. function onStateCheckChanged(evt) {
  1893. const state = evt.data;
  1894. const idx = _settings.selectedStates.indexOf(state);
  1895. if (evt.target.checked) {
  1896. if (idx === -1) _settings.selectedStates.push(state);
  1897. } else if (idx > -1) _settings.selectedStates.splice(idx, 1);
  1898. if (!_ignoreFetch) {
  1899. saveSettingsToStorage();
  1900. initLayersTab();
  1901. fetchFeatures();
  1902. }
  1903. }
  1904.  
  1905. function onLayerCheckboxChanged(checked) {
  1906. setEnabled(checked);
  1907. }
  1908.  
  1909. function setFillParcels(doFill) {
  1910. [LAYER_STYLES.parcels, LAYER_STYLES.state_parcels].forEach(style => {
  1911. style.fillOpacity = doFill ? 0.2 : 0;
  1912. });
  1913. }
  1914.  
  1915. function onFillParcelsCheckedChanged(evt) {
  1916. const { checked } = evt.target;
  1917. setFillParcels(checked);
  1918. _settings.fillParcels = checked;
  1919. saveSettingsToStorage();
  1920. fetchFeatures();
  1921. }
  1922.  
  1923. function onMapMove() {
  1924. if (_settings.enabled) fetchFeatures();
  1925. }
  1926.  
  1927. function onRefreshLayersClick() {
  1928. const $btn = $('#gis-layers-refresh');
  1929. if (!$btn.hasClass('fa-spin')) {
  1930. $btn.css({ cursor: 'auto' });
  1931. $btn.addClass('fa-spin');
  1932. init(false);
  1933. }
  1934. }
  1935.  
  1936. function onChevronClick(evt) {
  1937. const $target = $(evt.currentTarget);
  1938. $($target.children()[0])
  1939. .toggleClass('fa fa-fw fa-chevron-down')
  1940. .toggleClass('fa fa-fw fa-chevron-right');
  1941. $($target.siblings()[0]).toggleClass('collapse');
  1942. }
  1943.  
  1944. function doToggleABunch(evt, checkState) {
  1945. _ignoreFetch = true;
  1946. $(evt.target).closest('fieldset').find('input').prop('checked', !checkState).trigger('click');
  1947. _ignoreFetch = false;
  1948. saveSettingsToStorage();
  1949. if (evt.data) initLayersTab();
  1950. fetchFeatures();
  1951. }
  1952.  
  1953. function onSelectAllClick(evt) {
  1954. doToggleABunch(evt, true);
  1955. }
  1956.  
  1957. function onSelectNoneClick(evt) {
  1958. doToggleABunch(evt, false);
  1959. }
  1960.  
  1961. function onGisAddrDisplayChange(evt) {
  1962. _settings.addrLabelDisplay = evt.target.value;
  1963. saveSettingsToStorage();
  1964. fetchFeatures();
  1965. }
  1966.  
  1967. function onAddressDisplayShortcutKey() {
  1968. if (!$('#gisAddrDisplay-hn').is(':checked')) {
  1969. $('#gisAddrDisplay-hn').click();
  1970. } else {
  1971. $('#gisAddrDisplay-all').click();
  1972. }
  1973. }
  1974.  
  1975. function initLayer() {
  1976. const rules = _gisLayers.map(gisLayer => new OpenLayers.Rule({
  1977. filter: new OpenLayers.Filter.Comparison({
  1978. type: OpenLayers.Filter.Comparison.EQUAL_TO,
  1979. property: 'layerID',
  1980. value: gisLayer.id
  1981. }),
  1982. symbolizer: gisLayer.style
  1983. }));
  1984.  
  1985. setFillParcels(_settings.fillParcels);
  1986.  
  1987. const style = new OpenLayers.Style(DEFAULT_STYLE, { rules });
  1988. let existingLayer;
  1989. let uniqueName;
  1990.  
  1991. uniqueName = 'wmeGISLayersDefault';
  1992. existingLayer = W.map.getLayerByUniqueName(uniqueName);
  1993. if (existingLayer) W.map.removeLayer(existingLayer);
  1994. _mapLayer = new OpenLayers.Layer.Vector('GIS Layers - Default', {
  1995. uniqueName,
  1996. styleMap: new OpenLayers.StyleMap(style)
  1997. });
  1998.  
  1999. uniqueName = 'wmeGISLayersRoads';
  2000. existingLayer = W.map.getLayerByUniqueName(uniqueName);
  2001. if (existingLayer) W.map.removeLayer(existingLayer);
  2002. _roadLayer = new OpenLayers.Layer.Vector('GIS Layers - Roads', {
  2003. uniqueName,
  2004. styleMap: new OpenLayers.StyleMap(ROAD_STYLE)
  2005. });
  2006.  
  2007. _mapLayer.setVisibility(_settings.enabled);
  2008. _roadLayer.setVisibility(_settings.enabled);
  2009.  
  2010. W.map.addLayers([_roadLayer, _mapLayer]);
  2011. } // END InitLayer
  2012.  
  2013. function initLayersTab() {
  2014. const user = W.loginManager.user.userName.toLowerCase();
  2015. const states = _.uniq(_gisLayers.map(l => l.state)).filter(st => _settings.selectedStates.indexOf(st) > -1);
  2016.  
  2017. $('#panel-gis-state-layers').empty().append(
  2018. $('<div>', { class: 'controls-container' }).css({ 'padding-top': '0px' }).append(
  2019. $('<input>', { type: 'checkbox', id: 'only-show-applicable-gis-layers' }).change(
  2020. onOnlyShowApplicableLayersChanged
  2021. ).prop('checked', _settings.onlyShowApplicableLayers),
  2022. $('<label>', { for: 'only-show-applicable-gis-layers' })
  2023. .css({ 'white-space': 'pre-line' }).text('Only show applicable layers')
  2024. ),
  2025. $('.gis-layers-state-checkbox:checked').length === 0
  2026. ? $('<div>').text('Turn on layer categories in the Settings tab.')
  2027. : states.map(st => $('<fieldset>', {
  2028. id: `gis-layers-for-${st}`,
  2029. style: 'border:1px solid silver;padding:4px;border-radius:4px;-webkit-padding-before: 0;'
  2030. }).append(
  2031. $('<legend>', { style: 'margin-bottom:0px;border-bottom-style:none;width:auto;' })
  2032. .click(onChevronClick).append(
  2033. $('<i>', {
  2034. class: 'fa fa-fw fa-chevron-down',
  2035. style: 'cursor: pointer;font-size: 12px;margin-right: 4px'
  2036. }),
  2037. $('<span>', {
  2038. style: 'font-size:14px;font-weight:600;text-transform: uppercase; cursor: pointer'
  2039. }).text(STATES.toFullName(st))
  2040. ),
  2041. $('<div>', { id: `${st}_body` }).append(
  2042. $('<div>').css({ 'font-size': '11px' }).append(
  2043. $('<span>').append(
  2044. 'Select ',
  2045. $('<a>', { href: '#' })
  2046. .text('All')
  2047. .click(onSelectAllClick),
  2048. ' / ',
  2049. $('<a>', { href: '#' })
  2050. .text('None')
  2051. .click(onSelectNoneClick)
  2052. )
  2053. ),
  2054. $('<div>', { class: 'controls-container', style: 'padding-top:0px;' }).append(
  2055. _gisLayers.filter(l => (l.state === st && (!PRIVATE_LAYERS.hasOwnProperty(l.id)
  2056. || PRIVATE_LAYERS[l.id].indexOf(user) > -1)))
  2057. .map(gisLayer => {
  2058. const id = `gis-layer-${gisLayer.id}`;
  2059. return $('<div>', { class: 'controls-container', id: `${id}-container` })
  2060. .css({ 'padding-top': '0px', display: 'block' })
  2061. .append(
  2062. $('<input>', { type: 'checkbox', id })
  2063. .data('layer-id', gisLayer.id)
  2064. .change(onGisLayerToggleChanged)
  2065. .prop('checked', _settings.visibleLayers.indexOf(gisLayer.id) > -1),
  2066. $('<label>', { for: id, class: 'gis-state-layer-label' })
  2067. .css({ 'white-space': 'pre-line' })
  2068. .text(`${gisLayer.name}${gisLayer.restrictTo ? ' *' : ''}`)
  2069. .attr('title', gisLayer.restrictTo ? `Restricted to: ${gisLayer.restrictTo}` : '')
  2070. .contextmenu(evt => {
  2071. evt.preventDefault();
  2072. // TODO - enable the layer if it isn't already.
  2073. // Tried using click function on the evt target, but that didn't work.
  2074. _layerSettingsDialog.gisLayer = gisLayer;
  2075. _layerSettingsDialog.show();
  2076. })
  2077. );
  2078. })
  2079. )
  2080. )
  2081. ))
  2082. );
  2083. }
  2084.  
  2085. function initSettingsTab() {
  2086. const states = _.uniq(_gisLayers.map(l => l.state));
  2087. const createRadioBtn = (name, value, text, checked) => {
  2088. const id = `${name}-${value}`;
  2089. return [$('<input>', {
  2090. type: 'radio', id, name, value
  2091. }).prop('checked', checked), $('<label>', { for: id }).text(text).css({
  2092. paddingLeft: '15px', marginRight: '4px'
  2093. })];
  2094. };
  2095. $('#panel-gis-layers-settings').empty().append(
  2096. $('<fieldset>', {
  2097. style: 'border:1px solid silver;padding:8px;border-radius:4px;-webkit-padding-before: 0;margin-top:-8px;'
  2098. }).append(
  2099. $('<legend>', {
  2100. style: 'margin-bottom:0px;border-bottom-style:none;width:auto;'
  2101. }).append($('<span>', {
  2102. style: 'font-size:14px;font-weight:600;text-transform: uppercase;'
  2103. }).text('Labels')),
  2104. $('<div>', { id: 'labelSettings' }).append(
  2105. $('<div>', { class: 'controls-container' }).css({ 'padding-top': '2px' }).append(
  2106. $('<label>', { style: 'font-weight:normal;' }).text('Addresses:'),
  2107. createRadioBtn('gisAddrDisplay', 'hn', 'HN', _settings.addrLabelDisplay === 'hn'),
  2108. createRadioBtn('gisAddrDisplay', 'street', 'Street', _settings.addrLabelDisplay === 'street'),
  2109. createRadioBtn('gisAddrDisplay', 'all', 'Both', _settings.addrLabelDisplay === 'all'),
  2110. createRadioBtn('gisAddrDisplay', 'none', 'None', _settings.addrLabelDisplay === 'none'),
  2111. $('<i>', {
  2112. class: 'waze-tooltip',
  2113. id: 'gisAddrDisplayInfo',
  2114. 'data-toggle': 'tooltip',
  2115. style: 'margin-left:8px; font-size:12px',
  2116. 'data-placement': 'bottom',
  2117. title: `This may not work properly for all layers. Please report issues to ${SCRIPT_AUTHOR}.`
  2118. }).tooltip()
  2119. )
  2120. )
  2121. ),
  2122. $('<fieldset>', {
  2123. style: 'border:1px solid silver;padding:8px;border-radius:4px;-webkit-padding-before: 0;'
  2124. }).append(
  2125. $('<legend>', {
  2126. style: 'margin-bottom:0px;border-bottom-style:none;width:auto;'
  2127. }).append($('<span>', {
  2128. style: 'font-size:14px;font-weight:600;text-transform: uppercase;'
  2129. }).text('Layer Categories')),
  2130. $('<div>', { id: 'states_body' }).append(
  2131. $('<div>').css({ 'font-size': '11px' }).append(
  2132. $('<span>').append(
  2133. 'Select ',
  2134. $('<a>', { href: '#' }).text('All').click(true, onSelectAllClick),
  2135. ' / ',
  2136. $('<a>', { href: '#' }).text('None').click(true, onSelectNoneClick)
  2137. )
  2138. ),
  2139. $('<div>', { class: 'controls-container', style: 'padding-top:0px;' }).append(
  2140. states.map(st => {
  2141. const fullName = STATES.toFullName(st);
  2142. const id = `gis-layer-enable-state-${st}`;
  2143. return $('<div>', { class: 'controls-container' })
  2144. .css({ 'padding-top': '0px', display: 'block' })
  2145. .append(
  2146. $('<input>', { type: 'checkbox', id, class: 'gis-layers-state-checkbox' })
  2147. .change(st, onStateCheckChanged)
  2148. .prop('checked', _settings.selectedStates.indexOf(st) > -1),
  2149. $('<label>', { for: id }).css({ 'white-space': 'pre-line', color: '#777' }).text(fullName)
  2150. );
  2151. })
  2152. )
  2153. )
  2154. )
  2155. );
  2156. $('#panel-gis-layers-settings').append(
  2157. $('<fieldset>', { style: 'border:1px solid silver;padding:8px;border-radius:4px;-webkit-padding-before: 0;' })
  2158. .append(
  2159. $('<legend>', { style: 'margin-bottom:0px;border-bottom-style:none;width:auto;' })
  2160. .append(
  2161. $('<span>', { style: 'font-size:14px;font-weight:600;text-transform: uppercase;' })
  2162. .text('Appearance')
  2163. ),
  2164. $('<div>', { class: 'controls-container' }).css({ 'padding-top': '2px' }).append(
  2165. $('<input>', { type: 'checkbox', id: 'fill-parcels' })
  2166. .change(onFillParcelsCheckedChanged)
  2167. .prop('checked', _settings.fillParcels),
  2168. $('<label>', { for: 'fill-parcels' }).css({ 'white-space': 'pre-line', color: '#777' }).text('Fill parcels')
  2169. )
  2170. )
  2171. );
  2172. $('input[name=gisAddrDisplay]').change(onGisAddrDisplayChange);
  2173. }
  2174.  
  2175. function initTab(firstCall = true) {
  2176. initSettingsTab();
  2177. initLayersTab();
  2178. if (firstCall) {
  2179. if (!$('#gis-layers-power-btn').length) {
  2180. const color = _settings.enabled ? '#00bd00' : '#ccc';
  2181. $('a[href="#sidepanel-gis-l"]').prepend(
  2182. $('<span>', {
  2183. class: 'fa fa-power-off',
  2184. id: 'gis-layers-power-btn',
  2185. style: `margin-right: 5px;cursor: pointer;color: ${color};font-size: 13px;`,
  2186. title: 'Toggle GIS Layers'
  2187. }).click(evt => {
  2188. evt.stopPropagation();
  2189. setEnabled(!_settings.enabled);
  2190. })
  2191. );
  2192. }
  2193. $('#gis-layers-refresh').click(onRefreshLayersClick);
  2194. }
  2195. }
  2196.  
  2197. function initGui(firstCall = true) {
  2198. initLayer();
  2199.  
  2200. if (firstCall) {
  2201. const { user } = W.loginManager;
  2202. const content = $('<div>').append(
  2203. $('<span>', { style: 'font-size:14px;font-weight:600' }).text('GIS Layers'),
  2204. $('<span>', { style: 'font-size:11px;margin-left:10px;color:#aaa;' }).text(GM_info.script.version),
  2205. // <a href="https://docs.google.com/forms/d/e/1FAIpQLSevPQLz2ohu_LTge9gJ9Nv6PURmCmaSSjq0ayOJpGdRr2xI0g/viewform?usp=pp_url&entry.2116052852=test" target="_blank" style="color: #6290b7;font-size: 12px;margin-left: 8px;" title="Report broken layers, bugs, request new layers, script features">Report an issue</a>
  2206. $('<a>', {
  2207. href: REQUEST_FORM_URL.replace('{username}', user.userName),
  2208. target: '_blank',
  2209. style: 'color: #6290b7;font-size: 12px;margin-left: 8px;',
  2210. title: 'Report broken layers, bugs, request new layers, script features'
  2211. }).text('Submit a request'),
  2212. $('<span>', {
  2213. id: 'gis-layers-refresh',
  2214. class: 'fa fa-refresh',
  2215. style: 'float: right;',
  2216. 'data-toggle': 'tooltip',
  2217. title: 'Pull new layer info from master sheet and refresh all layers.'
  2218. }),
  2219. '<ul class="nav nav-tabs">'
  2220. + '<li class="active"><a data-toggle="tab" href="#panel-gis-state-layers" aria-expanded="true">'
  2221. + 'Layers'
  2222. + '</a></li>'
  2223. + '<li><a data-toggle="tab" href="#panel-gis-layers-settings" aria-expanded="true">'
  2224. + 'Settings'
  2225. + '</a></li> '
  2226. + '</ul>',
  2227. $('<div>', { class: 'tab-content', style: 'padding:8px;padding-top:2px' }).append(
  2228. $('<div>', { class: 'tab-pane active', id: 'panel-gis-state-layers', style: 'padding: 4px 0px 0px 0px' }),
  2229. $('<div>', { class: 'tab-pane', id: 'panel-gis-layers-settings', style: 'padding: 4px 0px 0px 0px' })
  2230. )
  2231. ).html();
  2232.  
  2233. new WazeWrap.Interface.Tab('GIS-L', content, initTab, null);
  2234. // Reduce panel div's padding to increase visible text space
  2235. $('#sidepanel-gis-l').css('padding', '6px');
  2236.  
  2237. WazeWrap.Interface.AddLayerCheckbox('Display', 'GIS Layers', _settings.enabled, onLayerCheckboxChanged);
  2238. // W.map.events.register('moveend', null, onMapMove);
  2239. WazeWrap.Events.register('moveend', null, onMapMove);
  2240. showScriptInfoAlert();
  2241. } else {
  2242. initTab(firstCall);
  2243. }
  2244. }
  2245.  
  2246. async function loadSpreadsheetAsync() {
  2247. let data;
  2248. try {
  2249. data = await $.getJSON(`${LAYER_DEF_SPREADSHEET_URL}?${DEC(API_KEY)}`);
  2250. } catch (err) {
  2251. throw new Error(`Spreadsheet call failed. (${err.status}: ${err.statusText})`);
  2252. }
  2253. const [[minVersion], fieldNames, ...layerDefRows] = data.values;
  2254. const REQUIRED_FIELD_NAMES = [
  2255. 'state', 'name', 'id', 'counties', 'url', 'where', 'labelFields',
  2256. 'processLabel', 'style', 'visibleAtZoom', 'labelsVisibleAtZoom', 'enabled',
  2257. 'restrictTo', 'oneTimeAlert'
  2258. ];
  2259. const result = { error: null };
  2260. const checkFieldNames = fldName => fieldNames.indexOf(fldName) > -1;
  2261.  
  2262. if (SCRIPT_VERSION < minVersion) {
  2263. result.error = `Script must be updated to at least version ${
  2264. minVersion} before layer definitions can be loaded.`;
  2265. } else if (fieldNames.length < REQUIRED_FIELD_NAMES.length) {
  2266. result.error = `Expected ${
  2267. REQUIRED_FIELD_NAMES.length} columns in layer definition data. Spreadsheet returned ${
  2268. fieldNames.length}.`;
  2269. } else if (!REQUIRED_FIELD_NAMES.every(fldName => checkFieldNames(fldName))) {
  2270. result.error = 'Script expected to see the following column names in the layer '
  2271. + `definition spreadsheet:\n${REQUIRED_FIELD_NAMES.join(', ')}\n`
  2272. + `But the spreadsheet returned these:\n${fieldNames.join(', ')}`;
  2273. }
  2274. if (!result.error) {
  2275. layerDefRows.filter(row => row.length).forEach(layerDefRow => {
  2276. const layerDef = { enabled: '0' };
  2277. fieldNames.forEach((fldName, fldIdx) => {
  2278. let value = layerDefRow[fldIdx];
  2279. if (value !== undefined && value.trim().length > 0) {
  2280. value = value.trim();
  2281. if (fldName === 'counties' || fldName === 'labelFields') {
  2282. value = value.split(',').map(item => item.trim());
  2283. } else if (fldName === 'processLabel') {
  2284. try {
  2285. // eslint-disable-next-line no-eval
  2286. value = eval(`(function(label, fieldValues){${value}})`);
  2287. } catch (ex) {
  2288. logError(`Error loading label processing function for layer "${
  2289. layerDef.id}".`);
  2290. logDebug(ex);
  2291. }
  2292. } else if (fldName === 'style') {
  2293. layerDef.isRoadLayer = value === 'roads';
  2294. if (LAYER_STYLES.hasOwnProperty(value)) {
  2295. value = LAYER_STYLES[value];
  2296. } else if (!layerDef.isRoadLayer) {
  2297. // If style is not defined, try to read in as JSON (custom style)
  2298. try {
  2299. value = JSON.parse(value);
  2300. } catch (ex) {
  2301. // ignore error
  2302. }
  2303. }
  2304. } else if (fldName === 'state') {
  2305. value = value ? value.toUpperCase() : value;
  2306. } else if (fldName === 'restrictTo') {
  2307. try {
  2308. const { user } = W.loginManager;
  2309. const values = value.split(',').map(v => v.trim().toLowerCase());
  2310. layerDef.notAllowed = !values.some(entry => {
  2311. const rankMatch = entry.match(/^r(\d)(\+am)?$/);
  2312. if (rankMatch) {
  2313. if (rankMatch[1] <= (user.rank + 1) && (!rankMatch[2] || user.isAreaManager)) {
  2314. return true;
  2315. }
  2316. } else if (entry === 'am' && user.isAreaManager) {
  2317. return true;
  2318. } else if (entry === user.userName.toLowerCase()) {
  2319. return true;
  2320. }
  2321. return false;
  2322. });
  2323. } catch (ex) {
  2324. logError(ex);
  2325. }
  2326. }
  2327. layerDef[fldName] = value;
  2328. } else if (fldName === 'labelFields') {
  2329. layerDef[fldName] = [''];
  2330. }
  2331. });
  2332. const enabled = layerDef.enabled && ['0', 'false', 'no', 'n'].indexOf(layerDef.enabled.toString().trim().toLowerCase()) === -1;
  2333. if (!layerDef.notAllowed && (enabled || layerDef.restrictTo)) {
  2334. _gisLayers.push(layerDef);
  2335. }
  2336. });
  2337. }
  2338.  
  2339. return result;
  2340. }
  2341.  
  2342. async function init(firstCall = true) {
  2343. _gisLayers = [];
  2344. if (firstCall) {
  2345. loadSettingsFromStorage();
  2346. installPathFollowingLabels();
  2347. // W.accelerators.events.listeners was removed in WME beta, so check for it here before calling WazeWrap.Interface.Shortcut
  2348. // Hopefully there will be a fix or workaround for this issue.
  2349. if (W.accelerators.events.listeners) {
  2350. new WazeWrap.Interface.Shortcut('GisLayersAddrDisplay', 'Toggle HN-only address labels (GIS Layers)',
  2351. 'layers', 'layersToggleGisAddressLabelDisplay', _settings.toggleHnsOnlyShortcut, onAddressDisplayShortcutKey, null).add();
  2352. }
  2353. window.addEventListener('beforeunload', saveSettingsToStorage, false);
  2354. _layerSettingsDialog = new LayerSettingsDialog();
  2355. }
  2356. const t0 = performance.now();
  2357. try {
  2358. const result = await loadSpreadsheetAsync();
  2359. if (result.error) {
  2360. logError(result.error);
  2361. return;
  2362. }
  2363. _layerRefinements.forEach(layerRefinement => {
  2364. const layerDef = _gisLayers.find(layerDef2 => layerDef2.id === layerRefinement.id);
  2365. if (layerDef) {
  2366. Object.keys(layerRefinement).forEach(fldName => {
  2367. const value = layerRefinement[fldName];
  2368. if (fldName !== 'id' && layerDef.hasOwnProperty(fldName)) {
  2369. logDebug(`The "${fldName}" property of layer "${
  2370. layerDef.id}" has a value hardcoded in the script, and also defined in the spreadsheet.`
  2371. + ' The spreadsheet value takes precedence.');
  2372. } else if (value) layerDef[fldName] = value;
  2373. });
  2374. } else {
  2375. logDebug(`Refined layer "${layerRefinement.id}" does not have a corresponding layer defined`
  2376. + ' in the spreadsheet. It can probably be removed from the script.');
  2377. }
  2378. });
  2379. logDebug(`Loaded ${_gisLayers.length} layer definitions in ${Math.round(performance.now() - t0)} ms.`);
  2380. initGui(firstCall);
  2381. fetchFeatures();
  2382. $('#gis-layers-refresh').removeClass('fa-spin').css({ cursor: 'pointer' });
  2383. log('Initialized.');
  2384. } catch (err) {
  2385. logError(err);
  2386. }
  2387. }
  2388.  
  2389. function bootstrap() {
  2390. if (W && W.loginManager && W.map && W.loginManager.user && W.model
  2391. && W.model.states && W.model.states.getObjectArray().length && WazeWrap && WazeWrap.Ready) {
  2392. log('Initializing...');
  2393. // WazeWrap.Interface.ShowScriptUpdate(GM_info.script.name, SCRIPT_VERSION, UPDATE_MESSAGE, GF_URL);
  2394. init();
  2395. } else {
  2396. log('Bootstrap failed. Trying again...');
  2397. setTimeout(() => {
  2398. bootstrap();
  2399. }, 1000);
  2400. }
  2401. }
  2402.  
  2403. bootstrap();
  2404.  
  2405. /*eslint-disable*/
  2406. function installPathFollowingLabels() {
  2407. // Copyright (c) 2015 by Jean-Marc.Viglino [at]ign.fr
  2408. // Dual-licensed under the CeCILL-B Licence (http://www.cecill.info/)
  2409. // and the Beerware license (http://en.wikipedia.org/wiki/Beerware),
  2410. // feel free to use and abuse it in your projects (the code, not the beer ;-).
  2411. //
  2412. //* Overwrite the SVG function to allow text along a path
  2413. //* setStyle function
  2414. //*
  2415. //* Add new options to the Openlayers.Style
  2416.  
  2417. // pathLabel: {String} Label to draw on the path
  2418. // pathLabelXOffset: {String} Offset along the line to start drawing text in pixel or %, default: "50%"
  2419. // pathLabelYOffset: {Number} Distance of the line to draw the text
  2420. // pathLabelCurve: {String} Smooth the line the label is drawn on (empty string for no)
  2421. // pathLabelReadable: {String} Make the label readable (empty string for no)
  2422.  
  2423. // * Extra standard values : all label and text values
  2424.  
  2425.  
  2426. // *
  2427. // * Method: removeChildById
  2428. // * Remove child in a node.
  2429. // *
  2430.  
  2431. function removeChildById(node, id) {
  2432. if (node.querySelector) {
  2433. var c = node.querySelector('#' + id);
  2434. if (c) node.removeChild(c);
  2435. return;
  2436. }
  2437. // For old browsers
  2438. var c = node.childNodes;
  2439. if (c) for (var i = 0; i < c.length; i++) {
  2440. if (c[i].id === id) {
  2441. node.removeChild(c[i]);
  2442. return;
  2443. }
  2444. }
  2445. }
  2446.  
  2447.  
  2448. // *
  2449. // * Method: setStyle
  2450. // * Use to set all the style attributes to a SVG node.
  2451. // *
  2452. // * Takes care to adjust stroke width and point radius to be
  2453. // * resolution-relative
  2454. // *
  2455. // * Parameters:
  2456. // * node - {SVGDomElement} An SVG element to decorate
  2457. // * style - {Object}
  2458. // * options - {Object} Currently supported options include
  2459. // * 'isFilled' {Boolean} and
  2460. // * 'isStroked' {Boolean}
  2461.  
  2462. var setStyle = OpenLayers.Renderer.SVG.prototype.setStyle;
  2463. OpenLayers.Renderer.SVG.LABEL_STARTOFFSET = { 'l': '0%', 'r': '100%', 'm': '50%' };
  2464.  
  2465. OpenLayers.Renderer.SVG.prototype.pathText = function (node, style, suffix) {
  2466. var label = this.nodeFactory(null, 'text');
  2467. label.setAttribute('id', node._featureId + '_' + suffix);
  2468. if (style.fontColor) label.setAttributeNS(null, 'fill', style.fontColor);
  2469. if (style.fontStrokeColor) label.setAttributeNS(null, 'stroke', style.fontStrokeColor);
  2470. if (style.fontStrokeWidth) label.setAttributeNS(null, 'stroke-width', style.fontStrokeWidth);
  2471. if (style.fontOpacity) label.setAttributeNS(null, 'opacity', style.fontOpacity);
  2472. if (style.fontFamily) label.setAttributeNS(null, 'font-family', style.fontFamily);
  2473. if (style.fontSize) label.setAttributeNS(null, 'font-size', style.fontSize);
  2474. if (style.fontWeight) label.setAttributeNS(null, 'font-weight', style.fontWeight);
  2475. if (style.fontStyle) label.setAttributeNS(null, 'font-style', style.fontStyle);
  2476. if (style.labelSelect === true) {
  2477. label.setAttributeNS(null, 'pointer-events', 'visible');
  2478. label._featureId = node._featureId;
  2479. } else {
  2480. label.setAttributeNS(null, 'pointer-events', 'none');
  2481. }
  2482.  
  2483. function getpath(pathStr, readeable) {
  2484. var npath = pathStr.split(',');
  2485. var pts = [];
  2486. if (!readeable || Number(npath[0]) - Number(npath[npath.length - 2]) < 0) {
  2487. while (npath.length) pts.push({ x: Number(npath.shift()), y: Number(npath.shift()) });
  2488. } else {
  2489. while (npath.length) pts.unshift({ x: Number(npath.shift()), y: Number(npath.shift()) });
  2490. }
  2491. return pts;
  2492. }
  2493.  
  2494. var path = this.nodeFactory(null, 'path');
  2495. var tpid = node._featureId + '_t' + suffix;
  2496. var tpath = node.getAttribute('points');
  2497. if (style.pathLabelCurve) {
  2498. var pts = getpath(tpath, style.pathLabelReadable);
  2499. var p = pts[0].x + ' ' + pts[0].y;
  2500. var dx, dy, s1, s2;
  2501. dx = (pts[0].x - pts[1].x) / 4;
  2502. dy = (pts[0].y - pts[1].y) / 4;
  2503. for (var i = 1; i < pts.length - 1; i++) {
  2504. p += ' C ' + (pts[i - 1].x - dx) + ' ' + (pts[i - 1].y - dy);
  2505. dx = (pts[i - 1].x - pts[i + 1].x) / 4;
  2506. dy = (pts[i - 1].y - pts[i + 1].y) / 4;
  2507. s1 = Math.sqrt(Math.pow(pts[i - 1].x - pts[i].x, 2) + Math.pow(pts[i - 1].y - pts[i].y, 2));
  2508. s2 = Math.sqrt(Math.pow(pts[i + 1].x - pts[i].x, 2) + Math.pow(pts[i + 1].y - pts[i].y, 2));
  2509. p += ' ' + (pts[i].x + s1 * dx / s2) + ' ' + (pts[i].y + s1 * dy / s2);
  2510. dx *= s2 / s1;
  2511. dy *= s2 / s1;
  2512. p += ' ' + pts[i].x + ' ' + pts[i].y;
  2513. }
  2514. p += ' C ' + (pts[i - 1].x - dx) + ' ' + (pts[i - 1].y - dy);
  2515. dx = (pts[i - 1].x - pts[i].x) / 4;
  2516. dy = (pts[i - 1].y - pts[i].y) / 4;
  2517. p += ' ' + (pts[i].x + dx) + ' ' + (pts[i].y + dy);
  2518. p += ' ' + pts[i].x + ' ' + pts[i].y;
  2519.  
  2520. path.setAttribute('d', 'M ' + p);
  2521. } else {
  2522. if (style.pathLabelReadable) {
  2523. var pts = getpath(tpath, style.pathLabelReadable);
  2524. var p = '';
  2525. for (var i = 0; i < pts.length; i++) p += ' ' + pts[i].x + ' ' + pts[i].y;
  2526. path.setAttribute('d', 'M ' + p);
  2527. } else path.setAttribute('d', 'M ' + tpath);
  2528. }
  2529. path.setAttribute('id', tpid);
  2530.  
  2531. var defs = this.createDefs();
  2532. removeChildById(defs, tpid);
  2533. defs.appendChild(path);
  2534.  
  2535. var textPath = this.nodeFactory(null, 'textPath');
  2536. textPath.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href', '#' + tpid);
  2537. var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign;
  2538. label.setAttributeNS(null, 'text-anchor', OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || 'middle');
  2539. textPath.setAttribute('startOffset', style.pathLabelXOffset || OpenLayers.Renderer.SVG.LABEL_STARTOFFSET[align[0]] || '50%');
  2540. label.setAttributeNS(null, 'dominant-baseline', OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || 'central');
  2541. if (style.pathLabelYOffset) label.setAttribute('dy', style.pathLabelYOffset);
  2542. //textPath.setAttribute('method','stretch');
  2543. //textPath.setAttribute('spacing','auto');
  2544.  
  2545. textPath.textContent = style.pathLabel;
  2546. label.appendChild(textPath);
  2547.  
  2548. removeChildById(this.textRoot, node._featureId + '_' + suffix);
  2549. this.textRoot.appendChild(label);
  2550. };
  2551.  
  2552. OpenLayers.Renderer.SVG.prototype.setStyle = function (node, style, options) {
  2553. if (node._geometryClass === 'OpenLayers.Geometry.LineString' && style.pathLabel) {
  2554. if (node._geometryClass === 'OpenLayers.Geometry.LineString' && style.pathLabel) {
  2555. var drawOutline = (!!style.labelOutlineWidth);
  2556. // First draw text in halo color and size and overlay the
  2557. // normal text afterwards
  2558. if (drawOutline) {
  2559. var outlineStyle = OpenLayers.Util.extend({}, style);
  2560. outlineStyle.fontColor = outlineStyle.labelOutlineColor;
  2561. outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor;
  2562. outlineStyle.fontStrokeWidth = style.labelOutlineWidth;
  2563. if (style.labelOutlineOpacity) outlineStyle.fontOpacity = style.labelOutlineOpacity;
  2564. delete outlineStyle.labelOutlineWidth;
  2565. this.pathText(node, outlineStyle, 'txtpath0');
  2566. }
  2567. this.pathText(node, style, 'txtpath');
  2568. setStyle.apply(this, arguments);
  2569. }
  2570. } else setStyle.apply(this, arguments);
  2571. return node;
  2572. };
  2573.  
  2574. // *
  2575. // * Method: drawGeometry
  2576. // * Remove the textpath if no geometry is drawn.
  2577. // *
  2578. // * Parameters:
  2579. // * geometry - {<OpenLayers.Geometry>}
  2580. // * style - {Object}
  2581. // * featureId - {String}
  2582. // *
  2583. // * Returns:
  2584. // * {Boolean} true if the geometry has been drawn completely; null if
  2585. // * incomplete; false otherwise
  2586.  
  2587. var drawGeometry = OpenLayers.Renderer.SVG.prototype.drawGeometry;
  2588. OpenLayers.Renderer.SVG.prototype.drawGeometry = function (geometry, style, id) {
  2589. var rendered = drawGeometry.apply(this, arguments);
  2590. if (rendered === false) {
  2591. removeChildById(this.textRoot, id + '_txtpath');
  2592. removeChildById(this.textRoot, id + '_txtpath0');
  2593. }
  2594. return rendered;
  2595. };
  2596.  
  2597. // *
  2598. // * Method: eraseGeometry
  2599. // * Erase a geometry from the renderer. In the case of a multi-geometry,
  2600. // * we cycle through and recurse on ourselves. Otherwise, we look for a
  2601. // * node with the geometry.id, destroy its geometry, and remove it from
  2602. // * the DOM.
  2603. // *
  2604. // * Parameters:
  2605. // * geometry - {<OpenLayers.Geometry>}
  2606. // * featureId - {String}
  2607.  
  2608. var eraseGeometry = OpenLayers.Renderer.SVG.prototype.eraseGeometry;
  2609. OpenLayers.Renderer.SVG.prototype.eraseGeometry = function (geometry, featureId) {
  2610. eraseGeometry.apply(this, arguments);
  2611. removeChildById(this.textRoot, featureId + '_txtpath');
  2612. removeChildById(this.textRoot, featureId + '_txtpath0');
  2613. };
  2614.  
  2615. }