Introduce a regulator trait

This will provide a common interface for Lorentz product regulators,
curvature regulators, and hopefully all the other regulators too.
This commit is contained in:
Aaron Fenyes 2025-03-27 00:29:27 -07:00
parent f1f87e97be
commit 25f446499b
3 changed files with 75 additions and 35 deletions

View file

@ -199,7 +199,7 @@ pub fn AddRemove() -> View {
.try_into() .try_into()
.unwrap() .unwrap()
); );
state.assembly.insert_new_regulator(subjects); state.assembly.insert_new_product_regulator(subjects);
state.selection.update(|sel| sel.clear()); state.selection.update(|sel| sel.clear());
} }
) { "🔗" } ) { "🔗" }

View file

@ -1,7 +1,7 @@
use nalgebra::{DMatrix, DVector, DVectorView, Vector3}; use nalgebra::{DMatrix, DVector, DVectorView, Vector3};
use rustc_hash::FxHashMap; use rustc_hash::FxHashMap;
use slab::Slab; use slab::Slab;
use std::{collections::BTreeSet, sync::atomic::{AtomicU64, Ordering}}; use std::{collections::BTreeSet, rc::Rc, sync::atomic::{AtomicU64, Ordering}};
use sycamore::prelude::*; use sycamore::prelude::*;
use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */ use web_sys::{console, wasm_bindgen::JsValue}; /* DEBUG */
@ -132,14 +132,35 @@ impl Element {
} }
} }
#[derive(Clone, Copy)] pub trait Regulator {
// get information
fn subjects(&self) -> Vec<ElementKey>;
fn measurement(&self) -> ReadSignal<f64>;
fn set_point(&self) -> Signal<SpecifiedValue>;
// write problem data
fn write_to_problem(&self, problem: &mut ConstraintProblem, elts: &Slab<Element>);
}
pub struct ProductRegulator { pub struct ProductRegulator {
pub subjects: [ElementKey; 2], pub subjects: [ElementKey; 2],
pub measurement: ReadSignal<f64>, pub measurement: ReadSignal<f64>,
pub set_point: Signal<SpecifiedValue> pub set_point: Signal<SpecifiedValue>
} }
impl ProductRegulator { impl Regulator for ProductRegulator {
fn subjects(&self) -> Vec<ElementKey> {
self.subjects.into()
}
fn measurement(&self) -> ReadSignal<f64> {
self.measurement
}
fn set_point(&self) -> Signal<SpecifiedValue> {
self.set_point
}
fn write_to_problem(&self, problem: &mut ConstraintProblem, elts: &Slab<Element>) { fn write_to_problem(&self, problem: &mut ConstraintProblem, elts: &Slab<Element>) {
self.set_point.with_untracked(|set_pt| { self.set_point.with_untracked(|set_pt| {
if let Some(val) = set_pt.value { if let Some(val) = set_pt.value {
@ -169,7 +190,7 @@ type AssemblyMotion<'a> = Vec<ElementMotion<'a>>;
pub struct Assembly { pub struct Assembly {
// elements and regulators // elements and regulators
pub elements: Signal<Slab<Element>>, pub elements: Signal<Slab<Element>>,
pub regulators: Signal<Slab<ProductRegulator>>, pub regulators: Signal<Slab<Rc<dyn Regulator>>>,
// solution variety tangent space. the basis vectors are stored in // solution variety tangent space. the basis vectors are stored in
// configuration matrix format, ordered according to the elements' column // configuration matrix format, ordered according to the elements' column
@ -240,19 +261,23 @@ impl Assembly {
); );
} }
fn insert_regulator(&self, regulator: ProductRegulator) { fn insert_regulator(&self, regulator: Rc<dyn Regulator>) {
let subjects = regulator.subjects; let subjects = regulator.subjects();
let key = self.regulators.update(|regs| regs.insert(regulator)); let key = self.regulators.update(
let subject_regulators = self.elements.with( |regs| regs.insert(regulator)
|elts| subjects.map(|subj| elts[subj].regulators) );
let subject_regulators: Vec<_> = self.elements.with(
|elts| subjects.into_iter().map(
|subj| elts[subj].regulators
).collect()
); );
for regulators in subject_regulators { for regulators in subject_regulators {
regulators.update(|regs| regs.insert(key)); regulators.update(|regs| regs.insert(key));
} }
} }
pub fn insert_new_regulator(self, subjects: [ElementKey; 2]) { pub fn insert_new_product_regulator(self, subjects: [ElementKey; 2]) {
// create and insert a new regulator // create and insert a new product regulator
let measurement = self.elements.map( let measurement = self.elements.map(
move |elts| { move |elts| {
let representations = subjects.map(|subj| elts[subj].representation); let representations = subjects.map(|subj| elts[subj].representation);
@ -264,26 +289,31 @@ impl Assembly {
} }
); );
let set_point = create_signal(SpecifiedValue::from_empty_spec()); let set_point = create_signal(SpecifiedValue::from_empty_spec());
self.insert_regulator(ProductRegulator { self.insert_regulator(Rc::new(ProductRegulator {
subjects: subjects, subjects: subjects,
measurement: measurement, measurement: measurement,
set_point: set_point set_point: set_point
}); }));
/* DEBUG */ /* DEBUG */
// print an updated list of regulators // print an updated list of regulators
console::log_1(&JsValue::from("Regulators:")); console::log_1(&JsValue::from("Regulators:"));
self.regulators.with(|regs| { self.regulators.with(|regs| {
for (_, reg) in regs.into_iter() { for (_, reg) in regs.into_iter() {
console::log_5( console::log_1(&JsValue::from(format!(
&JsValue::from(" "), " {:?}: {}",
&JsValue::from(reg.subjects[0]), reg.subjects(),
&JsValue::from(reg.subjects[1]), reg.set_point().with_untracked(
&JsValue::from(":"), |set_pt| {
&reg.set_point.with_untracked( let spec = &set_pt.spec;
|set_pt| JsValue::from(set_pt.spec.as_str()) if spec.is_empty() {
"__".to_string()
} else {
spec.clone()
}
}
) )
); )));
} }
}); });

View file

@ -1,4 +1,5 @@
use itertools::Itertools; use itertools::Itertools;
use std::rc::Rc;
use sycamore::prelude::*; use sycamore::prelude::*;
use web_sys::{ use web_sys::{
KeyboardEvent, KeyboardEvent,
@ -9,24 +10,32 @@ use web_sys::{
use crate::{ use crate::{
AppState, AppState,
assembly, assembly,
assembly::{ElementKey, ProductRegulator, RegulatorKey}, assembly::{ElementKey, Regulator, RegulatorKey},
specified::SpecifiedValue specified::SpecifiedValue
}; };
// an editable view of a regulator // an editable view of a regulator
#[component(inline_props)] #[component(inline_props)]
fn RegulatorInput(regulator: ProductRegulator) -> View { fn RegulatorInput(regulator: Rc<dyn Regulator>) -> View {
// get the regulator's measurement and set point signals
let measurement = regulator.measurement();
let set_point = regulator.set_point();
// the `valid` signal tracks whether the last entered value is a valid set
// point specification
let valid = create_signal(true); let valid = create_signal(true);
// the `value` signal holds the current set point specification
let value = create_signal( let value = create_signal(
regulator.set_point.with_untracked(|set_pt| set_pt.spec.clone()) set_point.with_untracked(|set_pt| set_pt.spec.clone())
); );
// this closure resets the input value to the regulator's set point // this `reset_value` closure resets the input value to the regulator's set
// specification // point specification
let reset_value = move || { let reset_value = move || {
batch(|| { batch(|| {
valid.set(true); valid.set(true);
value.set(regulator.set_point.with(|set_pt| set_pt.spec.clone())); value.set(set_point.with(|set_pt| set_pt.spec.clone()));
}) })
}; };
@ -39,7 +48,7 @@ fn RegulatorInput(regulator: ProductRegulator) -> View {
r#type="text", r#type="text",
class=move || { class=move || {
if valid.get() { if valid.get() {
regulator.set_point.with(|set_pt| { set_point.with(|set_pt| {
if set_pt.is_present() { if set_pt.is_present() {
"regulator-input constraint" "regulator-input constraint"
} else { } else {
@ -50,13 +59,13 @@ fn RegulatorInput(regulator: ProductRegulator) -> View {
"regulator-input invalid" "regulator-input invalid"
} }
}, },
placeholder=regulator.measurement.with(|result| result.to_string()), placeholder=measurement.with(|result| result.to_string()),
bind:value=value, bind:value=value,
on:change=move |_| { on:change=move |_| {
valid.set( valid.set(
match SpecifiedValue::try_from(value.get_clone_untracked()) { match SpecifiedValue::try_from(value.get_clone_untracked()) {
Ok(set_pt) => { Ok(set_pt) => {
regulator.set_point.set(set_pt); set_point.set(set_pt);
true true
} }
Err(_) => false Err(_) => false
@ -80,11 +89,12 @@ fn RegulatorInput(regulator: ProductRegulator) -> View {
fn RegulatorOutlineItem(regulator_key: RegulatorKey, element_key: ElementKey) -> View { fn RegulatorOutlineItem(regulator_key: RegulatorKey, element_key: ElementKey) -> View {
let state = use_context::<AppState>(); let state = use_context::<AppState>();
let assembly = &state.assembly; let assembly = &state.assembly;
let regulator = assembly.regulators.with(|regs| regs[regulator_key]); let regulator = assembly.regulators.with(|regs| regs[regulator_key].clone());
let other_subject = if regulator.subjects[0] == element_key { let subjects = regulator.subjects();
regulator.subjects[1] let other_subject = if subjects[0] == element_key {
subjects[1]
} else { } else {
regulator.subjects[0] subjects[0]
}; };
let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone()); let other_subject_label = assembly.elements.with(|elts| elts[other_subject].label.clone());
view! { view! {