// Configure your import map in config/importmap.rb. Read more: // https://github.com/rails/importmap-rails import "@hotwired/turbo-rails" /* Hide page before loaded for testing purposes */ function showPage(event) { document.documentElement.style.visibility="visible" } document.addEventListener('turbo:load', showPage) function groupMeasurements() { var tbody = document.getElementById('measurements'); if (!tbody) return; var prevTakenAt = null; Array.from(tbody.querySelectorAll('tr[data-taken-at]')) .filter(function(row) { return row.style.display !== 'none' }) .forEach(function(row) { var takenAt = row.dataset.takenAt; row.classList.toggle('grouped', takenAt !== null && takenAt === prevTakenAt); prevTakenAt = takenAt; }); } function buildCharts() { var container = document.getElementById('measurements-charts'); var dataEl = document.getElementById('charts-data'); if (!container || !dataEl) return; var readouts = JSON.parse(dataEl.textContent); container.innerHTML = ''; if (readouts.length === 0) return; // Data arrives sorted by taken_at from the server; group into per-quantity traces var quantities = new Map(); readouts.forEach(function(r) { if (!r.takenAt) return; if (!quantities.has(r.quantityId)) { quantities.set(r.quantityId, { name: r.quantityName, unit: r.unit, x: [], y: [] }); } var q = quantities.get(r.quantityId); q.x.push(r.takenAt); q.y.push(r.value); }); var traces = []; quantities.forEach(function(q) { traces.push({ x: q.x, y: q.y, mode: 'lines+markers', type: 'scatter', name: q.name + ' (' + q.unit + ')', marker: { size: 5 } }); }); var div = document.createElement('div'); div.className = 'chart-panel'; container.appendChild(div); Plotly.newPlot(div, traces, { xaxis: { type: 'date', tickformat: '%Y-%m-%d %H:%M' }, yaxis: {}, margin: { t: 20, r: 20, b: 80, l: 60 }, paper_bgcolor: 'transparent', plot_bgcolor: 'transparent', font: { family: 'system-ui' }, legend: { orientation: 'h', y: -0.25 } }, { responsive: true, displayModeBar: false }); } function getMeasurementsView() { return localStorage.getItem('measurements-view') || 'compact'; } function applyMeasurementsView(view) { document.body.dataset.measurementsView = view; } function setMeasurementsView(view) { localStorage.setItem('measurements-view', view); applyMeasurementsView(view); } window.setMeasurementsView = setMeasurementsView document.addEventListener('turbo:load', function() { if (document.getElementById('charts-data')) { buildCharts(); return; } var tbody = document.getElementById('measurements'); if (!tbody) return; groupMeasurements(); applyMeasurementsView(getMeasurementsView()); new MutationObserver(function() { groupMeasurements(); }).observe(tbody, { childList: true, subtree: true, attributes: true, attributeFilter: ['style'] }); }) function detailsChange(event) { var target = event.currentTarget var count = target.querySelectorAll('input:checked:not([disabled])').length var span = target.querySelector('summary > span') var button = target.querySelector('button') if (count > 0) { span.textContent = count + ' selected'; Turbo.StreamElement.prototype.enableElement(button) } else { span.textContent = span.getAttribute('data-prompt') Turbo.StreamElement.prototype.disableElement(button) } } window.detailsChange = detailsChange /* Close open
when focus lost */ function detailsClose(event) { if (!event.relatedTarget || event.relatedTarget.closest("details") != event.currentTarget) { event.currentTarget.removeAttribute("open") } } window.detailsClose = detailsClose window.detailsObserver = new MutationObserver((mutations) => { mutations[0].target.dispatchEvent(new Event('change', {bubbles: true})) }); function readoutUnitChanged(select) { var button = select.closest('tr').querySelector('.set-default-unit'); if (select.value && select.value !== select.dataset.defaultUnitId) { Turbo.StreamElement.prototype.enableElement(button); } else { Turbo.StreamElement.prototype.disableElement(button); } } window.readoutUnitChanged = readoutUnitChanged function setDefaultUnit(button) { var select = button.closest('tr').querySelector('select[data-default-unit-id]'); var form = document.createElement('form'); form.action = button.dataset.path; form.method = 'post'; form.dataset.turboStream = 'true'; var methodInput = document.createElement('input'); methodInput.type = 'hidden'; methodInput.name = '_method'; methodInput.value = 'patch'; var unitInput = document.createElement('input'); unitInput.type = 'hidden'; unitInput.name = 'quantity[default_unit_id]'; unitInput.value = select.value; form.appendChild(methodInput); form.appendChild(unitInput); form.addEventListener('turbo:submit-end', function(event) { if (event.detail.success) { select.dataset.defaultUnitId = select.value; readoutUnitChanged(select); } form.remove(); }); document.body.appendChild(form); form.requestSubmit(); } window.setDefaultUnit = setDefaultUnit function formValidate(event) { var id = event.submitter.getAttribute("data-validate") if (!id) return; var input = document.getElementById(id) if (!input.checkValidity()) { input.reportValidity() event.preventDefault() } } window.formValidate = formValidate /* Turbo stream actions */ Turbo.StreamElement.prototype.disableElement = function(element) { element.setAttribute("disabled", "disabled") element.setAttribute("aria-disabled", "true") element.setAttribute("tabindex", "-1") } Turbo.StreamActions.disable = function() { this.targetElements.forEach((e) => { this.disableElement(e) }) } Turbo.StreamElement.prototype.enableElement = function(element) { element.removeAttribute("disabled") element.removeAttribute("aria-disabled") // Assume 'tabindex' is not used explicitly, so removing it is safe element.removeAttribute("tabindex") } Turbo.StreamActions.enable = function() { this.targetElements.forEach((e) => { this.enableElement(e) }) } /* TODO: change to visibility = collapse to avoid width change? */ Turbo.StreamActions.hide = function() { this.targetElements.forEach((e) => { e.style.display = "none" }) } Turbo.StreamActions.show = function() { this.targetElements.forEach((e) => { e.style.removeProperty("display") }) } /* Turbo.StreamActions.collapse = function() { this.targetElements.forEach((e) => { e.style.visibility = "collapse" }) } */ Turbo.StreamActions.close_form = function() { this.targetElements.forEach((e) => { /* Move focus if there's no focus or focus inside form being closed */ const focused = document.activeElement if (!focused || (focused == document.body) || e.contains(focused)) { let nextForm = e.parentElement.querySelector(`#${e.id} ~ tr:has([autofocus])`) nextForm ??= e.parentElement.querySelector("tr:has([autofocus])") nextForm?.querySelector("[autofocus]").focus() } document.getElementById(e.getAttribute("data-form")).remove() if (e.hasAttribute("data-link")) { this.enableElement(document.getElementById(e.getAttribute("data-link"))) } if (e.hasAttribute("data-hidden-row")) { document.getElementById(e.getAttribute("data-hidden-row")).removeAttribute("style") } e.remove() }) } Turbo.StreamActions.unselect = function() { this.targetElements.forEach((e) => { e.checked = false this.enableElement(e) }) } function formProcessKey(event) { switch (event.key) { case "Escape": event.currentTarget.querySelector("a[name=cancel]").click() break case "Enter": event.currentTarget.querySelector("button[name=button]").click() event.preventDefault() break } } window.formProcessKey = formProcessKey function detailsProcessKey(event) { // TODO: up/down arrows to move focus to prev/next line switch (event.key) { case "Escape": if (event.currentTarget.hasAttribute("open")) { event.currentTarget.removeAttribute("open") event.stopPropagation() } break case "Enter": var button = event.currentTarget.querySelector("button:not([disabled])") if (button) { button.click() // Autofocus won't be respected unless target is blurred event.target.blur() event.preventDefault() event.stopPropagation() } break } } window.detailsProcessKey = detailsProcessKey; /* Items table drag and drop support */ var lastEnterTime function dragStart(event) { lastEnterTime = event.timeStamp var row = event.currentTarget row.closest("table").querySelectorAll("thead tr").forEach((tr) => { tr.toggleAttribute("hidden") }) event.dataTransfer.setData("text/plain", row.getAttribute("data-drag-path")) var rowRectangle = row.getBoundingClientRect() event.dataTransfer.setDragImage(row, event.x - rowRectangle.left, event.y - rowRectangle.top) event.dataTransfer.dropEffect = "move" } window.dragStart = dragStart /* * Drag tracking assumptions (based on FF 122.0 experience): * * Enter/Leave events at the same timeStamp may not be logically ordered * (e.g. E -> E -> L, not E -> L -> E), * * not every Enter event has corresponding Leave event, especially during * rapid pointer moves * NOTE: sometimes Leave is not emitted when pointer goes fast over table * and outside. This should probably be fixed in browser, than patched here. */ function dragEnter(event) { //console.log(event.timeStamp + " " + event.type + ": " + event.currentTarget.id) dragLeave(event) lastEnterTime = event.timeStamp const id = event.currentTarget.getAttribute("data-drop-id") document.getElementById(id).classList.add("dropzone") } window.dragEnter = dragEnter function dragOver(event) { event.preventDefault() } window.dragOver = dragOver function dragLeave(event) { //console.log(event.timeStamp + " " + event.type + ": " + event.currentTarget.id) // Leave has been accounted for by Enter at the same timestamp, processed earlier if (event.timeStamp <= lastEnterTime) return event.currentTarget.closest("table").querySelectorAll(".dropzone").forEach((tr) => { tr.classList.remove("dropzone") }) } window.dragLeave = dragLeave function dragEnd(event) { dragLeave(event) event.currentTarget.closest("table").querySelectorAll("thead tr").forEach((tr) => { tr.toggleAttribute("hidden") }) } window.dragEnd = dragEnd function drop(event) { event.preventDefault() var idParam = event.currentTarget.getAttribute("data-drop-id-param") var id = event.currentTarget.getAttribute("data-drop-id").split("_").pop() var form = document.createElement('form'); form.action = event.dataTransfer.getData("text/plain"); form.method = 'post'; form.dataset.turboStream = 'true'; var input = document.createElement('input'); input.type = 'hidden'; input.name = idParam; input.value = id; form.appendChild(input); form.addEventListener('turbo:submit-end', function() { form.remove(); }); document.body.appendChild(form); form.requestSubmit(); } window.drop = drop