mirror of
https://github.com/zed-industries/zed.git
synced 2024-11-08 07:35:01 +03:00
vim: substitute handles multibyte characters
And is now in its own file
This commit is contained in:
parent
16022e9c1a
commit
926acd6033
@ -1,5 +1,6 @@
|
|||||||
mod change;
|
mod change;
|
||||||
mod delete;
|
mod delete;
|
||||||
|
mod substitute;
|
||||||
mod yank;
|
mod yank;
|
||||||
|
|
||||||
use std::{borrow::Cow, cmp::Ordering, sync::Arc};
|
use std::{borrow::Cow, cmp::Ordering, sync::Arc};
|
||||||
@ -25,6 +26,7 @@ use workspace::Workspace;
|
|||||||
use self::{
|
use self::{
|
||||||
change::{change_motion, change_object},
|
change::{change_motion, change_object},
|
||||||
delete::{delete_motion, delete_object},
|
delete::{delete_motion, delete_object},
|
||||||
|
substitute::substitute,
|
||||||
yank::{yank_motion, yank_object},
|
yank::{yank_motion, yank_object},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -478,25 +480,6 @@ pub(crate) fn normal_replace(text: Arc<str>, cx: &mut WindowContext) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn substitute(vim: &mut Vim, count: usize, cx: &mut WindowContext) {
|
|
||||||
vim.update_active_editor(cx, |editor, cx| {
|
|
||||||
editor.transact(cx, |editor, cx| {
|
|
||||||
let selections = editor.selections.all::<Point>(cx);
|
|
||||||
for selection in selections.into_iter().rev() {
|
|
||||||
let end = if selection.start == selection.end {
|
|
||||||
selection.start + Point::new(0, count as u32)
|
|
||||||
} else {
|
|
||||||
selection.end
|
|
||||||
};
|
|
||||||
editor.buffer().update(cx, |buffer, cx| {
|
|
||||||
buffer.edit([(selection.start..end, "")], None, cx)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
});
|
|
||||||
vim.switch_mode(Mode::Insert, true, cx)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use gpui::TestAppContext;
|
use gpui::TestAppContext;
|
||||||
|
69
crates/vim/src/normal/substitute.rs
Normal file
69
crates/vim/src/normal/substitute.rs
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
use gpui::WindowContext;
|
||||||
|
use language::Point;
|
||||||
|
|
||||||
|
use crate::{motion::Motion, Mode, Vim};
|
||||||
|
|
||||||
|
pub fn substitute(vim: &mut Vim, count: usize, cx: &mut WindowContext) {
|
||||||
|
vim.update_active_editor(cx, |editor, cx| {
|
||||||
|
editor.set_clip_at_line_ends(false, cx);
|
||||||
|
editor.change_selections(None, cx, |s| {
|
||||||
|
s.move_with(|map, selection| {
|
||||||
|
if selection.start == selection.end {
|
||||||
|
Motion::Right.expand_selection(map, selection, count, true);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
});
|
||||||
|
editor.transact(cx, |editor, cx| {
|
||||||
|
let selections = editor.selections.all::<Point>(cx);
|
||||||
|
for selection in selections.into_iter().rev() {
|
||||||
|
editor.buffer().update(cx, |buffer, cx| {
|
||||||
|
buffer.edit([(selection.start..selection.end, "")], None, cx)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
});
|
||||||
|
editor.set_clip_at_line_ends(true, cx);
|
||||||
|
});
|
||||||
|
vim.switch_mode(Mode::Insert, true, cx)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use crate::{state::Mode, test::VimTestContext};
|
||||||
|
use indoc::indoc;
|
||||||
|
|
||||||
|
#[gpui::test]
|
||||||
|
async fn test_substitute(cx: &mut gpui::TestAppContext) {
|
||||||
|
let mut cx = VimTestContext::new(cx, true).await;
|
||||||
|
|
||||||
|
// supports a single cursor
|
||||||
|
cx.set_state(indoc! {"ˇabc\n"}, Mode::Normal);
|
||||||
|
cx.simulate_keystrokes(["s", "x"]);
|
||||||
|
cx.assert_editor_state("xˇbc\n");
|
||||||
|
|
||||||
|
// supports a selection
|
||||||
|
cx.set_state(indoc! {"a«bcˇ»\n"}, Mode::Visual { line: false });
|
||||||
|
cx.assert_editor_state("a«bcˇ»\n");
|
||||||
|
cx.simulate_keystrokes(["s", "x"]);
|
||||||
|
cx.assert_editor_state("axˇ\n");
|
||||||
|
|
||||||
|
// supports counts
|
||||||
|
cx.set_state(indoc! {"ˇabc\n"}, Mode::Normal);
|
||||||
|
cx.simulate_keystrokes(["2", "s", "x"]);
|
||||||
|
cx.assert_editor_state("xˇc\n");
|
||||||
|
|
||||||
|
// supports multiple cursors
|
||||||
|
cx.set_state(indoc! {"a«bcˇ»deˇffg\n"}, Mode::Normal);
|
||||||
|
cx.simulate_keystrokes(["2", "s", "x"]);
|
||||||
|
cx.assert_editor_state("axˇdexˇg\n");
|
||||||
|
|
||||||
|
// does not read beyond end of line
|
||||||
|
cx.set_state(indoc! {"ˇabc\n"}, Mode::Normal);
|
||||||
|
cx.simulate_keystrokes(["5", "s", "x"]);
|
||||||
|
cx.assert_editor_state("xˇ\n");
|
||||||
|
|
||||||
|
// it handles multibyte characters
|
||||||
|
cx.set_state(indoc! {"ˇcàfé\n"}, Mode::Normal);
|
||||||
|
cx.simulate_keystrokes(["4", "s", "x"]);
|
||||||
|
cx.assert_editor_state("xˇ\n");
|
||||||
|
}
|
||||||
|
}
|
@ -98,27 +98,3 @@ async fn test_buffer_search(cx: &mut gpui::TestAppContext) {
|
|||||||
assert_eq!(bar.query_editor.read(cx).text(cx), "jumps");
|
assert_eq!(bar.query_editor.read(cx).text(cx), "jumps");
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[gpui::test]
|
|
||||||
async fn test_substitute(cx: &mut gpui::TestAppContext) {
|
|
||||||
let mut cx = VimTestContext::new(cx, true).await;
|
|
||||||
|
|
||||||
// supports a single cursor
|
|
||||||
cx.set_state(indoc! {"ˇabc\n"}, Mode::Normal);
|
|
||||||
cx.simulate_keystrokes(["s", "x"]);
|
|
||||||
cx.assert_editor_state("xˇbc\n");
|
|
||||||
|
|
||||||
// supports a selection
|
|
||||||
cx.set_state(indoc! {"a«bcˇ»\n"}, Mode::Visual { line: false });
|
|
||||||
cx.simulate_keystrokes(["s", "x"]);
|
|
||||||
cx.assert_editor_state("axˇ\n");
|
|
||||||
|
|
||||||
// supports multiple cursors
|
|
||||||
cx.set_state(indoc! {"a«bcˇ»deˇfg\n"}, Mode::Normal);
|
|
||||||
cx.simulate_keystrokes(["s", "x"]);
|
|
||||||
cx.assert_editor_state("axˇdexˇg\n");
|
|
||||||
|
|
||||||
cx.set_state(indoc! {"ˇabc\n"}, Mode::Normal);
|
|
||||||
cx.simulate_keystrokes(["2", "s", "x"]);
|
|
||||||
cx.assert_editor_state("xˇc\n");
|
|
||||||
}
|
|
||||||
|
Loading…
Reference in New Issue
Block a user