use crate::{Vim, VimEvent};
use editor::{EditorBlurred, EditorFocused, EditorReleased};
use gpui::AppContext;

pub fn init(cx: &mut AppContext) {
    cx.subscribe_global(focused).detach();
    cx.subscribe_global(blurred).detach();
    cx.subscribe_global(released).detach();
}

fn focused(EditorFocused(editor): &EditorFocused, cx: &mut AppContext) {
    if let Some(previously_active_editor) = Vim::read(cx).active_editor.clone() {
        previously_active_editor.window().update(cx, |cx| {
            Vim::update(cx, |vim, cx| {
                vim.update_active_editor(cx, |previously_active_editor, cx| {
                    vim.unhook_vim_settings(previously_active_editor, cx)
                });
            });
        });
    }

    editor.window().update(cx, |cx| {
        Vim::update(cx, |vim, cx| {
            vim.set_active_editor(editor.clone(), cx);
            if vim.enabled {
                cx.emit_global(VimEvent::ModeChanged {
                    mode: vim.state().mode,
                });
            }
        });
    });
}

fn blurred(EditorBlurred(editor): &EditorBlurred, cx: &mut AppContext) {
    editor.window().update(cx, |cx| {
        Vim::update(cx, |vim, cx| {
            vim.workspace_state.recording = false;
            vim.workspace_state.recorded_actions.clear();
            if let Some(previous_editor) = vim.active_editor.clone() {
                if previous_editor == 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(EditorReleased(editor): &EditorReleased, cx: &mut AppContext) {
    editor.window().update(cx, |cx| {
        Vim::update(cx, |vim, _| {
            if let Some(previous_editor) = vim.active_editor.clone() {
                if previous_editor == editor.clone() {
                    vim.active_editor = None;
                    vim.editor_subscription = None;
                }
            }
            vim.editor_states.remove(&editor.id())
        });
    });
}

#[cfg(test)]
mod test {
    use crate::{test::VimTestContext, Vim};
    use editor::Editor;
    use gpui::View;
    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.add_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.read(|cx| window2.root(cx)).unwrap();

        cx.update(|cx| {
            let vim = Vim::read(cx);
            assert_eq!(vim.active_editor.unwrap().id(), editor2.id())
        });

        // no panic when blurring an editor in a different window.
        cx.update_editor(|editor1, cx| {
            editor1.focus_out(cx.handle().into_any(), cx);
        });
    }
}
