WME URCom

This script is for replying to user requests the goal is to speed up and simplify the process!

安裝腳本?
作者推薦腳本

您可能也會喜歡 WME Send to Slack

安裝腳本
// ==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, "&amp;");
	a = a.replace(/</g, "&lt;");
	a = a.replace(/>/g, "&gt;");
	a = a.replace(/"/g, "&quot;");
	a = a.replace(/'/g, "&#039;");
	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();
})