WME Form Filler

Use info from WME to automatically fill out related forms

Tính đến 24-06-2019. Xem phiên bản mới nhất.

  1. // ==UserScript==
  2. // @name WME Form Filler
  3. // @description Use info from WME to automatically fill out related forms
  4. // @namespace https://greasyfork.org/users/6605
  5. // @version 1.4.4.6
  6. // @homepage https://github.com/WazeDev/WME-Form-Filler
  7. // @supportURL https://github.com/WazeDev/WME-Form-Filler/issues
  8. // @include https://www.waze.com/editor
  9. // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
  10. // @author crazycaveman
  11. // @license MIT
  12. // @run-at document-end
  13. // @icon data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyJpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoV2luZG93cykiIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6NUIzRDdFNzAwRTlGMTFFNkIyRDZGMzNERUFDMUM1NDgiIHhtcE1NOkRvY3VtZW50SUQ9InhtcC5kaWQ6NUIzRDdFNzEwRTlGMTFFNkIyRDZGMzNERUFDMUM1NDgiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDo1QjNEN0U2RTBFOUYxMUU2QjJENkYzM0RFQUMxQzU0OCIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDo1QjNEN0U2RjBFOUYxMUU2QjJENkYzM0RFQUMxQzU0OCIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/PtdrqLIAAAOCSURBVHjatJdLaBNBGMdn81ITQ2mLNqlIKInGkpSgFooPpKGHIlZPJqgoIvQivooPfFUsxYNVW62KiuItPah48RaUSlFEaDzEqgeV+sBUxag1rZa8Nv6nfMKaZpNssx34kdnN7sx/5nvNCtFolAUCAW8ymdwtCIKZMZZh6jQBjIIL4EkikWA+n485HI7/HtLFYjHH0NDQnXQ6XQkBbAbaGrAsHo+PNDc3T/lTF4lEVouiWKnX62O4DgBRpYn1YBuoAkuxwBGNRjNVAP7QZjKTu/4V7FJ59euAEWjlHtBg2//ZXEOq1WpzaUyWz690Ms5jUDhZcrqm0+UYyAPuKxyH2zqohgC++j/gmcJxRot5yGAwFBTAr9+AtWrHIqKMBYNBFgqFmlKpVLympuax1+udIiBD9reUON/nLMcTtFotC4fDBoR8H5JSVV1dXVdjY+OxbAEpUA8GSxRQC95KrkUe6jABzwnbkQ/OmUwmPSJQzBbAw+YL6C5hcr7y72TO2TTeI9AA7oErELNBp9NF5HzgEziogtnNZM6j4BfoAfNBB0XZu1wCuAkWggMKJ7sBXmbd4xmwH1wHW8FKut/Hi5NcFIiUu/cpFNCfQ0ACHKGdOM0LHbx/DE54HJWX92XDMEy7oKRFc9z7SXSBBZiYWSyWs0aj8QMvzVarVTYRJckP1GhLwN7J7Ugkhv1+f4/H48mbiPi+LOKeWmIUtIKP4AxFAiNn/F0oE/KX5wB3iQLSoAmsp3sPwW0q+3kF8HL8HFgVTqrJqvn8upf6XMyhYotRhsKnVqEAXj9ikuv9wEX9m/mKWy4fWA6eKhSwEdylfjVop/4PcEJJOebbOAz2KDgdC1kr7ATl1O+gUFwMXhcjgNvuG7g8TQdcAXZQ/xW4RLvxXk6ARmagBhqoTHJvFdhCUSLX2iVjttH7h8GEEh9wggEwiw4mfirRA2Sia2CnzHjV9HuLCs5FOpwmlERBGU3Om41+50nCzJ4nFCM0eTftlI/XAJ778WHCeDou5kg2SNtXLzkXPAAnKbWeymOCzWCM+nwRGT55S0sLs9lszG635xQgSCphivq9OU7LnQUcUJRMzts4T0JozOl0Ci6Xq6AJ+GHhPGUuNRo/jJRTGZbPAyiLIZzRJrBVZXi4bQY+TsfNZvMLWQEVFRVht9u9CQJaIcCk8uf5GArQVRxAh+Ue+ivAAAY7DIf3WTuXAAAAAElFTkSuQmCC
  14. // @grant none
  15. // ==/UserScript==
  16.  
  17. /*global W, $, GM_info */
  18.  
  19. (function () {
  20. "use strict";
  21. var WMEFFName = GM_info.script.name;
  22. var WMEFFVersion = GM_info.script.version;
  23. var WMEFFIcon = "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyJpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoV2luZG93cykiIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6NUIzRDdFNzAwRTlGMTFFNkIyRDZGMzNERUFDMUM1NDgiIHhtcE1NOkRvY3VtZW50SUQ9InhtcC5kaWQ6NUIzRDdFNzEwRTlGMTFFNkIyRDZGMzNERUFDMUM1NDgiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDo1QjNEN0U2RTBFOUYxMUU2QjJENkYzM0RFQUMxQzU0OCIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDo1QjNEN0U2RjBFOUYxMUU2QjJENkYzM0RFQUMxQzU0OCIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/PtdrqLIAAAOCSURBVHjatJdLaBNBGMdn81ITQ2mLNqlIKInGkpSgFooPpKGHIlZPJqgoIvQivooPfFUsxYNVW62KiuItPah48RaUSlFEaDzEqgeV+sBUxag1rZa8Nv6nfMKaZpNssx34kdnN7sx/5nvNCtFolAUCAW8ymdwtCIKZMZZh6jQBjIIL4EkikWA+n485HI7/HtLFYjHH0NDQnXQ6XQkBbAbaGrAsHo+PNDc3T/lTF4lEVouiWKnX62O4DgBRpYn1YBuoAkuxwBGNRjNVAP7QZjKTu/4V7FJ59euAEWjlHtBg2//ZXEOq1WpzaUyWz690Ms5jUDhZcrqm0+UYyAPuKxyH2zqohgC++j/gmcJxRot5yGAwFBTAr9+AtWrHIqKMBYNBFgqFmlKpVLympuax1+udIiBD9reUON/nLMcTtFotC4fDBoR8H5JSVV1dXVdjY+OxbAEpUA8GSxRQC95KrkUe6jABzwnbkQ/OmUwmPSJQzBbAw+YL6C5hcr7y72TO2TTeI9AA7oErELNBp9NF5HzgEziogtnNZM6j4BfoAfNBB0XZu1wCuAkWggMKJ7sBXmbd4xmwH1wHW8FKut/Hi5NcFIiUu/cpFNCfQ0ACHKGdOM0LHbx/DE54HJWX92XDMEy7oKRFc9z7SXSBBZiYWSyWs0aj8QMvzVarVTYRJckP1GhLwN7J7Ugkhv1+f4/H48mbiPi+LOKeWmIUtIKP4AxFAiNn/F0oE/KX5wB3iQLSoAmsp3sPwW0q+3kF8HL8HFgVTqrJqvn8upf6XMyhYotRhsKnVqEAXj9ikuv9wEX9m/mKWy4fWA6eKhSwEdylfjVop/4PcEJJOebbOAz2KDgdC1kr7ATl1O+gUFwMXhcjgNvuG7g8TQdcAXZQ/xW4RLvxXk6ARmagBhqoTHJvFdhCUSLX2iVjttH7h8GEEh9wggEwiw4mfirRA2Sia2CnzHjV9HuLCs5FOpwmlERBGU3Om41+50nCzJ4nFCM0eTftlI/XAJ778WHCeDou5kg2SNtXLzkXPAAnKbWeymOCzWCM+nwRGT55S0sLs9lszG635xQgSCphivq9OU7LnQUcUJRMzts4T0JozOl0Ci6Xq6AJ+GHhPGUuNRo/jJRTGZbPAyiLIZzRJrBVZXi4bQY+TsfNZvMLWQEVFRVht9u9CQJaIcCk8uf5GArQVRxAh+Ue+ivAAAY7DIf3WTuXAAAAAElFTkSuQmCC";
  24. var forms = [];
  25.  
  26. function formfiller_bootstrap() {
  27. formfiller_log("Running bootstrap");
  28.  
  29. if (typeof W.app === "undefined" || !window.W.map) {
  30. setTimeout(formfiller_bootstrap, 500);
  31. return;
  32. }
  33. formfiller_log("Starting init");
  34. formfiller_init();
  35. }
  36.  
  37. function formfiller_init() {
  38. // Check document elements are ready
  39. var userInfo = document.getElementById("user-info");
  40. if (userInfo === null) {
  41. window.setTimeout(formfiller_init, 500);
  42. return;
  43. }
  44. var userTabs = document.getElementById("user-tabs");
  45. if (userTabs === null) {
  46. window.setTimeout(formfiller_init, 500);
  47. return;
  48. }
  49. var navTab = userInfo.getElementsByTagName("ul");
  50. if (navTab.length === 0) {
  51. window.setTimeout(formfiller_init, 500);
  52. return;
  53. }
  54. if (typeof navTab[0] === "undefined") {
  55. window.setTimeout(formfiller_init, 500);
  56. return;
  57. }
  58. var tabContent = userInfo.getElementsByTagName("div");
  59. if (tabContent.length === 0) {
  60. window.setTimeout(formfiller_init, 500);
  61. return;
  62. }
  63. if (typeof tabContent[0] === "undefined") {
  64. window.setTimeout(formfiller_init, 500);
  65. return;
  66. }
  67.  
  68. ff_addUserTab();
  69. ff_addFormBtn();
  70. var formFillerObserver = new MutationObserver(function (mutations) {
  71. mutations.forEach(function (mutation) {
  72. // Mutation is a NodeList and doesn't support forEach like an array
  73. for (var i = 0; i < mutation.addedNodes.length; i += 1) {
  74. var addedNode = mutation.addedNodes[i];
  75.  
  76. // Only fire up if it's a node
  77. if (addedNode.nodeType === Node.ELEMENT_NODE) {
  78. var selectionDiv = addedNode.querySelector("div.selection");
  79.  
  80. if (selectionDiv) {
  81. ff_addFormBtn();
  82. }
  83. }
  84. }
  85. });
  86. });
  87. formFillerObserver.observe(document.getElementById("edit-panel"), {
  88. childList: true,
  89. subtree: true
  90. });
  91. //W.selectionManager.events.register("selectionchanged", null, ff_addFormBtn);
  92. if (W.app.modeController) {
  93. W.app.modeController.model.bind("change:mode", function (model, modeId) {
  94. if (modeId === 0) {
  95. ff_addUserTab();
  96. }
  97. });
  98. }
  99.  
  100. // Unit switched (imperial/metric)
  101. if (W.prefs) {
  102. W.prefs.on("change:isImperial", ff_addUserTab);
  103. }
  104.  
  105. if (!W.selectionManager.getSelectedFeatures) {
  106. W.selectionManager.getSelectedFeatures = W.selectionManager.getSelectedItems;
  107. }
  108. formfiller_log("Init done");
  109. return;
  110. }
  111.  
  112. //Shamelessly copied from https://gist.github.com/CalebGrove/c285a9510948b633aa47
  113. function abbrState(input, to) {
  114. var states = [
  115. ["ALABAMA", "AL"],
  116. ["ALASKA", "AK"],
  117. ["ARIZONA", "AZ"],
  118. ["ARKANSAS", "AR"],
  119. ["CALIFORNIA", "CA"],
  120. ["COLORADO", "CO"],
  121. ["CONNECTICUT", "CT"],
  122. ["DELAWARE", "DE"],
  123. ["DISTRICT OF COLUMBIA", "DC"],
  124. ["FLORIDA", "FL"],
  125. ["GEORGIA", "GA"],
  126. ["HAWAII", "HI"],
  127. ["IDAHO", "ID"],
  128. ["ILLINOIS", "IL"],
  129. ["INDIANA", "IN"],
  130. ["IOWA", "IA"],
  131. ["KANSAS", "KS"],
  132. ["KENTUCKY", "KY"],
  133. ["LOUISIANA", "LA"],
  134. ["MAINE", "ME"],
  135. ["MARYLAND", "MD"],
  136. ["MASSACHUSETTS", "MA"],
  137. ["MICHIGAN", "MI"],
  138. ["MINNESOTA", "MN"],
  139. ["MISSISSIPPI", "MS"],
  140. ["MISSOURI", "MO"],
  141. ["MONTANA", "MT"],
  142. ["NEBRASKA", "NE"],
  143. ["NEVADA", "NV"],
  144. ["NEW HAMPSHIRE", "NH"],
  145. ["NEW JERSEY", "NJ"],
  146. ["NEW MEXICO", "NM"],
  147. ["NEW YORK", "NY"],
  148. ["NORTH CAROLINA", "NC"],
  149. ["NORTH DAKOTA", "ND"],
  150. ["OHIO", "OH"],
  151. ["OKLAHOMA", "OK"],
  152. ["OREGON", "OR"],
  153. ["PENNSYLVANIA", "PA"],
  154. ["RHODE ISLAND", "RI"],
  155. ["SOUTH CAROLINA", "SC"],
  156. ["SOUTH DAKOTA", "SD"],
  157. ["TENNESSEE", "TN"],
  158. ["TEXAS", "TX"],
  159. ["UTAH", "UT"],
  160. ["VERMONT", "VT"],
  161. ["VIRGINIA", "VA"],
  162. ["WASHINGTON", "WA"],
  163. ["WEST VIRGINIA", "WV"],
  164. ["WISCONSIN", "WI"],
  165. ["WYOMING", "WY"],
  166. ["PUERTO RICO", "PR"],
  167. ["VIRGIN ISLANDS (U.S.)", "VI"]
  168. ];
  169.  
  170. var i;
  171. if (to === "abbr") {
  172. input = input.toUpperCase();
  173. for (i = 0; i < states.length; i += 1) {
  174. if (states[i][0] === input) {
  175. return (states[i][1]);
  176. }
  177. }
  178. } else if (to === "name") {
  179. input = input.toUpperCase();
  180. for (i = 0; i < states.length; i += 1) {
  181. if (states[i][1] === input) {
  182. return (states[i][0]);
  183. }
  184. }
  185. }
  186. }
  187.  
  188. function formfiller_log(message) {
  189. if (typeof message === "string") {
  190. console.log("FormFiller: " + message);
  191. } else {
  192. console.log("FormFiller: ", message);
  193. }
  194. }
  195.  
  196. function ff_getStreetName(selection) {
  197. var streetName = [],
  198. i;
  199.  
  200. for (i = 0; i < selection.length; i += 1) {
  201. var newStreet = W.model.streets.getObjectById(selection[i].model.attributes.primaryStreetID);
  202. if (typeof newStreet === "undefined" || newStreet.name === null) {
  203. newStreet = { name: "No Name" };
  204. }
  205. if (streetName.indexOf(newStreet.name) === -1) {
  206. streetName.push(newStreet.name);
  207. }
  208. }
  209. return streetName.join(", ");
  210. }
  211.  
  212. function ff_getState(selection) {
  213. var stateName = "",
  214. i;
  215.  
  216. for (i = 0; i < selection.length; i += 1) {
  217. var cID = W.model.streets.getObjectById(selection[i].model.attributes.primaryStreetID).cityID;
  218. var sID = W.model.cities.getObjectById(cID).attributes.stateID;
  219. var newState = W.model.states.getObjectById(sID).name;
  220.  
  221. if (newState === "") {
  222. sID = W.model.cities.getObjectById(cID).attributes.countryID;
  223. newState = W.model.countries.getObjectById(sID).name;
  224. formfiller_log("cID: " + cID);
  225. formfiller_log("sID: " + sID);
  226. formfiller_log("newState: " + newState);
  227. }
  228.  
  229. if (stateName === "") {
  230. stateName = newState;
  231. } else if (stateName !== newState) {
  232. stateName = "";
  233. break;
  234. }
  235. }
  236. return stateName;
  237. }
  238.  
  239. function ff_getCity(selection) {
  240. var cityName = "",
  241. i;
  242. for (i = 0; i < selection.length; i += 1) {
  243. var cID = W.model.streets.getObjectById(selection[i].model.attributes.primaryStreetID).cityID;
  244. var newCity = W.model.cities.getObjectById(cID).attributes.name;
  245. if (cityName === "") {
  246. cityName = newCity;
  247. } else if (cityName !== newCity) {
  248. cityName = "";
  249. break;
  250. }
  251. }
  252. return cityName;
  253. }
  254.  
  255. function ff_getCounty(selection) {
  256. var county = "";
  257. // Disable county for now, currently broken
  258. /*var center = W.map.center.clone().transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  259. //formfiller_log("Getting county for "+center.lat.toString()+","+center.lon.toString());
  260. var xhr = new XMLHttpRequest();
  261. xhr.open("GET", "https://maps.googleapis.com/maps/api/geocode/json?latlng=" + center.lat + "," + center.lon, false);
  262. xhr.onload = function () {
  263. if (xhr.readyState === 4) {
  264. if (xhr.status === 200) {
  265. var response = JSON.parse(xhr.responseText);
  266. if (response.status !== "OK") {
  267. formfiller_log(`Error getting county name (${response.status})`);
  268. return county;
  269. }
  270. var addrComps = response.results[0].address_components;
  271. var comp;
  272. for (comp = 0; comp < addrComps.length; comp += 1) {
  273. if (addrComps[comp].types.indexOf("administrative_area_level_2") !== -1) {
  274. county = addrComps[comp].long_name;
  275. //formfiller_log("ff_getCounty: "+county);
  276. var countyIndex = (county.indexOf(" County") !== -1 ? county.indexOf(" County") : county.indexOf(" Parish"));
  277. if (countyIndex !== -1) {
  278. county = county.slice(0, countyIndex);
  279. }
  280. break;
  281. }
  282. }
  283. }
  284. }
  285. };
  286. xhr.send(null);*/
  287. return county;
  288. }
  289.  
  290. function ff_closureActive(selection) {
  291. var i;
  292. for (i = 0; i < selection.length; i += 1) {
  293. if (selection[i].model.hasClosures()) {
  294. if (W.model.roadClosures.getByAttributes({
  295. segID: selection[i].model.attributes.id
  296. })[0].active) {
  297. return true;
  298. }
  299. }
  300. }
  301. return false;
  302. }
  303.  
  304. function ff_getClosureInfo(seg) {
  305. var closureInfo = {
  306. closedDir: "",
  307. endDate: "",
  308. idFwd: "",
  309. idRev: "",
  310. closedReason: ""
  311. };
  312. var segID = seg.model.attributes.id;
  313. var closureList = W.model.roadClosures.getByAttributes({
  314. segID,
  315. active: true
  316. });
  317. var i;
  318.  
  319. for (i = 0; i < closureList.length; i += 1) {
  320. if (closureList[i].active === true) {
  321. if (closureInfo.endDate === "") {
  322. closureInfo.endDate = closureList[i].endDate;
  323. } else if (closureInfo.endDate > closureList[i].endDate) {
  324. closureInfo.endDate = closureList[i].endDate;
  325. }
  326. if (closureList[i].forward === true) {
  327. closureInfo.idFwd = closureList[i].id;
  328. } else {
  329. closureInfo.idRev = closureList[i].id;
  330. }
  331. if (closureInfo.closedReason === "") {
  332. closureInfo.closedReason = closureList[i].reason;
  333. }
  334. }
  335. }
  336.  
  337. if (closureInfo.idFwd !== "" && closureInfo.idRev !== "") {
  338. closureInfo.closedDir = "Two-Way";
  339. } else {
  340. closureInfo.closedDir = "One-Way";
  341. }
  342.  
  343. return closureInfo;
  344. }
  345.  
  346. function ff_createPermalink(selection) {
  347. //https://www.waze.com/editor/?env=usa&lon=-79.79248&lat=32.86150&layers=12709&zoom=5&mode=0&mapProblemFilter=1&mapUpdateRequestFilter=0&venueFilter=0&segments=504534141
  348. //https://www.waze.com/editor/?env=usa&lon=-79.79248&lat=32.86150&layers=12709&zoom=5&mode=0&mapProblemFilter=1&mapUpdateRequestFilter=0&venueFilter=0&venues=183632201.1836387542.3102948
  349. var permalink = "https://www.waze.com/editor/?",
  350. segIDs = [];
  351. var latLon = W.map.center.clone().transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  352. var lat = latLon.lat,
  353. lon = latLon.lon;
  354. var env = W.location ? W.location.code : W.app.getAppRegionCode();
  355. var type = "segments";
  356. var zoom = W.map.zoom;
  357. var zoomToRoadType = function (e) {
  358. switch (e) {
  359. case 0:
  360. case 1:
  361. return [];
  362. case 2:
  363. return [2, 3, 4, 6, 7, 15];
  364. case 3:
  365. return [2, 3, 4, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22];
  366. case 4:
  367. case 5:
  368. case 6:
  369. case 7:
  370. case 8:
  371. case 9:
  372. case 10:
  373. default:
  374. return [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22];
  375. }
  376. };
  377. var i;
  378.  
  379. //To get lat and long centered on segment
  380. if (selection.length === 1) {
  381. latLon = selection[0].model.getCenter().clone();
  382. latLon.transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  383. lat = latLon.y;
  384. lon = latLon.x;
  385. }
  386.  
  387. for (i = 0; i < selection.length; i += 1) {
  388. var segment = selection[i].model;
  389. if (segment.type === "segment") {
  390. segIDs.push(segment.attributes.id);
  391. if (zoomToRoadType(zoom) === 0 || zoomToRoadType(zoom).indexOf(segment.attributes.roadType) === -1) {
  392. alert("This zoom level (" + zoom.toString() + ") cannot be used for this road type! Please increase your zoom:\n" +
  393. "Streets: 4+\nOther drivable and Non-drivable: 3+\nHighways and PS: 2+");
  394. formfiller_log("Zoom level not correct for segment: " + zoom.toString() + " " + segment.attributes.roadType.toString());
  395. return;
  396. }
  397. }
  398. }
  399. permalink += "env=" + env + "&lon=" + lon + "&lat=" + lat + "&zoom=" + zoom.toString() + "&" + type + "=" + segIDs.join();
  400. return permalink;
  401. }
  402.  
  403. function ff_getLastEditor(selection) {
  404. var eID;
  405. var editorNames = "";
  406. var newEdName = "";
  407. //selection[0].model.attributes.updatedBy;
  408. selection.forEach(function (selected) {
  409. eID = selected.model.attributes.updatedBy;
  410. if (typeof eID !== "undefined") {
  411. formfiller_log("Unable to get updatedBy on " + selected.model.attributes.id);
  412. eID = selected.model.attributes.createdBy;
  413. }
  414. newEdName = W.model.users.getObjectById(eID).userName;
  415. if (editorNames.indexOf(newEdName) === -1) {
  416. editorNames += ", " + newEdName;
  417. }
  418.  
  419. });
  420. editorNames = editorNames.substr(2);
  421. return editorNames;
  422. }
  423.  
  424. function ff_createFormLink(formSel) {
  425. var selection = W.selectionManager.getSelectedFeatures();
  426. var formValues = {};
  427. var formFields = formSel.fields;
  428. var formLink = formSel.url + "?entry.";
  429. var formArgs = [];
  430. if (selection.length === 0 || selection[0].model.type !== "segment") {
  431. formfiller_log("No segments selected.");
  432. return;
  433. }
  434.  
  435. /*Fields expected:
  436. username
  437. permalink
  438. closedDir
  439. closureStatus
  440. closedReason
  441. endDate
  442. streetname
  443. fromStreet
  444. toStreet
  445. stateabbr
  446. county
  447. city
  448. source
  449. notes
  450. */
  451.  
  452. Object.keys(formFields).forEach((key, index) => {
  453. switch (key) {
  454. case "username":
  455. formValues[key] = W.loginManager.user.userName;
  456. break;
  457. case "permalink":
  458. formValues[key] = ff_createPermalink(selection);
  459. if (typeof formValues.permalink === "undefined") {
  460. formfiller_log("No permalink generated");
  461. return;
  462. }
  463. break;
  464. case "streetname":
  465. formValues[key] = ff_getStreetName(selection);
  466. break;
  467. case "editorName":
  468. formValues[key] = ff_getLastEditor(selection);
  469. break;
  470. case "stateabbr":
  471. formValues[key] = abbrState(ff_getState(selection), "abbr");
  472. break;
  473. case "state":
  474. formValues[key] = ff_getState(selection);
  475. break;
  476. case "county":
  477. formValues.county = ff_getCounty(selection);
  478. break;
  479. case "city":
  480. formValues[key] = ff_getCity(selection);
  481. break;
  482. case "notes":
  483. formValues[key] = "Form filled by " + WMEFFName + " v" + WMEFFVersion;
  484. break;
  485. case "closureStatus":
  486. if (selection[0].model.type === "segment") {
  487. if (ff_closureActive(selection)) {
  488. formValues.closureStatus = "CLOSED";
  489. var closureInfo = ff_getClosureInfo(selection[0]);
  490. formValues.closedDir = closureInfo.closedDir;
  491. formValues.closedReason = closureInfo.closedReason;
  492. formValues.endDate = closureInfo.endDate;
  493. } else {
  494. formValues.closureStatus = "REPORTED";
  495. formValues.closedDir = "Two-Way";
  496. formValues.closedReason = document.getElementById("ff-closure-reason").value;
  497. formValues.endDate = document.getElementById("ff-closure-endDate").value + "+" + document.getElementById("ff-closure-endTime").value;
  498. }
  499. }
  500. break;
  501. default:
  502. formfiller_log("Nothing defined for " + key);
  503. break;
  504. }
  505.  
  506. //Add entry to form URL, if there's something to add
  507. if (typeof formValues[key] !== "undefined" && formValues[key] !== "") {
  508. formArgs[index] = formFields[key] + "=" + encodeURIComponent(formValues[key]);
  509. }
  510. });
  511. formLink += formArgs.join("&entry.");
  512.  
  513. formfiller_log(formLink);
  514. return formLink;
  515. }
  516.  
  517. function ff_addFormBtn() {
  518. var selection = W.selectionManager.getSelectedFeatures();
  519. var ffDiv = document.createElement("div"),
  520. ffMnu = document.createElement("select"),
  521. ffBtn = document.createElement("button");
  522. var formWindowName = "WME Form Filler result",
  523. formWindowSpecs = "resizable=1,menubar=0,scrollbars=1,status=0,toolbar=0";
  524. var editPanel,
  525. selElem,
  526. formLink;
  527. ffDiv.id = "formfillerDiv";
  528. editPanel = document.getElementById("edit-panel");
  529. selElem = editPanel.getElementsByClassName("selection");
  530. if (selection.length === 0 || selection[0].model.type !== "segment") {
  531. //formfiller_log("No segments selected.");
  532. return;
  533. }
  534. if (document.getElementById("formfillerDiv")) {
  535. //formfiller_log("Div already created");
  536. return;
  537. }
  538.  
  539. forms = [{
  540. //https://docs.google.com/forms/d/e/1FAIpQLSduBiLMhbg6nRpsEVCTcVbV4eWmHDXdIKGtuaOvzy6NZLbSgw/viewform?entry.1553765347=username&entry.1264424583=CLOSED&entry.1811077109=permalink&entry.792657790=Two-Way&entry.345142186=reason&entry.1102521735=2016-09-20+03:00&entry.2015424420=street+name&entry.1547375393=from+street&entry.1335391716=to+street&entry.1867193205=SC&entry.1714138473=county&entry.1803937317=source&entry.1648634142=notes
  541. name: "USA VEOC closures",
  542. url: "https://docs.google.com/forms/d/e/1FAIpQLSduBiLMhbg6nRpsEVCTcVbV4eWmHDXdIKGtuaOvzy6NZLbSgw/viewform",
  543. fields: {
  544. username: "1553765347",
  545. closureStatus: "1264424583",
  546. permalink: "1811077109",
  547. closedDir: "792657790",
  548. closedReason: "345142186",
  549. endDate: "1102521735",
  550. streetname: "2015424420",
  551. fromStreet: "1547375393",
  552. toStreet: "1335391716",
  553. stateabbr: "1867193205",
  554. county: "1714138473",
  555. source: "1803937317",
  556. notes: "1648634142"
  557. }
  558. }, {
  559. name: "US Jane TTS Pronunciation",
  560. url: "https://docs.google.com/forms/d/e/1FAIpQLSeuCmC0zy7GEQDJQP5R8dndxYhXCkqzadrPgP89BvatVl1bdg/viewform",
  561. fields: {
  562. username: "324217272",
  563. state: "1065619417",
  564. issue: "1086951221",
  565. streetname: "1163516948",
  566. incorrectp: "1191620241",
  567. correctp: "1649051316",
  568. permalink: "2028167849",
  569. instructions: "2120232339",
  570. comments: "1917392591"
  571. }
  572. }, {
  573. //https://docs.google.com/forms/d/e/1FAIpQLSff7nsBw8qxCojBdxrjTPl6tercqyyzGy92Vif_SBdHkYDchw/viewform?entry.1204781462=Reporter&entry.828228572=Reported&entry.1647952662=Street+name+&entry.1501712688=From+street+&entry.2094306654=To+street+&entry.1414240321=Two-Way&entry.900957975=10/27/2016+00:00&entry.1051351191=Adams&entry.1093044522=City+&entry.1540676081=IDOT&entry.430378754=Reason+&entry.1754051160=Permalink+&entry.172235277=Source+&entry.1722909714=Notes+
  574. name: "IL closures",
  575. url: "https://docs.google.com/forms/d/e/1FAIpQLSff7nsBw8qxCojBdxrjTPl6tercqyyzGy92Vif_SBdHkYDchw/viewform",
  576. fields: {
  577. username: "1204781462",
  578. closureStatus: "828228572",
  579. permalink: "1754051160",
  580. closedDir: "1414240321",
  581. closedReason: "430378754",
  582. endDate: "900957975",
  583. streetname: "1647952662",
  584. fromStreet: "1501712688",
  585. toStreet: "2094306654",
  586. county: "1051351191",
  587. city: "1093044522",
  588. source: "172235277",
  589. notes: "1722909714"
  590. }
  591. }, {
  592. //https://docs.google.com/forms/d/e/1FAIpQLSd8wouEY6DhA4ifMb_Ub0rEE7X4TCM7WIEqFiT4Lq1hySxEWw/viewform?entry.1553765347=username&entry.1264424583=CLOSED&entry.1811077109=permalink&entry.792657790=Two-Way&entry.345142186=reason&entry.1102521735=2016-09-20+03:00&entry.2015424420=street+name&entry.1547375393=from+street&entry.1335391716=to+street&entry.1867193205=OK&entry.1714138473=county&entry.1803937317=source&entry.1648634142=notes
  593. name: "OK closures",
  594. url: "https://docs.google.com/forms/d/e/1FAIpQLSd8wouEY6DhA4ifMb_Ub0rEE7X4TCM7WIEqFiT4Lq1hySxEWw/viewform",
  595. fields: {
  596. username: "1553765347",
  597. closureStatus: "1264424583",
  598. permalink: "1811077109",
  599. closedDir: "792657790",
  600. closedReason: "345142186",
  601. endDate: "1102521735",
  602. streetname: "2015424420",
  603. fromStreet: "1547375393",
  604. toStreet: "1335391716",
  605. stateabbr: "1867193205",
  606. county: "1714138473",
  607. source: "1803937317",
  608. notes: "1648634142"
  609. }
  610. }, {
  611. //https://docs.google.com/forms/d/e/1FAIpQLScwEyNVqiHHdFjc4hr82zlFXW2bAsff9pqIzFUqT8Evh6YROg/viewform?usp=pp_url&entry.1553765347=kwrigh01&entry.1264424583=CLOSED&entry.1811077109=https://www.waze.com/editor/?env%3Dusa%26lon%3D-79.99979%26lat%3D37.89567%26zoom%3D5%26segments%3D82457306,82457308,82457338&entry.792657790=Two-Way&entry.345142186=Bridge+Work&entry.1102521735=2018-07-01+23:59&entry.2015424420=Morris+Hollow+Rd&entry.1547375393=1st&entry.1335391716=2nd&entry.1867193205=VA&entry.1714138473=Alleghany&entry.1803937317=VDOT&entry.1648634142=Test+Closure
  612. name: "VA Closures",
  613. url: "https://docs.google.com/forms/d/e/1FAIpQLScwEyNVqiHHdFjc4hr82zlFXW2bAsff9pqIzFUqT8Evh6YROg/viewform",
  614. fields: {
  615. username: "1553765347",
  616. closureStatus: "1264424583",
  617. permalink: "1811077109",
  618. closedDir: "792657790",
  619. closedReason: "345142186",
  620. endDate: "1102521735",
  621. streetname: "2015424420",
  622. fromStreet: "1547375393",
  623. toStreet: "1335391716",
  624. stateabbr: "1867193205",
  625. county: "1714138473",
  626. source: "1803937317",
  627. notes: "1648634142"
  628. }
  629. }, {
  630. //https://docs.google.com/forms/d/e/1FAIpQLSeiKY0KsO0xN69Asw77MARQFmxOy6zQXF-k2OQdWOfwtiCp7Q/viewform?entry.1204781462=ojlaw&entry.828228572=CLOSED&entry.1647952662=Test1&entry.1501712688=Test2&entry.2094306654=Test3&entry.1414240321=One-Way&entry.900957975=00/00/0000+00:00&entry.1051351191=Adams&entry.1093044522=Test4&entry.1540676081=City&entry.430378754=Test5&entry.1754051160=Test6&entry.172235277=Test7&entry.1722909714=Test8
  631. name: "WI closures",
  632. url: "https://docs.google.com/forms/d/e/1FAIpQLSeiKY0KsO0xN69Asw77MARQFmxOy6zQXF-k2OQdWOfwtiCp7Q/viewform",
  633. fields: {
  634. username: "1204781462",
  635. closureStatus: "828228572",
  636. permalink: "1754051160",
  637. closedDir: "1414240321",
  638. closedReason: "430378754",
  639. endDate: "900957975",
  640. streetname: "1647952662",
  641. fromStreet: "1501712688",
  642. toStreet: "2094306654",
  643. county: "1051351191",
  644. city: "1093044522",
  645. source: "172235277",
  646. notes: "1722909714"
  647. }
  648. }, {
  649. //https://docs.google.com/forms/d/e/1FAIpQLSf3YAdjscVvJXiU4KWq0e8J8XyCyYBDyharRoXW3bf6R4wH1w/viewform?usp=pp_url&entry.1553765347=kwrigh01&entry.1264424583=CLOSED&entry.1811077109=https://www.waze.com/editor/?env%3Dusa%26lon%3D-80.26934%26lat%3D39.39069%26zoom%3D5%26segments%3D504613052,55526394,55530967&entry.792657790=Two-Way&entry.345142186=Road+Slip&entry.1102521735=2018-06-01+16:00&entry.2015424420=Janes+Hill+Rd&entry.1547375393&entry.1335391716&entry.1714138473=Harrison+&entry.1803937317=WVDOT+&entry.1648634142=Closed+Indefinitely
  650. name: "WV Closures",
  651. url: "https://docs.google.com/forms/d/e/1FAIpQLSf3YAdjscVvJXiU4KWq0e8J8XyCyYBDyharRoXW3bf6R4wH1w/viewform",
  652. fields: {
  653. username: "1553765347",
  654. closureStatus: "1264424583",
  655. permalink: "1811077109",
  656. closedDir: "792657790",
  657. closedReason: "345142186",
  658. endDate: "1102521735",
  659. streetname: "2015424420",
  660. fromStreet: "1547375393",
  661. toStreet: "1335391716",
  662. county: "1714138473",
  663. source: "1803937317",
  664. notes: "1648634142"
  665. }
  666. }
  667. ];
  668.  
  669. forms.forEach(function (key, i) {
  670. ffMnu.options.add(new Option(forms[i].name, i));
  671. });
  672. ffBtn.innerHTML = "Go to Form";
  673. ffBtn.onclick = function () {
  674. //alert(ffMnu.options[ffMnu.selectedIndex].value+": "+forms[ffMnu.options[ffMnu.selectedIndex].value].name);
  675. ff_saveSettings();
  676. formLink = ff_createFormLink(forms[ffMnu.options[ffMnu.selectedIndex].value]);
  677. if (typeof formLink === "undefined") {
  678. return;
  679. }
  680.  
  681. if ($("#ff-open-in-tab").prop("checked")) {
  682. window.open(formLink, "_blank");
  683. } else {
  684. window.open(formLink, formWindowName, formWindowSpecs);
  685. }
  686. };
  687. ffDiv.appendChild(ffMnu);
  688. ffDiv.appendChild(ffBtn);
  689. selElem[0].appendChild(ffDiv);
  690.  
  691. return;
  692. }
  693.  
  694. function ff_loadSettings() {
  695. var todayDate = new Date(),
  696. futureDate = new Date(),
  697. daysInFuture = 3;
  698. var today = todayDate.getFullYear() + "-" + (todayDate.getMonth() + 1 < 10 ? "0" + (todayDate.getMonth() + 1) : todayDate.getMonth() + 1) + "-" + todayDate.getDate();
  699. futureDate.setDate(futureDate.getDate() + daysInFuture);
  700.  
  701. var ffOpenInTab = localStorage.getItem("ff-open-in-tab");
  702. if (ffOpenInTab === "1") {
  703. $("#ff-open-in-tab").trigger("click");
  704. }
  705. var ffClosureReason = localStorage.getItem("ff-closure-reason");
  706. if (ffClosureReason !== null) {
  707. $("#ff-closure-reason").val(ffClosureReason);
  708. }
  709. var ffClosureEndDate = localStorage.getItem("ff-closure-endDate");
  710. if (ffClosureEndDate !== null && ffClosureEndDate !== "" && ffClosureEndDate >= today) {
  711. $("#ff-closure-endDate").val(ffClosureEndDate);
  712. } else {
  713. var closureDate = futureDate.getFullYear() + "-" + (futureDate.getMonth() + 1 < 10 ? "0" + (futureDate.getMonth() + 1) : futureDate.getMonth() + 1) + "-" + (futureDate.getDate() < 10 ? "0" + futureDate.getDate() : futureDate.getDate());
  714. $("#ff-closure-endDate").val(closureDate);
  715. }
  716. var ffClosureEndTime = localStorage.getItem("ff-closure-endTime");
  717. if (ffClosureEndTime !== null && ffClosureEndTime !== "") {
  718. $("#ff-closure-endTime").val(ffClosureEndTime);
  719. }
  720. //formfiller_log("Settings loaded");
  721. return;
  722. }
  723.  
  724. function ff_saveSettings() {
  725. if ($("#ff-open-in-tab").prop("checked")) {
  726. localStorage.setItem("ff-open-in-tab", "1");
  727. } else {
  728. localStorage.setItem("ff-open-in-tab", "0");
  729. }
  730. localStorage.setItem("ff-closure-reason", $("#ff-closure-reason").val());
  731. localStorage.setItem("ff-closure-endDate", $("#ff-closure-endDate").val());
  732. localStorage.setItem("ff-closure-endTime", $("#ff-closure-endTime").val());
  733. //formfiller_log("Settings saved");
  734. return;
  735. }
  736.  
  737. function ff_addUserTab() {
  738. var userInfo = document.getElementById("user-info"),
  739. userTabs = document.getElementById("user-tabs"),
  740. navTabs = userTabs.getElementsByClassName("nav-tabs"),
  741. tabContent = userInfo.getElementsByClassName("tab-content");
  742. var ffTab = document.createElement("li"),
  743. ffPanel = document.createElement("div"),
  744. ffReason = document.createElement("div"),
  745. ffEndDate = document.createElement("div"),
  746. ffNewTabBox = document.createElement("input"),
  747. ffNewTabLabel = document.createElement("label"),
  748. ffTabInfo = document.createElement("div");
  749.  
  750. ffTab.innerHTML = '<a title="Form Filler" href="#sidepanel-formfill" data-toggle="tab"><img class="fa" src="' + WMEFFIcon + '" width="15px" /></a>';
  751. ffPanel.id = "sidepanel-formfill";
  752. ffPanel.className = "tab-pane";
  753.  
  754. ffTabInfo.innerHTML = '<b>' + WMEFFName + '</b> v' + WMEFFVersion;
  755.  
  756. ffNewTabBox.id = "ff-open-in-tab";
  757. ffNewTabBox.type = "checkbox";
  758. ffNewTabBox.name = "ff_open_tab";
  759.  
  760. ffNewTabLabel.innerHTML = "Open form in new tab";
  761. ffNewTabLabel.for = "ff_open_tab";
  762.  
  763. ffReason.className = "form-group";
  764. ffReason.innerHTML = '<label class="control-label" for="ff_closure_reason">Closures reason:</label><div class="controls"><input id="ff-closure-reason" class="form-control" type="text" name="ff_closure_reason"></div>';
  765.  
  766. ffEndDate.className = "form-group";
  767. ffEndDate.innerHTML = '<label class="control-label" for="ff_closure_endDate">Closures end:</label><div class="controls"><div class="date date-input-group input-group pull-left" style="width: 62%"><input id="ff-closure-endDate" class="form-control end-date" type="text" name="ff_closure_endDate"><span class="input-group-addon"><i class="fa fa-calendar"></i></span></div><div class="bootstrap-timepicker input-group style="width: 38%""><input id="ff-closure-endTime" class="end-time form-control" type="text" name="ff_closure_endTime"><span class="input-group-addon"><i class="fa fa-clock-o"></i></span></div></div>';
  768.  
  769. ffPanel.appendChild(ffTabInfo);
  770. ffPanel.appendChild(ffNewTabBox);
  771. ffPanel.appendChild(ffNewTabLabel);
  772. ffPanel.appendChild(ffReason);
  773. ffPanel.appendChild(ffEndDate);
  774. navTabs[0].appendChild(ffTab);
  775. tabContent[0].appendChild(ffPanel);
  776.  
  777. if (typeof $.fn.datepicker !== "undefined") {
  778. $("#ff-closure-endDate").datepicker({
  779. format: "yyyy-mm-dd",
  780. todayHighlight: true,
  781. autoclose: true
  782. });
  783. } else {
  784. if (typeof $.fn.daterangepicker !== "undefined") {
  785. $("#ff-closure-endDate").daterangepicker({
  786. singleDatePicker: true,
  787. locale: {
  788. format: "YYYY-MM-DD"
  789. }
  790. });
  791. }
  792. }
  793.  
  794. if (typeof $.fn.timepicker !== "undefined") {
  795. $("#ff-closure-endTime").timepicker({
  796. template: false,
  797. defaultTime: "00:00",
  798. showMeridian: false
  799. });
  800. }
  801.  
  802. ff_loadSettings();
  803. $("#ff-closure-reason").change(function () {
  804. ff_saveSettings();
  805. });
  806. $("#ff-closure-endDate").change(function () {
  807. ff_saveSettings();
  808. });
  809. $("#ff-closure-endTime").change(function () {
  810. ff_saveSettings();
  811. });
  812. $("#ff-open-in-tab").click(function () {
  813. ff_saveSettings();
  814. });
  815. }
  816.  
  817. setTimeout(formfiller_bootstrap, 2000);
  818. }());