|
|
|
|
@ -2,7 +2,7 @@
|
|
|
|
|
{% load i18n static custom_filters %}
|
|
|
|
|
{% block title %}
|
|
|
|
|
{% trans 'Add New Car' %} {% endblock %}
|
|
|
|
|
{% block content %}
|
|
|
|
|
{% block content %}
|
|
|
|
|
<style>
|
|
|
|
|
#video {
|
|
|
|
|
width: 100%;
|
|
|
|
|
@ -339,336 +339,435 @@
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<!---->
|
|
|
|
|
<script>
|
|
|
|
|
function getCookie(name) {
|
|
|
|
|
let cookieValue = null;
|
|
|
|
|
if (document.cookie && document.cookie !== "") {
|
|
|
|
|
const cookies = document.cookie.split(";");
|
|
|
|
|
for (let cookie of cookies) {
|
|
|
|
|
cookie = cookie.trim();
|
|
|
|
|
if (cookie.substring(0, name.length + 1) === name + "=") {
|
|
|
|
|
cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return cookieValue;
|
|
|
|
|
{% endblock content %}
|
|
|
|
|
{% block customJS %}
|
|
|
|
|
<script>
|
|
|
|
|
// Global variables
|
|
|
|
|
let codeReader;
|
|
|
|
|
let currentStream = null;
|
|
|
|
|
const csrfToken = getCookie("csrftoken");
|
|
|
|
|
const ajaxUrl = "{% url 'ajax_handler' request.dealer.slug %}";
|
|
|
|
|
|
|
|
|
|
// Initialize when page loads and after HTMX swaps
|
|
|
|
|
document.addEventListener('DOMContentLoaded', initPage);
|
|
|
|
|
document.addEventListener('htmx:afterSwap', initPage);
|
|
|
|
|
|
|
|
|
|
function initPage() {
|
|
|
|
|
// Get DOM elements
|
|
|
|
|
const elements = {
|
|
|
|
|
vinInput: document.getElementById("{{ form.vin.id_for_label }}"),
|
|
|
|
|
decodeVinBtn: document.getElementById("decodeVinBtn"),
|
|
|
|
|
makeSelect: document.getElementById("{{ form.id_car_make.id_for_label }}"),
|
|
|
|
|
modelSelect: document.getElementById("{{ form.id_car_model.id_for_label }}"),
|
|
|
|
|
yearSelect: document.getElementById("{{ form.year.id_for_label }}"),
|
|
|
|
|
serieSelect: document.getElementById("{{ form.id_car_serie.id_for_label }}"),
|
|
|
|
|
trimSelect: document.getElementById("{{ form.id_car_trim.id_for_label }}"),
|
|
|
|
|
equipmentSelect: document.getElementById("equipment_id"),
|
|
|
|
|
showSpecificationButton: document.getElementById("specification-btn"),
|
|
|
|
|
showEquipmentButton: document.getElementById("options-btn"),
|
|
|
|
|
specificationsContent: document.getElementById("specificationsContent"),
|
|
|
|
|
optionsContent: document.getElementById("optionsContent"),
|
|
|
|
|
generationContainer: document.getElementById("generation-div"),
|
|
|
|
|
closeButton: document.querySelector(".btn-close"),
|
|
|
|
|
scanVinBtn: document.getElementById("scan-vin-btn"),
|
|
|
|
|
videoElement: document.getElementById("video"),
|
|
|
|
|
resultDisplay: document.getElementById("result"),
|
|
|
|
|
fallbackButton: document.getElementById("ocr-fallback-btn")
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Initialize scanner if available
|
|
|
|
|
if (typeof ZXing !== 'undefined' && !codeReader) {
|
|
|
|
|
codeReader = new ZXing.BrowserMultiFormatReader();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add event listeners
|
|
|
|
|
setupEventListeners(elements);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function setupEventListeners(elements) {
|
|
|
|
|
// Remove existing listeners first to prevent duplicates
|
|
|
|
|
removeEventListeners(elements);
|
|
|
|
|
|
|
|
|
|
// Add new listeners
|
|
|
|
|
if (elements.decodeVinBtn) {
|
|
|
|
|
elements.decodeVinBtn.addEventListener("click", decodeVin);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.scanVinBtn) {
|
|
|
|
|
elements.scanVinBtn.addEventListener("click", () => {
|
|
|
|
|
elements.resultDisplay.textContent = "";
|
|
|
|
|
startScanner(elements.videoElement);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.fallbackButton) {
|
|
|
|
|
elements.fallbackButton.addEventListener("click", () => {
|
|
|
|
|
captureAndOCR(elements.videoElement, elements.vinInput);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.serieSelect) {
|
|
|
|
|
elements.serieSelect.addEventListener("change", () => {
|
|
|
|
|
const serie_id = elements.serieSelect.value;
|
|
|
|
|
const model_id = elements.modelSelect.value;
|
|
|
|
|
if (serie_id && model_id) loadTrims(serie_id, model_id, elements);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.trimSelect) {
|
|
|
|
|
elements.trimSelect.addEventListener("change", () => {
|
|
|
|
|
const trimId = elements.trimSelect.value;
|
|
|
|
|
elements.showSpecificationButton.disabled = !trimId;
|
|
|
|
|
elements.showEquipmentButton.disabled = !trimId;
|
|
|
|
|
if (trimId) loadSpecifications(trimId, elements);
|
|
|
|
|
loadEquipment(trimId, elements);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.equipmentSelect) {
|
|
|
|
|
elements.equipmentSelect.addEventListener("change", () => {
|
|
|
|
|
const equipmentId = elements.equipmentSelect.value;
|
|
|
|
|
if (equipmentId) loadOptions(equipmentId, elements);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.closeButton) {
|
|
|
|
|
elements.closeButton.addEventListener("click", () => closeModal(elements.scanVinBtn));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.makeSelect) {
|
|
|
|
|
elements.makeSelect.addEventListener("change", (e) => {
|
|
|
|
|
loadModels(e.target.value, elements.modelSelect.value, elements);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (elements.modelSelect) {
|
|
|
|
|
elements.modelSelect.addEventListener("change", (e) => {
|
|
|
|
|
loadSeries(e.target.value, elements.yearSelect.value, elements);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function removeEventListeners(elements) {
|
|
|
|
|
// Remove all event listeners to prevent duplicates
|
|
|
|
|
const events = [
|
|
|
|
|
{ element: elements.decodeVinBtn, event: "click", func: decodeVin },
|
|
|
|
|
{ element: elements.scanVinBtn, event: "click" },
|
|
|
|
|
{ element: elements.fallbackButton, event: "click" },
|
|
|
|
|
{ element: elements.serieSelect, event: "change" },
|
|
|
|
|
{ element: elements.trimSelect, event: "change" },
|
|
|
|
|
{ element: elements.equipmentSelect, event: "change" },
|
|
|
|
|
{ element: elements.closeButton, event: "click" },
|
|
|
|
|
{ element: elements.makeSelect, event: "change" },
|
|
|
|
|
{ element: elements.modelSelect, event: "change" }
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
events.forEach(item => {
|
|
|
|
|
if (item.element) {
|
|
|
|
|
item.element.removeEventListener(item.event, item.func || null);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
document.addEventListener("DOMContentLoaded", function () {
|
|
|
|
|
const csrfToken = getCookie("csrftoken");
|
|
|
|
|
|
|
|
|
|
const vinInput = document.getElementById("{{ form.vin.id_for_label }}");
|
|
|
|
|
const decodeVinBtn = document.getElementById("decodeVinBtn");
|
|
|
|
|
const makeSelect = document.getElementById("{{ form.id_car_make.id_for_label }}");
|
|
|
|
|
const modelSelect = document.getElementById("{{ form.id_car_model.id_for_label }}");
|
|
|
|
|
const yearSelect = document.getElementById("{{ form.year.id_for_label }}");
|
|
|
|
|
const serieSelect = document.getElementById("{{ form.id_car_serie.id_for_label }}");
|
|
|
|
|
const trimSelect = document.getElementById("{{ form.id_car_trim.id_for_label }}");
|
|
|
|
|
const equipmentSelect = document.getElementById("equipment_id")
|
|
|
|
|
const showSpecificationButton = document.getElementById("specification-btn");
|
|
|
|
|
const showEquipmentButton = document.getElementById("options-btn")
|
|
|
|
|
const specificationsContent = document.getElementById("specificationsContent");
|
|
|
|
|
const optionsContent = document.getElementById("optionsContent")
|
|
|
|
|
const generationContainer = document.getElementById("generation-div")
|
|
|
|
|
|
|
|
|
|
const ajaxUrl = "{% url 'ajax_handler' request.dealer.slug %}";
|
|
|
|
|
|
|
|
|
|
const closeButton = document.querySelector(".btn-close");
|
|
|
|
|
const scanVinBtn = document.getElementById("scan-vin-btn");
|
|
|
|
|
const videoElement = document.getElementById("video");
|
|
|
|
|
const resultDisplay = document.getElementById("result");
|
|
|
|
|
const fallbackButton = document.getElementById("ocr-fallback-btn");
|
|
|
|
|
let codeReader;
|
|
|
|
|
codeReader = new ZXing.BrowserMultiFormatReader();
|
|
|
|
|
let currentStream = null;
|
|
|
|
|
|
|
|
|
|
function closeModal() {
|
|
|
|
|
stopScanner();
|
|
|
|
|
try {
|
|
|
|
|
const scannerModal = document.getElementById("scannerModal");
|
|
|
|
|
if (scannerModal) {
|
|
|
|
|
document.activeElement.blur();
|
|
|
|
|
scannerModal.setAttribute("inert", "true");
|
|
|
|
|
const modalInstance = bootstrap.Modal.getInstance(scannerModal);
|
|
|
|
|
if (modalInstance) modalInstance.hide();
|
|
|
|
|
if (scanVinBtn) scanVinBtn.focus();
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Error closing scanner modal:", err);
|
|
|
|
|
}
|
|
|
|
|
// Cookie helper function
|
|
|
|
|
function getCookie(name) {
|
|
|
|
|
let cookieValue = null;
|
|
|
|
|
if (document.cookie && document.cookie !== "") {
|
|
|
|
|
const cookies = document.cookie.split(";");
|
|
|
|
|
for (let cookie of cookies) {
|
|
|
|
|
cookie = cookie.trim();
|
|
|
|
|
if (cookie.substring(0, name.length + 1) === name + "=") {
|
|
|
|
|
cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return cookieValue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function decodeVin() {
|
|
|
|
|
const vinNumber = vinInput.value.trim();
|
|
|
|
|
if (vinNumber.length !== 17) {
|
|
|
|
|
Swal.fire("error", "{% trans 'Please enter a valid VIN.' %}");
|
|
|
|
|
/*alert("{% trans 'Please enter a valid VIN.' %}");*/
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
showLoading();
|
|
|
|
|
try {
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=decode_vin&vin_no=${vinNumber}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
if (data.success) {
|
|
|
|
|
hideLoading();
|
|
|
|
|
await updateFields(data.data);
|
|
|
|
|
} else {
|
|
|
|
|
hideLoading();
|
|
|
|
|
Swal.fire("{% trans 'error' %}", data.error);
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("Error decoding VIN:", error);
|
|
|
|
|
hideLoading();
|
|
|
|
|
Swal.fire("error", "{% trans 'An error occurred while decoding the VIN.' %}");
|
|
|
|
|
}
|
|
|
|
|
// VIN Decoding functions
|
|
|
|
|
async function decodeVin() {
|
|
|
|
|
const vinInput = document.getElementById("{{ form.vin.id_for_label }}");
|
|
|
|
|
const vinNumber = vinInput.value.trim();
|
|
|
|
|
|
|
|
|
|
if (vinNumber.length !== 17) {
|
|
|
|
|
Swal.fire("error", "{% trans 'Please enter a valid VIN.' %}");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
showLoading();
|
|
|
|
|
try {
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=decode_vin&vin_no=${vinNumber}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
|
|
|
|
|
if (data.success) {
|
|
|
|
|
hideLoading();
|
|
|
|
|
await updateFields(data.data);
|
|
|
|
|
} else {
|
|
|
|
|
hideLoading();
|
|
|
|
|
Swal.fire("{% trans 'error' %}", data.error);
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("Error decoding VIN:", error);
|
|
|
|
|
hideLoading();
|
|
|
|
|
Swal.fire("error", "{% trans 'An error occurred while decoding the VIN.' %}");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function updateFields(vinData) {
|
|
|
|
|
const elements = {
|
|
|
|
|
makeSelect: document.getElementById("{{ form.id_car_make.id_for_label }}"),
|
|
|
|
|
modelSelect: document.getElementById("{{ form.id_car_model.id_for_label }}"),
|
|
|
|
|
yearSelect: document.getElementById("{{ form.year.id_for_label }}"),
|
|
|
|
|
generationContainer: document.getElementById("generation-div")
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
console.log(vinData);
|
|
|
|
|
if (vinData.make_id && elements.makeSelect) {
|
|
|
|
|
elements.makeSelect.value = vinData.make_id;
|
|
|
|
|
document.getElementById("make-check").innerHTML = "✓";
|
|
|
|
|
await loadModels(vinData.make_id);
|
|
|
|
|
}
|
|
|
|
|
if (vinData.model_id && elements.modelSelect) {
|
|
|
|
|
elements.modelSelect.value = vinData.model_id;
|
|
|
|
|
document.getElementById("model-check").innerHTML = "✓";
|
|
|
|
|
await loadSeries(vinData.model_id, vinData.year);
|
|
|
|
|
}
|
|
|
|
|
if (vinData.year && elements.yearSelect) {
|
|
|
|
|
elements.yearSelect.value = vinData.year;
|
|
|
|
|
document.getElementById("year-check").innerHTML = "✓";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Scanner functions
|
|
|
|
|
async function startScanner(videoElement) {
|
|
|
|
|
if (!codeReader) return;
|
|
|
|
|
|
|
|
|
|
codeReader
|
|
|
|
|
.decodeFromVideoDevice(null, videoElement, async (result, err) => {
|
|
|
|
|
if (result) {
|
|
|
|
|
document.getElementById("{{ form.vin.id_for_label }}").value = result.text;
|
|
|
|
|
closeModal();
|
|
|
|
|
await decodeVin();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function updateFields(vinData) {
|
|
|
|
|
console.log(vinData);
|
|
|
|
|
if (vinData.make_id) {
|
|
|
|
|
makeSelect.value = vinData.make_id;
|
|
|
|
|
document.getElementById("make-check").innerHTML = "✓";
|
|
|
|
|
await loadModels(vinData.make_id);
|
|
|
|
|
}
|
|
|
|
|
if (vinData.model_id) {
|
|
|
|
|
modelSelect.value = vinData.model_id;
|
|
|
|
|
document.getElementById("model-check").innerHTML = "✓";
|
|
|
|
|
|
|
|
|
|
await loadSeries(vinData.model_id, vinData.year);
|
|
|
|
|
}
|
|
|
|
|
if (vinData.year) {
|
|
|
|
|
yearSelect.value = vinData.year;
|
|
|
|
|
document.getElementById("year-check").innerHTML = "✓";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Start the scanner
|
|
|
|
|
async function startScanner() {
|
|
|
|
|
codeReader
|
|
|
|
|
.decodeFromVideoDevice(null, videoElement, async (result, err) => {
|
|
|
|
|
let res = await result;
|
|
|
|
|
if (result) {
|
|
|
|
|
vinInput.value = result.text;
|
|
|
|
|
closeModal();
|
|
|
|
|
await decodeVin();
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
.catch(console.error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function captureAndOCR() {
|
|
|
|
|
const canvas = document.createElement("canvas");
|
|
|
|
|
const context = canvas.getContext("2d");
|
|
|
|
|
canvas.width = videoElement.videoWidth;
|
|
|
|
|
canvas.height = videoElement.videoHeight;
|
|
|
|
|
context.drawImage(videoElement, 0, 0, canvas.width, canvas.height);
|
|
|
|
|
Tesseract.recognize(canvas.toDataURL("image/png"), "eng")
|
|
|
|
|
.then(({ data: { text } }) => {
|
|
|
|
|
const vin = text.match(/[A-HJ-NPR-Z0-9]{17}/);
|
|
|
|
|
if (vin) vinInput.value = vin[0];
|
|
|
|
|
closeModal();
|
|
|
|
|
decodeVin();
|
|
|
|
|
})
|
|
|
|
|
.catch((err) => console.error("OCR Error:", err));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function stopScanner() {
|
|
|
|
|
if (currentStream) {
|
|
|
|
|
currentStream.getTracks().forEach((track) => track.stop());
|
|
|
|
|
currentStream = null;
|
|
|
|
|
}
|
|
|
|
|
codeReader.reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function resetDropdown(dropdown, placeholder) {
|
|
|
|
|
dropdown.innerHTML = `<option value="">${placeholder}</option>`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadModels(makeId) {
|
|
|
|
|
resetDropdown(modelSelect, '{% trans "Select" %}');
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_models&make_id=${makeId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((model) => {
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = model.id_car_model;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? model.name : model.arabic_name;
|
|
|
|
|
modelSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadSeries(modelId, year) {
|
|
|
|
|
resetDropdown(serieSelect, '{% trans "Select" %}');
|
|
|
|
|
resetDropdown(trimSelect, '{% trans "Select" %}');
|
|
|
|
|
specificationsContent.innerHTML = "";
|
|
|
|
|
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_series&model_id=${modelId}&year=${year}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
console.log(data)
|
|
|
|
|
data.forEach((serie) => {
|
|
|
|
|
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = serie.id_car_serie;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? serie.name : serie.name;
|
|
|
|
|
generationContainer.innerHTML = serie.generation_name
|
|
|
|
|
serieSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadTrims(serie_id, model_id) {
|
|
|
|
|
resetDropdown(trimSelect, '{% trans "Select" %}');
|
|
|
|
|
specificationsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_trims&serie_id=${serie_id}&model_id=${model_id}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((trim) => {
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = trim.id_car_trim;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? trim.name : trim.name;
|
|
|
|
|
trimSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
showSpecificationButton.disabled = !this.value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadEquipment(trimId){
|
|
|
|
|
resetDropdown(equipmentSelect, '{% trans "Select" %}');
|
|
|
|
|
optionsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_equipments&trim_id=${trimId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
'X-Requested-With': 'XMLHttpRequest',
|
|
|
|
|
'X-CSRFToken': csrfToken
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((equipment) => {
|
|
|
|
|
const option = document.createElement('option');
|
|
|
|
|
option.value = equipment.id_car_equipment;
|
|
|
|
|
option.textContent = equipment.name;
|
|
|
|
|
equipmentSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadSpecifications(trimId) {
|
|
|
|
|
specificationsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_specifications&trim_id=${trimId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((spec) => {
|
|
|
|
|
const parentDiv = document.createElement("div");
|
|
|
|
|
parentDiv.innerHTML = `<strong>${spec.parent_name}</strong>`;
|
|
|
|
|
spec.specifications.forEach((s) => {
|
|
|
|
|
const specDiv = document.createElement("div");
|
|
|
|
|
specDiv.innerHTML = `• ${s.s_name}: ${s.s_value} ${s.s_unit}`;
|
|
|
|
|
parentDiv.appendChild(specDiv);
|
|
|
|
|
});
|
|
|
|
|
specificationsContent.appendChild(parentDiv);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadOptions(equipmentId) {
|
|
|
|
|
optionsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_options&equipment_id=${equipmentId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((parent) => {
|
|
|
|
|
const parentDiv = document.createElement("div");
|
|
|
|
|
parentDiv.innerHTML = `<strong>${parent.parent_name}</strong>`;
|
|
|
|
|
parent.options.forEach((option) => {
|
|
|
|
|
const optDiv = document.createElement("div");
|
|
|
|
|
optDiv.innerHTML = `• ${option.option_name}`;
|
|
|
|
|
parentDiv.appendChild(optDiv);
|
|
|
|
|
});
|
|
|
|
|
optionsContent.appendChild(parentDiv);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scanVinBtn.addEventListener("click", () => {
|
|
|
|
|
resultDisplay.textContent = "";
|
|
|
|
|
startScanner();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
fallbackButton.addEventListener("click", () => {
|
|
|
|
|
captureAndOCR();
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
serieSelect.addEventListener("change", () => {
|
|
|
|
|
const serie_id = serieSelect.value;
|
|
|
|
|
const model_id = modelSelect.value;
|
|
|
|
|
if (serie_id && model_id) loadTrims(serie_id, model_id);
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
trimSelect.addEventListener("change", () => {
|
|
|
|
|
const trimId = trimSelect.value
|
|
|
|
|
showSpecificationButton.disabled = !trimId
|
|
|
|
|
showEquipmentButton.disabled = !trimId
|
|
|
|
|
if (trimId) loadSpecifications(trimId)
|
|
|
|
|
loadEquipment(trimId)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
equipmentSelect.addEventListener("change", () => {
|
|
|
|
|
const equipmentId = equipmentSelect.value
|
|
|
|
|
if (equipmentId) loadOptions(equipmentId)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
closeButton.addEventListener("click", closeModal);
|
|
|
|
|
makeSelect.addEventListener("change", (e) => {
|
|
|
|
|
loadModels(e.target.value, modelSelect.value);
|
|
|
|
|
})
|
|
|
|
|
modelSelect.addEventListener("change", (e) => {
|
|
|
|
|
loadSeries(e.target.value, yearSelect.value);
|
|
|
|
|
})
|
|
|
|
|
decodeVinBtn.addEventListener("click", decodeVin);
|
|
|
|
|
})
|
|
|
|
|
.catch(console.error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function captureAndOCR(videoElement, vinInput) {
|
|
|
|
|
const canvas = document.createElement("canvas");
|
|
|
|
|
const context = canvas.getContext("2d");
|
|
|
|
|
canvas.width = videoElement.videoWidth;
|
|
|
|
|
canvas.height = videoElement.videoHeight;
|
|
|
|
|
context.drawImage(videoElement, 0, 0, canvas.width, canvas.height);
|
|
|
|
|
|
|
|
|
|
function showLoading() {
|
|
|
|
|
Swal.fire({
|
|
|
|
|
title: "{% trans 'Please Wait' %}",
|
|
|
|
|
text: "{% trans 'Loading' %}...",
|
|
|
|
|
allowOutsideClick: false,
|
|
|
|
|
didOpen: () => {
|
|
|
|
|
Swal.showLoading();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
if (typeof Tesseract !== 'undefined') {
|
|
|
|
|
Tesseract.recognize(canvas.toDataURL("image/png"), "eng")
|
|
|
|
|
.then(({ data: { text } }) => {
|
|
|
|
|
const vin = text.match(/[A-HJ-NPR-Z0-9]{17}/);
|
|
|
|
|
if (vin) vinInput.value = vin[0];
|
|
|
|
|
closeModal();
|
|
|
|
|
decodeVin();
|
|
|
|
|
})
|
|
|
|
|
.catch((err) => console.error("OCR Error:", err));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function closeModal(scanVinBtn) {
|
|
|
|
|
stopScanner();
|
|
|
|
|
try {
|
|
|
|
|
const scannerModal = document.getElementById("scannerModal");
|
|
|
|
|
if (scannerModal) {
|
|
|
|
|
document.activeElement.blur();
|
|
|
|
|
scannerModal.setAttribute("inert", "true");
|
|
|
|
|
const modalInstance = bootstrap.Modal.getInstance(scannerModal);
|
|
|
|
|
if (modalInstance) modalInstance.hide();
|
|
|
|
|
if (scanVinBtn) scanVinBtn.focus();
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Error closing scanner modal:", err);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function hideLoading() {
|
|
|
|
|
Swal.close();
|
|
|
|
|
}
|
|
|
|
|
function stopScanner() {
|
|
|
|
|
if (currentStream) {
|
|
|
|
|
currentStream.getTracks().forEach((track) => track.stop());
|
|
|
|
|
currentStream = null;
|
|
|
|
|
}
|
|
|
|
|
if (codeReader) codeReader.reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function notify(tag, msg) {
|
|
|
|
|
Swal.fire({
|
|
|
|
|
icon: tag,
|
|
|
|
|
titleText: msg,
|
|
|
|
|
});
|
|
|
|
|
// Data loading functions
|
|
|
|
|
function resetDropdown(dropdown, placeholder) {
|
|
|
|
|
if (dropdown) dropdown.innerHTML = `<option value="">${placeholder}</option>`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadModels(makeId, modelSelect, elements) {
|
|
|
|
|
if (!modelSelect) return;
|
|
|
|
|
|
|
|
|
|
resetDropdown(modelSelect, '{% trans "Select" %}');
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_models&make_id=${makeId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
data.forEach((model) => {
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = model.id_car_model;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? model.name : model.arabic_name;
|
|
|
|
|
modelSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadSeries(modelId, year, elements) {
|
|
|
|
|
if (!elements?.serieSelect) return;
|
|
|
|
|
|
|
|
|
|
resetDropdown(elements.serieSelect, '{% trans "Select" %}');
|
|
|
|
|
resetDropdown(elements.trimSelect, '{% trans "Select" %}');
|
|
|
|
|
if (elements.specificationsContent) elements.specificationsContent.innerHTML = "";
|
|
|
|
|
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_series&model_id=${modelId}&year=${year}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
console.log(data);
|
|
|
|
|
|
|
|
|
|
data.forEach((serie) => {
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = serie.id_car_serie;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? serie.name : serie.name;
|
|
|
|
|
if (elements.generationContainer) elements.generationContainer.innerHTML = serie.generation_name;
|
|
|
|
|
elements.serieSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadTrims(serie_id, model_id, elements) {
|
|
|
|
|
if (!elements?.trimSelect) return;
|
|
|
|
|
|
|
|
|
|
resetDropdown(elements.trimSelect, '{% trans "Select" %}');
|
|
|
|
|
if (elements.specificationsContent) elements.specificationsContent.innerHTML = "";
|
|
|
|
|
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_trims&serie_id=${serie_id}&model_id=${model_id}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
{% endblock %}
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
|
|
|
|
|
data.forEach((trim) => {
|
|
|
|
|
const option = document.createElement("option");
|
|
|
|
|
option.value = trim.id_car_trim;
|
|
|
|
|
option.textContent = document.documentElement.lang === "en" ? trim.name : trim.name;
|
|
|
|
|
elements.trimSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (elements.showSpecificationButton) {
|
|
|
|
|
elements.showSpecificationButton.disabled = !elements.trimSelect.value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadEquipment(trimId, elements) {
|
|
|
|
|
if (!elements?.equipmentSelect) return;
|
|
|
|
|
|
|
|
|
|
resetDropdown(elements.equipmentSelect, '{% trans "Select" %}');
|
|
|
|
|
if (elements.optionsContent) elements.optionsContent.innerHTML = "";
|
|
|
|
|
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_equipments&trim_id=${trimId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
'X-Requested-With': 'XMLHttpRequest',
|
|
|
|
|
'X-CSRFToken': csrfToken
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
|
|
|
|
|
data.forEach((equipment) => {
|
|
|
|
|
const option = document.createElement('option');
|
|
|
|
|
option.value = equipment.id_car_equipment;
|
|
|
|
|
option.textContent = equipment.name;
|
|
|
|
|
elements.equipmentSelect.appendChild(option);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadSpecifications(trimId, elements) {
|
|
|
|
|
if (!elements?.specificationsContent) return;
|
|
|
|
|
|
|
|
|
|
elements.specificationsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_specifications&trim_id=${trimId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
|
|
|
|
|
data.forEach((spec) => {
|
|
|
|
|
const parentDiv = document.createElement("div");
|
|
|
|
|
parentDiv.innerHTML = `<strong>${spec.parent_name}</strong>`;
|
|
|
|
|
spec.specifications.forEach((s) => {
|
|
|
|
|
const specDiv = document.createElement("div");
|
|
|
|
|
specDiv.innerHTML = `• ${s.s_name}: ${s.s_value} ${s.s_unit}`;
|
|
|
|
|
parentDiv.appendChild(specDiv);
|
|
|
|
|
});
|
|
|
|
|
elements.specificationsContent.appendChild(parentDiv);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadOptions(equipmentId, elements) {
|
|
|
|
|
if (!elements?.optionsContent) return;
|
|
|
|
|
|
|
|
|
|
elements.optionsContent.innerHTML = "";
|
|
|
|
|
const response = await fetch(`${ajaxUrl}?action=get_options&equipment_id=${equipmentId}`, {
|
|
|
|
|
headers: {
|
|
|
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
|
|
|
"X-CSRFToken": csrfToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
|
|
|
|
|
data.forEach((parent) => {
|
|
|
|
|
const parentDiv = document.createElement("div");
|
|
|
|
|
parentDiv.innerHTML = `<strong>${parent.parent_name}</strong>`;
|
|
|
|
|
parent.options.forEach((option) => {
|
|
|
|
|
const optDiv = document.createElement("div");
|
|
|
|
|
optDiv.innerHTML = `• ${option.option_name}`;
|
|
|
|
|
parentDiv.appendChild(optDiv);
|
|
|
|
|
});
|
|
|
|
|
elements.optionsContent.appendChild(parentDiv);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// UI Helper functions
|
|
|
|
|
function showLoading() {
|
|
|
|
|
Swal.fire({
|
|
|
|
|
title: "{% trans 'Please Wait' %}",
|
|
|
|
|
text: "{% trans 'Loading' %}...",
|
|
|
|
|
allowOutsideClick: false,
|
|
|
|
|
didOpen: () => {
|
|
|
|
|
Swal.showLoading();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function hideLoading() {
|
|
|
|
|
Swal.close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function notify(tag, msg) {
|
|
|
|
|
Swal.fire({
|
|
|
|
|
icon: tag,
|
|
|
|
|
titleText: msg,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
{% endblock customJS %}
|