Fix some edge-cases in vim visual delete (#12131)

Release Notes:

- vim: Fixed `shift-d` in visual and visual block mode.
This commit is contained in:
Conrad Irwin 2024-05-22 12:54:41 -06:00 committed by GitHub
parent c084b6aade
commit c2f650fe49
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 66 additions and 6 deletions

View File

@ -493,8 +493,8 @@
"shift-o": "vim::OtherEnd",
"d": "vim::VisualDelete",
"x": "vim::VisualDelete",
"shift-d": "vim::VisualDelete",
"shift-x": "vim::VisualDelete",
"shift-d": "vim::VisualDeleteLine",
"shift-x": "vim::VisualDeleteLine",
"y": "vim::VisualYank",
"shift-y": "vim::VisualYank",
"p": "vim::Paste",

View File

@ -30,6 +30,7 @@ actions!(
ToggleVisualLine,
ToggleVisualBlock,
VisualDelete,
VisualDeleteLine,
VisualYank,
OtherEnd,
SelectNext,
@ -55,7 +56,13 @@ pub fn register(workspace: &mut Workspace, _: &mut ViewContext<Workspace>) {
workspace.register_action(|_, _: &VisualDelete, cx| {
Vim::update(cx, |vim, cx| {
vim.record_current_action(cx);
delete(vim, cx);
delete(vim, false, cx);
});
});
workspace.register_action(|_, _: &VisualDeleteLine, cx| {
Vim::update(cx, |vim, cx| {
vim.record_current_action(cx);
delete(vim, true, cx);
});
});
workspace.register_action(|_, _: &VisualYank, cx| {
@ -355,10 +362,10 @@ pub fn other_end(_: &mut Workspace, _: &OtherEnd, cx: &mut ViewContext<Workspace
});
}
pub fn delete(vim: &mut Vim, cx: &mut WindowContext) {
pub fn delete(vim: &mut Vim, line_mode: bool, cx: &mut WindowContext) {
vim.update_active_editor(cx, |vim, editor, cx| {
let mut original_columns: HashMap<_, _> = Default::default();
let line_mode = editor.selections.line_mode;
let line_mode = line_mode || editor.selections.line_mode;
editor.transact(cx, |editor, cx| {
editor.change_selections(Some(Autoscroll::fit()), cx, |s| {
@ -369,6 +376,17 @@ pub fn delete(vim: &mut Vim, cx: &mut WindowContext) {
position = movement::left(map, position);
}
original_columns.insert(selection.id, position.to_point(map).column);
if vim.state().mode == Mode::VisualBlock {
*selection.end.column_mut() = map.line_len(selection.end.row())
} else if vim.state().mode != Mode::VisualLine {
selection.start = DisplayPoint::new(selection.start.row(), 0);
if selection.end.row() == map.max_point().row() {
selection.end = map.max_point()
} else {
*selection.end.row_mut() += 1;
*selection.end.column_mut() = 0;
}
}
}
selection.goal = SelectionGoal::None;
});
@ -569,7 +587,7 @@ pub fn select_match(
Some(Operator::Change) => substitute(vim, None, false, cx),
Some(Operator::Delete) => {
vim.stop_recording();
delete(vim, cx)
delete(vim, false, cx)
}
Some(Operator::Yank) => yank(vim, cx),
_ => {} // Ignoring other operators
@ -1197,4 +1215,36 @@ mod test {
cx.simulate_shared_keystrokes(".").await;
cx.shared_state().await.assert_eq("aa ˇx aa aa aa");
}
#[gpui::test]
async fn test_visual_shift_d(cx: &mut gpui::TestAppContext) {
let mut cx = NeovimBackedTestContext::new(cx).await;
cx.set_shared_state(indoc! {
"The ˇquick brown
fox jumps over
the lazy dog
"
})
.await;
cx.simulate_shared_keystrokes("v down shift-d").await;
cx.shared_state().await.assert_eq(indoc! {
"the ˇlazy dog\n"
});
cx.set_shared_state(indoc! {
"The ˇquick brown
fox jumps over
the lazy dog
"
})
.await;
cx.simulate_shared_keystrokes("ctrl-v down shift-d").await;
cx.shared_state().await.assert_eq(indoc! {
"Theˇ•
fox
the lazy dog
"
});
}
}

View File

@ -0,0 +1,10 @@
{"Put":{"state":"The ˇquick brown\nfox jumps over\nthe lazy dog\n"}}
{"Key":"v"}
{"Key":"down"}
{"Key":"shift-d"}
{"Get":{"state":"the ˇlazy dog\n","mode":"Normal"}}
{"Put":{"state":"The ˇquick brown\nfox jumps over\nthe lazy dog\n"}}
{"Key":"ctrl-v"}
{"Key":"down"}
{"Key":"shift-d"}
{"Get":{"state":"Theˇ \nfox \nthe lazy dog\n","mode":"Normal"}}