Main/resources/[creator]/jobs_creator/html/index.js

4454 lines
140 KiB
JavaScript
Raw Normal View History

2025-06-07 08:51:21 +02:00
/*
*/
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();
}
}
});