use crate::{insert::NormalBefore, Vim}; use editor::{Editor, EditorEvent}; use gpui::{Action, AppContext, Entity, EntityId, View, ViewContext, WindowContext}; pub fn init(cx: &mut AppContext) { cx.observe_new_views(|_, cx: &mut ViewContext| { let editor = cx.view().clone(); cx.subscribe(&editor, |_, editor, event: &EditorEvent, cx| match event { EditorEvent::Focused => cx.window_context().defer(|cx| focused(editor, cx)), EditorEvent::Blurred => cx.window_context().defer(|cx| blurred(editor, cx)), _ => {} }) .detach(); let id = cx.view().entity_id(); cx.on_release(move |_, _, cx| released(id, cx)).detach(); }) .detach(); } fn focused(editor: View, cx: &mut WindowContext) { if Vim::read(cx).active_editor.clone().is_some() { Vim::update(cx, |vim, cx| { vim.update_active_editor(cx, |previously_active_editor, cx| { vim.unhook_vim_settings(previously_active_editor, cx) }); }); } Vim::update(cx, |vim, cx| { vim.set_active_editor(editor.clone(), cx); }); } fn blurred(editor: View, cx: &mut WindowContext) { Vim::update(cx, |vim, cx| { vim.stop_recording_immediately(NormalBefore.boxed_clone()); if let Some(previous_editor) = vim.active_editor.clone() { if previous_editor .upgrade() .is_some_and(|previous| previous == editor.clone()) { vim.clear_operator(cx); vim.active_editor = None; vim.editor_subscription = None; } } editor.update(cx, |editor, cx| vim.unhook_vim_settings(editor, cx)) }); } fn released(entity_id: EntityId, cx: &mut AppContext) { cx.update_global(|vim: &mut Vim, _| { if vim .active_editor .as_ref() .is_some_and(|previous| previous.entity_id() == entity_id) { vim.active_editor = None; vim.editor_subscription = None; } vim.editor_states.remove(&entity_id) }); } #[cfg(test)] mod test { use crate::{test::VimTestContext, Vim}; use editor::Editor; use gpui::{Context, Entity, VisualTestContext}; use language::Buffer; // regression test for blur called with a different active editor #[gpui::test] async fn test_blur_focus(cx: &mut gpui::TestAppContext) { let mut cx = VimTestContext::new(cx, true).await; let buffer = cx.new_model(|_| Buffer::new(0, 0, "a = 1\nb = 2\n")); let window2 = cx.add_window(|cx| Editor::for_buffer(buffer, None, cx)); let editor2 = cx .update(|cx| { window2.update(cx, |_, cx| { cx.activate_window(); cx.focus_self(); cx.view().clone() }) }) .unwrap(); cx.run_until_parked(); cx.update(|cx| { let vim = Vim::read(cx); assert_eq!( vim.active_editor.as_ref().unwrap().entity_id(), editor2.entity_id(), ) }); // no panic when blurring an editor in a different window. cx.update_editor(|editor1, cx| { editor1.handle_blur(cx); }); } // regression test for focus_in/focus_out being called on window activation #[gpui::test] async fn test_focus_across_windows(cx: &mut gpui::TestAppContext) { let mut cx = VimTestContext::new(cx, true).await; let mut cx1 = VisualTestContext::from_window(cx.window, &cx); let editor1 = cx.editor.clone(); let buffer = cx.new_model(|_| Buffer::new(0, 0, "a = 1\nb = 2\n")); let (editor2, cx2) = cx.add_window_view(|cx| Editor::for_buffer(buffer, None, cx)); editor2.update(cx2, |_, cx| { cx.focus_self(); cx.activate_window(); }); cx.run_until_parked(); cx1.update(|cx| { assert_eq!( Vim::read(cx).active_editor.as_ref().unwrap().entity_id(), editor2.entity_id(), ) }); cx1.update(|cx| { cx.activate_window(); }); cx.run_until_parked(); cx.update(|cx| { assert_eq!( Vim::read(cx).active_editor.as_ref().unwrap().entity_id(), editor1.entity_id(), ) }); } }