Added substitution config switch

This commit is contained in:
2023-08-20 16:19:57 -07:00
parent 6e3609d85e
commit 6969d17cce
4 changed files with 110 additions and 101 deletions

View File

@ -21,8 +21,10 @@ pub fn parse(
context: &Context, s: &String
) -> Result<Expression, (LineLocation, DaisyError)> {
let expressions = stage::tokenize(context, s);
let (_, expressions) = stage::find_subs(expressions);
let mut expressions = stage::tokenize(context, s);
if context.config.enable_substituion {
(_, expressions) = stage::find_subs(expressions);
}
let g = stage::groupify(context, expressions)?;
let g = stage::treeify(context, g)?;
@ -34,6 +36,7 @@ pub fn parse_no_context(s: &String) -> Result<Expression, (LineLocation, DaisyEr
}
pub fn substitute(context: &Context, s: &String) -> String {
if !context.config.enable_substituion { return s.clone(); }
let (_, s) = substitute_cursor(context, s, s.chars().count());
return s;
}
@ -43,34 +46,45 @@ pub fn substitute_cursor(
s: &String, // The string to substitute
c: usize // Location of the cursor right now
) -> (
usize, // Location of cursor in substituted string
usize, // New cursor
String // String with substitutions
) {
if !context.config.enable_substituion { return (c, s.clone()); }
if s == "" { return (c, s.clone()) }
let mut new_s = s.clone();
let l = s.chars().count();
let expressions = stage::tokenize(context, s);
let (mut subs, _) = stage::find_subs(expressions);
let mut new_c = l - c;
let mut new_c = c.clone();
while subs.len() > 0 {
let r = subs.pop_back().unwrap();
// Apply substitutions in reverse order
// r is the current substitution: (linelocation, string)
let r = subs.pop_back().unwrap();
if { // Don't substitute if our cursor is inside the substitution
c >= r.0.pos &&
c < r.0.pos+r.0.len
} { continue; }
if c < r.0.pos {
let ct = r.1.chars().count();
if ct >= r.0.len {
if new_c >= ct - r.0.len {
new_c += ct - r.0.len
}
} else {
new_c -= r.0.len - ct
// If this substitution is before our cursor,
// we need to adjust our cursor's position.
if c > r.0.pos {
let c_o = r.0.len; // Old length
let c_n = r.1.chars().count(); // New length
if c_n > c_o {
// Move cursor right by difference
new_c += c_n - c_o;
} else if c_n < c_o {
// Move cursor left by difference
if new_c >= c_o - c_n {
new_c -= c_o - c_n;
} else { new_c = 0; }
}
}

View File

@ -6,6 +6,64 @@ use super::super::{
};
fn sub_string(s: &str) -> Option<&'static str> {
let r = match s {
/* Only found in operator tokens */
"*" => "×",
"/" => "÷",
"sqrt" => "",
"rt" => "",
/* Only found in word tokens */
// Greek letters
"alpha" => "α",
"beta" => "β",
"gamma" => "γ",
"delta" => "δ",
"epsilon" => "ε",
"zeta" => "ζ",
"eta" => "η",
"theta" => "θ",
//"iota" => {Some("ι")}, // looks just like i
//"kappa" => {Some("κ")}, // looks just like k
"lambda" => "λ",
"mu" => "μ",
//"nu" => {Some("ν")}, // looks just like v
"xi" => "ξ",
//"omicron" => {Some("ο")}, // looks exactly like o
"pi" => "π",
"rho" => "ρ",
"sigma" => "σ",
"tau" => "τ",
//"upsilon" => {Some("υ")}, // looks just like u
"phi" => "φ",
"chi" => "χ",
//"psi" => {Some("ψ")}, Conflict with pound / square inch
"omega" => "ω",
// Constants
"epsilon_zero" => "ε₀",
"eps_zero" => "ε₀",
"g_zero" => "g₀",
"mu_zero" => "μ₀",
"h_bar" => "",
// Misc
"deg" => "°",
_ => { return None; }
};
return Some(r);
}
pub fn find_subs(
mut g: VecDeque<Token>,
) -> (
@ -24,62 +82,19 @@ pub fn find_subs(
while g.len() > 0 {
let mut t = g.pop_front().unwrap();
let target: Option<&str> = match &mut t {
Token::Operator(_, s) => {
let target = match &s[..] {
"*" => {Some("×")},
"/" => {Some("÷")},
"sqrt" => {Some("")},
"rt" => {Some("")},
_ => {None}
};
let target = sub_string(s);
// Update token contents too.
// This makes sure that errors also contain the updated text.
// This makes errors and printouts use the updated string.
if target.is_some() { *s = String::from(target.unwrap()); }
target
},
Token::Word(_, s) => {
let target = match &s[..] {
// Greek letters
"alpha" => {Some("α")},
"beta" => {Some("β")},
"gamma" => {Some("γ")},
"delta" => {Some("δ")},
"epsilon" => {Some("ε")},
"zeta" => {Some("ζ")},
"eta" => {Some("η")},
"theta" => {Some("θ")},
//"iota" => {Some("ι")},
//"kappa" => {Some("κ")},
"lambda" => {Some("λ")},
"mu" => {Some("μ")},
//"nu" => {Some("ν")},
"xi" => {Some("ξ")},
//"omicron" => {Some("ο")},
"pi" => {Some("π")},
"rho" => {Some("ρ")},
"sigma" => {Some("σ")},
"tau" => {Some("τ")},
//"upsilon" => {Some("υ")},
"phi" => {Some("φ")},
"chi" => {Some("χ")},
//"psi" => {Some("ψ")}, Conflict with pound / square inch
"omega" => {Some("ω")},
// Constants
"epsilon_zero" => {Some("ε₀")},
"eps_zero" => {Some("ε₀")},
"g_zero" => {Some("g₀")},
"mu_zero" => {Some("μ₀")},
"h_bar" => {Some("")},
// Misc
"deg" => {Some("°")}
_ => {None}
};
let target = sub_string(s);
if target.is_some() { *s = String::from(target.unwrap()); }
target
},