Detailed changes
@@ -168,6 +168,7 @@
"^": "vim::FirstNonWhitespace",
"o": "vim::InsertLineBelow",
"shift-o": "vim::InsertLineAbove",
+ "~": "vim::ChangeCase",
"v": [
"vim::SwitchMode",
{
@@ -297,6 +298,7 @@
"y": "vim::VisualYank",
"p": "vim::VisualPaste",
"s": "vim::Substitute",
+ "~": "vim::ChangeCase",
"r": [
"vim::PushOperator",
"Replace"
@@ -1,3 +1,4 @@
+mod case;
mod change;
mod delete;
mod scroll;
@@ -23,6 +24,7 @@ use log::error;
use workspace::Workspace;
use self::{
+ case::change_case,
change::{change_motion, change_object},
delete::{delete_motion, delete_object},
substitute::substitute,
@@ -44,6 +46,7 @@ actions!(
Paste,
Yank,
Substitute,
+ ChangeCase,
]
);
@@ -53,6 +56,7 @@ pub fn init(cx: &mut AppContext) {
cx.add_action(insert_end_of_line);
cx.add_action(insert_line_above);
cx.add_action(insert_line_below);
+ cx.add_action(change_case);
cx.add_action(|_: &mut Workspace, _: &Substitute, cx| {
Vim::update(cx, |vim, cx| {
let times = vim.pop_number_operator(cx);
@@ -0,0 +1,64 @@
+use gpui::ViewContext;
+use language::Point;
+use workspace::Workspace;
+
+use crate::{motion::Motion, normal::ChangeCase, Vim};
+
+pub fn change_case(_: &mut Workspace, _: &ChangeCase, cx: &mut ViewContext<Workspace>) {
+ Vim::update(cx, |vim, cx| {
+ let count = vim.pop_number_operator(cx);
+ vim.update_active_editor(cx, |editor, cx| {
+ editor.set_clip_at_line_ends(false, cx);
+ editor.transact(cx, |editor, 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);
+ }
+ })
+ });
+ let selections = editor.selections.all::<Point>(cx);
+ for selection in selections.into_iter().rev() {
+ let snapshot = editor.buffer().read(cx).snapshot(cx);
+ editor.buffer().update(cx, |buffer, cx| {
+ let range = selection.start..selection.end;
+ let text = snapshot
+ .text_for_range(selection.start..selection.end)
+ .flat_map(|s| s.chars())
+ .flat_map(|c| {
+ if c.is_lowercase() {
+ c.to_uppercase().collect::<Vec<char>>()
+ } else {
+ c.to_lowercase().collect::<Vec<char>>()
+ }
+ })
+ .collect::<String>();
+
+ buffer.edit([(range, text)], None, cx)
+ })
+ }
+ });
+ editor.set_clip_at_line_ends(true, cx);
+ });
+ })
+}
+
+#[cfg(test)]
+mod test {
+ use crate::{state::Mode, test::VimTestContext};
+ use indoc::indoc;
+
+ #[gpui::test]
+ async fn test_change_case(cx: &mut gpui::TestAppContext) {
+ let mut cx = VimTestContext::new(cx, true).await;
+ cx.set_state(indoc! {"ˇabC\n"}, Mode::Normal);
+ cx.simulate_keystrokes(["~"]);
+ cx.assert_editor_state("AˇbC\n");
+ cx.simulate_keystrokes(["2", "~"]);
+ cx.assert_editor_state("ABcˇ\n");
+
+ cx.set_state(indoc! {"a😀C«dÉ1*fˇ»\n"}, Mode::Normal);
+ cx.simulate_keystrokes(["~"]);
+ cx.assert_editor_state("a😀CDé1*Fˇ\n");
+ }
+}
@@ -6,14 +6,14 @@ use crate::{motion::Motion, Mode, Vim};
pub fn substitute(vim: &mut Vim, count: Option<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| {
+ editor.change_selections(None, cx, |s| {
+ s.move_with(|map, selection| {
+ if selection.start == selection.end {
+ Motion::Right.expand_selection(map, selection, count, true);
+ }
+ })
+ });
let selections = editor.selections.all::<Point>(cx);
for selection in selections.into_iter().rev() {
editor.buffer().update(cx, |buffer, cx| {
@@ -63,7 +63,11 @@ mod test {
// 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");
+ cx.simulate_keystrokes(["4", "s"]);
+ cx.assert_editor_state("ˇ\n");
+
+ // should transactionally undo selection changes
+ cx.simulate_keystrokes(["escape", "u"]);
+ cx.assert_editor_state("ˇcàfé\n");
}
}