WME Form Filler

Use info from WME to automatically fill out related forms

当前为 2018-04-24 提交的版本,查看 最新版本

  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.0
  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 
  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 = "";
  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. if (!W.selectionManager.getSelectedFeatures) {
  101. W.selectionManager.getSelectedFeatures = W.selectionManager.getSelectedItems;
  102. }
  103. formfiller_log("Init done");
  104. return;
  105. }
  106.  
  107. //Shamelessly copied from https://gist.github.com/CalebGrove/c285a9510948b633aa47
  108. function abbrState(input, to) {
  109. var states = [
  110. ["ALABAMA", "AL"],
  111. ["ALASKA", "AK"],
  112. ["ARIZONA", "AZ"],
  113. ["ARKANSAS", "AR"],
  114. ["CALIFORNIA", "CA"],
  115. ["COLORADO", "CO"],
  116. ["CONNECTICUT", "CT"],
  117. ["DELAWARE", "DE"],
  118. ["DISTRICT OF COLUMBIA", "DC"],
  119. ["FLORIDA", "FL"],
  120. ["GEORGIA", "GA"],
  121. ["HAWAII", "HI"],
  122. ["IDAHO", "ID"],
  123. ["ILLINOIS", "IL"],
  124. ["INDIANA", "IN"],
  125. ["IOWA", "IA"],
  126. ["KANSAS", "KS"],
  127. ["KENTUCKY", "KY"],
  128. ["LOUISIANA", "LA"],
  129. ["MAINE", "ME"],
  130. ["MARYLAND", "MD"],
  131. ["MASSACHUSETTS", "MA"],
  132. ["MICHIGAN", "MI"],
  133. ["MINNESOTA", "MN"],
  134. ["MISSISSIPPI", "MS"],
  135. ["MISSOURI", "MO"],
  136. ["MONTANA", "MT"],
  137. ["NEBRASKA", "NE"],
  138. ["NEVADA", "NV"],
  139. ["NEW HAMPSHIRE", "NH"],
  140. ["NEW JERSEY", "NJ"],
  141. ["NEW MEXICO", "NM"],
  142. ["NEW YORK", "NY"],
  143. ["NORTH CAROLINA", "NC"],
  144. ["NORTH DAKOTA", "ND"],
  145. ["OHIO", "OH"],
  146. ["OKLAHOMA", "OK"],
  147. ["OREGON", "OR"],
  148. ["PENNSYLVANIA", "PA"],
  149. ["RHODE ISLAND", "RI"],
  150. ["SOUTH CAROLINA", "SC"],
  151. ["SOUTH DAKOTA", "SD"],
  152. ["TENNESSEE", "TN"],
  153. ["TEXAS", "TX"],
  154. ["UTAH", "UT"],
  155. ["VERMONT", "VT"],
  156. ["VIRGINIA", "VA"],
  157. ["WASHINGTON", "WA"],
  158. ["WEST VIRGINIA", "WV"],
  159. ["WISCONSIN", "WI"],
  160. ["WYOMING", "WY"],
  161. ["PUERTO RICO", "PR"],
  162. ["VIRGIN ISLANDS (U.S.)", "VI"]
  163. ];
  164.  
  165. var i;
  166. if (to === "abbr") {
  167. input = input.toUpperCase();
  168. for (i = 0; i < states.length; i += 1) {
  169. if (states[i][0] === input) {
  170. return (states[i][1]);
  171. }
  172. }
  173. } else if (to === "name") {
  174. input = input.toUpperCase();
  175. for (i = 0; i < states.length; i += 1) {
  176. if (states[i][1] === input) {
  177. return (states[i][0]);
  178. }
  179. }
  180. }
  181. }
  182.  
  183. function formfiller_log(message) {
  184. if (typeof message === "string") {
  185. console.log("FormFiller: " + message);
  186. } else {
  187. console.log("FormFiller: ", message);
  188. }
  189. }
  190.  
  191. function ff_getStreetName(selection) {
  192. var streetName = "",
  193. i;
  194.  
  195. for (i = 0; i < selection.length; i += 1) {
  196. var newStreet = W.model.streets.get(selection[i].model.attributes.primaryStreetID);
  197. if (typeof newStreet === "undefined" || newStreet.name === null) {
  198. newStreet = "No Name";
  199. }
  200. if (streetName === "") {
  201. streetName = newStreet.name;
  202. } else if (streetName !== newStreet.name) {
  203. streetName += ", " + newStreet.name;
  204. }
  205. }
  206. return streetName;
  207. }
  208.  
  209. function ff_getState(selection) {
  210. var stateName = "",
  211. i;
  212.  
  213. for (i = 0; i < selection.length; i += 1) {
  214. var cID = W.model.streets.get(selection[i].model.attributes.primaryStreetID).cityID;
  215. var sID = W.model.cities.get(cID).attributes.stateID;
  216. var newState = W.model.states.get(sID).name;
  217.  
  218. if (newState === "") {
  219. sID = W.model.cities.get(cID).attributes.countryID;
  220. newState = W.model.countries.get(sID).name;
  221. formfiller_log("cID: " + cID);
  222. formfiller_log("sID: " + sID);
  223. formfiller_log("newState: " + newState);
  224. }
  225.  
  226. if (stateName === "") {
  227. stateName = newState;
  228. } else if (stateName !== newState) {
  229. stateName = "";
  230. break;
  231. }
  232. }
  233. return stateName;
  234. }
  235.  
  236. function ff_getCity(selection) {
  237. var cityName = "",
  238. i;
  239. for (i = 0; i < selection.length; i += 1) {
  240. var cID = W.model.streets.get(selection[i].model.attributes.primaryStreetID).cityID;
  241. var newCity = W.model.cities.get(cID).attributes.name;
  242. if (cityName === "") {
  243. cityName = newCity;
  244. } else if (cityName !== newCity) {
  245. cityName = "";
  246. break;
  247. }
  248. }
  249. return cityName;
  250. }
  251.  
  252. function ff_getCounty(selection) {
  253. var county = "";
  254. var center = W.map.center.clone().transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  255. //formfiller_log("Getting county for "+center.lat.toString()+","+center.lon.toString());
  256. var xhr = new XMLHttpRequest();
  257. xhr.open("GET", "https://maps.googleapis.com/maps/api/geocode/json?latlng=" + center.lat + "," + center.lon, false);
  258. xhr.onload = function () {
  259. if (xhr.readyState === 4) {
  260. if (xhr.status === 200) {
  261. var response = JSON.parse(xhr.responseText);
  262. var addrComps = response.results[0].address_components;
  263. var comp;
  264. for (comp = 0; comp < addrComps.length; comp += 1) {
  265. if (addrComps[comp].types.indexOf("administrative_area_level_2") !== -1) {
  266. county = addrComps[comp].long_name;
  267. //formfiller_log("ff_getCounty: "+county);
  268. var countyIndex = (county.indexOf(" County") !== -1 ? county.indexOf(" County") : county.indexOf(" Parish"));
  269. if (countyIndex !== -1) {
  270. county = county.slice(0, countyIndex);
  271. }
  272. break;
  273. }
  274. }
  275. }
  276. }
  277. };
  278. xhr.send(null);
  279. return county;
  280.  
  281. //Async call. Figure this out!
  282. /*return $.getJSON("https://maps.googleapis.com/maps/api/geocode/json?latlng="+center.lat+","+center.lon, function(data) {
  283. if (data.status === "OK")
  284. {
  285. var addrComps = data.results[0].address_components;
  286. for (comp = 0; comp < addrComps.length; comp+=1)
  287. {
  288. if (addrComps[comp].types.indexOf("administrative_area_level_2") !== -1)
  289. {
  290. county = addrComps[comp].long_name;
  291. county = county.slice(0,county.indexOf(" County"));
  292. formfiller_log("JSON func "+county);
  293. break;
  294. }
  295. }
  296. }
  297. if (county === "")
  298. county = "Not found";
  299. formfiller_log("Got county");
  300. formfiller_log(county);
  301. return county;
  302. });*/
  303. }
  304.  
  305. function ff_closureActive(selection) {
  306. var i;
  307. for (i = 0; i < selection.length; i += 1) {
  308. if (selection[i].model.hasClosures()) {
  309. if (W.model.roadClosures.getByAttributes({
  310. segID: selection[i].model.attributes.id
  311. })[0].active) {
  312. return true;
  313. }
  314. }
  315. }
  316. return false;
  317. }
  318.  
  319. function ff_getClosureInfo(seg) {
  320. var closureInfo = {
  321. closedDir: "",
  322. endDate: "",
  323. idFwd: "",
  324. idRev: "",
  325. closedReason: ""
  326. };
  327. var segID = seg.model.attributes.id;
  328. var closureList = W.model.roadClosures.getByAttributes({
  329. segID: segID,
  330. active: true
  331. });
  332. var i;
  333.  
  334. for (i = 0; i < closureList.length; i += 1) {
  335. if (closureList[i].active === true) {
  336. if (closureInfo.endDate === "") {
  337. closureInfo.endDate = closureList[i].endDate;
  338. } else if (closureInfo.endDate > closureList[i].endDate) {
  339. closureInfo.endDate = closureList[i].endDate;
  340. }
  341. if (closureList[i].forward === true) {
  342. closureInfo.idFwd = closureList[i].id;
  343. } else {
  344. closureInfo.idRev = closureList[i].id;
  345. }
  346. if (closureInfo.closedReason === "") {
  347. closureInfo.closedReason = closureList[i].closedReason;
  348. }
  349. }
  350. }
  351.  
  352. if (closureInfo.idFwd !== "" && closureInfo.idRev !== "") {
  353. closureInfo.closedDir = "Two-Way";
  354. } else {
  355. closureInfo.closedDir = "One-Way";
  356. }
  357.  
  358. return closureInfo;
  359. }
  360.  
  361. function ff_createPermalink(selection) {
  362. //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
  363. //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
  364. var permalink = "https://www.waze.com/editor/?",
  365. segIDs = [];
  366. var latLon = W.map.center.clone().transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  367. var lat = latLon.lat,
  368. lon = latLon.lon;
  369. var env = W.location.code;
  370. var type = "segments";
  371. var zoom = W.map.zoom;
  372. var zoomToRoadType = W.Config.segments.zoomToRoadType;
  373. var i;
  374.  
  375. //To get lat and long centered on segment
  376. if (selection.length === 1) {
  377. latLon = selection[0].model.getCenter().clone();
  378. latLon.transform(W.map.projection.projCode, W.map.displayProjection.projCode);
  379. lat = latLon.y;
  380. lon = latLon.x;
  381. }
  382.  
  383. for (i = 0; i < selection.length; i += 1) {
  384. var segment = selection[i].model;
  385. if (segment.type === "segment") {
  386. segIDs.push(segment.attributes.id);
  387. if (zoomToRoadType[zoom] !== -1 && zoomToRoadType[zoom].indexOf(segment.attributes.roadType) === -1) {
  388. alert("This zoom level (" + zoom.toString() + ") cannot be used for this road type! Please increase your zoom:\n" +
  389. "Streets: 4+\nOther drivable and Non-drivable: 3+\nHighways and PS: 2+");
  390. formfiller_log("Zoom level not correct for segment: " + zoom.toString() + " " + segment.attributes.roadType.toString());
  391. return;
  392. }
  393. }
  394. }
  395. permalink += "env=" + env + "&lon=" + lon + "&lat=" + lat + "&zoom=" + zoom.toString() + "&" + type + "=" + segIDs.join();
  396. return permalink;
  397. }
  398.  
  399. function ff_getLastEditor(selection) {
  400. var eID;
  401. var editorNames = "";
  402. var newEdName = "";
  403. //selection[0].model.attributes.updatedBy;
  404. selection.forEach(function (selected) {
  405. eID = selected.model.attributes.updatedBy;
  406. if (typeof eID !== "undefined") {
  407. formfiller_log("Unable to get updatedBy on " + selected.model.attributes.id);
  408. eID = selected.model.attributes.createdBy;
  409. }
  410. newEdName = W.model.users.get(eID).userName;
  411. if (editorNames.indexOf(newEdName) === -1) {
  412. editorNames += ", " + newEdName;
  413. }
  414.  
  415. });
  416. editorNames = editorNames.substr(2);
  417. return editorNames;
  418. }
  419.  
  420. function ff_createFormLink(formSel) {
  421. var selection = W.selectionManager.getSelectedFeatures();
  422. var formValues = {};
  423. var formFields = formSel.fields;
  424. var formLink = formSel.url + "?entry.";
  425. var formArgs = [];
  426. if (selection.length === 0 || selection[0].model.type !== "segment") {
  427. formfiller_log("No segments selected.");
  428. return;
  429. }
  430.  
  431. /*Fields expected:
  432. username
  433. permalink
  434. closedDir
  435. closureStatus
  436. closedReason
  437. endDate
  438. streetname
  439. fromStreet
  440. toStreet
  441. stateabbr
  442. county
  443. city
  444. source
  445. notes
  446. */
  447.  
  448. Object.keys(formFields).forEach(function (key, index) {
  449. switch (key) {
  450. case "username":
  451. formValues[key] = W.loginManager.user.userName;
  452. break;
  453. case "permalink":
  454. formValues[key] = ff_createPermalink(selection);
  455. if (typeof formValues.permalink === "undefined") {
  456. formfiller_log("No permalink generated");
  457. return;
  458. }
  459. break;
  460. case "streetname":
  461. formValues[key] = ff_getStreetName(selection);
  462. break;
  463. case "editorName":
  464. formValues[key] = ff_getLastEditor(selection);
  465. break;
  466. case "stateabbr":
  467. formValues[key] = abbrState(ff_getState(selection), "abbr");
  468. break;
  469. case "state":
  470. formValues[key] = ff_getState(selection);
  471. break;
  472. case "county":
  473. formValues.county = ff_getCounty(selection);
  474. break;
  475. case "city":
  476. formValues[key] = ff_getCity(selection);
  477. break;
  478. case "notes":
  479. formValues[key] = "Form filled by " + WMEFFName + " v" + WMEFFVersion;
  480. break;
  481. case "closureStatus":
  482. if (selection[0].model.type === "segment") {
  483. if (ff_closureActive(selection)) {
  484. formValues.closureStatus = "CLOSED";
  485. var closureInfo = ff_getClosureInfo(selection[0]);
  486. formValues.closedDir = closureInfo.closedDir;
  487. formValues.closedReason = closureInfo.closedReason;
  488. formValues.endDate = closureInfo.endDate;
  489. } else {
  490. formValues.closureStatus = "REPORTED";
  491. formValues.closedDir = "Two-Way";
  492. formValues.closedReason = document.getElementById("ff-closure-reason").value;
  493. formValues.endDate = document.getElementById("ff-closure-endDate").value + "+" + document.getElementById("ff-closure-endTime").value;
  494. }
  495. }
  496. break;
  497. default:
  498. formfiller_log("Nothing defined for " + key);
  499. break;
  500. }
  501.  
  502. //Add entry to form URL, if there's something to add
  503. if (typeof formValues[key] !== "undefined" && formValues[key] !== "") {
  504. formArgs[index] = formFields[key] + "=" + encodeURIComponent(formValues[key]);
  505. }
  506. });
  507. formLink += formArgs.join("&entry.");
  508.  
  509. formfiller_log(formLink);
  510. return formLink;
  511. }
  512.  
  513. function ff_addFormBtn() {
  514. var selection = W.selectionManager.getSelectedFeatures();
  515. var ffDiv = document.createElement("div"),
  516. ffMnu = document.createElement("select"),
  517. ffBtn = document.createElement("button");
  518. var formWindowName = "WME Form Filler result",
  519. formWindowSpecs = "resizable=1,menubar=0,scrollbars=1,status=0,toolbar=0";
  520. var editPanel,
  521. selElem,
  522. formLink;
  523. ffDiv.id = "formfillerDiv";
  524. editPanel = document.getElementById("edit-panel");
  525. selElem = editPanel.getElementsByClassName("selection");
  526. if (selection.length === 0 || selection[0].model.type !== "segment") {
  527. //formfiller_log("No segments selected.");
  528. return;
  529. }
  530. if (document.getElementById("formfillerDiv")) {
  531. //formfiller_log("Div already created");
  532. return;
  533. }
  534.  
  535. forms = [{
  536. //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
  537. name: "USA VEOC closures",
  538. url: "https://docs.google.com/forms/d/e/1FAIpQLSduBiLMhbg6nRpsEVCTcVbV4eWmHDXdIKGtuaOvzy6NZLbSgw/viewform",
  539. fields: {
  540. username: "1553765347",
  541. closureStatus: "1264424583",
  542. permalink: "1811077109",
  543. closedDir: "792657790",
  544. closedReason: "345142186",
  545. endDate: "1102521735",
  546. streetname: "2015424420",
  547. fromStreet: "1547375393",
  548. toStreet: "1335391716",
  549. stateabbr: "1867193205",
  550. county: "1714138473",
  551. source: "1803937317",
  552. notes: "1648634142"
  553. }
  554. }, {
  555. name: "US Jane TTS Pronunciation",
  556. url: "https://docs.google.com/forms/d/e/1FAIpQLSeuCmC0zy7GEQDJQP5R8dndxYhXCkqzadrPgP89BvatVl1bdg/viewform",
  557. fields: {
  558. username: "324217272",
  559. state: "1065619417",
  560. issue: "1086951221",
  561. streetname: "1163516948",
  562. incorrectp: "1191620241",
  563. correctp: "1649051316",
  564. permalink: "2028167849",
  565. instructions: "2120232339",
  566. comments: "1917392591"
  567. }
  568. }, {
  569. //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+
  570. name: "Illinois event/weather closures",
  571. url: "https://docs.google.com/forms/d/e/1FAIpQLSff7nsBw8qxCojBdxrjTPl6tercqyyzGy92Vif_SBdHkYDchw/viewform",
  572. fields: {
  573. username: "1204781462",
  574. closureStatus: "828228572",
  575. permalink: "1754051160",
  576. closedDir: "1414240321",
  577. closedReason: "430378754",
  578. endDate: "900957975",
  579. streetname: "1647952662",
  580. fromStreet: "1501712688",
  581. toStreet: "2094306654",
  582. county: "1051351191",
  583. city: "1093044522",
  584. source: "172235277",
  585. notes: "1722909714"
  586. }
  587. }, {
  588. //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
  589. name: "WV closures",
  590. url: "https://docs.google.com/forms/d/e/1FAIpQLSf3YAdjscVvJXiU4KWq0e8J8XyCyYBDyharRoXW3bf6R4wH1w/viewform",
  591. fields: {
  592. username: "1553765347",
  593. closureStatus: "1264424583",
  594. permalink: "1811077109",
  595. closedDir: "792657790",
  596. closedReason: "345142186",
  597. endDate: "1102521735",
  598. streetname: "2015424420",
  599. fromStreet: "1547375393",
  600. toStreet: "1335391716",
  601. county: "1714138473",
  602. source: "1803937317",
  603. notes: "1648634142"
  604. }
  605. }, {
  606. //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
  607. name: "Wisconsin event/weather closures",
  608. url: "https://docs.google.com/forms/d/e/1FAIpQLSeiKY0KsO0xN69Asw77MARQFmxOy6zQXF-k2OQdWOfwtiCp7Q/viewform",
  609. fields: {
  610. username: "1204781462",
  611. closureStatus: "828228572",
  612. permalink: "1754051160",
  613. closedDir: "1414240321",
  614. closedReason: "430378754",
  615. endDate: "900957975",
  616. streetname: "1647952662",
  617. fromStreet: "1501712688",
  618. toStreet: "2094306654",
  619. county: "1051351191",
  620. city: "1093044522",
  621. source: "172235277",
  622. notes: "1722909714"
  623. }
  624. }
  625. /*{
  626. //https://docs.google.com/forms/d/e/1FAIpQLScY_5WKyYTqvH1fdiBThqLO4DRIzFzgdBtBexw5-iKL_LOzBw/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
  627. name: "USA Weather related closures",
  628. url: "https://docs.google.com/forms/d/e/1FAIpQLScY_5WKyYTqvH1fdiBThqLO4DRIzFzgdBtBexw5-iKL_LOzBw/viewform",
  629. username: "1553765347",
  630. closureStatus: "1264424583",
  631. permalink: "1811077109",
  632. closedDir: "792657790",
  633. closedReason: "345142186",
  634. endDate: "1102521735",
  635. streetname: "2015424420",
  636. fromStreet: "1547375393",
  637. toStreet: "1335391716",
  638. state: "1867193205",
  639. county: "1714138473",
  640. source: "1803937317",
  641. notes: "1648634142",
  642. }*/
  643. ];
  644.  
  645. forms.forEach(function (key, i) {
  646. ffMnu.options.add(new Option(forms[i].name, i));
  647. });
  648. ffBtn.innerHTML = "Go to Form";
  649. ffBtn.onclick = function () {
  650. //alert(ffMnu.options[ffMnu.selectedIndex].value+": "+forms[ffMnu.options[ffMnu.selectedIndex].value].name);
  651. ff_saveSettings();
  652. formLink = ff_createFormLink(forms[ffMnu.options[ffMnu.selectedIndex].value]);
  653. if (typeof formLink === "undefined") {
  654. return;
  655. }
  656.  
  657. if ($("#ff-open-in-tab").prop("checked")) {
  658. window.open(formLink, "_blank");
  659. } else {
  660. window.open(formLink, formWindowName, formWindowSpecs);
  661. }
  662. };
  663. ffDiv.appendChild(ffMnu);
  664. ffDiv.appendChild(ffBtn);
  665. selElem[0].appendChild(ffDiv);
  666.  
  667. return;
  668. }
  669.  
  670. function ff_loadSettings() {
  671. var todayDate = new Date(),
  672. futureDate = new Date(),
  673. daysInFuture = 3;
  674. var today = todayDate.getFullYear() + "-" + (todayDate.getMonth() + 1 < 10 ? "0" + (todayDate.getMonth() + 1) : todayDate.getMonth() + 1) + "-" + todayDate.getDate();
  675. futureDate.setDate(futureDate.getDate() + daysInFuture);
  676.  
  677. var ffOpenInTab = localStorage.getItem("ff-open-in-tab");
  678. if (ffOpenInTab === "1") {
  679. $("#ff-open-in-tab").trigger("click");
  680. }
  681. var ffClosureReason = localStorage.getItem("ff-closure-reason");
  682. if (ffClosureReason !== null) {
  683. $("#ff-closure-reason").val(ffClosureReason);
  684. }
  685. var ffClosureEndDate = localStorage.getItem("ff-closure-endDate");
  686. if (ffClosureEndDate !== null && ffClosureEndDate !== "" && ffClosureEndDate >= today) {
  687. $("#ff-closure-endDate").val(ffClosureEndDate);
  688. } else {
  689. var closureDate = futureDate.getFullYear() + "-" + (futureDate.getMonth() + 1 < 10 ? "0" + (futureDate.getMonth() + 1) : futureDate.getMonth() + 1) + "-" + (futureDate.getDate() < 10 ? "0" + futureDate.getDate() : futureDate.getDate());
  690. $("#ff-closure-endDate").val(closureDate);
  691. }
  692. var ffClosureEndTime = localStorage.getItem("ff-closure-endTime");
  693. if (ffClosureEndTime !== null && ffClosureEndTime !== "") {
  694. $("#ff-closure-endTime").val(ffClosureEndTime);
  695. }
  696. //formfiller_log("Settings loaded");
  697. return;
  698. }
  699.  
  700. function ff_saveSettings() {
  701. if ($("#ff-open-in-tab").prop("checked")) {
  702. localStorage.setItem("ff-open-in-tab", "1");
  703. } else {
  704. localStorage.setItem("ff-open-in-tab", "0");
  705. }
  706. localStorage.setItem("ff-closure-reason", $("#ff-closure-reason").val());
  707. localStorage.setItem("ff-closure-endDate", $("#ff-closure-endDate").val());
  708. localStorage.setItem("ff-closure-endTime", $("#ff-closure-endTime").val());
  709. //formfiller_log("Settings saved");
  710. return;
  711. }
  712.  
  713. function ff_addUserTab() {
  714. var userInfo = document.getElementById("user-info"),
  715. userTabs = document.getElementById("user-tabs"),
  716. navTabs = userTabs.getElementsByClassName("nav-tabs"),
  717. tabContent = userInfo.getElementsByClassName("tab-content");
  718. var ffTab = document.createElement("li"),
  719. ffPanel = document.createElement("div"),
  720. ffReason = document.createElement("div"),
  721. ffEndDate = document.createElement("div"),
  722. ffNewTabBox = document.createElement("input"),
  723. ffNewTabLabel = document.createElement("label"),
  724. ffTabInfo = document.createElement("div");
  725.  
  726. ffTab.innerHTML = '<a title="Form Filler" href="#sidepanel-formfill" data-toggle="tab"><img class="fa" src="' + WMEFFIcon + '" width="15px" /></a>';
  727. ffPanel.id = "sidepanel-formfill";
  728. ffPanel.className = "tab-pane";
  729.  
  730. ffTabInfo.innerHTML = '<b>' + WMEFFName + '</b> v' + WMEFFVersion;
  731.  
  732. ffNewTabBox.id = "ff-open-in-tab";
  733. ffNewTabBox.type = "checkbox";
  734. ffNewTabBox.name = "ff_open_tab";
  735.  
  736. ffNewTabLabel.innerHTML = "Open form in new tab";
  737. ffNewTabLabel.for = "ff_open_tab";
  738.  
  739. ffReason.className = "form-group";
  740. 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>';
  741.  
  742. ffEndDate.className = "form-group";
  743. 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>';
  744.  
  745. ffPanel.appendChild(ffTabInfo);
  746. ffPanel.appendChild(ffNewTabBox);
  747. ffPanel.appendChild(ffNewTabLabel);
  748. ffPanel.appendChild(ffReason);
  749. ffPanel.appendChild(ffEndDate);
  750. navTabs[0].appendChild(ffTab);
  751. tabContent[0].appendChild(ffPanel);
  752.  
  753. if (typeof $.fn.datepicker !== "undefined") {
  754. $("#ff-closure-endDate").datepicker({
  755. format: "yyyy-mm-dd",
  756. todayHighlight: true,
  757. autoclose: true
  758. });
  759. } else {
  760. if (typeof $.fn.daterangepicker !== "undefined") {
  761. $("#ff-closure-endDate").daterangepicker({
  762. singleDatePicker: true,
  763. locale: {
  764. format: "YYYY-MM-DD"
  765. }
  766. });
  767. }
  768. }
  769.  
  770. if (typeof $.fn.timepicker !== "undefined") {
  771. $("#ff-closure-endTime").timepicker({
  772. template: false,
  773. defaultTime: "00:00",
  774. showMeridian: false
  775. });
  776. }
  777.  
  778. ff_loadSettings();
  779. $("#ff-closure-reason").change(function () {
  780. ff_saveSettings();
  781. });
  782. $("#ff-closure-endDate").change(function () {
  783. ff_saveSettings();
  784. });
  785. $("#ff-closure-endTime").change(function () {
  786. ff_saveSettings();
  787. });
  788. $("#ff-open-in-tab").click(function () {
  789. ff_saveSettings();
  790. });
  791. }
  792.  
  793. setTimeout(formfiller_bootstrap, 2000);
  794. }());