diff --git a/crates/haku-wasm/src/lib.rs b/crates/haku-wasm/src/lib.rs index b45e065..416092f 100644 --- a/crates/haku-wasm/src/lib.rs +++ b/crates/haku-wasm/src/lib.rs @@ -279,11 +279,24 @@ extern "C" fn haku_status_string(code: StatusCode) -> *const i8 { .as_ptr() } +#[derive(Debug)] +struct CompileStats { + ast_size: usize, +} + +#[derive(Debug)] +struct RunnableBrush { + chunk_id: ChunkId, + closure_spec: ClosureSpec, + + compile_stats: CompileStats, +} + #[derive(Debug, Default)] enum BrushState { #[default] Default, - Ready(ChunkId, ClosureSpec), + Ready(RunnableBrush), } #[derive(Debug, Default)] @@ -431,7 +444,13 @@ unsafe extern "C" fn haku_compile_brush( Ok(chunk_id) => chunk_id, Err(_) => return StatusCode::TooManyChunks, }; - brush.state = BrushState::Ready(chunk_id, closure_spec); + brush.state = BrushState::Ready(RunnableBrush { + chunk_id, + closure_spec, + compile_stats: CompileStats { + ast_size: ast.len(), + }, + }); info!("brush compiled into {chunk_id:?}"); @@ -470,7 +489,7 @@ unsafe extern "C" fn haku_begin_brush(instance: *mut Instance, brush: *const Bru let instance = &mut *instance; let brush = &*brush; - let BrushState::Ready(chunk_id, closure_spec) = brush.state else { + let BrushState::Ready(runnable) = &brush.state else { panic!("brush is not compiled and ready to be used"); }; @@ -479,10 +498,10 @@ unsafe extern "C" fn haku_begin_brush(instance: *mut Instance, brush: *const Bru instance.reset_exception(); instance.trampoline = None; - let Ok(closure_id) = instance - .vm - .create_ref(Ref::Closure(Closure::chunk(chunk_id, closure_spec))) - else { + let Ok(closure_id) = instance.vm.create_ref(Ref::Closure(Closure::chunk( + runnable.chunk_id, + runnable.closure_spec, + ))) else { return StatusCode::OutOfRefSlots; }; @@ -574,3 +593,26 @@ unsafe extern "C" fn haku_cont_dotter( ) }) } + +#[unsafe(no_mangle)] +unsafe extern "C" fn haku_stat_ast_size(brush: *const Brush) -> usize { + match &(*brush).state { + BrushState::Default => 0, + BrushState::Ready(runnable) => runnable.compile_stats.ast_size, + } +} + +#[unsafe(no_mangle)] +unsafe extern "C" fn haku_stat_num_refs(instance: *const Instance) -> usize { + (*instance).vm.num_refs() +} + +#[unsafe(no_mangle)] +unsafe extern "C" fn haku_stat_remaining_fuel(instance: *const Instance) -> usize { + (*instance).vm.remaining_fuel() +} + +#[unsafe(no_mangle)] +unsafe extern "C" fn haku_stat_remaining_memory(instance: *const Instance) -> usize { + (*instance).vm.remaining_memory() +} diff --git a/crates/haku/src/vm.rs b/crates/haku/src/vm.rs index ce9df0e..59aa97b 100644 --- a/crates/haku/src/vm.rs +++ b/crates/haku/src/vm.rs @@ -73,10 +73,18 @@ impl Vm { } } + pub fn num_refs(&self) -> usize { + self.refs.len() + } + pub fn remaining_fuel(&self) -> usize { self.fuel } + pub fn remaining_memory(&self) -> usize { + self.memory + } + pub fn set_fuel(&mut self, fuel: usize) { self.fuel = fuel; } diff --git a/scripts/mkicon.fish b/scripts/mkicon.fish index 5fbca61..406f1b4 100755 --- a/scripts/mkicon.fish +++ b/scripts/mkicon.fish @@ -1,7 +1,13 @@ #!/usr/bin/env fish -set filename $argv[1] -set icon_name (basename $filename .svg) -set icon_base64 (svgcleaner --stdout $filename 2>/dev/null | base64 -w0) +function mkicon + set -l filename $argv[1] + set -l icon_name (basename $filename .svg) + set -l icon_base64 (svgcleaner --stdout $filename 2>/dev/null | base64 -w0) -printf "--icon-%s: url('data:image/svg+xml;base64,%s');" "$icon_name" "$icon_base64" + printf "--icon-%s: url('data:image/svg+xml;base64,%s');\n" "$icon_name" "$icon_base64" +end + +for arg in $argv + mkicon $arg +end diff --git a/static/base.css b/static/base.css index 933649d..5118f27 100644 --- a/static/base.css +++ b/static/base.css @@ -166,12 +166,20 @@ pre:has(code) { /* Icons */ :root { - --icon-rkgk-grayscale: url(""); + --icon-brackets: url(""); + --icon-droplet: url(""); --icon-external-link: url(""); + --icon-memory: url(""); + --icon-object: url(""); + --icon-rkgk-grayscale: url(""); + + --icon-brackets-white: url(""); + --icon-droplet-white: url(""); + --icon-object-white: url(""); + --icon-memory-white: url(""); } .icon { - display: inline-block; vertical-align: middle; width: 16px; height: 16px; @@ -179,10 +187,35 @@ pre:has(code) { background-repeat: no-repeat; background-position: 50% 50%; - &.icon-rkgk-grayscale { - background-image: var(--icon-rkgk-grayscale); + &.icon-brackets { + background-image: var(--icon-brackets); + } + &.icon-droplet { + background-image: var(--icon-droplet); } &.icon-external-link { background-image: var(--icon-external-link); } + &.icon-memory { + background-image: var(--icon-memory); + } + &.icon-object { + background-image: var(--icon-object); + } + &.icon-rkgk-grayscale { + background-image: var(--icon-rkgk-grayscale); + } + + &.icon-brackets-white { + background-image: var(--icon-brackets-white); + } + &.icon-droplet-white { + background-image: var(--icon-droplet-white); + } + &.icon-memory-white { + background-image: var(--icon-memory-white); + } + &.icon-object-white { + background-image: var(--icon-object-white); + } } diff --git a/static/brush-cost.js b/static/brush-cost.js new file mode 100644 index 0000000..31e2263 --- /dev/null +++ b/static/brush-cost.js @@ -0,0 +1,81 @@ +class Gauge extends HTMLElement { + constructor(iconName, label) { + super(); + + this.iconName = iconName; + this.label = label; + } + + connectedCallback() { + this.role = "progressbar"; + this.classList.add("icon", `icon-${this.iconName}`); + + this.full = this.appendChild(document.createElement("div")); + this.full.classList.add("full", "icon", `icon-${this.iconName}-white`); + } + + setValue(value, valueMax) { + let clampedNormalized = Math.max(0, Math.min(1, value / valueMax)); + this.style.setProperty("--progress", `${clampedNormalized * 100}%`); + this.title = `${this.label}: ${value} / ${valueMax} (${Math.ceil((value / valueMax) * 100)}%)`; + } +} + +customElements.define("rkgk-brush-cost-gauge", Gauge); + +export class BrushCostGauges extends HTMLElement { + constructor() { + super(); + } + + connectedCallback() { + this.codeSizeGauge = this.appendChild( + createGauge({ + className: "code-size", + iconName: "brackets", + label: "Code size", + }), + ); + this.fuelGauge = this.appendChild( + createGauge({ + className: "fuel", + iconName: "droplet", + label: "Fuel", + }), + ); + this.objectsGauge = this.appendChild( + createGauge({ + className: "objects", + iconName: "object", + label: "Objects", + }), + ); + this.memoryGauge = this.appendChild( + createGauge({ + className: "memory", + iconName: "memory", + label: "Bulk memory", + }), + ); + + this.codeSizeGauge.setValue(0); + this.fuelGauge.setValue(0); + this.objectsGauge.setValue(0); + this.memoryGauge.setValue(0); + } + + update(stats) { + this.codeSizeGauge.setValue(stats.astSize, stats.astSizeMax); + this.fuelGauge.setValue(stats.fuel, stats.fuelMax); + this.objectsGauge.setValue(stats.numRefs, stats.numRefsMax); + this.memoryGauge.setValue(stats.memory, stats.memoryMax); + } +} + +customElements.define("rkgk-brush-cost-gauges", BrushCostGauges); + +function createGauge({ className, iconName, label }) { + let gauge = new Gauge(iconName, label); + gauge.classList.add(className); + return gauge; +} diff --git a/static/haku.js b/static/haku.js index efede63..2a734df 100644 --- a/static/haku.js +++ b/static/haku.js @@ -257,4 +257,24 @@ export class Haku { return { status: "ok" }; } + + get astSize() { + if (this.#pBrush != 0) { + return w.haku_stat_ast_size(this.#pBrush); + } else { + return 0; + } + } + + get numRefs() { + return w.haku_stat_num_refs(this.#pInstance); + } + + get remainingFuel() { + return w.haku_stat_remaining_fuel(this.#pInstance); + } + + get remainingMemory() { + return w.haku_stat_remaining_memory(this.#pInstance); + } } diff --git a/static/icon/brackets-white.svg b/static/icon/brackets-white.svg new file mode 100644 index 0000000..06e166d --- /dev/null +++ b/static/icon/brackets-white.svg @@ -0,0 +1,3 @@ + + + diff --git a/static/icon/brackets.svg b/static/icon/brackets.svg new file mode 100644 index 0000000..7404df9 --- /dev/null +++ b/static/icon/brackets.svg @@ -0,0 +1,3 @@ + + + diff --git a/static/icon/droplet-white.svg b/static/icon/droplet-white.svg new file mode 100644 index 0000000..ea06ed0 --- /dev/null +++ b/static/icon/droplet-white.svg @@ -0,0 +1,3 @@ + + + diff --git a/static/icon/droplet.svg b/static/icon/droplet.svg new file mode 100644 index 0000000..e17734d --- /dev/null +++ b/static/icon/droplet.svg @@ -0,0 +1,3 @@ + + + diff --git a/static/icon/memory-white.svg b/static/icon/memory-white.svg new file mode 100644 index 0000000..202e9e3 --- /dev/null +++ b/static/icon/memory-white.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/static/icon/memory.svg b/static/icon/memory.svg new file mode 100644 index 0000000..2e84fe3 --- /dev/null +++ b/static/icon/memory.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/static/icon/object-white.svg b/static/icon/object-white.svg new file mode 100644 index 0000000..e270309 --- /dev/null +++ b/static/icon/object-white.svg @@ -0,0 +1,4 @@ + + + + diff --git a/static/icon/object.svg b/static/icon/object.svg new file mode 100644 index 0000000..9054acb --- /dev/null +++ b/static/icon/object.svg @@ -0,0 +1,4 @@ + + + + diff --git a/static/icon/rkgk design.zip b/static/icon/rkgk design.zip new file mode 100644 index 0000000..1921ee5 Binary files /dev/null and b/static/icon/rkgk design.zip differ diff --git a/static/index.css b/static/index.css index 1b67091..b13f26b 100644 --- a/static/index.css +++ b/static/index.css @@ -15,7 +15,7 @@ main { height: 100%; position: relative; - &>.fullscreen { + & > .fullscreen { width: 100%; height: 100%; max-width: 100%; @@ -25,23 +25,26 @@ main { top: 0; } - &>.panels { + & > .panels { --right-width: 384px; /* Overridden by JavaScript */ - + box-sizing: border-box; padding: 16px; display: grid; - grid-template-columns: [left] 1fr [right-resize] auto [right] minmax(0, var(--right-width)); + grid-template-columns: [left] 1fr [right-resize] auto [right] minmax( + 0, + var(--right-width) + ); /* Pass all events through. Children may receive events as normal. */ pointer-events: none; - &>* { + & > * { pointer-events: all; } - &>.right { + & > .right { grid-column: right / right; min-height: 0; @@ -53,29 +56,41 @@ main { pointer-events: none; - &>* { + & > * { min-width: 0; min-height: 0; } - &>rkgk-resize-handle { + & > rkgk-resize-handle { pointer-events: auto; } - &>.docked>rkgk-brush-editor { + & > .docked > rkgk-brush-editor { max-height: 100%; pointer-events: auto; } - &>.floating>rkgk-brush-preview { - width: 128px; - height: 128px; - pointer-events: auto; + & > .floating { + display: flex; + flex-direction: column; + + gap: 12px; + + & > rkgk-brush-preview { + width: 128px; + height: 128px; + pointer-events: auto; + } + + & > rkgk-brush-cost-gauges { + width: 100%; + pointer-events: auto; + } } } } - &>rkgk-canvas-renderer { + & > rkgk-canvas-renderer { width: 100%; height: 100%; position: absolute; @@ -83,7 +98,7 @@ main { top: 0; } - &>rkgk-reticle-renderer { + & > rkgk-reticle-renderer { width: 100%; height: 100%; position: absolute; @@ -92,7 +107,7 @@ main { overflow: hidden; } - &>#js-loading { + & > #js-loading { background-color: var(--color-panel-background); display: flex; @@ -111,7 +126,7 @@ rkgk-resize-handle { cursor: col-resize; - &>.visual { + & > .visual { width: 2px; height: 100%; background-color: var(--color-brand-blue); @@ -119,7 +134,8 @@ rkgk-resize-handle { opacity: 0%; } - &:hover>.visual, &.dragging>.visual { + &:hover > .visual, + &.dragging > .visual { opacity: 100%; } } @@ -130,7 +146,7 @@ rkgk-resize-handle { rkgk-canvas-renderer { display: block; - &>canvas { + & > canvas { display: block; } } @@ -142,7 +158,7 @@ rkgk-reticle-renderer { pointer-events: none; - &>.reticles { + & > .reticles { position: relative; } } @@ -153,15 +169,15 @@ rkgk-reticle-cursor { position: absolute; display: block; - &>.container { - &>.arrow { + & > .container { + & > .arrow { width: 24px; height: 24px; background-color: var(--color); clip-path: path("M 0,0 L 13,13 L 6,13 L 0,19 Z"); } - &>.nickname { + & > .nickname { position: absolute; top: 20px; left: 8px; @@ -188,7 +204,7 @@ rkgk-code-editor { overflow: auto; - &>.layer { + & > .layer { position: absolute; left: 0; top: 0; @@ -203,7 +219,7 @@ rkgk-code-editor { display: flex; flex-direction: column; - &>.line { + & > .line { flex-shrink: 0; white-space: pre-wrap; @@ -211,14 +227,14 @@ rkgk-code-editor { } } - &>.layer-gutter { + & > .layer-gutter { user-select: none; counter-reset: line; color: transparent; - &>.line { + & > .line { display: flex; flex-direction: row; @@ -241,27 +257,27 @@ rkgk-code-editor { } } - &>.layer:not(.layer-gutter) { + & > .layer:not(.layer-gutter) { margin-left: var(--gutter-width); width: calc(100% - var(--gutter-width)); } - &>.layer-error-squiggles { + & > .layer-error-squiggles { color: transparent; - &>.line { - &>.squiggle { + & > .line { + & > .squiggle { text-decoration: underline wavy black; text-decoration-skip-ink: none; } - &>.squiggle-error { + & > .squiggle-error { text-decoration-color: var(--color-error); } } } - &>textarea { + & > textarea { display: block; width: calc(100% - var(--gutter-width)); margin: 0; @@ -294,8 +310,8 @@ rkgk-brush-editor.rkgk-panel { gap: 4px; position: relative; - - &>.text-area { + + & > .text-area { display: block; width: 100%; margin: 0; @@ -306,18 +322,19 @@ rkgk-brush-editor.rkgk-panel { box-sizing: border-box; } - &>.errors:empty, &>.error-header:empty { + & > .errors:empty, + & > .error-header:empty { display: none; } - &>.error-header { + & > .error-header { margin: 0; margin-top: 0.5em; font-size: 1rem; color: var(--color-error); } - &>.errors { + & > .errors { margin: 0; color: var(--color-error); white-space: pre-wrap; @@ -334,13 +351,19 @@ rkgk-brush-preview { display: block; position: relative; - background: - repeating-conic-gradient(var(--checkerboard-light) 0% 25%, var(--checkerboard-dark) 0% 50%) + background: repeating-conic-gradient( + var(--checkerboard-light) 0% 25%, + var(--checkerboard-dark) 0% 50% + ) 50% 50% / var(--checkerboard-size) var(--checkerboard-size); border-radius: 4px; + & > canvas { + border-radius: 4px; + } + &.error { - &>canvas { + & > canvas { display: none; } @@ -354,10 +377,53 @@ rkgk-brush-preview { } } +/* Brush cost gauges */ + +rkgk-brush-cost-gauges, +rkgk-brush-cost-gauges.rkgk-panel { + --gauge-size: 20px; + + height: var(--gauge-size); + border-radius: 4px; + + display: flex; + flex-direction: row; + + overflow: clip; /* clip corners */ + + & > rkgk-brush-cost-gauge { + display: block; + height: var(--gauge-size); + flex-grow: 1; + + & > .full { + width: 100%; + height: 100%; + + clip-path: xywh(0 0 var(--progress) 100%); + + background-color: var(--gauge-color); + } + + &.code-size { + --gauge-color: var(--color-brand-blue); + } + &.fuel { + --gauge-color: #f44096; + } + &.objects { + --gauge-color: #fd9916; + } + &.memory { + --gauge-color: #5aca40; + } + } +} + /* Welcome screen */ rkgk-welcome { - &>dialog { + & > dialog { h3 { margin: 0.5rem 0; font-size: 2rem; @@ -369,7 +435,8 @@ rkgk-welcome { /* Connection status dialogs */ rkgk-connection-status { - &>dialog[name='logging-in-dialog'][open], &>dialog[name='disconnected-dialog'][open] { + & > dialog[name="logging-in-dialog"][open], + & > dialog[name="disconnected-dialog"][open] { border: none; outline: none; background: none; @@ -379,8 +446,8 @@ rkgk-connection-status { align-items: center; } - &>dialog[name='error-dialog'][open] { - & textarea[name='error-text'] { + & > dialog[name="error-dialog"][open] { + & textarea[name="error-text"] { box-sizing: border-box; width: 100%; resize: none; @@ -394,7 +461,7 @@ rkgk-connection-status { .menu-bar { --border-radius: 4px; - + display: flex; align-items: center; box-sizing: border-box; @@ -403,9 +470,9 @@ rkgk-connection-status { height: 24px; border-radius: var(--border-radius); - &>a { + & > a { display: block; - + color: var(--color-text); padding: 4px 8px; text-decoration: none; @@ -432,11 +499,10 @@ rkgk-connection-status { } } - &>hr { + & > hr { height: 100%; margin: 0; border: none; border-right: 1px solid var(--color-panel-border); } } - diff --git a/static/index.js b/static/index.js index 5434f61..58ac40e 100644 --- a/static/index.js +++ b/static/index.js @@ -18,6 +18,7 @@ let canvasRenderer = main.querySelector("rkgk-canvas-renderer"); let reticleRenderer = main.querySelector("rkgk-reticle-renderer"); let brushEditor = main.querySelector("rkgk-brush-editor"); let brushPreview = main.querySelector("rkgk-brush-preview"); +let brushCostGauges = main.querySelector("rkgk-brush-cost-gauges"); let welcome = main.querySelector("rkgk-welcome"); let connectionStatus = main.querySelector("rkgk-connection-status"); @@ -251,12 +252,15 @@ function readUrl(urlString) { let compileResult = currentUser.setBrush(brushEditor.code); brushEditor.renderHakuResult("Compilation", compileResult); + brushCostGauges.update(currentUser.getStats(session.wallInfo)); + if (compileResult.status != "ok") { brushPreview.setErrorFlag(); return; } brushPreview.renderBrush(currentUser.haku).then((previewResult) => { + brushCostGauges.update(currentUser.getStats(session.wallInfo)); if (previewResult.status == "error") { brushEditor.renderHakuResult( previewResult.phase == "eval" ? "Evaluation" : "Rendering", diff --git a/static/online-users.js b/static/online-users.js index 16f39a0..9424412 100644 --- a/static/online-users.js +++ b/static/online-users.js @@ -89,6 +89,19 @@ export class User { return false; } } + + getStats(wallInfo) { + return { + astSize: this.haku.astSize, + astSizeMax: wallInfo.hakuLimits.ast_capacity, + numRefs: this.haku.numRefs, + numRefsMax: wallInfo.hakuLimits.ref_capacity, + fuel: wallInfo.hakuLimits.fuel - this.haku.remainingFuel, + fuelMax: wallInfo.hakuLimits.fuel, + memory: wallInfo.hakuLimits.memory - this.haku.remainingMemory, + memoryMax: wallInfo.hakuLimits.memory, + }; + } } export class OnlineUsers extends EventTarget { diff --git a/template/index.hbs.html b/template/index.hbs.html index 4aeff19..f9d846a 100644 --- a/template/index.hbs.html +++ b/template/index.hbs.html @@ -21,6 +21,7 @@