diff options
Diffstat (limited to 'src/macro_processor/macro_processor.rs')
-rw-r--r-- | src/macro_processor/macro_processor.rs | 35 |
1 files changed, 35 insertions, 0 deletions
diff --git a/src/macro_processor/macro_processor.rs b/src/macro_processor/macro_processor.rs index ad230c0..7839963 100644 --- a/src/macro_processor/macro_processor.rs +++ b/src/macro_processor/macro_processor.rs @@ -339,6 +339,20 @@ fn smp_builtin_format_time( Ok(format!("{}", dt.format(&args[0]))) } +fn macro_is_whitespace_deleting(s: &str) -> bool { + s.chars().nth(s.len() - 1) == Some('_') +} + +fn macro_name_clean<'a>(macro_name: &'a str) -> &'a str { + let mut macro_name = macro_name; + if macro_is_whitespace_deleting(macro_name) { + let mut macro_chars = macro_name.chars(); + macro_chars.next_back(); + macro_name = macro_chars.as_str(); + } + macro_name +} + /// Types of macros, this is to make it easy to store both functions and strings #[derive(Clone)] pub enum MacroType { @@ -498,6 +512,9 @@ impl MacroProcessor { /// * `macro_name` - Name of macro to expand if it exists /// * `args` - List of arguments parsed along with macro invokation (empty list if no arguments were parsed) fn expand_macro(&mut self, macro_name: &str, args: &mut [String]) -> Result<String, SMPError> { + // Ignore trailing underscore in macro name, the parser will pop a space in front if + // present, but we should ignore it for finding the macro. + let macro_name = macro_name_clean(macro_name); let Some(macro_body) = self.macros.get(macro_name) else { if args.len() == 0 { return Ok(format!("{}", macro_name)); @@ -600,6 +617,12 @@ impl MacroProcessor { parens_level += 1; state = ParserState::InMacroArgs; } else { + if macro_is_whitespace_deleting(¯o_name) { + if output.chars().last() == Some(' ') { + output.pop(); + } + macro_name = macro_name_clean(¯o_name).to_string(); + } if self.macros.contains_key(¯o_name) { highlight_debug!("\x1b[32m\x1b[7m", input, (macro_name_start -> i)); } @@ -622,6 +645,12 @@ impl MacroProcessor { } ParserState::InMacroArgs => { if (c == ')') && (parens_level == 1) { + if macro_is_whitespace_deleting(¯o_name) { + if output.chars().last() == Some(' ') { + output.pop(); + } + macro_name = macro_name_clean(¯o_name).to_string(); + } highlight_debug!("\x1b[32m\x1b[7m", input, (macro_name_start -> i)); parens_level = 0; macro_args.push(argument.trim().to_string()); @@ -649,6 +678,12 @@ impl MacroProcessor { // Handle cases where the text ends with a macro without arguments if !macro_name.is_empty() { + if macro_is_whitespace_deleting(¯o_name) { + if output.chars().last() == Some(' ') { + output.pop(); + } + macro_name = macro_name_clean(¯o_name).to_string(); + } output.push_str(&self.expand_macro(¯o_name, &mut [])?); } |