All checks were successful
Docker Image CI / build (push) Successful in 3m45s
110 lines
3.2 KiB
JavaScript
110 lines
3.2 KiB
JavaScript
import { Controller } from "/lib/hotwired/stimulus/dist/stimulus.js";
|
|
|
|
export default class TransactionItemFormController extends Controller {
|
|
static targets = ["barcodeButton", "barcodeData", "barcodeFormat", "barcodeFormField", "brand", "option"];
|
|
|
|
#scanning = false;
|
|
#scanIntervalId;
|
|
#stream;
|
|
|
|
connect() {
|
|
if ('BarcodeDetector' in globalThis) {
|
|
this.barcodeFormFieldTarget.hidden = false;
|
|
if (this.barcodeDataTarget.value) {
|
|
this.brandTarget.autofocus = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
disconnect() {
|
|
this.barcodeFormFieldTarget.hidden = true;
|
|
this.brandTarget.autofocus = true;
|
|
this.stopScanning();
|
|
}
|
|
|
|
filterNames(event) {
|
|
for (const option of this.optionTargets) {
|
|
if (!event.target.value) {
|
|
option.disabled = false;
|
|
continue;
|
|
}
|
|
|
|
const value = option.getAttribute("data-brand");
|
|
option.disabled = value !== event.target.value;
|
|
}
|
|
}
|
|
|
|
setPriceAndQuantity(event) {
|
|
const { brand, name, price, quantity, unit } = event.target.form.elements;
|
|
|
|
if (!brand.value || !name.value) {
|
|
price.value = "";
|
|
quantity.value = !unit.value ? "1" : "";
|
|
return;
|
|
}
|
|
|
|
const option = this.optionTargets.find(option =>
|
|
option.getAttribute("data-brand") === brand.value &&
|
|
option.value === name.value);
|
|
|
|
if (option != null) {
|
|
if (!price.value) {
|
|
price.value = option.getAttribute("data-price");
|
|
}
|
|
if (quantity.value || (!unit.value && quantity.value === "1")) {
|
|
quantity.value = option.getAttribute("data-quantity") || (!unit.value ? "1" : "");
|
|
}
|
|
}
|
|
}
|
|
|
|
async scanBarcode(event) {
|
|
event?.preventDefault();
|
|
|
|
if (this.#scanning) {
|
|
return;
|
|
}
|
|
|
|
this.#scanning = true;
|
|
this.barcodeDataTarget.value = "";
|
|
this.barcodeFormatTarget.value = "";
|
|
|
|
this.#stream = await navigator.mediaDevices.getUserMedia({ video: { facingMode: "environment" } });
|
|
|
|
const video = document.createElement("video");
|
|
video.srcObject = this.#stream;
|
|
await video.play();
|
|
|
|
const detector = new BarcodeDetector();
|
|
|
|
this.#scanIntervalId = setInterval(async () => {
|
|
const barcodes = await detector.detect(video);
|
|
if (barcodes.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const barcode = barcodes[0];
|
|
this.barcodeDataTarget.value = barcode.rawValue;
|
|
this.barcodeFormatTarget.value = barcode.format;
|
|
|
|
this.stopScanning();
|
|
|
|
const form = this.element.closest("form");
|
|
for (const element of form.elements) {
|
|
element.disabled = !element.name.startsWith("barcode");
|
|
}
|
|
|
|
form.requestSubmit(this.barcodeButtonTarget);
|
|
}, 250);
|
|
}
|
|
|
|
stopScanning() {
|
|
if (this.#scanning) {
|
|
this.#stream?.getTracks()
|
|
.forEach(track => track.stop());
|
|
|
|
clearInterval(this.#scanIntervalId);
|
|
this.#scanning = false;
|
|
}
|
|
}
|
|
}
|