// ==UserScript==
// @name WME URCom
// @icon 
// @namespace https://gitlab.com/WMEScripts
// @version 2025.02.23.01
// @description This script is for replying to user requests the goal is to speed up and simplify the process!
// @match *://*.waze.com/*editor*
// @exclude *://*.waze.com/user/editor*
// @author tunisiano187 based on Rick Zabel '2014, maintained by GyllieGyllie
// @license MIT/BSD/X11
// @compatible chrome firefox
// @connect gitlab.com
// @connect *
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @supportURL mailto:incoming+WMEScripts/[email protected]
// @grant GM_xmlhttpRequest
// @grant unsafeWindow
// ==/UserScript==
/* global $, GM_info, GM_xmlhttpRequest, WazeWrap, unsafeWindow, getWmeSdk */
const ScriptName = GM_info.script.name;
const URCommentVersion = GM_info.script.version;//var URCommentVersion = "1.5.4"; //branched from 0.9.3
const URCommentUpdateMessage = "yes"; // yes alert the user, no has a silent update.
/* Changelog
1.9.2 - Sizing Signature field
1.9.3 - Details about the Signature and the refresh need
1.9.4 - Relocate Signature settings
1.9.5 - Auto email errors
1.9.6 - Pictures in pills
1.9.7 - Pills corrections
1.9.8 - Title and pills updates
1.9.9 - Credits et corrections
1.9.10 - Bug closure text correction
1.9.11 - Design and some corrections
1.9.12 - icons before comments in the comments tab
1.9.13 - Don't complete comments, disable textarea and hide send button id the UR is locked (Out of editing area) or closed
1.9.14 - Adding Francais Belgique à la liste
2018.07.18.01 - New versionning numbers to ease the checks for last update
2018.08.04.01 - Adding the NL version
2018.08.08.01 - Signature even if no predefined answer found
2018.08.12.01 - Local version update
2018.08.15.01 - Adding the ability to add Titles in the comments part
2018.08.15.02 - Translator's name
2018.08.15.03 - Removing the open database option for now
2018.08.15.04 - Bad comment break the script
2018.08.15.05 - Title spaces
2018.08.15.06 - Test text moved
2018.08.16.01-03 - Design changes
2018.08.16.04 - Bug correction
2018.08.17.01 - Handling the ending dot on comments title that breaks the autocomplete
2018.08.18.01-03 - Change URCom logo + Design
2018.08.21.01
2018.09.02.01 - undefined mask
2019.05.19.01 - Adapt timeout to avoid useless messages
2019.08.15.01 - Adapt to the new WME version
2019.11.24.01 - Yellow Background if message from the user
2020.01.18.01 - Support of unknown reports
2020.01.18.02 - remove alert
2020.01.13.01 - Solving signature's return problem
2020.02.15.01 - Error version number
2020.04.12.01 - Add user's message in the (..) answer
2020.08.07.01 - Remove the line that hide the textarea
2023.06.04.01 - New design based on current WME styling
2023.06.16.01 - Major code overhaul & bug fixing
2023.06.16.02 - Fix migration logic being broken
2023.06.18.01 - Filter cleanup, small fixes, new features
2023.07.03.01 - Add feature to create custom responses
2023.08.03.01 - Fixes after new WME Update
2023.08.03.02 - Fixes after WME update rollback
2023.08.15.01 - Change how custom list data is stored to fix all browser support
2023.08.16.01 - Small fix to changes from yesterday
2023.08.21.01 - Small fix in migration logic being broken if you use script first time
2024.03.22.01 - Fix auto response no longer working
2024.05.29.01 - Fix some issues after WME update
2025.02.12.01 - Major overhaul, no more separate language scripts & prep for multi language
>>>>>>> URComments(dev).user.js
*/
let URCommentVersionUpdateNotes = "UR Comments has been updated to " + URCommentVersion + "<br />";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "In this update we've done a major update to the script. Since a few months it's no longer possible to load custom translations & repsonses from separate TM Scripts. In this update we have fully redone the system and all previously maintained translations/responses are now default integrated into the main script.";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br /><br />" + "<b>Things that changed:</b>";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- No more need for additional scripts for other languages";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- Filters are now fully removed, as almost all of it is supported by WME filtering";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- Script language & responses are no longer linked, you can use the script in a different language than the responses";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br /><br />" + "<b>Steps to take to use the script again:</b>";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- Remove any old additional script you had installed for your custom language/responses";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- Navigate to the settings of the script and select in which language the script should operate";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- On the text tab, select which default list you want to use";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- That's it, you should now be able to use URCom like before";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br /><br />" + "<b>Any issues/feedback?</b>";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br />" + "- If you found a bug, want another language to be supported or any other suggestions please report them on <a href='https://gitlab.com/WMEScripts/URComments-French/-/issues' target='_blank'>our Gitlab</a>";
URCommentVersionUpdateNotes = URCommentVersionUpdateNotes + "<br /><br />";
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////
////// Define items that need to be at the root level so they can be used inside functions
//////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
let wmeSDK;
let migrate = false;
const options = loadOptions();
// If we need to migrate, do the migration now
if (migrate) {
migrateOptions();
}
// Now validate the options are ok
validateOptions(options);
let DB;
const SupportedLanguages = [
"English",
"Nederlands",
"Français",
];
const defaultLists = new Map([
["English", "URComDefault"],
["Nederlands", "Nederlands"],
["Francais Belgique", "Francais_Belgique"],
["Luxembourg", "Luxembourg"]
])
let CustomLists = [];
let CustomListText = [];
let InitReady = false;
let ShowTextEditFields = false;
let ShowTextOptionButtons = true;
let TextEditIndex = undefined;
let DeleteTextMode = false;
// Array that holds the comments
let URCommentsArray;
// Waze swaps out the close button on the UR window after you click send. we use this to grab the close and compare to the new one
let CloseButtonHolder = "";
// Since we are scanning for open UR I need to keep track of the current urID so the comments can be overridden
let UrCommentLasturID = "";
// This is used to hold the info about the previous tab, before we auto switched tabs
let PreviousTab = null;
let PreviousTabPosition = null;
// Return to zoom instead of zoom 0
let ReturnToCurrentZoom = "";
let translation;
let knownLanguages;
//var UrCommentsIcon = '';
const UrCommentsIcon = '';
const UrCommentsTextPic = '<img alt="" style="margin-bottom: 1px; height: 12px;" src="\" />'; // Picture of the number of comments
const UrCommentsTimePic = '<img alt="" style=\"margin-bottom: 1px; height: 12px;" src=\"\" />'; // Delay picture
const URC_img_waze_editor = '<img alt="" style="height: 15px; margin-right: 5px;" src="" />';
const MagnifyingIcon = '<img alt="" style="margin-bottom: 1px; height: 12px;" src="" />';
const RefreshIcon = '<img alt="" style="margin-bottom: 1px; height: 12px;" src="\n" />';
// Keep references to our html elements
let scriptContentPane;
let nConfirm;
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////
////// bootstrap
//////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
async function URComments_init() {
if (!WazeWrap.Ready) {
setTimeout(URComments_init, 250);
return;
}
await initDatabase();
console.log("[URCom] Loading data");
await loadTranslation();
await loadResponses();
const dsRequest = DB
.transaction("data-store")
.objectStore("data-store")
.get("languages");
dsRequest.onsuccess = (event) => {
if (event.target.result?.responses && event.target.result.responses.length > 0) {
const data = event.target.result.responses[0];
console.log(data);
knownLanguages = data.languages;
} else {
knownLanguages = [];
}
};
while (!dataReady()) {
await sleep(100);
}
console.log("[URCom] Data loaded");
const request = DB
.transaction("custom_lists")
.objectStore("custom_lists")
.getAll();
request.onsuccess = (event) => {
if (event.target.result) {
const results = [...event.target.result];
results.sort((a, b) => {
return a.order - b.order
});
CustomLists = [];
defaultLists.forEach((value, key) => {
CustomLists.push("* " + key);
});
results.forEach(list => {
CustomLists.push(list.name);
})
CustomLists.push(translate("list.new"));
loadCustomText(() => {
InitReady = true;
});
}
};
startCode();
displayChangelog();
}
function loadCustomText(callback) {
if (options.CustomTextList) {
if (defaultLists.has(options.CustomTextList)) {
callback();
return;
}
CustomListText = [];
// Verify it still exists
if (CustomLists.indexOf(options.CustomTextList) === -1) {
alert(translate("prompts.list-gone"));
options.CustomTextList = "English";
saveOptions(options);
loadResponses().then(callback);
} else {
console.log("[URCom] Loading custom text: " + options.CustomTextList);
const request = DB
.transaction("custom_lists")
.objectStore("custom_lists")
.get(options.CustomTextList);
request.onsuccess = (event) => {
if (event.target.result) {
const results = [...event.target.result.responses];
for (let i = 0; i < results.length; i++) {
const result = results[i];
CustomListText.push({
sorting: result.sorting,
name: result.name,
text: result.text,
status: result.status,
});
}
CustomListText.sort((a, b) => {
return a.sorting - b.sorting
});
}
callback();
};
}
} else {
callback();
}
}
function displayChangelog() {
if (!WazeWrap.Interface) {
setTimeout(displayChangelog, 1000);
return;
}
if (URCommentUpdateMessage === "yes") {
// Alert the user in URComment version updates
if (localStorage.getItem('URComVersion') === URCommentVersion) {
console.log("[URCom] Version - " + URCommentVersion);
} else {
WazeWrap.Interface.ShowScriptUpdate(ScriptName, URCommentVersion, URCommentVersionUpdateNotes + "<br />", "https://gitlab.com/WMEScripts/URComments-French");
const updateName = "#wmeurcom" + URCommentVersion.replaceAll(".", "");
$(updateName + " .WWSUFooter a").text("Gitlab")
localStorage.setItem('URComVersion', URCommentVersion);
}
}
}
async function refreshUI() {
await loadTranslation();
await loadResponses();
// Clear the html from the tabs
$("#sidepanel-Comments").html('');
// Reload the content
init();
}
async function loadTranslation() {
// Language isn't know so fall back to English
if (!SupportedLanguages.includes(options.Language)) {
options.Language = "English";
saveOptions(options);
}
console.log("[URCom] Selected language: " + options.Language);
let lang = "en";
switch (options.Language) {
case "Nederlands": {
lang = "nl";
break;
}
case "Français": {
lang = "fr";
break;
}
}
await makeHTTPRequest('GET', "https://gitlab.com/WMEScripts/URComments-French/-/raw/master/i18n/" + lang + ".json")
.then((response) => {
translation = response;
console.log("[URCom] " + translate("example"));
})
.catch((error) => {
console.log(error);
})
console.log("[URCom] Language loaded!");
}
async function loadResponses() {
// We are using a custom list
if (!defaultLists.has(options.CustomTextList)) {
return;
}
console.log("[URCom] Selected default list: " + options.CustomTextList);
const fileName = defaultLists.get(options.CustomTextList)
await makeHTTPRequest('GET', "https://gitlab.com/WMEScripts/URComments-French/-/raw/master/presets/" + fileName + ".json")
.then((response) => {
URCommentsArray = response;
console.log("[URCom] Loaded default responses.");
})
.catch((error) => {
console.log(error);
})
console.log("[URCom] Default list loaded!");
}
function dataReady() {
return !!knownLanguages;
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////
////// init
//////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
function applyCss() {
// CSS
// Expand the UR textarea so we can verfiy what comment we clicked on. special thanks to SeekingSerenity
let g = '.ur-comment-list .comment-list { bottom: 200px !important; } .ur-comment-list .new-comment-form textarea { height: 140px !important; } .ur-comment-list .new-comment-form { height: 200px !important; }';
const wazeHeight = $('.view-area').height();
if (wazeHeight > 500) {
// beta-editor comment textarea
g = g + ' .new-comment-form .new-comment-text { height: 140px !important; }';
}
// css for making pane fit
g = g + '#sidepanel-Comments { width: 100% !important; }';
// css for items in my tab that are in a label
g = g + '#sidepanel-Comments label { cursor:pointer; margin:0px 0px 0px; vertical-align: middle;font-size: 10px;}';
// css for checkboxes
g = g + ' #sidepanel-Comments .URCommentsCheckbox { text-decoration:none; cursor:pointer; color: #000000; margin:0px 0px 0px; vertical-align: middle; font-size: 12px;}';
// css for our comments,
g = g + ' #sidepanel-Comments .URComments { text-decoration:none; cursor:pointer; color: #000000; font-size: 12px;}'; // margin-top: 5px;
// css for our presets,
g = g + ' #sidepanel-Comments .URCommentsPresets { text-decoration:none; cursor:pointer; color: #000000; font-size: 10px;}';
// css for our nav tabs,
g = g + ' #comments-tab22 ul { font-size: 12px; padding: 0px;}';
// css for our nav tabs links,
g = g + ' #comments-tab22 a { padding: 3px !important ; margin-right: 0px !important;}';
// keep the padding on our nav tabs
g = g + ' #comments-tab22 nav-tabs { padding: 0px !important;}';
// css for non selected UR opacity
g = g + " .olMap.problem-selected .map-problem:not(.selected) { opacity: .5 !important;}";
// css to fix the beta editors UR window
g = g + ' .problem-edit .section .title { line-height: 15px !important; }';
g = g + ' .problem-edit .header { line-height: 15px !important; padding: 0px 15px!important; }';
g = g + ' .problem-edit .section .content { padding: 5px !important;}';
// css to undo some of the changes from maximizer
if ($("#sidebar").width() < 300) {
//g = g + '#sidebar { max-width: 290px !important;}';
g = g + ' .show-sidebar .row-fluid .fluid-fixed {margin-left: 290px !important;}';
}
// move description up a div
g = g + ' .problem-edit .description { background-color: white; overflow-x: hidden; overflow-y: auto; max-height: 80px; border-bottom: 2px solid #e9e9e9;}';
// Make some spacing between checkboxes in settings
g = g + ' .urcom-option { margin-top: 10px; }';
// Our button style
g = g + ' .urcom-button { margin: 5px 0; border: 1px solid rgba(27, 31, 35, 0.15); border-radius: 6px; box-shadow: rgba(27, 31, 35, 0.04) 0 1px 0, rgba(255, 255, 255, 0.25) 0 1px 0 inset; background-color: #FAFBFC; cursor: pointer; font-size: 14px; font-weight: 500; line-height: 20px; list-style: none; padding: 3px 10px; position: relative; transition: background-color 0.2s cubic-bezier(0.3, 0, 0.5, 1); }'
// Append our css to the head
$("head").append($('<style id="URCOM-CSS" type="text/css">' + g + '</style>'));
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////
////
//// Option Methods
////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
function addCommentTexts(container) {
if (!ShowTextEditFields) {
// Custom Text
addDropdownSettings(container, "list.name.tooltip", "list.name.title", 'CustomTextList', CustomLists, changeCustomTextList);
const header = $('<h5 style="margin-top: 20px;">' + translate("list.responses") + '</h5>');
container.append(header);
}
// Start generating the comment list and mouse click handlers
if (options.CustomTextList && !defaultLists.has(options.CustomTextList)) {
if (ShowTextEditFields) {
let headerText;
if (TextEditIndex !== undefined && TextEditIndex !== -1) {
headerText = translate("list.edit");
} else {
headerText = translate("list.response.title.title");
}
// Add options to setup custom text
const header = $('<h5 style="margin-top: 20px;">' + headerText + '</h5>');
container.append(header);
const titleInput = $('<wz-text-input id="customTextTitle"></wz-text-input>');
const titleWrapper = $('<div class="urcom-option"><span Title="' + translate("list.response.title.tooltip") + '">' + translate("list.response.title.title") + '</span></div>').append(titleInput);
container.append(titleWrapper);
const textInput = $('<wz-textarea id="customTextText" style="height: 130px; width: 270px;"></wz-textarea>');
const textWrapper = $('<div class="urcom-option"><span Title="' + translate("list.response.response.tooltip") + '">' + translate("list.response.response.title") + '</span></div>').append(textInput);
container.append(textWrapper);
const selectWrapper = $('<div class="urcom-option" title="' + translate("list.response.status.tooltip") + '"></div>');
selectWrapper.append('<span>' + translate("list.response.status.title") + '</span>');
selectWrapper.append('<br />');
const select = $('<select name="customTextStatus" id="customTextStatus" style="width: 100%;"></select>');
select.append('<option value="open">open</option>');
select.append('<option value="open">closed</option>');
select.append('<option value="open">notidentified</option>');
selectWrapper.append(select);
container.append(selectWrapper);
const saveButton = $('<wz-button size="sm" style="width: 100%; margin-top: 10px;">' + translate("save") + '</wz-button>');
saveButton.on('click', saveCustomText);
container.append(saveButton);
const cancelButton = $('<wz-button size="sm" style="width: 100%; margin-top: 10px;">' + translate("cancel") + '</wz-button>');
cancelButton.on('click', cancelTextEdit);
container.append(cancelButton);
} else {
// Go over the array and generate comment divs
for (let i = 0; i < CustomListText.length; i++) {
const text = CustomListText[i];
let reply = text.text;
addComments(container, i, {
title: text.name,
status: text.status,
response: reply
});
}
if (ShowTextOptionButtons) {
const options = $('<h5 style="margin-top: 20px;">' + translate("list.options") + '</h5>');
container.append(options);
const addItem = $('<div style="width: 100%; margin-top: 20px; color: green" class="urcom-button">' + translate("list.add") + '</div>');
addItem.on('click', addNewItem);
container.append(addItem);
const editButton = $('<div style="width: 100%; margin-top: 10px; color: orange" class="urcom-button">' + translate("list.edit") + '</div>');
editButton.on('click', editItem);
container.append(editButton);
const editList = $('<div style="width: 100%; margin-top: 10px; color: orange" class="urcom-button">' + translate("list.edit-name") + '</div>');
editList.on('click', renameList);
container.append(editList);
const dangerZone = $('<h5 style="margin-top: 20px; color:darkred">' + translate("list.danger") + '</h5>');
container.append(dangerZone);
let text;
if (DeleteTextMode) {
text = translate("list.cancel-delete");
} else {
text = translate("list.delete-response");
}
const deleteTextButton = $('<div style="width: 100%; margin-top: 10px; color: orangered" class="urcom-button">' + text + '</div>');
deleteTextButton.on('click', deleteTextItem);
container.append(deleteTextButton);
const deleteButton = $('<div style="width: 100%; margin-top: 10px; color: orangered" class="urcom-button">' + translate("list.delete-list") + '</div>');
deleteButton.on('click', deleteCustomList);
container.append(deleteButton);
}
}
} else {
// CurrentIndex is used to keep count of total arrays pairs which is used on the div's id tags
let CurrentIndex = 1;
// Go over the array and generate comment divs
for (let i = 0; i < URCommentsArray.length; i++) {
const comment = URCommentsArray[i];
addComments(container, CurrentIndex, comment)
// inc the CurrentIndex
CurrentIndex++;
}
}
// Add 2 br to the end of the list for lower resolution monitors
container.append('<br /><br />');
}
function addComments(container, CurrentIndex, commentObject) {
const title = commentObject.title ?? (commentObject.custom === 'new-line' ? '<br />' : '');
const urStatus = commentObject.status?.toLowerCase() ?? '';
const text = commentObject.response ?? '';
// Setup the comment color var
let textColor;
if (urStatus === "open") {
// Black
textColor = "#000000";
} else if (urStatus === "solved") {
// Green
textColor = "#008F00";
} else if (urStatus === "notidentified") {
// Orange
textColor = "#E68A00";
} else {
// Red - not defined and that is a problem
textColor = "#CC0000";
}
// Escaping titles and comments with escapeHtml(comment) so we can display items with special char as html;
let comment = text ? escapeHtml(text) : '';
let addClicks = false;
// normal comment link
let commentWrapper;
let clickText;
let doubleClickText;
if (!commentObject.custom) {
// This is a real button
commentWrapper = $('<div id="URComments-comment' + CurrentIndex + '" style="margin: 5px 0; border: 1px solid rgba(27, 31, 35, 0.15); border-radius: 6px; box-shadow: rgba(27, 31, 35, 0.04) 0 1px 0, rgba(255, 255, 255, 0.25) 0 1px 0 inset; background-color: #FAFBFC; cursor: pointer; font-size: 14px; font-weight: 500; line-height: 20px; list-style: none; padding: 3px 10px; position: relative; transition: background-color 0.2s cubic-bezier(0.3, 0, 0.5, 1);">');
if (DeleteTextMode) {
commentWrapper.css("border-color", "darkred");
commentWrapper.css("background-color", "orangered");
} else if (TextEditIndex === -1) {
commentWrapper.css("background-color", "yellow");
}
addClicks = true;
} else {
// These are titles/empty lines
commentWrapper = $('<div id="URComments-comment' + CurrentIndex + '"></div>')
}
if (commentObject.custom === "title") {
const titleHeader = $('<h5 style="color:' + textColor + '; text-decoration: underline; margin-top:15px;" title="title: ' + title + ' Action: ' + urStatus + '; comment: ' + comment + ' ">' + title + '</h5>');
commentWrapper.append(titleHeader);
} else {
clickText = $('<a class="URComments" style="color:' + textColor + '" title="title: ' + title + ' Action: ' + urStatus + '; comment: ' + comment + ' ">' + title + '</a>');
commentWrapper.append(clickText);
if (commentObject.custom !== "new-line") {
if (commentObject.urType === -2 && options.DBLClk7DCAutoSend || options.DBLClkAll) {
doubleClickText = $('<a id="URComments-commentDBLCLK' + CurrentIndex + '" class="URComments" style="color:' + textColor + '" title="' + translate("options.double-click.tooltip", { title }) + '"> ' + translate("options.double-click.title") + '</a>');
commentWrapper.append('<br />').append(doubleClickText);
}
}
}
// Add comment to list
container.append(commentWrapper);
if (addClicks) {
if (DeleteTextMode) {
commentWrapper.on('click', executeDeleteTextItem(CurrentIndex));
} else if (TextEditIndex === -1) {
commentWrapper.on('click', selectEditItem(CurrentIndex));
} else if (!TextEditIndex) {
// Set urID to zero so we don't freak out the functions expecting a UR ID
let urID = 0;
// Create the click function for each comment
commentWrapper.on('click', autoZoomIN(title, text, urStatus, urID));
// Create the double click function for each comment
if (doubleClickText) {
// Use this to click send automatically
doubleClickText.on('dblclick', autoZoomIN(title, text, urStatus, urID, "AutoSendComment"));
}
}
}
}
function addOptions(container) {
// Language select
addDropdownSettings(container, "", "settings.language.title", 'Language', SupportedLanguages, changeLanguage);
const settings = $('<h4 style="margin-top: 20px;">' + translate("tabs.settings") + '</h4>');
container.append(settings);
// Auto Reply
addBooleanSettings(container, "settings.auto-comment.tooltip", "settings.auto-comment.title", 'AutoSetNewComment');
// Automated Reminder / Closure
addBooleanSettings(container, "settings.auto-reminder.tooltip", "settings.auto-reminder.title", 'UrCommentAutoSet4dayComment');
// Zoom in on open
addBooleanSettings(container, "settings.auto-zoom.tooltip", "settings.auto-zoom.title", 'NewZoomIn');
// Auto Change State
addBooleanSettingsCallback(container, "settings.auto-status.tooltip", "settings.auto-status.title", 'AutoClickURStatus', (event) => {
toggleBoolean(event);
if (!options.AutoClickURStatus && options.SaveAfterComment) {
options.SaveAfterComment = false;
}
if (options.DBLClk7DCAutoSend || options.DBLClkAll) {
alert(translate("prompts.require-auto-click")); //"URComments to use the double click links you must have the autoset UR status option enabled"
options.AutoClickURStatus = true;
}
saveOptions(options);
refreshUI();
});
// Auto Save On Close
addBooleanSettingsCallback(container, "settings.auto-save.tooltip", "settings.auto-save.title", 'SaveAfterComment', (event) => {
toggleBoolean(event);
if (options.SaveAfterComment && !options.AutoClickURStatus) {
// This is required!
options.AutoClickURStatus = true;
saveOptions(options);
refreshUI();
}
});
// Auto Close UR Screen
addBooleanSettings(container, "settings.auto-close.tooltip", "settings.auto-close.title", 'UrCommentAutoCloseComment');
// Zoom after close
addBooleanSettings(container, "settings.auto-zoom-close.tooltip", "settings.auto-zoom-close.title", 'ZoomOutAfterComment');
// Switch to comment tab auto
addBooleanSettings(container, "settings.auto-tab.tooltip", "settings.auto-tab.title", 'AutoSwitchToURCommentsTab');
// Double clicking the 7 day close comment will auto send the 7 day close comment
addBooleanSettingsCallback(container, "settings.double-click-close.tooltip", "settings.double-click-close.title", 'DBLClk7DCAutoSend', (event) => {
toggleBoolean(event);
if (options.DBLClk7DCAutoSend) {
// This is required!
options.AutoClickURStatus = true;
saveOptions(options);
refreshUI();
}
});
// Double clicking comments will auto send comments
addBooleanSettingsCallback(container, "settings.double-click-send.tooltip", "settings.double-click-send.title", 'DBLClkAll', (event) => {
toggleBoolean(event);
if (options.DBLClkAll) {
// This is required!
options.AutoClickURStatus = true;
saveOptions(options);
refreshUI();
}
});
// Disable Next Button
addBooleanSettings(container, "settings.disable-buttons.tooltip", "settings.disable-buttons.title", 'UrCommentDisableURDoneBtn');
// Enable or disable ur pill counts
//addBooleanSettingsCallback(container, "settings.pill.tooltip", "settings.pill.title", 'URCommentsPillEnabled', (event) => {
// toggleBoolean(event);
// FilterURs();
//});
// Hide the AA description
addBooleanSettings(container, "settings.hide-aa.tooltip", "settings.hide-aa.title", 'HideAADescription');
// Days used to filter UR (reminder days / close days)
addTextNumberSettings(container, "", "settings.reminder-days.title", "ReminderDays");
addTextNumberSettings(container, "", "settings.close-days.title", "CloseDays");
// Greeting
addTextAreaSettings(container, "settings.greeting.tooltip", "settings.greeting.title", 'Greeting');
// Signature
addTextAreaSettings(container, "settings.signature.tooltip", "settings.signature.title", 'Signature');
const contactLabel = $('<div class="urcom-option"><span Title="' + translate("settings.credits.tooltip") + '" style="padding-bottom: 10px; line-height: 15px; font-weight: bold;"><br>' + translate("settings.credits.title") + ' :</span></div>');
container.append(contactLabel);
const report = $('<div class="urcom-option"><span Title="' + translate("settings.report.tooltip") + '" style="line-height: 15px;"><a href="https://gitlab.com/WMEScripts/URComments-French/-/issues" target="_blank">' + translate("settings.report.title") + '</a></span></div>'); // Maintainer
container.append(report);
const maintainer = $('<div class="urcom-option"><span Title="' + translate("settings.maintainer.tooltip") + '" style="line-height: 15px;">' + URC_img_waze_editor + translate("settings.maintainer.title") + ': @GyllieGyllie</span></div>'); // Maintainer
container.append(maintainer);
const founder = $('<div class="urcom-option"><span Title="' + translate("settings.founder.tooltip") + '" style="line-height: 15px;">' + URC_img_waze_editor + translate("settings.founder.title") + ': @tunisiano187</span></div>'); // Dev
container.append(founder);
}
// Save what comment list is selected
function changeLanguage(event) {
const selected = event.target.value;
if (selected !== "") {
options.Language = selected;
saveOptions(options);
refreshUI();
}
}
// Save what custom comment list is selected
function changeCustomTextList(event) {
const selected = event.target.value;
if (selected === translate("list.new")) {
addCustomList();
} else if (selected !== "") {
if (selected.startsWith("* ")) {
options.CustomTextList = selected.replace("* ", "");
saveOptions(options);
loadResponses().then(refreshUI);
} else {
options.CustomTextList = selected;
saveOptions(options);
loadCustomText(() => {
refreshUI();
})
}
}
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////
////
//// Option Logic
////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
function getDefaultOptions() {
return {
AutoSetNewComment: true,
UrCommentAutoSet4dayComment: true,
NewZoomIn: true,
AutoClickURStatus: true,
SaveAfterComment: true,
UrCommentAutoCloseComment: true,
ZoomOutAfterComment: true,
AutoSwitchToURCommentsTab: true,
DBLClk7DCAutoSend: false,
DBLClkAll: false,
UrCommentDisableURDoneBtn: true,
HideAADescription: true,
Signature: "",
Greeting: "",
URComShowPS: false,
URCommentsPS: "",
BoilerPlateCreators: "URComDefault",
URCommentsFilterEnabled: true,
URCommentsPillEnabled: true,
URCommentsHideNotMyUR: true,
URCommentsShowPastClose: true,
URCommentsHideInbetween: true,
URCommentsHideReminderNeeded: false,
URCommentsHideReplies: false,
URCommentsHideCloseNeeded: false,
URCommentsHideInital: false,
URCommentsHideWithoutDescript: false,
URCommentsHideWithDescript: false,
URCommentsHideClosed: true,
ReminderDays: 4,
CloseDays: 3,
CustomTextList: '',
Language: "English",
}
}
function loadOptions() {
let text = localStorage.getItem("URCom-Options");
let options;
if (text) {
options = JSON.parse(text);
} else {
options = getDefaultOptions();
migrate = true;
}
return options;
}
function validateOptions(options) {
const defaultOptions = getDefaultOptions();
// Add missing options
for (let key in defaultOptions) {
if (!(key in options)) {
options[key] = defaultOptions[key]
}
}
if (options.DBLClkAll) {
options.AutoClickURStatus = true;
}
if (options.DBLClk7DCAutoSend) {
options.AutoClickURStatus = true;
}
if (isNaN(options.ReminderDays) || options.ReminderDays < 0) {
options.ReminderDays = 0;
}
if (isNaN(options.CloseDays) || options.CloseDays < 0) {
options.CloseDays = 0;
}
if (options.CustomTextList === "None") {
options.CustomTextList = "Default";
}
}
function saveOptions(options) {
const optionsJson = JSON.stringify(options);
localStorage.setItem("URCom-Options", optionsJson);
}
function toggleBoolean(event) {
options[event.target.id] = event.target.checked;
saveOptions(options);
}
function changeText(event) {
options[event.target.id] = event.target.value;
saveOptions(options);
}
function migrateOptions() {
migrateTextOption('BoilerPlateCreators', 'BoilerPlateCreators');
migrateBooleanOptions('AutoSetNewComment', 'AutoSetNewComment');
migrateBooleanOptions('UrCommentAutoSet4dayComment', 'UrCommentAutoSet4dayComment');
migrateBooleanOptions('UrCommentAutoCloseComment', 'UrCommentAutoCloseComment');
migrateBooleanOptions('NewZoomIn', 'NewZoomIn');
migrateBooleanOptions('AutoClickURStatus', 'AutoClickURStatus');
migrateBooleanOptions('SaveAfterComment', 'SaveAfterComment');
migrateBooleanOptions('ZoomOutAfterComment', 'ZoomOutAfterComment');
migrateBooleanOptions('AutoSwitchToURCommentsTab', 'AutoSwitchToURCommentsTab');
migrateBooleanOptions('DBLClk7DCAutoSend', 'DBLClk7DCAutoSend');
migrateBooleanOptions('DBLClkAll', 'DBLClkAll');
migrateBooleanOptions('UrCommentDisableURDoneBtn', 'UrCommentDisableURDoneBtn');
migrateTextOption('Signature', 'Signature');
migrateBooleanOptions('URComShowPS', 'URComShowPS');
migrateTextOption('URCommentsPS', 'URCommentsPS');
migrateBooleanOptions('URCommentsFilterEnabled', 'URCommentsFilterEnabled');
migrateBooleanOptions('URCommentsPillEnabled', 'URCommentsPillEnabled');
migrateBooleanOptions('URCommentsHideNotMyUR', 'URCommentsHideNotMyUR');
migrateBooleanOptions('URCommentsShowPastClose', 'URCommentsShowPastClose');
migrateBooleanOptions('URCommentsHideInbetween', 'URCommentsHideInbetween');
migrateBooleanOptions('URCommentsHideReminderNeeded', 'URCommentsHideReminderNeeded');
migrateBooleanOptions('URCommentsHideReplies', 'URCommentsHideReplies');
migrateBooleanOptions('URCommentsHideCloseNeeded', 'URCommentsHideCloseNeeded');
migrateBooleanOptions('URCommentsHideInital', 'URCommentsHideInital');
migrateBooleanOptions('URCommentsHideWithoutDescript', 'URCommentsHideWithoutDescript');
migrateBooleanOptions('URCommentsHideWithDescript', 'URCommentsHideWithDescript');
migrateBooleanOptions('URCommentsHideClosed', 'URCommentsHideClosed');
migrateNumericOption('ReminderDays', 'ReminderDays');
migrateNumericOption('CloseDays', 'CloseDays');
saveOptions(options);
}
function migrateBooleanOptions(oldOption, newOption) {
const value = localStorage.getItem(oldOption);
if (value !== null && value !== undefined) {
options[newOption] = value === "yes";
}
}
function migrateNumericOption(oldOption, newOption) {
const value = localStorage.getItem(oldOption);
if (value !== null && value !== undefined) {
const number = Number(value);
if (!isNaN(number)) {
options[newOption] = number;
}
}
}
function migrateTextOption(oldOption, newOption) {
const value = localStorage.getItem(oldOption);
if (value !== null && value !== undefined) {
options[newOption] = value;
}
}
function addDropdownSettings(container, title, label, name, values, callback) {
let currentValue = options[name];
const wrapper = $('<div class="urcom-option" title="' + translate(title) + '"></div>');
wrapper.append('<span>' + translate(label) + '</span>');
wrapper.append('<br />');
const select = $('<select name="' + name + '" id="' + name + '" style="width: 100%;"></select>');
for (let i = 0; i < values.length; i++) {
select.append('<option value="' + values[i] + '" ' + (currentValue === values[i].replace("* ", "") ? 'selected="selected"' : '') + '>' + values[i] + '</option>');
}
// Create call back for the select
select.on('change', callback);
wrapper.append(select);
container.append(wrapper);
}
function addBooleanSettings(container, title, label, name) {
addBooleanSettingsCallback(container, title, label, name, toggleBoolean);
}
function addBooleanSettingsCallback(container, title, label, name, clickHandler) {
const currentValue = options[name];
const checkbox = $('<wz-checkbox id="' + name + '" Title="' + translate(title) + '" name="types" disabled="false" checked="' + currentValue + '">' + translate(label) + '</wz-checkbox>');
const optionHtml = $('<div class="urcom-option"></div>').append(checkbox);
container.append(optionHtml);
checkbox.on('click', clickHandler);
}
function addTextNumberSettings(container, title, label, name) {
const currentValue = options[name];
const textInput = $('<wz-text-input type="number" min="0" max="999" id="' + name + '" value="' + currentValue + '"></wz-text-input>');
const optionHtml = $('<div class="urcom-option"><span Title="' + translate(title) + '">' + translate(label) + '</span></div>').append(textInput);
container.append(optionHtml);
textInput.on('change', changeText);
}
function addTextAreaSettings(container, title, label, name) {
const currentValue = options[name];
const textArea = $('<wz-textarea id="' + name + '" style="height: 130px; width: 270px;" value="' + currentValue + '"></wz-textarea>');
const optionHtml = $('<div class="urcom-option"><span Title="' + translate(title) + '">' + translate(label) + '</span></div>').append(textArea);
container.append(optionHtml);
textArea.on('change', changeText);
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////
////
//// UR Comment functions
////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
function addCustomList() {
let name = prompt("Enter the name for your list");
if (!name) {
refreshUI();
return;
}
// Make sure it doesn't exist
if (CustomLists.indexOf(name) >= 0) {
alert(translate("prompts.list-name-exists"));
return;
}
// Insert into the DB
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
objectStore.add({
name: name,
order: CustomLists.length - 2,
responses: []
});
CustomLists.splice(CustomLists.length - 1, 1);
CustomLists.push(name);
CustomLists.push(translate("list.new"));
options.CustomTextList = name;
saveOptions(options);
CustomListText = [];
refreshUI();
alert(translate("prompts.list-created"));
}
function renameList() {
let name = prompt(translate("prompts.list-enter-name"), options.CustomTextList);
if (!name) {
return;
}
// Make sure it doesn't exist
if (CustomLists.indexOf(name) >= 0) {
alert(translate("prompts.list-name-exists"));
return;
}
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
const request = objectStore.get(options.CustomTextList);
request.onerror = (event) => {
console.log(event);
alert(translate("prompts.list-update-failed"));
};
request.onsuccess = (event) => {
const data = event.target.result;
data.name = name;
const requestUpdate = objectStore.add(data);
requestUpdate.onerror = (event) => {
console.log(event);
alert(translate("prompts.list-update-failed"));
};
requestUpdate.onsuccess = (event) => {
objectStore.delete(options.CustomTextList);
const index = CustomLists.indexOf(options.CustomTextList);
CustomLists[index] = name;
options.CustomTextList = name;
saveOptions(options);
refreshUI();
}
}
}
function cancelTextEdit() {
ShowTextEditFields = false;
TextEditIndex = undefined;
refreshUI();
}
function saveCustomText() {
const titleInput = $("#customTextTitle")[0];
const textInput = $("#customTextText")[0];
const statusSelect = $("#customTextStatus")[0];
const title = titleInput.value;
const text = textInput.value;
const statusIndex = statusSelect.selectedIndex;
let status;
switch (statusIndex) {
case 0:
default:
status = "open";
break;
case 1:
status = "closed";
break;
case 2:
status = "notidentified";
break;
}
if (!title || !text || !status) {
alert(translate("prompts.missing-data"));
return;
}
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
const request = objectStore.get(options.CustomTextList);
request.onerror = (event) => {
alert(translate("prompts.response-failed"));
};
request.onsuccess = (event) => {
const data = event.target.result;
if (TextEditIndex !== undefined && TextEditIndex !== -1) {
// Update text
data.responses.forEach(response => {
if (response.sorting === TextEditIndex) {
response.name = title;
response.text = text;
response.status = status;
}
})
} else {
data.responses.push({
sorting: CustomListText.length,
name: title,
text: text,
status: status,
})
}
const requestUpdate = objectStore.put(data);
requestUpdate.onerror = (event) => {
alert(translate("prompts.response-failed"));
};
requestUpdate.onsuccess = (event) => {
ShowTextEditFields = false;
TextEditIndex = undefined;
loadCustomText(() => {
refreshUI();
});
}
}
}
function deleteCustomList() {
const result = confirm(translate("prompts.confirm-delete-list"));
if (!result) {
return;
}
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
const request = objectStore.delete(options.CustomTextList);
request.onerror = (event) => {
alert(translate("prompts.delete-list-failed"));
};
request.onsuccess = (event) => {
// Fall back to default list
CustomLists = CustomLists.filter(name => name !== options.CustomTextList);
CustomListText = [];
// Switch back to default list
options.CustomTextList = defaultLists.keys().next().value;
console.log("[URCom] Fallback to " + options.CustomTextList + " after list delete");
saveOptions(options);
loadResponses().then(refreshUI);
}
}
function deleteTextItem() {
DeleteTextMode = !DeleteTextMode;
refreshUI();
if (DeleteTextMode) {
showWMEMessage(translate("prompts.response-delete-click"), 5000);
}
}
function executeDeleteTextItem(index) {
return function() {
const text = CustomListText[index];
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
const request = objectStore.get(options.CustomTextList);
request.onerror = (event) => {
alert(translate("prompts.response-delete-failed"));
};
request.onsuccess = (event) => {
const data = event.target.result;
// Remove the one we don't need
data.responses = data.responses.filter(response => response.name !== text.name || response.text !== text.text || response.status !== text.status);
// Update sorting
for (let i = 0; i < data.responses.length; i++) {
data.responses[i].sorting = i;
}
const requestUpdate = objectStore.put(data);
requestUpdate.onerror = (event) => {
alert(translate("prompts.response-delete-failed"));
};
requestUpdate.onsuccess = (event) => {
DeleteTextMode = false;
loadCustomText(() => {
refreshUI();
});
}
}
}
}
function migrateDb(oldDb) {
return new Promise(async resolve => {
let migrating = true;
oldDb.transaction(async (tx) => {
await tx.executeSql('SELECT * FROM custom_comments', [], async function (tx, results) {
const len = results.rows.length;
console.log("[URCom] Migration found " + len + " existing lists");
for (let i = 0; i < len; i++) {
let migratingList = true;
const name = results.rows.item(i).name;
const responses = [];
await tx.executeSql('SELECT * FROM custom_comments_text WHERE list = ? ORDER BY sorting', [name], function (tx, results) {
const len = results.rows.length;
for (let j = 0; j < len; j++) {
const result = results.rows.item(j);
responses.push({
sorting: result.sorting,
name: result.name,
text: result.text,
status: result.status,
});
}
const objectStore = DB
.transaction("custom_lists", "readwrite")
.objectStore("custom_lists");
objectStore.add({
name: name,
order: i,
responses: responses
});
migratingList = false;
});
while (migratingList) {
await sleep(1);
}
}
migrating = false;
});
}, (error) => {
// Ignore since prolly table doesn't exist
migrating = false;
});
while (migrating) {
await sleep(1);
}
oldDb.transaction((tx) => {
tx.executeSql('DELETE FROM custom_comments', []);
});
resolve();
});
}
function addNewItem() {
ShowTextEditFields = true;
refreshUI()
}
function editItem() {
ShowTextOptionButtons = false;
TextEditIndex = -1;
refreshUI();
showWMEMessage(translate("prompts.response-edit-click"), 5000);
}
function selectEditItem(index) {
return async function() {
const text = CustomListText[index];
if (!text) {
TextEditIndex = undefined;
return;
}
ShowTextEditFields = true;
ShowTextOptionButtons = true;
TextEditIndex = index;
// Refresh first
await refreshUI();
// Then update the settings
const titleInput = $("#customTextTitle")[0];
const textInput = $("#customTextText")[0];
const statusSelect = $("#customTextStatus")[0];
titleInput.value = text.name;
textInput.value = text.text;
switch (text.status) {
case "open":
statusSelect.selectedIndex = 0;
break;
case "closed":
statusSelect.selectedIndex = 1;
break;
case "notidentified":
statusSelect.selectedIndex = 2;
break;
}
}
}
function escapeHtml(a) {
a = a.replace(/&/g, "&");
a = a.replace(/</g, "<");
a = a.replace(/>/g, ">");
a = a.replace(/"/g, """);
a = a.replace(/'/g, "'");
return a;
}
function autoZoomIN(title, comment, urStatus, urID, AutoSendComment) {
return function() {
let URCommentsUnsavedDetected = false;
const hasUnsavedChanges = wmeSDK.Editing.getUnsavedChangesCount();
// Detect unsaved changed if there are and the auto save option is on abort adding comments to the UR
if (hasUnsavedChanges > 0 && options.SaveAfterComment && urStatus.toLowerCase() !== "open") {
URCommentsUnsavedDetected = true;
alert(translate("prompts.unsaved-changes")); //"UrComments has detected that you have unsaved changes!\n\nWith the Auto Save option enabled and with unsaved changes you cannot send comments that would require the script to save. Please save your changes and then re-click the comment you wish to send."
}
// Get urID for manually clicked comments
if (urID === 0 || urID === "" || urID === undefined) {
urID = $(".map-problem.marker-selected").data("id");
}
// Check to see if the auto zoom in option is enabled, if it is start the zooming in process
if (options.NewZoomIn && AutoSendComment !== "AutoSendComment" && URCommentsUnsavedDetected === false) {
// Predefined zoom threshold for auto zoom
const zoom = 15;
// Do not zoom back out if we are already zoomed in and just happen to be re-clicking on a UR.
// or we have the map set good for a 4-day reminder
const WazeCurrentZoom = getWazeMapZoomLevel();
if (WazeCurrentZoom < zoom) {
goToURById(urID, zoom);
}
setTimeout(postURComment(title, comment, urStatus, AutoSendComment, urID), 1);
} else if (URCommentsUnsavedDetected === false) {
// auto zoom in is disabled jump to postURComment
// we have to use set timeout here because we need the return function() in PostURComment
// for when we are zooming in and out for the reminder
// since we are not zooming here jump rigth to PostURComment
setTimeout(postURComment(title, comment, urStatus, AutoSendComment, urID), 1);
}
};
}
function getWazeMapZoomLevel() {
return W.map.mapState.mapLocation.zoom;
}
function goToURById(urID, zoom) {
// save zoom so we can return this the current zoom level
ReturnToCurrentZoom = getWazeMapZoomLevel();
const geo = getUrById(urID).attributes.geometry;
W.map.getOLMap().moveTo([geo.x, geo.y], zoom);
}
function postURComment(title, comment, urStatus, autoSendComment, urID) {
// The user clicked on a comment link
return function() {
// Swap out special text
comment = stringSwap(comment, urID);
$('.new-comment-form .send-button').on('click', urCommentZoomOutCheck(title, urStatus));
// Check if the comment text area is present if not alert the user to open a UR
const commentArea = $($(".new-comment-text")[0]);
if (commentArea) {
commentArea.val(comment).change();
commentArea.keyup();
if(!$('.no-comments')) {
commentArea.css("backgroundColor","Yellow");
}
// Click the UR status
setTimeout(checkIfClickStatus(urStatus, autoSendComment), 100);
} else {
// We were unable to find an open UR
showWMEMessage(translate("prompts.no-comment-box"), 5000); //"URComments: Can not find the comment box! In order for this script to work you need to have a user request open."
}
};
}
function urCommentZoomOutCheck(Title, URStatus) {
return function() {
// This is the new place for zooming out and will still be happening while the comment is sending
if (options.ZoomOutAfterComment) {
setTimeout(setZoomCloseUR(ReturnToCurrentZoom, "LeaveOpen"), 0);
}
setTimeout(urCommentSendBtnClicked(Title, URStatus), 20);
}
}
function setZoomCloseUR(zoom, b) {
// This sets the map zoom; 0 is all the way out; 10 is all the way in but next to useless (the map and sat views disappear);
// the closest zoom that shows the sat and map is zoom 9
return function() {
W.map.setCenter(W.map.getCenter(), zoom);
// Close ur if zooming out to
if (b === "CloseUR") {
$('.close-panel')[0].click();
}
};
}
function stringSwap(string, urID) {
if (urID > 0) {
const ur = getUrById(urID);
string = string.replace("$URD", ur.attributes.typeText + ' : ' + ur.attributes.description);
string = string.replace(/ /g, "");
string = string.replace(/ /g, "");
//string = string.replace(/(\r\n|\n)/gm, "");
} else if ($(".description .content").length > 0) {
string = string.replace("$URD", $(".description .content").html());
string = string.replace(/ /g, "");
string = string.replace(/ /g, "");
//string = string.replace(/(\r\n|\n)/gm, "");
string = string.replace("$USERNAME", W.loginManager.user.getUsername());
} else if (string.indexOf("$URD") >= 0) {
string = string.replace(" \"$URD\"", "");
}
// Legacy, remove old signature
string = string.replace('$SIGNATURE', '');
const greeting = options.Greeting;
if (greeting && greeting !== "" && greeting !== null && greeting !== "undefined") {
string = greeting + "\r\n" + string;
}
const signature = options.Signature;
if (signature && signature !== "" && signature !== null && signature !== "undefined") {
string = string + "\r\n" + signature;
}
// Selected segments
let streetName = "";
if (W.selectionManager.getSelectedFeatures().length >= 1 && W.selectionManager.getSelectedFeatures().length <= 2) {
for (let i = 0; i < W.selectionManager.getSelectedFeatures().length; i++) {
if (W.selectionManager.getSelectedFeatures()[i].model.CLASS_NAME === "W.Feature.Vector.Segment") {
const primaryStreetID = W.selectionManager.getSelectedFeatures()[i].model.attributes.primaryStreetID;
if (i === 0) {
streetName = W.model.streets.objects[primaryStreetID].name;
} else {
streetName = "the intersection of " + streetName + " and " + W.model.streets.objects[primaryStreetID].name;
}
}
}
string = string.replace("$SELSEGS", streetName);
}
return string;
}
function urCommentSendBtnClicked(title, urStatus) {
// Waze is weird and after clicking send button the close button had to be refound, which takes a few seconds for the new close button to be drawn
// so we wait 1500 milliseconds before looking for the close button
// since we are passing vars to the next function we have to pass this to handler function so it doesn't happen on click
// The above is still true but what i have found is that it sometimes takes a while from clicking send to the comment actually posting.
// There was time when i closed the comment before it actually posted and it would have to be redone. So added a check and a timeout and recheck if the textarea isn't empty.
// Afterwards we can re-grab and click the close button
return function() {
// Grab the close button to compare to later
CloseButtonHolder = $(".problem-panel-navigation button.close-button");
// Check to see if the comments went through before saving or closing the comment
const textArea = $(".new-comment-text");
if (textArea.val() !== "" && textArea.val() !== undefined) {
setTimeout(urCommentSendBtnClicked(title, urStatus), 20);
} else {
closeUR(title, urStatus);
}
};
}
function closeUR(title, urStatus) {
return function() {
if (urStatus.toLowerCase() === "solved" || urStatus.toLowerCase() === "notidentified") {
// This clicks the waze save btn
if (options.AutoClickURStatus && options.SaveAfterComment) {
// Click save
$('.toolbar #save-button').trigger('click');
} else if (options.UrCommentAutoCloseComment) {
$('.close-panel')[0].click();
}
} else {
// When not saving you have to click close.
if (options.UrCommentAutoCloseComment) {
$('.close-panel')[0].click();
}
}
//this is the new place for zooming out and will still be happening while the comment is sending
//zoom out option - if the user option is set to reload map after posting a comment reply
if (options.ZoomOutAfterComment) {
//urcToConsole("zoom out closeUR");
setTimeout(setZoomCloseUR(ReturnToCurrentZoom, "LeaveOpen"), 0);
}
setTimeout(urcUREvent_onObjectsAdded('URCommentsReload'), 0);
};
}
function urcUREvent_onObjectsAdded(a) {
return function() {
// Check if the pill count are enabled otherwise abort this function
if (!options.URCommentsPillEnabled) {
return
}
if (a === "timedout" && window.HideUR === "running") {
window.HideUR = "stopped";
a = "timedout";
}
if (window.HideUR === "stopped" && a !== "timedout") {
clearTimeout(window.buffertedtimeout);
clearTimeout(window.buffertedtimeout2);
// Certain events need to be launched a bit later then others
if (a === "moveend" || a === "zoomend") {
window.buffertedtimeout = setTimeout(FilterURs, 2000);
} else {
window.buffertedtimeout = setTimeout(FilterURs, 500);
}
window.HideUR = "running";
window.BufferFillterURCommand = false;
window.buffertedtimeout2 = setTimeout(urcUREvent_onObjectsAdded("timedout"), 30000);
}
};
}
function FilterURs() {
const allUrs = W.model.mapUpdateRequests.objects;
const allUrIDs = Object.keys(allUrs);
// Abort filtering URs if the list is empty
if (allUrIDs.length === 0) {
window.HideUR = "stopped";
setTimeout(urcUREvent_onObjectsAdded("list is empty retry"), 5000);
return;
}
let allID = '';
let countInCurrentList = 0;
let delay = 0;
let languageCount = knownLanguages && knownLanguages.length;
for (let urID of allUrIDs) {
const ur = allUrs[urID];
const language = ur.attributes.reporterLanguage;
if (knownLanguages && knownLanguages.indexOf(language) === -1) {
knownLanguages.push(language);
}
// Make sure we can see the UR
if (onScreen(ur)) {
if (allID !== '') {
allID += ",";
}
allID = allID + urID;
countInCurrentList = countInCurrentList + 1;
}
// Batch the current list
if (countInCurrentList >= 500) {
// Send this batch in a timeout
setTimeout(FilterURs2(allID), delay);
countInCurrentList = 0;
allID = "";
delay = delay + 1000;
}
}
// Keep track of possible languages, to later support different list per ur language
if (knownLanguages && languageCount !== knownLanguages.length) {
const objectStore = DB
.transaction("data-store", "readwrite")
.objectStore("data-store");
const requestUpdate = objectStore.put({
id: "languages",
languages: knownLanguages
});
requestUpdate.onerror = (event) => {
console.log("[URCom] Failed to update known languages");
};
requestUpdate.onsuccess = (event) => {
console.log("[URCom] Updated known languages");
};
}
setTimeout(FilterURs2(allID), delay);
}
function FilterURs2(allID) {
return function() {
const CloseDays = options.CloseDays;
const ReminderDays = options.ReminderDays;
const ClosePlusReminderDays = CloseDays + ReminderDays;
const EditorID = W.loginManager.user.getID(); // Editor's id number
if (allID !== "") {
// This is the link that works but I think it is grabbing a cached version of the page
//https://www.waze.com/row-Descartes/app/MapProblems/UpdateRequests?ids=20006756
const urSessionURL = 'https://' + window.location.host + W.Config.api_base + '/MapProblems/UpdateRequests?ids=' + allID + '&time=' + (new Date()).getTime();
let LastCommenterUserID = '0';
// Send a request to load the UR data
$.ajax({
dataType: "json",
url: urSessionURL,
success: function(json) {
try {
const urs = json.updateRequestSessions.objects;
const layer = W.map.layers.find((layer) => layer.name === "update_requests");
for (let jsnObj = 0; jsnObj < urs.length; jsnObj++) {
const urDetail = urs[jsnObj];
const id = urDetail.id;
const ur = W.model.mapUpdateRequests.objects[id];
let skip = false;
let urStyle = 'shown';
let lastCommentDateTime = 0;
const numberOfComments = urDetail.comments.length;
let CountsBGColor = '#FFFF99'; // Light yellow
const urBubble = $('image#' + id);
if (numberOfComments > 0) {
let reporterResponded = false;
let LastComment = '';
for (let comment of urDetail.comments) {
if (comment.userID === -1) {
reporterResponded = true;
}
const commentText = comment.text;
lastCommentDateTime = comment.createdOn;
LastCommenterUserID = comment.userID;
LastComment = commentText;
// Only white if logged in editor has the last editor comment
if (LastCommenterUserID > 1) {
if (LastCommenterUserID === EditorID) {
CountsBGColor = "white";
} else {
CountsBGColor = "#FFFF99"; //light yellow
}
}
}
}
let lastCommentAge = uroDateToDays(lastCommentDateTime);
if (ur.attributes.open === false) {
skip = true;
}
if (skip === false && ur.type === "mapUpdateRequest") {
if (numberOfComments > 0) {
// Last comment was a user reply; change the pin color to blue or peach
if (LastCommenterUserID < 1) {
if (CountsBGColor === "white") {
CountsBGColor = "#79B5C7"; // Light blue
} else {
CountsBGColor = "#FFCC99"; // Peach
}
} else if (lastCommentAge > ClosePlusReminderDays && CountsBGColor !== "white") {
CountsBGColor = "#FF8B8B"; // Light red
}
if (CountsBGColor === "#FFFF99" && numberOfComments >= 1 && lastCommentAge >= ClosePlusReminderDays && LastCommenterUserID > 1) {
CountsBGColor = "#FF8B8B"; // Light red
}
}
}
if (lastCommentAge === undefined) {
lastCommentAge = "?";
}
let Message = numberOfComments + UrCommentsTextPic + " " + lastCommentAge + UrCommentsTimePic;
// If the ur has more than 0 comments reorder the ur stacking
if (numberOfComments > 0) {
let MessageOffset = Message.length;
MessageOffset = MessageOffset - (UrCommentsTextPic.length - 3)
MessageOffset = MessageOffset - (UrCommentsTimePic.length - 4)
if (MessageOffset < 10) {
MessageOffset = MessageOffset * 1.4;
} else {
MessageOffset = MessageOffset * 2.3;
}
MessageOffset = "-" + MessageOffset + "px";
if (CountsBGColor === "#CCCCCC") { // Light grey
urBubble.css({
'z-index': '999'
});
} else if (CountsBGColor === "white" || CountsBGColor === "#79B5C7") {
urBubble.css({
'z-index': '998'
});
//).style.visibility
} else if (CountsBGColor === "#FF8B8B") { // Light red
urBubble.css({
'z-index': '997'
});
}
/*
CountsBGColor = "#FFFF99"; //light yellow = UR that are waiting for the last editor but still has time before the Close days setting ex 7 days
CountsBGColor = "#CCCCCC"; //light grey = URO styled notes ex [NOTE]
CountsBGColor = "#FFCC99"; //peach = another editor before the close days has gone by has user reply
CountsBGColor = "#FF8B8B"; //light red = past the close days setting and is now able to be taken over by another editor
CountsBGColor = "white"; = UR that "belong to the editor logged in that need work
CountsBGColor = "#79B5C7"; //light blue = UR that "belong to the editor logged in that have user replies
White = The editor logged in has comments on the UR and the UR needs work
light blue = The editor logged in has comments on the UR and the UR's last comment was a user reply
light red = past the close days setting and is now able to be taken over by another editor
light yellow = UR that are waiting for the another editor to send a reminder or to be closed but still has time before the Close days setting (ex 7 days)
peach = UR that another editor had commented on but the last comment is a user reply
*/
if (options.URCommentsPillEnabled) {
const featureId = layer.features.find((feature) => feature.attributes.wazeFeature.id === id).geometry.id;
const urCountBubble = $('image#' + featureId);
// Add URC bubble
if (urCountBubble.hasClass('URCounts') === false) {
urBubble.append($("<div>").css("clear", "both")
.css("margin-bottom", "10px")
.append($("<div>").html(Message)
.css("color", "black")
.css("background", CountsBGColor)
.css("position", "absolute")
.css("top", "30px")
.css("height", "18px")
.css("right", MessageOffset)
.css("display", "block")
.css("width", "auto")
.css("white-space", "nowrap")
.css("padding-left", "5px")
.css("padding-right", "5px")
.css("border", "1px solid")
.css("border-radius", "25px")
.addClass('URCounts')
)
);
} else {
urCountBubble.html(Message);
urCountBubble.css("background", CountsBGColor);
urCountBubble.css("right", MessageOffset);
}
// End of URC bubble
}
}
urStyle = 'visible';
urBubble.css("visibility", urStyle);
}
window.HideUR = "stopped";
} catch (e) {
console.log(e);
setTimeout(FilterURs, 1000);
}
}
});
} else {
window.HideUR = "stopped";
}
};
}
function onScreen(obj) {
// Geometry was removed from W.model.mapUpdateRequests.objects[id]
// but can be found in W.model.mapUpdateRequests.objects[id].geometry
const geo = obj.getOLGeometry();
if (geo) {
return W.map.getExtent().intersectsBounds(geo.getBounds());
}
return false;
}
function uroDateToDays(dateToConvert) {
const dateNow = new Date();
const elapsedSinceEpoch = dateNow.getTime();
const elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
dateNow.setHours(0);
dateNow.setMinutes(0);
dateNow.setSeconds(0);
dateNow.setMilliseconds(0);
const elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
if (elapsedSinceEvent < elapsedSinceMidnight) {
// event occurred today...
return 0;
} else {
// event occurred at some point prior to midnight this morning, so return a minimum value of 1...
return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
}
}
function uroGetCommentAge(commentObj) {
if (commentObj.createdOn === null) {
return -1;
}
return uroDateToDays(commentObj.createdOn);
}
function showWMEMessage(message, delay) {
const dateNow = new Date();
let TrickRemove = dateNow.getTime();
const width = message.length * 10;
// Message holder div
const c = '<div id="URCMessage" style="width: 100%; font-size: 15px; font-weight: bold; margin-left: auto; margin-right: auto; position: absolute; top: 0; left: 10px; z-index: 1000;"></div>';
$("#map").append(c);
// Message div
const d = '<div id="URCMapNote' + TrickRemove + '" style="width: ' + width + 'px; font-size: 15px; font-weight: bold; margin-left: auto; margin-right: auto; background-color: orange;"><center><b>' + message + '</b></center></div>';
$("#URCMessage").append(d);
// Remove messages after the duration has passed
$("#URCMapNote" + TrickRemove).show().delay(delay).queue(function(n) {
$(this).remove();
n();
});
}
function checkIfClickStatus(URStatus, AutoSendComment) {
return function() {
if (options.AutoClickURStatus) {
// Bypass the confirm function so other site messages do not come up!
confirmToConsole();
const openInput = $("input[value='open']");
// Click the ur status options (Not identified solved, open)
if (openInput) {
if (URStatus.toLowerCase() === "notidentified") {
// Click Not identified
$("input[value='not-identified']")[0].click();
} else if (URStatus.toLowerCase() === "solved") {
// Click solved
$("input[value='solved']")[0].click();
} else {
// Click back on open just encase the wrong reply was clicked previously
openInput[0].click();
}
}
// Restores confirm function so other site messages come up!
restoreConfirmToConfirm();
}
if (AutoSendComment === "AutoSendComment") {
setTimeout(uRVerifyStatusSet(URStatus, AutoSendComment), 1);
}
};
}
function confirmToConsole() {
// Override the default action of the standard confirm by writing confirm to a new function nconfirm, making a 'fake' confirm and then restoring confirm by copying the nconfirm back over confirm!
nConfirm = confirm;
confirm = function(msg) {
// if ((I18n.translations[I18n.locale].update_requests.panel.confirm == msg) && (uroGetCBChecked('_cbDisablePendingQuestions') == true)) {
// urcToConsole('URComment confirm redirected to console: ' + msg);
return true;
};
}
function restoreConfirmToConfirm() {
// Restore the normal action of confirm by writing nconfirm to back over confirm, so the site is able to send user messages outside of my script!
confirm = nConfirm;
}
function uRVerifyStatusSet(URStatus, AutoSendComment) {
return function() {
let URStatusOk = true;
if (options.AutoClickURStatus) {
if (URStatus.toLowerCase() === "notidentified") {
// Click Not identified
if ($("input[value='not-identified']").is(":checked") === false) {
URStatusOk = false;
}
} else if (URStatus.toLowerCase() === "solved") {
// Click solved
if ($("input[value='solved']").is(":checked") === false) {
URStatusOk = false;
}
} else {
// Click back on open just encase the wrong reply was clicked previously
if ($("input[value='open']").is(":checked") === false) {
URStatusOk = false;
}
}
}
if (URStatusOk) {
setTimeout(autoClickSend(URStatus, AutoSendComment), 10);
} else {
setTimeout(uRVerifyStatusSet(URStatus, AutoSendComment), 150);
}
};
}
function autoClickSend(URStatus, AutoSendComment) {
return function() {
if (AutoSendComment === "AutoSendComment") {
$('.new-comment-form .send-button').trigger('click');
}
};
}
// This is the small crosshair in the UR window
function crossHairsClicked(urID, zoom) {
return function() {
setTimeout(() => goToURById(urID, zoom), 100);
};
}
function getUrById(id) {
/*
const urs = W.model.mapUpdateRequests.getObjectArray();
for (let i = 0; i < urs.length; i++) {
if (urs[i].attributes?.id === id) {
return urs[i];
}
}
return null;
*/
return W.model.mapUpdateRequests.objects[id];
}
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
//Make HTTP Requests
function makeHTTPRequest(type, url) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: type,
url: url,
headers: {"Referer": document.location.href},
onload: (response) => {
resolve(JSON.parse(response.response));
},
onerror: (error) => {
reject(error);
}
});
});
}
function translate(s, data) {
s = s.replace(/\[(\w+)\]/g, '.$1'); // convert indexes to properties
s = s.replace(/^\./, ''); // strip a leading dot
let current = translation;
const a = s.split('.');
for (let i = 0, n = a.length; i < n; ++i) {
let k = a[i];
if (k in current) {
current = current[k];
} else {
return s;
}
}
if (current && data) {
for (const [key, value] of Object.entries(data)) {
current = current.replace("{{" + key + "}}", value);
}
}
return current;
}
///////////////////////////////////////////////////////////////////////////////////
//////
////// Background task(s)
//////
///////////////////////////////////////////////////////////////////////////////////
// Look for the UR window to be open
function lookForOpenedUR() {
// Relaunch checking for open ur - moved this up here so the script would keep trying if I had an error
setTimeout(lookForOpenedUR, 500);
let urID = W.map.getLayerByName('update_requests').features.filter((feature) => feature.attributes.wazeFeature.isSelected)[0]?.attributes?.wazeFeature?.id;
if (typeof urID !== 'undefined' && urID > 0) {
// New or same ur (re)open(ed) so now we can do stuff; this prevents this function to continuously applying settings to a ur
if (UrCommentLasturID !== urID && $(".new-comment-text").length !== 0) {
// grab the current UR ID for next time
UrCommentLasturID = urID;
// Auto expand ur comments list and text area
const conversation = $("#panel-container .problem-edit .conversation");
if (conversation.length !== 0) {
conversation.removeClass('collapsed');
}
// Auto expand ur comments description
const description = $("#panel-container .problem-edit .description");
if (description.length !== 0) {
description.removeClass('collapsed');
if (options.HideAADescription) {
const content = $("#panel-container .problem-edit .description .content");
if (content.length !== 0) {
let contentText = content.text();
// Does the AA tag exist?
if (contentText.indexOf("Reported from AAOS") >= 0) {
if (contentText === "Reported from AAOS") {
// Full description so just hide it
description.css("display", "none");
} else {
// Subtract if from the content
content.text(contentText.replace("Reported from AAOS", ""));
}
}
}
}
}
// Disable the stupid buttons at the bottom of the UR window done / next
const nextButton = $("#panel-container .content .navigation");
if (nextButton.length > 0 && options.UrCommentDisableURDoneBtn) {
nextButton.css('display', 'none');
}
// Attach an even listener to the UR center button to take us to the UR
$('#panel-container .header .panel-header-actions .focus').on('click', crossHairsClicked(urID, 15));
const UR = getUrById(urID);
// Grab the report language
const language = W.model.mapUpdateRequests.objects[urID].attributes.reporterLanguage;
console.log("[URCom] Report language is: " + language);
// Autofill in comment
// Check what type of message to insert into the ur
let comments;
if (W.model.updateRequestSessions.objects[urID].hasOwnProperty('comments')) {
comments = W.model.updateRequestSessions.objects[urID].comments;
} else {
comments = W.model.updateRequestSessions.objects[urID].attributes.comments;
}
const commentCount = comments.length;
// If number of comment is zero assume this is a new ur
if (!UR.attributes.hasComments) {
const noPermissionAlert = $(".no-permissions-alert").length > 0;
// Initial, zero comments
if (options.AutoSetNewComment && !noPermissionAlert) {
// This will be on of the types of UR that a user can choose from when submitting a UR
const urType = UR.attributes.type;
console.log("[URCom] UR Type: " + urType);
if (options.CustomTextList && !defaultLists.has(options.CustomTextList)) {
for (let i = 0; i < CustomListText.length; i++) {
let text = CustomListText[i];
if (text.name.toLowerCase().replace('.', '') === translate("ur-types." + urType).toLowerCase()) {
let reply = text.text;
setTimeout(autoZoomIN(text.name, reply, text.status, urID), 0);
break;
}
}
} else {
// Loop through the comment array for a comment that matches the request type.
for (let i = 0; i < URCommentsArray.length; i++) {
const comment = URCommentsArray[i];
if (comment.urType === "" + urType) {
setTimeout(autoZoomIN(comment.title, comment.response, comment.status, urID), 0);
break;
}
}
}
//var inmyzone = 1;
} else if (noPermissionAlert){
//$(".new-comment-text").prop('disabled', true);
//$(".send-button").hide();
//var inmyzone = 0;
}
} else {
// Reminder & close section
const LastCommenterUserID = comments[comments.length - 1].userID;
// uro days old
let commentDaysOld = uroGetCommentAge(comments[commentCount - 1]);
// Do we want to auto set reminders
if (options.UrCommentAutoSet4dayComment) {
// -- REMINDER SECTION -- \\
const ReminderDays = options.ReminderDays;
// Only 1 comment has been added, reminder days is higher than 0, reminder days has passed, and comment isn't from reporter (aka userid != -1)
if (commentCount === 1 && ReminderDays > 0 && commentDaysOld >= ReminderDays && LastCommenterUserID > 1) {
const comment = URCommentsArray.find((c) => c.urType === "-1");
setTimeout(autoZoomIN(comment.title, comment.response, comment.status, urID), 0);
}
// -- CLOSE SECTION -- \\
const CloseDays = options.CloseDays;
if (commentCount > 1 && CloseDays > 0 && commentDaysOld >= CloseDays && LastCommenterUserID > 1) {
const comment = URCommentsArray.find((c) => c.urType === "-2");
setTimeout(autoZoomIN(comment.title, comment.response, comment.status, urID), 0);
}
}
// Don't allow auto filling of the close message because it clicks the not identified option and causes trouble when the Ur window is shut/closed the next save will mark it as not identified, without a comment.
}
/*if ($(".no-permissions-alert")[0] || $(".close-details")[0]){
$(".new-comment-text").prop('disabled', true);
$(".new-comment-text").hide();
$(".send-button").hide();
}*/
/*if($(".new-comment-text").val()==="" && URComSignature.trim()!=="")
{
$(".new-comment-text").val("\n\n" + URComSignature);
}*/
const WazeCurrentZoom = getWazeMapZoomLevel();
// Predefined zoom threshold for auto zoom
const zoom = 15;
if (commentCount === 0 && options.NewZoomIn) {
// Zoom in new
// Do not zoom back out if we are already zoomed in and just happen to be re-clicking on a UR.
if (WazeCurrentZoom < zoom) {
goToURById(urID, zoom);
}
}
// Jump to bottom in full UR section
const topSection = $('.problem-data').parent();
topSection.scrollTop(topSection[0].scrollHeight);
// Jump to bottom in conversation section
const conversationSection = $('.conversation-view .comment-list');
conversationSection.scrollTop(conversationSection[0].scrollHeight);
// Auto switch to ur comments tab
if (options.AutoSwitchToURCommentsTab) {
// Grab the active tab
PreviousTab = $("#user-tabs > ul > li.active > a");
// Grab the active tabs scroll position
const tabContent = $('#sidebar div.tab-content');
PreviousTabPosition = tabContent.scrollTop();
tabContent.scrollTop(0); // Reset scroll pos
// Open the scripts tab
if (!$("[data-for=userscript_tab]")[0].selected) {
$(".w-icon.w-icon-script").click();
}
// Make UR Comments tab active
$("span[title='URCom Settings']").click();
}
}
} else {
// Reset the id if a ur is not open so we can set the tab for the same ur
UrCommentLasturID = "";
// Switch tab back
if (options.AutoSwitchToURCommentsTab) {
// Verify that we had found a tab
if (PreviousTab !== null) {
$(PreviousTab).click(); // Click back on the previous tab
$('#sidebar div.tab-content').scrollTop( PreviousTabPosition ); // Set scroll pos
// Clear out the previous tab holder
PreviousTab = null;
PreviousTabPosition = null;
}
}
}
}
function URCLoadUpdateRequestsEvents() {
window.HideUR = "stopped";
//from Cardyin 5/26/2016
W.model.mapUpdateRequests.on("objectschanged", urcUREvent_onObjectsAdded('mapUpdateRequests objectschanged')); // needed
W.model.updateRequestSessions.on("objectsadded", urcUREvent_onObjectsAdded('updateRequestSessions objectsadded')); // needed
W.map.events.register("moveend", null, urcUREvent_onObjectsAdded('moveend')); //needed
W.map.events.register("zoomend", null, urcUREvent_onObjectsAdded('zoomend')); //needed
setTimeout(urcUREvent_onObjectsAdded('pageload'), 500);
}
function createSettingsTab() {
applyCss();
// -- Set up the tab for the script
wmeSDK.Sidebar.registerScriptTab().then(({ tabLabel, tabPane }) => {
console.log("[URCom] Register settings tab");
tabLabel.innerHTML = '<img src="' + UrCommentsIcon + '" height="16px" title="URCom" alt="URCom"/> URCom';
tabLabel.title = 'URCom Settings';
tabPane.innerHTML = '<div id="sidepanel-Comments"></div>';
scriptContentPane = $('#sidepanel-Comments');
init();
});
}
function init() {
// Add the content to the comments tab
const tabs = $('<wz-tabs fixed="true"></wz-tabs>');
const textTab = $('<wz-tab is-active label="' + translate("tabs.comments") + '" tooltip="' + translate("tabs.comments") + '"></wz-tab>');
const textTabContent = $('<div id="sidepanel-URComments-list"></div>')
textTab.append(textTabContent);
const settingsTab = $('<wz-tab label="' + translate("tabs.settings") + '" tooltip="' + translate("tabs.settings") + '"></wz-tab>');
const settingsTabContent = $('<div id="sidepanel-URComments-settings"></div>')
settingsTab.append(settingsTabContent);
tabs.append(textTab);
tabs.append(settingsTab);
scriptContentPane.append(tabs);
////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////
////// Draw Options
//////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
addCommentTexts(textTabContent);
addOptions(settingsTabContent);
////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////
////// Launch background task(s)
//////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Start looking for opened UR's window
setTimeout(lookForOpenedUR, 1000);
setTimeout(URCLoadUpdateRequestsEvents, 0);
$('#sidepanel-Comments').closest('section').css('padding', '0px');
}
function startCode() {
if ($("#user-tabs").length && InitReady) {
createSettingsTab();
} else {
setTimeout(startCode, 2000);
}
}
async function initDatabase() {
const request = window.indexedDB.open("URCom", 2);
request.onerror = (event) => {
console.error("[URCom] Why didn't you allow my web app to use IndexedDB?!");
};
request.onupgradeneeded = (event) => {
console.log(event);
const db = event.target.result;
if (event.newVersion >= 1 && event.oldVersion < 1) {
const customListStore = db.createObjectStore("custom_lists", { keyPath: "name" });
customListStore.transaction.oncomplete = (event) => {
console.log("[URCom] Created list store");
}
}
if (event.newVersion >= 2 && event.oldVersion < 2) {
const dataStore = db.createObjectStore("data-store", { keyPath: "id" })
dataStore.transaction.oncomplete = (event) => {
console.log("[URCom] Created data store");
}
}
}
request.onsuccess = (event) => {
console.log("[URCom] DB ready");
DB = event.target.result;
};
while (!DB) {
await sleep(10);
}
if ("openDatabase" in window) {
try {
let created = false;
const oldDb = openDatabase(
"URCom", // actual database name. Opens existing or creates.
"1", // version number. *Must* be correct.
"Database to store custom URCom data", //
10 * 1024 * 1024, // Size of the DB
() => {
created = true;
}
);
if (!created) {
console.log("[URCom] Migrating");
await migrateDb(oldDb);
console.log("[URCom] Migration over");
}
} catch (e) {}
}
}
unsafeWindow.SDK_INITIALIZED.then(async () => {
// initialize the sdk with your script id and script name
wmeSDK = getWmeSdk({scriptId: "urcom", scriptName: "UR Comments"});
await URComments_init();
})