4454 lines
		
	
	
		
			No EOL
		
	
	
		
			140 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			4454 lines
		
	
	
		
			No EOL
		
	
	
		
			140 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/*
 | 
						|
██╗░░░██╗████████╗██╗██╗░░░░░░██████╗
 | 
						|
██║░░░██║╚══██╔══╝██║██║░░░░░██╔════╝
 | 
						|
██║░░░██║░░░██║░░░██║██║░░░░░╚█████╗░
 | 
						|
██║░░░██║░░░██║░░░██║██║░░░░░░╚═══██╗
 | 
						|
╚██████╔╝░░░██║░░░██║███████╗██████╔╝
 | 
						|
░╚═════╝░░░░╚═╝░░░╚═╝╚══════╝╚═════╝░
 | 
						|
*/
 | 
						|
const resName = GetParentResourceName();
 | 
						|
 | 
						|
const allJobsStatisticsDiv = document.getElementById('all-jobs-statistics').getContext('2d');
 | 
						|
const jobStatisticsDiv = document.getElementById('job-statistics-chart').getContext('2d');
 | 
						|
 | 
						|
let TRANSLATIONS = {};
 | 
						|
let ENGLISH_TRANSLATIONS = {};
 | 
						|
let MARKERS_INFOS = {};
 | 
						|
let ACTIONS_LABELS = {};
 | 
						|
 | 
						|
