mirror of
https://github.com/Part-DB/Part-DB-server.git
synced 2026-03-02 21:39:35 +00:00
Allow to scan labels anywhere on the page
This commit is contained in:
parent
bebd603117
commit
f3dab36bbe
3 changed files with 154 additions and 5 deletions
|
|
@ -19,12 +19,17 @@
|
||||||
|
|
||||||
import {Controller} from "@hotwired/stimulus";
|
import {Controller} from "@hotwired/stimulus";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Purpose of this controller is to allow users to input non-printable characters like EOT, FS, etc. in a form field and submit them correctly with the form.
|
||||||
|
* The visible input field encodes non-printable characters via their Unicode Control picture representation, e.g. \n becomes ␊ and \t becomes ␉, so that they can be displayed in the input field without breaking the form submission.
|
||||||
|
* The actual value of the field, which is submitted with the form, is stored in a hidden input and contains the non-printable characters in their original form.
|
||||||
|
*/
|
||||||
export default class extends Controller {
|
export default class extends Controller {
|
||||||
|
|
||||||
_hiddenInput;
|
_hiddenInput;
|
||||||
|
|
||||||
connect() {
|
connect() {
|
||||||
this.element.addEventListener("change", this._update.bind(this));
|
this.element.addEventListener("input", this._update.bind(this));
|
||||||
|
|
||||||
// We use a hidden input to store the actual value of the field, which is submitted with the form.
|
// We use a hidden input to store the actual value of the field, which is submitted with the form.
|
||||||
// The visible input is just for user interaction and can contain non-printable characters, which are not allowed in the hidden input.
|
// The visible input is just for user interaction and can contain non-printable characters, which are not allowed in the hidden input.
|
||||||
|
|
|
||||||
136
assets/controllers/helpers/scan_special_char_controller.js
Normal file
136
assets/controllers/helpers/scan_special_char_controller.js
Normal file
|
|
@ -0,0 +1,136 @@
|
||||||
|
/*
|
||||||
|
* This file is part of Part-DB (https://github.com/Part-DB/Part-DB-symfony).
|
||||||
|
*
|
||||||
|
* Copyright (C) 2019 - 2026 Jan Böhmer (https://github.com/jbtronics)
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Controller } from "@hotwired/stimulus"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This controller listens for a special non-printable character (SOH / ASCII 1) to be entered anywhere on the page,
|
||||||
|
* which is then used as a trigger to submit the following characters as a barcode / scan input.
|
||||||
|
*/
|
||||||
|
export default class extends Controller {
|
||||||
|
connect() {
|
||||||
|
// Optional: Log to confirm global attachment
|
||||||
|
console.log("Scanner listener active")
|
||||||
|
|
||||||
|
this.isCapturing = false
|
||||||
|
this.buffer = ""
|
||||||
|
|
||||||
|
window.addEventListener("keypress", this.handleKeydown.bind(this))
|
||||||
|
}
|
||||||
|
|
||||||
|
initialize() {
|
||||||
|
this.isCapturing = false
|
||||||
|
this.buffer = ""
|
||||||
|
this.timeoutId = null
|
||||||
|
}
|
||||||
|
|
||||||
|
handleKeydown(event) {
|
||||||
|
|
||||||
|
// Ignore if the user is typing in a form field
|
||||||
|
const isInput = ["INPUT", "TEXTAREA", "SELECT"].includes(event.target.tagName) ||
|
||||||
|
event.target.isContentEditable;
|
||||||
|
if (isInput) return
|
||||||
|
|
||||||
|
// 1. Detect Start of Header (SOH / Ctrl+A)
|
||||||
|
if (event.key === "\x01" || event.keyCode === 1) {
|
||||||
|
this.startCapturing(event)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Process characters if in capture mode
|
||||||
|
if (this.isCapturing) {
|
||||||
|
this.resetTimeout() // Push the expiration back with every keypress
|
||||||
|
|
||||||
|
if (event.key === "Enter" || event.keyCode === 13) {
|
||||||
|
|
||||||
|
this.finishCapturing(event)
|
||||||
|
} else if (event.key.length === 1) {
|
||||||
|
this.buffer += event.key
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
startCapturing(event) {
|
||||||
|
this.isCapturing = true
|
||||||
|
this.buffer = ""
|
||||||
|
this.resetTimeout()
|
||||||
|
event.preventDefault()
|
||||||
|
console.debug("Scan character detected. Capture started...")
|
||||||
|
}
|
||||||
|
|
||||||
|
finishCapturing(event) {
|
||||||
|
event.preventDefault()
|
||||||
|
const data = this.buffer;
|
||||||
|
this.stopCapturing()
|
||||||
|
this.processCapture(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
stopCapturing() {
|
||||||
|
this.isCapturing = false
|
||||||
|
this.buffer = ""
|
||||||
|
if (this.timeoutId) clearTimeout(this.timeoutId)
|
||||||
|
console.debug("Capture cleared/finished.")
|
||||||
|
}
|
||||||
|
|
||||||
|
resetTimeout() {
|
||||||
|
if (this.timeoutId) clearTimeout(this.timeoutId)
|
||||||
|
|
||||||
|
this.timeoutId = setTimeout(() => {
|
||||||
|
if (this.isCapturing) {
|
||||||
|
console.warn("Capture timed out. Resetting buffer.")
|
||||||
|
this.stopCapturing()
|
||||||
|
}
|
||||||
|
}, 500)
|
||||||
|
}
|
||||||
|
|
||||||
|
processCapture(data) {
|
||||||
|
if (!data) return
|
||||||
|
|
||||||
|
console.debug("Captured scan data: " + data)
|
||||||
|
|
||||||
|
const scanInput = document.getElementById("scan_dialog_input");
|
||||||
|
if (scanInput) { //When we are on the scan dialog page, submit the form there
|
||||||
|
this._submitScanForm(data);
|
||||||
|
} else { //Otherwise use our own form (e.g. on the part list page)
|
||||||
|
this.element.querySelector("input[name='input']").value = data;
|
||||||
|
this.element.requestSubmit();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
_submitScanForm(data) {
|
||||||
|
const scanInput = document.getElementById("scan_dialog_input");
|
||||||
|
if (!scanInput) {
|
||||||
|
console.error("Scan input field not found!")
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
scanInput.value = data;
|
||||||
|
scanInput.dispatchEvent(new Event('input', { bubbles: true }));
|
||||||
|
|
||||||
|
const form = document.getElementById("scan_dialog_form");
|
||||||
|
if (!form) {
|
||||||
|
console.error("Scan form not found!")
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
form.requestSubmit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="{{ app.request.locale | replace({"_": "-"}) }}"
|
<html lang="{{ app.request.locale | replace({"_": "-"}) }}"
|
||||||
{# For the UX translator, just use the language part (before the _. should be 2 chars), otherwise it finds no translations #}
|
{# For the UX translator, just use the language part (before the _. should be 2 chars), otherwise it finds no translations #}
|
||||||
data-symfony-ux-translator-locale="{{ app.request.locale|u.truncate(2) }}">
|
data-symfony-ux-translator-locale="{{ app.request.locale|u.truncate(2) }}">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
|
|
@ -73,9 +73,17 @@
|
||||||
{{ encore_entry_script_tags('webauthn_tfa') }}
|
{{ encore_entry_script_tags('webauthn_tfa') }}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
</head>
|
</head>
|
||||||
<body data-base-url="{{ path('homepage', {'_locale': app.request.locale}) }}"
|
<body data-base-url="{{ path('homepage', {'_locale': app.request.locale}) }}"
|
||||||
data-locale="{{ app.request.locale|default("en")|slice(0,2) }}"
|
data-locale="{{ app.request.locale|default("en")|slice(0,2) }}"
|
||||||
data-keybindings-special-characters="{{ settings_instance('keybindings').enableSpecialCharacters ? 'true' : 'false' }}">
|
data-keybindings-special-characters="{{ settings_instance('keybindings').enableSpecialCharacters ? 'true' : 'false' }}">
|
||||||
|
|
||||||
|
{# Listen for the special #}
|
||||||
|
{% if is_granted("@tools.label_scanner") %}
|
||||||
|
<form class="d-none" {{ stimulus_controller('helpers/scan_special_char') }} action="{{ path("scan_dialog") }}">
|
||||||
|
<input name="input" type="hidden">
|
||||||
|
</form>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
{% block body %}
|
{% block body %}
|
||||||
<header>
|
<header>
|
||||||
<turbo-frame id="navbar-frame" target="content" data-turbo-action="advance">
|
<turbo-frame id="navbar-frame" target="content" data-turbo-action="advance">
|
||||||
|
|
@ -121,13 +129,13 @@
|
||||||
|
|
||||||
<!-- Back to top button -->
|
<!-- Back to top button -->
|
||||||
<button id="back-to-top" class="btn btn-primary back-to-top btn-sm" role="button" title="{% trans %}back_to_top{% endtrans %}"
|
<button id="back-to-top" class="btn btn-primary back-to-top btn-sm" role="button" title="{% trans %}back_to_top{% endtrans %}"
|
||||||
{{ stimulus_controller('common/back_to_top') }} {{ stimulus_action('common/back_to_top', 'backToTop') }}>
|
{{ stimulus_controller('common/back_to_top') }} {{ stimulus_action('common/back_to_top', 'backToTop') }}>
|
||||||
<i class="fas fa-angle-up fa-fw"></i>
|
<i class="fas fa-angle-up fa-fw"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
{# Must be outside of the sidebar or it will be hidden too #}
|
{# Must be outside of the sidebar or it will be hidden too #}
|
||||||
<button class="btn btn-outline-secondary btn-sm p-0 d-md-block d-none" type="button" id="sidebar-toggle-button" title="{% trans %}sidebar.big.toggle{% endtrans %}"
|
<button class="btn btn-outline-secondary btn-sm p-0 d-md-block d-none" type="button" id="sidebar-toggle-button" title="{% trans %}sidebar.big.toggle{% endtrans %}"
|
||||||
{{ stimulus_controller('common/hide_sidebar') }} {{ stimulus_action('common/hide_sidebar', 'toggleSidebar') }} style="--fa-width: 10px;">
|
{{ stimulus_controller('common/hide_sidebar') }} {{ stimulus_action('common/hide_sidebar', 'toggleSidebar') }} style="--fa-width: 10px;">
|
||||||
<i class="fas fa-angle-left"></i>
|
<i class="fas fa-angle-left"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue