Added binary functions to the prelude.

This commit is contained in:
PaddiM8 2020-05-29 15:04:50 +02:00
parent 32cd3a4e9e
commit 7c42368501
2 changed files with 66 additions and 12 deletions

View File

@ -145,6 +145,11 @@ impl Visitor<f64, f64> for Interpreter {
let x = self.visit_expr(&expressions[0]);
self.prelude.call_unary_func(identifier, x)
}
2 => {
let x = self.visit_expr(&expressions[0]);
let y = self.visit_expr(&expressions[1]);
self.prelude.call_binary_func(identifier, x, y)
}
_ => None,
};

View File

@ -26,37 +26,57 @@ impl UnaryFuncInfo {
fn call(&self, x: f64, angle_unit: &Unit) -> f64 {
let func = *self.func;
match self.func_type {
FuncType::Trig => func(self.from_angle_unit(x, angle_unit)),
FuncType::InverseTrig => self.to_angle_unit(func(x), angle_unit),
FuncType::Trig => func(from_angle_unit(x, angle_unit)),
FuncType::InverseTrig => to_angle_unit(func(x), angle_unit),
FuncType::Other => func(x),
}
}
}
struct BinaryFuncInfo {
func: Box<fn(f64, f64) -> f64>,
func_type: FuncType,
}
fn to_angle_unit(&self, x: f64, angle_unit: &Unit) -> f64 {
impl BinaryFuncInfo {
fn call(&self, x: f64, y: f64, angle_unit: &Unit) -> f64 {
let func = *self.func;
match self.func_type {
FuncType::Trig => func(
from_angle_unit(x, angle_unit),
from_angle_unit(y, angle_unit),
),
FuncType::InverseTrig => to_angle_unit(func(x, y), angle_unit),
FuncType::Other => func(x, y),
}
}
}
fn to_angle_unit(x: f64, angle_unit: &Unit) -> f64 {
match angle_unit {
Unit::Radians => x,
Unit::Degrees => x.to_degrees(),
}
}
fn from_angle_unit(&self, x: f64, angle_unit: &Unit) -> f64 {
fn from_angle_unit(x: f64, angle_unit: &Unit) -> f64 {
match angle_unit {
Unit::Radians => x,
Unit::Degrees => x.to_radians(),
}
}
}
pub struct Prelude {
pub angle_unit: Unit,
unary: HashMap<String, UnaryFuncInfo>,
binary: HashMap<String, BinaryFuncInfo>,
}
impl Prelude {
pub fn new() -> Self {
Prelude {
unary: HashMap::new(),
angle_unit: math_parser::DEFAULT_ANGLE_UNIT,
unary: HashMap::new(),
binary: HashMap::new(),
}
}
@ -146,6 +166,27 @@ impl Prelude {
}
}
}
pub fn call_binary_func(&mut self, name: &str, x: f64, y: f64) -> Option<f64> {
let misc_func: Option<fn(f64, f64) -> f64> = match name {
"max" => Some(funcs::max),
"min" => Some(funcs::min),
_ => None,
};
if let Some(func) = misc_func {
let func_info = BinaryFuncInfo {
func: Box::new(func),
func_type: FuncType::Other,
};
let value = func_info.call(x, y, &self.angle_unit);
self.binary.insert(name.to_string(), func_info);
return Some(value);
} else {
None
}
}
}
mod funcs {
@ -248,6 +289,14 @@ mod funcs {
x.ln()
}
pub fn max(x: f64, y: f64) -> f64 {
x.max(y)
}
pub fn min(x: f64, y: f64) -> f64 {
x.min(y)
}
pub fn round(x: f64) -> f64 {
x.round()
}