diff --git a/core/src/num/unit.rs b/core/src/num/unit.rs index bd0d7486..8d05be73 100644 --- a/core/src/num/unit.rs +++ b/core/src/num/unit.rs @@ -728,6 +728,7 @@ impl Value { }) } + #[allow(clippy::too_many_lines)] pub(crate) fn simplify( self, attrs: Attrs, @@ -832,17 +833,25 @@ impl Value { simplifiable: self.simplifiable, }; - if result.unit.has_pos_and_neg_base_unit_exponents() { + if result.unit.components.len() > 1 + && !result + .unit + .components + .iter() + .any(|c| c.unit.singular_name == "rad" || c.unit.singular_name == "radian") + { // try and replace unit with a default one, e.g. `kilogram` or `ampere` let (hashmap, _) = result.unit.to_hashmap_and_scale(int)?; - let mut base_units = hashmap + if let Ok(mut base_units) = hashmap .into_iter() .map(|(k, v)| v.try_as_i64(int).map(|v| format!("{}^{v}", k.name()))) - .collect::, _>>()?; - base_units.sort(); - if let Some(new_unit) = lookup_default_unit(&base_units.join(" ")) { - let rhs = query_unit_static(new_unit, attrs, ctx, int)?.expect_num()?; - return result.convert_to(rhs, int); + .collect::, _>>() + { + base_units.sort(); + if let Some(new_unit) = lookup_default_unit(&base_units.join(" ")) { + let rhs = query_unit_static(new_unit, attrs, ctx, int)?.expect_num()?; + return result.convert_to(rhs, int); + } } } @@ -977,28 +986,6 @@ impl Unit { Ok(Self { components: cs }) } - fn has_pos_and_neg_base_unit_exponents(&self) -> bool { - if self.components.len() <= 1 { - return false; - } - - let mut pos = HashSet::new(); - let mut neg = HashSet::new(); - for comp in &self.components { - let component_sign = comp.exponent > 0.into(); - for (base, base_exp) in &comp.unit.base_units { - let base_sign = base_exp > &0.into(); - let combined_sign = component_sign == base_sign; // xnor - if combined_sign { - pos.insert(base); - } else { - neg.insert(base); - } - } - } - pos.intersection(&neg).next().is_some() - } - pub(crate) fn equal_to(&self, rhs: &str) -> bool { if self.components.len() != 1 { return false; @@ -1011,7 +998,7 @@ impl Unit { prefix.is_empty() && name == rhs } - /// guarantees that base units with an cancelled exponents do not appear in the hashmap + /// base units with cancelled exponents do not appear in the hashmap fn to_hashmap_and_scale(&self, int: &I) -> Result { let mut hashmap = HashMap::::new(); let mut scale = Complex::from(1); diff --git a/core/src/num/unit/named_unit.rs b/core/src/num/unit/named_unit.rs index a33579a7..923b804f 100644 --- a/core/src/num/unit/named_unit.rs +++ b/core/src/num/unit/named_unit.rs @@ -14,7 +14,7 @@ use crate::{ #[derive(Clone, Eq, PartialEq)] pub(crate) struct NamedUnit { prefix: Cow<'static, str>, - singular_name: Cow<'static, str>, + pub(super) singular_name: Cow<'static, str>, plural_name: Cow<'static, str>, alias: bool, pub(super) base_units: HashMap, diff --git a/core/src/units/builtin.rs b/core/src/units/builtin.rs index 81c24127..ef3f0045 100644 --- a/core/src/units/builtin.rs +++ b/core/src/units/builtin.rs @@ -841,8 +841,8 @@ const DEFAULT_UNITS: &[(&str, &str)] = &[ ("pascal", "kilogram^1 meter^-1 second^-2"), ("joule", "kilogram^1 meter^2 second^-2"), ("watt", "kilogram^1 meter^2 second^-3"), - ("ohm", "ampere^-2 kilogram meter^2 second^-3"), - ("volt", "ampere^-1 kilogram meter^2 second^-3"), + ("ohm", "ampere^-2 kilogram^1 meter^2 second^-3"), + ("volt", "ampere^-1 kilogram^1 meter^2 second^-3"), ("liter", "meter^3"), ]; diff --git a/core/tests/integration_tests.rs b/core/tests/integration_tests.rs index 5615fed9..2435de59 100644 --- a/core/tests/integration_tests.rs +++ b/core/tests/integration_tests.rs @@ -5812,3 +5812,13 @@ fn ohms_law() { fn simplification_sec_hz() { test_eval("c/(145MHz)", "approx. 2.0675341931 meters"); } + +#[test] +fn simplification_ohms() { + test_eval("4556 ohm * ampere", "4556 volts"); +} + +#[test] +fn simplification_ohms_2() { + test_eval("4556 volt / ampere", "4556 ohms"); +}