WME RRC AutoLock

Locks RRCs and Cameras to set level instead of autolock to rank of editor

目前為 2020-07-26 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name WME RRC AutoLock
  3. // @namespace https://github.com/jm6087
  4. // @version 2020.07.25.00
  5. // @description Locks RRCs and Cameras to set level instead of autolock to rank of editor
  6. // @author jm6087
  7. // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
  8. // @require https://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js
  9. // @require https://gf.qytechs.cn/scripts/27254-clipboard-js/code/clipboardjs.js
  10. // @grant none
  11. // ==/UserScript==
  12.  
  13. /* global W */
  14. /* global WazeWrap */
  15. /* global $ */
  16. /* global I18n */
  17. /* global wazedevtoastr */
  18. /* global _ */
  19.  
  20. (function() {
  21. 'use strict';
  22. var UPDATE_NOTES = `Locks (adjustable) RRCs to L4 and Cameras to L5 upon selection.<br><br>
  23. 2020.07.12.00 Sets lock ranks to country minimum instead of N/A
  24. <br><br>
  25. Thanks for Dude495, TheCre8r, and SkiDooGuy for their assistance and encouragement`
  26.  
  27.  
  28. // PREVIOUS NOTES
  29. // with assistance and encouragment from Dude495, TheCre8r, and SkiDooGuy
  30.  
  31. // 2020.07.12.00 Sets lock ranks to country minimum instead of N/A
  32. // 2020.07.11.00 Refresh country settings/defaults
  33. // 2020.07.07.00 Can now add countries without version update. Added lock all on screen button.
  34. // 2020.07.04.00 - Added Pakistan lock rank
  35. // 2020.07.02.00 - Tab color change when there are RRCs or ECs on screen that are not set to lock level
  36. // 2020.06.24.01 Dropped camera min to 3
  37. // 2020.06.23.00 Fixed banner issue
  38. // 2020.06.21.01 - Released to editors
  39. // 2020.06.18.02 - Added check to see if RRC/camera are within editable areas
  40. // 2020.06.18.00 - More code clean up
  41. // 2020.06.17.00 - Code clean up
  42. // 2020.06.16.01 - Added WazeWrap storage. (Thanks Daniel)
  43. // 2020.06.16.00 - Minor changes
  44. // Changed a little text in panel at recomendation of Dude495
  45. // Added option for changing lock level
  46. // BUG fix
  47. // Script currently conflicts with WME Tiles Update. Not allowing unverified RRCs to autolock initially<br><br>
  48. // The enable script now works and persists thanks to dude495
  49. // I think I got the enable button to default to checked. Still working on persisting through refresh when disabled
  50. // Fixed items that juliansean pointed out
  51.  
  52. // Variables that designate beta version - Do no copy to other versions
  53. var TAB_NAME = 'RRC-AL';
  54. let sPanel = `#sidepanel-rrc-al`;
  55. const STORE_NAME = "RRCSettings";
  56. let LS = 1594558757308;
  57.  
  58. const CountrySS = 'https://sheets.googleapis.com/v4/spreadsheets/1wPb4tqTsES7EgAyxVqRRsRiWBDurld5NzN7IdC4pnSo/values/CountryMinimumLocks/?key='+atob('QUl6YVN5QXUxcl84ZDBNdkJUdEFwQ2VZdndDUXR6M2I0cmhWZFNn');
  59. const BetaSS = 'https://sheets.googleapis.com/v4/spreadsheets/1wPb4tqTsES7EgAyxVqRRsRiWBDurld5NzN7IdC4pnSo/values/Beta/?key='+atob('QUl6YVN5QXUxcl84ZDBNdkJUdEFwQ2VZdndDUXR6M2I0cmhWZFNn');
  60. var BETA_TESTERS = [];
  61. var COUNTRYID = [];
  62. let countQty;
  63.  
  64. // Other variables
  65. var VERSION = GM_info.script.version;
  66. var SCRIPT_NAME = GM_info.script.name;
  67. const USER = {name: null, rank:null};
  68. var RRCAutoLockSettings;
  69. var LastEditorUserName;
  70. var CameraType;
  71. var CameraTypeWW;
  72. const mapCenter = {lon: null, lat: null};
  73. const Lang = "en-US";
  74. var newCleanPL;
  75. var OpenBrack;
  76. var ClosedBrack;
  77. var RRCAutoLock;
  78. var newLockLevel;
  79. let UpdateObj;
  80. var SelModel;
  81. var RRCAutolockRankplusOne;
  82. var RRCAutoLockRankOverLock;
  83. var modelRank; // -- DBSOONER
  84. var manAuto;
  85. var SelMan;
  86. var count;
  87. var evalCount;
  88. var CountryID;
  89. var tabColor;
  90. var originalLon;
  91. var movedLon;
  92. var originalZoom;
  93. var movedZoom;
  94. var RRCscreenCount;
  95. var ECscreenCount;
  96. var RRCmin;
  97. var ECmin;
  98. var CountryName;
  99. var forceDefault;
  100.  
  101. function RRCscreenLock(){
  102. const extentGeometry = W.map.getOLMap().getExtent().toGeometry();
  103. count = 0;
  104. evalCount = 0;
  105. CameraTypeWW = "Railroad Crossing";
  106. _.each(W.model.railroadCrossings.getObjectArray(), v => {
  107. if (extentGeometry.intersects(v.geometry)) {
  108. evalCount++;
  109. if (count < countQty) {
  110. SelModel = v;
  111. manAuto = "Auto";
  112. RRCsharedLock();
  113. }
  114. }
  115. })
  116. ScreenlockCompleted();
  117. }
  118.  
  119. function ECscreenLock(){
  120. const extentGeometry = W.map.getOLMap().getExtent().toGeometry();
  121. count = 0;
  122. evalCount = 0;
  123. CameraTypeWW = "Enforcement Camera";
  124. _.each(W.model.cameras.getObjectArray(), v => {
  125. if (extentGeometry.intersects(v.geometry)) {
  126. evalCount++;
  127. if (count < countQty) {
  128. SelModel = v;
  129. manAuto = "Auto";
  130. RRCsharedLock();
  131. }
  132. }
  133. })
  134. ScreenlockCompleted();
  135. }
  136.  
  137. function ScreenlockCompleted() {
  138. wazedevtoastr.options.timeOut = 5000;
  139. if (count == 0) {
  140. WazeWrap.Alerts.info(SCRIPT_NAME, ['All ' + CameraTypeWW + ' are already at assigned lock level, above', 'your edit rank, or the ' + CameraTypeWW + ' are not in your editable area', 'No changes were made'].join('\n'));
  141. }else{
  142. originalLon = "";
  143. RRCscreenMove()
  144. WazeWrap.Alerts.success(SCRIPT_NAME, ['RRC-AL locked ' + count + " (of " + evalCount + ") " + CameraTypeWW, ' that were equal to or below ' + USER.rank, 'They are now locked at lock level ' + newLockLevel].join('\n'));
  145. console.log (SCRIPT_NAME, "RCC Screenlock completed");
  146. }
  147. }
  148.  
  149. function setRRCAutoLock() {
  150. SelMan = W.selectionManager;
  151. if (SelMan.getSelectedFeatures().length > 0){ // Determines if there is an item selected
  152. SelModel = SelMan.getSelectedFeatures()[0].model;
  153. manAuto = "Manual";
  154. RRCsharedLock();
  155. originalLon = "";
  156. RRCscreenMove()
  157. }
  158. }
  159.  
  160. function RRCsharedLock () {
  161.  
  162. wazedevtoastr.options.timeOut = 2500; // Used to adjust the timing of the WW banner messages
  163. if ((SelModel.type !== 'camera') && (SelModel.type !== 'railroadCrossing')) return; // Suggested by DBSOONER - exits script if the object selected is not what I want
  164.  
  165. if (SelModel.type === 'camera'){ // Determines camera type is Enforcement Camera
  166. CameraType = 'camera';
  167. CameraTypeWW = 'Enforcement Camera';
  168. modelRank = ($('#ECAutoLockLevelOption')[0].value - 1);
  169. }else{
  170. if (SelModel.type === 'railroadCrossing'){
  171. CameraType = 'railroadCrossing';
  172. CameraTypeWW = 'Railroad Crossing';
  173. modelRank = ($('#RRCAutoLockLevelOption')[0].value - 1);
  174. }
  175. }
  176. if (USER.rank >= SelModel.attributes.rank + 1 && SelModel.arePropertiesEditable() == false){ // Checking to see if the the editor is high enough rank and if the so, then checking to see if the camera is editable. If not, then must not be in EA.
  177. if (manAuto == "Manual") {
  178. wazedevtoastr.options.timeOut = 6000;
  179. WazeWrap.Alerts.error(SCRIPT_NAME, [CameraTypeWW + ' does not appear to be in your edit area.', 'Please check your Editable Areas layer to ensure you have edit rights'].join('\n'));
  180. }
  181. }else{
  182. //checks to see if Enabled is checked
  183. if (RRCAutoLockSettings.RRCAutoLockEnabled == false && CameraType == 'railroadCrossing') // Warning message is valid and MUST be there
  184. return console.log(SCRIPT_NAME, CameraTypeWW + " is disabled");
  185. if (RRCAutoLockSettings.ECAutoLockEnabled == false && CameraType == 'camera') // Warning message is valid and MUST be there
  186. return console.log(SCRIPT_NAME, CameraTypeWW + " is disabled");
  187. console.log(SCRIPT_NAME, "Script is enabled");
  188.  
  189. RRCAutoLockRankOverLock = SelModel.attributes.rank + 1; // Checks rank of selected RRC or EC
  190. if ((SelModel.attributes.unapproved == true) || (SelModel.attributes.lockRank == null)) { // Checks to see if selected RRC or EC is unverified
  191. RRCAutolockRankplusOne = ("Auto (" + (SelModel.attributes.rank + 1)+")"); // If unverified, sets the text for WW with Auto(lock number)
  192. }else{
  193. RRCAutolockRankplusOne = SelModel.attributes.lockRank + 1; // If already verified, sets text for WW to lock number only
  194. };
  195. if (USER.rank < modelRank + 1){
  196. newLockLevel = USER.rank;
  197. if (manAuto == "Manual") {
  198. RRCAutoLock = $(`input[id^="lockRank-${USER.rank - 1}"]`); // Sets variable for which lockrank to click if the user rank is less than the settings
  199. }else{
  200. if (manAuto == "Auto") {
  201. RRCAutoLock = USER.rank - 1;
  202. }
  203. }
  204. }else{
  205. newLockLevel = modelRank + 1;
  206. if (manAuto == "Manual") {
  207. RRCAutoLock = $(`input[id^="lockRank-${modelRank}"]`); // Sets variable for which lockrank to click
  208. }else{
  209. if (manAuto == "Auto") {
  210. RRCAutoLock = modelRank;
  211. }
  212. }
  213. }
  214. if (manAuto == "Manual") {
  215. if (SelMan.hasSelectedFeatures() && SelModel.type === CameraType){
  216. // Finds ID number of the last editor
  217. let lastEditBy = SelModel.attributes.updatedBy;
  218. if (lastEditBy === null) {
  219. lastEditBy = SelModel.attributes.createdBy;
  220. }
  221. if (lastEditBy == undefined) {
  222. LastEditorUserName = USER.name;
  223. }else{
  224. // Finds the UserName based off the ID of last editor
  225. LastEditorUserName = W.model.users.objects[lastEditBy].userName;
  226. }
  227. }
  228. }
  229.  
  230. // Checks to see if User Rank is higher/equal to object lock AND if object is not equal to dropdown lock level in panel
  231. if ((USER.rank >= (SelModel.attributes.rank + 1)) && (SelModel.attributes.lockRank != modelRank)){
  232. if (manAuto == "Manual") {
  233. RRCAutoLock[0].click();
  234. }else{
  235. if (manAuto == "Auto") {
  236. if (SelModel.attributes.unapproved == false || SelModel.type == "camera") {
  237. count++;
  238. W.model.actionManager.add(new UpdateObj(SelModel, { lockRank: RRCAutoLock }));
  239. }else{
  240. console.log (SCRIPT_NAME, CameraTypeWW, "ID", SelModel.attributes.id, "skipped since it is unapproved");
  241. }
  242. }
  243. }
  244. // Checks to see if WazeWrap banner Enabled is checked
  245. if (manAuto == "Manual") {
  246. if (RRCAutoLockSettings.RRCAutoLockWazeWrapSuccessEnabled == true){
  247. console.log(SCRIPT_NAME, "WazeWrap is enabled");
  248. WazeWrap.Alerts.success(SCRIPT_NAME, [CameraTypeWW + ' changed from lock level ' + RRCAutolockRankplusOne + ' to ' + newLockLevel, 'Last edited by ' + LastEditorUserName].join('\n'));
  249. }
  250. }
  251. console.log(SCRIPT_NAME, "Version #", VERSION, "-", CameraTypeWW + "ID", SelModel.attributes.id, "Lock level changed from", RRCAutolockRankplusOne , "to", newLockLevel, "Last Edited by", LastEditorUserName);
  252. }else{
  253. // Checks to see if User rank is greater or equal to object lock level AND if object is already equal to dropdown lock level in panel
  254. if (USER.rank >= (SelModel.attributes.rank + 1) && SelModel.attributes.lockRank == modelRank){
  255. if (manAuto == "Manual") {
  256. if (RRCAutoLockSettings.RRCAutoLockWazeWrapInfoEnabled == true){ // Check to see if WW banner enabled
  257. console.log(SCRIPT_NAME, "WazeWrap is enabled");
  258. WazeWrap.Alerts.info(SCRIPT_NAME, [CameraTypeWW + ' lock not changed, already at lock level ' + RRCAutolockRankplusOne, 'Last edited by ' + LastEditorUserName].join('\n'));
  259. }
  260. }
  261. console.log (SCRIPT_NAME, "Version #", VERSION, "-", CameraTypeWW, "ID ", SelModel.attributes.id, "lock not changed, already at lock level", RRCAutolockRankplusOne);
  262. }else{
  263. // Checks to see if object is locked above User rank
  264. if (USER.rank < (SelModel.attributes.rank + 1)){
  265. if (manAuto == "Manual") {
  266. wazedevtoastr.options.timeOut = 5000;
  267. if (RRCAutoLockRankOverLock > 5){
  268. WazeWrap.Alerts.error(SCRIPT_NAME, [CameraTypeWW + ' is locked above your rank', 'You will need assistance from an Rank ' + RRCAutoLockRankOverLock + ' editor', 'Last edited by ' + LastEditorUserName].join('\n'));
  269. }else{
  270. WazeWrap.Alerts.error(SCRIPT_NAME, [CameraTypeWW + ' is locked above your rank', 'You will need assistance from at least a Rank ' + RRCAutoLockRankOverLock + ' editor', 'Last edited by ' + LastEditorUserName].join('\n'));
  271. console.log (SCRIPT_NAME, "Version #", VERSION, " - ", CameraTypeWW, " ID ", SelModel.attributes.id , " is locked above editor rank");
  272. }
  273. }
  274. }
  275. }
  276. }
  277. }
  278. manAuto = null;
  279. }
  280.  
  281. function RRCAutoLockTab()
  282. {
  283. var $RRCsection = $("<div>");
  284. $RRCsection.html([
  285. '<div>',
  286. '<h4 style="margin-bottom:0px;"><b>'+ SCRIPT_NAME +'</b></h4>',
  287. VERSION +'</br>',
  288. '<div id="countryName"></div>',
  289. '<button id="force-country-settings" title="Reset country defaults" class="btn"><i class="fa fa-refresh"></i> Reset country settings/defaults <i class="fa fa-refresh"></i></button>',
  290. '<div class="form-group"></div>',
  291. '<b><input type="checkbox" id="RRCAutoLockCheckbox"> RRC lock enabled</b></br>',
  292. '<b><id="RRCAutoLockLevelValue">RRC lock level: <select id="RRCAutoLockLevelOption"></b></br>',
  293. '<option value="0">N/A</option>',
  294. '<option value="1">1</option>',
  295. '<option value="2">2</option>',
  296. '<option value="3">3</option>',
  297. '<option value="4">4</option>',
  298. '<option value="5">5</option>',
  299. '<option value="6">6</option>',
  300. '</select></br>',
  301. '<b><div id="RRCscreenCount"></div></b></br>',
  302. '<b><input type="checkbox" id="ECAutoLockCheckbox"> Enforcement camera lock enabled</b></br>',
  303. '<b><id="ECAutoLockLevelValue">Enforcement camera lock level: <select id="ECAutoLockLevelOption"></b></br>',
  304. '<option value="0">N/A</option>',
  305. '<option value="1">1</option>',
  306. '<option value="2">2</option>',
  307. '<option value="3">3</option>',
  308. '<option value="4">4</option>',
  309. '<option value="5">5</option>',
  310. '<option value="6">6</option>',
  311. '</select></br>',
  312. '<b><div id="ECscreenCount"></div></b></br>',
  313. '<b><input type="checkbox" id="RRCAutoLockWazeWrapSuccessCheckbox"> Alerts: Success</b></br>',
  314. '<b><input type="checkbox" id="RRCAutoLockWazeWrapInfoCheckbox"> Alerts: Info</b></br>',
  315. '<b><div id="WMETUWarning"></div></b>',
  316. '<b><input type="checkbox" id="TUWARNING"> Disable URMP / TU Warning</b></br>',
  317. '<b><h4>Your WME window was last refreshed at:</h4></b>',
  318. '<b><h4><div id="CurrentDate"></div></h4></b></br>',
  319. '<div class="form-group"></br>',
  320. '<div id="panelCountQty"></div></br>',
  321. '<div><input type="button" id="RRC-Screen-Lock" title="RRC Screen Lock" value="Lock all RRCs" class="btn btn-danger btn-xs RRC-Button"></div></br>',
  322. '<div><input type="button" id="EC-Screen-Lock" title="EC Screen Lock" value="Lock all Enforcement Cameras" class="btn btn-danger btn-xs RRC-Button"></div></div></br>',
  323. // BETA USER FEATURE BELOW
  324. ////////////////////////////////////////////////////////////////////////////////////////////////
  325. '<div class="form-group">', // BETA USER FEATURE
  326. '<b><div id="BETAonly">The features below only show for editors listed as Beta testers<div></b></br>', // BETA USER FEATURE
  327. '<div id="discord">', // BETA USER FEATURE
  328. '<b><input type="checkbox" id="DiscordPermalinkCheckbox"> Create PL with < > for Discord.</div></b></br>', // https://www.w3schools.com/bootstrap/bootstrap_buttons.asp
  329. '<input type="button" id="Permalink-Button-Name" title="PL" value="Copy Clean PL to your clipboard" class="btn btn-info btn-xs RRC-Button"></br></br>', // BETA USER FEATURE
  330. '<input type="button" id="Permalink-Button-Input" title="PL" value="Clean PL from another editor" class="btn btn-info btn-xs RRC-Button"></br></br>', // BETA USER FEATURE
  331. '<div class="form-group">',
  332. '<b><h5><div id="USERedits"><div></h5></b></br></div></div>', // BETA USER FEATURE
  333. '</div>', // BETA USER FEATURE
  334. '<p><div id="sheet"><a href="https://docs.google.com/spreadsheets/d/1wPb4tqTsES7EgAyxVqRRsRiWBDurld5NzN7IdC4pnSo/edit#gid=1675959019" target="_blank">Sheet</a></div></p>',
  335. '<div>',
  336. ].join(' '));
  337.  
  338. new WazeWrap.Interface.Tab(TAB_NAME, $RRCsection.html(), RRCAutoLockInitializeSettings);
  339. $("#Permalink-Button-Name").click(CleanPermaLink); // BETA USER FEATURE
  340. $("#Permalink-Button-Input").click(inputPermaLink); // BETA USER FEATURE
  341. $("#RRC-Screen-Lock").click(RRCscreenLock); //
  342. $("#EC-Screen-Lock").click(ECscreenLock); //
  343. $("#force-country-settings").click(forceCountrySetting); //
  344. }
  345.  
  346. // BETA USER FEATURE BELOW
  347. /////////////////////////////////////////////////////////////////////////////////////////
  348. function CleanPermaLink(){
  349. let selectedID;
  350. let PLselFeat = W.selectionManager.getSelectedFeatures();
  351. let LatLonCenter = W.map.getCenter();
  352. let center4326 = WazeWrap.Geometry.ConvertTo4326(LatLonCenter.lon, LatLonCenter.lat);
  353. let PLurl = 'https://www.waze.com/' + I18n.currentLocale() + '/editor?env=' + W.app.getAppRegionCode() + "&lon=";
  354. if (RRCAutoLockSettings.DiscordPermalink == true){
  355. OpenBrack = "<";
  356. ClosedBrack = ">";
  357. }else{
  358. OpenBrack = "";
  359. ClosedBrack = "";
  360. }
  361. if (PLselFeat.length > 0){
  362. let selectedType = PLselFeat[0].model.type;
  363. if (selectedType == 'segment') {
  364. selectedID = $('#'+selectedType+'-edit-general > ul > li:contains("ID:")')[0].textContent.match(/\d.*/)[0];
  365. }else{
  366. selectedID = PLselFeat[0].model.attributes.id;
  367. }
  368. newCleanPL = OpenBrack + PLurl + center4326.lon.toFixed(5) + "&lat=" + center4326.lat.toFixed(5) + "&zoom=5&" + selectedType + "s=" + selectedID + ClosedBrack;
  369. }else{
  370. newCleanPL = OpenBrack + PLurl + center4326.lon.toFixed(5) + "&lat=" + center4326.lat.toFixed(5) + "&zoom=5" + ClosedBrack;
  371. }
  372. copyToClipboard();
  373. }
  374.  
  375. function inputPermaLink(){
  376. // Add WazeWrap Prompt box to grab PL and then clean it up
  377. WazeWrap.Alerts.prompt(SCRIPT_NAME, "Paste your PL", "", OKinputPermaLink, cancelInputPermaLink); // Prompts to enter a PL
  378. }
  379. function OKinputPermaLink(){
  380. let inputData = $(".toast-prompt-input")[0].value;
  381. var regexs = {
  382. 'wazeurl': new RegExp('(?:http(?:s):\/\/)?(?:www\.|beta\.)?waze\.com\/(?:.*?\/)?(editor|livemap)[-a-zA-Z0-9@:%_\+,.~#?&\/\/=]*', "ig")
  383. };
  384. if (inputData.match(regexs.wazeurl)){
  385. let PLurl = 'https://www.waze.com/' + I18n.currentLocale() + '/editor?env=' + W.app.getAppRegionCode() + "&lon=";
  386. var inputSegsVen;
  387. let params = inputData.match(/lon=(-?\d*.\d*)&lat=(-?\d*.\d*)/);
  388. let inputLon = params[1];
  389. let inputLat = params[2];
  390. let inputSegs = inputData.match(/&segments=(.*)(?:&|$)/);
  391. let inputVenue = inputData.match(/&venues=(.*)(?:&|$)/);
  392. let inputRRC = inputData.match(/&railroadCrossings=(.*)(?:&|$)/)
  393. if ((inputSegs == null) || (inputVenue == null)) (inputSegsVen = "");
  394. if (inputSegs != null) (inputSegsVen = "&segments=" + inputSegs[1]);
  395. if (inputVenue != null) (inputSegsVen = "&venues=" + inputVenue[1]);
  396. if (inputRRC != null) (inputSegsVen = "&railroadCrossings=" + inputRRC[1]);
  397. if (RRCAutoLockSettings.DiscordPermalink == true){
  398. OpenBrack = "<";
  399. ClosedBrack = ">";
  400. }else{
  401. OpenBrack = "";
  402. ClosedBrack = "";
  403. }
  404. newCleanPL = OpenBrack +PLurl + inputLon + "&lat=" + inputLat + "&zoom=6" + inputSegsVen + ClosedBrack;
  405. copyToClipboard();
  406. console.log (SCRIPT_NAME, 'Inputed PL now clean ' + newCleanPL);
  407. }else{
  408. wazedevtoastr.options.timeout = 2000;
  409. WazeWrap.Alerts.info(SCRIPT_NAME, "That did not appear to be a valid permalink");
  410. }
  411. }
  412. function cancelInputPermaLink(){
  413. console.log (SCRIPT_NAME, "cancel button");
  414. }
  415.  
  416. function copyToClipboard(){
  417. // NEXT 4 LINES COPIES CLEAN PL TO CLIPBOARD
  418. var copied = $('<textarea id="PLcopy" rows="1" cols="1">').val(newCleanPL/*.replace(/\_*\n/g, '\n')*/).appendTo('body').select(); // Creates temp text box with the PL
  419. document.execCommand('copy'); // Copies the PL to clipboard
  420. var rembox = document.getElementById('PLcopy');
  421. document.body.removeChild(rembox); // Deletes temp text box
  422. wazedevtoastr.options.timeOut = 1500;
  423. WazeWrap.Alerts.info(SCRIPT_NAME, 'PL saved to your clipboard');
  424. console.log(SCRIPT_NAME, newCleanPL + ' copied to your clipboard');
  425. }
  426. ////////////////////////////////////
  427. // BETA USER FEATURE ABOVE
  428.  
  429. function disabledOptions() { // Disables the drop down if the enabled option is off.
  430. $('#RRCAutoLockLevelOption')[0].disabled = !RRCAutoLockSettings.RRCAutoLockEnabled;
  431. $('#ECAutoLockLevelOption')[0].disabled = !RRCAutoLockSettings.ECAutoLockEnabled;
  432. if (RRCAutoLockSettings.RRCAutoLockEnabled == false) {
  433. document.getElementById("RRC-Screen-Lock").disabled = true;
  434. }else{
  435. document.getElementById("RRC-Screen-Lock").disabled = false;
  436. }
  437. if (RRCAutoLockSettings.ECAutoLockEnabled == false) {
  438. document.getElementById("EC-Screen-Lock").disabled = true;
  439. }else{
  440. document.getElementById("EC-Screen-Lock").disabled = false;
  441. }
  442. }
  443.  
  444. /*-- START SETTINGS --*/
  445. async function loadSettings() {
  446. let loadedSettings = $.parseJSON(localStorage.getItem(STORE_NAME)); // Loads settings from local storage, allows settings to persist with refresh
  447. const defaultSettings = { // sets default values for tab options
  448. RRCAutoLockLevelOption: "0",
  449. ECAutoLockLevelOption: "0",
  450. RRCAutoLockWazeWrapSuccessEnabled: true,
  451. RRCAutoLockWazeWrapInfoEnabled: true,
  452. RRCAutoLockEnabled: true,
  453. ECAutoLockEnabled: true,
  454. DiscordPermalink: true,
  455. TUWARNING: false,
  456. lastSaved: "",
  457. lastVersion: ""
  458. };
  459. RRCAutoLockSettings = loadedSettings ? loadedSettings : defaultSettings;
  460. for (let prop in defaultSettings) {
  461. if (!RRCAutoLockSettings.hasOwnProperty(prop)) {
  462. RRCAutoLockSettings[prop] = defaultSettings[prop];
  463. }
  464. }
  465. if (RRCAutoLockSettings.lastSaved <= LS) { // Clears local storage and resets to defaults if older version is found
  466. RRCAutoLockSettings = defaultSettings;
  467. await WazeWrap.Remote.SaveSettings(STORE_NAME, JSON.stringify(defaultSettings));
  468. localStorage.removeItem(STORE_NAME); // Clears local storage and resets to defaults if older version is found
  469. localStorage.setItem(STORE_NAME, JSON.stringify(RRCAutoLockSettings)); // saves settings to local storage for persisting when refreshed
  470. }
  471. const serverSettings = await WazeWrap.Remote.RetrieveSettings(STORE_NAME); //Settings stored to WazeWrap
  472. if (serverSettings && (serverSettings.lastSaved > RRCAutoLockSettings.lastSaved)) { // checks to see if WazeWrap stored settings are newer than what is stored in local storage
  473. $.extend(true, RRCAutoLockSettings, serverSettings);
  474. localStorage.setItem(STORE_NAME, JSON.stringify(RRCAutoLockSettings)); // saves settings to local storage for persisting when refreshed
  475. }
  476. if (RRCAutoLockSettings.ECAutoLockLevelOption == 0 || RRCAutoLockSettings.RRCAutoLockLevelOption == 0) {
  477. forceDefault = true;
  478. forceCountrySetting();
  479. console.log(SCRIPT_NAME, "Settings set to country minimum by default");
  480. }
  481. console.log(SCRIPT_NAME, "Settings Loaded");
  482. }
  483. function saveSettings() {
  484. if (localStorage) {
  485. RRCAutoLockSettings.lastVersion = VERSION;
  486. if (forceDefault != true) {
  487. RRCAutoLockSettings.RRCAutoLockEnabled = $('#RRCAutoLockCheckbox')[0].checked;
  488. RRCAutoLockSettings.ECAutoLockEnabled = $('#ECAutoLockCheckbox')[0].checked;
  489. RRCAutoLockSettings.RRCAutoLockWazeWrapSuccessEnabled = $('#RRCAutoLockWazeWrapSuccessCheckbox')[0].checked;
  490. RRCAutoLockSettings.RRCAutoLockWazeWrapInfoEnabled = $('#RRCAutoLockWazeWrapInfoCheckbox')[0].checked;
  491. }else{
  492. RRCAutoLockSettings.RRCAutoLockEnabled = true;
  493. RRCAutoLockSettings.ECAutoLockEnabled = true;
  494. RRCAutoLockSettings.RRCAutoLockWazeWrapSuccessEnabled = true;
  495. RRCAutoLockSettings.RRCAutoLockWazeWrapInfoEnabled = true;
  496. }
  497. RRCAutoLockSettings.RRCAutoLockLevelOption = $('#RRCAutoLockLevelOption')[0].value;
  498. RRCAutoLockSettings.ECAutoLockLevelOption = $('#ECAutoLockLevelOption')[0].value;
  499. RRCAutoLockSettings.lastSaved = Date.now();
  500. RRCAutoLockSettings.DiscordPermalink = $('#DiscordPermalinkCheckbox')[0].checked;
  501. RRCAutoLockSettings.TUWARNING = $('#TUWARNING')[0].checked;
  502. disabledOptions();
  503. localStorage.setItem(STORE_NAME, JSON.stringify(RRCAutoLockSettings)); // saves settings to local storage for persisting when refreshed
  504. WazeWrap.Remote.SaveSettings(STORE_NAME, JSON.stringify(RRCAutoLockSettings)); // saves settings to WazeWrap
  505. console.log(SCRIPT_NAME, 'Settings Saved '+ JSON.stringify(RRCAutoLockSettings));
  506. }
  507. }
  508. async function RRCAutoLockInitializeSettings(){
  509. USER.rank = W.loginManager.user.rank + 1;
  510. USER.name = W.loginManager.user.userName;
  511. let dte = Date();
  512. UpdateObj = require('Waze/Action/UpdateObject');
  513. await loadSettings();
  514. $('#RRCAutoLockUsername').text(USER.name);
  515. $('#RRCAutoLockRank').text(USER.rank);
  516. $('#RRCAutoLockTotalEdits').text(W.loginManager.user.totalEdits);
  517. $('#RRCAutoLockTotalPoints').text(W.loginManager.user.totalPoints);
  518. $('#RRCAutoLockCheckbox')[0].checked = RRCAutoLockSettings.RRCAutoLockEnabled;
  519. $('#ECAutoLockCheckbox')[0].checked = RRCAutoLockSettings.ECAutoLockEnabled;
  520. $('#RRCAutoLockWazeWrapSuccessCheckbox')[0].checked = RRCAutoLockSettings.RRCAutoLockWazeWrapSuccessEnabled;
  521. $('#RRCAutoLockWazeWrapInfoCheckbox')[0].checked = RRCAutoLockSettings.RRCAutoLockWazeWrapInfoEnabled;
  522. $('#RRCAutoLockLevelOption')[0].value = RRCAutoLockSettings.RRCAutoLockLevelOption;
  523. $('#ECAutoLockLevelOption')[0].value = RRCAutoLockSettings.ECAutoLockLevelOption;
  524. $('#DiscordPermalinkCheckbox')[0].checked = RRCAutoLockSettings.DiscordPermalink;
  525. $('#TUWARNING')[0].checked = RRCAutoLockSettings.TUWARNING;
  526. disabledOptions()
  527. setBetaFeatures(USER.name);
  528. setTimeout (loadCountryID, 2000);
  529. loadCountryID(CountryID);
  530. console.log(SCRIPT_NAME, "- Tab Created & User Rank is ",USER.rank);
  531. $('#RRCAutoLockCheckbox')[0].onchange = function() {
  532. console.log(SCRIPT_NAME, "RRCAutoLockCheckbox Settings changed");
  533. saveSettings();
  534. };
  535. $('#ECAutoLockCheckbox')[0].onchange = function() {
  536. console.log(SCRIPT_NAME, "ECAutoLockCheckbox Settings changed");
  537. saveSettings();
  538. };
  539. $('#RRCAutoLockWazeWrapSuccessCheckbox')[0].onchange = function() {
  540. console.log(SCRIPT_NAME, "RRCAutoLockWazeWrapSuccessCheckbox Settings changed");
  541. saveSettings();
  542. };
  543. $('#RRCAutoLockWazeWrapInfoCheckbox')[0].onchange = function() {
  544. console.log(SCRIPT_NAME, "RRCAutoLockWazeWrapInfoCheckbox Settings changed");
  545. saveSettings();
  546. };
  547. $('#RRCAutoLockLevelOption')[0].onchange = function() {
  548. let x = $('#RRCAutoLockLevelOption')[0].value;
  549. undoAction();
  550. console.log(SCRIPT_NAME, "RRCAutoLockLevelValue Settings changed to L" + x);
  551. saveSettings();
  552. };
  553. $('#ECAutoLockLevelOption')[0].onchange = function() {
  554. let x = $('#ECAutoLockLevelOption')[0].value;
  555. undoAction();
  556. console.log(SCRIPT_NAME, "ECAutoLockLevelValue Settings changed to L" + x);
  557. saveSettings();
  558. };
  559. $('#DiscordPermalinkCheckbox')[0].onchange = function() {
  560. console.log(SCRIPT_NAME, "Discord PL option changed");
  561. saveSettings();
  562. };
  563. $('#TUWARNING')[0].onchange = function() {
  564. console.log(SCRIPT_NAME, "TUWARNING");
  565. saveSettings();
  566. };
  567. if (($('#Info_server')[0]) || ($('#sidepanel-urt')[0])) { $('#WMETUWarning')[0].innerHTML = 'WME Tile Update and/or UR-MP Script Detected;<br>WMETU and UR-MP are known to cause problems with this script.<br>Disable WMETU and/or UR-MP if you experience any issues.';
  568. if (RRCAutoLockSettings.TUWARNING == false) {
  569. wazedevtoastr.options.timeOut = 8000;
  570. WazeWrap.Alerts.warning(SCRIPT_NAME, ["WME Tile Update and/or UR-MP Script Detected;","WMETU and UR-MP are known to cause problems with this script.","Disable WMETU and/or UR-MP if you experience any issues."].join('\n'));
  571. }
  572. }else {
  573. $('#WMETUWarning')[0].textContent = ''};
  574. $('#CurrentDate')[0].textContent = dte;
  575. document.getElementById("force-country-settings").style.padding = "1px";
  576. document.getElementById("RRC-Screen-Lock").style.padding = "1px";
  577. document.getElementById("EC-Screen-Lock").style.padding = "1px";
  578. document.getElementById("Permalink-Button-Name").style.padding = "1px";
  579. document.getElementById("Permalink-Button-Input").style.padding = "1px";
  580. }
  581. async function loadCountry() {
  582. await $.getJSON(CountrySS, function(cdata){
  583. COUNTRYID = cdata;
  584. console.log(SCRIPT_NAME, 'Country loaded....');
  585. });
  586. }
  587. async function loadBetaUsers() {
  588. await $.getJSON(BetaSS, function(ldata){
  589. BETA_TESTERS = ldata;
  590. console.log('RRC-AL Beta Users Loaded....');
  591. });
  592. }
  593. function getCountryFromSheet(CountryID){
  594. let mapped = COUNTRYID.values.slice(0).reverse().map(obj =>{
  595. return {ctry: obj[0].trim(), ctryID: obj[1].trim(), ctryRRC: obj[2].trim(), ctryEC: obj[3].trim()
  596. }
  597. });
  598. for(let i=0; i<mapped.length; i++){
  599. if(mapped[i].ctryID == CountryID) {
  600. return mapped[i];
  601. }
  602. }
  603. return null;
  604. }
  605.  
  606. function getFromSheetList(editorName){
  607. let mapped = BETA_TESTERS.values.slice(0).reverse().map(obj =>{
  608. return {username: obj[0].trim()
  609. }
  610. });
  611. countQty = [mapped[mapped.length-3], mapped[mapped.length-2]];
  612. for(let i=0; i<mapped.length; i++){
  613. if(mapped[i].username.toLowerCase() === editorName.toLowerCase()) {
  614. return mapped[i];
  615. }
  616. }
  617. return null;
  618. }
  619.  
  620. function setBetaFeatures(user) {
  621. let entry = getFromSheetList(user);
  622. if (entry == null) {
  623. countQty = countQty[1].username;
  624. $("#DiscordPermalinkCheckbox").hide();
  625. $('#Permalink-Button-Name').hide();
  626. $('#Permalink-Button-Input').hide();
  627. $('#discord').hide();
  628. $('#BETAonly').hide();
  629. $('#USERedits').hide();
  630. $('#sheet').hide();
  631. console.log(SCRIPT_NAME, "Not a beta user");
  632. }else{
  633. countQty = countQty[0].username;
  634. $('#USERedits')[0].textContent = 'Current Edit Count for '+ USER.name + ' - ' + W.loginManager.user.totalEdits;
  635. console.log(SCRIPT_NAME, "Beta features loaded");
  636. }
  637. $('#panelCountQty')[0].textContent = 'Lock up to ' + countQty + ' RRCs or ECs on screen';
  638. }
  639.  
  640. function loadCountryID() { // comment out the hide for each lock to show
  641.  
  642. var max = W.loginManager.user.rank + 1;
  643. CountryName = W.model.topCountry.name;
  644. if (W.model.topCountry.id == 235) CountryName = W.model.topState.name + ', USA';
  645. let cEntry = getCountryFromSheet(CountryID);
  646. if (RRCmin == null) {
  647.  
  648. $("#RRCAutoLockLevelOption option[value='0']").show();
  649. $("#RRCAutoLockLevelOption option[value='1']").show();
  650. $("#RRCAutoLockLevelOption option[value='2']").show();
  651. $("#RRCAutoLockLevelOption option[value='3']").show();
  652. $("#RRCAutoLockLevelOption option[value='4']").show();
  653. $("#RRCAutoLockLevelOption option[value='5']").show();
  654. $("#RRCAutoLockLevelOption option[value='6']").show();
  655. $("#ECAutoLockLevelOption option[value='0']").show();
  656. $("#ECAutoLockLevelOption option[value='1']").show();
  657. $("#ECAutoLockLevelOption option[value='2']").show();
  658. $("#ECAutoLockLevelOption option[value='3']").show();
  659. $("#ECAutoLockLevelOption option[value='4']").show();
  660. $("#ECAutoLockLevelOption option[value='5']").show();
  661. $("#ECAutoLockLevelOption option[value='6']").show();
  662.  
  663. // Sets country ID for RRC and EC minimum
  664.  
  665. if (cEntry == null) {
  666. RRCmin = 4;
  667. ECmin = 4;
  668. var ctry = "NOT LISTED ON SHEET";
  669. }else{
  670. if (cEntry.ctryRRC == null) {
  671. RRCmin = 4;
  672. }else{
  673. RRCmin = cEntry.ctryRRC;
  674. ctry = cEntry.ctry;
  675. }
  676. if (cEntry.ctryEC == null) {
  677. ECmin = 4;
  678. }else{
  679. ECmin = cEntry.ctryEC;
  680. }
  681. }
  682.  
  683. console.log(SCRIPT_NAME, 'Country ID is', CountryID, '-', ctry, ', the minimum RRC lock level is set to', RRCmin, 'and max rank set at', max);
  684. console.log(SCRIPT_NAME, 'Country ID is', CountryID, '-', ctry, ', the minimum EC lock level is set to', ECmin, 'and max rank set at', max);
  685.  
  686. if (max < RRCmin) {
  687. // wazedevtoastr.options.timeOut = 5000;
  688. // WazeWrap.Alerts.warning(SCRIPT_NAME, ["It appears that your user rank of R" + max,"is less than the minimum lock level of " + RRCmin + " for your country for Railroad Crossings"].join('\n'));
  689. RRCmin = 10;
  690. }
  691. if (max < ECmin) {
  692. // wazedevtoastr.options.timeOut = 5000;
  693. // WazeWrap.Alerts.warning(SCRIPT_NAME, ["It appears that your user rank of R" + max,"is less than the minimum lock level of " + ECmin + " for your country for Enforcement Cameras"].join('\n'));
  694. ECmin = 10;
  695. }
  696.  
  697. if (RRCmin >= 2) {
  698. $("#RRCAutoLockLevelOption option[value='0']").hide();
  699. $("#RRCAutoLockLevelOption option[value='1']").hide();
  700. if (RRCmin >= 3) {
  701. $("#RRCAutoLockLevelOption option[value='2']").hide();
  702. if (RRCmin >= 4) {
  703. $("#RRCAutoLockLevelOption option[value='3']").hide();
  704. if (RRCmin >= 5) {
  705. $("#RRCAutoLockLevelOption option[value='4']").hide();
  706. if (RRCmin >= 6) {
  707. $("#RRCAutoLockLevelOption option[value='5']").hide();
  708. if (RRCmin == 10) {
  709. $("#RRCAutoLockLevelOption option[value='6']").hide();
  710. $("#RRCAutoLockLevelOption option[value='0']").show();
  711. }
  712. }
  713. }
  714. }
  715. }
  716. }
  717.  
  718. if (ECmin >= 2) {
  719. $("#ECAutoLockLevelOption option[value='0']").hide();
  720. $("#ECAutoLockLevelOption option[value='1']").hide();
  721. if (ECmin >= 3) {
  722. $("#ECAutoLockLevelOption option[value='2']").hide();
  723. if (ECmin >= 4) {
  724. $("#ECAutoLockLevelOption option[value='3']").hide();
  725. if (ECmin >= 5) {
  726. $("#ECAutoLockLevelOption option[value='4']").hide();
  727. if (ECmin >= 6) {
  728. $("#ECAutoLockLevelOption option[value='5']").hide();
  729. if (ECmin == 10) {
  730. $("#ECAutoLockLevelOption option[value='6']").hide();
  731. $("#ECAutoLockLevelOption option[value='0']").show();
  732. }
  733. }
  734. }
  735. }
  736. }
  737. }
  738. if (max <= 5) {
  739. $("#ECAutoLockLevelOption option[value='6']").hide();
  740. $("#RRCAutoLockLevelOption option[value='6']").hide();
  741. if (max == 4) {
  742. $("#ECAutoLockLevelOption option[value='5']").hide();
  743. $("#RRCAutoLockLevelOption option[value='5']").hide();
  744. if (max == 3) {
  745. $("#ECAutoLockLevelOption option[value='4']").hide();
  746. $("#RRCAutoLockLevelOption option[value='4']").hide();
  747. if (max == 2) {
  748. $("#ECAutoLockLevelOption option[value='3']").hide();
  749. $("#RRCAutoLockLevelOption option[value='3']").hide();
  750. if (max == 1) {
  751. $("#ECAutoLockLevelOption option[value='2']").hide();
  752. $("#RRCAutoLockLevelOption option[value='2']").hide();
  753. }
  754. }
  755. }
  756. }
  757. }
  758. }
  759. $('#countryName')[0].textContent = 'Country setting is for ' + CountryName;
  760. }
  761.  
  762. function forceCountrySetting(){
  763. CountryID = 'forced refresh';
  764. checkCountry();
  765. $('#RRCAutoLockLevelOption')[0].value = RRCmin;
  766. $('#ECAutoLockLevelOption')[0].value = ECmin;
  767. saveSettings();
  768. undoAction();
  769. console.log(SCRIPT_NAME, "forceCountrySetting activated, reset RRC to L" + RRCmin + " and EC to L" + ECmin);
  770. }
  771.  
  772. function undoAction(){
  773. originalLon = 0;
  774. RRCscreenMove();
  775. }
  776.  
  777. function RRCscreenMove(tries = 1) {
  778. let RRClockCount = 0;
  779. let EClockCount = 0;
  780.  
  781. movedLon = W.map.getCenter().lon;
  782. movedZoom = W.map.getZoom();
  783. if ((originalLon != movedLon) || (originalZoom != movedZoom)) {
  784. originalLon = movedLon;
  785. originalZoom = movedZoom;
  786. const extentGeometry = W.map.getOLMap().getExtent().toGeometry();
  787.  
  788. //Changes the background color of the tab.
  789. modelRank = ($('#RRCAutoLockLevelOption')[0].value);
  790. _.each(W.model.railroadCrossings.getObjectArray(), v => {
  791. if (extentGeometry.intersects(v.geometry)) {
  792. var RRClockrank = v.attributes.lockRank + 1;
  793. var RRCunapproved = v.attributes.unapproved;
  794. if ((RRClockrank != modelRank) || (RRCunapproved == true)) {
  795. RRClockCount++
  796. $(`a[href$='${sPanel}']`).css('background-color', '#ffa07a');
  797. $(`a[href$='${sPanel}']`).text('RRC-' + RRClockCount + '/EC-' + EClockCount);
  798. tabColor = 1
  799. }
  800. }
  801. })
  802.  
  803. if (RRClockCount > 0) {
  804. $('#RRCscreenCount')[0].innerHTML = 'There are ' + RRClockCount + ' RRCs needing locked';
  805. }else{
  806. $('#RRCscreenCount')[0].innerHTML = '';
  807. }
  808.  
  809. //Changes the background color of the tab.
  810. modelRank = ($('#ECAutoLockLevelOption')[0].value);
  811. _.each(W.model.cameras.getObjectArray(), v => {
  812. if (extentGeometry.intersects(v.geometry)) {
  813. var EClockrank = v.attributes.lockRank + 1;
  814. var ECunapproved = v.attributes.unapproved;
  815. if ((EClockrank != modelRank) || (ECunapproved == true)) {
  816. EClockCount++
  817. $(`a[href$='${sPanel}']`).css('background-color', '#ffa07a');
  818. $(`a[href$='${sPanel}']`).text('RRC-' + RRClockCount + '/EC-' + EClockCount);
  819. tabColor = 1
  820. }
  821. }
  822. })
  823.  
  824. if (tabColor != 1) {
  825. $(`a[href$='${sPanel}']`).css('background-color', '#e9e9e9');
  826. $(`a[href$='${sPanel}']`).text(TAB_NAME);
  827. }
  828.  
  829. if (EClockCount > 0) {
  830. $('#ECscreenCount')[0].innerHTML = 'There are ' + EClockCount + ' ECs needing locked';
  831. }else{
  832. $('#ECscreenCount')[0].innerHTML = '';
  833. }
  834. tabColor = 0
  835. checkCountry();
  836. }
  837. }
  838.  
  839. function checkCountry(tries = 1){
  840. setTimeout (RRCscreenMove, 3000);
  841. if (W.model.topCountry) {
  842. let newLocationID = W.model.topCountry.id;
  843. if (newLocationID == 235) newLocationID = W.model.topState.id;
  844. if (newLocationID != CountryID) {
  845. console.log(SCRIPT_NAME, 'function RRCscreenMove - Country ID is', CountryID, 'newLocationID =',newLocationID);
  846. CountryID = newLocationID;
  847. loadCountry();
  848. RRCmin = null;
  849. ECmin = null;
  850. loadCountryID();
  851. }
  852. }else if (tries < 2000)
  853. setTimeout(function () {checkCountry(++tries);}, 200);
  854. }
  855.  
  856. function initialCountrySetup(tries = 1) {
  857. if (W.model.topCountry) {
  858. CountryID = W.model.topCountry.id;
  859. CountryName = W.model.topCountry.name;
  860. if (CountryID == 235) {
  861. CountryID = W.model.topState.id;
  862. CountryName = W.model.topState.name;
  863. }
  864. loadCountry();
  865. console.log(SCRIPT_NAME, 'function: initialCountrySetup - Country ID is', CountryID);
  866. }else{
  867. if (tries < 2000)
  868. setTimeout(function () {initialCountrySetup(++tries);}, 1000);
  869. }
  870. }
  871.  
  872. async function bootstrap(tries = 1) {
  873. if (W && W.map && W.model && W.loginManager.user && $ && WazeWrap.Ready ) {
  874. await initialCountrySetup();
  875. await loadBetaUsers();
  876. await RRCAutoLockTab();
  877. await loadCountry();
  878. originalLon = W.map.getCenter().lon;
  879. originalZoom = W.map.getZoom();
  880. WazeWrap.Events.register("selectionchanged", null, setRRCAutoLock);
  881. WazeWrap.Events.register("moveend", null, RRCscreenMove);
  882. WazeWrap.Events.register("afterundoaction", null, undoAction);
  883. WazeWrap.Interface.ShowScriptUpdate(SCRIPT_NAME, VERSION, UPDATE_NOTES);
  884. console.log(SCRIPT_NAME, "loaded");
  885. } else if (tries < 1000)
  886. setTimeout(function () {bootstrap(++tries);}, 200);
  887. }
  888. bootstrap();
  889. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址