function translateEverything() {
 | 
						|
	$("body").find("[data-translation-id], [data-bs-toggle='tooltip']").each(function() {
 | 
						|
		let translationId = $(this).data("translationId")
 | 
						|
 | 
						|
		if( $(this).data("bsToggle") == "tooltip" ) {
 | 
						|
			if ($(this).tooltip) {
 | 
						|
				$(this).tooltip('dispose');
 | 
						|
			}
 | 
						|
	
 | 
						|
			$(this).attr("data-bs-title", getLocalizedText(translationId));
 | 
						|
			$(this).tooltip();
 | 
						|
		} else if( $(this).hasClass("form-control") ) {
 | 
						|
			$(this).attr("placeholder", getLocalizedText(translationId));
 | 
						|
		} else {
 | 
						|
			$(this).prop("innerHTML", getLocalizedText(translationId));
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	MARKERS_INFOS = {
 | 
						|
		["stash"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:stash"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:stash_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: false
 | 
						|
		},
 | 
						|
		["armory"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:armory"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:armory_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true,
 | 
						|
			framework: "ESX"
 | 
						|
		},
 | 
						|
		["safe"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:safe"),
 | 
						|
			settingsName: null,
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: false,
 | 
						|
			framework: "ESX"
 | 
						|
		},
 | 
						|
		["garage"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:temporary_garage"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:temporary_garage_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["garage_buyable"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:garage_buyable"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:garage_buyable_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["garage_owned"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:garage_owned"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:garage_owned_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["boss"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:boss"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:boss_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: false,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["wardrobe"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:wardrobe"),
 | 
						|
			settingsName: null,
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["job_outfit"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:job_outfit"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:job_outfit_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["shop"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:shop"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:shop_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["market"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:market"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:market_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["harvest"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:harvest"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:harvest_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["process"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:process"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:process_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["crafting_table"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:crafting_table"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:crafting_table_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["teleport"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:teleport"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:teleport_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["weapon_upgrader"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:weapon_upgrader"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:weapon_upgrader_settings"),
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["duty"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:duty"),
 | 
						|
			settingsName: null,
 | 
						|
			isJobMarker: true,
 | 
						|
			isPublicMarker: false,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
		["job_shop"]: { 
 | 
						|
			label: getLocalizedText("menu:dynamic:marker_info:job_shop"),
 | 
						|
			settingsName: getLocalizedText("menu:dynamic:marker_info:job_shop_settings"),
 | 
						|
			isJobMarker: false,
 | 
						|
			isPublicMarker: true,
 | 
						|
			requiresDataUpdate: true
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	ACTIONS_LABELS = [
 | 
						|
		{action: "actionsMenuEnabled", label: getLocalizedText("menu:job_settings:can_open_actions_menu")},
 | 
						|
		{action: "canRepairVehicles", label: getLocalizedText("menu:job_settings:enable_repair_vehicles")},
 | 
						|
		{action: "canCheckDrivingLicense", label: getLocalizedText("menu:job_settings:enable_check_driving_license")},
 | 
						|
		{action: "canCheckWeaponLicense", label: getLocalizedText("menu:job_settings:enable_check_weapon_license")},
 | 
						|
		{action: "enableBilling", label: getLocalizedText("menu:job_settings:enable_billing")},
 | 
						|
		{action: "canHandcuff", label: getLocalizedText("menu:job_settings:enable_handcuff_players")},
 | 
						|
		{action: "placeableObjects", label: getLocalizedText("menu:job_settings:placeable_objects")},
 | 
						|
		{action: "canLockpickCars", label: getLocalizedText("menu:job_settings:enable_lockpick_players")},
 | 
						|
		{action: "whitelisted", label: getLocalizedText("menu:job_settings:whitelisted")},
 | 
						|
		{action: "canRevive", label: getLocalizedText("menu:job_settings:enable_revive")},
 | 
						|
		{action: "canWashVehicles", label: getLocalizedText("menu:job_settings:enable_wash_vehicles")},
 | 
						|
		{action: "canRob", label: getLocalizedText("menu:job_settings:enable_search_players")},
 | 
						|
		{action: "canCheckIdentity", label: getLocalizedText("menu:job_settings:enable_check_identities")},
 | 
						|
		{action: "canImpoundVehicles", label: getLocalizedText("menu:job_settings:enable_impound_vehicles")},
 | 
						|
		{action: "canCheckVehicleOwner", label: getLocalizedText("menu:job_settings:enable_check_vehicle_owner")},
 | 
						|
		{action: "canHeal", label: getLocalizedText("menu:job_settings:enable_heal")},
 | 
						|
	]
 | 
						|
} 
 | 
						|
 | 
						|
async function refreshTranslations(locale) {
 | 
						|
	let rawEnglishTranslations = await $.get("menu_translations/en.json");
 | 
						|
	ENGLISH_TRANSLATIONS = typeof rawEnglishTranslations == "object" ? rawEnglishTranslations : JSON.parse(rawEnglishTranslations);
 | 
						|
 | 
						|
	let rawTranslations = await $.get(`menu_translations/${locale}.json`);
 | 
						|
	TRANSLATIONS = typeof rawTranslations == "object" ? rawTranslations : JSON.parse(rawTranslations);
 | 
						|
 | 
						|
	translateEverything();
 | 
						|
}
 | 
						|
 | 
						|
async function loadTranslations() {
 | 
						|
	const locale = await $.post(`https://${resName}/getLocale`);
 | 
						|
 | 
						|
	refreshTranslations(locale);
 | 
						|
} loadTranslations();
 | 
						|
 | 
						|
function getLocalizedText(text) {
 | 
						|
	return TRANSLATIONS[text] || ENGLISH_TRANSLATIONS[text] || text;
 | 
						|
}
 | 
						|
 | 
						|
async function getFramework() {
 | 
						|
	const framework = await $.post(`https://${resName}/getFramework`);
 | 
						|
	return framework;
 | 
						|
}
 | 
						|
 | 
						|
const allJobsStatisticsChart = new Chart(allJobsStatisticsDiv, {
 | 
						|
	type: 'bar',
 | 
						|
 | 
						|
	data: {
 | 
						|
		labels: "Jobs",
 | 
						|
		datasets: []
 | 
						|
	},
 | 
						|
 | 
						|
	options: {
 | 
						|
		scales: {
 | 
						|
			y: {
 | 
						|
				beginAtZero: true
 | 
						|
			}
 | 
						|
		},
 | 
						|
	}
 | 
						|
});
 | 
						|
 | 
						|
const jobStatisticsChart = new Chart(jobStatisticsDiv, {
 | 
						|
	type: 'bar',
 | 
						|
 | 
						|
	data: {
 | 
						|
		labels: "Ranks",
 | 
						|
		datasets: []
 | 
						|
	},
 | 
						|
 | 
						|
	options: {
 | 
						|
		scales: {
 | 
						|
			y: {
 | 
						|
				beginAtZero: true
 | 
						|
			}
 | 
						|
		},
 | 
						|
	}
 | 
						|
});
 | 
						|
 | 
						|
// Fetch all the forms we want to apply custom Bootstrap validation styles to
 | 
						|
var forms = document.querySelectorAll('.needs-validation')
 | 
						|
 | 
						|
// Loop over them and prevent submission
 | 
						|
Array.prototype.slice.call(forms)
 | 
						|
.forEach(function (form) {
 | 
						|
	form.addEventListener('submit', function (event) {
 | 
						|
 | 
						|
	event.preventDefault();
 | 
						|
 | 
						|
	form.classList.add('was-validated')
 | 
						|
	}, false)
 | 
						|
})
 | 
						|
 | 
						|
$(".max-two-decimals").change(function() {
 | 
						|
	if(isNaN(this.value)) {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	let number = parseFloat(this.value);
 | 
						|
 | 
						|
	if(number) {
 | 
						|
		this.value = number.toFixed(2);
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function getValidId(text) {
 | 
						|
	text = text.toLowerCase();
 | 
						|
	text = text.replace(/[^a-zA-Z0-9]/g,'_');
 | 
						|
 | 
						|
	return text
 | 
						|
}
 | 
						|
 | 
						|
$(".id").on("input", function() {
 | 
						|
	let text = $(this).val();
 | 
						|
	
 | 
						|
 | 
						|
	$(this).val( getValidId(text) );
 | 
						|
})
 | 
						|
 | 
						|
function componentToHex(c) {
 | 
						|
	var hex = c.toString(16);
 | 
						|
 | 
						|
	return hex.length == 1 ? "0" + hex : hex;
 | 
						|
  }
 | 
						|
  
 | 
						|
function rgbToHex(r, g, b) {
 | 
						|
	return "#" + componentToHex(r) + componentToHex(g) + componentToHex(b);
 | 
						|
}  
 | 
						|
 | 
						|
function hexToRgb(hex) {
 | 
						|
	var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
 | 
						|
 | 
						|
	return result ? {
 | 
						|
	  r: parseInt(result[1], 16),
 | 
						|
	  g: parseInt(result[2], 16),
 | 
						|
	  b: parseInt(result[3], 16)
 | 
						|
	} : null;
 | 
						|
}
 | 
						|
 | 
						|
function getVehicleLabel(vehicleModel, cb) {
 | 
						|
	$.post(`https://${resName}/get-vehicle-label`, JSON.stringify({vehicleModel: vehicleModel}), cb)
 | 
						|
}
 | 
						|
 | 
						|
// Disables enter
 | 
						|
$(document).on("keypress", 'form', function (e) {
 | 
						|
    var code = e.keyCode || e.which;
 | 
						|
    if (code == 13) {
 | 
						|
        e.preventDefault();
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
});
 | 
						|
 | 
						|
/* Dialogs */
 | 
						|
function inputDialog(title, data, yesCallback) {
 | 
						|
	$("#input-text-dialog-title").text(title);
 | 
						|
 | 
						|
	var inputModal = $("#input-text-dialog-modal");
 | 
						|
 | 
						|
	// Removes old data
 | 
						|
	inputModal.find(".modal-body form").empty();
 | 
						|
 | 
						|
	data.forEach(currentData => {
 | 
						|
		var id = currentData.id;
 | 
						|
		var label = currentData.label;
 | 
						|
		var defaultValue = currentData.default !== undefined ? currentData.default : "";
 | 
						|
 | 
						|
		var type = currentData.type || "text";
 | 
						|
 | 
						|
		if (type === "button") {
 | 
						|
			var button = $(`<button type="button" class="mt-3 btn ${currentData.buttonType}">${label}</button>`);
 | 
						|
 | 
						|
			button.click(function () {
 | 
						|
				currentData.buttonCallback(inputModal);
 | 
						|
			});
 | 
						|
 | 
						|
			inputModal.find(".modal-body form").append(button);
 | 
						|
		} else if (type === "select") {
 | 
						|
 | 
						|
			var select = $(`<select class="form-select" data-id="${id}" aria-label="Select"></select>`)
 | 
						|
			var options = currentData.options
 | 
						|
 | 
						|
			options.forEach(option => {
 | 
						|
				var optionElement = $(`<option value="${option.value}">${option.label}</option>`)
 | 
						|
				select.append(optionElement);
 | 
						|
			})
 | 
						|
 | 
						|
			inputModal.find(".modal-body form").append(select);
 | 
						|
		} else if (type === "switch") {
 | 
						|
 | 
						|
			var switchInput = $(`
 | 
						|
			<div class="form-check form-switch mt-3">
 | 
						|
				<label class="form-check-label" for="modal-${id}">${label}</label>
 | 
						|
				<input class="form-check-input" type="checkbox" id="modal-${id}" data-id="${id}">
 | 
						|
			</div>`);
 | 
						|
 | 
						|
			inputModal.find(".modal-body form").append(switchInput);
 | 
						|
 | 
						|
			$(`#modal-${id}`).prop("checked", defaultValue)
 | 
						|
		} else {
 | 
						|
			let isAnId = currentData.isId || false
 | 
						|
			var formGroup = $(
 | 
						|
				`<div class="form-group">
 | 
						|
					<label for="modal-${id}" class="col-form-label">${label}</label>
 | 
						|
					<input id="modal-${id}" type="${type}" class="form-control" data-id="${id}" value="${defaultValue}">
 | 
						|
					<div class="invalid-feedback">Can't be empty</div>
 | 
						|
				</div>`
 | 
						|
			)
 | 
						|
 | 
						|
			inputModal.find(".modal-body form").append(formGroup)
 | 
						|
 | 
						|
			let inputDiv = $(`#modal-${id}`);
 | 
						|
 | 
						|
			inputDiv.unbind('input');
 | 
						|
 | 
						|
			if(isAnId) {
 | 
						|
				inputDiv.on('input', function() {
 | 
						|
					let text = inputDiv.val();
 | 
						|
					text = text.toLowerCase();
 | 
						|
					text = text.replace(/[^a-zA-Z0-9]/g,'_');
 | 
						|
 | 
						|
					inputDiv.val(text);
 | 
						|
				});
 | 
						|
			}
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	inputModal.modal("show");
 | 
						|
 | 
						|
	$("#input-text-dialog-confirm").off('click');
 | 
						|
	$("#input-text-dialog-confirm").click(function () {
 | 
						|
		var areInputsValid = true;
 | 
						|
 | 
						|
		var inputs = inputModal.find(".form-control, .form-select, :checkbox");
 | 
						|
		var length = inputs.length;
 | 
						|
 | 
						|
		var data = {};
 | 
						|
 | 
						|
		inputs.each(function (index, element) {
 | 
						|
			var text = $(element).val();
 | 
						|
			if (text || $(element).is(":checkbox")) {
 | 
						|
				$(element).removeClass("is-invalid");
 | 
						|
				var id = $(element).data("id");
 | 
						|
 | 
						|
				if ($(element).is(":checkbox")) {
 | 
						|
					data[id] = $(element).is(':checked');
 | 
						|
				} else {
 | 
						|
					data[id] = text;
 | 
						|
				}
 | 
						|
			} else {
 | 
						|
				$(element).addClass("is-invalid");
 | 
						|
				areInputsValid = false;
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		inputModal.modal("hide");
 | 
						|
		yesCallback(data);
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function showError(msg){
 | 
						|
	$("#notification-message").text(msg);
 | 
						|
	$("#notification-modal").modal("show");
 | 
						|
}
 | 
						|
 | 
						|
function accountsDialog(cb) {
 | 
						|
	let accountsModal = $("#input-accounts-dialog-modal");
 | 
						|
	let accountsList = $("#accounts-list");
 | 
						|
 | 
						|
	$.post(`https://${resName}/getAllAccounts`, {}, function (accounts) {
 | 
						|
		accountsList.empty();
 | 
						|
 | 
						|
		for(const[accountName, accountLabel] of Object.entries(accounts)) {
 | 
						|
			let accountDiv = $(`
 | 
						|
				<li class="list-group-item list-group-item-action" value=${accountName}>${accountLabel}</li>
 | 
						|
			`);
 | 
						|
 | 
						|
			accountDiv.click(function() {
 | 
						|
				accountsModal.modal("hide");
 | 
						|
				cb(accountName);
 | 
						|
			});
 | 
						|
 | 
						|
			accountsList.append(accountDiv);
 | 
						|
		}
 | 
						|
 | 
						|
		accountsModal.modal("show");
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
async function getCurrentCoordsAndHeading() {
 | 
						|
	return new Promise((resolve, reject) => {
 | 
						|
		$.post(`https://${resName}/get-current-coords`, {}, function (coords) {
 | 
						|
			resolve(coords);
 | 
						|
		});
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
██████╗░░█████╗░███╗░░██╗██╗░░██╗░██████╗
 | 
						|
██╔══██╗██╔══██╗████╗░██║██║░██╔╝██╔════╝
 | 
						|
██████╔╝███████║██╔██╗██║█████═╝░╚█████╗░
 | 
						|
██╔══██╗██╔══██║██║╚████║██╔═██╗░░╚═══██╗
 | 
						|
██║░░██║██║░░██║██║░╚███║██║░╚██╗██████╔╝
 | 
						|
╚═╝░░╚═╝╚═╝░░╚═╝╚═╝░░╚══╝╚═╝░░╚═╝╚═════╝░
 | 
						|
*/
 | 
						|
let ranksDataTable = $("#ranks-container").DataTable( {
 | 
						|
	"lengthMenu": [5, 10, 15, 20],
 | 
						|
	"pageLength": 10,
 | 
						|
	"createdRow": function ( row, data, index ) {
 | 
						|
		$(row).addClass("clickable");
 | 
						|
 | 
						|
		$(row).click(function() {
 | 
						|
			let editRankModal = $("#edit-rank");
 | 
						|
 | 
						|
			let rankId = data[0];
 | 
						|
			let rankLabel = data[1];
 | 
						|
			let rankName = data[2];
 | 
						|
			let rankGrade = data[3];
 | 
						|
			let rankSalary = data[4];
 | 
						|
 | 
						|
			$("#edit-rank-label").val(rankLabel);
 | 
						|
			$("#edit-rank-name").val(rankName);
 | 
						|
			$("#edit-rank-grade").val(rankGrade);
 | 
						|
			$("#edit-rank-salary").val(rankSalary);
 | 
						|
 | 
						|
			editRankModal.data("rankId", rankId);
 | 
						|
 | 
						|
			editRankModal.modal("show");
 | 
						|
		})
 | 
						|
	},
 | 
						|
	"columnDefs": [{"defaultContent": "???", "targets": "_all"}]
 | 
						|
} );
 | 
						|
 | 
						|
async function deleteRank(jobName, rankId) {
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:do_you_want_to_delete_the_rank"))) return;
 | 
						|
 | 
						|
	const data = await $.post(`https://${resName}/delete-rank`, JSON.stringify({ rankId: rankId }));
 | 
						|
 | 
						|
	// Removes deleted row
 | 
						|
	if (data.isSuccessful) {
 | 
						|
		refreshJobRanks(jobName)
 | 
						|
	} else {
 | 
						|
		showError(data.message)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
function fillJobRanks(ranks) {
 | 
						|
	ranksDataTable.clear();
 | 
						|
 | 
						|
	for(const[_, gradeData] of Object.entries(ranks)) {
 | 
						|
		ranksDataTable.row.add( [gradeData.id, gradeData.label, gradeData.name, gradeData.grade, gradeData.salary] );
 | 
						|
	}
 | 
						|
 | 
						|
	ranksDataTable.draw();
 | 
						|
}
 | 
						|
 | 
						|
$("#edit-rank-form").submit(function(event) {
 | 
						|
	if (!this.checkValidity()) {
 | 
						|
		event.preventDefault();
 | 
						|
		event.stopPropagation();
 | 
						|
 | 
						|
		return;
 | 
						|
	} else {
 | 
						|
		$(this).removeClass("was-validated");
 | 
						|
		$("#create-rank").modal("hide");
 | 
						|
	}
 | 
						|
	
 | 
						|
	let data = {
 | 
						|
		rankLabel: $("#edit-rank-label").val(),
 | 
						|
		rankName: $("#edit-rank-name").val(),
 | 
						|
		rankGrade: parseInt( $("#edit-rank-grade").val() ),
 | 
						|
		rankSalary: parseInt( $("#edit-rank-salary").val() ),
 | 
						|
		rankId: $("#edit-rank").data("rankId")
 | 
						|
	}
 | 
						|
 | 
						|
	$.post(`https://${resName}/updateRank`, JSON.stringify(data), function (resultData) {
 | 
						|
		// Updates old row field with new values
 | 
						|
		if (resultData.isSuccessful) {
 | 
						|
			refreshJobRanks( $("#edit-job").data("jobName") );
 | 
						|
		} else {
 | 
						|
			showError(resultData.message)
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	$("#edit-rank").modal("hide");
 | 
						|
})
 | 
						|
 | 
						|
$("#edit-rank-delete-btn").click(function() {
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
	let rankId = $("#edit-rank").data("rankId");
 | 
						|
 | 
						|
	deleteRank(jobName, rankId);
 | 
						|
})
 | 
						|
 | 
						|
$("#create-rank-btn").click(function() {
 | 
						|
	// Reset previous inputs
 | 
						|
	$("#create-rank-label").val("");
 | 
						|
	$("#create-rank-name").val("");
 | 
						|
	$("#create-rank-grade").val("");
 | 
						|
	$("#create-rank-salary").val("");
 | 
						|
 | 
						|
	$("#create-rank").modal("show");
 | 
						|
});
 | 
						|
 | 
						|
// Automatically fills the id from the label
 | 
						|
$("#create-rank-label").on("input", function(){
 | 
						|
	let label = $(this).val();
 | 
						|
 | 
						|
	let id = getValidId(label);
 | 
						|
 | 
						|
	$("#create-rank-name").val(id);
 | 
						|
})
 | 
						|
 | 
						|
// Automatically fills the id from the label
 | 
						|
$("#edit-rank-label").on("input", function(){
 | 
						|
	let label = $(this).val();
 | 
						|
 | 
						|
	let id = getValidId(label);
 | 
						|
 | 
						|
	$("#edit-rank-name").val(id);
 | 
						|
})
 | 
						|
 | 
						|
$("#create-rank-form").submit(function(event) {
 | 
						|
	if(isThereAnyErrorInForm(event)) return;
 | 
						|
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
 | 
						|
	let data = {
 | 
						|
		rankLabel: $("#create-rank-label").val(),
 | 
						|
		rankName: $("#create-rank-name").val(),
 | 
						|
		rankGrade: parseInt( $("#create-rank-grade").val() ),
 | 
						|
		rankSalary: parseInt( $("#create-rank-salary").val() ),
 | 
						|
		jobName: jobName
 | 
						|
	}
 | 
						|
 | 
						|
	$.post(`https://${resName}/create-new-rank`, JSON.stringify(data), function (data) {
 | 
						|
		if (data.isSuccessful) {
 | 
						|
			refreshJobRanks(jobName)
 | 
						|
		} else {
 | 
						|
			showError(data.message)
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	$("#create-rank").modal("hide");
 | 
						|
})
 | 
						|
 | 
						|
function refreshJobRanks(jobName) {
 | 
						|
	$.post(`https://${resName}/retrieveJobRanks`, JSON.stringify({ jobName: jobName }), function (ranks) {
 | 
						|
		if (ranks) {
 | 
						|
			fillJobRanks(ranks)
 | 
						|
		}
 | 
						|
	});
 | 
						|
}
 | 
						|
/*
 | 
						|
░░░░░██╗░█████╗░██████╗░░██████╗
 | 
						|
░░░░░██║██╔══██╗██╔══██╗██╔════╝
 | 
						|
░░░░░██║██║░░██║██████╦╝╚█████╗░
 | 
						|
██╗░░██║██║░░██║██╔══██╗░╚═══██╗
 | 
						|
╚█████╔╝╚█████╔╝██████╦╝██████╔╝
 | 
						|
░╚════╝░░╚════╝░╚═════╝░╚═════╝░
 | 
						|
*/
 | 
						|
var allJobs = {};
 | 
						|
 | 
						|
async function refresh() {
 | 
						|
	fillPublicMarkers();
 | 
						|
 | 
						|
	const jobs = await $.post(`https://${resName}/getJobsData`, {});
 | 
						|
 | 
						|
	allJobs = jobs;
 | 
						|
		
 | 
						|
	fillJobs(jobs);
 | 
						|
}
 | 
						|
 | 
						|
$("#create-job-btn").click(function () {
 | 
						|
	inputDialog(getLocalizedText("menu:create_new_job"), [
 | 
						|
		{ id: "jobLabel", label: getLocalizedText("menu:dynamic:new_job_label") },
 | 
						|
		{ id: "jobName", label: getLocalizedText("menu:dynamic:new_job_name"), isId: true },
 | 
						|
	], function (data) {
 | 
						|
		$.post(`https://${resName}/create-new-job`, JSON.stringify({ jobLabel: data.jobLabel, jobName: data.jobName }), function (data) {
 | 
						|
			if (data.isSuccessful) {
 | 
						|
				refresh()
 | 
						|
			} else {
 | 
						|
				showError(data.message)
 | 
						|
			}
 | 
						|
		});
 | 
						|
	});
 | 
						|
})
 | 
						|
 | 
						|
function exitFromEditJob(){
 | 
						|
	// Resets current active tab (ranks is the default one)
 | 
						|
	$("#edit-job").find(".nav-link, .tab-pane").each(function() {
 | 
						|
		if($(this).data("isDefault") == "1") {
 | 
						|
			$(this).addClass(["active", "show"])
 | 
						|
		} else {
 | 
						|
			$(this).removeClass(["active", "show"])
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	refresh();
 | 
						|
 | 
						|
	$("#edit-job").hide();
 | 
						|
	$("#job-creator").show();
 | 
						|
}
 | 
						|
 | 
						|
async function deleteJob(jobName) {
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:dynamic:delete_job_confirm"))) return;
 | 
						|
	
 | 
						|
	const data = await $.post(`https://${resName}/delete-job`, JSON.stringify({ jobName: jobName }));
 | 
						|
 | 
						|
	data.isSuccessful ? exitFromEditJob() : showError(data.message);
 | 
						|
}
 | 
						|
 | 
						|
$("#delete-job-btn").click(function() {
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
	deleteJob(jobName);
 | 
						|
})
 | 
						|
 | 
						|
$("#edit-job-label").on("input", function(){
 | 
						|
	let label = $(this).val();
 | 
						|
 | 
						|
	let id = getValidId(label);
 | 
						|
 | 
						|
	$("#edit-job-name").val(id);
 | 
						|
})
 | 
						|
 | 
						|
function editJob(jobData) {
 | 
						|
	let editJobDiv = $("#edit-job");
 | 
						|
 | 
						|
	editJobDiv.data("jobName", jobData.name);
 | 
						|
 | 
						|
	// Chart with ranks 
 | 
						|
	loadRanksDistrubutions();
 | 
						|
 | 
						|
	// Job settings
 | 
						|
	$("#edit-job-label").val(jobData.label);
 | 
						|
	$("#edit-job-name").val(jobData.name);
 | 
						|
 | 
						|
	// Fills actions in settings
 | 
						|
	if(jobData.actions) {
 | 
						|
		for(const[actionName, isEnabled] of Object.entries(jobData.actions)) {
 | 
						|
			$("#job-settings-actions").find(`[data-action-name=${actionName}]`).prop("checked", isEnabled)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// Fill placeable objects
 | 
						|
	$("#job-placeable-objects-list").empty();
 | 
						|
	$("#job-placeable-objects-no-objects-text").show(); // It will be hidden if there are objects
 | 
						|
 | 
						|
	for(const [model, objectData] of Object.entries(jobData.actions?.placeableObjects || {})) {
 | 
						|
		addPlaceableObject(model, objectData.label, objectData.limit);
 | 
						|
	}
 | 
						|
 | 
						|
	$("#edit-job-text").text(jobData.label);
 | 
						|
 | 
						|
	fillJobRanks(jobData.ranks);
 | 
						|
	fillJobMarkers();
 | 
						|
 | 
						|
	$("#job-creator").hide();
 | 
						|
	editJobDiv.show();
 | 
						|
 | 
						|
	jobTour();
 | 
						|
}
 | 
						|
 | 
						|
$("#edit-job-close-btn").click(exitFromEditJob);
 | 
						|
 | 
						|
function addPlaceableObject(model, label, limit) {
 | 
						|
	const div = $(`
 | 
						|
		<div class="row g-2 row-cols-auto align-items-center justify-content-center my-2 placeable-object">
 | 
						|
			<button type="button" class="btn btn-danger delete-object-btn me-4" ><i class="bi bi-trash-fill"></i></button>	
 | 
						|
 | 
						|
			<div class="form-floating col-3">
 | 
						|
				<input type="text" class="form-control object-model" placeholder="Model" required>
 | 
						|
				<label>${ getLocalizedText("menu:object_model") }</label>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<div class="form-floating col-3">
 | 
						|
				<input type="text" class="form-control object-label" placeholder="label" required>
 | 
						|
				<label>${ getLocalizedText("menu:label") }</label>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<div class="form-floating col-2">
 | 
						|
				<input type="number" min="1" class="form-control object-limit" placeholder="Limit" required>
 | 
						|
				<label>${ getLocalizedText("menu:limit_per_player") }</label>
 | 
						|
			</div>
 | 
						|
		</div>
 | 
						|
	`);
 | 
						|
 | 
						|
	div.find(".delete-object-btn").click(function() {
 | 
						|
		div.remove();
 | 
						|
 | 
						|
		if($("#job-placeable-objects-list").children().length == 0) {
 | 
						|
			$("#job-placeable-objects-no-objects-text").show();
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	div.find(".object-model").val(model || "");
 | 
						|
	div.find(".object-label").val(label || "");
 | 
						|
	div.find(".object-limit").val(limit || 1);
 | 
						|
 | 
						|
	$("#job-placeable-objects-no-objects-text").hide();
 | 
						|
 | 
						|
	$("#job-placeable-objects-list").append(div);
 | 
						|
}
 | 
						|
 | 
						|
$("#job-placeable-objects-add-btn").click(function() {
 | 
						|
	addPlaceableObject();
 | 
						|
});
 | 
						|
 | 
						|
$("#job-settings").submit(function(event) {
 | 
						|
	if (!this.checkValidity()) {
 | 
						|
		event.stopPropagation();
 | 
						|
		event.preventDefault();
 | 
						|
		return;
 | 
						|
	} else {
 | 
						|
		$(this).removeClass("was-validated");
 | 
						|
	}
 | 
						|
 | 
						|
	let actions = {};
 | 
						|
 | 
						|
	$("#job-settings-actions").find(".job-action").each(function() {
 | 
						|
		let actionName = $(this).data("actionName");
 | 
						|
 | 
						|
		actions[actionName] = $(this).prop("checked");
 | 
						|
	})
 | 
						|
 | 
						|
	let placeableObjects = {};
 | 
						|
	
 | 
						|
	$("#job-placeable-objects-list").find(".placeable-object").each(function() {
 | 
						|
		let model = $(this).find(".object-model").val();
 | 
						|
 | 
						|
		placeableObjects[model] = {
 | 
						|
			label: $(this).find(".object-label").val(),
 | 
						|
			limit: parseInt( $(this).find(".object-limit").val() )
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	actions["placeableObjects"] = placeableObjects;
 | 
						|
 | 
						|
	let oldJobName = $("#edit-job").data("jobName");
 | 
						|
	let jobName = $("#edit-job-name").val();
 | 
						|
	let jobLabel = $("#edit-job-label").val();
 | 
						|
 | 
						|
	let data = {
 | 
						|
		oldJobName,
 | 
						|
		jobName,
 | 
						|
		jobLabel,
 | 
						|
		actions
 | 
						|
	}
 | 
						|
 | 
						|
	$.post(`https://${resName}/update-job`, JSON.stringify(data), function (resultData) {
 | 
						|
		// Updates old job with new values
 | 
						|
		if (resultData.isSuccessful) {
 | 
						|
			exitFromEditJob();
 | 
						|
		} else {
 | 
						|
			showError(resultData.message)
 | 
						|
		}
 | 
						|
	});
 | 
						|
})
 | 
						|
 | 
						|
let jobsDataTable = $("#jobs-container").DataTable( {
 | 
						|
	"lengthMenu": [5, 10, 15, 20],
 | 
						|
	"pageLength": 10,
 | 
						|
	"createdRow": function ( row, data, index ) {
 | 
						|
		$(row).addClass("clickable");
 | 
						|
 | 
						|
		$(row).click(function() {
 | 
						|
			let jobName = data[1];
 | 
						|
 | 
						|
			editJob(allJobs[jobName])
 | 
						|
		})
 | 
						|
	},
 | 
						|
	"columnDefs": [{"defaultContent": "???", "targets": "_all"}]
 | 
						|
} );
 | 
						|
 | 
						|
function fillJobs(jobs) {
 | 
						|
	jobsDataTable.clear();
 | 
						|
 | 
						|
	for (const [_, job] of Object.entries(jobs)) {
 | 
						|
		jobsDataTable.row.add( [job.label, job.name, Object.entries(job.ranks).length] );
 | 
						|
	}
 | 
						|
 | 
						|
	jobsDataTable.draw();
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
███╗░░░███╗░█████╗░██████╗░██╗░░██╗███████╗██████╗░░██████╗
 | 
						|
████╗░████║██╔══██╗██╔══██╗██║░██╔╝██╔════╝██╔══██╗██╔════╝
 | 
						|
██╔████╔██║███████║██████╔╝█████═╝░█████╗░░██████╔╝╚█████╗░
 | 
						|
██║╚██╔╝██║██╔══██║██╔══██╗██╔═██╗░██╔══╝░░██╔══██╗░╚═══██╗
 | 
						|
██║░╚═╝░██║██║░░██║██║░░██║██║░╚██╗███████╗██║░░██║██████╔╝
 | 
						|
╚═╝░░░░░╚═╝╚═╝░░╚═╝╚═╝░░╚═╝╚═╝░░╚═╝╚══════╝╚═╝░░╚═╝╚═════╝░
 | 
						|
*/
 | 
						|
let allMarkers = {};
 | 
						|
 | 
						|
let jobMarkersDataTable = $("#job-markers-container").DataTable( {
 | 
						|
	"lengthMenu": [5, 10, 15, 20],
 | 
						|
	"pageLength": 10,
 | 
						|
	"createdRow": function ( row, data, index ) {
 | 
						|
		$(row).addClass("clickable");
 | 
						|
 | 
						|
		$(row).click(function() {
 | 
						|
			let markerId = data[0];
 | 
						|
 | 
						|
			editMarker(markerId);
 | 
						|
		})
 | 
						|
	},
 | 
						|
	"columnDefs": [{"defaultContent": "???", "targets": "_all"}]
 | 
						|
} );
 | 
						|
 | 
						|
let publicMarkersDataTable = $("#public-markers-container").DataTable( {
 | 
						|
	"lengthMenu": [5, 10, 15, 20],
 | 
						|
	"pageLength": 10,
 | 
						|
	"createdRow": function ( row, data, index ) {
 | 
						|
		$(row).addClass("clickable");
 | 
						|
 | 
						|
		$(row).click(function() {
 | 
						|
			let markerId = data[0];
 | 
						|
 | 
						|
			editMarker(markerId);
 | 
						|
		})
 | 
						|
	},
 | 
						|
	"columnDefs": [{"defaultContent": "???", "targets": "_all"}]
 | 
						|
} );
 | 
						|
 | 
						|
function fillJobMarkers(markerToEdit) {
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
 | 
						|
	$.post(`https://${resName}/retrieveJobMarkers`, JSON.stringify({ jobName: jobName }), function (markers) {
 | 
						|
		allMarkers = {};
 | 
						|
 | 
						|
		// Manually insert each marker to avoid wrong conversion between lua table first index (1) and javascript object first index (0)
 | 
						|
		for(const[k, markerData] of Object.entries(markers)) {
 | 
						|
			allMarkers[markerData.id] = markerData;
 | 
						|
		}
 | 
						|
 | 
						|
		jobMarkersDataTable.clear();
 | 
						|
 | 
						|
		for(const[_, markerData] of Object.entries(markers)) {
 | 
						|
			let typeLabel = MARKERS_INFOS[markerData.type]?.label;
 | 
						|
			
 | 
						|
			jobMarkersDataTable.row.add( [markerData.id || "Unknown", markerData.label || "No name", typeLabel, markerData.coords.x, markerData.coords.y, markerData.coords.z] );
 | 
						|
		}
 | 
						|
	
 | 
						|
		jobMarkersDataTable.draw();
 | 
						|
 | 
						|
		if(markerToEdit) {
 | 
						|
			editMarker(markerToEdit);
 | 
						|
		}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function fillPublicMarkers(markerToEdit) {
 | 
						|
	$.post(`https://${resName}/retrieveJobMarkers`, JSON.stringify({ jobName: "public_marker" }), function (markers) {
 | 
						|
		allMarkers = {};
 | 
						|
 | 
						|
		// Manually insert each marker to avoid wrong conversion between lua table first index (1) and javascript object first index (0)
 | 
						|
		for(const[k, markerData] of Object.entries(markers)) {
 | 
						|
			allMarkers[markerData.id] = markerData;
 | 
						|
		}
 | 
						|
		
 | 
						|
		publicMarkersDataTable.clear();
 | 
						|
 | 
						|
		for(const[_, markerData] of Object.entries(markers)) {
 | 
						|
			let typeLabel = MARKERS_INFOS[markerData.type]?.label;
 | 
						|
			
 | 
						|
			publicMarkersDataTable.row.add( [markerData.id, markerData.label || "No name", typeLabel, markerData.coords.x, markerData.coords.y, markerData.coords.z] );
 | 
						|
		}
 | 
						|
	
 | 
						|
		publicMarkersDataTable.draw();
 | 
						|
 | 
						|
		if(markerToEdit) {
 | 
						|
			editMarker(markerToEdit);
 | 
						|
		}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function getOffDutyName(name) {
 | 
						|
    return "off_" + name;
 | 
						|
}
 | 
						|
 | 
						|
function getOnDutyName(name) {
 | 
						|
    return name.replace("off_", "");
 | 
						|
}
 | 
						|
 | 
						|
function isOffDutyName(name) {
 | 
						|
    return name.includes("off_");
 | 
						|
}
 | 
						|
 | 
						|
async function createJobMarker(jobName) {
 | 
						|
	let allowedMarkers = [];
 | 
						|
 | 
						|
	for(const[markerType, markerInfo] of Object.entries(MARKERS_INFOS)) {
 | 
						|
		if(markerInfo.framework != undefined && markerInfo.framework != await getFramework()) continue;
 | 
						|
		if(!markerInfo.isJobMarker) continue;
 | 
						|
 | 
						|
		allowedMarkers.push({
 | 
						|
			value: markerType,
 | 
						|
			label: markerInfo.label
 | 
						|
		})
 | 
						|
	}
 | 
						|
 | 
						|
	inputDialog(getLocalizedText("menu:dynamic:job_markers:create_marker"), [
 | 
						|
		{ id: "markerType", label: "Marker Type:", type: "select", options: allowedMarkers },
 | 
						|
		{ id: "label", label: getLocalizedText("menu:dynamic:job_markers:label"), type: "text", default: getLocalizedText("menu:dynamic:default") },
 | 
						|
		{ id: "markerCoordsX", label: getLocalizedText("menu:dynamic:job_markers:coord_x"), type: "number" },
 | 
						|
		{ id: "markerCoordsY", label: getLocalizedText("menu:dynamic:job_markers:coord_y"), type: "number" },
 | 
						|
		{ id: "markerCoordsZ", label: getLocalizedText("menu:dynamic:job_markers:coord_z"), type: "number" },
 | 
						|
		{
 | 
						|
			id: "currentCoordsBtn", label: getLocalizedText("menu:dynamic:job_markers:current_coords"), type: "button", buttonType: "btn-success",
 | 
						|
			buttonCallback: async function (modal) {
 | 
						|
				const coords = await placeEntity();
 | 
						|
				if(!coords) return;
 | 
						|
 | 
						|
				modal.find("#modal-markerCoordsX").val(coords.x)
 | 
						|
				modal.find("#modal-markerCoordsY").val(coords.y)
 | 
						|
				modal.find("#modal-markerCoordsZ").val(coords.z)
 | 
						|
			}
 | 
						|
		},
 | 
						|
		{ id: "markerMinGrade", label: getLocalizedText("menu:dynamic:job_markers:min_grade"), type: "number", default: 0 },
 | 
						|
	], async function (data) {
 | 
						|
		data.jobName = jobName
 | 
						|
 | 
						|
		const resultData = await $.post(`https://${resName}/create-marker`, JSON.stringify(data));
 | 
						|
 | 
						|
		if (resultData.isSuccessful) {
 | 
						|
			fillJobMarkers(resultData.markerId);
 | 
						|
 | 
						|
			if(!isOffDutyName(jobName) && await getFramework() == "ESX" && data.markerType == "duty") {
 | 
						|
				const duplicateForOffDuty = await swal({
 | 
						|
					title: getLocalizedText("menu:note"),
 | 
						|
					text: getLocalizedText("menu:do_you_want_to_create_it_also_for_off_duty_job"),
 | 
						|
					icon: "info",
 | 
						|
					buttons: true,
 | 
						|
					dangerMode: false,
 | 
						|
				});
 | 
						|
 | 
						|
				if(!duplicateForOffDuty) return;
 | 
						|
 | 
						|
				data.jobName = getOffDutyName(jobName);
 | 
						|
 | 
						|
				const resultData = await $.post(`https://${resName}/create-marker`, JSON.stringify(data));
 | 
						|
 | 
						|
				if (resultData.isSuccessful) {
 | 
						|
					fillJobMarkers(resultData.markerId);
 | 
						|
				} else {
 | 
						|
					showError(resultData.message)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			showError(resultData.message)
 | 
						|
		}
 | 
						|
	})
 | 
						|
}
 | 
						|
$("#create-job-marker-btn").click(function() {
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
 | 
						|
	createJobMarker(jobName);
 | 
						|
})
 | 
						|
 | 
						|
async function createPublicMarker() {
 | 
						|
	let allowedMarkers = [];
 | 
						|
 | 
						|
	for(const[markerType, markerInfo] of Object.entries(MARKERS_INFOS)) {
 | 
						|
		if(markerInfo.framework == undefined || markerInfo.framework == await getFramework()) {
 | 
						|
			if(markerInfo.isPublicMarker) {
 | 
						|
				allowedMarkers.push({
 | 
						|
					value: markerType,
 | 
						|
					label: markerInfo.label
 | 
						|
				})
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	inputDialog(getLocalizedText("menu:dynamic:public_markers:create_public_marker"), [
 | 
						|
		{ id: "markerType", label: "Marker Type:", type: "select", options: allowedMarkers },
 | 
						|
		{ id: "label", label: getLocalizedText("menu:dynamic:public_markers:label"), type: "text", default: getLocalizedText("menu:dynamic:default") },
 | 
						|
		{ id: "markerCoordsX", label: getLocalizedText("menu:dynamic:public_markers:coord_x"), type: "number" },
 | 
						|
		{ id: "markerCoordsY", label: getLocalizedText("menu:dynamic:public_markers:coord_y"), type: "number" },
 | 
						|
		{ id: "markerCoordsZ", label: getLocalizedText("menu:dynamic:public_markers:coord_z"), type: "number" },
 | 
						|
		{
 | 
						|
			id: "currentCoordsBtn", label: getLocalizedText("menu:dynamic:public_markers:current_coords"), type: "button", buttonType: "btn-success",
 | 
						|
			buttonCallback: async function (modal) {
 | 
						|
				const coords = await placeEntity();
 | 
						|
				if(!coords) return;
 | 
						|
 | 
						|
				modal.find("#modal-markerCoordsX").val(coords.x)
 | 
						|
				modal.find("#modal-markerCoordsY").val(coords.y)
 | 
						|
				modal.find("#modal-markerCoordsZ").val(coords.z)
 | 
						|
			}
 | 
						|
		},
 | 
						|
	], function (data) {
 | 
						|
		$.post(`https://${resName}/create-public-marker`, JSON.stringify(data), function (data) {
 | 
						|
			if (data.isSuccessful) {
 | 
						|
				fillPublicMarkers(data.markerId);
 | 
						|
			} else {
 | 
						|
				showError(data.message)
 | 
						|
			}
 | 
						|
		});
 | 
						|
	})
 | 
						|
}
 | 
						|
$("#create-public-marker-btn").click(createPublicMarker);
 | 
						|
 | 
						|
function hideAllMarkerSettings() {
 | 
						|
	$("#edit-marker-form").find(".edit-marker-settings").each(function() {
 | 
						|
		$(this).find(".form-control").each(function(index, element) {
 | 
						|
			if($(element).prop("required")) {
 | 
						|
				$(element).prop("required", false);
 | 
						|
				$(element).data("isActuallyRequired", true);
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		$(this).hide();
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function showMarkerSettings(markerType) {
 | 
						|
	hideAllMarkerSettings();
 | 
						|
 | 
						|
	$("#edit-marker-form").find(`.edit-marker-settings`).each(function() {
 | 
						|
		if( $(this).data("markerType") != markerType ) return;
 | 
						|
 | 
						|
		$(this).find(".form-control").each(function(index, element) {
 | 
						|
			if( $(element).data("isActuallyRequired") ) {
 | 
						|
				$(element).prop("required", true);
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		$(this).show();
 | 
						|
 | 
						|
		return;
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function editMarker(markerId) {
 | 
						|
	$("#edit-marker-form").removeClass("was-validated");
 | 
						|
 | 
						|
	let markerData = allMarkers[markerId];
 | 
						|
 | 
						|
	if(!markerData) {
 | 
						|
		console.log("Couldn't find marker " + markerId);
 | 
						|
		
 | 
						|
		console.log(`Existing markers for job '${ $("#edit-job").data("jobName") }': `)
 | 
						|
 | 
						|
		for(const[key, value] of Object.entries(allMarkers)) {
 | 
						|
			console.log(`	Marker ID '${key}'`);
 | 
						|
		}
 | 
						|
 | 
						|
		
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	let isMarkerPublic = markerData.jobName == "public_marker";
 | 
						|
	
 | 
						|
	$("#edit-marker-form").data("isPublic", isMarkerPublic);
 | 
						|
 | 
						|
	$("#edit-marker-label").val(markerData.label);
 | 
						|
	$("#edit-marker-type").val(markerData.markerType);
 | 
						|
 | 
						|
	changeMarkerGradeType(markerData.gradesType);
 | 
						|
 | 
						|
	if(markerData.gradesType === "specificGrades") {
 | 
						|
		setSpecificGrades(markerData.specificGrades);
 | 
						|
	} else if(markerData.gradesType === "minimumGrade") {
 | 
						|
		$("#edit-marker-min-grade").val(markerData.minGrade);
 | 
						|
	}
 | 
						|
 | 
						|
	$("#edit-marker-scale-x").val(markerData.scale.x);
 | 
						|
	$("#edit-marker-scale-y").val(markerData.scale.y);
 | 
						|
	$("#edit-marker-scale-z").val(markerData.scale.z);
 | 
						|
	
 | 
						|
	$("#edit-marker-coord-x").val(markerData.coords.x);
 | 
						|
	$("#edit-marker-coord-y").val(markerData.coords.y);
 | 
						|
	$("#edit-marker-coord-z").val(markerData.coords.z);
 | 
						|
	
 | 
						|
	$("#edit-marker-color").val(rgbToHex(markerData.color.r, markerData.color.g, markerData.color.b));
 | 
						|
	$("#edit-marker-alpha").val(markerData.color.alpha);
 | 
						|
	
 | 
						|
	// Map blip
 | 
						|
	$("#edit-marker-sprite-id").val(markerData.blip?.spriteId);
 | 
						|
	$("#edit-marker-blip-color").val(markerData.blip?.color);
 | 
						|
	$("#edit-marker-blip-scale").val(markerData.blip?.scale);
 | 
						|
	toggleBlipInputs(markerData.blip?.spriteId ? true : false);
 | 
						|
 | 
						|
	// NPC
 | 
						|
	$("#edit-marker-npc-model").val(markerData.ped?.model);
 | 
						|
	$("#edit-marker-npc-heading").val(markerData.ped?.heading);
 | 
						|
	toggleNpcInputs(markerData.ped?.model ? true : false);
 | 
						|
	
 | 
						|
	// Object
 | 
						|
	$("#edit-marker-object-model").val(markerData.object?.model);
 | 
						|
	$("#edit-marker-object-heading").val(markerData.object?.heading);
 | 
						|
	toggleObjectInputs(markerData.object?.model ? true : false);
 | 
						|
 | 
						|
	// Minimum grade / specific grade inputs
 | 
						|
	$("#edit-marker-min-grade").prop("disabled", isMarkerPublic);
 | 
						|
	$("input[name=markerGradeType]").prop("disabled", isMarkerPublic);
 | 
						|
	$("#edit-marker-min-grade").val(markerData.minGrade || 0);
 | 
						|
 | 
						|
	let settingsLabel = MARKERS_INFOS[markerData.type].settingsName;
 | 
						|
 | 
						|
	if(settingsLabel) {
 | 
						|
		$("#specific-marker-settings").show();
 | 
						|
		$("#specific-marker-settings-label").text(settingsLabel);
 | 
						|
		
 | 
						|
		showMarkerSettings(markerData.type);
 | 
						|
	} else {
 | 
						|
		$("#specific-marker-settings").hide();
 | 
						|
		hideAllMarkerSettings();
 | 
						|
	}
 | 
						|
 | 
						|
	setCurrentMarkerData(markerData.type, markerData.data);
 | 
						|
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
 | 
						|
	editMarkerModal.modal("show");
 | 
						|
	editMarkerModal.data("markerId", markerId);
 | 
						|
}
 | 
						|
 | 
						|
function changeMarkerGradeType(type) {
 | 
						|
	$(`input[type=radio][name="markerGradeType"][value="${type}"]`).prop("checked", true);
 | 
						|
 | 
						|
	switch(type) {
 | 
						|
		case "minimumGrade": {
 | 
						|
			$("#edit-marker-min-grade").prop("required", true);
 | 
						|
 | 
						|
			$("#edit-marker-min-grade-form").show();
 | 
						|
			$("#edit-marker-specific-grades-form").hide();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "specificGrades": {
 | 
						|
			$("#edit-marker-min-grade").prop("required", false);
 | 
						|
 | 
						|
			$("#edit-marker-min-grade-form").hide();
 | 
						|
			$("#edit-marker-specific-grades-form").show();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$("input[type=radio][name=markerGradeType").change(function() {
 | 
						|
	let gradesType = $(this).val();
 | 
						|
 | 
						|
	changeMarkerGradeType(gradesType);
 | 
						|
})
 | 
						|
 | 
						|
function addVehicleToTemporaryGarage(vehicleModel, vehicleData) {
 | 
						|
	vehicleData = vehicleData || {}
 | 
						|
 | 
						|
	let vehiclesTable = $("#garage-data-model-tbody")
 | 
						|
	
 | 
						|
	let vehiclePrimaryColor = vehicleData.primaryColor || "#111111"
 | 
						|
	let vehicleSecondaryColor = vehicleData.secondaryColor || "#111111"
 | 
						|
	let livery = vehicleData.livery;
 | 
						|
 | 
						|
	let customPlate = vehicleData.plate || "";
 | 
						|
 | 
						|
	let deleteVehicleBtn = $(`<button class="btn btn-outline-danger" type="button">${getLocalizedText("menu:dynamic:marker:temporary_garage:delete")}</button>`)
 | 
						|
 | 
						|
	let vehicleTableRow = $(`
 | 
						|
		<tr class="vehicle">
 | 
						|
			<th scope="row">
 | 
						|
				<p class="text-center vehicle-model">${vehicleModel}</p>
 | 
						|
			</th>
 | 
						|
			<td>
 | 
						|
				<p class="text-center vehicle-label">${vehicleModel}</p>
 | 
						|
			</td>
 | 
						|
			<td>
 | 
						|
				<input type="color" class="form-control form-control-color vehicle-primary-color" value="${vehiclePrimaryColor}" title="Primary Color">
 | 
						|
			</td>
 | 
						|
			<td>
 | 
						|
				<input type="color" class="form-control form-control-color vehicle-secondary-color" value="${vehicleSecondaryColor}" title="Secondary Color">
 | 
						|
			</td>
 | 
						|
			<td>
 | 
						|
				<input type="number" class="form-control vehicle-livery" value="${livery}" placeholder="N/A">
 | 
						|
			</td>
 | 
						|
			<td>
 | 
						|
				<input type="text" class="form-control vehicle-custom-plate" value="${customPlate}" placeholder="${getLocalizedText("menu:dynamic:marker:temporary_garage:random_plate")}" title="Custom plate">
 | 
						|
			</td>
 | 
						|
			<td class="delete-vehicle-btn"/>
 | 
						|
		</tr>
 | 
						|
	`)
 | 
						|
 | 
						|
	getVehicleLabel(vehicleModel, (vehicleLabel) => {
 | 
						|
		vehicleTableRow.find(".vehicle-label").text(vehicleLabel)
 | 
						|
	})
 | 
						|
 | 
						|
	vehiclesTable.append(vehicleTableRow);
 | 
						|
 | 
						|
	vehicleTableRow.find(".delete-vehicle-btn").append(deleteVehicleBtn)
 | 
						|
 | 
						|
	$(deleteVehicleBtn).click(function () {
 | 
						|
		$(this).parents(".vehicle").remove();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
$("#garage-owned-current-coords").click(function() {
 | 
						|
	$.post(`https://${resName}/get-current-coords`, {}, function (data) {
 | 
						|
		var coords = data.coords;
 | 
						|
		var heading = data.heading;
 | 
						|
 | 
						|
		if (coords) {
 | 
						|
			$("#garage-owned-spawnpoint-x").val(coords.x)
 | 
						|
			$("#garage-owned-spawnpoint-y").val(coords.y)
 | 
						|
			$("#garage-owned-spawnpoint-z").val(coords.z)
 | 
						|
		}
 | 
						|
 | 
						|
		if (heading) {
 | 
						|
			$("#garage-owned-heading").val(heading)
 | 
						|
		}
 | 
						|
	})
 | 
						|
});
 | 
						|
 | 
						|
function toggleBlipInputs(isEnabled) {
 | 
						|
	$("#edit-marker-map-blip").prop("checked", isEnabled);
 | 
						|
 | 
						|
	$("#edit-marker-map-blip-inputs").find(".form-control").each(function() {
 | 
						|
		$(this).prop("disabled", !isEnabled);
 | 
						|
		$(this).prop("required", isEnabled);
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function toggleNpcInputs(isEnabled) {
 | 
						|
	$("#edit-marker-npc").prop("checked", isEnabled);
 | 
						|
 | 
						|
	$("#edit-marker-npc-inputs").find(".form-control").each(function() {
 | 
						|
		$(this).prop("disabled", !isEnabled);
 | 
						|
		$(this).prop("required", isEnabled);
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
function toggleObjectInputs(isEnabled) {
 | 
						|
	$("#edit-marker-object").prop("checked", isEnabled);
 | 
						|
 | 
						|
	$("#edit-marker-object-inputs").find(".form-control").each(function() {
 | 
						|
		$(this).prop("disabled", !isEnabled);
 | 
						|
		$(this).prop("required", isEnabled);
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// Boss marker settings
 | 
						|
function toggleBossWashMoneyOptions(isEnabled) {
 | 
						|
	$("#boss-data-washmoney-percentage").prop("required", isEnabled);
 | 
						|
	$("#boss-data-washmoney-percentage").prop("disabled", !isEnabled);
 | 
						|
	$("#boss-data-washmoney-to-society-account").prop("disabled", !isEnabled);
 | 
						|
}
 | 
						|
 | 
						|
$("#boss-data-washmoney").change(function() {
 | 
						|
	toggleBossWashMoneyOptions( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
 | 
						|
// Harvestable items settings
 | 
						|
$("#harvest-item-requires-tool").change(function() {
 | 
						|
	const isEnabled = $(this).prop("checked");
 | 
						|
	$("#harvest-item-tool-div").find("input, button").prop("required", isEnabled).prop("disabled", !isEnabled);
 | 
						|
 | 
						|
	$("#harvest-item-tool-lose-on-use").prop("disabled", !isEnabled);
 | 
						|
 | 
						|
	if(!isEnabled) {
 | 
						|
		$("#harvest-item-tool-lose-on-use").prop("checked", false).change();
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#harvest-item-tool-lose-on-use").change(function() {
 | 
						|
	const isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	$("#harvest-item-tool-lose-quantity")
 | 
						|
		.prop("required", isEnabled)
 | 
						|
		.prop("disabled", !isEnabled);
 | 
						|
 | 
						|
	$("#harvest-item-tool-lose-probability")
 | 
						|
		.prop("required", isEnabled)
 | 
						|
		.prop("disabled", !isEnabled);
 | 
						|
})
 | 
						|
 | 
						|
$("#harvest-disappear-after-use").change(function() {
 | 
						|
	toggleDisappearsAfterUse( $(this).prop("checked") );
 | 
						|
	$("#harvest-disappear-seconds")
 | 
						|
		.prop("required", isEnabled)
 | 
						|
		.prop("disabled", !isEnabled);
 | 
						|
})
 | 
						|
 | 
						|
function toggleDisappearsAfterUse(isEnabled) {
 | 
						|
	$("#harvest-disappear-after-use").prop("checked", isEnabled);
 | 
						|
 | 
						|
	$("#harvest-disappear-seconds").prop("required", isEnabled);
 | 
						|
	$("#harvest-disappear-seconds").prop("disabled", !isEnabled);
 | 
						|
}
 | 
						|
 | 
						|
$("#harvest-requires-minimum-account-money").change(function() {
 | 
						|
	let isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	$("#harvest-minimum-account-name-div").find("input, button").prop("disabled", !isEnabled);
 | 
						|
	$("#harvest-minimum-account-name-div").find("input, button").prop("required", isEnabled);
 | 
						|
});
 | 
						|
 | 
						|
// Marker generic settings
 | 
						|
$("#edit-marker-map-blip").change(function(){
 | 
						|
	let isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	toggleBlipInputs(isEnabled);
 | 
						|
})
 | 
						|
 | 
						|
$("#edit-marker-npc").change(function(){
 | 
						|
	let isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	toggleNpcInputs(isEnabled);
 | 
						|
})
 | 
						|
 | 
						|
$("#edit-marker-object").change(function(){
 | 
						|
	let isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	toggleObjectInputs(isEnabled);
 | 
						|
})
 | 
						|
 | 
						|
$("#edit-marker-current-coords-btn").click(async function(){
 | 
						|
	const npcModel = $("#edit-marker-npc").prop("checked") ? $("#edit-marker-npc-model").val() : null;
 | 
						|
	const objectModel = $("#edit-marker-object") ? $("#edit-marker-object-model").val() : null;
 | 
						|
 | 
						|
	if(npcModel) {
 | 
						|
		var data = await placeEntity(npcModel, "ped");
 | 
						|
	} else if(objectModel) {	
 | 
						|
		var data = await placeEntity(objectModel, "object");
 | 
						|
	} else {
 | 
						|
		var data = await placeEntity();
 | 
						|
	}
 | 
						|
	
 | 
						|
	if(!data) return;
 | 
						|
 | 
						|
	if(npcModel || objectModel) {
 | 
						|
		$("#edit-marker-coord-x").val(data.coords.x)
 | 
						|
		$("#edit-marker-coord-y").val(data.coords.y)
 | 
						|
		$("#edit-marker-coord-z").val(data.coords.z)
 | 
						|
		
 | 
						|
		if(npcModel) 
 | 
						|
			$("#edit-marker-npc-heading").val(data.heading);
 | 
						|
 | 
						|
		if(objectModel) 
 | 
						|
			$("#edit-marker-object-heading").val(data.heading);
 | 
						|
 | 
						|
	} else {
 | 
						|
		$("#edit-marker-coord-x").val(data.x)
 | 
						|
		$("#edit-marker-coord-y").val(data.y)
 | 
						|
		$("#edit-marker-coord-z").val(data.z)
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#delete-marker-btn").click(async function(){
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
	let markerId = editMarkerModal.data("markerId");
 | 
						|
 | 
						|
	editMarkerModal.modal("hide");
 | 
						|
	if(!markerId) return;
 | 
						|
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:do_you_want_to_delete_this_marker"))) return;
 | 
						|
 | 
						|
	const data = await $.post(`https://${resName}/delete-marker`, JSON.stringify({ markerId: markerId }))
 | 
						|
 | 
						|
	if (!data.isSuccessful) {
 | 
						|
		showError(data.message);
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	if( $("#edit-marker-form").data("isPublic") ) {
 | 
						|
		fillPublicMarkers();
 | 
						|
	} else {
 | 
						|
		fillJobMarkers();
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
async function getOpenExternalClothingMenu() {
 | 
						|
	return await $.post(`https://${resName}/getOpenExternalClothingMenu`);
 | 
						|
}
 | 
						|
 | 
						|
async function getInventoryScriptUsed() {
 | 
						|
	return await $.post(`https://${resName}/getInventoryScriptUsed`);
 | 
						|
}
 | 
						|
 | 
						|
async function setCurrentMarkerData(markerType, markerData) {
 | 
						|
	// Resets animations from old opened marker
 | 
						|
	$("#edit-marker-dialog-modal").data("animations", []);
 | 
						|
	
 | 
						|
	markerData = markerData || {};
 | 
						|
	
 | 
						|
	switch(markerType) {
 | 
						|
		case "garage": {
 | 
						|
			let spawnpointsDiv = $("#temporary-garage-spawnpoints-list");
 | 
						|
			spawnpointsDiv.empty();
 | 
						|
			if (markerData.spawnPoints) {
 | 
						|
				markerData.spawnPoints.forEach(spawnPointData => {
 | 
						|
					addGarageSpawnpoint(spawnpointsDiv, spawnPointData);
 | 
						|
				})
 | 
						|
			}
 | 
						|
 | 
						|
			$("#garage-data-model-tbody").empty();
 | 
						|
 | 
						|
			if (markerData.vehicles) {
 | 
						|
				for (const [vehicleModel, vehicleData] of Object.entries(markerData.vehicles)) {
 | 
						|
					addVehicleToTemporaryGarage(vehicleModel, vehicleData)
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "garage_buyable": {
 | 
						|
			let spawnpointsDiv = $("#buyable-garage-spawnpoints-list");
 | 
						|
			spawnpointsDiv.empty();
 | 
						|
			if (markerData.spawnPoints) {
 | 
						|
				markerData.spawnPoints.forEach(spawnPointData => {
 | 
						|
					addGarageSpawnpoint(spawnpointsDiv, spawnPointData);
 | 
						|
				})
 | 
						|
			}
 | 
						|
 | 
						|
			$("#garage-buyable-vehicles").empty();
 | 
						|
 | 
						|
			if(markerData.vehicles) {
 | 
						|
				for (const [vehicleName, price] of Object.entries(markerData.vehicles)) {
 | 
						|
					var vehicleInputGroup = $(`<div class="input-group mt-1 vehicle"></div>`)
 | 
						|
 | 
						|
					var vehicleInput = $(`<input type="text" class="form-control vehicle-model" placeholder="Vehicle model" disabled value="${vehicleName}">`);
 | 
						|
					var moneySpan = $(`<span class="input-group-text">$</span>`)
 | 
						|
					var vehiclePrice = $(`<input type="text" class="form-control vehicle-price" placeholder="Vehicle price" disabled value="${price}">`);
 | 
						|
					var deleteVehicle = $(`<button class="btn btn-outline-danger" type="button">Delete</button>`);
 | 
						|
 | 
						|
					vehicleInputGroup.append(vehicleInput, moneySpan, vehiclePrice, deleteVehicle);
 | 
						|
 | 
						|
					$(deleteVehicle).click(function () {
 | 
						|
						$(this).parent().remove();
 | 
						|
					})
 | 
						|
 | 
						|
					$("#garage-buyable-vehicles").append(vehicleInputGroup)
 | 
						|
				}
 | 
						|
			}
 | 
						|
			
 | 
						|
			$("#garage-buyable-vehicle-model").val("");
 | 
						|
			$("#garage-buyable-vehicle-price").val("");
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "garage_owned": {
 | 
						|
			let spawnpointsDiv = $("#owned-garage-spawnpoints-list");
 | 
						|
			spawnpointsDiv.empty();
 | 
						|
			if (markerData.spawnPoints) {
 | 
						|
				markerData.spawnPoints.forEach(spawnPointData => {
 | 
						|
					addGarageSpawnpoint(spawnpointsDiv, spawnPointData);
 | 
						|
				})
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		
 | 
						|
		case "boss": {
 | 
						|
			let canWashMoney = markerData.canWashMoney;
 | 
						|
			let canWithdraw = markerData.canWithdraw;
 | 
						|
			let canDeposit = markerData.canDeposit;
 | 
						|
			let canEmployees = markerData.canEmployees;
 | 
						|
			let canGrades = markerData.canGrades;
 | 
						|
 | 
						|
			toggleBossWashMoneyOptions(canWashMoney);
 | 
						|
 | 
						|
			if(canWashMoney) {
 | 
						|
				$("#boss-data-washmoney-percentage").val(markerData.washMoneyReturnPercentage);
 | 
						|
				$("#boss-data-washmoney-to-society-account").prop("checked", markerData.washMoneyGoesToSocietyAccount);
 | 
						|
			}
 | 
						|
 | 
						|
			$("#boss-data-washmoney").prop('checked', canWashMoney);
 | 
						|
			$("#boss-data-withdraw").prop('checked', canWithdraw);
 | 
						|
			$("#boss-data-deposit").prop('checked', canDeposit);
 | 
						|
			$("#boss-data-employees").prop('checked', canEmployees);
 | 
						|
			$("#boss-data-grades").prop('checked', canGrades);
 | 
						|
			
 | 
						|
			$("#boss-data-max-salary").val(markerData.maxSalary);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "shop": {
 | 
						|
			setChoosenObject( $("#shop-new-item-name-div"), null );
 | 
						|
 | 
						|
			$("#shop-data-item-price").val("")
 | 
						|
			$('#shop-data-items-container').empty();
 | 
						|
 | 
						|
			if (markerData.itemsOnSale) {
 | 
						|
				markerData.itemsOnSale.forEach(itemData => {
 | 
						|
					addItemToShop(itemData.object, itemData)
 | 
						|
				})
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "market": {
 | 
						|
			$("#market-modal-items-tbody").empty();
 | 
						|
 | 
						|
			markerData?.items?.forEach(item => {
 | 
						|
				addItemToMarket(item.object, item.minPrice, item.maxPrice, item.blackMoney, item.sellTime);
 | 
						|
			})
 | 
						|
 | 
						|
			$("#market-modal-society-percentage").val(markerData.percentageForSociety);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "crafting_table": {
 | 
						|
			let craftablesDiv = $("#craftables");
 | 
						|
 | 
						|
			craftablesDiv.empty();
 | 
						|
 | 
						|
			if(markerData.craftablesItems) {
 | 
						|
				markerData.craftablesItems.forEach(craftableItem => {
 | 
						|
					addNewCraftableToCraftables(craftableItem.resultObject, craftableItem)
 | 
						|
				})
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "armory": {
 | 
						|
			$("#armory-is-shared").prop("checked", markerData.isShared);
 | 
						|
			$("#armory-refill-on-take").prop("checked", markerData.refillOnTake);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "job_outfit": {
 | 
						|
			$("#job-outfit-outfits").empty();
 | 
						|
 | 
						|
			if(!markerData.outfits) return;
 | 
						|
 | 
						|
			const openExternalClothingMenu = await getOpenExternalClothingMenu();
 | 
						|
 | 
						|
			if(openExternalClothingMenu) {
 | 
						|
				hideAllMarkerSettings();
 | 
						|
				$("#specific-marker-settings").hide();
 | 
						|
				
 | 
						|
				swal(getLocalizedText("menu:note"), getLocalizedText("menu:job_outfit_warning"), "info");
 | 
						|
				return;
 | 
						|
			}
 | 
						|
			
 | 
						|
			markerData.outfits.forEach(outfit => {
 | 
						|
				createNewOutfit(outfit)
 | 
						|
			})	
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "teleport": {
 | 
						|
			if(markerData.teleportCoords) {
 | 
						|
				$("#teleport-x").val(markerData.teleportCoords.x);
 | 
						|
				$("#teleport-y").val(markerData.teleportCoords.y);
 | 
						|
				$("#teleport-z").val(markerData.teleportCoords.z);
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "harvest": {
 | 
						|
			$("#harvest-modal-items").empty();
 | 
						|
			
 | 
						|
			if(markerData.harvestableItems) {
 | 
						|
				markerData.harvestableItems.forEach(item => {
 | 
						|
					addItemToHarvestable(item.object, item.minQuantity, item.maxQuantity, item.time, item.chances);
 | 
						|
				});
 | 
						|
			}
 | 
						|
			
 | 
						|
			if(markerData.animations) {
 | 
						|
				$("#edit-marker-dialog-modal").data("animations", markerData.animations)
 | 
						|
			}
 | 
						|
 | 
						|
			$("#harvest-item-requires-tool").prop("checked", markerData.itemTool != undefined).change();
 | 
						|
			setChoosenObject( $("#harvest-item-tool-div"), markerData.itemTool )
 | 
						|
 | 
						|
			$("#harvest-item-tool-lose-on-use").prop("checked", markerData.itemToolLoseQuantity).change();
 | 
						|
			$("#harvest-item-tool-lose-quantity").val(markerData.itemToolLoseQuantity);
 | 
						|
			$("#harvest-item-tool-lose-probability").val(markerData.itemToolLoseProbability);
 | 
						|
			
 | 
						|
			toggleDisappearsAfterUse(markerData.disappearAfterUse);
 | 
						|
			$("#harvest-disappear-seconds").val(markerData.disappearSeconds);
 | 
						|
 | 
						|
			$("#harvest-requires-minimum-account-money").prop("checked", markerData.requiresMinimumAccountMoney || false).change();
 | 
						|
			$("#harvest-minimum-account-name").val(markerData.minimumAccountName);
 | 
						|
			$("#harvest-minimum-account-amount").val(markerData.minimumAccountAmount);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "process": {
 | 
						|
			setChoosenObject( $("#process-modal-remove-item-div"), markerData.itemToRemove);
 | 
						|
			$("#item-to-remove-quantity").val(markerData?.itemToRemoveQuantity || 1);
 | 
						|
 | 
						|
			setChoosenObject( $("#process-modal-give-item-div"), markerData.itemToAdd);
 | 
						|
			$("#item-to-add-quantity").val(markerData?.itemToAddQuantity || 1);
 | 
						|
 | 
						|
			$("#process-time").val(markerData?.timeToProcess || 5);
 | 
						|
			$("#edit-marker-dialog-modal").data("animations", markerData?.animations)
 | 
						|
			$("#process-requires-minimum-account-money").prop("checked", markerData.requiresMinimumAccountMoney || false).change();
 | 
						|
			$("#process-minimum-account-name").val(markerData.minimumAccountName);
 | 
						|
			$("#process-minimum-account-amount").val(markerData.minimumAccountAmount);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "weapon_upgrader": {
 | 
						|
			if(markerData.componentsPrices) {
 | 
						|
				// set empty all, so disabled components will remain disabled
 | 
						|
				$("#weapon-upgrader-components").find(".form-control").each(function() {
 | 
						|
					$(this).val("");
 | 
						|
				})
 | 
						|
				
 | 
						|
				for(const [componentName, componentPrice] of Object.entries(markerData.componentsPrices)) {
 | 
						|
					$("#weapon-upgrader-components").find(`[data-componentname=${componentName}]`).val(componentPrice);
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			if(markerData.tintsPrices) {
 | 
						|
				// set empty all, so disabled tints will remain disabled
 | 
						|
				$("#weapon-upgrader-tints").find(".form-control").each(function() {
 | 
						|
					$(this).val("");
 | 
						|
				})
 | 
						|
 | 
						|
				for(const [tintId, tintPrice] of Object.entries(markerData.tintsPrices)) {
 | 
						|
					$("#weapon-upgrader-tints").find(`[data-tintid=${tintId}]`).val(tintPrice);
 | 
						|
				}
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "job_shop": {
 | 
						|
			let allJobsSelect = $("#job-shop-all-jobs");
 | 
						|
			
 | 
						|
			allJobsSelect.find(".job-shop-job").remove();
 | 
						|
 | 
						|
			const jobs = await $.post(`https://${resName}/getJobsData`);
 | 
						|
			let jobsRanks = {}
 | 
						|
 | 
						|
			for(const [_, jobData] of Object.entries(jobs)) {
 | 
						|
				if(jobData.name) {
 | 
						|
					let jobDiv = $(`<option class="job-shop-job" value="${jobData.name}">${jobData.label}</option>`);
 | 
						|
 | 
						|
					jobsRanks[jobData.name] = jobData.ranks
 | 
						|
					allJobsSelect.append(jobDiv);
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			allJobsSelect.data("ranks", jobsRanks);
 | 
						|
			
 | 
						|
			$("#job-shop-all-jobs").val(markerData.allowedJob).change();
 | 
						|
			$("#job-shop-all-ranks").val(markerData.minimumRank);
 | 
						|
 | 
						|
			break;
 | 
						|
		} 
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
function getCurrentMarkerData(markerType) {
 | 
						|
	let markerData = {};
 | 
						|
 | 
						|
	switch(markerType) {
 | 
						|
		case "garage": {		
 | 
						|
			let vehicles = {};
 | 
						|
			
 | 
						|
			let vehiclesObject = $("#garage-data-model-tbody").children(".vehicle");
 | 
						|
		
 | 
						|
			vehiclesObject.each(function (index, element) {
 | 
						|
				let vehicleModel = $(element).find(".vehicle-model").text();
 | 
						|
				let vehiclePrimaryColor = $(element).find(".vehicle-primary-color").val();
 | 
						|
				let vehicleSecondaryColor = $(element).find(".vehicle-secondary-color").val();
 | 
						|
				let vehicleLivery = parseInt( $(element).find(".vehicle-livery").val() );
 | 
						|
				let vehiclePlate = $(element).find(".vehicle-custom-plate").val();
 | 
						|
		
 | 
						|
				vehicles[vehicleModel] = {
 | 
						|
					primaryColor: vehiclePrimaryColor,
 | 
						|
					secondaryColor: vehicleSecondaryColor,
 | 
						|
					livery: vehicleLivery || null,
 | 
						|
					plate: vehiclePlate || null
 | 
						|
				}
 | 
						|
			})
 | 
						|
			
 | 
						|
			markerData = {
 | 
						|
				vehicles: vehicles,
 | 
						|
				spawnPoints: getGarageSpawnpoints( $("#temporary-garage-spawnpoints-list") ),
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "garage_buyable": {		
 | 
						|
			let vehicles = {};
 | 
						|
			let vehiclesObject = $("#garage-buyable-vehicles").find(".vehicle");
 | 
						|
	
 | 
						|
			vehiclesObject.each(function (index, element) {
 | 
						|
				let model = $(element).find(".vehicle-model").val();
 | 
						|
				let price = parseInt( $(element).find(".vehicle-price").val() );
 | 
						|
	
 | 
						|
				vehicles[model] = price;
 | 
						|
			});
 | 
						|
 | 
						|
			markerData = {
 | 
						|
				spawnPoints: getGarageSpawnpoints( $("#buyable-garage-spawnpoints-list") ),
 | 
						|
				vehicles: vehicles
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "garage_owned": {
 | 
						|
			markerData = {
 | 
						|
				spawnPoints: getGarageSpawnpoints( $("#owned-garage-spawnpoints-list") ),
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "boss": {
 | 
						|
			let canWashMoney = $("#boss-data-washmoney").prop('checked')
 | 
						|
			let canWithdraw = $("#boss-data-withdraw").prop('checked')
 | 
						|
			let canDeposit = $("#boss-data-deposit").prop('checked')
 | 
						|
			let canEmployees = $("#boss-data-employees").prop('checked')
 | 
						|
			let canGrades = $("#boss-data-grades").prop('checked')
 | 
						|
		
 | 
						|
			if(canWashMoney) {
 | 
						|
				var washMoneyReturnPercentage = parseInt($("#boss-data-washmoney-percentage").val());
 | 
						|
				var washMoneyGoesToSocietyAccount = $("#boss-data-washmoney-to-society-account").prop("checked");
 | 
						|
			}
 | 
						|
		
 | 
						|
			markerData = {
 | 
						|
				canWashMoney: canWashMoney,
 | 
						|
				washMoneyReturnPercentage: washMoneyReturnPercentage,
 | 
						|
				washMoneyGoesToSocietyAccount: washMoneyGoesToSocietyAccount,
 | 
						|
				canWithdraw: canWithdraw,
 | 
						|
				canDeposit: canDeposit,
 | 
						|
				canEmployees: canEmployees,
 | 
						|
				canGrades: canGrades,
 | 
						|
				maxSalary: parseInt( $("#boss-data-max-salary").val() ),
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "shop": {
 | 
						|
			let itemsOnSale = [];
 | 
						|
 | 
						|
			let itemsContainer = $("#shop-data-items-container")
 | 
						|
		
 | 
						|
			itemsContainer.children().each(function (index, element) {
 | 
						|
				let object = $(element).data("object");
 | 
						|
				let itemPrice = $(element).data("item-price");
 | 
						|
				let blackMoney = $(element).find(".item-blackmoney").prop("checked");
 | 
						|
	
 | 
						|
				if (!object || itemPrice === undefined) return;
 | 
						|
 | 
						|
				itemsOnSale.push({
 | 
						|
					object,
 | 
						|
					price: itemPrice,
 | 
						|
					blackMoney
 | 
						|
				})
 | 
						|
			})
 | 
						|
 | 
						|
			markerData = {
 | 
						|
				itemsOnSale: itemsOnSale
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "market": {
 | 
						|
			let items = [];
 | 
						|
 | 
						|
			let marketBodyDiv = $("#market-modal-items-tbody")
 | 
						|
 | 
						|
			let itemsDivs = marketBodyDiv.children(".item")
 | 
						|
 | 
						|
			itemsDivs.each(function(index, element) {
 | 
						|
				let itemDiv = $(element)
 | 
						|
 | 
						|
				let object = itemDiv.data("object");
 | 
						|
				let itemMinPrice = parseInt(itemDiv.find(".item-min-price").text());
 | 
						|
				let itemMaxPrice = parseInt(itemDiv.find(".item-max-price").text());
 | 
						|
				let blackMoney = itemDiv.find(".item-blackmoney").prop("checked")
 | 
						|
				let sellTime = parseInt(itemDiv.find(".item-selltime").val());
 | 
						|
 | 
						|
				items.push({
 | 
						|
					object,
 | 
						|
					minPrice: itemMinPrice,
 | 
						|
					maxPrice: itemMaxPrice,
 | 
						|
					blackMoney: blackMoney,
 | 
						|
					sellTime: sellTime
 | 
						|
				});
 | 
						|
			});
 | 
						|
 | 
						|
			markerData = { 
 | 
						|
				items: items,
 | 
						|
				percentageForSociety: parseInt( $("#market-modal-society-percentage").val() )
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "crafting_table": {
 | 
						|
			let craftablesItems = [];
 | 
						|
				
 | 
						|
			$("#craftables").children().each(function(craftableIndex, craftableElement) {
 | 
						|
				let craftable = $(craftableElement);
 | 
						|
		
 | 
						|
				let resultObject = getChoosenObject( craftable.find('.craftable-object-div') );
 | 
						|
				let resultItemQuantity = parseInt( craftable.find('.result-item-quantity').val() );
 | 
						|
				let resultItemCraftingTime = parseInt( craftable.find('.result-item-crafting-time').val() );
 | 
						|
		
 | 
						|
				let recipes = [];
 | 
						|
 | 
						|
				craftable.find('.craft-ingredients').children().each(function(ingredientIndex, ingredientElement){
 | 
						|
					var ingredientDiv = $(ingredientElement)
 | 
						|
		
 | 
						|
					var ingredientObject = ingredientDiv.data("object");
 | 
						|
					var ingredientQuantity = ingredientDiv.data("item-quantity");
 | 
						|
					var loseOnUse = ingredientDiv.find(".lose-on-use-checkbox").prop("checked")
 | 
						|
					
 | 
						|
					recipes.push({
 | 
						|
						object: ingredientObject,
 | 
						|
						quantity: ingredientQuantity,
 | 
						|
						loseOnUse
 | 
						|
					});
 | 
						|
				})
 | 
						|
 | 
						|
				craftablesItems.push({
 | 
						|
					resultObject,
 | 
						|
					recipes,
 | 
						|
					animations: craftable.data("animations"),
 | 
						|
					quantity: resultItemQuantity || 1,
 | 
						|
					craftingTime: resultItemCraftingTime || 8
 | 
						|
				});
 | 
						|
			})
 | 
						|
 | 
						|
			markerData = { craftablesItems }
 | 
						|
		
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "armory": {
 | 
						|
			markerData = {
 | 
						|
				isShared: $("#armory-is-shared").prop("checked"),
 | 
						|
				refillOnTake: $("#armory-refill-on-take").prop("checked")
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "job_outfit": {
 | 
						|
			let jobOutfitsDiv = $("#job-outfit-outfits");
 | 
						|
			let outfitsDiv = jobOutfitsDiv.children();
 | 
						|
		
 | 
						|
			let outfits = [];
 | 
						|
 | 
						|
			outfitsDiv.each(function(index, outfitDivRaw){
 | 
						|
				let outfitDiv = $(outfitDivRaw);
 | 
						|
	
 | 
						|
				let outfit = getOutfitFromOutfitDiv(outfitDiv);
 | 
						|
	
 | 
						|
				outfits.push(outfit);
 | 
						|
			})
 | 
						|
	
 | 
						|
			markerData = {
 | 
						|
				outfits: outfits
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "teleport": {
 | 
						|
			markerData = {
 | 
						|
				teleportCoords: {
 | 
						|
					x: parseFloat( $(`#teleport-x`).val() ),
 | 
						|
					y: parseFloat( $(`#teleport-y`).val() ),
 | 
						|
					z: parseFloat( $(`#teleport-z`).val() ),
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "harvest": {
 | 
						|
			let harvestableItems = [];
 | 
						|
 | 
						|
			$("#harvest-modal-items").find(".harvestable-item").each(function(index, element) {
 | 
						|
				let harvestableItem = {
 | 
						|
					object: getChoosenObject( $(this).find(".harvest-object-div") ),
 | 
						|
					minQuantity: parseInt( $(this).find(".harvest-item-min-quantity").val() ),
 | 
						|
					maxQuantity: parseInt( $(this).find(".harvest-item-max-quantity").val() ),
 | 
						|
					time: parseInt( $(this).find(".harvest-item-time").val() ),
 | 
						|
					chances: parseInt( $(this).find(".harvest-item-chance").val()),
 | 
						|
				};
 | 
						|
 | 
						|
				harvestableItems.push(harvestableItem);
 | 
						|
			})
 | 
						|
 | 
						|
			if($("#harvest-item-requires-tool").prop("checked")) {
 | 
						|
				var itemTool = getChoosenObject( $("#harvest-item-tool-div") );
 | 
						|
			}
 | 
						|
 | 
						|
			if($("#harvest-item-tool-lose-on-use").prop("checked")) {
 | 
						|
				var itemToolLoseQuantity = parseInt($("#harvest-item-tool-lose-quantity").val());
 | 
						|
				var itemToolLoseProbability = parseInt( $("#harvest-item-tool-lose-probability").val() );
 | 
						|
			}
 | 
						|
 | 
						|
			markerData = {
 | 
						|
				harvestableItems,
 | 
						|
				animations: $("#edit-marker-dialog-modal").data("animations"),
 | 
						|
 | 
						|
				itemTool,
 | 
						|
				itemToolLoseQuantity,
 | 
						|
				itemToolLoseProbability,
 | 
						|
 | 
						|
				disappearAfterUse: $("#harvest-disappear-after-use").prop("checked"),
 | 
						|
				disappearSeconds: parseInt( $("#harvest-disappear-seconds").val() ),
 | 
						|
 | 
						|
				requiresMinimumAccountMoney: $("#harvest-requires-minimum-account-money").prop("checked"),
 | 
						|
				minimumAccountName: $("#harvest-minimum-account-name").val(),
 | 
						|
				minimumAccountAmount: parseInt( $("#harvest-minimum-account-amount").val() ),
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "process": {
 | 
						|
			markerData = {
 | 
						|
				itemToRemove: getChoosenObject( $("#process-modal-remove-item-div") ),
 | 
						|
				itemToRemoveQuantity: parseInt($("#item-to-remove-quantity").val()),
 | 
						|
				itemToAdd: getChoosenObject( $("#process-modal-give-item-div") ),
 | 
						|
				itemToAddQuantity: parseInt($("#item-to-add-quantity").val()),
 | 
						|
				timeToProcess: parseInt($("#process-time").val()),
 | 
						|
				animations: $("#edit-marker-dialog-modal").data("animations"),
 | 
						|
 | 
						|
				requiresMinimumAccountMoney: $("#process-requires-minimum-account-money").prop("checked"),
 | 
						|
				minimumAccountName: $("#process-minimum-account-name").val(),
 | 
						|
				minimumAccountAmount: parseInt( $("#process-minimum-account-amount").val() )
 | 
						|
			}
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "weapon_upgrader": {
 | 
						|
			let weaponsComponentsDiv = $("#weapon-upgrader-components");
 | 
						|
			let weaponsTintsDiv = $("#weapon-upgrader-tints");
 | 
						|
			
 | 
						|
			let componentsPrices = {};
 | 
						|
			let tintsPrices = {};
 | 
						|
 | 
						|
			weaponsComponentsDiv.find(".form-control").each(function(index, element) {
 | 
						|
				let componentDiv = $(element)
 | 
						|
 | 
						|
				let componentName = componentDiv.data("componentname")
 | 
						|
				let componentPrice = componentDiv.val()
 | 
						|
 | 
						|
				if(componentPrice) {
 | 
						|
					componentsPrices[componentName] = componentPrice
 | 
						|
				}
 | 
						|
			})
 | 
						|
 | 
						|
			weaponsTintsDiv.find(".form-control").each(function(index, element) {
 | 
						|
				let tintDiv = $(element)
 | 
						|
 | 
						|
				let tintId = parseInt(tintDiv.data("tintid"))
 | 
						|
				let tintPrice = parseInt(tintDiv.val())
 | 
						|
 | 
						|
				if(tintPrice) {
 | 
						|
					tintsPrices[tintId] = tintPrice
 | 
						|
				}
 | 
						|
			})
 | 
						|
 | 
						|
			markerData = {
 | 
						|
				componentsPrices: componentsPrices,
 | 
						|
				tintsPrices: tintsPrices
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "job_shop": {
 | 
						|
			let jobName = $("#job-shop-all-jobs").val();
 | 
						|
			if(!jobName) return;
 | 
						|
 | 
						|
			let rankGrade = parseInt($("#job-shop-all-ranks").val());
 | 
						|
			
 | 
						|
			if(jobName && !isNaN(rankGrade)) {
 | 
						|
				markerData = {
 | 
						|
					allowedJob: jobName,
 | 
						|
					minimumRank: rankGrade
 | 
						|
				}
 | 
						|
			}
 | 
						|
			
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return markerData;
 | 
						|
}
 | 
						|
 | 
						|
$("#edit-marker-form").submit(function(event){
 | 
						|
	if (!this.checkValidity()) {
 | 
						|
		event.stopPropagation();
 | 
						|
		event.preventDefault();
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
	let markerId = editMarkerModal.data("markerId");
 | 
						|
 | 
						|
	let isPublicMarker = $("#edit-marker-form").data("isPublic");
 | 
						|
 | 
						|
	let enableMarkerBlip = $("#edit-marker-map-blip").prop("checked");
 | 
						|
 | 
						|
	if(enableMarkerBlip) {
 | 
						|
		var blipSpriteId = 	parseInt( $("#edit-marker-sprite-id").val() );
 | 
						|
		var blipColor = 	parseInt( $("#edit-marker-blip-color").val() );
 | 
						|
		var blipScale = 	parseFloat( $("#edit-marker-blip-scale").val() );
 | 
						|
	}
 | 
						|
 | 
						|
	let color = hexToRgb( $("#edit-marker-color").val() );
 | 
						|
 | 
						|
	let enableNPC = $("#edit-marker-npc").prop("checked");
 | 
						|
 | 
						|
	if(enableNPC) {
 | 
						|
		var npcModel = $("#edit-marker-npc-model").val();
 | 
						|
		var npcHeading = parseFloat( $("#edit-marker-npc-heading").val() );
 | 
						|
	}
 | 
						|
 | 
						|
	let enableObject = $("#edit-marker-object").prop("checked");
 | 
						|
 | 
						|
	if(enableObject) {
 | 
						|
		var objectModel = $("#edit-marker-object-model").val();
 | 
						|
	}
 | 
						|
 | 
						|
	let gradesType = $("input[type=radio][name=markerGradeType]:checked").val();
 | 
						|
 | 
						|
	if(gradesType === "minimumGrade") {
 | 
						|
		var minGrade = parseInt( $("#edit-marker-min-grade").val() );
 | 
						|
	} else if(gradesType === "specificGrades") {
 | 
						|
		var specificGrades = $("#edit-marker-specific-grades").data("grades") || {};
 | 
						|
	}
 | 
						|
 | 
						|
	let data = {
 | 
						|
		markerId: markerId,
 | 
						|
 | 
						|
		label: $("#edit-marker-label").val(),
 | 
						|
 | 
						|
		gradesType: gradesType,
 | 
						|
 | 
						|
		minGrade: minGrade,
 | 
						|
		specificGrades: specificGrades,
 | 
						|
 | 
						|
		coords: {
 | 
						|
			x: parseFloat( $("#edit-marker-coord-x").val() ),
 | 
						|
			y: parseFloat( $("#edit-marker-coord-y").val() ),
 | 
						|
			z: parseFloat( $("#edit-marker-coord-z").val() ),
 | 
						|
		},
 | 
						|
 | 
						|
		blip: {
 | 
						|
			spriteId: blipSpriteId,
 | 
						|
			color: blipColor,
 | 
						|
			scale: blipScale,
 | 
						|
		},
 | 
						|
		
 | 
						|
		markerType: parseInt( $("#edit-marker-type").val() ),
 | 
						|
 | 
						|
		scale: {
 | 
						|
			x: parseFloat( $("#edit-marker-scale-x").val() ),
 | 
						|
			y: parseFloat( $("#edit-marker-scale-y").val() ),
 | 
						|
			z: parseFloat( $("#edit-marker-scale-z").val() )
 | 
						|
		},
 | 
						|
		
 | 
						|
		color: {
 | 
						|
			r: color.r,
 | 
						|
			g: color.g,
 | 
						|
			b: color.b,
 | 
						|
			alpha: parseInt( $("#edit-marker-alpha").val() )
 | 
						|
		},
 | 
						|
 | 
						|
		ped: {
 | 
						|
			model: npcModel,
 | 
						|
			heading: npcHeading
 | 
						|
		},
 | 
						|
 | 
						|
		object: {
 | 
						|
			model: objectModel,
 | 
						|
			heading: parseFloat( $("#edit-marker-object-heading").val() )
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	let markerData = getCurrentMarkerData(allMarkers[markerId].type);
 | 
						|
 | 
						|
	$.post(`https://${resName}/update-marker`, JSON.stringify(data), function (data) {
 | 
						|
		if (data.isSuccessful) {
 | 
						|
 | 
						|
			let markerType = allMarkers[markerId].type
 | 
						|
 | 
						|
			if(MARKERS_INFOS[markerType].requiresDataUpdate) {
 | 
						|
				$.post(`https://${resName}/update-marker-data`, JSON.stringify({ markerId: markerId, markerData: markerData }), function (data) {
 | 
						|
					if (data.isSuccessful) {
 | 
						|
						editMarkerModal.modal("hide");
 | 
						|
					} else {
 | 
						|
						showError(data.message)
 | 
						|
					}
 | 
						|
	
 | 
						|
					if(isPublicMarker) {
 | 
						|
						fillPublicMarkers();
 | 
						|
					} else {
 | 
						|
						fillJobMarkers();
 | 
						|
					}
 | 
						|
				});
 | 
						|
			} else {
 | 
						|
				editMarkerModal.modal("hide");
 | 
						|
 | 
						|
				if(isPublicMarker) {
 | 
						|
					fillPublicMarkers();
 | 
						|
				} else {
 | 
						|
					fillJobMarkers();
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			showError(data.message)
 | 
						|
		}
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
function gradesDialog(enabledGrades) {
 | 
						|
	return new Promise(resolve => {
 | 
						|
		let gradesModal = $("#input-grades-dialog-modal")
 | 
						|
		let gradesList = $("#grades-list");
 | 
						|
		gradesList.empty();
 | 
						|
	
 | 
						|
		let jobName = $("#edit-job").data("jobName");
 | 
						|
	
 | 
						|
		$.post(`https://${resName}/getJobGrades`, JSON.stringify({jobName: jobName}), function(jobGrades) {
 | 
						|
			for( const[_, gradeData] of Object.entries(jobGrades) ) {
 | 
						|
				let gradeDiv = $(`
 | 
						|
					<div class="form-check fs-3">
 | 
						|
						<input class="form-check-input grade" type="checkbox" value="${gradeData.grade}" ${enabledGrades[gradeData.grade] && "checked" || null}>
 | 
						|
						<label class="form-check-label">
 | 
						|
							${gradeData.grade} - ${gradeData.label}
 | 
						|
						</label>
 | 
						|
					</div>
 | 
						|
				`);
 | 
						|
	
 | 
						|
				gradesList.append(gradeDiv);
 | 
						|
			}
 | 
						|
	
 | 
						|
			gradesModal.modal("show");
 | 
						|
		})
 | 
						|
 | 
						|
		
 | 
						|
	
 | 
						|
		let confirmButton = $("#input-grades-dialog-confirm");
 | 
						|
		
 | 
						|
		confirmButton.unbind("click");
 | 
						|
		confirmButton.click(function() {
 | 
						|
			let grades = {};
 | 
						|
			$(".grade:checked").each(function() {
 | 
						|
				grades[ parseInt($(this).val()) ] = true;
 | 
						|
			})
 | 
						|
	
 | 
						|
			gradesModal.modal("hide");
 | 
						|
	
 | 
						|
			resolve(grades);		
 | 
						|
		})
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
function setSpecificGrades(grades) {
 | 
						|
	let specificGradesDiv = $("#edit-marker-specific-grades");
 | 
						|
	
 | 
						|
	specificGradesDiv.data("grades", grades);
 | 
						|
	specificGradesDiv.val( Object.keys(grades).join(","));
 | 
						|
}
 | 
						|
 | 
						|
$("#edit-marker-specific-grades-choose-btn").click(async function() {
 | 
						|
	let enabledGrades = $("#edit-marker-specific-grades").data("grades") || {};
 | 
						|
 | 
						|
	let grades = await gradesDialog(enabledGrades);
 | 
						|
 | 
						|
	setSpecificGrades(grades);
 | 
						|
})
 | 
						|
 | 
						|
$("#marker-data-add-vehicle").click(function () {
 | 
						|
	var modelDiv = $("#marker-data-vehicle-model");
 | 
						|
	var model = modelDiv.val();
 | 
						|
 | 
						|
	// Clears input
 | 
						|
	modelDiv.val("");
 | 
						|
 | 
						|
	if (model) {
 | 
						|
		modelDiv.removeClass("is-invalid");
 | 
						|
 | 
						|
		addVehicleToTemporaryGarage(model)
 | 
						|
	} else {
 | 
						|
		modelDiv.addClass("is-invalid");
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function getGarageSpawnpoints(listDiv) {
 | 
						|
	let spawnpoints = [];
 | 
						|
 | 
						|
	listDiv.find(".spawnpoint-div").each(function() {
 | 
						|
		let spawnpointData = {
 | 
						|
			coords: {
 | 
						|
				x: parseFloat( $(this).find(".spawnpoint-coords-x").val() ),
 | 
						|
				y: parseFloat( $(this).find(".spawnpoint-coords-y").val() ),
 | 
						|
				z: parseFloat( $(this).find(".spawnpoint-coords-z").val() ),
 | 
						|
			},
 | 
						|
 | 
						|
			heading: parseFloat( $(this).find(".spawnpoint-heading").val() ),	
 | 
						|
			radius: parseInt( $(this).find(".spawnpoint-radius").val() )
 | 
						|
		}
 | 
						|
 | 
						|
		spawnpoints.push(spawnpointData);
 | 
						|
	})
 | 
						|
 | 
						|
	return spawnpoints;
 | 
						|
}
 | 
						|
 | 
						|
function addGarageSpawnpoint(listDiv, spawnpointData) {
 | 
						|
	let div = $(`
 | 
						|
		<div class="input-group spawnpoint-div my-2">
 | 
						|
			<button type="button" class="btn-close delete-spawnpoint-btn my-auto me-3"></button>	
 | 
						|
			<span class="input-group-text" data-translation-id="menu:coords">Coords</span>
 | 
						|
			<input type="number" step="0.01" class="form-control max-two-decimals spawnpoint-coords-x" placeholder="X" required>
 | 
						|
			<input type="number" step="0.01" class="form-control max-two-decimals spawnpoint-coords-y" placeholder="Y" required>
 | 
						|
			<input type="number" step="0.01" class="form-control max-two-decimals spawnpoint-coords-z" placeholder="Z" required>
 | 
						|
			<span class="input-group-text" data-translation-id="menu:heading">Heading</span>
 | 
						|
			<input type="number" step="0.01" class="form-control max-two-decimals spawnpoint-heading" placeholder="Heading" required>	
 | 
						|
			<button type="button" class="btn btn-secondary col-auto current-coords-btn" data-bs-toggle="tooltip" data-bs-placement="top" title="Do this when inside a vehicle"><i class="bi bi-arrow-down-square"></i></button>						
 | 
						|
			<span class="input-group-text ms-2" data-translation-id="menu:radius">Radius</span>
 | 
						|
			<input type="number" class="form-control spawnpoint-radius" placeholder="Radius" required value="5">	
 | 
						|
		</div>
 | 
						|
	`);
 | 
						|
 | 
						|
	div.find(".delete-spawnpoint-btn").click(function() {
 | 
						|
		div.remove();
 | 
						|
	});
 | 
						|
 | 
						|
	div.find(".current-coords-btn").click(async function() {
 | 
						|
		const data = await getCurrentCoordsAndHeading();
 | 
						|
 | 
						|
		div.find(".spawnpoint-coords-x").val(data.coords.x);
 | 
						|
		div.find(".spawnpoint-coords-y").val(data.coords.y);
 | 
						|
		div.find(".spawnpoint-coords-z").val(data.coords.z);
 | 
						|
		div.find(".spawnpoint-heading").val(data.heading);
 | 
						|
	});
 | 
						|
			
 | 
						|
 | 
						|
	if(spawnpointData) {
 | 
						|
		div.find(".spawnpoint-coords-x").val(spawnpointData.coords.x);
 | 
						|
		div.find(".spawnpoint-coords-y").val(spawnpointData.coords.y);
 | 
						|
		div.find(".spawnpoint-coords-z").val(spawnpointData.coords.z);
 | 
						|
		div.find(".spawnpoint-heading").val(spawnpointData.heading);
 | 
						|
		div.find(".spawnpoint-radius").val(spawnpointData.radius);
 | 
						|
	}
 | 
						|
 | 
						|
	listDiv.append(div);
 | 
						|
}
 | 
						|
 | 
						|
$("#temporary-garage-add-spawnpoint").click(function() {
 | 
						|
	addGarageSpawnpoint( $("#temporary-garage-spawnpoints-list") );
 | 
						|
})
 | 
						|
 | 
						|
$("#buyable-garage-add-spawnpoint").click(function() {
 | 
						|
	addGarageSpawnpoint( $("#buyable-garage-spawnpoints-list") );
 | 
						|
})
 | 
						|
 | 
						|
$("#owned-garage-add-spawnpoint").click(function() {
 | 
						|
	addGarageSpawnpoint( $("#owned-garage-spawnpoints-list") );
 | 
						|
})
 | 
						|
 | 
						|
function addItemToShop(object, itemData) {
 | 
						|
	var newItem = $(`
 | 
						|
		<tr class="border">
 | 
						|
			<td class="col">
 | 
						|
				<p class="text-center">${object.name}</p>
 | 
						|
			</td>
 | 
						|
			
 | 
						|
			<td class="col">
 | 
						|
				<p class="text-center">$${itemData.price}</p>
 | 
						|
			</td>
 | 
						|
 | 
						|
			<td class="col">
 | 
						|
				<div class="form-check form-switch col-md-4 offset-md-4">
 | 
						|
					<input class="form-check-input item-blackmoney" type="checkbox"}>
 | 
						|
				</div>
 | 
						|
			</td>
 | 
						|
			
 | 
						|
			<td class="col">
 | 
						|
				<button type="button" class="btn btn-outline-danger col-12">Remove</button>
 | 
						|
			</td>
 | 
						|
		</tr>
 | 
						|
	`)
 | 
						|
 | 
						|
	newItem.data("item-price", parseInt(itemData.price));
 | 
						|
 | 
						|
	newItem.find(".item-blackmoney").prop("checked", itemData.blackMoney);
 | 
						|
 | 
						|
	newItem.find(".btn").click(function () { newItem.remove(); })
 | 
						|
 | 
						|
	newItem.data("object", object);
 | 
						|
 | 
						|
	$('#shop-data-items-container').append(newItem)
 | 
						|
}
 | 
						|
 | 
						|
$("#shop-data-add-btn").click(function () {
 | 
						|
	var isEverythingValid = true
 | 
						|
 | 
						|
	let choosenObject = getChoosenObject( $("#shop-new-item-name-div") );
 | 
						|
	var itemPriceDiv = $("#shop-data-item-price")
 | 
						|
	var itemName = choosenObject.name;
 | 
						|
	var itemPrice = itemPriceDiv.val();
 | 
						|
 | 
						|
	let itemData = {
 | 
						|
		price: itemPrice
 | 
						|
	}
 | 
						|
 | 
						|
	if (!itemName) return;
 | 
						|
 | 
						|
	if (!itemPrice) {
 | 
						|
		itemPriceDiv.addClass("is-invalid")
 | 
						|
		isEverythingValid = false
 | 
						|
	} else {
 | 
						|
		itemPriceDiv.removeClass("is-invalid")
 | 
						|
	}
 | 
						|
 | 
						|
	if (!isEverythingValid) return;
 | 
						|
 | 
						|
	addItemToShop(choosenObject, itemData);
 | 
						|
})
 | 
						|
 | 
						|
$("#garage-buyable-add-vehicle").click(function () {
 | 
						|
	var isEverythingValid = true;
 | 
						|
	
 | 
						|
	var modelDiv = $("#garage-buyable-vehicle-model");
 | 
						|
	var model = modelDiv.val();
 | 
						|
	var priceDiv = $("#garage-buyable-vehicle-price");
 | 
						|
	var price = parseInt(priceDiv.val());
 | 
						|
 | 
						|
	// Clears input
 | 
						|
	modelDiv.val("");
 | 
						|
	priceDiv.val("");
 | 
						|
 | 
						|
	if (model) {
 | 
						|
		modelDiv.removeClass("is-invalid");
 | 
						|
	} else {
 | 
						|
		modelDiv.addClass("is-invalid");
 | 
						|
		isEverythingValid = false
 | 
						|
	}
 | 
						|
 | 
						|
	if(price) {
 | 
						|
		priceDiv.removeClass("is-invalid")
 | 
						|
	} else {
 | 
						|
		priceDiv.addClass("is-invalid")
 | 
						|
		isEverythingValid = false
 | 
						|
	}
 | 
						|
 | 
						|
	if(isEverythingValid) {
 | 
						|
		var vehicleInputGroup = $(`<div class="input-group mt-1 vehicle"></div>`)
 | 
						|
 | 
						|
		var vehicleInput = $(`<input type="text" class="form-control vehicle-model" placeholder="Vehicle model" disabled value="${model}">`);
 | 
						|
		var moneySpan = $(`<span class="input-group-text">$</span>`)
 | 
						|
		var vehiclePrice = $(`<input type="text" class="form-control vehicle-price" placeholder="Vehicle price" disabled value="${price}">`);
 | 
						|
		var deleteVehicle = $(`<button class="btn btn-outline-danger" type="button">Delete</button>`);
 | 
						|
 | 
						|
		vehicleInputGroup.append(vehicleInput, moneySpan, vehiclePrice, deleteVehicle);
 | 
						|
 | 
						|
		$(deleteVehicle).click(function () {
 | 
						|
			$(this).parent().remove();
 | 
						|
		})
 | 
						|
 | 
						|
		$("#garage-buyable-vehicles").append(vehicleInputGroup)
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#garage-buyable-current-coords").click(function() {
 | 
						|
	$.post(`https://${resName}/get-current-coords`, {}, function (data) {
 | 
						|
		var coords = data.coords;
 | 
						|
		var heading = data.heading;
 | 
						|
 | 
						|
		if (coords) {
 | 
						|
			$("#garage-buyable-spawnpoint-x").val(coords.x)
 | 
						|
			$("#garage-buyable-spawnpoint-y").val(coords.y)
 | 
						|
			$("#garage-buyable-spawnpoint-z").val(coords.z)
 | 
						|
		}
 | 
						|
 | 
						|
		if (heading) {
 | 
						|
			$("#garage-buyable-heading").val(heading)
 | 
						|
		}
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
function addNewCraftableToCraftables(object, craftingData) {
 | 
						|
	var craftablesDiv = $("#craftables");
 | 
						|
 | 
						|
	var newCraftable = $(`
 | 
						|
		<div class="container mb-5">
 | 
						|
			<div class="d-flex gap-3 align-items-center">
 | 
						|
				<div class="choose-object-div craftable-object-div input-group">
 | 
						|
					<input type="text" class="form-control choose-object-label" placeholder="Result item" readonly>
 | 
						|
					<button type="button" class="btn btn-secondary choose-object-btn" data-bs-toggle="tooltip" data-bs-placement="top" title="${getLocalizedText("menu:dialog:choose_item")}"><i class="bi bi-list-ul"></i></button>
 | 
						|
				</div>
 | 
						|
 | 
						|
				<div class="input-group mb-3 mt-2">
 | 
						|
					<span class="input-group-text ms-2">${getLocalizedText("menu:dynamic:marker:crafting_table:quantity")}</span>
 | 
						|
					<input type="number" class="form-control result-item-quantity" placeholder="Item quantity" value="${craftingData?.quantity || 1}" required>
 | 
						|
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:marker:crafting_table:time_to_craft")}</span>
 | 
						|
					<input type="number" class="form-control result-item-crafting-time" placeholder="Seconds" value="${craftingData?.craftingTime || 8}" required>
 | 
						|
				</div>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<table class="table">
 | 
						|
				<thead>
 | 
						|
					<tr>
 | 
						|
						<th scope="col">${getLocalizedText("menu:dynamic:marker:crafting_table:item_id")}</th>
 | 
						|
						<th scope="col">${getLocalizedText("menu:dynamic:marker:crafting_table:item_quantity")}</th>
 | 
						|
						<th scope="col">${getLocalizedText("menu:dynamic:marker:crafting_table:lose_on_use")}</th>
 | 
						|
						<th scope="col">${getLocalizedText("menu:dynamic:marker:crafting_table:remove")}</th>
 | 
						|
					</tr>
 | 
						|
				</thead>
 | 
						|
				<tbody class="craft-ingredients">
 | 
						|
 | 
						|
				</tbody>
 | 
						|
			</table>
 | 
						|
 | 
						|
			<div class="d-flex align-content-center">
 | 
						|
				<div class="choose-object-div ingredient-object-div input-group" data-metadata-disabled>
 | 
						|
					<input type="text" class="form-control choose-object-label" placeholder="New ingredient" readonly>
 | 
						|
					<button type="button" class="btn btn-secondary choose-object-btn" data-bs-toggle="tooltip" data-bs-placement="top" title="${getLocalizedText("menu:dialog:choose_item")}"><i class="bi bi-list-ul"></i></button>
 | 
						|
				</div>
 | 
						|
 | 
						|
				<div class="input-group">
 | 
						|
					<span class="input-group-text ms-2">${getLocalizedText("menu:dynamic:marker:crafting_table:item_quantity")}</span>
 | 
						|
					<input type="number" class="form-control crafting-table-item-quantity">
 | 
						|
					<button type="button" class="btn btn-success ms-2 add-to-craft-btn">${getLocalizedText("menu:dynamic:marker:crafting_table:add_to_craft")}</button>
 | 
						|
				</div>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<div class="d-flex justify-content-between mt-5 mb-2">
 | 
						|
				<button type="button" class="btn btn-danger remove-craft-btn">${getLocalizedText("menu:dynamic:marker:crafting_table:remove_craft")}</button>
 | 
						|
				<button type="button" class="btn btn-secondary animations-craft-btn">${getLocalizedText("menu:dynamic:marker:crafting_table:animations")}</button>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<hr>
 | 
						|
		</div>
 | 
						|
	`)
 | 
						|
 | 
						|
	loadChooseObjectDivs( newCraftable.find(".craftable-object-div"), object)
 | 
						|
	loadChooseObjectDivs( newCraftable.find(".ingredient-object-div"))
 | 
						|
 | 
						|
	newCraftable.find(".remove-craft-btn").click(function() {
 | 
						|
		newCraftable.remove();
 | 
						|
	})
 | 
						|
 | 
						|
	newCraftable.find(".animations-craft-btn").click(function() {
 | 
						|
		inputAnimations(newCraftable.data("animations"), function(animations) {
 | 
						|
			newCraftable.data("animations", animations)
 | 
						|
		});
 | 
						|
	})
 | 
						|
 | 
						|
	var addToCraftBtn = newCraftable.find(".add-to-craft-btn")
 | 
						|
	addToCraftBtn.click(function() {
 | 
						|
		var itemQuantityDiv = newCraftable.find(".crafting-table-item-quantity");
 | 
						|
 | 
						|
		const object = getChoosenObject( newCraftable.find(".ingredient-object-div") );
 | 
						|
		var itemQuantity = parseInt( itemQuantityDiv.val() );
 | 
						|
 | 
						|
		if(!object) return;
 | 
						|
 | 
						|
		if(!itemQuantity || itemQuantity <= 0) {
 | 
						|
			itemQuantityDiv.addClass("is-invalid");
 | 
						|
			return;
 | 
						|
		} else {
 | 
						|
			itemQuantityDiv.removeClass("is-invalid");
 | 
						|
		}
 | 
						|
 | 
						|
		var craftIngredientsDiv = newCraftable.find(".craft-ingredients");
 | 
						|
 | 
						|
		let newCraftItem = $(`
 | 
						|
			<tr data-item-quantity=${itemQuantity}>
 | 
						|
				<th scope="row">${object.name}</td>
 | 
						|
				<td>${itemQuantity}</td>
 | 
						|
				<td>
 | 
						|
					<div class="form-check form-switch fs-4">
 | 
						|
						<input class="form-check-input lose-on-use-checkbox" type="checkbox" checked>
 | 
						|
					</div>
 | 
						|
				</td>
 | 
						|
				<td><button type="button" class="btn btn-outline-danger remove-ingredient-btn">Remove</button></td>
 | 
						|
			</tr>
 | 
						|
		`)
 | 
						|
 | 
						|
		newCraftItem.data("object", object);
 | 
						|
 | 
						|
		newCraftItem.find('.remove-ingredient-btn').click(function() {
 | 
						|
			newCraftItem.remove()
 | 
						|
		})
 | 
						|
 | 
						|
		craftIngredientsDiv.append(newCraftItem)
 | 
						|
 | 
						|
		setChoosenObject( newCraftable.find(".ingredient-object-div"), null )
 | 
						|
		itemQuantityDiv.val("");
 | 
						|
	})
 | 
						|
 | 
						|
	if(craftingData?.recipes) {
 | 
						|
		let craftIngredientsDiv = newCraftable.find('.craft-ingredients');
 | 
						|
		
 | 
						|
		craftingData.recipes.forEach(ingredientData => {
 | 
						|
			let newCraftItem = $(`
 | 
						|
				<tr data-item-quantity=${ingredientData.quantity}>
 | 
						|
					<th scope="row">${ingredientData.object.name}</td>
 | 
						|
					<td>${ingredientData.quantity}</td>
 | 
						|
					<td>
 | 
						|
						<div class="form-check form-switch fs-4">
 | 
						|
							<input class="form-check-input lose-on-use-checkbox" type="checkbox">
 | 
						|
						</div>
 | 
						|
					</td>
 | 
						|
					<td><button type="button" class="btn btn-outline-danger remove-ingredient-btn">Remove</button></td>
 | 
						|
				</tr>
 | 
						|
			`)
 | 
						|
 | 
						|
			newCraftItem.data("object", ingredientData.object);
 | 
						|
 | 
						|
			let loseOnUse = ingredientData.loseOnUse
 | 
						|
			newCraftItem.find(".lose-on-use-checkbox").prop("checked", loseOnUse)
 | 
						|
 | 
						|
			newCraftItem.find('.remove-ingredient-btn').click(function() {
 | 
						|
				newCraftItem.remove()
 | 
						|
			})
 | 
						|
 | 
						|
			craftIngredientsDiv.append(newCraftItem);
 | 
						|
		})
 | 
						|
	}
 | 
						|
 | 
						|
	newCraftable.data("animations", craftingData?.animations || undefined)
 | 
						|
 | 
						|
	craftablesDiv.append(newCraftable);
 | 
						|
}
 | 
						|
 | 
						|
$("#crafting-table-new-craft-btn").click(function() {
 | 
						|
	addNewCraftableToCraftables()
 | 
						|
})
 | 
						|
 | 
						|
$("#delete-armory-content").click(async function(){
 | 
						|
	let armoryModal = $("#armory-data-modal");
 | 
						|
	
 | 
						|
	let markerId = armoryModal.data("markerId")
 | 
						|
	
 | 
						|
	armoryModal.modal("hide");
 | 
						|
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:do_you_want_to_delete_armory_content"))) return;
 | 
						|
 | 
						|
	const data = $.post(`https://${resName}/delete-armory-inventory`, JSON.stringify({ markerId: markerId }));
 | 
						|
 | 
						|
	if(!data.isSuccessful) {
 | 
						|
		showError(data.message)
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#delete-stash-inventory").click(async function() {
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
	let markerId = editMarkerModal.data("markerId");
 | 
						|
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:do_you_want_to_delete_stash_content"))) return;
 | 
						|
 | 
						|
	const data = await $.post(`https://${resName}/delete-stash-inventory`, JSON.stringify({ markerId: markerId }));
 | 
						|
 | 
						|
	if(!data.isSuccessful) {
 | 
						|
		showError(data.message)
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function createNewOutfit(outfit){
 | 
						|
	outfit = outfit || {};
 | 
						|
 | 
						|
	outfit.tshirt_1 = outfit.tshirt_1 || 0;
 | 
						|
	outfit.tshirt_2 = outfit.tshirt_2 || 0;
 | 
						|
 | 
						|
	outfit.torso_1 = outfit.torso_1 || 0;
 | 
						|
	outfit.torso_2 = outfit.torso_2 || 0;
 | 
						|
 | 
						|
	outfit.decals_1 = outfit.decals_1 || 0;
 | 
						|
	outfit.decals_2 = outfit.decals_2 || 0;
 | 
						|
 | 
						|
	outfit.arms = outfit.arms || 0;
 | 
						|
	outfit.arms_2 = outfit.arms_2 || 0;
 | 
						|
 | 
						|
	outfit.pants_1 = outfit.pants_1 || 0;
 | 
						|
	outfit.pants_2 = outfit.pants_2 || 0;
 | 
						|
 | 
						|
	outfit.shoes_1 = outfit.shoes_1 || 0;
 | 
						|
	outfit.shoes_2 = outfit.shoes_2 || 0;
 | 
						|
 | 
						|
	outfit.mask_1 = outfit.mask_1 || 0;
 | 
						|
	outfit.mask_2 = outfit.mask_2 || 0;
 | 
						|
 | 
						|
	outfit.bproof_1 = outfit.bproof_1 || 0;
 | 
						|
	outfit.bproof_2 = outfit.bproof_2 || 0;
 | 
						|
 | 
						|
	outfit.chain_1 = outfit.chain_1 || 0;
 | 
						|
	outfit.chain_2 = outfit.chain_2 || 0;
 | 
						|
 | 
						|
	outfit.helmet_1 = outfit.helmet_1 || -1;
 | 
						|
	outfit.helmet_2 = outfit.helmet_2 || 0;
 | 
						|
 | 
						|
	outfit.glasses_1 = outfit.glasses_1 || 0;
 | 
						|
	outfit.glasses_2 = outfit.glasses_2 || 0;
 | 
						|
 | 
						|
	outfit.bags_1 = outfit.bags_1 || 0;
 | 
						|
	outfit.bags_2 = outfit.bags_2 || 0;
 | 
						|
 | 
						|
	let outfitsDiv = $("#job-outfit-outfits")
 | 
						|
	let outfitId = $("#job-outfit-outfits").children().length + 1;
 | 
						|
	
 | 
						|
	let outfitDiv = $(`
 | 
						|
		<div class="accordion accordion-flush mt-1" id="job-outfit-id-${outfitId}" data-label="${outfit.label}">
 | 
						|
			<div class="accordion-item">
 | 
						|
				<h2 class="accordion-header">
 | 
						|
					<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse"
 | 
						|
						data-bs-target="#job-outfit-content-id-${outfitId}" aria-expanded="false">
 | 
						|
						${outfit.label}
 | 
						|
					</button>
 | 
						|
				</h2>
 | 
						|
				<div id="job-outfit-content-id-${outfitId}" class="accordion-collapse collapse" data-bs-parent="#job-outfit-id-${outfitId}">
 | 
						|
					<div class="accordion-body container">
 | 
						|
						<div class="outfit">
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">T-Shirt</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-tshirt" value=${outfit.tshirt_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-tshirt-color" value=${outfit.tshirt_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Torso</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-torso" value=${outfit.torso_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-torso-color" value=${outfit.torso_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Decals</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-decals" value=${outfit.decals_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-decals-color" value=${outfit.decals_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Arms</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-arms" value=${outfit.arms}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-arms-color" value=${outfit.arms_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Pants</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-pants" value=${outfit.pants_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-pants-color" value=${outfit.pants_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Shoes</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-shoes" value=${outfit.shoes_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-shoes-color" value=${outfit.shoes_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Mask</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-mask" value=${outfit.mask_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-mask-color" value=${outfit.mask_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Bulletproof</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-bulletproof" value=${outfit.bproof_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-bulletproof-color" value=${outfit.bproof_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Chain</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-chain" value=${outfit.chain_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-chain-color" value=${outfit.chain_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Helmet/Hat</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-helmet" value=${outfit.helmet_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-helmet-color" value=${outfit.helmet_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Glasses</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-glasses" value=${outfit.glasses_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-glasses-color" value=${outfit.glasses_2}>
 | 
						|
							</div>
 | 
						|
							<div class="input-group mb-3 col">
 | 
						|
								<span class="input-group-text col">Bag</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-bag" value=${outfit.bags_1}>
 | 
						|
								<span class="input-group-text col">Color</span>
 | 
						|
								<input type="number" class="form-control float-end col outfit-bag-color" value=${outfit.bags_2}>
 | 
						|
							</div>
 | 
						|
						</div>
 | 
						|
 | 
						|
						<button type="button" class="btn btn-success mt-1 current-outfit-btn">Current outfit</button>
 | 
						|
 | 
						|
						<button type="button" class="btn btn-danger mt-1 delete-outfit-btn">Delete outfit</button>
 | 
						|
					</div>
 | 
						|
				</div>
 | 
						|
			</div>
 | 
						|
		</div>
 | 
						|
	`)
 | 
						|
 | 
						|
	outfitDiv.find(".current-outfit-btn").click(function(){
 | 
						|
		$.post(`https://${resName}/get-current-outfit`, JSON.stringify({}), function (data) {
 | 
						|
			outfitDiv.find(".outfit-tshirt").val(data.tshirt_1)
 | 
						|
			outfitDiv.find(".outfit-tshirt-color").val(data.tshirt_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-torso").val(data.torso_1)
 | 
						|
			outfitDiv.find(".outfit-torso-color").val(data.torso_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-decals").val(data.decals_1)
 | 
						|
			outfitDiv.find(".outfit-decals-color").val(data.decals_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-arms").val(data.arms)
 | 
						|
			outfitDiv.find(".outfit-arms-color").val(data.arms_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-pants").val(data.pants_1)
 | 
						|
			outfitDiv.find(".outfit-pants-color").val(data.pants_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-shoes").val(data.shoes_1)
 | 
						|
			outfitDiv.find(".outfit-shoes-color").val(data.shoes_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-mask").val(data.mask_1)
 | 
						|
			outfitDiv.find(".outfit-mask-color").val(data.mask_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-bulletproof").val(data.bproof_1)
 | 
						|
			outfitDiv.find(".outfit-bulletproof-color").val(data.bproof_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-chain").val(data.chain_1)
 | 
						|
			outfitDiv.find(".outfit-chain-color").val(data.chain_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-helmet").val(data.helmet_1)
 | 
						|
			outfitDiv.find(".outfit-helmet-color").val(data.helmet_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-glasses").val(data.glasses_1)
 | 
						|
			outfitDiv.find(".outfit-glasses-color").val(data.glasses_2)
 | 
						|
 | 
						|
			outfitDiv.find(".outfit-bag").val(data.bags_1)
 | 
						|
			outfitDiv.find(".outfit-bag-color").val(data.bags_2)
 | 
						|
		});
 | 
						|
	})
 | 
						|
 | 
						|
	outfitDiv.find(".delete-outfit-btn").click(async function(){
 | 
						|
		if(! await confirmDeletion(getLocalizedText("menu:do_you_want_to_delete_outfit"))) return;
 | 
						|
 | 
						|
		outfitDiv.remove();
 | 
						|
	})
 | 
						|
 | 
						|
	$(outfitsDiv).append(outfitDiv)
 | 
						|
}
 | 
						|
 | 
						|
function getOutfitFromOutfitDiv(outfitDiv) {
 | 
						|
	let outfit = {}
 | 
						|
 | 
						|
	outfit.label = outfitDiv.data("label")
 | 
						|
 | 
						|
	outfit.tshirt_1 = parseInt(outfitDiv.find(".outfit-tshirt").val()) || 0;
 | 
						|
	outfit.tshirt_2 = parseInt(outfitDiv.find(".outfit-tshirt-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.torso_1 = parseInt(outfitDiv.find(".outfit-torso").val()) || 0;
 | 
						|
	outfit.torso_2 = parseInt(outfitDiv.find(".outfit-torso-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.decals_1 = parseInt(outfitDiv.find(".outfit-decals").val()) || 0;
 | 
						|
	outfit.decals_2 = parseInt(outfitDiv.find(".outfit-decals-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.arms = parseInt(outfitDiv.find(".outfit-arms").val()) || 0;
 | 
						|
	outfit.arms_2 = parseInt(outfitDiv.find(".outfit-arms-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.pants_1 = parseInt(outfitDiv.find(".outfit-pants").val()) || 0;
 | 
						|
	outfit.pants_2 = parseInt(outfitDiv.find(".outfit-pants-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.shoes_1 = parseInt(outfitDiv.find(".outfit-shoes").val()) || 0;
 | 
						|
	outfit.shoes_2 = parseInt(outfitDiv.find(".outfit-shoes-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.mask_1 = parseInt(outfitDiv.find(".outfit-mask").val()) || 0;
 | 
						|
	outfit.mask_2 = parseInt(outfitDiv.find(".outfit-mask-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.bproof_1 = parseInt(outfitDiv.find(".outfit-bulletproof").val()) || 0;
 | 
						|
	outfit.bproof_2 = parseInt(outfitDiv.find(".outfit-bulletproof-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.chain_1 = parseInt(outfitDiv.find(".outfit-chain").val()) || 0;
 | 
						|
	outfit.chain_2 = parseInt(outfitDiv.find(".outfit-chain-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.helmet_1 = parseInt(outfitDiv.find(".outfit-helmet").val()) || -1;
 | 
						|
	outfit.helmet_2 = parseInt(outfitDiv.find(".outfit-helmet-color").val()) || 0
 | 
						|
 | 
						|
	outfit.glasses_1 = parseInt(outfitDiv.find(".outfit-glasses").val()) || 0;
 | 
						|
	outfit.glasses_2 = parseInt(outfitDiv.find(".outfit-glasses-color").val()) || 0;
 | 
						|
 | 
						|
	outfit.bags_1 = parseInt(outfitDiv.find(".outfit-bag").val()) || 0;
 | 
						|
	outfit.bags_2 = parseInt(outfitDiv.find(".outfit-bag-color").val()) || 0;
 | 
						|
 | 
						|
	return outfit;
 | 
						|
}
 | 
						|
 | 
						|
$("#job-outfit-new-outfit-btn").click(function(){
 | 
						|
	inputDialog("Create new outfit", [
 | 
						|
		{ id: "outfitLabel", label: "New outfit label:" },
 | 
						|
	], function (data) {
 | 
						|
		let outfit = {label: data.outfitLabel}
 | 
						|
		createNewOutfit(outfit);
 | 
						|
	});
 | 
						|
})
 | 
						|
 | 
						|
$("#teleport-current-coords-btn").click(function(){
 | 
						|
	$.post(`https://${resName}/get-current-coords`, {}, function (data) {
 | 
						|
		var coords = data.coords;
 | 
						|
 | 
						|
		if (coords) {
 | 
						|
			$(`#teleport-x`).val(coords.x)
 | 
						|
			$(`#teleport-y`).val(coords.y)
 | 
						|
			$(`#teleport-z`).val(coords.z)
 | 
						|
		}
 | 
						|
	})
 | 
						|
});
 | 
						|
 | 
						|
function addItemToMarket(object, itemMinPrice, itemMaxPrice, blackMoney = false, sellTime = 0) {
 | 
						|
	let itemsTableDiv = $("#market-modal-items-tbody");
 | 
						|
 | 
						|
	let itemDiv = $(`
 | 
						|
		<tr class="item">
 | 
						|
			<th class="item-name">${object.name}</th>
 | 
						|
			
 | 
						|
			<td class="item-min-price">${itemMinPrice}</td>
 | 
						|
			<td class="item-max-price">${itemMaxPrice}</td>
 | 
						|
 | 
						|
			<td>
 | 
						|
				<input class="form-check-input item-blackmoney fs-4" type="checkbox">
 | 
						|
			</td>
 | 
						|
 | 
						|
			<td>
 | 
						|
				<input type="number" class="form-control item-selltime" placeholder="Seconds" value=${sellTime}>
 | 
						|
			</td>
 | 
						|
 | 
						|
			<td class="delete-btn"></td>
 | 
						|
		</tr>
 | 
						|
	`)
 | 
						|
 | 
						|
	let deleteBtn = $(`<button type="button" class="btn btn-outline-danger">Remove</button>`);
 | 
						|
 | 
						|
	deleteBtn.click(function(){ itemDiv.remove(); })
 | 
						|
 | 
						|
	itemDiv.find(".delete-btn").append(deleteBtn);
 | 
						|
	
 | 
						|
	itemDiv.find(".item-blackmoney").prop("checked", blackMoney);
 | 
						|
 | 
						|
	itemDiv.data("object", object);
 | 
						|
 | 
						|
	itemsTableDiv.append(itemDiv);
 | 
						|
}
 | 
						|
 | 
						|
$("#market-modal-new-item-btn").click(function(){
 | 
						|
	let itemIdDiv = $("#market-modal-new-item-name-input");
 | 
						|
	const chooseObjectDiv = $("#market-new-item-name-div");
 | 
						|
	let object = getChoosenObject(chooseObjectDiv);
 | 
						|
 | 
						|
	let itemMinPriceDiv = $("#market-modal-new-item-min-price-input");
 | 
						|
	let itemMaxPriceDiv = $("#market-modal-new-item-max-price-input");
 | 
						|
 | 
						|
	let itemMinPrice = itemMinPriceDiv.val();
 | 
						|
	let itemMaxPrice = itemMaxPriceDiv.val();
 | 
						|
 | 
						|
	if(object) {
 | 
						|
		itemIdDiv.removeClass("is-invalid");
 | 
						|
	} else {
 | 
						|
		itemIdDiv.addClass("is-invalid");
 | 
						|
	}
 | 
						|
 | 
						|
	if(itemMinPrice) {
 | 
						|
		itemMinPriceDiv.removeClass("is-invalid");
 | 
						|
	} else {
 | 
						|
		itemMinPriceDiv.addClass("is-invalid");
 | 
						|
	}
 | 
						|
 | 
						|
	if(itemMaxPrice) {
 | 
						|
		itemMaxPriceDiv.removeClass("is-invalid");
 | 
						|
	} else {
 | 
						|
		itemMaxPriceDiv.addClass("is-invalid");
 | 
						|
	}
 | 
						|
 | 
						|
	if(object && itemMinPrice && itemMaxPrice) {
 | 
						|
		setChoosenObject(chooseObjectDiv, null);
 | 
						|
		itemMinPriceDiv.val("");
 | 
						|
		itemMaxPriceDiv.val("");
 | 
						|
 | 
						|
		addItemToMarket(object, itemMinPrice, itemMaxPrice);
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function inputAnimations(animations, cb) {
 | 
						|
	let animationsModal = $("#animations-modal");
 | 
						|
	let animationsForm = $("#animations-modal-form");
 | 
						|
 | 
						|
	$("#animations").empty();
 | 
						|
 | 
						|
	if(animations && animations.length > 0) {
 | 
						|
		animations.forEach(animation => {
 | 
						|
			addAnimation(animation)
 | 
						|
		})
 | 
						|
	} else {
 | 
						|
		addAnimation();
 | 
						|
	}
 | 
						|
 | 
						|
	animationsForm.unbind();
 | 
						|
	animationsForm.submit(function(event) {
 | 
						|
		if (!this.checkValidity()) {
 | 
						|
			event.stopPropagation();
 | 
						|
			event.preventDefault();
 | 
						|
			return;
 | 
						|
		} else {
 | 
						|
			$(animationsForm).removeClass("was-validated");
 | 
						|
		}
 | 
						|
 | 
						|
		let animations = [];
 | 
						|
 | 
						|
		$("#animations").find(".animation-container").each(function() {
 | 
						|
			let currentAnimDiv = $(this);
 | 
						|
 | 
						|
			let animType = currentAnimDiv.find(".animation-type").prop("checked");
 | 
						|
 | 
						|
			// If it's a scenario
 | 
						|
			if(animType) {
 | 
						|
				animations.push({
 | 
						|
					type: "scenario",
 | 
						|
					scenarioName: currentAnimDiv.find(".scenario-name").val(),
 | 
						|
					scenarioDuration: parseInt(currentAnimDiv.find(".scenario-duration").val())
 | 
						|
				})
 | 
						|
			} else { // If it's an animation
 | 
						|
				animations.push({
 | 
						|
					type: "animation",
 | 
						|
					animDict: currentAnimDiv.find(".animation-dictionary").val(),
 | 
						|
					animName: currentAnimDiv.find(".animation-name").val(),
 | 
						|
					animDuration: parseInt(currentAnimDiv.find(".animation-duration").val())
 | 
						|
				})
 | 
						|
			}
 | 
						|
		})
 | 
						|
 | 
						|
		animationsModal.modal("hide");
 | 
						|
 | 
						|
		cb(animations)
 | 
						|
	})
 | 
						|
 | 
						|
	animationsModal.modal("show");
 | 
						|
}
 | 
						|
 | 
						|
function addAnimation(animation) {
 | 
						|
	if(!animation) {
 | 
						|
		animation = {};
 | 
						|
	}
 | 
						|
 | 
						|
	let fullAnimContainer = $(`
 | 
						|
		<div class="animation-container mb-5">
 | 
						|
			<div class="form-check form-switch">
 | 
						|
				<input class="form-check-input animation-type" type="checkbox">
 | 
						|
				<label class="form-check-label animation-type-label">Animation</label>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<div class="animation">
 | 
						|
				<div class="input-group">
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:animations:animation_dictionary")}</span>
 | 
						|
					<input type="text" class="form-control animation-dictionary" value="${animation.animDict || ""}" required>
 | 
						|
				</div>
 | 
						|
				<div class="input-group mt-1">
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:animations:animation_name")}</span>
 | 
						|
					<input type="text" class="form-control animation-name" value="${animation.animName || ""}" required>
 | 
						|
				</div>
 | 
						|
				<div class="input-group mt-1">
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:animations:animation_duration")}</span>
 | 
						|
					<input type="number" class="form-control animation-duration" min="1" value="${animation.animDuration || ""}" required>
 | 
						|
				</div>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<div class="scenario" style="display: none">
 | 
						|
				<div class="input-group">
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:animations:scenario_name")}</span>
 | 
						|
					<input type="text" class="form-control scenario-name" value="${animation.scenarioName || ""}">
 | 
						|
				</div>
 | 
						|
				<div class="input-group mt-1">
 | 
						|
					<span class="input-group-text">${getLocalizedText("menu:dynamic:animations:scenario_duration")}</span>
 | 
						|
					<input type="number" class="form-control scenario-duration" min="1" value="${animation.scenarioDuration || ""}">
 | 
						|
				</div>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<button type="button" class="btn btn-warning float-end btn-sm mt-1 remove-anim-btn">${getLocalizedText("menu:dynamic:animations:remove")}</button>
 | 
						|
		</div>
 | 
						|
	`);
 | 
						|
 | 
						|
	let scenarioContainer = fullAnimContainer.find(".scenario");
 | 
						|
	let animationContainer = fullAnimContainer.find(".animation");
 | 
						|
	let switchLabel = fullAnimContainer.find(".animation-type-label");
 | 
						|
	let switchInput = fullAnimContainer.find(".animation-type");
 | 
						|
 | 
						|
	let removeAnimBtn = fullAnimContainer.find(".remove-anim-btn");
 | 
						|
 | 
						|
	fullAnimContainer.find(".animation-type").change(function() {
 | 
						|
		let isChecked = $(this).prop("checked");
 | 
						|
 | 
						|
		if(isChecked) {
 | 
						|
			scenarioContainer.show();
 | 
						|
			scenarioContainer.find(".form-control").prop("required", true);
 | 
						|
 | 
						|
			animationContainer.hide();
 | 
						|
			animationContainer.find(".form-control").prop("required", false);
 | 
						|
 | 
						|
			switchLabel.text("Scenario");
 | 
						|
		} else {
 | 
						|
			scenarioContainer.hide();
 | 
						|
			scenarioContainer.find(".form-control").prop("required", false);
 | 
						|
 | 
						|
			animationContainer.show();
 | 
						|
			animationContainer.find(".form-control").prop("required", true);
 | 
						|
 | 
						|
			switchLabel.text("Animation");
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	removeAnimBtn.click(function() {
 | 
						|
		$(this).parent().remove();
 | 
						|
	})
 | 
						|
 | 
						|
	if(animation && animation.type == "scenario") {
 | 
						|
		switchInput.prop("checked", true).trigger("change");
 | 
						|
	}
 | 
						|
 | 
						|
	$("#animations").append(fullAnimContainer);
 | 
						|
}
 | 
						|
 | 
						|
$("#add-animation-btn").click(function(){
 | 
						|
	addAnimation();
 | 
						|
})
 | 
						|
 | 
						|
$("#harvest-add-item-btn").click(function() {
 | 
						|
	addItemToHarvestable();
 | 
						|
});
 | 
						|
 | 
						|
$("#harvest-animations-btn").click(function() {
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
	inputAnimations(editMarkerModal.data("animations"), function(animations) {
 | 
						|
		editMarkerModal.data("animations", animations)
 | 
						|
	});
 | 
						|
});
 | 
						|
 | 
						|
$("#harvest-disappear-after-use").change(function(){
 | 
						|
	toggleDisappearsAfterUse( $(this).prop("checked") );
 | 
						|
});
 | 
						|
 | 
						|
$("#harvest-choose-account-btn").click(function() {
 | 
						|
	accountsDialog(accountName => {
 | 
						|
		$("#harvest-minimum-account-name").val(accountName);
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
function addItemToHarvestable(object, minQuantity = "", maxQuantity = "", secondsToHarvest = "", chances = "") {
 | 
						|
	var itemDiv = $(`
 | 
						|
	<div class="harvestable-item mb-3">
 | 
						|
		<div class="d-flex gap-3 align-items-center mb-2">
 | 
						|
			<div class="choose-object-div harvest-object-div input-group">
 | 
						|
				<input type="text" class="form-control choose-object-label" placeholder="Item name" readonly>
 | 
						|
				<button type="button" class="btn btn-secondary choose-object-btn" data-bs-toggle="tooltip" data-bs-placement="top" title="${getLocalizedText("menu:dialog:choose_item")}"><i class="bi bi-list-ul"></i></button>
 | 
						|
			</div>
 | 
						|
			<div class="input-group">
 | 
						|
				<span class="input-group-text">${getLocalizedText("menu:dynamic:harvest:min_quantity")}</span>
 | 
						|
				<input type="number" class="form-control harvest-item-min-quantity" placeholder="1" required value=${minQuantity}>
 | 
						|
			</div>
 | 
						|
			<div class="input-group">
 | 
						|
				<span class="input-group-text">${getLocalizedText("menu:dynamic:harvest:max_quantity")}</span>
 | 
						|
				<input type="number" class="form-control harvest-item-max-quantity" placeholder="1" required value=${maxQuantity}>
 | 
						|
			</div>
 | 
						|
		</div>
 | 
						|
 | 
						|
		<div class="d-flex gap-3 align-items-center mb-3">
 | 
						|
			<div class="input-group">
 | 
						|
				<span class="input-group-text">${getLocalizedText("menu:dynamic:harvest:time_to_harvest")}</span>
 | 
						|
				<input type="number" class="form-control harvest-item-time" placeholder="5" required value=${secondsToHarvest}>
 | 
						|
			</div>	
 | 
						|
			<div class="input-group">
 | 
						|
				<span class="input-group-text">${getLocalizedText("menu:dynamic:harvest:probabilty")}</span>
 | 
						|
				<input type="number" class="form-control harvest-item-chance" placeholder="50%" required value=${chances}>
 | 
						|
			</div>
 | 
						|
 | 
						|
			<button type="button" class="btn btn-sm btn-danger rounded remove-harvestable-item-btn">Remove</button>
 | 
						|
		</div>
 | 
						|
 | 
						|
		<hr>
 | 
						|
	</div>
 | 
						|
	`);
 | 
						|
 | 
						|
	$(itemDiv).find(".remove-harvestable-item-btn").click(function() {
 | 
						|
		$(this).parents(".harvestable-item").remove();
 | 
						|
	})
 | 
						|
 | 
						|
	loadChooseObjectDivs( $(itemDiv).find(".choose-object-div"), object);
 | 
						|
 | 
						|
	$("#harvest-modal-items").append(itemDiv);
 | 
						|
}
 | 
						|
 | 
						|
$("#process-animations-btn").click(function() {
 | 
						|
	let editMarkerModal = $("#edit-marker-dialog-modal");
 | 
						|
 | 
						|
	inputAnimations(editMarkerModal.data("animations"), function(animations) {
 | 
						|
		editMarkerModal.data("animations", animations)
 | 
						|
	});
 | 
						|
})
 | 
						|
 | 
						|
$("#process-requires-minimum-account-money").change(function() {
 | 
						|
	let isEnabled = $(this).prop("checked");
 | 
						|
 | 
						|
	$("#process-minimum-account-name-div").find("input, button").prop("disabled", !isEnabled);
 | 
						|
	$("#process-minimum-account-name-div").find("input, button").prop("required", isEnabled);
 | 
						|
});
 | 
						|
 | 
						|
$("#process-choose-account-btn").click(function() {
 | 
						|
	accountsDialog(accountName => {
 | 
						|
		$("#process-minimum-account-name").val(accountName);
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
$("#job-shop-all-jobs").change(function() {
 | 
						|
	let jobRanks = $(this).data("ranks");
 | 
						|
	let jobName = $(this).val();
 | 
						|
 | 
						|
	let allRanksSelect = $("#job-shop-all-ranks");
 | 
						|
 | 
						|
	allRanksSelect.find(".job-shop-rank").remove();
 | 
						|
 | 
						|
	let ranksArray = Array.isArray(jobRanks[jobName]) ? jobRanks[jobName] : Object.values(jobRanks[jobName]);
 | 
						|
 | 
						|
	ranksArray.forEach(rank => {
 | 
						|
		let rankDiv = $(`<option class="job-shop-rank" value=${rank.grade}>${rank.label}</option>`);
 | 
						|
 | 
						|
		allRanksSelect.append(rankDiv);
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
function loadAllJobsOnlinePlayers() {
 | 
						|
	$.post(`https://${resName}/getAllJobsOnlinePlayers`, {}, function(jobsOnlinePlayers) {
 | 
						|
		allJobsStatisticsChart.data.labels = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0] = {};
 | 
						|
		allJobsStatisticsChart.data.datasets[0].data = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0].label = getLocalizedText("menu:online_players");
 | 
						|
		allJobsStatisticsChart.data.datasets[0].backgroundColor = "rgba(26, 188, 156, 0.5)";
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderColor = "rgba(26, 188, 156, 0.8)";
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderWidth = 1;
 | 
						|
		allJobsStatisticsChart.options.scales.y.ticks.callback = function(value, index, values) {
 | 
						|
			if (Number.isInteger(value)) { return value; } 
 | 
						|
		}
 | 
						|
 | 
						|
		let maximumValue = 0;
 | 
						|
 | 
						|
		jobsOnlinePlayers.forEach(jobData => {
 | 
						|
			allJobsStatisticsChart.data.labels.push(jobData.label);
 | 
						|
 | 
						|
			allJobsStatisticsChart.data.datasets[0].data.push(jobData.playersCount);
 | 
						|
 | 
						|
			if (jobData.playersCount > maximumValue) {
 | 
						|
				maximumValue = jobData.playersCount;
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		allJobsStatisticsChart.options.scales.y.suggestedMax = maximumValue + 1;
 | 
						|
 | 
						|
		allJobsStatisticsChart.update();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
function loadAllJobsTotalPlayers() {
 | 
						|
	$.post(`https://${resName}/getAllJobsTotalPlayers`, {}, function(jobsTotalPlayers) {
 | 
						|
		allJobsStatisticsChart.data.labels = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0] = {};
 | 
						|
		allJobsStatisticsChart.data.datasets[0].data = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0].label = getLocalizedText("menu:total_players");
 | 
						|
		allJobsStatisticsChart.data.datasets[0].backgroundColor = "rgba(26, 188, 156, 0.5)";
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderColor = "rgba(26, 188, 156, 0.8)";
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderWidth = 1;
 | 
						|
		allJobsStatisticsChart.options.scales.y.ticks.callback = function(value, index, values) {
 | 
						|
			if (Number.isInteger(value)) { return value; } 
 | 
						|
		}
 | 
						|
 | 
						|
		let maximumValue = 0;
 | 
						|
 | 
						|
		jobsTotalPlayers.forEach(jobData => {
 | 
						|
			allJobsStatisticsChart.data.labels.push(jobData.label);
 | 
						|
 | 
						|
			allJobsStatisticsChart.data.datasets[0].data.push(jobData.playersCount);
 | 
						|
 | 
						|
			if (jobData.playersCount > maximumValue) {
 | 
						|
				maximumValue = jobData.playersCount;
 | 
						|
			}
 | 
						|
		});
 | 
						|
		allJobsStatisticsChart.options.scales.y.suggestedMax = maximumValue + 1;
 | 
						|
 | 
						|
		allJobsStatisticsChart.update();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
function loadAllJobsSocietyMoney() {
 | 
						|
	$.post(`https://${resName}/getJobsSocietyMoney`, {}, function(jobsSocietyMoney) {
 | 
						|
		allJobsStatisticsChart.data.labels = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0] = {};
 | 
						|
		allJobsStatisticsChart.data.datasets[0].data = [];
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0].label = getLocalizedText("menu:society_money");
 | 
						|
		allJobsStatisticsChart.data.datasets[0].backgroundColor = "rgba(241, 196, 15, 0.4)";
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderColor = "rgba(241, 196, 15, 0.8)";
 | 
						|
 | 
						|
		allJobsStatisticsChart.data.datasets[0].borderWidth = 1;
 | 
						|
		allJobsStatisticsChart.options.scales.y.ticks.callback = function(value, index, values) {
 | 
						|
			return "$" + value;
 | 
						|
		}
 | 
						|
 | 
						|
		let maximumValue = 0;
 | 
						|
 | 
						|
		jobsSocietyMoney.forEach(jobData => {
 | 
						|
			allJobsStatisticsChart.data.labels.push(jobData.label);
 | 
						|
 | 
						|
			allJobsStatisticsChart.data.datasets[0].data.push(jobData.money);
 | 
						|
 | 
						|
			if (jobData.money > maximumValue) {
 | 
						|
				maximumValue = jobData.money;
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		allJobsStatisticsChart.options.scales.y.suggestedMax = maximumValue + maximumValue * 0.10;
 | 
						|
 | 
						|
		allJobsStatisticsChart.update();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
$('input[type=radio][name=all-jobs-statistics-type]').change(function() {
 | 
						|
    let action = $(this).val();
 | 
						|
 | 
						|
	switch(action) {
 | 
						|
		case "online-players": {
 | 
						|
			loadAllJobsOnlinePlayers();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "total-players": {
 | 
						|
			loadAllJobsTotalPlayers();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		case "society-money": {
 | 
						|
			loadAllJobsSocietyMoney();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
});
 | 
						|
 | 
						|
function loadRanksDistrubutions(){
 | 
						|
	let jobName = $("#edit-job").data("jobName");
 | 
						|
 | 
						|
	$.post(`https://${resName}/getRanksDistribution`, JSON.stringify({jobName: jobName }), function(rankDistribution) {
 | 
						|
		jobStatisticsChart.data.labels = [];
 | 
						|
 | 
						|
		jobStatisticsChart.data.datasets[0] = {};
 | 
						|
		jobStatisticsChart.data.datasets[0].data = [];
 | 
						|
 | 
						|
		jobStatisticsChart.data.datasets[0].label = getLocalizedText("menu:chart:players_in_rank");
 | 
						|
 | 
						|
		jobStatisticsChart.data.datasets[0].backgroundColor = "rgba(26, 188, 156, 0.5)";
 | 
						|
		jobStatisticsChart.data.datasets[0].borderColor = "rgba(26, 188, 156, 0.8)";
 | 
						|
		jobStatisticsChart.data.datasets[0].borderWidth = 1;
 | 
						|
 | 
						|
		jobStatisticsChart.options.scales.y.ticks.callback = function(value, index, values) {
 | 
						|
			if (Number.isInteger(value)) { return value; } 
 | 
						|
		}
 | 
						|
		
 | 
						|
		let maximumValue = 0;
 | 
						|
 | 
						|
		rankDistribution.forEach(rankData => {
 | 
						|
			jobStatisticsChart.data.labels.push(rankData.label);
 | 
						|
 | 
						|
			jobStatisticsChart.data.datasets[0].data.push(rankData.playersCount);
 | 
						|
 | 
						|
			if (rankData.playersCount > maximumValue) {
 | 
						|
				maximumValue = rankData.playersCount;
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		jobStatisticsChart.options.scales.y.suggestedMax = maximumValue + 1;
 | 
						|
 | 
						|
		jobStatisticsChart.update();
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
async function settingsTour() {
 | 
						|
	const introName = "settingsTour";
 | 
						|
	if(await hasIntroBeenViewed(introName)) return;
 | 
						|
 | 
						|
	const tour = new Shepherd.Tour({
 | 
						|
		useModalOverlay: true,
 | 
						|
		defaultStepOptions: {
 | 
						|
			scrollTo: true
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:settings"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#settings',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:settings_external_names"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#settings-external-scripts-names',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:settings_modules"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#settings-modules',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.start();
 | 
						|
 | 
						|
	registerIntroView(introName);
 | 
						|
}
 | 
						|
$("#settings-tab").click(settingsTour);
 | 
						|
 | 
						|
async function jobTour() {
 | 
						|
	const introName = "jobTour";
 | 
						|
	if(await hasIntroBeenViewed(introName)) return;
 | 
						|
 | 
						|
	const tour = new Shepherd.Tour({
 | 
						|
		useModalOverlay: true,
 | 
						|
		defaultStepOptions: {
 | 
						|
			scrollTo: true
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:ranks"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#create-rank-btn',
 | 
						|
			on: 'left'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
	
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:job_settings"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#job-settings-tab',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:okay"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.start();
 | 
						|
 | 
						|
	registerIntroView(introName);
 | 
						|
}
 | 
						|
 | 
						|
async function mainMenuTour() {
 | 
						|
	const introName = "mainMenuTour";
 | 
						|
	if(await hasIntroBeenViewed(introName)) return;
 | 
						|
 | 
						|
	const tour = new Shepherd.Tour({
 | 
						|
		useModalOverlay: true,
 | 
						|
		defaultStepOptions: {
 | 
						|
			scrollTo: true
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:welcome"),
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:jobs"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#create-job-btn',
 | 
						|
			on: 'left'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:jobs_table"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#jobs-container',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:next"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.addStep({
 | 
						|
		text: getLocalizedText("menu:intro:public_markers"),
 | 
						|
		attachTo: {
 | 
						|
			element: '#public-markers-tab',
 | 
						|
			on: 'right'
 | 
						|
		},
 | 
						|
		buttons: [
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:skip"),
 | 
						|
				action: tour.cancel,
 | 
						|
				secondary: true
 | 
						|
			},
 | 
						|
			{
 | 
						|
				text: getLocalizedText("menu:intro:okay"),
 | 
						|
				action: tour.next
 | 
						|
			}
 | 
						|
		]
 | 
						|
	});
 | 
						|
 | 
						|
	tour.start();
 | 
						|
 | 
						|
	registerIntroView(introName);
 | 
						|
}
 | 
						|
 | 
						|
// [[ NEXUS ]]
 | 
						|
const voteJobRater = raterJs({
 | 
						|
	starSize: 35,
 | 
						|
	element: document.querySelector("#vote-job-rater"),
 | 
						|
	rateCallback: async function rateCallback(rating, done) {
 | 
						|
		const jobId = $("#nexus-modal").data("jobInstance").id;
 | 
						|
		const success = await $.post(`https://${resName}/nexus/rateJob`, JSON.stringify({rating, jobId}));
 | 
						|
		if(success) voteJobRater.setRating(rating);
 | 
						|
 | 
						|
		done();
 | 
						|
	}
 | 
						|
});
 | 
						|
 | 
						|
const averageJobVotes = raterJs({
 | 
						|
	starSize: 20,
 | 
						|
	readOnly: true,
 | 
						|
	element: document.querySelector("#nexus-modal-job-average-rating"),
 | 
						|
});
 | 
						|
 | 
						|
$("#nexus-import-job-btn").click(async function() {
 | 
						|
	const jobId = $("#nexus-modal").data("jobInstance").id;
 | 
						|
 | 
						|
	const response = await $.post(`https://${resName}/nexus/importJob`, JSON.stringify({jobId}));
 | 
						|
	$("#nexus-modal").modal("hide");
 | 
						|
 | 
						|
	if(response === true) refresh();
 | 
						|
 | 
						|
	showServerResponse(response);
 | 
						|
});
 | 
						|
 | 
						|
let nexusDataTable = $("#nexus-jobs-container").DataTable({
 | 
						|
	"lengthMenu": [5, 10, 15, 20],
 | 
						|
	"pageLength": 10,
 | 
						|
	"order": [[3, 'desc'], [4, 'desc']],
 | 
						|
	"createdRow": function (row, data, index) {
 | 
						|
		$(row).addClass("clickable");
 | 
						|
		$(row).click(function () {
 | 
						|
			const jobInstance = $(this).data("jobInstance");
 | 
						|
			showJobInstance(jobInstance);
 | 
						|
			$("#nexus-modal").modal("show");
 | 
						|
		});
 | 
						|
	},
 | 
						|
	"columnDefs": [{ "defaultContent": "???", "targets": "_all" }]
 | 
						|
});
 | 
						|
  
 | 
						|
function showJobInstance(jobInstance) {
 | 
						|
	$("#nexus-modal").data("jobInstance", jobInstance);
 | 
						|
 | 
						|
	$("#nexus-modal-job-listing-label").text(jobInstance.label);
 | 
						|
	$("#nexus-modal-job-label").text(jobInstance.jobConfiguration.label);
 | 
						|
	$("#nexus-modal-job-name").text(jobInstance.jobConfiguration.name);
 | 
						|
	$("#nexus-modal-job-description").text(jobInstance.description || getLocalizedText("menu:nexus:no_description"));
 | 
						|
	$("#nexus-modal-job-author").text(jobInstance.author);
 | 
						|
 | 
						|
	// Votes
 | 
						|
	if(jobInstance?.votes?.total > 0) {
 | 
						|
		averageJobVotes.setRating(jobInstance?.votes.averageRating);
 | 
						|
	} else {
 | 
						|
		averageJobVotes.setRating(0);
 | 
						|
	}
 | 
						|
 | 
						|
	$("#nexus-modal-job-total-votes").text(jobInstance.votes?.total || 0);
 | 
						|
 | 
						|
	// This server vote
 | 
						|
	voteJobRater.setRating(0);
 | 
						|
 | 
						|
	// Actions
 | 
						|
	const actionsListDiv = $("#nexus-modal-job-actions-list");
 | 
						|
	actionsListDiv.empty();
 | 
						|
 | 
						|
	let currentRow = null;
 | 
						|
 | 
						|
	for(let i=0; i < ACTIONS_LABELS.length; i++) {
 | 
						|
		const {action, label} = ACTIONS_LABELS[i];
 | 
						|
		const actionValue = jobInstance.jobConfiguration.actions[action];
 | 
						|
		const isEnabled = action == "placeableObjects" ? actionValue.length > 0 : actionValue;
 | 
						|
 | 
						|
		if(i % 2 === 0) {
 | 
						|
			actionsListDiv.append(currentRow);
 | 
						|
			currentRow = null;
 | 
						|
		}
 | 
						|
 | 
						|
		const emoji = isEnabled ? "✅" : "❌";
 | 
						|
 | 
						|
		if(currentRow == null) currentRow = $("<p class='d-flex justify-content-between'></p>");
 | 
						|
		
 | 
						|
		currentRow.append( i % 2 !== 0 ? `<span>${label} ${emoji}</span>` : `<span>${emoji} ${label}</span>`);
 | 
						|
	}
 | 
						|
 | 
						|
	if (currentRow !== null) actionsListDiv.append(currentRow);
 | 
						|
 | 
						|
	// Ranks
 | 
						|
	const ranksListDiv = $("#nexus-modal-job-ranks-list");
 | 
						|
	ranksListDiv.empty();
 | 
						|
 | 
						|
	for(const [rank, rankInfo] of Object.entries(jobInstance.jobConfiguration.ranks)) {
 | 
						|
		const rankDiv = $(`
 | 
						|
			<li class="list-group-item d-flex justify-content-between align-items-center">
 | 
						|
				${rankInfo.label}
 | 
						|
				<span class="badge bg-primary rounded-pill">${rank}</span>
 | 
						|
			</li>
 | 
						|
		`);
 | 
						|
 | 
						|
		ranksListDiv.append(rankDiv);
 | 
						|
	}
 | 
						|
 | 
						|
	// Markers count
 | 
						|
	const markersCount = jobInstance.jobMarkers ? Object.keys(jobInstance.jobMarkers).length : 0;
 | 
						|
	$("#nexus-modal-job-included-markers-count").text(markersCount);
 | 
						|
}
 | 
						|
 | 
						|
$("#upload-job-to-nexus-btn").click(async function() {
 | 
						|
	const jobName = await singleJobDialog();
 | 
						|
	if(!jobName) return;
 | 
						|
 | 
						|
	$("#nexus-modal-upload").data("jobName", jobName);
 | 
						|
 | 
						|
	$("#nexus-upload-label").val("");
 | 
						|
	$("#nexus-upload-description").val("");
 | 
						|
	$("#nexus-upload-include-job-markers").prop("checked", true).change();
 | 
						|
 | 
						|
	$("#nexus-upload-accept-tos").prop("checked", false)
 | 
						|
	$("#nexus-upload-accept-sharing-data").prop("checked", false)
 | 
						|
	
 | 
						|
	const allJobs = await $.post(`https://${resName}/getJobsData`);
 | 
						|
	const jobData = allJobs[jobName];
 | 
						|
 | 
						|
	$("#nexus-upload-label-shared-job-label").val(jobData.label)
 | 
						|
	$("#nexus-upload-label-shared-job-id").val(jobData.name)
 | 
						|
 | 
						|
	const jobMarkers = await $.post(`https://${resName}/retrieveJobMarkers`, JSON.stringify({jobName}));
 | 
						|
 | 
						|
	$("#nexus-upload-included-markers-count").val(Object.keys(jobMarkers).length)
 | 
						|
 | 
						|
	$("#nexus-modal-upload").modal("show");
 | 
						|
});
 | 
						|
 | 
						|
$("#nexus-upload-include-job-markers").change(function() {
 | 
						|
	const isChecked = $(this).prop("checked");
 | 
						|
	$("#nexus-upload-included-markers-count-div").toggle(isChecked);
 | 
						|
});
 | 
						|
 | 
						|
$("#nexus-upload-form").submit(async function(event) {
 | 
						|
	if(isThereAnyErrorInForm(event)) return;
 | 
						|
 | 
						|
	const dataToUpload = {
 | 
						|
		jobName: $("#nexus-modal-upload").data("jobName"),
 | 
						|
		label: $("#nexus-upload-label").val(),
 | 
						|
		description: $("#nexus-upload-description").val(),
 | 
						|
		includeJobMarkers: $("#nexus-upload-include-job-markers").prop("checked"),
 | 
						|
	}
 | 
						|
 | 
						|
	const result = await $.post(`https://${resName}/nexus/uploadJob`, JSON.stringify(dataToUpload));
 | 
						|
 | 
						|
	if(result == true) {
 | 
						|
		swal("Success", getLocalizedText("menu:nexus:upload_success"), "success");
 | 
						|
		resetNexus();
 | 
						|
	} else {
 | 
						|
		swal("Error", result, "error");
 | 
						|
	}
 | 
						|
 | 
						|
	$("#nexus-modal-upload").modal("hide");
 | 
						|
});
 | 
						|
 | 
						|
$("#enter-in-nexus-btn").click(async function() {
 | 
						|
	$("#nexus-login").find(".spinner-border").show();
 | 
						|
	$("#enter-in-nexus-label").text("...");
 | 
						|
 | 
						|
	const jobs = await $.get(`https://${resName}/nexus/getJobs`);
 | 
						|
	if(!jobs) {
 | 
						|
		swal("Error", getLocalizedText("menu:nexus:not_available"), "error");
 | 
						|
		resetNexus();
 | 
						|
		return;
 | 
						|
	} 
 | 
						|
	
 | 
						|
	nexusDataTable.clear();
 | 
						|
 | 
						|
	for(const[_, jobInstance] of Object.entries(jobs)) {
 | 
						|
		const roundedAverageRating = jobInstance?.votes?.averageRating ? Math.round(jobInstance.votes.averageRating) : 0;
 | 
						|
		const ratingStars = jobInstance?.votes?.total ? "⭐".repeat(roundedAverageRating) : getLocalizedText("menu:nexus:not_rated");
 | 
						|
		const limitedDescription = jobInstance.description?.length > 30 ? jobInstance.description.substring(0, 30) + "..." : jobInstance.description;
 | 
						|
		const jobMarkersCount = jobInstance.jobMarkers ? Object.keys(jobInstance.jobMarkers).length : 0;
 | 
						|
 | 
						|
		const rawRow = nexusDataTable.row.add( [jobInstance.label, limitedDescription, jobMarkersCount, ratingStars, jobInstance.votes?.total || 0, jobInstance.author] );
 | 
						|
 | 
						|
		const rowDiv = $(rawRow.node());
 | 
						|
		$(rowDiv).data("jobInstance", jobInstance);
 | 
						|
	}
 | 
						|
 | 
						|
	nexusDataTable.draw();
 | 
						|
 | 
						|
	$("#nexus-login").hide();
 | 
						|
	$("#nexus-container").show();
 | 
						|
})
 | 
						|
 | 
						|
function resetNexus() {
 | 
						|
	$("#nexus-login").show();
 | 
						|
	$("#nexus-login").find(".spinner-border").hide();
 | 
						|
	$("#enter-in-nexus-label").text("Enter in Nexus");
 | 
						|
 | 
						|
	$("#nexus-container").hide();
 | 
						|
}
 | 
						|
 | 
						|
async function init(version, fullConfig) {
 | 
						|
	$("#job-creator-version").text(version);
 | 
						|
 | 
						|
	loadSettings(fullConfig);
 | 
						|
 | 
						|
	resetNexus()
 | 
						|
 | 
						|
	$("#job-creator").show();
 | 
						|
	
 | 
						|
	await refresh();
 | 
						|
 | 
						|
	mainMenuTour();
 | 
						|
	refreshAnnouncements();
 | 
						|
}
 | 
						|
 | 
						|
function exit() {
 | 
						|
	// Resets current active tab (jobs is the default one)
 | 
						|
	$("#job-creator").find(".nav-link, .tab-pane").each(function() {
 | 
						|
		if($(this).data("isDefault") == "1") {
 | 
						|
			$(this).addClass(["active", "show"])
 | 
						|
		} else {
 | 
						|
			$(this).removeClass(["active", "show"])
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	$("#job-creator").hide();
 | 
						|
 | 
						|
	$.post(`https://${resName}/exit`)
 | 
						|
}
 | 
						|
 | 
						|
window.addEventListener('message', (event) => {
 | 
						|
	let data = event.data;
 | 
						|
	let action = data.action;
 | 
						|
 | 
						|
	if (action == 'show') {
 | 
						|
		init(data.version, data.fullConfig);
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#close-main-btn").click(function () {
 | 
						|
	exit()
 | 
						|
})
 | 
						|
 | 
						|
/*
 | 
						|
 
 | 
						|
░██████╗███████╗████████╗████████╗██╗███╗░░██╗░██████╗░░██████╗
 | 
						|
██╔════╝██╔════╝╚══██╔══╝╚══██╔══╝██║████╗░██║██╔════╝░██╔════╝
 | 
						|
╚█████╗░█████╗░░░░░██║░░░░░░██║░░░██║██╔██╗██║██║░░██╗░╚█████╗░
 | 
						|
░╚═══██╗██╔══╝░░░░░██║░░░░░░██║░░░██║██║╚████║██║░░╚██╗░╚═══██╗
 | 
						|
██████╔╝███████╗░░░██║░░░░░░██║░░░██║██║░╚███║╚██████╔╝██████╔╝
 | 
						|
╚═════╝░╚══════╝░░░╚═╝░░░░░░╚═╝░░░╚═╝╚═╝░░╚══╝░╚═════╝░╚═════╝░
 | 
						|
*/
 | 
						|
 | 
						|
function addJobInWhitelistForOffDutyJobs(jobName) {
 | 
						|
	const listDiv = $("#settings_whitelistForOffdutyJobs");
 | 
						|
 | 
						|
	listDiv.append(`
 | 
						|
		<li class="list-group-item d-flex justify-content-between align-items-center job" data-job-name="${jobName}">
 | 
						|
			${jobName}
 | 
						|
			<button class="btn remove-job"><i class="bi bi-x"></i></button>
 | 
						|
		</li>
 | 
						|
	`);
 | 
						|
 | 
						|
	listDiv.find(".remove-job").click(function() {
 | 
						|
		$(this).parent().remove();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_addNewWhitelistedJobForOffduty").click(function() {
 | 
						|
	let inputDiv = $("#settings_newWhitelistedJobForOffduty");
 | 
						|
 | 
						|
	let jobName = inputDiv.val();
 | 
						|
	if(!jobName) return;
 | 
						|
	
 | 
						|
	inputDiv.val("");
 | 
						|
	addJobInWhitelistForOffDutyJobs(jobName);
 | 
						|
});
 | 
						|
 | 
						|
function getWhitelistedJobsForOffduty() {
 | 
						|
	let jobs = {};
 | 
						|
 | 
						|
	$("#settings_whitelistForOffdutyJobs").find(".job").each(function() {
 | 
						|
		let jobName = $(this).data("jobName");
 | 
						|
		jobs[jobName] = true;
 | 
						|
	});
 | 
						|
 | 
						|
	return jobs;
 | 
						|
}
 | 
						|
 | 
						|
function clearLicensesInList(type) {
 | 
						|
	if(type === "driver") {
 | 
						|
		var listDiv = $("#settings_drivingLicensesList");
 | 
						|
	} else if(type === "weapon") {
 | 
						|
		var listDiv = $("#settings_weaponLicensesList");
 | 
						|
	}
 | 
						|
 | 
						|
	if(listDiv) {
 | 
						|
		listDiv.empty();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
function addNewLicenseInList(licenseName, type) {
 | 
						|
	if(type === "driver") {
 | 
						|
		var listDiv = $("#settings_drivingLicensesList");
 | 
						|
	} else if(type === "weapon") {
 | 
						|
		var listDiv = $("#settings_weaponLicensesList");
 | 
						|
	}
 | 
						|
 | 
						|
	if(listDiv) {
 | 
						|
		listDiv.append(`
 | 
						|
			<li class="list-group-item d-flex justify-content-between align-items-center license" data-license="${licenseName}">
 | 
						|
				${licenseName}
 | 
						|
				<button class="btn remove-license"><i class="bi bi-x"></i></button>
 | 
						|
			</li>
 | 
						|
		`);
 | 
						|
 | 
						|
		listDiv.find(".remove-license").click(function() {
 | 
						|
			$(this).parent().remove();
 | 
						|
		});
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_addNewDrivingLicenseBtn").click(function() {
 | 
						|
	let drivingLicenseName = $("#settings_newDrivingLicense").val();
 | 
						|
	
 | 
						|
	if(drivingLicenseName) {
 | 
						|
		$("#settings_newDrivingLicense").val("");
 | 
						|
		addNewLicenseInList(drivingLicenseName, "driver");
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
$("#settings_addNewWeaponLicenseBtn").click(function() {
 | 
						|
	let weaponLicenseName = $("#settings_newWeaponLicense").val();
 | 
						|
	
 | 
						|
	if(weaponLicenseName) {
 | 
						|
		$("#settings_newWeaponLicense").val("");
 | 
						|
		addNewLicenseInList(weaponLicenseName, "weapon");
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function getSettingsLicenses(type) {
 | 
						|
	if(type === "driver") {
 | 
						|
		var listDiv = $("#settings_drivingLicensesList");
 | 
						|
	} else if(type === "weapon") {
 | 
						|
		var listDiv = $("#settings_weaponLicensesList");
 | 
						|
	}
 | 
						|
 | 
						|
	let licenses = {};
 | 
						|
 | 
						|
	listDiv.find(".license").each(function(index, element) {
 | 
						|
		licenses[ $(element).data("license") ] = true;
 | 
						|
	})
 | 
						|
 | 
						|
	return licenses;
 | 
						|
}
 | 
						|
 | 
						|
function getSeparatedDiscordWebhooks() {
 | 
						|
	let webhooks = {};
 | 
						|
 | 
						|
	$("#settings_specific_webhooks").find(".form-control").each(function(index, element) {
 | 
						|
		let markerType = $(element).data("markerType");
 | 
						|
		let webhook = $(element).val();
 | 
						|
 | 
						|
		if(webhook) {
 | 
						|
			webhooks[markerType] = webhook;
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	return webhooks;
 | 
						|
}
 | 
						|
 | 
						|
$("#settings").submit(async function(event) {
 | 
						|
	if(isThereAnyErrorInForm(event)) return;
 | 
						|
 | 
						|
	let clientSettings = {
 | 
						|
		markerDistance: parseFloat( $("#settings_markerDistance").val() ),
 | 
						|
		use3Dtext: $("#settings_use3Dtext").prop("checked"),
 | 
						|
		textSize: parseInt( $("#settings_textSize").val() ),
 | 
						|
		textFont: parseInt( $("#settings_textFont").val() ),
 | 
						|
		carLockpickTime: parseInt( $("#settings_carLockpickTime").val() ),
 | 
						|
		enableAlarmWhenLockpicking: $("#settings_enableAlarmWhenLockpicking").prop("checked"),
 | 
						|
		useJSFourIdCard: $("#settings_useJSFourIdCard").prop("checked"),
 | 
						|
		canUseActionsMenuWhileOffDuty: $("#settings_canUseActionsMenuWhileOffDuty").prop("checked"),
 | 
						|
		licenses: {
 | 
						|
			driver: getSettingsLicenses("driver"),
 | 
						|
			weapon: getSettingsLicenses("weapon"),
 | 
						|
		},
 | 
						|
		marketSellOnePerTime: $("#settings_marketSellOnePerTime").prop("checked"),
 | 
						|
		menuPosition: $("#settings_menuPosition").val(),
 | 
						|
		actionsMenuKey: $("#settings_actionsMenuKey").val(),
 | 
						|
		freezeWhenSoftHandcuffed: $("#settings_freezeWhenSoftHandcuffed").prop("checked"),
 | 
						|
		freezeWhenHardHandcuffed: $("#settings_freezeWhenHardHandcuffed").prop("checked"),
 | 
						|
		searchRequiresHandcuffState: $("#settings_searchRequiresHandcuffState").prop("checked"),
 | 
						|
		targetingScript:$("#settings-targeting-script").val(),
 | 
						|
		whitelistedControlsWhileHandcuffed: getAllWhitelistedControlsWhileHandcuffed(),
 | 
						|
		toggleDrag: {
 | 
						|
			enabled: $("#toggle-drag-enabled").prop("checked"),
 | 
						|
			key: $("#toggle-drag-default-key").val()
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	let sharedSettings = {
 | 
						|
		allowAfkFarming: $("#settings_allowAfkFarming").prop("checked"),
 | 
						|
		locale: $("#settings_locale").val(),
 | 
						|
		externalScriptsNames: getIntegrationSettings(),
 | 
						|
		modules: getModulesSettings(),
 | 
						|
		handcuffsEnableSelfRelease: $("#settings_handcuffsEnableSelfRelease").prop("checked"),
 | 
						|
	}
 | 
						|
 | 
						|
	let serverSettings = {
 | 
						|
		unemployedJob: $("#settings_unemployedJob").val(),
 | 
						|
		unemployedGrade: parseInt( $("#settings_unemployedGrade").val() ),
 | 
						|
 | 
						|
		acePermission: $("#settings_acePermission").val(),
 | 
						|
 | 
						|
		areDiscordLogsActive: $("#settings_isDiscordLogActive").prop("checked"),
 | 
						|
		mainDiscordWebhook: $("#settings_discordWebhook").val(),
 | 
						|
		specificWebhooks: getSeparatedDiscordWebhooks(),
 | 
						|
 | 
						|
		handcuffRequireItem: $("#settings_handcuffRequireItem").prop("checked"),
 | 
						|
		handcuffsItemName: $("#settings_handcuffsItemName").val(),
 | 
						|
		handcuffsRemoveOnUse: $("#settings_handcuffsRemoveOnUse").prop("checked"),
 | 
						|
 | 
						|
		lockpickCarRequireItem: $("#settings_lockpickCarRequireItem").prop("checked"),
 | 
						|
		lockpickItemName: $("#settings_lockpickItemName").val(),
 | 
						|
		lockpickRemoveOnUse: $("#settings_lockpickRemoveOnUse").prop("checked"),
 | 
						|
 | 
						|
		robbableAccounts: getRobbableAccounts(),
 | 
						|
 | 
						|
		canAlwaysCarryItem: $("#settings_canAlwaysCarryItem").prop("checked"),
 | 
						|
 | 
						|
		depositableInSafeAccounts: getDepositableInSafeAccounts(),
 | 
						|
 | 
						|
		repairVehicleRequireItem: $("#settings_repairVehicleRequireItem").prop("checked"),
 | 
						|
		repairVehicleItemName: $("#settings_repairVehicleItemName").val(),
 | 
						|
		repairVehicleRemoveOnUse: $("#settings_repairVehicleRemoveOnUse").prop("checked"),
 | 
						|
 | 
						|
		cleanVehicleRequireItem: $("#settings_cleanVehicleRequireItem").prop("checked"),
 | 
						|
		cleanVehicleItemName: $("#settings_cleanVehicleItemName").val(),
 | 
						|
		cleanVehicleRemoveOnUse: $("#settings_cleanVehicleRemoveOnUse").prop("checked"),
 | 
						|
 | 
						|
		healRequireItem: $("#settings_healRequireItem").prop("checked"),
 | 
						|
		healItemName: $("#settings_healItemName").val(),
 | 
						|
		healRemoveOnUse: $("#settings_healRemoveOnUse").prop("checked"),
 | 
						|
		
 | 
						|
		reviveRequireItem: $("#settings_reviveRequireItem").prop("checked"),
 | 
						|
		reviveItemName: $("#settings_reviveItemName").val(),
 | 
						|
		reviveRemoveOnuse: $("#settings_reviveRemoveOnUse").prop("checked"),
 | 
						|
 | 
						|
		enablePropertyOutfits: $("#settings_enablePropertyOutfits").prop("checked"),
 | 
						|
 | 
						|
		parkAllOwnedVehiclesOnRestart: $("#settings_parkAllOwnedVehiclesOnRestart").prop("checked"),
 | 
						|
		
 | 
						|
		whitelistedJobsForOffduty: getWhitelistedJobsForOffduty(),
 | 
						|
 | 
						|
		blackMoney: getBlackMoneySettings(),
 | 
						|
	}
 | 
						|
 | 
						|
	const response = await $.post(`https://${resName}/saveSettings`, JSON.stringify({
 | 
						|
		clientSettings: clientSettings,
 | 
						|
		serverSettings: serverSettings,
 | 
						|
		sharedSettings: sharedSettings
 | 
						|
	}));
 | 
						|
	showServerResponse(response);
 | 
						|
 | 
						|
	refreshTranslations(sharedSettings.locale);
 | 
						|
})
 | 
						|
 | 
						|
async function toggleDiscordLogsInSettings(enable) {
 | 
						|
	$("#settings_discordWebhook").prop("disabled", !enable);
 | 
						|
	$("#settings_discordWebhook").prop("required", enable);
 | 
						|
	
 | 
						|
	$("#settings_specific_webhooks").find(`.form-control`).prop("disabled", !enable);
 | 
						|
 | 
						|
	if(await getInventoryScriptUsed() !== "default" || await getFramework() !== "ESX") {
 | 
						|
		$("#settings_specific_webhooks").find(`.form-control[data-marker-type="stash"]`).prop("disabled", true);
 | 
						|
		$("#settings_specific_webhooks").find(`.form-control[data-marker-type="armory"]`).prop("disabled", true);
 | 
						|
		$("#settings_specific_webhooks").find(`.form-control[data-marker-type="safe"]`).prop("disabled", true);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_isDiscordLogActive").change(function() {
 | 
						|
	let enabled = $(this).prop("checked");
 | 
						|
 | 
						|
	toggleDiscordLogsInSettings(enabled);
 | 
						|
})
 | 
						|
 | 
						|
$(`input[type=radio][name=black-money-worth-type]`).change(function() {
 | 
						|
	const worthType = $(this).val();
 | 
						|
	const isMetadata = worthType === "metadata";
 | 
						|
 | 
						|
	$("#settings-black-money-metadata-field-id-div").toggle(isMetadata);
 | 
						|
	$("#settings-black-money-metadata-field-id").prop("required", isMetadata);
 | 
						|
});
 | 
						|
 | 
						|
function setBlackMoneySettings(blackMoney) {
 | 
						|
	$(`input[type=radio][name=black-money-worth-type][value=${blackMoney.worthType}]`).prop("checked", true).change();
 | 
						|
	setChoosenObject( $("#settings-choose-black-money-div"), blackMoney.object);
 | 
						|
	$("#settings-black-money-metadata-field-id").val(blackMoney.metadataFieldId);
 | 
						|
}
 | 
						|
 | 
						|
function getBlackMoneySettings() {
 | 
						|
	return {
 | 
						|
		worthType: $(`input[type=radio][name=black-money-worth-type]:checked`).val(),
 | 
						|
		object: getChoosenObject( $("#settings-choose-black-money-div") ),
 | 
						|
		metadataFieldId: $("#settings-black-money-metadata-field-id").val()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
async function loadSettings(fullConfig) {
 | 
						|
	// CLIENT
 | 
						|
	$("#settings_markerDistance").val(fullConfig.markerDistance);
 | 
						|
 | 
						|
	// 3D Text
 | 
						|
	$("#settings_use3Dtext").prop("checked", fullConfig.use3Dtext);
 | 
						|
	$("#settings_textSize").val(fullConfig.textSize);
 | 
						|
	setTomSelectValue("#settings_textFont", fullConfig.textFont)
 | 
						|
	
 | 
						|
	// Targeting
 | 
						|
	setTomSelectValue("#settings-targeting-script", fullConfig.targetingScript)
 | 
						|
 | 
						|
	// Black money
 | 
						|
	setBlackMoneySettings(fullConfig.blackMoney);
 | 
						|
 | 
						|
	// Car lockpicking
 | 
						|
	$("#settings_carLockpickTime").val(fullConfig.carLockpickTime);
 | 
						|
	$("#settings_enableAlarmWhenLockpicking").prop("checked", fullConfig.enableAlarmWhenLockpicking);
 | 
						|
 | 
						|
	$("#settings_useJSFourIdCard").prop("checked", fullConfig.useJSFourIdCard);
 | 
						|
	$("#settings_canUseActionsMenuWhileOffDuty").prop("checked", fullConfig.canUseActionsMenuWhileOffDuty);
 | 
						|
	$("#settings_marketSellOnePerTime").prop("checked", fullConfig.marketSellOnePerTime);
 | 
						|
 | 
						|
	// Whitelisted jobs for off duty
 | 
						|
	$("#off-duty-jobs-whitelist").toggle( await getFramework() == "ESX" );
 | 
						|
	$("#settings_whitelistForOffdutyJobs").empty();
 | 
						|
	for(const jobName of Object.keys(fullConfig.whitelistedJobsForOffduty || {})) {
 | 
						|
		addJobInWhitelistForOffDutyJobs(jobName)
 | 
						|
	}
 | 
						|
 | 
						|
	// Weapon/Driving licenses
 | 
						|
	if(fullConfig.licenses) {
 | 
						|
		for(const[licenseType, licenses] of Object.entries(fullConfig.licenses)) {
 | 
						|
			clearLicensesInList(licenseType);
 | 
						|
 | 
						|
			Object.keys(licenses).forEach(licenseName => {
 | 
						|
				addNewLicenseInList(licenseName, licenseType);
 | 
						|
			})
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// Menu position
 | 
						|
	setTomSelectValue("#settings_menuPosition", fullConfig.menuPosition)
 | 
						|
 | 
						|
	// Actions key
 | 
						|
	$("#settings_actionsMenuKey").val(fullConfig.actionsMenuKey);
 | 
						|
 | 
						|
	loadIntegrationsSettings(fullConfig.externalScriptsNames);
 | 
						|
	loadModulesSettings(fullConfig.modules);
 | 
						|
 | 
						|
	// SHARED
 | 
						|
	$("#settings_allowAfkFarming").prop("checked", fullConfig.allowAfkFarming);
 | 
						|
	setTomSelectValue("#settings_locale", fullConfig.locale)
 | 
						|
 | 
						|
	// Generic
 | 
						|
	$("#settings_unemployedJob").val(fullConfig.unemployedJob);
 | 
						|
	$("#settings_unemployedGrade").val(fullConfig.unemployedGrade);
 | 
						|
	$("#settings_acePermission").val(fullConfig.acePermission);
 | 
						|
 | 
						|
	// Discord logs webhooks
 | 
						|
	$("#settings_isDiscordLogActive").prop("checked", fullConfig.areDiscordLogsActive);
 | 
						|
	toggleDiscordLogsInSettings(fullConfig.areDiscordLogsActive);
 | 
						|
 | 
						|
	$("#settings_discordWebhook").val(fullConfig.mainDiscordWebhook);
 | 
						|
 | 
						|
	// Handcuffs
 | 
						|
	toggleSettingsHandcuffsRequireItem(fullConfig.handcuffRequireItem);
 | 
						|
	$("#settings_handcuffsItemName").val(fullConfig.handcuffsItemName)
 | 
						|
	$("#settings_handcuffsRemoveOnUse").prop("checked", fullConfig.handcuffsRemoveOnUse)
 | 
						|
	$("#settings_handcuffsEnableSelfRelease").prop("checked", fullConfig.handcuffsEnableSelfRelease);
 | 
						|
 | 
						|
	$("#settings_freezeWhenSoftHandcuffed").prop("checked", fullConfig.freezeWhenSoftHandcuffed);
 | 
						|
	$("#settings_freezeWhenHardHandcuffed").prop("checked", fullConfig.freezeWhenHardHandcuffed);
 | 
						|
 | 
						|
	// Whitelisted controls while handcuffed
 | 
						|
	$("#whitelisted-controls-while-handcuffed-list").empty();
 | 
						|
 | 
						|
	fullConfig.whitelistedControlsWhileHandcuffed.forEach(control => {
 | 
						|
		addWhitelistedControlWhileHandcuffed(control);
 | 
						|
	})
 | 
						|
 | 
						|
	// Toggle drag
 | 
						|
	$("#toggle-drag-enabled").prop("checked", fullConfig.toggleDrag.enabled);
 | 
						|
	$("#toggle-drag-default-key").val(fullConfig.toggleDrag.key);
 | 
						|
 | 
						|
	// Search
 | 
						|
	$("#settings_searchRequiresHandcuffState").prop("checked", fullConfig.searchRequiresHandcuffState);
 | 
						|
 | 
						|
	// Lockpick
 | 
						|
	toggleSettingsLockpickRequireItem(fullConfig.lockpickCarRequireItem);
 | 
						|
	$("#settings_lockpickItemName").val(fullConfig.lockpickItemName)
 | 
						|
	$("#settings_lockpickRemoveOnUse").prop("checked", fullConfig.lockpickRemoveOnUse)
 | 
						|
 | 
						|
	// Vehicle repair
 | 
						|
	toggleSettingsVehicleRepairRequireItem(fullConfig.repairVehicleRequireItem);
 | 
						|
	$("#settings_repairVehicleItemName").val(fullConfig.repairVehicleItemName);
 | 
						|
	$("#settings_repairVehicleRemoveOnUse").prop("checked", fullConfig.repairVehicleRemoveOnUse);
 | 
						|
 | 
						|
	// Vehicle cleaning
 | 
						|
	toggleSettingsVehicleCleaningRequireItem(fullConfig.cleanVehicleRequireItem);
 | 
						|
	$("#settings_cleanVehicleItemName").val(fullConfig.cleanVehicleItemName);
 | 
						|
	$("#settings_cleanVehicleRemoveOnUse").prop("checked", fullConfig.cleanVehicleRemoveOnUse);
 | 
						|
 | 
						|
	// Healing
 | 
						|
	toggleSettingsHealingRequireItem(fullConfig.healRequireItem);
 | 
						|
	$("#settings_healItemName").val(fullConfig.healItemName);
 | 
						|
	$("#settings_healRemoveOnUse").prop("checked", fullConfig.healRemoveOnUse);
 | 
						|
 | 
						|
	// Reviving
 | 
						|
	toggleSettingsRevivingRequireItem(fullConfig.reviveRequireItem);
 | 
						|
	$("#settings_reviveItemName").val(fullConfig.reviveItemName);
 | 
						|
	$("#settings_reviveRemoveOnUse").prop("checked", fullConfig.reviveRemoveOnuse);
 | 
						|
 | 
						|
	// Robbable accounts
 | 
						|
	$("#settings_robbableAccountsList").empty();
 | 
						|
	if(fullConfig.robbableAccounts) {
 | 
						|
		fullConfig.robbableAccounts.forEach(account => {
 | 
						|
			addNewRobbableAccountInList(account)
 | 
						|
		});
 | 
						|
	}
 | 
						|
 | 
						|
	$("#settings_canAlwaysCarryItem").prop("checked", fullConfig.canAlwaysCarryItem);
 | 
						|
 | 
						|
	// Depositable accounts
 | 
						|
	$("#settings_depositableAccountsList").empty();
 | 
						|
	if(fullConfig.depositableInSafeAccounts) {
 | 
						|
		fullConfig.depositableInSafeAccounts.forEach(account => {
 | 
						|
			addNewDepositableAccountInList(account)
 | 
						|
		});
 | 
						|
	}
 | 
						|
 | 
						|
	$("#settings_enablePropertyOutfits").prop("checked", fullConfig.enablePropertyOutfits);
 | 
						|
 | 
						|
	$("#settings_parkAllOwnedVehiclesOnRestart").prop("checked", fullConfig.parkAllOwnedVehiclesOnRestart);
 | 
						|
 | 
						|
	for(const[markerType, webhook] of Object.entries(fullConfig.specificWebhooks)) {
 | 
						|
		$("#settings_specific_webhooks").find(`[data-marker-type="${markerType}"]`).val(webhook);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$("#settingsRestoreDefaultBtn").click(async function() {
 | 
						|
	if(! await confirmDeletion(getLocalizedText("menu:are_you_sure_to_restore_settings"))) return;
 | 
						|
 | 
						|
	const defaultConfiguration = await $.post(`https://${resName}/getDefaultConfiguration`);
 | 
						|
	loadSettings(defaultConfiguration);
 | 
						|
})
 | 
						|
 | 
						|
function addNewRobbableAccountInList(accountName) {
 | 
						|
	var listDiv = $("#settings_robbableAccountsList");
 | 
						|
 | 
						|
	if(listDiv) {
 | 
						|
		listDiv.append(`
 | 
						|
			<li class="list-group-item d-flex justify-content-between align-items-center robbable-account" data-account="${accountName}">
 | 
						|
				${accountName}
 | 
						|
				<button class="btn remove-account"><i class="bi bi-x"></i></button>
 | 
						|
			</li>
 | 
						|
		`);
 | 
						|
 | 
						|
		listDiv.find(".remove-account").click(function() {
 | 
						|
			$(this).parent().remove();
 | 
						|
		});
 | 
						|
	}
 | 
						|
}
 | 
						|
$("#settings_addNewRobbableAccountBtn").click(function() {
 | 
						|
	let robbableAccount = $("#settings_newRobbableAccount").val();
 | 
						|
	
 | 
						|
	if(robbableAccount) {
 | 
						|
		$("#settings_newRobbableAccount").val("");
 | 
						|
		addNewRobbableAccountInList(robbableAccount);
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function getRobbableAccounts() {
 | 
						|
	var listDiv = $("#settings_robbableAccountsList");
 | 
						|
 | 
						|
	let robbableAccounts = [];
 | 
						|
 | 
						|
	listDiv.find(".robbable-account").each(function(index, element) {
 | 
						|
		robbableAccounts.push( $(element).data("account") );
 | 
						|
	})
 | 
						|
 | 
						|
	return robbableAccounts;
 | 
						|
}
 | 
						|
 | 
						|
function addNewDepositableAccountInList(accountName) {
 | 
						|
	var listDiv = $("#settings_depositableAccountsList");
 | 
						|
 | 
						|
	if(listDiv) {
 | 
						|
		listDiv.append(`
 | 
						|
			<li class="list-group-item d-flex justify-content-between align-items-center depositable-account" data-account="${accountName}">
 | 
						|
				${accountName}
 | 
						|
				<button class="btn remove-account"><i class="bi bi-x"></i></button>
 | 
						|
			</li>
 | 
						|
		`);
 | 
						|
 | 
						|
		listDiv.find(".remove-account").click(function() {
 | 
						|
			$(this).parent().remove();
 | 
						|
		});
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_addNewDepositableAccountBtn").click(function() {
 | 
						|
	let depositableAccount = $("#settings_newDepositableAccount").val();
 | 
						|
	
 | 
						|
	if(depositableAccount) {
 | 
						|
		$("#settings_newDepositableAccount").val("");
 | 
						|
		addNewDepositableAccountInList(depositableAccount);
 | 
						|
	}
 | 
						|
})
 | 
						|
 | 
						|
function getDepositableInSafeAccounts() {
 | 
						|
	var listDiv = $("#settings_depositableAccountsList");
 | 
						|
 | 
						|
	let depositableAccounts = [];
 | 
						|
 | 
						|
	listDiv.find(".depositable-account").each(function(index, element) {
 | 
						|
		depositableAccounts.push( $(element).data("account") );
 | 
						|
	})
 | 
						|
 | 
						|
	return depositableAccounts;
 | 
						|
}
 | 
						|
 | 
						|
// Handcuffs settings
 | 
						|
function toggleSettingsHandcuffsRequireItem(enable) {
 | 
						|
	$("#settings_handcuffRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_handcuffsItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_handcuffsItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_handcuffsRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_handcuffRequireItem").change(function() {
 | 
						|
	toggleSettingsHandcuffsRequireItem( $("#settings_handcuffRequireItem").prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
function getAllWhitelistedControlsWhileHandcuffed() {
 | 
						|
	let controls = [];
 | 
						|
 | 
						|
	$("#whitelisted-controls-while-handcuffed-list").find(".control-btn").each(function(index, element) {
 | 
						|
		const controlNumber = parseInt( $(element).val() );
 | 
						|
		if(controlNumber) controls.push(controlNumber);
 | 
						|
	})
 | 
						|
 | 
						|
	return controls;
 | 
						|
}
 | 
						|
 | 
						|
function addWhitelistedControlWhileHandcuffed(control="") {
 | 
						|
	const div = $(`
 | 
						|
	<div class="d-flex align-items-center justify-content-center mt-2">
 | 
						|
		<button type="button" class="btn-close me-3" ></button>	
 | 
						|
 | 
						|
		<div class="form-floating col-2">
 | 
						|
			<input type="number" class="form-control clickable control-btn" required placeholder="" value="${control}">
 | 
						|
			<label>${ getLocalizedText("menu:control_number") }</label>
 | 
						|
		</div>
 | 
						|
	</div>
 | 
						|
	`);
 | 
						|
 | 
						|
	div.find(".btn-close").click(function() {
 | 
						|
		div.remove();
 | 
						|
	});
 | 
						|
 | 
						|
	div.find(".control-btn").click(async function() {
 | 
						|
		const controlNumber = await controlsDialog();
 | 
						|
		if(!controlNumber) return;
 | 
						|
 | 
						|
		$(this).val(controlNumber);
 | 
						|
	});
 | 
						|
 | 
						|
	$("#whitelisted-controls-while-handcuffed-list").append(div);
 | 
						|
}
 | 
						|
 | 
						|
$("#add-whitelisted-control-while-handcuffed-btn").click(function() {
 | 
						|
	addWhitelistedControlWhileHandcuffed();
 | 
						|
})
 | 
						|
 | 
						|
// Lockpick settings
 | 
						|
function toggleSettingsLockpickRequireItem(enable) {
 | 
						|
	$("#settings_lockpickCarRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_lockpickItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_lockpickItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_lockpickRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_lockpickCarRequireItem").change(function() {
 | 
						|
	toggleSettingsLockpickRequireItem( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
// Vehicle repair settings
 | 
						|
function toggleSettingsVehicleRepairRequireItem(enable) {
 | 
						|
	$("#settings_repairVehicleRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_repairVehicleItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_repairVehicleItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_repairVehicleRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_repairVehicleRequireItem").change(function() {
 | 
						|
	toggleSettingsVehicleRepairRequireItem( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
// Vehicle cleaning settings
 | 
						|
function toggleSettingsVehicleCleaningRequireItem(enable) {
 | 
						|
	$("#settings_cleanVehicleRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_cleanVehicleItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_cleanVehicleItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_cleanVehicleRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_cleanVehicleRequireItem").change(function() {
 | 
						|
	toggleSettingsVehicleCleaningRequireItem( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
// Healing settings
 | 
						|
function toggleSettingsHealingRequireItem(enable) {
 | 
						|
	$("#settings_healRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_healItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_healItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_healRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_healRequireItem").change(function() {
 | 
						|
	toggleSettingsHealingRequireItem( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
// Reviving settings
 | 
						|
function toggleSettingsRevivingRequireItem(enable) {
 | 
						|
	$("#settings_reviveRequireItem").prop("checked", enable);
 | 
						|
 | 
						|
	$("#settings_reviveItemName").prop("disabled", !enable);
 | 
						|
	$("#settings_reviveItemName").prop("required", enable);
 | 
						|
 | 
						|
	$("#settings_reviveRemoveOnUse").prop("disabled", !enable);
 | 
						|
}
 | 
						|
 | 
						|
$("#settings_reviveRequireItem").change(function() {
 | 
						|
	toggleSettingsRevivingRequireItem( $(this).prop("checked") );
 | 
						|
})
 | 
						|
 | 
						|
// Closes menu when clicking ESC
 | 
						|
$(document).on('keyup', function(e) {
 | 
						|
	if (e.key == "Escape") {
 | 
						|
		if( $("#job-creator").is(":visible") ) {
 | 
						|
			exit();
 | 
						|
		} else if( $("#edit-job").is(":visible") ) {
 | 
						|
			exitFromEditJob();
 | 
						|
		}
 | 
						|
	}
 | 
						|
}); |