Generalize constraints to observables #48
4 changed files with 65 additions and 65 deletions
|
@ -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<String>`
|
||||
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<String> for SpecifiedValue {
|
||||
type Error = ParseFloatError;
|
||||
|
||||
fn try_from(spec: String) -> Result<Self, Self::Error> {
|
||||
if spec.is_empty() {
|
||||
Ok(Absent)
|
||||
} else {
|
||||
spec.parse::<f64>().map(
|
||||
|value| Present { spec: spec, value: value }
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Copy)]
|
||||
pub struct Regulator {
|
||||
pub subjects: (ElementKey, ElementKey),
|
||||
|
|
|
@ -3,6 +3,7 @@ mod assembly;
|
|||
mod display;
|
||||
mod engine;
|
||||
mod outline;
|
||||
mod specified;
|
||||
|
||||
use rustc_hash::FxHashSet;
|
||||
use sycamore::prelude::*;
|
||||
|
|
|
@ -12,10 +12,9 @@ use crate::{
|
|||
assembly::{
|
||||
ElementKey,
|
||||
Regulator,
|
||||
RegulatorKey,
|
||||
SpecifiedValue,
|
||||
SpecifiedValue::*
|
||||
}
|
||||
RegulatorKey
|
||||
},
|
||||
specified::{SpecifiedValue, SpecifiedValue::{Absent, Present}}
|
||||
};
|
||||
|
||||
// an editable view of a regulator
|
||||
|
|
56
app-proto/src/specified.rs
Normal file
56
app-proto/src/specified.rs
Normal file
|
@ -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<String>`
|
||||
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)
|
||||
glen marked this conversation as resolved
|
||||
//
|
||||
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<String> for SpecifiedValue {
|
||||
type Error = ParseFloatError;
|
||||
|
||||
fn try_from(spec: String) -> Result<Self, Self::Error> {
|
||||
if spec.is_empty() {
|
||||
Ok(Absent)
|
||||
} else {
|
||||
spec.parse::<f64>().map(
|
||||
|value| Present { spec: spec, value: value }
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Add table
Reference in a new issue
try_from("")
? Wouldn't it be cleaner reuse either to implement this as the appropriate unwrapping of try_from(String::new()), or else in the try_from case implement the empty string case by returning the result of this function? We really only want the format of the canonical absent SpecifiedValue to be contained in one place, it seems to me.08ec838
. If we eventually have more than one way to specify an absent value, it might make sense to switch to a "canonically specified absent value" function, but ensuring consistency withtry_from
would get more complicated. For that reason, I'd want to make that switch in the context of actually wanting it, rather than guessing at what we might want in the future.from_empty_spec
did have potential for inconsistency withtry_from
. Commit08ec838
implements your suggestion of havingtry_from
callfrom_empty_spec
when the specification is empty.All right. Having the syntactic "trivial" generator of an unspecified SpecifiedValue (i.e. the one specified by an empty string) rather than a semantic one like "the canonical absent SpecifiedValue" seems a slightly odd emphasis to me (on syntax over semantics), but as you say, at the moment they are more or less equivalent because in fact there is only one syntactically correct specification of an absent SpecifiedValue. Hence resolving.