mirror of
https://github.com/ellite/Wallos.git
synced 2026-01-06 13:58:11 -05:00
1158 lines
38 KiB
JavaScript
1158 lines
38 KiB
JavaScript
const deleteSvgContent = `
|
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 48 48" id="Recycle-Bin-2--Streamline-Plump.svg" height="48" width="48">
|
|
<g id="recycle-bin-2--remove-delete-empty-bin-trash-garbage">
|
|
<path id="Union" class="accent-color" d="M43.318 15.934a1.5 1.5 0 0 0 -1.618 -1.591c-3.016 0.246 -8.46 0.52 -17.721 0.52 -9.215 0 -14.65 -0.271 -17.675 -0.516a1.5 1.5 0 0 0 -1.618 1.59c0.888 13.84 1.74 21.07 2.253 24.547 0.332 2.252 1.85 4.217 4.226 4.788 2.445 0.588 6.55 1.227 12.837 1.227 6.286 0 10.392 -0.64 12.837 -1.227 2.375 -0.57 3.894 -2.536 4.226 -4.788 0.513 -3.477 1.365 -10.708 2.253 -24.55Z" stroke-width="1"/>
|
|
<path id="Union_2" class="main-color" d="M23.37 1a8 8 0 0 0 -7.034 4.188c-3.411 0.072 -6 0.182 -7.814 0.282 -2.312 0.127 -4.692 1.242 -5.7 3.605 -0.244 0.57 -0.475 1.212 -0.663 1.919 -0.68 2.548 1.302 4.622 3.657 4.822 3.057 0.258 8.614 0.548 18.161 0.548 9.549 0 15.106 -0.29 18.162 -0.549 2.374 -0.2 4.291 -2.261 3.751 -4.785a16.68 16.68 0 0 0 -0.294 -1.167c-0.824 -2.831 -3.517 -4.277 -6.188 -4.411a260.66 260.66 0 0 0 -7.744 -0.264A8 8 0 0 0 24.631 1H23.37Z" stroke-width="1"/>
|
|
<path id="Vector_831_Stroke" class="main-color" fill-rule="evenodd" d="M17.8 23.01a2 2 0 0 1 2.19 1.791l1 10a2 2 0 0 1 -3.98 0.398l-1 -10a2 2 0 0 1 1.79 -2.189Z" clip-rule="evenodd" stroke-width="1"/>
|
|
<path id="Vector_832_Stroke" class="main-color" fill-rule="evenodd" d="M30.2 23.01a2 2 0 0 0 -2.19 1.791l-1 10a2 2 0 0 0 3.98 0.398l1 -10a2 2 0 0 0 -1.79 -2.189Z" clip-rule="evenodd" stroke-width="1"/>
|
|
</g>
|
|
</svg>
|
|
`;
|
|
|
|
const editSvgContent = `<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 48 48" id="File-Check-Alternate--Streamline-Plump.svg" height="48" width="48">
|
|
<g id="file-check-alternate--file-common-check">
|
|
<path id="Subtract" class="accent-color" d="M13.582 2.137C16.326 1.823 20.685 1.5 27 1.5a165 165 0 0 1 5.13 0.077 1.5 1.5 0 0 1 0.4 0.068c1.098 0.343 4.029 1.564 8.123 5.578 3.862 3.787 5.195 6.563 5.63 7.781a1.5 1.5 0 0 1 0.087 0.45c0.08 2.153 0.13 4.655 0.13 7.546 0 7.57 -0.343 12.478 -0.669 15.432 -0.32 2.9 -2.518 5.1 -5.413 5.431 -2.744 0.314 -7.103 0.637 -13.418 0.637 -1.044 0 -2.035 -0.009 -2.974 -0.025A14.458 14.458 0 0 0 28.5 34c0 -8.008 -6.492 -14.5 -14.5 -14.5a14.44 14.44 0 0 0 -6.492 1.531c0.053 -6.464 0.364 -10.773 0.66 -13.463 0.32 -2.9 2.519 -5.1 5.414 -5.431Z" stroke-width="1"></path>
|
|
<path id="Intersect" class="main-color" d="M46.348 15.25c-2.42 -0.001 -6.57 -0.04 -8.948 -0.268 -2.598 -0.249 -4.641 -2.321 -4.896 -4.975 -0.214 -2.233 -0.253 -5.99 -0.254 -8.421 0.095 0.01 0.188 0.03 0.28 0.059 1.098 0.343 4.029 1.564 8.123 5.578 3.862 3.787 5.195 6.563 5.63 7.781 0.029 0.08 0.05 0.163 0.065 0.246Z" stroke-width="1"></path>
|
|
<path id="Subtract_2" class="main-color" fill-rule="evenodd" d="M14 46c6.627 0 12 -5.373 12 -12s-5.373 -12 -12 -12S2 27.373 2 34s5.373 12 12 12Z" clip-rule="evenodd" stroke-width="1"></path>
|
|
<path id="Subtract_3" class="accent-color" fill-rule="evenodd" d="M20.611 31.185a2 2 0 1 0 -3.222 -2.37l-4.413 6.002L10.5 32.01a2 2 0 0 0 -3 2.647l4.118 4.666a2 2 0 0 0 3.111 -0.138l5.882 -8Z" clip-rule="evenodd" stroke-width="1"></path>
|
|
</g>
|
|
</svg>`;
|
|
|
|
function saveBudget() {
|
|
const button = document.getElementById("saveBudget");
|
|
button.disabled = true;
|
|
|
|
const budget = document.getElementById("budget").value;
|
|
|
|
fetch('endpoints/user/budget.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: JSON.stringify({budget: budget}),
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate('unknown_error'));
|
|
})
|
|
.finally(() => {
|
|
button.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function addMemberButton(memberId) {
|
|
const addButton = document.getElementById("addMember");
|
|
addButton.disabled = true;
|
|
|
|
fetch("endpoints/household/household.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({action: "add"}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
showErrorMessage(translate("failed_add_member"));
|
|
throw new Error(translate("network_response_error"));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
const newMemberId = responseData.householdId;
|
|
const container = document.getElementById("householdMembers");
|
|
|
|
const div = document.createElement("div");
|
|
div.className = "form-group-inline";
|
|
div.dataset.memberid = newMemberId;
|
|
|
|
const input = document.createElement("input");
|
|
input.type = "text";
|
|
input.placeholder = translate("member");
|
|
input.name = "member";
|
|
input.value = translate("member");
|
|
|
|
const emailInput = document.createElement("input");
|
|
emailInput.type = "text";
|
|
emailInput.placeholder = translate("email");
|
|
emailInput.name = "email";
|
|
emailInput.value = "";
|
|
|
|
const editLink = document.createElement("button");
|
|
editLink.className = "image-button medium";
|
|
editLink.name = "save";
|
|
editLink.onclick = () => editMember(newMemberId);
|
|
editLink.innerHTML = editSvgContent;
|
|
editLink.title = translate("save_member");
|
|
|
|
const deleteLink = document.createElement("button");
|
|
deleteLink.className = "image-button medium";
|
|
deleteLink.name = "delete";
|
|
deleteLink.onclick = () => removeMember(newMemberId);
|
|
deleteLink.innerHTML = deleteSvgContent;
|
|
deleteLink.title = translate("delete_member");
|
|
|
|
div.appendChild(input);
|
|
div.appendChild(emailInput);
|
|
div.appendChild(editLink);
|
|
div.appendChild(deleteLink);
|
|
|
|
container.appendChild(div);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate("failed_add_member"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("failed_add_member"));
|
|
})
|
|
.finally(() => {
|
|
addButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
function removeMember(memberId) {
|
|
fetch("endpoints/household/household.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: "delete",
|
|
memberId: memberId,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate("network_response_error"));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
const divToRemove = document.querySelector(`[data-memberid="${memberId}"]`);
|
|
if (divToRemove) divToRemove.remove();
|
|
showSuccessMessage(responseData.message);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate("failed_remove_member"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("failed_remove_member"));
|
|
});
|
|
}
|
|
|
|
|
|
function editMember(memberId) {
|
|
const saveButton = document.querySelector(`div[data-memberid="${memberId}"] button[name="save"]`);
|
|
const memberNameElement = document.querySelector(`div[data-memberid="${memberId}"] input[name="member"]`);
|
|
const memberEmailElement = document.querySelector(`div[data-memberid="${memberId}"] input[name="email"]`);
|
|
|
|
if (!memberNameElement) return;
|
|
|
|
saveButton.classList.add("disabled");
|
|
saveButton.disabled = true;
|
|
|
|
const memberName = memberNameElement.value;
|
|
const memberEmail = memberEmailElement ? memberEmailElement.value : "";
|
|
|
|
fetch("endpoints/household/household.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: "edit",
|
|
memberId: memberId,
|
|
name: memberName,
|
|
email: memberEmail,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
showErrorMessage(translate("failed_save_member"));
|
|
throw new Error(translate("network_response_error"));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
showSuccessMessage(responseData.message);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate("failed_save_member"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("failed_save_member"));
|
|
})
|
|
.finally(() => {
|
|
saveButton.classList.remove("disabled");
|
|
saveButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function addCategoryButton(categoryId) {
|
|
const addButton = document.getElementById("addCategory");
|
|
addButton.disabled = true;
|
|
|
|
fetch('endpoints/categories/category.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({action: 'add'}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
showErrorMessage(translate('failed_add_category'));
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
const newCategoryId = responseData.categoryId;
|
|
const container = document.getElementById("categories");
|
|
|
|
const row = document.createElement("div");
|
|
row.className = "form-group-inline";
|
|
row.dataset.categoryid = newCategoryId;
|
|
|
|
const dragIcon = document.createElement("div");
|
|
dragIcon.className = "drag-icon";
|
|
|
|
const input = document.createElement("input");
|
|
input.type = "text";
|
|
input.placeholder = translate('category');
|
|
input.name = "category";
|
|
input.value = translate('category');
|
|
|
|
const editLink = document.createElement("button");
|
|
editLink.className = "image-button medium";
|
|
editLink.name = "save";
|
|
editLink.onclick = function () {
|
|
editCategory(newCategoryId);
|
|
};
|
|
editLink.innerHTML = editSvgContent;
|
|
editLink.title = translate('save_member');
|
|
|
|
const deleteLink = document.createElement("button");
|
|
deleteLink.className = "image-button medium";
|
|
deleteLink.name = "delete";
|
|
deleteLink.onclick = function () {
|
|
removeCategory(newCategoryId);
|
|
};
|
|
deleteLink.innerHTML = deleteSvgContent;
|
|
deleteLink.title = translate('delete_member');
|
|
|
|
row.appendChild(dragIcon);
|
|
row.appendChild(input);
|
|
row.appendChild(editLink);
|
|
row.appendChild(deleteLink);
|
|
container.appendChild(row);
|
|
} else {
|
|
showErrorMessage(responseData.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate('failed_add_category'));
|
|
})
|
|
.finally(() => {
|
|
addButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function removeCategory(categoryId) {
|
|
fetch('endpoints/categories/category.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: 'delete',
|
|
categoryId: categoryId,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
const divToRemove = document.querySelector(`[data-categoryid="${categoryId}"]`);
|
|
if (divToRemove) divToRemove.remove();
|
|
showSuccessMessage(responseData.message);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate('failed_remove_category'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate('failed_remove_category'));
|
|
});
|
|
}
|
|
|
|
|
|
function editCategory(categoryId) {
|
|
const saveButton = document.querySelector(`div[data-categoryid="${categoryId}"] button[name="save"]`);
|
|
const inputElement = document.querySelector(`div[data-categoryid="${categoryId}"] input[name="category"]`);
|
|
|
|
if (!inputElement) return;
|
|
|
|
saveButton.classList.add("disabled");
|
|
saveButton.disabled = true;
|
|
|
|
const categoryName = inputElement.value;
|
|
|
|
fetch('endpoints/categories/category.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: 'edit',
|
|
categoryId: categoryId,
|
|
name: categoryName,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
saveButton.classList.remove("disabled");
|
|
saveButton.disabled = false;
|
|
|
|
if (!response.ok) {
|
|
showErrorMessage(translate('failed_save_category'));
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
showSuccessMessage(responseData.message);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate('failed_save_category'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate('failed_save_category'));
|
|
saveButton.classList.remove("disabled");
|
|
saveButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function addCurrencyButton(currencyId) {
|
|
const addButton = document.getElementById("addCurrency");
|
|
addButton.disabled = true;
|
|
|
|
fetch('endpoints/currency/currency.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({action: 'add'}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(responseData => {
|
|
if (responseData.success) {
|
|
const newCurrencyId = responseData.currencyId;
|
|
const container = document.getElementById("currencies");
|
|
|
|
const div = document.createElement("div");
|
|
div.className = "form-group-inline";
|
|
div.dataset.currencyid = newCurrencyId;
|
|
|
|
const inputSymbol = document.createElement("input");
|
|
inputSymbol.type = "text";
|
|
inputSymbol.placeholder = "$";
|
|
inputSymbol.name = "symbol";
|
|
inputSymbol.value = "$";
|
|
inputSymbol.classList.add("short");
|
|
|
|
const inputName = document.createElement("input");
|
|
inputName.type = "text";
|
|
inputName.placeholder = translate('currency');
|
|
inputName.name = "currency";
|
|
inputName.value = translate('currency');
|
|
|
|
const inputCode = document.createElement("input");
|
|
inputCode.type = "text";
|
|
inputCode.placeholder = translate('currency_code');
|
|
inputCode.name = "code";
|
|
inputCode.value = "CODE";
|
|
|
|
const editLink = document.createElement("button");
|
|
editLink.className = "image-button medium";
|
|
editLink.name = "save";
|
|
editLink.onclick = function () {
|
|
editCurrency(newCurrencyId);
|
|
};
|
|
editLink.innerHTML = editSvgContent;
|
|
editLink.title = translate('save_member');
|
|
|
|
const deleteLink = document.createElement("button");
|
|
deleteLink.className = "image-button medium";
|
|
deleteLink.name = "delete";
|
|
deleteLink.onclick = function () {
|
|
removeCurrency(newCurrencyId);
|
|
};
|
|
deleteLink.innerHTML = deleteSvgContent;
|
|
deleteLink.title = translate('delete_member');
|
|
|
|
div.appendChild(inputSymbol);
|
|
div.appendChild(inputName);
|
|
div.appendChild(inputCode);
|
|
div.appendChild(editLink);
|
|
div.appendChild(deleteLink);
|
|
|
|
container.appendChild(div);
|
|
} else {
|
|
showErrorMessage(responseData.message || translate('failed_add_currency'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate('failed_add_currency'));
|
|
})
|
|
.finally(() => {
|
|
addButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
function removeCurrency(currencyId) {
|
|
fetch('endpoints/currency/currency.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: 'delete',
|
|
currencyId: currencyId,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
const divToRemove = document.querySelector(`[data-currencyid="${currencyId}"]`);
|
|
if (divToRemove) divToRemove.remove();
|
|
} else {
|
|
showErrorMessage(data.message || translate('failed_remove_currency'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(error.message || translate('failed_remove_currency'));
|
|
});
|
|
}
|
|
|
|
function editCurrency(currencyId) {
|
|
const saveButton = document.querySelector(`div[data-currencyid="${currencyId}"] button[name="save"]`);
|
|
const inputSymbolElement = document.querySelector(`div[data-currencyid="${currencyId}"] input[name="symbol"]`);
|
|
const inputNameElement = document.querySelector(`div[data-currencyid="${currencyId}"] input[name="currency"]`);
|
|
const inputCodeElement = document.querySelector(`div[data-currencyid="${currencyId}"] input[name="code"]`);
|
|
|
|
if (!inputNameElement) return;
|
|
|
|
saveButton.classList.add("disabled");
|
|
saveButton.disabled = true;
|
|
|
|
const currencyName = inputNameElement.value;
|
|
const currencySymbol = inputSymbolElement.value;
|
|
const currencyCode = inputCodeElement.value;
|
|
|
|
fetch('endpoints/currency/currency.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
action: 'edit',
|
|
currencyId: currencyId,
|
|
name: currencyName,
|
|
symbol: currencySymbol,
|
|
code: currencyCode,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate('network_response_error'));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(data => {
|
|
saveButton.classList.remove("disabled");
|
|
saveButton.disabled = false;
|
|
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message || translate('failed_save_currency'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(error.message || translate('failed_save_currency'));
|
|
saveButton.classList.remove("disabled");
|
|
saveButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
function togglePayment(paymentId) {
|
|
const element = document.querySelector(`div[data-paymentid="${paymentId}"]`);
|
|
|
|
if (element.dataset.inUse === "yes") {
|
|
return showErrorMessage(translate("cant_disable_payment_in_use"));
|
|
}
|
|
|
|
const newEnabledState = element.dataset.enabled === "1" ? "0" : "1";
|
|
const paymentMethodName = element.querySelector(".payment-name").innerText;
|
|
|
|
fetch("endpoints/payments/toggle.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
"X-CSRF-Token": window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
paymentId: paymentId,
|
|
enabled: newEnabledState,
|
|
}),
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate("network_response_error"));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(data => {
|
|
if (data.success) {
|
|
element.dataset.enabled = newEnabledState;
|
|
showSuccessMessage(`${paymentMethodName} ${data.message}`);
|
|
} else {
|
|
showErrorMessage(data.message || translate("failed_save_payment_method"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(error.message || translate("failed_save_payment_method"));
|
|
});
|
|
}
|
|
|
|
document.body.addEventListener('click', function (e) {
|
|
let targetElement = e.target;
|
|
do {
|
|
if (targetElement.classList && targetElement.classList.contains('payments-payment')) {
|
|
let targetChild = e.target;
|
|
do {
|
|
if (targetChild.classList && (targetChild.classList.contains('payment-name') || targetChild.classList.contains('drag-icon'))) {
|
|
return;
|
|
}
|
|
targetChild = targetChild.parentNode;
|
|
} while (targetChild && targetChild !== targetElement);
|
|
|
|
const paymentId = targetElement.dataset.paymentid;
|
|
togglePayment(paymentId);
|
|
return;
|
|
}
|
|
targetElement = targetElement.parentNode;
|
|
} while (targetElement);
|
|
});
|
|
|
|
document.body.addEventListener('blur', function (e) {
|
|
let targetElement = e.target;
|
|
if (targetElement.classList && targetElement.classList.contains('payment-name')) {
|
|
const paymentId = targetElement.closest('.payments-payment').dataset.paymentid;
|
|
const newName = targetElement.textContent;
|
|
renamePayment(paymentId, newName);
|
|
}
|
|
}, true);
|
|
|
|
function renamePayment(paymentId, newName) {
|
|
const name = newName.trim();
|
|
if (!name) return;
|
|
|
|
const formData = new FormData();
|
|
formData.append("paymentId", paymentId);
|
|
formData.append("name", name);
|
|
|
|
fetch("endpoints/payments/rename.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"X-CSRF-Token": window.csrfToken,
|
|
},
|
|
body: formData,
|
|
})
|
|
.then(response => {
|
|
if (!response.ok) {
|
|
throw new Error(translate("network_response_error"));
|
|
}
|
|
return response.json();
|
|
})
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(`${newName} ${data.message}`);
|
|
} else {
|
|
showErrorMessage(data.message || translate("failed_save_payment_method"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("unknown_error"));
|
|
});
|
|
}
|
|
|
|
|
|
document.body.addEventListener('keypress', function (e) {
|
|
let targetElement = e.target;
|
|
if (targetElement.classList && targetElement.classList.contains('payment-name')) {
|
|
if (e.key === 'Enter') {
|
|
e.preventDefault();
|
|
targetElement.blur();
|
|
}
|
|
}
|
|
});
|
|
|
|
function handleFileSelect(event) {
|
|
const fileInput = event.target;
|
|
const iconPreview = document.querySelector('.icon-preview');
|
|
const iconImg = iconPreview.querySelector('img');
|
|
const iconUrl = document.querySelector("#icon-url");
|
|
iconUrl.value = "";
|
|
|
|
if (fileInput.files && fileInput.files[0]) {
|
|
const reader = new FileReader();
|
|
|
|
reader.onload = function (e) {
|
|
iconImg.src = e.target.result;
|
|
iconImg.style.display = 'block';
|
|
};
|
|
|
|
reader.readAsDataURL(fileInput.files[0]);
|
|
}
|
|
}
|
|
|
|
function setSearchButtonStatus() {
|
|
|
|
const nameInput = document.querySelector("#paymentname");
|
|
const hasSearchTerm = nameInput.value.trim().length > 0;
|
|
const iconSearchButton = document.querySelector("#icon-search-button");
|
|
if (hasSearchTerm) {
|
|
iconSearchButton.classList.remove("disabled");
|
|
} else {
|
|
iconSearchButton.classList.add("disabled");
|
|
}
|
|
|
|
}
|
|
|
|
function searchPaymentIcon() {
|
|
const nameInput = document.querySelector("#paymentname");
|
|
const searchTerm = nameInput.value.trim();
|
|
if (searchTerm !== "") {
|
|
const iconSearchPopup = document.querySelector("#icon-search-results");
|
|
iconSearchPopup.classList.add("is-open");
|
|
const imageSearchUrl = `endpoints/payments/search.php?search=${searchTerm}`;
|
|
fetch(imageSearchUrl)
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.imageUrls) {
|
|
displayImageResults(data.imageUrls);
|
|
} else if (data.error) {
|
|
console.error(data.error);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(translate('error_fetching_image_results'), error);
|
|
});
|
|
} else {
|
|
nameInput.focus();
|
|
}
|
|
}
|
|
|
|
function displayImageResults(imageSources) {
|
|
const iconResults = document.querySelector("#icon-search-images");
|
|
iconResults.innerHTML = "";
|
|
|
|
imageSources.forEach(src => {
|
|
const img = document.createElement("img");
|
|
img.src = src;
|
|
img.onclick = function () {
|
|
selectWebIcon(src);
|
|
};
|
|
img.onerror = function () {
|
|
this.parentNode.removeChild(this);
|
|
};
|
|
iconResults.appendChild(img);
|
|
});
|
|
}
|
|
|
|
function selectWebIcon(url) {
|
|
closeIconSearch();
|
|
const iconPreview = document.querySelector("#form-icon");
|
|
const iconUrl = document.querySelector("#icon-url");
|
|
iconPreview.src = url;
|
|
iconPreview.style.display = 'block';
|
|
iconUrl.value = url;
|
|
}
|
|
|
|
function closeIconSearch() {
|
|
const iconSearchPopup = document.querySelector("#icon-search-results");
|
|
iconSearchPopup.classList.remove("is-open");
|
|
const iconResults = document.querySelector("#icon-search-images");
|
|
iconResults.innerHTML = "";
|
|
}
|
|
|
|
function resetFormIcon() {
|
|
const iconPreview = document.querySelector("#form-icon");
|
|
iconPreview.src = "";
|
|
iconPreview.style.display = 'none';
|
|
}
|
|
|
|
function reloadPaymentMethods() {
|
|
const paymentsContainer = document.querySelector("#payments-list");
|
|
const paymentMethodsEndpoint = "endpoints/payments/get.php";
|
|
|
|
fetch(paymentMethodsEndpoint)
|
|
.then(response => response.text())
|
|
.then(data => {
|
|
paymentsContainer.innerHTML = data;
|
|
});
|
|
}
|
|
|
|
function addPaymentMethod() {
|
|
closeIconSearch();
|
|
const addPaymentMethodEndpoint = "endpoints/payments/add.php";
|
|
const paymentMethodForm = document.querySelector("#payments-form");
|
|
const submitButton = document.querySelector("#add-payment-button");
|
|
|
|
submitButton.disabled = true;
|
|
const formData = new FormData(paymentMethodForm);
|
|
formData.append("action", "add");
|
|
|
|
fetch(addPaymentMethodEndpoint, {
|
|
method: "POST",
|
|
headers: {
|
|
"X-CSRF-Token": window.csrfToken,
|
|
},
|
|
body: formData,
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
paymentMethodForm.reset();
|
|
resetFormIcon();
|
|
reloadPaymentMethods();
|
|
} else {
|
|
showErrorMessage(data.message || translate("failed_add_payment_method"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("unknown_error"));
|
|
})
|
|
.finally(() => {
|
|
submitButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function deletePaymentMethod(paymentId) {
|
|
fetch(`endpoints/payments/delete.php?id=${paymentId}`, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
"X-CSRF-Token": window.csrfToken,
|
|
},
|
|
body: JSON.stringify({ id: paymentId }),
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
var paymentToRemove = document.querySelector('.payments-payment[data-paymentid="' + paymentId + '"]');
|
|
if (paymentToRemove) {
|
|
paymentToRemove.remove();
|
|
}
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch((error) => {
|
|
console.error('Error:', error);
|
|
});
|
|
}
|
|
|
|
function savePaymentMethodsSorting() {
|
|
const paymentMethods = document.getElementById("payments-list");
|
|
const paymentMethodIds = Array.from(paymentMethods.children).map(
|
|
paymentMethod => paymentMethod.dataset.paymentid
|
|
);
|
|
|
|
const formData = new FormData();
|
|
paymentMethodIds.forEach(id => formData.append("paymentMethodIds[]", id));
|
|
formData.append("action", "sort");
|
|
|
|
fetch("endpoints/payments/sort.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"X-CSRF-Token": window.csrfToken,
|
|
},
|
|
body: formData,
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message || translate("failed_sort_payment_methods"));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("unknown_error"));
|
|
});
|
|
}
|
|
|
|
|
|
var el = document.getElementById('payments-list');
|
|
var sortable = Sortable.create(el, {
|
|
handle: '.drag-icon',
|
|
ghostClass: 'sortable-ghost',
|
|
delay: 500,
|
|
delayOnTouchOnly: true,
|
|
touchStartThreshold: 5,
|
|
onEnd: function (evt) {
|
|
savePaymentMethodsSorting();
|
|
},
|
|
});
|
|
|
|
|
|
document.addEventListener('DOMContentLoaded', function () {
|
|
|
|
var removePaymentButtons = document.querySelectorAll(".delete-payment-method");
|
|
removePaymentButtons.forEach(function (button) {
|
|
button.addEventListener('click', function (event) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
let paymentId = event.target.getAttribute('data-paymentid');
|
|
deletePaymentMethod(paymentId);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
function addFixerKeyButton() {
|
|
const addButton = document.getElementById("addFixerKey");
|
|
addButton.disabled = true;
|
|
|
|
const apiKeyInput = document.querySelector("#fixerKey");
|
|
const apiKey = apiKeyInput.value.trim();
|
|
const provider = document.querySelector("#fixerProvider").value;
|
|
const convertCurrencyCheckbox = document.querySelector("#convertcurrency");
|
|
|
|
fetch("endpoints/currency/fixer_api_key.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({
|
|
api_key: apiKey,
|
|
provider: provider,
|
|
}),
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
addButton.disabled = false;
|
|
convertCurrencyCheckbox.disabled = false;
|
|
|
|
fetch("endpoints/currency/update_exchange.php", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: new URLSearchParams({force: "true"}),
|
|
}).catch(console.error);
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
addButton.disabled = false;
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("unknown_error"));
|
|
addButton.disabled = false;
|
|
});
|
|
}
|
|
|
|
|
|
function storeSettingsOnDB(endpoint, value) {
|
|
fetch('endpoints/settings/' + endpoint + '.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: JSON.stringify({ "value": value })
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
});
|
|
}
|
|
|
|
function setShowMonthlyPrice() {
|
|
const showMonthlyPriceCheckbox = document.querySelector("#monthlyprice");
|
|
const value = showMonthlyPriceCheckbox.checked;
|
|
|
|
storeSettingsOnDB('monthly_price', value);
|
|
}
|
|
|
|
function setConvertCurrency() {
|
|
const convertCurrencyCheckbox = document.querySelector("#convertcurrency");
|
|
const value = convertCurrencyCheckbox.checked;
|
|
|
|
storeSettingsOnDB('convert_currency', value);
|
|
}
|
|
|
|
function setRemoveBackground() {
|
|
const removeBackgroundCheckbox = document.querySelector("#removebackground");
|
|
const value = removeBackgroundCheckbox.checked;
|
|
|
|
storeSettingsOnDB('remove_background', value);
|
|
}
|
|
|
|
function setHideDisabled() {
|
|
const hideDisabledCheckbox = document.querySelector("#hidedisabled");
|
|
const value = hideDisabledCheckbox.checked;
|
|
|
|
storeSettingsOnDB('hide_disabled', value);
|
|
}
|
|
|
|
function setDisabledToBottom() {
|
|
const disabledToBottomCheckbox = document.querySelector("#disabledtobottom");
|
|
const value = disabledToBottomCheckbox.checked;
|
|
|
|
storeSettingsOnDB('disabled_to_bottom', value);
|
|
}
|
|
|
|
function setShowOriginalPrice() {
|
|
const showOriginalPriceCheckbox = document.querySelector("#showoriginalprice");
|
|
const value = showOriginalPriceCheckbox.checked;
|
|
|
|
storeSettingsOnDB('show_original_price', value);
|
|
}
|
|
|
|
function setMobileNavigation() {
|
|
const mobileNavigationCheckbox = document.querySelector("#mobilenavigation");
|
|
const value = mobileNavigationCheckbox.checked;
|
|
|
|
storeSettingsOnDB('mobile_navigation', value);
|
|
}
|
|
|
|
function setShowSubscriptionProgress() {
|
|
const showSubscriptionProgressCheckbox = document.querySelector("#showsubscriptionprogress");
|
|
const value = showSubscriptionProgressCheckbox.checked;
|
|
|
|
storeSettingsOnDB('subscription_progress', value);
|
|
}
|
|
|
|
function saveCategorySorting() {
|
|
const categories = document.getElementById("categories");
|
|
const categoryIds = Array.from(categories.children).map(c => c.dataset.categoryid);
|
|
|
|
const formData = new FormData();
|
|
categoryIds.forEach(categoryId => formData.append("categoryIds[]", categoryId));
|
|
formData.append("action", "sort");
|
|
|
|
fetch("endpoints/categories/category.php", {
|
|
method: "POST",
|
|
headers: {"X-CSRF-Token": window.csrfToken},
|
|
body: formData,
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error(error);
|
|
showErrorMessage(translate("unknown_error"));
|
|
});
|
|
}
|
|
|
|
|
|
var el = document.getElementById('categories');
|
|
var sortable = Sortable.create(el, {
|
|
handle: '.drag-icon',
|
|
ghostClass: 'sortable-ghost',
|
|
delay: 500,
|
|
delayOnTouchOnly: true,
|
|
touchStartThreshold: 5,
|
|
onEnd: function (evt) {
|
|
saveCategorySorting();
|
|
},
|
|
});
|
|
|
|
function fetch_ai_models() {
|
|
const endpoint = 'endpoints/ai/fetch_models.php';
|
|
const type = document.querySelector("#ai_type").value;
|
|
const api_key = document.querySelector("#ai_api_key").value.trim();
|
|
const ollama_host = document.querySelector("#ai_ollama_host").value.trim();
|
|
const modelSelect = document.querySelector("#ai_model");
|
|
|
|
fetch(endpoint, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: JSON.stringify({ type, api_key, ollama_host })
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
modelSelect.innerHTML = '';
|
|
data.models.forEach(model => {
|
|
const option = document.createElement('option');
|
|
option.value = model.id;
|
|
option.textContent = model.name;
|
|
modelSelect.appendChild(option);
|
|
});
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
showErrorMessage(translate('unknown_error'));
|
|
});
|
|
}
|
|
|
|
function toggleAiInputs() {
|
|
const aiTypeSelect = document.getElementById("ai_type");
|
|
const apiKeyInput = document.getElementById("ai_api_key");
|
|
const ollamaHostInput = document.getElementById("ai_ollama_host");
|
|
const type = aiTypeSelect.value;
|
|
if (type === "ollama") {
|
|
apiKeyInput.classList.add("hidden");
|
|
ollamaHostInput.classList.remove("hidden");
|
|
} else {
|
|
apiKeyInput.classList.remove("hidden");
|
|
ollamaHostInput.classList.add("hidden");
|
|
}
|
|
}
|
|
|
|
function saveAiSettingsButton() {
|
|
const aiEnabled = document.querySelector("#ai_enabled").checked;
|
|
const aiType = document.querySelector("#ai_type").value;
|
|
const aiApiKey = document.querySelector("#ai_api_key").value.trim();
|
|
const aiOllamaHost = document.querySelector("#ai_ollama_host").value.trim();
|
|
const aiModel = document.querySelector("#ai_model").value;
|
|
|
|
fetch('endpoints/ai/save_settings.php', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
},
|
|
body: JSON.stringify({ ai_enabled: aiEnabled, ai_type: aiType, api_key: aiApiKey, ollama_host: aiOllamaHost, model: aiModel })
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
const runAiActionButton = document.querySelector("#runAiRecommendations");
|
|
if (data.enabled) {
|
|
runAiActionButton.classList.remove("hidden");
|
|
} else {
|
|
runAiActionButton.classList.add("hidden");
|
|
}
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
showErrorMessage(translate('unknown_error'));
|
|
});
|
|
}
|
|
|
|
function runAiRecommendations() {
|
|
const endpoint = 'endpoints/ai/generate_recommendations.php';
|
|
const button = document.querySelector("#runAiRecommendations");
|
|
const spinner = document.querySelector("#aiSpinner");
|
|
|
|
button.classList.add("hidden");
|
|
spinner.classList.remove("hidden");
|
|
|
|
fetch(endpoint, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': window.csrfToken,
|
|
}
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showSuccessMessage(data.message);
|
|
} else {
|
|
showErrorMessage(data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
showErrorMessage(translate('unknown_error'));
|
|
})
|
|
.finally(() => {
|
|
button.classList.remove("hidden");
|
|
spinner.classList.add("hidden");
|
|
});
|
|
|
|
} |