From fb8e391587f3e3ffde9e7c7d68940fd24dc80a6f Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sat, 25 Jan 2025 13:00:18 -0800 Subject: [PATCH 01/25] Generalize constraints to observables --- app-proto/main.css | 8 +++---- app-proto/src/add_remove.rs | 44 ++++++++++++++++++++++++++----------- app-proto/src/assembly.rs | 18 ++++++++++----- app-proto/src/outline.rs | 42 +++++++++++++++++++++++++---------- 4 files changed, 77 insertions(+), 35 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index b9fc0a1..9c45c29 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -131,10 +131,6 @@ details[open]:has(li) .element-switch::after { color: var(--text-invalid); } -.constraint > input[type=checkbox] { - margin: 0px 8px 0px 0px; -} - .constraint > input[type=text] { color: inherit; background-color: inherit; @@ -154,6 +150,10 @@ details[open]:has(li) .element-switch::after { font-style: normal; } +.constrained > .status::after, details:has(.constrained):not([open]) .status::after { + content: '🔗'; +} + .invalid > .status::after, details:has(.invalid):not([open]) .status::after { content: '⚠'; color: var(--text-invalid); diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index ba02e65..360a12d 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -1,7 +1,17 @@ use sycamore::prelude::*; use web_sys::{console, wasm_bindgen::JsValue}; -use crate::{engine, AppState, assembly::{Assembly, Constraint, Element}}; +use crate::{ + engine, + AppState, + assembly::{ + Assembly, + Constraint, + ConstraintRole, + Element + }, + engine::Q +}; /* DEBUG */ // load an example assembly for testing. this code will be removed once we've @@ -190,15 +200,23 @@ pub fn AddRemove() -> View { (subject_vec[0].clone(), subject_vec[1].clone()) } ); - let lorentz_prod = create_signal(0.0); - let lorentz_prod_valid = create_signal(false); - let active = create_signal(true); + let measured = state.assembly.elements.map( + move |elts| { + let reps = ( + elts[subjects.0].representation.get_clone(), + elts[subjects.1].representation.get_clone() + ); + reps.0.dot(&(&*Q * reps.1)) + } + ); + let desired = create_signal(0.0); + let role = create_signal(ConstraintRole::Measure); state.assembly.insert_constraint(Constraint { subjects: subjects, - lorentz_prod: lorentz_prod, - lorentz_prod_text: create_signal(String::new()), - lorentz_prod_valid: lorentz_prod_valid, - active: active, + measured: measured, + desired: desired, + desired_text: create_signal(String::new()), + role: role, }); state.selection.update(|sel| sel.clear()); @@ -212,19 +230,19 @@ pub fn AddRemove() -> View { &JsValue::from(cst.subjects.0), &JsValue::from(cst.subjects.1), &JsValue::from(":"), - &JsValue::from(cst.lorentz_prod.get_untracked()) + &JsValue::from(cst.desired.get_untracked()) ); } }); - // update the realization when the constraint becomes active - // and valid, or is edited while active and valid + // update the realization when the observable becomes + // constrained, or is edited while constrained create_effect(move || { console::log_1(&JsValue::from( format!("Constraint ({}, {}) updated", subjects.0, subjects.1) )); - lorentz_prod.track(); - if active.get() && lorentz_prod_valid.get() { + desired.track(); + if role.with(|r| matches!(r, ConstraintRole::Constrain)) { state.assembly.realize(); } }); diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 7073c9e..37bd484 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -111,13 +111,19 @@ impl Element { } } +pub enum ConstraintRole { + Measure, + Constrain, + Invalid +} + #[derive(Clone)] pub struct Constraint { pub subjects: (ElementKey, ElementKey), - pub lorentz_prod: Signal, - pub lorentz_prod_text: Signal, - pub lorentz_prod_valid: Signal, - pub active: Signal + pub measured: ReadSignal, + pub desired: Signal, + pub desired_text: Signal, + pub role: Signal } // the velocity is expressed in uniform coordinates @@ -230,11 +236,11 @@ impl Assembly { let mut gram_to_be = PartialMatrix::new(); self.constraints.with_untracked(|csts| { for (_, cst) in csts { - if cst.active.get_untracked() && cst.lorentz_prod_valid.get_untracked() { + if cst.role.with_untracked(|role| matches!(role, ConstraintRole::Constrain)) { let subjects = cst.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, cst.lorentz_prod.get_untracked()); + gram_to_be.push_sym(row, col, cst.desired.get_untracked()); } } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index a6e968d..8459262 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -8,7 +8,16 @@ use web_sys::{ wasm_bindgen::JsCast }; -use crate::{AppState, assembly, assembly::{Constraint, ConstraintKey, ElementKey}}; +use crate::{ + AppState, + assembly, + assembly::{ + Constraint, + ConstraintKey, + ConstraintRole::*, + ElementKey + } +}; // an editable view of the Lorentz product representing a constraint #[component(inline_props)] @@ -16,16 +25,22 @@ fn LorentzProductInput(constraint: Constraint) -> View { view! { input( r#type="text", - bind:value=constraint.lorentz_prod_text, + placeholder=constraint.measured.with(|result| result.to_string()), + bind:value=constraint.desired_text, on:change=move |event: Event| { let target: HtmlInputElement = event.target().unwrap().unchecked_into(); - match target.value().parse::() { - Ok(lorentz_prod) => batch(|| { - constraint.lorentz_prod.set(lorentz_prod); - constraint.lorentz_prod_valid.set(true); - }), - Err(_) => constraint.lorentz_prod_valid.set(false) - }; + let value = target.value(); + if value.is_empty() { + constraint.role.set(Measure); + } else { + match target.value().parse::() { + Ok(desired) => batch(|| { + constraint.desired.set(desired); + constraint.role.set(Constrain); + }), + Err(_) => constraint.role.set(Invalid) + }; + } } ) } @@ -43,12 +58,15 @@ fn ConstraintOutlineItem(constraint_key: ConstraintKey, element_key: ElementKey) constraint.subjects.0 }; let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); - let class = constraint.lorentz_prod_valid.map( - |&lorentz_prod_valid| if lorentz_prod_valid { "constraint" } else { "constraint invalid" } + let class = constraint.role.map( + |role| match role { + Measure => "constraint", + Constrain => "constraint constrained", + Invalid => "constraint invalid" + } ); view! { li(class=class.get()) { - input(r#type="checkbox", bind:checked=constraint.active) div(class="constraint-label") { (other_subject_label) } LorentzProductInput(constraint=constraint) div(class="status") -- 2.43.0 From 677ef47544d1b99cfd82989dcdb83b9b6176d838 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sun, 26 Jan 2025 10:50:15 -0800 Subject: [PATCH 02/25] Rename constraints to observables --- app-proto/main.css | 14 ++++---- app-proto/src/add_remove.rs | 26 +++++++------- app-proto/src/assembly.rs | 46 ++++++++++++------------ app-proto/src/outline.rs | 70 ++++++++++++++++++------------------- 4 files changed, 78 insertions(+), 78 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index 9c45c29..e41fcd9 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -77,12 +77,12 @@ summary.selected { background-color: var(--selection-highlight); } -summary > div, .constraint { +summary > div, .observable { padding-top: 4px; padding-bottom: 4px; } -.element, .constraint { +.element, .observable { display: flex; flex-grow: 1; padding-left: 8px; @@ -107,7 +107,7 @@ details[open]:has(li) .element-switch::after { flex-grow: 1; } -.constraint-label { +.observable-label { flex-grow: 1; } @@ -123,22 +123,22 @@ details[open]:has(li) .element-switch::after { width: 56px; } -.constraint { +.observable { font-style: italic; } -.constraint.invalid { +.observable.invalid { color: var(--text-invalid); } -.constraint > input[type=text] { +.observable > input[type=text] { color: inherit; background-color: inherit; border: 1px solid var(--border); border-radius: 2px; } -.constraint.invalid > input[type=text] { +.observable.invalid > input[type=text] { border-color: var(--border-invalid); } diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index 360a12d..07b4c28 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -6,8 +6,8 @@ use crate::{ AppState, assembly::{ Assembly, - Constraint, - ConstraintRole, + Observable, + ObservableRole, Element }, engine::Q @@ -210,8 +210,8 @@ pub fn AddRemove() -> View { } ); let desired = create_signal(0.0); - let role = create_signal(ConstraintRole::Measure); - state.assembly.insert_constraint(Constraint { + let role = create_signal(ObservableRole::Measure); + state.assembly.insert_observable(Observable { subjects: subjects, measured: measured, desired: desired, @@ -221,16 +221,16 @@ pub fn AddRemove() -> View { state.selection.update(|sel| sel.clear()); /* DEBUG */ - // print updated constraint list - console::log_1(&JsValue::from("Constraints:")); - state.assembly.constraints.with(|csts| { - for (_, cst) in csts.into_iter() { + // print updated observable list + console::log_1(&JsValue::from("Observables:")); + state.assembly.observables.with(|obsls| { + for (_, obs) in obsls.into_iter() { console::log_5( &JsValue::from(" "), - &JsValue::from(cst.subjects.0), - &JsValue::from(cst.subjects.1), + &JsValue::from(obs.subjects.0), + &JsValue::from(obs.subjects.1), &JsValue::from(":"), - &JsValue::from(cst.desired.get_untracked()) + &JsValue::from(obs.desired.get_untracked()) ); } }); @@ -239,10 +239,10 @@ pub fn AddRemove() -> View { // constrained, or is edited while constrained create_effect(move || { console::log_1(&JsValue::from( - format!("Constraint ({}, {}) updated", subjects.0, subjects.1) + format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); desired.track(); - if role.with(|r| matches!(r, ConstraintRole::Constrain)) { + if role.with(|r| matches!(r, ObservableRole::Constrain)) { state.assembly.realize(); } }); diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 37bd484..e438934 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -7,9 +7,9 @@ use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ use crate::engine::{realize_gram, local_unif_to_std, ConfigSubspace, PartialMatrix}; -// the types of the keys we use to access an assembly's elements and constraints +// the types of the keys we use to access an assembly's elements and observables pub type ElementKey = usize; -pub type ConstraintKey = usize; +pub type ObservableKey = usize; pub type ElementColor = [f32; 3]; @@ -26,7 +26,7 @@ pub struct Element { pub label: String, pub color: ElementColor, pub representation: Signal>, - pub constraints: Signal>, + pub observables: Signal>, // a serial number, assigned by `Element::new`, that uniquely identifies // each element @@ -61,7 +61,7 @@ impl Element { label: label, color: color, representation: create_signal(representation), - constraints: create_signal(BTreeSet::default()), + observables: create_signal(BTreeSet::default()), serial: serial, column_index: None } @@ -111,19 +111,19 @@ impl Element { } } -pub enum ConstraintRole { +pub enum ObservableRole { Measure, Constrain, Invalid } #[derive(Clone)] -pub struct Constraint { +pub struct Observable { pub subjects: (ElementKey, ElementKey), pub measured: ReadSignal, pub desired: Signal, pub desired_text: Signal, - pub role: Signal + pub role: Signal } // the velocity is expressed in uniform coordinates @@ -137,9 +137,9 @@ type AssemblyMotion<'a> = Vec>; // a complete, view-independent description of an assembly #[derive(Clone)] pub struct Assembly { - // elements and constraints + // elements and observables pub elements: Signal>, - pub constraints: Signal>, + pub observables: Signal>, // solution variety tangent space. the basis vectors are stored in // configuration matrix format, ordered according to the elements' column @@ -161,13 +161,13 @@ impl Assembly { pub fn new() -> Assembly { Assembly { elements: create_signal(Slab::new()), - constraints: create_signal(Slab::new()), + observables: create_signal(Slab::new()), tangent: create_signal(ConfigSubspace::zero(0)), elements_by_id: create_signal(FxHashMap::default()) } } - // --- inserting elements and constraints --- + // --- inserting elements and observables --- // insert an element into the assembly without checking whether we already // have an element with the same identifier. any element that does have the @@ -210,14 +210,14 @@ impl Assembly { ); } - pub fn insert_constraint(&self, constraint: Constraint) { - let subjects = constraint.subjects; - let key = self.constraints.update(|csts| csts.insert(constraint)); - let subject_constraints = self.elements.with( - |elts| (elts[subjects.0].constraints, elts[subjects.1].constraints) + pub fn insert_observable(&self, observable: Observable) { + let subjects = observable.subjects; + let key = self.observables.update(|obsls| obsls.insert(observable)); + let subject_observables = self.elements.with( + |elts| (elts[subjects.0].observables, elts[subjects.1].observables) ); - subject_constraints.0.update(|csts| csts.insert(key)); - subject_constraints.1.update(|csts| csts.insert(key)); + subject_observables.0.update(|obsls| obsls.insert(key)); + subject_observables.1.update(|obsls| obsls.insert(key)); } // --- realization --- @@ -234,13 +234,13 @@ impl Assembly { let (gram, guess) = self.elements.with_untracked(|elts| { // set up the off-diagonal part of the Gram matrix let mut gram_to_be = PartialMatrix::new(); - self.constraints.with_untracked(|csts| { - for (_, cst) in csts { - if cst.role.with_untracked(|role| matches!(role, ConstraintRole::Constrain)) { - let subjects = cst.subjects; + self.observables.with_untracked(|obsls| { + for (_, obs) in obsls { + if obs.role.with_untracked(|role| matches!(role, ObservableRole::Constrain)) { + let subjects = obs.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, cst.desired.get_untracked()); + gram_to_be.push_sym(row, col, obs.desired.get_untracked()); } } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 8459262..5444974 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -12,33 +12,33 @@ use crate::{ AppState, assembly, assembly::{ - Constraint, - ConstraintKey, - ConstraintRole::*, + Observable, + ObservableKey, + ObservableRole::*, ElementKey } }; -// an editable view of the Lorentz product representing a constraint +// an editable view of the Lorentz product representing an observable #[component(inline_props)] -fn LorentzProductInput(constraint: Constraint) -> View { +fn ObservableInput(observable: Observable) -> View { view! { input( r#type="text", - placeholder=constraint.measured.with(|result| result.to_string()), - bind:value=constraint.desired_text, + placeholder=observable.measured.with(|result| result.to_string()), + bind:value=observable.desired_text, on:change=move |event: Event| { let target: HtmlInputElement = event.target().unwrap().unchecked_into(); let value = target.value(); if value.is_empty() { - constraint.role.set(Measure); + observable.role.set(Measure); } else { match target.value().parse::() { Ok(desired) => batch(|| { - constraint.desired.set(desired); - constraint.role.set(Constrain); + observable.desired.set(desired); + observable.role.set(Constrain); }), - Err(_) => constraint.role.set(Invalid) + Err(_) => observable.role.set(Invalid) }; } } @@ -46,29 +46,29 @@ fn LorentzProductInput(constraint: Constraint) -> View { } } -// a list item that shows a constraint in an outline view of an element +// a list item that shows an observable in an outline view of an element #[component(inline_props)] -fn ConstraintOutlineItem(constraint_key: ConstraintKey, element_key: ElementKey) -> View { +fn ObservableOutlineItem(observable_key: ObservableKey, element_key: ElementKey) -> View { let state = use_context::(); let assembly = &state.assembly; - let constraint = assembly.constraints.with(|csts| csts[constraint_key].clone()); - let other_subject = if constraint.subjects.0 == element_key { - constraint.subjects.1 + let observable = assembly.observables.with(|obsls| obsls[observable_key].clone()); + let other_subject = if observable.subjects.0 == element_key { + observable.subjects.1 } else { - constraint.subjects.0 + observable.subjects.0 }; let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); - let class = constraint.role.map( + let class = observable.role.map( |role| match role { - Measure => "constraint", - Constrain => "constraint constrained", - Invalid => "constraint invalid" + Measure => "observable", + Constrain => "observable constrained", + Invalid => "observable invalid" } ); view! { li(class=class.get()) { - div(class="constraint-label") { (other_subject_label) } - LorentzProductInput(constraint=constraint) + div(class="observable-label") { (other_subject_label) } + ObservableInput(observable=observable) div(class="status") } } @@ -92,9 +92,9 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { ).collect::>() ) }; - let constrained = element.constraints.map(|csts| csts.len() > 0); - let constraint_list = element.constraints.map( - |csts| csts.clone().into_iter().collect() + let observed = element.observables.map(|obsls| obsls.len() > 0); + let observable_list = element.observables.map( + |obsls| obsls.clone().into_iter().collect() ); let details_node = create_node_ref(); view! { @@ -109,7 +109,7 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { state.select(key, event.shift_key()); event.prevent_default(); }, - "ArrowRight" if constrained.get() => { + "ArrowRight" if observed.get() => { let _ = details_node .get() .unchecked_into::() @@ -156,16 +156,16 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { div(class="status") } } - ul(class="constraints") { + ul(class="observables") { Keyed( - list=constraint_list, - view=move |cst_key| view! { - ConstraintOutlineItem( - constraint_key=cst_key, + list=observable_list, + view=move |obs_key| view! { + ObservableOutlineItem( + observable_key=obs_key, element_key=key ) }, - key=|cst_key| cst_key.clone() + key=|obs_key| obs_key.clone() ) } } @@ -174,8 +174,8 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { } // a component that lists the elements of the current assembly, showing the -// constraints on each element as a collapsible sub-list. its implementation -// is based on Kate Morley's HTML + CSS tree views: +// observables associated with each element as a collapsible sub-list. its +// implementation is based on Kate Morley's HTML + CSS tree views: // // https://iamkate.com/code/tree-views/ // -- 2.43.0 From af2724f934e6cea8e4528428d2b4c8d3aff5b6de Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sun, 26 Jan 2025 17:48:32 -0800 Subject: [PATCH 03/25] Rename `ObservableRole` variants Also rename corresponding CSS classes and add methods to check roles. --- app-proto/main.css | 8 ++++---- app-proto/src/add_remove.rs | 4 ++-- app-proto/src/assembly.rs | 22 ++++++++++++++++++---- app-proto/src/outline.rs | 12 ++++++------ 4 files changed, 30 insertions(+), 16 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index e41fcd9..f544e19 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -127,7 +127,7 @@ details[open]:has(li) .element-switch::after { font-style: italic; } -.observable.invalid { +.observable.invalid-constraint { color: var(--text-invalid); } @@ -138,7 +138,7 @@ details[open]:has(li) .element-switch::after { border-radius: 2px; } -.observable.invalid > input[type=text] { +.observable.invalid-constraint > input[type=text] { border-color: var(--border-invalid); } @@ -150,11 +150,11 @@ details[open]:has(li) .element-switch::after { font-style: normal; } -.constrained > .status::after, details:has(.constrained):not([open]) .status::after { +.valid-constraint > .status::after, details:has(.valid-constraint):not([open]) .status::after { content: '🔗'; } -.invalid > .status::after, details:has(.invalid):not([open]) .status::after { +.invalid-constraint > .status::after, details:has(.invalid-constraint):not([open]) .status::after { content: '⚠'; color: var(--text-invalid); } diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index 07b4c28..640925a 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -210,7 +210,7 @@ pub fn AddRemove() -> View { } ); let desired = create_signal(0.0); - let role = create_signal(ObservableRole::Measure); + let role = create_signal(ObservableRole::Measurement); state.assembly.insert_observable(Observable { subjects: subjects, measured: measured, @@ -242,7 +242,7 @@ pub fn AddRemove() -> View { format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); desired.track(); - if role.with(|r| matches!(r, ObservableRole::Constrain)) { + if role.with(|rl| rl.is_valid_constraint()) { state.assembly.realize(); } }); diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index e438934..ba2f9c5 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -112,9 +112,17 @@ impl Element { } pub enum ObservableRole { - Measure, - Constrain, - Invalid + Measurement, + Constraint(bool) +} + +impl ObservableRole { + pub fn is_valid_constraint(&self) -> bool { + match self { + ObservableRole::Measurement => false, + ObservableRole::Constraint(valid) => *valid + } + } } #[derive(Clone)] @@ -126,6 +134,12 @@ pub struct Observable { pub role: Signal } +impl Observable { + fn role_is_valid_constraint_untracked(&self) -> bool { + self.role.with_untracked(|role| role.is_valid_constraint()) + } +} + // the velocity is expressed in uniform coordinates pub struct ElementMotion<'a> { pub key: ElementKey, @@ -236,7 +250,7 @@ impl Assembly { let mut gram_to_be = PartialMatrix::new(); self.observables.with_untracked(|obsls| { for (_, obs) in obsls { - if obs.role.with_untracked(|role| matches!(role, ObservableRole::Constrain)) { + if obs.role_is_valid_constraint_untracked() { let subjects = obs.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 5444974..d2f9058 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -31,14 +31,14 @@ fn ObservableInput(observable: Observable) -> View { let target: HtmlInputElement = event.target().unwrap().unchecked_into(); let value = target.value(); if value.is_empty() { - observable.role.set(Measure); + observable.role.set(Measurement); } else { match target.value().parse::() { Ok(desired) => batch(|| { observable.desired.set(desired); - observable.role.set(Constrain); + observable.role.set(Constraint(true)); }), - Err(_) => observable.role.set(Invalid) + Err(_) => observable.role.set(Constraint(false)) }; } } @@ -60,9 +60,9 @@ fn ObservableOutlineItem(observable_key: ObservableKey, element_key: ElementKey) let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); let class = observable.role.map( |role| match role { - Measure => "observable", - Constrain => "observable constrained", - Invalid => "observable invalid" + Measurement => "observable", + Constraint(true) => "observable valid-constraint", + Constraint(false) => "observable invalid-constraint" } ); view! { -- 2.43.0 From dc8330df6a3ce414648028993a43a2ee58456468 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sun, 9 Feb 2025 22:25:44 -0800 Subject: [PATCH 04/25] Revise observable styling Distinguish constraints from observables using dark background rather than marker. Customize focus highlighting. Drop the input type selector that used to make styling apply to the Lorentz product field but not the constraint activation check box. --- app-proto/main.css | 29 +++++++++++++++++++++-------- 1 file changed, 21 insertions(+), 8 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index f544e19..46bdf84 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -3,7 +3,8 @@ --text-bright: white; --text-invalid: #f58fc2; /* bright pink */ --border: #555; /* light gray */ - --border-focus: #aaa; /* bright gray */ + --border-focus-dark: #aaa; /* bright gray */ + --border-focus-light: white; --border-invalid: #70495c; /* dusky pink */ --selection-highlight: #444; /* medium gray */ --page-background: #222; /* dark gray */ @@ -131,14 +132,24 @@ details[open]:has(li) .element-switch::after { color: var(--text-invalid); } -.observable > input[type=text] { +.observable > input { color: inherit; background-color: inherit; border: 1px solid var(--border); border-radius: 2px; } -.observable.invalid-constraint > input[type=text] { +.observable > input::placeholder { + color: inherit; + opacity: 54%; + font-style: italic; +} + +.observable.valid-constraint > input { + background-color: var(--display-background); +} + +.observable.invalid-constraint > input { border-color: var(--border-invalid); } @@ -150,10 +161,6 @@ details[open]:has(li) .element-switch::after { font-style: normal; } -.valid-constraint > .status::after, details:has(.valid-constraint):not([open]) .status::after { - content: '🔗'; -} - .invalid-constraint > .status::after, details:has(.invalid-constraint):not([open]) .status::after { content: '⚠'; color: var(--text-invalid); @@ -171,5 +178,11 @@ canvas { } canvas:focus { - border-color: var(--border-focus); + border-color: var(--border-focus-dark); + outline: none; +} + +input:focus { + border-color: var(--border-focus-light); + outline: none; } \ No newline at end of file -- 2.43.0 From de7122d871b9bb9d4e7caf2a9a47f655bd6920fb Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 12 Feb 2025 10:37:48 -0800 Subject: [PATCH 05/25] Label observable type Right now, there's only one type of observable, so the label can be hard-coded. --- app-proto/main.css | 7 ++++++- app-proto/src/outline.rs | 1 + 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/app-proto/main.css b/app-proto/main.css index 46bdf84..a1ff9da 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -24,7 +24,7 @@ body { display: flex; flex-direction: column; float: left; - width: 450px; + width: 500px; height: 100vh; margin: 0px; padding: 0px; @@ -128,6 +128,11 @@ details[open]:has(li) .element-switch::after { font-style: italic; } +.observable-type { + padding: 2px 8px 0px 8px; + font-size: 10pt; +} + .observable.invalid-constraint { color: var(--text-invalid); } diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index d2f9058..f341357 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -68,6 +68,7 @@ fn ObservableOutlineItem(observable_key: ObservableKey, element_key: ElementKey) view! { li(class=class.get()) { div(class="observable-label") { (other_subject_label) } + div(class="observable-type") { "Inversive distance" } ObservableInput(observable=observable) div(class="status") } -- 2.43.0 From 24139ad5e9a1822cc0e240a6021d6a91e5ddd22a Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 12 Feb 2025 11:35:07 -0800 Subject: [PATCH 06/25] Rename observables to regulators --- app-proto/main.css | 20 +++++----- app-proto/src/add_remove.rs | 26 ++++++------- app-proto/src/assembly.rs | 54 +++++++++++++-------------- app-proto/src/outline.rs | 74 ++++++++++++++++++------------------- 4 files changed, 87 insertions(+), 87 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index a1ff9da..d71ce4a 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -78,12 +78,12 @@ summary.selected { background-color: var(--selection-highlight); } -summary > div, .observable { +summary > div, .regulator { padding-top: 4px; padding-bottom: 4px; } -.element, .observable { +.element, .regulator { display: flex; flex-grow: 1; padding-left: 8px; @@ -108,7 +108,7 @@ details[open]:has(li) .element-switch::after { flex-grow: 1; } -.observable-label { +.regulator-label { flex-grow: 1; } @@ -124,37 +124,37 @@ details[open]:has(li) .element-switch::after { width: 56px; } -.observable { +.regulator { font-style: italic; } -.observable-type { +.regulator-type { padding: 2px 8px 0px 8px; font-size: 10pt; } -.observable.invalid-constraint { +.regulator.invalid-constraint { color: var(--text-invalid); } -.observable > input { +.regulator > input { color: inherit; background-color: inherit; border: 1px solid var(--border); border-radius: 2px; } -.observable > input::placeholder { +.regulator > input::placeholder { color: inherit; opacity: 54%; font-style: italic; } -.observable.valid-constraint > input { +.regulator.valid-constraint > input { background-color: var(--display-background); } -.observable.invalid-constraint > input { +.regulator.invalid-constraint > input { border-color: var(--border-invalid); } diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index 640925a..d8d6163 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -6,8 +6,8 @@ use crate::{ AppState, assembly::{ Assembly, - Observable, - ObservableRole, + Regulator, + RegulatorRole, Element }, engine::Q @@ -210,8 +210,8 @@ pub fn AddRemove() -> View { } ); let desired = create_signal(0.0); - let role = create_signal(ObservableRole::Measurement); - state.assembly.insert_observable(Observable { + let role = create_signal(RegulatorRole::Measurement); + state.assembly.insert_regulator(Regulator { subjects: subjects, measured: measured, desired: desired, @@ -221,22 +221,22 @@ pub fn AddRemove() -> View { state.selection.update(|sel| sel.clear()); /* DEBUG */ - // print updated observable list - console::log_1(&JsValue::from("Observables:")); - state.assembly.observables.with(|obsls| { - for (_, obs) in obsls.into_iter() { + // print updated regulator list + console::log_1(&JsValue::from("Regulators:")); + state.assembly.regulators.with(|regs| { + for (_, reg) in regs.into_iter() { console::log_5( &JsValue::from(" "), - &JsValue::from(obs.subjects.0), - &JsValue::from(obs.subjects.1), + &JsValue::from(reg.subjects.0), + &JsValue::from(reg.subjects.1), &JsValue::from(":"), - &JsValue::from(obs.desired.get_untracked()) + &JsValue::from(reg.desired.get_untracked()) ); } }); - // update the realization when the observable becomes - // constrained, or is edited while constrained + // update the realization when the regulator becomes + // a constraint, or is edited while acting as a constraint create_effect(move || { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index ba2f9c5..7c9418e 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -7,9 +7,9 @@ use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ use crate::engine::{realize_gram, local_unif_to_std, ConfigSubspace, PartialMatrix}; -// the types of the keys we use to access an assembly's elements and observables +// the types of the keys we use to access an assembly's elements and regulators pub type ElementKey = usize; -pub type ObservableKey = usize; +pub type RegulatorKey = usize; pub type ElementColor = [f32; 3]; @@ -26,7 +26,7 @@ pub struct Element { pub label: String, pub color: ElementColor, pub representation: Signal>, - pub observables: Signal>, + pub regulators: Signal>, // a serial number, assigned by `Element::new`, that uniquely identifies // each element @@ -61,7 +61,7 @@ impl Element { label: label, color: color, representation: create_signal(representation), - observables: create_signal(BTreeSet::default()), + regulators: create_signal(BTreeSet::default()), serial: serial, column_index: None } @@ -111,30 +111,30 @@ impl Element { } } -pub enum ObservableRole { +pub enum RegulatorRole { Measurement, Constraint(bool) } -impl ObservableRole { +impl RegulatorRole { pub fn is_valid_constraint(&self) -> bool { match self { - ObservableRole::Measurement => false, - ObservableRole::Constraint(valid) => *valid + RegulatorRole::Measurement => false, + RegulatorRole::Constraint(valid) => *valid } } } #[derive(Clone)] -pub struct Observable { +pub struct Regulator { pub subjects: (ElementKey, ElementKey), pub measured: ReadSignal, pub desired: Signal, pub desired_text: Signal, - pub role: Signal + pub role: Signal } -impl Observable { +impl Regulator { fn role_is_valid_constraint_untracked(&self) -> bool { self.role.with_untracked(|role| role.is_valid_constraint()) } @@ -151,9 +151,9 @@ type AssemblyMotion<'a> = Vec>; // a complete, view-independent description of an assembly #[derive(Clone)] pub struct Assembly { - // elements and observables + // elements and regulators pub elements: Signal>, - pub observables: Signal>, + pub regulators: Signal>, // solution variety tangent space. the basis vectors are stored in // configuration matrix format, ordered according to the elements' column @@ -175,13 +175,13 @@ impl Assembly { pub fn new() -> Assembly { Assembly { elements: create_signal(Slab::new()), - observables: create_signal(Slab::new()), + regulators: create_signal(Slab::new()), tangent: create_signal(ConfigSubspace::zero(0)), elements_by_id: create_signal(FxHashMap::default()) } } - // --- inserting elements and observables --- + // --- inserting elements and regulators --- // insert an element into the assembly without checking whether we already // have an element with the same identifier. any element that does have the @@ -224,14 +224,14 @@ impl Assembly { ); } - pub fn insert_observable(&self, observable: Observable) { - let subjects = observable.subjects; - let key = self.observables.update(|obsls| obsls.insert(observable)); - let subject_observables = self.elements.with( - |elts| (elts[subjects.0].observables, elts[subjects.1].observables) + pub fn insert_regulator(&self, regulator: Regulator) { + let subjects = regulator.subjects; + let key = self.regulators.update(|regs| regs.insert(regulator)); + let subject_regulators = self.elements.with( + |elts| (elts[subjects.0].regulators, elts[subjects.1].regulators) ); - subject_observables.0.update(|obsls| obsls.insert(key)); - subject_observables.1.update(|obsls| obsls.insert(key)); + subject_regulators.0.update(|regs| regs.insert(key)); + subject_regulators.1.update(|regs| regs.insert(key)); } // --- realization --- @@ -248,13 +248,13 @@ impl Assembly { let (gram, guess) = self.elements.with_untracked(|elts| { // set up the off-diagonal part of the Gram matrix let mut gram_to_be = PartialMatrix::new(); - self.observables.with_untracked(|obsls| { - for (_, obs) in obsls { - if obs.role_is_valid_constraint_untracked() { - let subjects = obs.subjects; + self.regulators.with_untracked(|regs| { + for (_, reg) in regs { + if reg.role_is_valid_constraint_untracked() { + let subjects = reg.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, obs.desired.get_untracked()); + gram_to_be.push_sym(row, col, reg.desired.get_untracked()); } } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index f341357..6e3e0d3 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -12,33 +12,33 @@ use crate::{ AppState, assembly, assembly::{ - Observable, - ObservableKey, - ObservableRole::*, + Regulator, + RegulatorKey, + RegulatorRole::*, ElementKey } }; -// an editable view of the Lorentz product representing an observable +// an editable view of a regulator #[component(inline_props)] -fn ObservableInput(observable: Observable) -> View { +fn RegulatorInput(regulator: Regulator) -> View { view! { input( r#type="text", - placeholder=observable.measured.with(|result| result.to_string()), - bind:value=observable.desired_text, + placeholder=regulator.measured.with(|result| result.to_string()), + bind:value=regulator.desired_text, on:change=move |event: Event| { let target: HtmlInputElement = event.target().unwrap().unchecked_into(); let value = target.value(); if value.is_empty() { - observable.role.set(Measurement); + regulator.role.set(Measurement); } else { match target.value().parse::() { Ok(desired) => batch(|| { - observable.desired.set(desired); - observable.role.set(Constraint(true)); + regulator.desired.set(desired); + regulator.role.set(Constraint(true)); }), - Err(_) => observable.role.set(Constraint(false)) + Err(_) => regulator.role.set(Constraint(false)) }; } } @@ -46,30 +46,30 @@ fn ObservableInput(observable: Observable) -> View { } } -// a list item that shows an observable in an outline view of an element +// a list item that shows a regulator in an outline view of an element #[component(inline_props)] -fn ObservableOutlineItem(observable_key: ObservableKey, element_key: ElementKey) -> View { +fn RegulatorOutlineItem(regulator_key: RegulatorKey, element_key: ElementKey) -> View { let state = use_context::(); let assembly = &state.assembly; - let observable = assembly.observables.with(|obsls| obsls[observable_key].clone()); - let other_subject = if observable.subjects.0 == element_key { - observable.subjects.1 + let regulator = assembly.regulators.with(|regs| regs[regulator_key].clone()); + let other_subject = if regulator.subjects.0 == element_key { + regulator.subjects.1 } else { - observable.subjects.0 + regulator.subjects.0 }; let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); - let class = observable.role.map( + let class = regulator.role.map( |role| match role { - Measurement => "observable", - Constraint(true) => "observable valid-constraint", - Constraint(false) => "observable invalid-constraint" + Measurement => "regulator", + Constraint(true) => "regulator valid-constraint", + Constraint(false) => "regulator invalid-constraint" } ); view! { li(class=class.get()) { - div(class="observable-label") { (other_subject_label) } - div(class="observable-type") { "Inversive distance" } - ObservableInput(observable=observable) + div(class="regulator-label") { (other_subject_label) } + div(class="regulator-type") { "Inversive distance" } + RegulatorInput(regulator=regulator) div(class="status") } } @@ -93,9 +93,9 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { ).collect::>() ) }; - let observed = element.observables.map(|obsls| obsls.len() > 0); - let observable_list = element.observables.map( - |obsls| obsls.clone().into_iter().collect() + let regulated = element.regulators.map(|regs| regs.len() > 0); + let regulator_list = element.regulators.map( + |regs| regs.clone().into_iter().collect() ); let details_node = create_node_ref(); view! { @@ -110,7 +110,7 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { state.select(key, event.shift_key()); event.prevent_default(); }, - "ArrowRight" if observed.get() => { + "ArrowRight" if regulated.get() => { let _ = details_node .get() .unchecked_into::() @@ -157,16 +157,16 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { div(class="status") } } - ul(class="observables") { + ul(class="regulators") { Keyed( - list=observable_list, - view=move |obs_key| view! { - ObservableOutlineItem( - observable_key=obs_key, + list=regulator_list, + view=move |reg_key| view! { + RegulatorOutlineItem( + regulator_key=reg_key, element_key=key ) }, - key=|obs_key| obs_key.clone() + key=|reg_key| reg_key.clone() ) } } @@ -174,9 +174,9 @@ fn ElementOutlineItem(key: ElementKey, element: assembly::Element) -> View { } } -// a component that lists the elements of the current assembly, showing the -// observables associated with each element as a collapsible sub-list. its -// implementation is based on Kate Morley's HTML + CSS tree views: +// a component that lists the elements of the current assembly, showing each +// element's regulators in a collapsible sub-list. its implementation is based +// on Kate Morley's HTML + CSS tree views: // // https://iamkate.com/code/tree-views/ // -- 2.43.0 From b3e4e902f39d4abc195878755203799fe1302fa1 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 12 Feb 2025 11:55:45 -0800 Subject: [PATCH 07/25] Rename `Regulator` fields --- app-proto/src/add_remove.rs | 14 +++++++------- app-proto/src/assembly.rs | 8 ++++---- app-proto/src/outline.rs | 8 ++++---- 3 files changed, 15 insertions(+), 15 deletions(-) diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index d8d6163..3280dac 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -200,7 +200,7 @@ pub fn AddRemove() -> View { (subject_vec[0].clone(), subject_vec[1].clone()) } ); - let measured = state.assembly.elements.map( + let measurement = state.assembly.elements.map( move |elts| { let reps = ( elts[subjects.0].representation.get_clone(), @@ -209,13 +209,13 @@ pub fn AddRemove() -> View { reps.0.dot(&(&*Q * reps.1)) } ); - let desired = create_signal(0.0); + let set_point = create_signal(0.0); let role = create_signal(RegulatorRole::Measurement); state.assembly.insert_regulator(Regulator { subjects: subjects, - measured: measured, - desired: desired, - desired_text: create_signal(String::new()), + measurement: measurement, + set_point: set_point, + set_point_text: create_signal(String::new()), role: role, }); state.selection.update(|sel| sel.clear()); @@ -230,7 +230,7 @@ pub fn AddRemove() -> View { &JsValue::from(reg.subjects.0), &JsValue::from(reg.subjects.1), &JsValue::from(":"), - &JsValue::from(reg.desired.get_untracked()) + &JsValue::from(reg.set_point.get_untracked()) ); } }); @@ -241,7 +241,7 @@ pub fn AddRemove() -> View { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); - desired.track(); + set_point.track(); if role.with(|rl| rl.is_valid_constraint()) { state.assembly.realize(); } diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 7c9418e..c052a3b 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -128,9 +128,9 @@ impl RegulatorRole { #[derive(Clone)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), - pub measured: ReadSignal, - pub desired: Signal, - pub desired_text: Signal, + pub measurement: ReadSignal, + pub set_point: Signal, + pub set_point_text: Signal, pub role: Signal } @@ -254,7 +254,7 @@ impl Assembly { let subjects = reg.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, reg.desired.get_untracked()); + gram_to_be.push_sym(row, col, reg.set_point.get_untracked()); } } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 6e3e0d3..8dcda93 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -25,8 +25,8 @@ fn RegulatorInput(regulator: Regulator) -> View { view! { input( r#type="text", - placeholder=regulator.measured.with(|result| result.to_string()), - bind:value=regulator.desired_text, + placeholder=regulator.measurement.with(|result| result.to_string()), + bind:value=regulator.set_point_text, on:change=move |event: Event| { let target: HtmlInputElement = event.target().unwrap().unchecked_into(); let value = target.value(); @@ -34,8 +34,8 @@ fn RegulatorInput(regulator: Regulator) -> View { regulator.role.set(Measurement); } else { match target.value().parse::() { - Ok(desired) => batch(|| { - regulator.desired.set(desired); + Ok(set_pt) => batch(|| { + regulator.set_point.set(set_pt); regulator.role.set(Constraint(true)); }), Err(_) => regulator.role.set(Constraint(false)) -- 2.43.0 From fef4127f69cff3e86e5070aacd209c863993c09b Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 17 Feb 2025 14:01:27 -0800 Subject: [PATCH 08/25] Only sync regulator inputs on change This lets us infer a regulator's role from whether it has a set point and what text specifies the set point. --- app-proto/src/add_remove.rs | 10 +++------- app-proto/src/assembly.rs | 38 +++++++++++++----------------------- app-proto/src/outline.rs | 39 +++++++++++++------------------------ 3 files changed, 30 insertions(+), 57 deletions(-) diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index 3280dac..c96d5ab 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -7,7 +7,6 @@ use crate::{ assembly::{ Assembly, Regulator, - RegulatorRole, Element }, engine::Q @@ -209,14 +208,12 @@ pub fn AddRemove() -> View { reps.0.dot(&(&*Q * reps.1)) } ); - let set_point = create_signal(0.0); - let role = create_signal(RegulatorRole::Measurement); + let set_point = create_signal(None); state.assembly.insert_regulator(Regulator { subjects: subjects, measurement: measurement, set_point: set_point, - set_point_text: create_signal(String::new()), - role: role, + set_point_spec: create_signal(String::new()) }); state.selection.update(|sel| sel.clear()); @@ -241,8 +238,7 @@ pub fn AddRemove() -> View { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); - set_point.track(); - if role.with(|rl| rl.is_valid_constraint()) { + if set_point.with(|set_pt| set_pt.is_some()) { state.assembly.realize(); } }); diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index c052a3b..c1eba1e 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -111,32 +111,17 @@ impl Element { } } -pub enum RegulatorRole { - Measurement, - Constraint(bool) -} - -impl RegulatorRole { - pub fn is_valid_constraint(&self) -> bool { - match self { - RegulatorRole::Measurement => false, - RegulatorRole::Constraint(valid) => *valid - } - } -} - -#[derive(Clone)] +#[derive(Clone, Copy)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), pub measurement: ReadSignal, - pub set_point: Signal, - pub set_point_text: Signal, - pub role: Signal + pub set_point: Signal>, + pub set_point_spec: Signal } impl Regulator { - fn role_is_valid_constraint_untracked(&self) -> bool { - self.role.with_untracked(|role| role.is_valid_constraint()) + pub fn has_no_set_point_spec(&self) -> bool { + self.set_point_spec.with(|spec| spec.is_empty()) } } @@ -250,11 +235,14 @@ impl Assembly { let mut gram_to_be = PartialMatrix::new(); self.regulators.with_untracked(|regs| { for (_, reg) in regs { - if reg.role_is_valid_constraint_untracked() { - let subjects = reg.subjects; - let row = elts[subjects.0].column_index.unwrap(); - let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, reg.set_point.get_untracked()); + match reg.set_point.get_untracked() { + Some(set_pt) => { + let subjects = reg.subjects; + let row = elts[subjects.0].column_index.unwrap(); + let col = elts[subjects.1].column_index.unwrap(); + gram_to_be.push_sym(row, col, set_pt); + }, + None => () } } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 8dcda93..d892f0f 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -1,8 +1,6 @@ use itertools::Itertools; use sycamore::prelude::*; use web_sys::{ - Event, - HtmlInputElement, KeyboardEvent, MouseEvent, wasm_bindgen::JsCast @@ -14,7 +12,6 @@ use crate::{ assembly::{ Regulator, RegulatorKey, - RegulatorRole::*, ElementKey } }; @@ -22,25 +19,17 @@ use crate::{ // an editable view of a regulator #[component(inline_props)] fn RegulatorInput(regulator: Regulator) -> View { + let value = create_signal(regulator.set_point_spec.get_clone_untracked()); + create_effect(move || value.set(regulator.set_point_spec.get_clone())); view! { input( r#type="text", placeholder=regulator.measurement.with(|result| result.to_string()), - bind:value=regulator.set_point_text, - on:change=move |event: Event| { - let target: HtmlInputElement = event.target().unwrap().unchecked_into(); - let value = target.value(); - if value.is_empty() { - regulator.role.set(Measurement); - } else { - match target.value().parse::() { - Ok(set_pt) => batch(|| { - regulator.set_point.set(set_pt); - regulator.role.set(Constraint(true)); - }), - Err(_) => regulator.role.set(Constraint(false)) - }; - } + bind:value=value, + on:change=move |_| { + let value_val = value.get_clone_untracked(); + regulator.set_point.set(value_val.parse::().ok()); + regulator.set_point_spec.set(value_val); } ) } @@ -51,20 +40,20 @@ fn RegulatorInput(regulator: Regulator) -> View { fn RegulatorOutlineItem(regulator_key: RegulatorKey, element_key: ElementKey) -> View { let state = use_context::(); let assembly = &state.assembly; - let regulator = assembly.regulators.with(|regs| regs[regulator_key].clone()); + let regulator = assembly.regulators.with(|regs| regs[regulator_key]); let other_subject = if regulator.subjects.0 == element_key { regulator.subjects.1 } else { regulator.subjects.0 }; let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); - let class = regulator.role.map( - |role| match role { - Measurement => "regulator", - Constraint(true) => "regulator valid-constraint", - Constraint(false) => "regulator invalid-constraint" + let class = create_memo(move || { + match regulator.set_point.get() { + None if regulator.has_no_set_point_spec() => "regulator", + None => "regulator invalid-constraint", + Some(_) => "regulator valid-constraint" } - ); + }); view! { li(class=class.get()) { div(class="regulator-label") { (other_subject_label) } -- 2.43.0 From 302d93638d36ccb6a7b75aba1f9d3c06e7c6f81d Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Tue, 18 Feb 2025 01:27:11 -0800 Subject: [PATCH 09/25] Require regulators to have valid specifications When an invalid specification is entered into a regulator input, keep it confined to that input. Reset a regulator input by pressing *escape*. --- app-proto/main.css | 15 +++++------ app-proto/src/assembly.rs | 7 +----- app-proto/src/outline.rs | 53 +++++++++++++++++++++++++++++++-------- 3 files changed, 49 insertions(+), 26 deletions(-) diff --git a/app-proto/main.css b/app-proto/main.css index d71ce4a..7ba1206 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -133,28 +133,25 @@ details[open]:has(li) .element-switch::after { font-size: 10pt; } -.regulator.invalid-constraint { - color: var(--text-invalid); -} - -.regulator > input { +.regulator-input { color: inherit; background-color: inherit; border: 1px solid var(--border); border-radius: 2px; } -.regulator > input::placeholder { +.regulator-input::placeholder { color: inherit; opacity: 54%; font-style: italic; } -.regulator.valid-constraint > input { +.regulator-input.constraint { background-color: var(--display-background); } -.regulator.invalid-constraint > input { +.regulator-input.invalid { + color: var(--text-invalid); border-color: var(--border-invalid); } @@ -166,7 +163,7 @@ details[open]:has(li) .element-switch::after { font-style: normal; } -.invalid-constraint > .status::after, details:has(.invalid-constraint):not([open]) .status::after { +.regulator:has(.invalid) > .status::after, details:has(.invalid):not([open]) .status::after { content: '⚠'; color: var(--text-invalid); } diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index c1eba1e..e8a8dc0 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -111,6 +111,7 @@ impl Element { } } +// `set_point_spec` must always be a valid specification of `set_point` #[derive(Clone, Copy)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), @@ -119,12 +120,6 @@ pub struct Regulator { pub set_point_spec: Signal } -impl Regulator { - pub fn has_no_set_point_spec(&self) -> bool { - self.set_point_spec.with(|spec| spec.is_empty()) - } -} - // the velocity is expressed in uniform coordinates pub struct ElementMotion<'a> { pub key: ElementKey, diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index d892f0f..b0347f6 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -19,17 +19,55 @@ use crate::{ // an editable view of a regulator #[component(inline_props)] fn RegulatorInput(regulator: Regulator) -> View { + let valid = create_signal(true); let value = create_signal(regulator.set_point_spec.get_clone_untracked()); - create_effect(move || value.set(regulator.set_point_spec.get_clone())); + + // this closure resets the input value to the regulator's set point + // specification, which is always a valid specification + let reset_value = move || { + batch(|| { + valid.set(true); + value.set(regulator.set_point_spec.get_clone()); + }) + }; + + // reset the input value whenever the regulator's set point specification + // is updated + create_effect(reset_value); + view! { input( r#type="text", + class=move || { + if valid.get() { + match regulator.set_point.get() { + Some(_) => "regulator-input constraint", + None => "regulator-input" + } + } else { + "regulator-input invalid" + } + }, placeholder=regulator.measurement.with(|result| result.to_string()), bind:value=value, on:change=move |_| { let value_val = value.get_clone_untracked(); - regulator.set_point.set(value_val.parse::().ok()); - regulator.set_point_spec.set(value_val); + match value_val.parse::() { + Err(_) if !value_val.is_empty() => valid.set(false), + set_pt => batch(|| { + regulator.set_point.set(set_pt.ok()); + regulator.set_point_spec.set(value_val); + valid.set(true); + }) + }; + }, + on:keydown={ + move |event: KeyboardEvent| { + match event.key().as_str() { + "Escape" => reset_value(), + _ => () + } + } } ) } @@ -47,15 +85,8 @@ fn RegulatorOutlineItem(regulator_key: RegulatorKey, element_key: ElementKey) -> regulator.subjects.0 }; let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); - let class = create_memo(move || { - match regulator.set_point.get() { - None if regulator.has_no_set_point_spec() => "regulator", - None => "regulator invalid-constraint", - Some(_) => "regulator valid-constraint" - } - }); view! { - li(class=class.get()) { + li(class="regulator") { div(class="regulator-label") { (other_subject_label) } div(class="regulator-type") { "Inversive distance" } RegulatorInput(regulator=regulator) -- 2.43.0 From bbd0835a8fee603a28de61054aa9637c74fdc08c Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Tue, 18 Feb 2025 12:28:55 -0800 Subject: [PATCH 10/25] Move set point spec validation into `Regulator` --- app-proto/src/assembly.rs | 13 +++++++++++++ app-proto/src/outline.rs | 14 +++----------- 2 files changed, 16 insertions(+), 11 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index e8a8dc0..20b893d 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -120,6 +120,19 @@ pub struct Regulator { pub set_point_spec: Signal } +impl Regulator { + pub fn try_specify_set_point(&self, spec: String) -> bool { + match spec.parse::() { + Err(_) if !spec.is_empty() => false, + set_pt => { + self.set_point.set(set_pt.ok()); + self.set_point_spec.set(spec); + true + } + } + } +} + // the velocity is expressed in uniform coordinates pub struct ElementMotion<'a> { pub key: ElementKey, diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index b0347f6..a48377b 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -50,17 +50,9 @@ fn RegulatorInput(regulator: Regulator) -> View { }, placeholder=regulator.measurement.with(|result| result.to_string()), bind:value=value, - on:change=move |_| { - let value_val = value.get_clone_untracked(); - match value_val.parse::() { - Err(_) if !value_val.is_empty() => valid.set(false), - set_pt => batch(|| { - regulator.set_point.set(set_pt.ok()); - regulator.set_point_spec.set(value_val); - valid.set(true); - }) - }; - }, + on:change=move |_| valid.set( + regulator.try_specify_set_point(value.get_clone_untracked()) + ), on:keydown={ move |event: KeyboardEvent| { match event.key().as_str() { -- 2.43.0 From f2e84fb64a87356e7ad43927dc3d2b7bd46bcd7b Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Tue, 18 Feb 2025 13:29:10 -0800 Subject: [PATCH 11/25] Enforce the validity of set point specifications Make a regulator's set point specification private, and split the set point into a private writable signal and a public read-only signal. The set point can now be initialized only through the factory method `insert_new_regulator` and changed only through the setter method `try_specify_set_point`, which both ensure that the set point specification is valid and consistent with the set point. --- app-proto/src/add_remove.rs | 47 +---------------------- app-proto/src/assembly.rs | 75 ++++++++++++++++++++++++++++++++++--- app-proto/src/outline.rs | 4 +- 3 files changed, 74 insertions(+), 52 deletions(-) diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index c96d5ab..cdc35e6 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -6,10 +6,8 @@ use crate::{ AppState, assembly::{ Assembly, - Regulator, Element - }, - engine::Q + } }; /* DEBUG */ @@ -199,49 +197,8 @@ pub fn AddRemove() -> View { (subject_vec[0].clone(), subject_vec[1].clone()) } ); - let measurement = state.assembly.elements.map( - move |elts| { - let reps = ( - elts[subjects.0].representation.get_clone(), - elts[subjects.1].representation.get_clone() - ); - reps.0.dot(&(&*Q * reps.1)) - } - ); - let set_point = create_signal(None); - state.assembly.insert_regulator(Regulator { - subjects: subjects, - measurement: measurement, - set_point: set_point, - set_point_spec: create_signal(String::new()) - }); + state.assembly.insert_new_regulator(subjects); state.selection.update(|sel| sel.clear()); - - /* DEBUG */ - // print updated regulator list - console::log_1(&JsValue::from("Regulators:")); - state.assembly.regulators.with(|regs| { - for (_, reg) in regs.into_iter() { - console::log_5( - &JsValue::from(" "), - &JsValue::from(reg.subjects.0), - &JsValue::from(reg.subjects.1), - &JsValue::from(":"), - &JsValue::from(reg.set_point.get_untracked()) - ); - } - }); - - // update the realization when the regulator becomes - // a constraint, or is edited while acting as a constraint - create_effect(move || { - console::log_1(&JsValue::from( - format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) - )); - if set_point.with(|set_pt| set_pt.is_some()) { - state.assembly.realize(); - } - }); } ) { "🔗" } select(bind:value=assembly_name) { /* DEBUG */ // example assembly chooser diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 20b893d..21421b6 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -5,7 +5,7 @@ use std::{collections::BTreeSet, sync::atomic::{AtomicU64, Ordering}}; use sycamore::prelude::*; use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ -use crate::engine::{realize_gram, local_unif_to_std, ConfigSubspace, PartialMatrix}; +use crate::engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix}; // the types of the keys we use to access an assembly's elements and regulators pub type ElementKey = usize; @@ -111,21 +111,38 @@ impl Element { } } -// `set_point_spec` must always be a valid specification of `set_point` +// `set_point_spec` is always a valid specification of `set_point` +// ┌────────────┬─────────────────────────────────────────────────────┐ +// │`set_point` │ `set_point_spec` │ +// ┝━━━━━━━━━━━━┿━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┥ +// │`Some(x)` │ a string that parses to the floating-point value `x`│ +// ├────────────┼─────────────────────────────────────────────────────┤ +// │`None` │ the empty string │ +// └────────────┴─────────────────────────────────────────────────────┘ #[derive(Clone, Copy)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), pub measurement: ReadSignal, - pub set_point: Signal>, - pub set_point_spec: Signal + pub set_point: ReadSignal>, + + set_point_writable: Signal>, + set_point_spec: Signal } impl Regulator { + pub fn get_set_point_spec_clone(&self) -> String { + self.set_point_spec.get_clone() + } + + pub fn get_set_point_spec_clone_untracked(&self) -> String { + self.set_point_spec.get_clone_untracked() + } + pub fn try_specify_set_point(&self, spec: String) -> bool { match spec.parse::() { Err(_) if !spec.is_empty() => false, set_pt => { - self.set_point.set(set_pt.ok()); + self.set_point_writable.set(set_pt.ok()); self.set_point_spec.set(spec); true } @@ -227,6 +244,54 @@ impl Assembly { subject_regulators.1.update(|regs| regs.insert(key)); } + pub fn insert_new_regulator(self, subjects: (ElementKey, ElementKey)) { + // create and insert a new regulator + let measurement = self.elements.map( + move |elts| { + let reps = ( + elts[subjects.0].representation.get_clone(), + elts[subjects.1].representation.get_clone() + ); + reps.0.dot(&(&*Q * reps.1)) + } + ); + let set_point_writable = create_signal(None); + let set_point = set_point_writable.split().0; + self.insert_regulator(Regulator { + subjects: subjects, + measurement: measurement, + set_point: set_point, + set_point_writable: set_point_writable, + set_point_spec: create_signal(String::new()) + }); + + /* DEBUG */ + // print updated regulator list + console::log_1(&JsValue::from("Regulators:")); + self.regulators.with(|regs| { + for (_, reg) in regs.into_iter() { + console::log_5( + &JsValue::from(" "), + &JsValue::from(reg.subjects.0), + &JsValue::from(reg.subjects.1), + &JsValue::from(":"), + &JsValue::from(reg.set_point.get_untracked()) + ); + } + }); + + // update the realization when the regulator becomes a constraint, or is + // edited while acting as a constraint + create_effect(move || { + console::log_1(&JsValue::from( + format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) + )); + if set_point.with(|set_pt| set_pt.is_some()) { + self.realize(); + } + }); + } + // --- realization --- pub fn realize(&self) { diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index a48377b..4898aba 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -20,14 +20,14 @@ use crate::{ #[component(inline_props)] fn RegulatorInput(regulator: Regulator) -> View { let valid = create_signal(true); - let value = create_signal(regulator.set_point_spec.get_clone_untracked()); + let value = create_signal(regulator.get_set_point_spec_clone_untracked()); // this closure resets the input value to the regulator's set point // specification, which is always a valid specification let reset_value = move || { batch(|| { valid.set(true); - value.set(regulator.set_point_spec.get_clone()); + value.set(regulator.get_set_point_spec_clone()); }) }; -- 2.43.0 From c54b6bc16599124f08686d52b3193422f140e2c9 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Tue, 18 Feb 2025 13:37:30 -0800 Subject: [PATCH 12/25] Tie invalidity indicator to regulator input Invalid attempts to specify a regulator's set point are now local to each view of the regulator, and don't affect the regulator model. In particular, a regulator will be valid and in force even when one of its regulator input views is showing an invalid specification attempt. The invalidity indicator should therefore be tied to the input showing the invalid specification, not to the whole regulator outline item. --- app-proto/main.css | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app-proto/main.css b/app-proto/main.css index 7ba1206..4726a27 100644 --- a/app-proto/main.css +++ b/app-proto/main.css @@ -163,7 +163,7 @@ details[open]:has(li) .element-switch::after { font-style: normal; } -.regulator:has(.invalid) > .status::after, details:has(.invalid):not([open]) .status::after { +.regulator-input.invalid + .status::after, details:has(.invalid):not([open]) .status::after { content: '⚠'; color: var(--text-invalid); } -- 2.43.0 From befadd25c9d35c6a9e782d9ada981002ebc91fc4 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 19 Feb 2025 01:22:42 -0800 Subject: [PATCH 13/25] Get the read-only set point signal more simply --- app-proto/src/assembly.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 21421b6..8403406 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -256,7 +256,7 @@ impl Assembly { } ); let set_point_writable = create_signal(None); - let set_point = set_point_writable.split().0; + let set_point = *set_point_writable; self.insert_regulator(Regulator { subjects: subjects, measurement: measurement, -- 2.43.0 From 6c31a25822ca9d5b8dc3a50dc7ffac4e94f19afc Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sun, 23 Feb 2025 23:51:24 -0800 Subject: [PATCH 14/25] Consolidate set point data --- app-proto/src/assembly.rs | 127 +++++++++++++++++++++++++------------- app-proto/src/outline.rs | 23 ++++--- 2 files changed, 97 insertions(+), 53 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 8403406..2ccfc69 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -1,7 +1,11 @@ use nalgebra::{DMatrix, DVector, DVectorView, Vector3}; use rustc_hash::FxHashMap; use slab::Slab; -use std::{collections::BTreeSet, sync::atomic::{AtomicU64, Ordering}}; +use std::{ + collections::BTreeSet, + num::ParseFloatError, + sync::atomic::{AtomicU64, Ordering} +}; use sycamore::prelude::*; use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ @@ -111,41 +115,75 @@ impl Element { } } -// `set_point_spec` is always a valid specification of `set_point` -// ┌────────────┬─────────────────────────────────────────────────────┐ -// │`set_point` │ `set_point_spec` │ -// ┝━━━━━━━━━━━━┿━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┥ -// │`Some(x)` │ a string that parses to the floating-point value `x`│ -// ├────────────┼─────────────────────────────────────────────────────┤ -// │`None` │ the empty string │ -// └────────────┴─────────────────────────────────────────────────────┘ +// to construct a `SpecifiedValue` that might be `Present`, use the associated +// function `try_from`. this ensures that `spec` is always a valid specification +// of `value` according to the format discussed above the implementation of +// `TryFrom` +pub enum SpecifiedValue { + Absent, + Present { + spec: String, + value: f64 + } +} + +use SpecifiedValue::*; + +impl SpecifiedValue { + // get the specification for this value. the associated function `try_from` + // is essentially a left inverse of this method: + // + // SpecifiedValue::try_from(x.spec()) == Ok(x) + // + pub fn spec(&self) -> String { + match self { + Absent => String::new(), + Present { spec, .. } => spec.clone() + } + } + + fn is_present(&self) -> bool { + match self { + Absent => false, + Present { .. } => true + } + } +} + +// we can try to turn a specification string into a `SpecifiedValue`. if the +// specification is empty, the `SpecifiedValue` is `Absent`. if the +// specification parses to a floating-point value `x`, the `SpecifiedValue` is +// `Present`, with a `value` of `x`, and the specification is stored in `spec`. +// these are the only valid specifications; any other produces an error +impl TryFrom for SpecifiedValue { + type Error = ParseFloatError; + + fn try_from(spec: String) -> Result { + if spec.is_empty() { + Ok(Absent) + } else { + spec.parse::().map( + |value| Present { spec: spec, value: value } + ) + } + } +} + #[derive(Clone, Copy)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), pub measurement: ReadSignal, - pub set_point: ReadSignal>, - - set_point_writable: Signal>, - set_point_spec: Signal + pub set_point: Signal } impl Regulator { - pub fn get_set_point_spec_clone(&self) -> String { - self.set_point_spec.get_clone() - } - - pub fn get_set_point_spec_clone_untracked(&self) -> String { - self.set_point_spec.get_clone_untracked() - } - - pub fn try_specify_set_point(&self, spec: String) -> bool { - match spec.parse::() { - Err(_) if !spec.is_empty() => false, - set_pt => { - self.set_point_writable.set(set_pt.ok()); - self.set_point_spec.set(spec); + pub fn try_set(&self, set_pt_spec: String) -> bool { + match SpecifiedValue::try_from(set_pt_spec) { + Ok(set_pt) => { + self.set_point.set(set_pt); true } + Err(_) => false, } } } @@ -255,18 +293,15 @@ impl Assembly { reps.0.dot(&(&*Q * reps.1)) } ); - let set_point_writable = create_signal(None); - let set_point = *set_point_writable; + let set_point = create_signal(Absent); self.insert_regulator(Regulator { subjects: subjects, measurement: measurement, - set_point: set_point, - set_point_writable: set_point_writable, - set_point_spec: create_signal(String::new()) + set_point: set_point }); /* DEBUG */ - // print updated regulator list + // print an updated list of regulators console::log_1(&JsValue::from("Regulators:")); self.regulators.with(|regs| { for (_, reg) in regs.into_iter() { @@ -275,7 +310,9 @@ impl Assembly { &JsValue::from(reg.subjects.0), &JsValue::from(reg.subjects.1), &JsValue::from(":"), - &JsValue::from(reg.set_point.get_untracked()) + &JsValue::from(reg.set_point.with_untracked( + |set_pt| set_pt.spec() + )) ); } }); @@ -286,7 +323,7 @@ impl Assembly { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); - if set_point.with(|set_pt| set_pt.is_some()) { + if set_point.with(|set_pt| set_pt.is_present()) { self.realize(); } }); @@ -308,15 +345,17 @@ impl Assembly { let mut gram_to_be = PartialMatrix::new(); self.regulators.with_untracked(|regs| { for (_, reg) in regs { - match reg.set_point.get_untracked() { - Some(set_pt) => { - let subjects = reg.subjects; - let row = elts[subjects.0].column_index.unwrap(); - let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, set_pt); - }, - None => () - } + reg.set_point.with_untracked(|set_pt| { + match set_pt { + Absent => (), + Present { value, .. } => { + let subjects = reg.subjects; + let row = elts[subjects.0].column_index.unwrap(); + let col = elts[subjects.1].column_index.unwrap(); + gram_to_be.push_sym(row, col, *value); + } + }; + }); } }); diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 4898aba..04bda9a 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -10,9 +10,10 @@ use crate::{ AppState, assembly, assembly::{ + ElementKey, Regulator, RegulatorKey, - ElementKey + SpecifiedValue::* } }; @@ -20,14 +21,16 @@ use crate::{ #[component(inline_props)] fn RegulatorInput(regulator: Regulator) -> View { let valid = create_signal(true); - let value = create_signal(regulator.get_set_point_spec_clone_untracked()); + let value = create_signal( + regulator.set_point.with_untracked(|set_pt| set_pt.spec()) + ); // this closure resets the input value to the regulator's set point - // specification, which is always a valid specification + // specification let reset_value = move || { batch(|| { valid.set(true); - value.set(regulator.get_set_point_spec_clone()); + value.set(regulator.set_point.with(|set_pt| set_pt.spec())); }) }; @@ -40,10 +43,12 @@ fn RegulatorInput(regulator: Regulator) -> View { r#type="text", class=move || { if valid.get() { - match regulator.set_point.get() { - Some(_) => "regulator-input constraint", - None => "regulator-input" - } + regulator.set_point.with(|set_pt| { + match set_pt { + Absent => "regulator-input", + Present { .. } => "regulator-input constraint" + } + }) } else { "regulator-input invalid" } @@ -51,7 +56,7 @@ fn RegulatorInput(regulator: Regulator) -> View { placeholder=regulator.measurement.with(|result| result.to_string()), bind:value=value, on:change=move |_| valid.set( - regulator.try_specify_set_point(value.get_clone_untracked()) + regulator.try_set(value.get_clone_untracked()) ), on:keydown={ move |event: KeyboardEvent| { -- 2.43.0 From c368a38803a735ff915e92609de28732f2e0348d Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 26 Feb 2025 22:22:58 -0800 Subject: [PATCH 15/25] Make `insert_regulator` private --- app-proto/src/assembly.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 2ccfc69..739dfc1 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -272,7 +272,7 @@ impl Assembly { ); } - pub fn insert_regulator(&self, regulator: Regulator) { + fn insert_regulator(&self, regulator: Regulator) { let subjects = regulator.subjects; let key = self.regulators.update(|regs| regs.insert(regulator)); let subject_regulators = self.elements.with( -- 2.43.0 From 7cbd92618b867fcf6d52cb755d454a492768c349 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Thu, 27 Feb 2025 00:26:54 -0800 Subject: [PATCH 16/25] Improve the comments in the `assembly` module Also, remove trailing comma. --- app-proto/src/assembly.rs | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 739dfc1..50b0783 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -30,8 +30,11 @@ pub struct Element { pub label: String, pub color: ElementColor, pub representation: Signal>, + + // the regulators that affect this element. the ambient assembly is + // responsible for keeping this set up to date pub regulators: Signal>, - + // a serial number, assigned by `Element::new`, that uniquely identifies // each element pub serial: u64, @@ -117,7 +120,7 @@ impl Element { // to construct a `SpecifiedValue` that might be `Present`, use the associated // function `try_from`. this ensures that `spec` is always a valid specification -// of `value` according to the format discussed above the implementation of +// of `value` according to the format discussed at the implementation of // `TryFrom` pub enum SpecifiedValue { Absent, @@ -154,7 +157,8 @@ impl SpecifiedValue { // specification is empty, the `SpecifiedValue` is `Absent`. if the // specification parses to a floating-point value `x`, the `SpecifiedValue` is // `Present`, with a `value` of `x`, and the specification is stored in `spec`. -// these are the only valid specifications; any other produces an error +// these are currently the only valid specifications; any other produces an +// error impl TryFrom for SpecifiedValue { type Error = ParseFloatError; @@ -183,7 +187,7 @@ impl Regulator { self.set_point.set(set_pt); true } - Err(_) => false, + Err(_) => false } } } -- 2.43.0 From 874c823dbef83542228783361789c82740a205b0 Mon Sep 17 00:00:00 2001 From: glen Date: Sat, 1 Mar 2025 01:27:45 +0000 Subject: [PATCH 17/25] Minor wording changes in description of Element.regulators. --- app-proto/src/assembly.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 50b0783..d30a371 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -31,8 +31,8 @@ pub struct Element { pub color: ElementColor, pub representation: Signal>, - // the regulators that affect this element. the ambient assembly is - // responsible for keeping this set up to date + // All regulators with this element as a subject. The assembly owning + // this element is responsible for keeping this set up to date. pub regulators: Signal>, // a serial number, assigned by `Element::new`, that uniquely identifies -- 2.43.0 From 894931a6e77815b5f2aabd6ba8bb1055d4e9ba52 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 3 Mar 2025 11:43:43 -0800 Subject: [PATCH 18/25] Replace `try_set` with `set_if_ok` --- app-proto/src/assembly.rs | 12 +++++------- app-proto/src/outline.rs | 9 ++++++--- 2 files changed, 11 insertions(+), 10 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index d30a371..9b123f5 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -181,13 +181,11 @@ pub struct Regulator { } impl Regulator { - pub fn try_set(&self, set_pt_spec: String) -> bool { - match SpecifiedValue::try_from(set_pt_spec) { - Ok(set_pt) => { - self.set_point.set(set_pt); - true - } - Err(_) => false + /* TO DO */ + // if it's called for, add a `set` method that takes a bare SpecifiedValue + pub fn set_if_ok(&self, set_pt_result: Result) { + if let Ok(set_pt) = set_pt_result { + self.set_point.set(set_pt); } } } diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 04bda9a..8f81a82 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -13,6 +13,7 @@ use crate::{ ElementKey, Regulator, RegulatorKey, + SpecifiedValue, SpecifiedValue::* } }; @@ -55,9 +56,11 @@ fn RegulatorInput(regulator: Regulator) -> View { }, placeholder=regulator.measurement.with(|result| result.to_string()), bind:value=value, - on:change=move |_| valid.set( - regulator.try_set(value.get_clone_untracked()) - ), + on:change=move |_| { + let set_pt_result = SpecifiedValue::try_from(value.get_clone_untracked()); + valid.set(set_pt_result.is_ok()); + regulator.set_if_ok(set_pt_result); + }, on:keydown={ move |event: KeyboardEvent| { match event.key().as_str() { -- 2.43.0 From 309b0881dfdc63e73b10db0641b272b2350b6cb3 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 3 Mar 2025 12:22:43 -0800 Subject: [PATCH 19/25] Move `SpecifiedValue` into its own module --- app-proto/src/assembly.rs | 66 +++----------------------------------- app-proto/src/main.rs | 1 + app-proto/src/outline.rs | 7 ++-- app-proto/src/specified.rs | 56 ++++++++++++++++++++++++++++++++ 4 files changed, 65 insertions(+), 65 deletions(-) create mode 100644 app-proto/src/specified.rs diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 9b123f5..01dee9d 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -1,15 +1,14 @@ use nalgebra::{DMatrix, DVector, DVectorView, Vector3}; use rustc_hash::FxHashMap; use slab::Slab; -use std::{ - collections::BTreeSet, - num::ParseFloatError, - sync::atomic::{AtomicU64, Ordering} -}; +use std::{collections::BTreeSet, sync::atomic::{AtomicU64, Ordering}}; use sycamore::prelude::*; use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ -use crate::engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix}; +use crate::{ + engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix}, + specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}} +}; // the types of the keys we use to access an assembly's elements and regulators pub type ElementKey = usize; @@ -118,61 +117,6 @@ impl Element { } } -// to construct a `SpecifiedValue` that might be `Present`, use the associated -// function `try_from`. this ensures that `spec` is always a valid specification -// of `value` according to the format discussed at the implementation of -// `TryFrom` -pub enum SpecifiedValue { - Absent, - Present { - spec: String, - value: f64 - } -} - -use SpecifiedValue::*; - -impl SpecifiedValue { - // get the specification for this value. the associated function `try_from` - // is essentially a left inverse of this method: - // - // SpecifiedValue::try_from(x.spec()) == Ok(x) - // - pub fn spec(&self) -> String { - match self { - Absent => String::new(), - Present { spec, .. } => spec.clone() - } - } - - fn is_present(&self) -> bool { - match self { - Absent => false, - Present { .. } => true - } - } -} - -// we can try to turn a specification string into a `SpecifiedValue`. if the -// specification is empty, the `SpecifiedValue` is `Absent`. if the -// specification parses to a floating-point value `x`, the `SpecifiedValue` is -// `Present`, with a `value` of `x`, and the specification is stored in `spec`. -// these are currently the only valid specifications; any other produces an -// error -impl TryFrom for SpecifiedValue { - type Error = ParseFloatError; - - fn try_from(spec: String) -> Result { - if spec.is_empty() { - Ok(Absent) - } else { - spec.parse::().map( - |value| Present { spec: spec, value: value } - ) - } - } -} - #[derive(Clone, Copy)] pub struct Regulator { pub subjects: (ElementKey, ElementKey), diff --git a/app-proto/src/main.rs b/app-proto/src/main.rs index f961504..6ab3e49 100644 --- a/app-proto/src/main.rs +++ b/app-proto/src/main.rs @@ -3,6 +3,7 @@ mod assembly; mod display; mod engine; mod outline; +mod specified; use rustc_hash::FxHashSet; use sycamore::prelude::*; diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 8f81a82..31837bb 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -12,10 +12,9 @@ use crate::{ assembly::{ ElementKey, Regulator, - RegulatorKey, - SpecifiedValue, - SpecifiedValue::* - } + RegulatorKey + }, + specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}} }; // an editable view of a regulator diff --git a/app-proto/src/specified.rs b/app-proto/src/specified.rs new file mode 100644 index 0000000..ea873e9 --- /dev/null +++ b/app-proto/src/specified.rs @@ -0,0 +1,56 @@ +use std::num::ParseFloatError; + +// to construct a `SpecifiedValue` that might be `Present`, use the associated +// function `try_from`. this ensures that `spec` is always a valid specification +// of `value` according to the format discussed at the implementation of +// `TryFrom` +pub enum SpecifiedValue { + Absent, + Present { + spec: String, + value: f64 + } +} + +use SpecifiedValue::{Absent, Present}; + +impl SpecifiedValue { + // get the specification for this value. the associated function `try_from` + // is essentially a left inverse of this method: + // + // SpecifiedValue::try_from(x.spec()) == Ok(x) + // + pub fn spec(&self) -> String { + match self { + Absent => String::new(), + Present { spec, .. } => spec.clone() + } + } + + pub fn is_present(&self) -> bool { + match self { + Absent => false, + Present { .. } => true + } + } +} + +// we can try to turn a specification string into a `SpecifiedValue`. if the +// specification is empty, the `SpecifiedValue` is `Absent`. if the +// specification parses to a floating-point value `x`, the `SpecifiedValue` is +// `Present`, with a `value` of `x`, and the specification is stored in `spec`. +// these are currently the only valid specifications; any other produces an +// error +impl TryFrom for SpecifiedValue { + type Error = ParseFloatError; + + fn try_from(spec: String) -> Result { + if spec.is_empty() { + Ok(Absent) + } else { + spec.parse::().map( + |value| Present { spec: spec, value: value } + ) + } + } +} \ No newline at end of file -- 2.43.0 From c58fed073db9748d4c1e4d81fdb757286c7e6a25 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 3 Mar 2025 12:31:30 -0800 Subject: [PATCH 20/25] Drop the `is_present` utility method --- app-proto/src/assembly.rs | 2 +- app-proto/src/specified.rs | 7 ------- 2 files changed, 1 insertion(+), 8 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 01dee9d..002b420 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -269,7 +269,7 @@ impl Assembly { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); - if set_point.with(|set_pt| set_pt.is_present()) { + if set_point.with(|set_pt| matches!(set_pt, Present { .. })) { self.realize(); } }); diff --git a/app-proto/src/specified.rs b/app-proto/src/specified.rs index ea873e9..a943f2b 100644 --- a/app-proto/src/specified.rs +++ b/app-proto/src/specified.rs @@ -26,13 +26,6 @@ impl SpecifiedValue { Present { spec, .. } => spec.clone() } } - - pub fn is_present(&self) -> bool { - match self { - Absent => false, - Present { .. } => true - } - } } // we can try to turn a specification string into a `SpecifiedValue`. if the -- 2.43.0 From 8b4a72c60c13955a8680a50c995cb288d36bf722 Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 3 Mar 2025 15:15:30 -0800 Subject: [PATCH 21/25] Simplify `match` to `if let` --- app-proto/src/assembly.rs | 15 ++++++--------- 1 file changed, 6 insertions(+), 9 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 002b420..f9b4620 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -292,15 +292,12 @@ impl Assembly { self.regulators.with_untracked(|regs| { for (_, reg) in regs { reg.set_point.with_untracked(|set_pt| { - match set_pt { - Absent => (), - Present { value, .. } => { - let subjects = reg.subjects; - let row = elts[subjects.0].column_index.unwrap(); - let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, *value); - } - }; + if let Present { value, .. } = set_pt { + let subjects = reg.subjects; + let row = elts[subjects.0].column_index.unwrap(); + let col = elts[subjects.1].column_index.unwrap(); + gram_to_be.push_sym(row, col, *value); + } }); } }); -- 2.43.0 From 84bfdefccbc090c6e5a758671a94416986cb621a Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Mon, 3 Mar 2025 23:10:28 -0800 Subject: [PATCH 22/25] Rewrite `SpecifiedValue` as a read-only structure --- app-proto/Cargo.toml | 1 + app-proto/src/assembly.rs | 16 +++++------ app-proto/src/outline.rs | 13 ++++----- app-proto/src/specified.rs | 55 +++++++++++++++++--------------------- 4 files changed, 41 insertions(+), 44 deletions(-) diff --git a/app-proto/Cargo.toml b/app-proto/Cargo.toml index c11fef4..8000327 100644 --- a/app-proto/Cargo.toml +++ b/app-proto/Cargo.toml @@ -13,6 +13,7 @@ itertools = "0.13.0" js-sys = "0.3.70" lazy_static = "1.5.0" nalgebra = "0.33.0" +readonly = "0.2.12" rustc-hash = "2.0.0" slab = "0.4.9" sycamore = "0.9.0-beta.3" diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index f9b4620..0881741 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -7,7 +7,7 @@ use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ use crate::{ engine::{Q, local_unif_to_std, realize_gram, ConfigSubspace, PartialMatrix}, - specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}} + specified::SpecifiedValue }; // the types of the keys we use to access an assembly's elements and regulators @@ -239,7 +239,7 @@ impl Assembly { reps.0.dot(&(&*Q * reps.1)) } ); - let set_point = create_signal(Absent); + let set_point = create_signal(SpecifiedValue::from_empty_spec()); self.insert_regulator(Regulator { subjects: subjects, measurement: measurement, @@ -256,9 +256,9 @@ impl Assembly { &JsValue::from(reg.subjects.0), &JsValue::from(reg.subjects.1), &JsValue::from(":"), - &JsValue::from(reg.set_point.with_untracked( - |set_pt| set_pt.spec() - )) + ®.set_point.with_untracked( + |set_pt| JsValue::from(set_pt.spec.as_str()) + ) ); } }); @@ -269,7 +269,7 @@ impl Assembly { console::log_1(&JsValue::from( format!("Updated constraint with subjects ({}, {})", subjects.0, subjects.1) )); - if set_point.with(|set_pt| matches!(set_pt, Present { .. })) { + if set_point.with(|set_pt| set_pt.is_present()) { self.realize(); } }); @@ -292,11 +292,11 @@ impl Assembly { self.regulators.with_untracked(|regs| { for (_, reg) in regs { reg.set_point.with_untracked(|set_pt| { - if let Present { value, .. } = set_pt { + if let Some(val) = set_pt.value { let subjects = reg.subjects; let row = elts[subjects.0].column_index.unwrap(); let col = elts[subjects.1].column_index.unwrap(); - gram_to_be.push_sym(row, col, *value); + gram_to_be.push_sym(row, col, val); } }); } diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 31837bb..bc920f3 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -14,7 +14,7 @@ use crate::{ Regulator, RegulatorKey }, - specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}} + specified::SpecifiedValue }; // an editable view of a regulator @@ -22,7 +22,7 @@ use crate::{ fn RegulatorInput(regulator: Regulator) -> View { let valid = create_signal(true); let value = create_signal( - regulator.set_point.with_untracked(|set_pt| set_pt.spec()) + regulator.set_point.with_untracked(|set_pt| set_pt.spec.clone()) ); // this closure resets the input value to the regulator's set point @@ -30,7 +30,7 @@ fn RegulatorInput(regulator: Regulator) -> View { let reset_value = move || { batch(|| { valid.set(true); - value.set(regulator.set_point.with(|set_pt| set_pt.spec())); + value.set(regulator.set_point.with(|set_pt| set_pt.spec.clone())); }) }; @@ -44,9 +44,10 @@ fn RegulatorInput(regulator: Regulator) -> View { class=move || { if valid.get() { regulator.set_point.with(|set_pt| { - match set_pt { - Absent => "regulator-input", - Present { .. } => "regulator-input constraint" + if set_pt.is_present() { + "regulator-input constraint" + } else { + "regulator-input" } }) } else { diff --git a/app-proto/src/specified.rs b/app-proto/src/specified.rs index a943f2b..0422291 100644 --- a/app-proto/src/specified.rs +++ b/app-proto/src/specified.rs @@ -1,48 +1,43 @@ use std::num::ParseFloatError; -// to construct a `SpecifiedValue` that might be `Present`, use the associated -// function `try_from`. this ensures that `spec` is always a valid specification -// of `value` according to the format discussed at the implementation of -// `TryFrom` -pub enum SpecifiedValue { - Absent, - Present { - spec: String, - value: f64 - } +// a real number described by a specification string. since the structure is +// read-only, we can guarantee that `spec` always specifies `value` in the +// following format +// ┌──────────────────────────────────────────────────────┬───────────┐ +// │ `spec` │ `value` │ +// ┝━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┿━━━━━━━━━━━┥ +// │ a string that parses to the floating-point value `x` │ `Some(x)` │ +// ├──────────────────────────────────────────────────────┼───────────┤ +// │ the empty string │ `None` │ +// └──────────────────────────────────────────────────────┴───────────┘ +#[readonly::make] +pub struct SpecifiedValue { + pub spec: String, + pub value: Option } -use SpecifiedValue::{Absent, Present}; - impl SpecifiedValue { - // get the specification for this value. the associated function `try_from` - // is essentially a left inverse of this method: - // - // SpecifiedValue::try_from(x.spec()) == Ok(x) - // - pub fn spec(&self) -> String { - match self { - Absent => String::new(), - Present { spec, .. } => spec.clone() - } + pub fn from_empty_spec() -> SpecifiedValue { + SpecifiedValue { spec: String::new(), value: None } + } + + pub fn is_present(&self) -> bool { + matches!(self.value, Some(_)) } } -// we can try to turn a specification string into a `SpecifiedValue`. if the -// specification is empty, the `SpecifiedValue` is `Absent`. if the -// specification parses to a floating-point value `x`, the `SpecifiedValue` is -// `Present`, with a `value` of `x`, and the specification is stored in `spec`. -// these are currently the only valid specifications; any other produces an -// error +// a `SpecifiedValue` can be constructed from a specification string, formatted +// as described in the comment on the structure definition. the result is `Ok` +// if the specification is properly formatted, and `Error` if not impl TryFrom for SpecifiedValue { type Error = ParseFloatError; fn try_from(spec: String) -> Result { if spec.is_empty() { - Ok(Absent) + Ok(SpecifiedValue { spec: spec, value: None }) } else { spec.parse::().map( - |value| Present { spec: spec, value: value } + |value| SpecifiedValue { spec: spec, value: Some(value) } ) } } -- 2.43.0 From 6eeeb1c6fddc2c4122ec2eaf9146a0c839bf7def Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Wed, 5 Mar 2025 15:05:00 -0800 Subject: [PATCH 23/25] Bring back `try_set` as inline code This essentially reverts commit 894931a and then inlines `try_set` in the one place where it's used. If we ever want to reuse that code, we'll factor it out again. --- app-proto/src/assembly.rs | 10 ---------- app-proto/src/outline.rs | 12 +++++++++--- 2 files changed, 9 insertions(+), 13 deletions(-) diff --git a/app-proto/src/assembly.rs b/app-proto/src/assembly.rs index 0881741..18176df 100644 --- a/app-proto/src/assembly.rs +++ b/app-proto/src/assembly.rs @@ -124,16 +124,6 @@ pub struct Regulator { pub set_point: Signal } -impl Regulator { - /* TO DO */ - // if it's called for, add a `set` method that takes a bare SpecifiedValue - pub fn set_if_ok(&self, set_pt_result: Result) { - if let Ok(set_pt) = set_pt_result { - self.set_point.set(set_pt); - } - } -} - // the velocity is expressed in uniform coordinates pub struct ElementMotion<'a> { pub key: ElementKey, diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index bc920f3..8e40140 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -57,9 +57,15 @@ fn RegulatorInput(regulator: Regulator) -> View { placeholder=regulator.measurement.with(|result| result.to_string()), bind:value=value, on:change=move |_| { - let set_pt_result = SpecifiedValue::try_from(value.get_clone_untracked()); - valid.set(set_pt_result.is_ok()); - regulator.set_if_ok(set_pt_result); + valid.set( + match SpecifiedValue::try_from(value.get_clone_untracked()) { + Ok(set_pt) => { + regulator.set_point.set(set_pt); + true + } + Err(_) => false + } + ) }, on:keydown={ move |event: KeyboardEvent| { -- 2.43.0 From b9db7a569987cf9333f588fcdff2759d62da5bcc Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sat, 8 Mar 2025 15:02:57 -0800 Subject: [PATCH 24/25] Make `use` declarations more compact --- app-proto/src/add_remove.rs | 5 +---- app-proto/src/outline.rs | 6 +----- 2 files changed, 2 insertions(+), 9 deletions(-) diff --git a/app-proto/src/add_remove.rs b/app-proto/src/add_remove.rs index cdc35e6..5fed411 100644 --- a/app-proto/src/add_remove.rs +++ b/app-proto/src/add_remove.rs @@ -4,10 +4,7 @@ use web_sys::{console, wasm_bindgen::JsValue}; use crate::{ engine, AppState, - assembly::{ - Assembly, - Element - } + assembly::{Assembly, Element} }; /* DEBUG */ diff --git a/app-proto/src/outline.rs b/app-proto/src/outline.rs index 8e40140..002baea 100644 --- a/app-proto/src/outline.rs +++ b/app-proto/src/outline.rs @@ -9,11 +9,7 @@ use web_sys::{ use crate::{ AppState, assembly, - assembly::{ - ElementKey, - Regulator, - RegulatorKey - }, + assembly::{ElementKey, Regulator, RegulatorKey}, specified::SpecifiedValue }; -- 2.43.0 From 08ec838334a5fc1bfe7961ba7047c3b60c8e70fb Mon Sep 17 00:00:00 2001 From: Aaron Fenyes Date: Sat, 8 Mar 2025 15:11:37 -0800 Subject: [PATCH 25/25] Consolidate constructions from empty specification To reduce the potential for inconsistency, we should only have one piece of code that constructs a `SpecifiedValue` from the empty specification. --- app-proto/src/specified.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app-proto/src/specified.rs b/app-proto/src/specified.rs index 0422291..cfe7fc3 100644 --- a/app-proto/src/specified.rs +++ b/app-proto/src/specified.rs @@ -34,7 +34,7 @@ impl TryFrom for SpecifiedValue { fn try_from(spec: String) -> Result { if spec.is_empty() { - Ok(SpecifiedValue { spec: spec, value: None }) + Ok(SpecifiedValue::from_empty_spec()) } else { spec.parse::().map( |value| SpecifiedValue { spec: spec, value: Some(value) } -- 2.43.0