diff --git a/book/src/generated/typable-cmd.md b/book/src/generated/typable-cmd.md index 7d3622256b3d8..24aa2ab127519 100644 --- a/book/src/generated/typable-cmd.md +++ b/book/src/generated/typable-cmd.md @@ -88,3 +88,4 @@ | `:move`, `:mv` | Move the current buffer and its corresponding file to a different path | | `:yank-diagnostic` | Yank diagnostic(s) under primary cursor to register, or clipboard by default | | `:read`, `:r` | Load a file into buffer | +| `:trim-trailing-whitespace`, `:trim` | Delete whitespace | diff --git a/helix-term/src/commands/typed.rs b/helix-term/src/commands/typed.rs index 68ba9bab556e9..e2bd955b393eb 100644 --- a/helix-term/src/commands/typed.rs +++ b/helix-term/src/commands/typed.rs @@ -2520,6 +2520,97 @@ fn read(cx: &mut compositor::Context, args: &[Cow], event: PromptEvent) -> Ok(()) } +fn trim_whitespace_impl(doc: &Document, selection: &Selection) -> Transaction { + /// Find the last non-whitespace char of a line + fn find_trailing(l: RopeSlice) -> Option { + // Handle empty docs + if l.len_chars() == 0 { + return None; + } + + // Returns the left-wise beginning of the trailing whitespace + // It is +1 the index of that char so that char is not deleted + l.chars_at(l.len_chars()) + .reversed() + .position(|ch| !ch.is_whitespace()) + .map(|n| l.len_chars() - n) + .or(Some(0)) + } + + let mut deletions: Vec = Vec::new(); + let mut delete = |start, end| { + // Don't push empty changes + if start != end { + deletions.push((start, end)); + } + }; + + // Assume ranges are in order and not overlapping + for range in selection.ranges().iter().rev() { + let slice = range.slice(doc.text().slice(..)); + let lines = slice.lines_at(slice.len_lines()).reversed(); + + // Cap the `end` to not delete the line ending + let end_account_le = |line: RopeSlice, n: usize| { + let le_len = helix_core::line_ending::get_line_ending(&line) + .map(|le| le.len_chars()) + .unwrap_or(0); + // Map `end` with respect to the whole doc + range.from() + n - le_len + }; + + // Ignore empty lines if `trailing` is true. + // If not `trailing`, delete trailing whitespace on lines. + let mut trailing = true; + for (idx, line) in lines + .enumerate() + + { + // This subtraction cannot underflow because len_lines must be `> 0` to enter + // this loop and `idx` is always `< len_lines`. + let line_number = len_lines - 1 - idx; + if trailing { + // `n @ 1..` will ignore `Some(0)` from empty lines + if let Some(n @ 1..) = find_trailing(line) { + let start = range.from() + slice.line_to_char(line_number) + n; + // Needed to retain the last EOL of the selection, which would be selected. e.g. in `%:trim` + let end = end_account_le(slice, slice.len_chars()); + delete(start, end); + trailing = false; + } + } else if let Some(n) = find_trailing(line) { + let start = range.from() + slice.line_to_char(line_number) + n; + let end = end_account_le(line, slice.line_to_char(line_number + 1)); + delete(start, end); + } + } + + // Delete empty selections + if trailing { + let start = range.from(); + let end = end_account_le(slice, slice.len_chars()); + delete(start, end); + } + } + Transaction::delete(doc.text(), deletions.into_iter().rev()) +} + +fn trim_whitespace( + cx: &mut compositor::Context, + _args: &[Cow], + event: PromptEvent, +) -> anyhow::Result<()> { + if event != PromptEvent::Validate { + return Ok(()); + } + let (view, doc) = current!(cx.editor); + let selection = doc.selection(view.id); + let tx = trim_whitespace_impl(doc, selection); + doc.apply(&tx, view.id); + doc.append_changes_to_history(view); + Ok(()) +} + pub const TYPABLE_COMMAND_LIST: &[TypableCommand] = &[ TypableCommand { name: "quit", @@ -3141,6 +3232,14 @@ pub const TYPABLE_COMMAND_LIST: &[TypableCommand] = &[ fun: read, signature: CommandSignature::positional(&[completers::filename]), }, + TypableCommand { + name: "trim-trailing-whitespace", + aliases: &["trim"], + doc: "Delete trailing whitespace from the current selections", + fun: trim_whitespace, + signature: CommandSignature::none(), + + }, ]; pub static TYPABLE_COMMAND_MAP: Lazy> =