forked from StudioInfinity/dyna3
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:
parent
f1f87e97be
commit
25f446499b
3 changed files with 75 additions and 35 deletions
|
@ -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());
|
||||||
}
|
}
|
||||||
) { "🔗" }
|
) { "🔗" }
|
||||||
|
|
|
@ -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| {
|
||||||
®.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()
|
||||||
|
}
|
||||||
|
}
|
||||||
)
|
)
|
||||||
);
|
)));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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! {
|
||||||
|
|
Loading…
Add table
Reference in a new issue