feat: ota progress bar

fix: can't set settings with inverse/scale
This commit is contained in:
2025-04-22 19:14:51 +02:00
parent 92b52e08f7
commit 5801e5fbbe
4 changed files with 64 additions and 42 deletions

View File

@@ -477,10 +477,14 @@ export class CharaDevice {
return Number(await this.send(1, ["RAM"]).then(([bytes]) => bytes)); return Number(await this.send(1, ["RAM"]).then(([bytes]) => bytes));
} }
async updateFirmware(file: File | Blob): Promise<void> { async updateFirmware(
file: ArrayBuffer,
progress: (transferred: number, total: number) => void,
): Promise<void> {
while (this.lock) { while (this.lock) {
await this.lock; await this.lock;
} }
let resolveLock: (result: true) => void; let resolveLock: (result: true) => void;
this.lock = new Promise<true>((resolve) => { this.lock = new Promise<true>((resolve) => {
resolveLock = resolve; resolveLock = resolve;
@@ -510,46 +514,46 @@ export class CharaDevice {
}); });
return it; return it;
}); });
} finally {
writer.releaseLock();
}
// Wait for the device to be ready // Wait for the device to be ready
const signal = await this.reader.read(); const signal = await this.reader.read();
serialLog.update((it) => { serialLog.update((it) => {
it.push({ it.push({
type: "output", type: "output",
value: signal.value!.trim(), value: signal.value!.trim(),
});
return it;
}); });
return it;
});
await file.stream().pipeTo(this.port.writable!); const chunkSize = 128;
for (let i = 0; i < file.byteLength; i += chunkSize) {
const chunk = file.slice(i, i + chunkSize);
await writer.write(new Uint8Array(chunk));
progress(i + chunk.byteLength, file.byteLength);
}
serialLog.update((it) => { serialLog.update((it) => {
it.push({ it.push({
type: "input", type: "input",
value: `...${file.size} bytes`, value: `...${file.byteLength} bytes`,
});
return it;
}); });
return it;
});
const result = (await this.reader.read()).value!.trim(); const result = (await this.reader.read()).value!.trim();
serialLog.update((it) => { serialLog.update((it) => {
it.push({ it.push({
type: "output", type: "output",
value: result!, value: result!,
});
return it;
}); });
return it;
});
if (result !== "OTA OK") { if (result !== "OTA OK") {
throw new Error(result); throw new Error(result);
} }
const writer2 = this.port.writable!.getWriter(); await writer.write(new TextEncoder().encode(`RST RESTART\r\n`));
try {
await writer2.write(new TextEncoder().encode(`RST RESTART\r\n`));
serialLog.update((it) => { serialLog.update((it) => {
it.push({ it.push({
type: "input", type: "input",
@@ -558,7 +562,7 @@ export class CharaDevice {
return it; return it;
}); });
} finally { } finally {
writer2.releaseLock(); writer.releaseLock();
} }
await this.suspend(); await this.suspend();

View File

@@ -49,6 +49,8 @@ export const setting: Action<
if (isNumeric) { if (isNumeric) {
value = Number(node.value); value = Number(node.value);
if (Number.isNaN(value)) return; if (Number.isNaN(value)) return;
if (min !== undefined) value = Math.max(min, value);
if (max !== undefined) value = Math.min(max, value);
value = Math.floor( value = Math.floor(
inverse !== undefined inverse !== undefined
? inverse / value ? inverse / value
@@ -56,8 +58,6 @@ export const setting: Action<
? value / scale ? value / scale
: value, : value,
); );
if (min !== undefined) value = Math.max(min, value);
if (max !== undefined) value = Math.min(max, value);
} else { } else {
value = node.checked ? 1 : 0; value = node.checked ? 1 : 0;
} }

View File

@@ -13,6 +13,7 @@
let unsafeUpdate = $state(false); let unsafeUpdate = $state(false);
let terminalOutput = $state(""); let terminalOutput = $state("");
let progress = $state(0);
let step = $state(0); let step = $state(0);
let eraseAll = $state(false); let eraseAll = $state(false);
@@ -28,9 +29,11 @@
try { try {
const file = await fetch( const file = await fetch(
`${data.meta.path}/${data.meta.update.ota}`, `${data.meta.path}/${data.meta.update.ota}`,
).then((it) => it.blob()); ).then((it) => it.arrayBuffer());
await port.updateFirmware(file); await port.updateFirmware(file, (transferred, total) => {
progress = transferred / total;
});
success = true; success = true;
} catch (e) { } catch (e) {
@@ -194,7 +197,9 @@
<section> <section>
<button <button
class="update-button" class="update-button"
class:working class:working={working && (progress <= 0 || progress >= 1)}
class:progress={working && progress > 0 && progress < 1}
style:--progress="{progress * 100}%"
class:primary={!buttonError} class:primary={!buttonError}
class:error={buttonError} class:error={buttonError}
disabled={working || $serialPort === undefined || !isCorrectDevice} disabled={working || $serialPort === undefined || !isCorrectDevice}
@@ -422,6 +427,7 @@
background: none; background: none;
} }
&.progress,
&.working { &.working {
border-color: transparent; border-color: transparent;
} }
@@ -445,15 +451,23 @@
height: 30%; height: 30%;
width: 120%; width: 120%;
} }
}
&.progress::after {
z-index: -2;
position: absolute;
left: 0;
content: "";
background: var(--md-sys-color-primary);
opacity: 0.2;
height: 100%;
width: var(--progress);
}
}
.version { .version {
color: var(--md-sys-color-secondary); color: var(--md-sys-color-secondary);
} }
.incorrect-device { .incorrect-device {
color: var(--md-sys-color-error); color: var(--md-sys-color-error);
} }

View File

@@ -132,7 +132,11 @@
type="number" type="number"
min={settingValue(item.range[0], item)} min={settingValue(item.range[0], item)}
max={settingValue(item.range[1], item)} max={settingValue(item.range[1], item)}
step={settingValue(item.step, item)} step={item.inverse !== undefined ||
item.scale !== undefined ||
item.step === undefined
? undefined
: settingValue(item.step, item)}
use:setting={{ use:setting={{
id: item.id, id: item.id,
inverse: item.inverse, inverse: item.inverse,