//! # A term that can be the input for calculation //! //! Short description //! //! Details //! //! ## Section 1 //! //! ## Section 2 //// ATTRIBUTES //////////////////////////////////////////////////////////////////////////////////// // we want docs #![warn(missing_docs)] #![warn(rustdoc::missing_crate_level_docs)] // we want Debug everywhere. #![warn(missing_debug_implementations)] // enable clippy's extra lints, the pedantic version #![warn(clippy::pedantic)] use std::collections::VecDeque; //// IMPORTS /////////////////////////////////////////////////////////////////////////////////////// pub use super::{Error, Result, Value, base::{self, *}}; use crate::logger::*; //// TYPES ///////////////////////////////////////////////////////////////////////////////////////// //// CONSTANTS ///////////////////////////////////////////////////////////////////////////////////// //// STATICS /////////////////////////////////////////////////////////////////////////////////////// //// MACROS //////////////////////////////////////////////////////////////////////////////////////// //// ENUMS ///////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////// /// ## Parsed value to be calculated /// /// This enum represents anything that goes to the output queue of [`Term::prepare()`] and will /// then be used to actually calculate something in [`Term::process()`]. #[derive(Debug)] enum Token { /// Some kind of operator Operator(Operator), /// A concrete value that we can calculate something with. May be a constant, integer, float, /// etc. Value(super::base::Value), /// A variable of some kind that will be present in the result Variable(char), } //// STRUCTS /////////////////////////////////////////////////////////////////////////////////////// /// ## Term that can be calculated /// /// Represents a signular term, that can be calculated. Terms will be evaluated by the [`Term::prepare`] /// function, afterwards calculated (as much as possible) in the [`Term::process`] function. /// #[derive(Debug)] pub struct Term { /// the original expression to calculate pub original: String, /// the calculated result, may be of diffrent types, see [`crate::math::calculator::result`]. pub result: Option>, ///////////////////////////////////// ///// internal values following ///// ///////////////////////////////////// operator_stack: Vec, output_queue: VecDeque } //// IMPLEMENTATION //////////////////////////////////////////////////////////////////////////////// impl Term { /// Build a new term from an expression /// /// Invalid terms will result in an [`Err`]. pub fn new(orig: String) -> Result { return Ok( Term { original: orig, result: None, operator_stack: Vec::new(), output_queue: VecDeque::new() } ) } /// Prepare the term for the processing. pub fn prepare(&mut self) -> Result<()> { // TODO: shunting yard // Storage for unfinished tokens let mut unfinished_chars: Vec = Vec::new(); for (index, c) in self.original.chars().enumerate() { if !c.is_alphanumeric() { // TODO: allow any unicode char to be a variable warn!("'{c}' is not a valid character, only alphanumeric input is allowed."); return Err(Error::SyntaxError); } // this will be a mess, but it has to be before i can sort the mess. match c { _ => { warn!("The meaning of '{c}' could not be identified."); return Err(Error::SyntaxError); } } } Ok(()) } /// Process a prepared term, calculating it's result pub fn process(&mut self) -> Result<()> { debug!("processing term: {:#?}", self); debug!("queue: {:#?}", self.output_queue); // TODO: process RPN and set result self.result = Some(Ok(19.into())); Ok(()) } /// Convert a character into a token /// /// Returns: A tuple with a [`Token`] and a [`bool`]. If the bool is false, the [`Token`] is /// marked as "incomplete", meaning that the character cannot be used yet. fn to_tok(s: Vec) -> Result { Ok(19.into()) } /// only leave relevant chars for calculation fn filter(s: String) -> String { let mut filtered = String::new(); for c in s.chars() { if !Self::is_ignore(&c) { filtered.push(c); } } return filtered } /// check if we should ignore this character fn is_ignore(c: &char) -> bool { match *c { ' ' => true, _ => false } } } //////////////////////////////////////////////////////////////////////////////////////////////////// /// Helper methods for Tokens impl Token { } //////////////////////////////////////////////////////////////////////////////////////////////////// impl From for Token where T: Into, T: PrimInt, u128: TryFrom { fn from(value: T) -> Self { Token::Value(base::Value::from(value)) } } //// PUBLIC FUNCTIONS ////////////////////////////////////////////////////////////////////////////// //// PRIVATE FUNCTIONS ///////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////