#![allow(clippy::significant_drop_tightening)]
use {
super::super::*,
crate::ids::EXECUTE_FIND_CHAR,
reovim_driver_command::{ArgValue, Command, CommandContext, CommandHandler, CommandResult},
reovim_driver_session::SessionRuntime,
reovim_kernel::api::v1::Position,
};
use {
reovim_driver_session::{ClientId, ExtensionMap, Session, WindowLayout, testing::StubExecutor},
reovim_kernel::{
api::{
ModeStack,
v1::{
Buffer, BufferId, HistoryRing, Jumplist, KernelContext, MarkBank, ModeId, ModuleId,
RegisterBank,
},
},
testing::create_test_context,
},
};
struct TestState {
session: Session,
mode_stack: ModeStack,
windows: WindowLayout,
extensions: ExtensionMap,
compositor: Option<Box<dyn reovim_driver_layout::RootCompositor>>,
tabs: reovim_driver_session::TabPageSet,
registers: RegisterBank,
clipboard_history: HistoryRing,
local_marks: MarkBank,
jumplist: Jumplist,
active_buffer: Option<BufferId>,
terminal_size: (u16, u16),
}
impl TestState {
fn with_buffer(buffer_id: Option<BufferId>) -> Self {
let home_mode = ModeId::new(ModuleId::new("test"), "normal");
let session = Session::new(ClientId::new(1), home_mode.clone());
let mode_stack = ModeStack::new(home_mode);
let mut windows = WindowLayout::empty();
let extensions = ExtensionMap::new();
let mut window = reovim_driver_session::Window::new();
if let Some(buffer_id) = buffer_id {
window.buffer_id = Some(buffer_id);
}
windows.add(window);
Self {
session,
mode_stack,
windows,
extensions,
compositor: None,
tabs: reovim_driver_session::TabPageSet::new(),
registers: RegisterBank::new(),
clipboard_history: HistoryRing::new(),
local_marks: MarkBank::new(),
jumplist: Jumplist::new(),
active_buffer: None,
terminal_size: (80, 24),
}
}
fn runtime<'a>(&'a mut self, kernel: &'a KernelContext) -> SessionRuntime<'a> {
SessionRuntime::new(
&mut self.session,
reovim_driver_session::ClientContext {
mode_stack: &mut self.mode_stack,
windows: &mut self.windows,
extensions: &mut self.extensions,
compositor: &mut self.compositor,
tabs: &mut self.tabs,
registers: &mut self.registers,
clipboard_history: &mut self.clipboard_history,
local_marks: &mut self.local_marks,
jumplist: &mut self.jumplist,
active_buffer: &mut self.active_buffer,
terminal_size: &mut self.terminal_size,
},
kernel,
&StubExecutor,
)
}
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_command_metadata() {
let cmd = ExecuteFindChar;
assert_eq!(cmd.id(), EXECUTE_FIND_CHAR);
assert!(cmd.description().contains("find-char"));
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_command_id_module() {
let cmd = ExecuteFindChar;
assert_eq!(cmd.id().module().as_str(), "vim");
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_command_id_name() {
let cmd = ExecuteFindChar;
assert_eq!(cmd.id().name(), "execute-find-char");
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_command_description_not_empty() {
let cmd = ExecuteFindChar;
assert!(!cmd.description().is_empty());
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_no_find_char_arg() {
let ctx = create_test_context();
let args = CommandContext::new();
let mut state = TestState::with_buffer(None);
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert!(matches!(result, CommandResult::Error(_)));
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_no_buffer() {
let ctx = create_test_context();
let mut args = CommandContext::new();
args.set("find_char", ArgValue::Char('x'));
let mut state = TestState::with_buffer(None);
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert!(matches!(result, CommandResult::Error(_)));
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_forward() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('o'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 4);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_not_found() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('z'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 0);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_backward() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('h'));
args.set("find_direction", ArgValue::String("backward".to_string()));
let mut state = TestState::with_buffer(Some(buffer_id));
if let Some(w) = state.windows.active_mut() {
w.cursor = Position::new(0, 5).into();
}
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 0);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_till() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('o'));
args.set("find_inclusive", ArgValue::Bool(false));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 3);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_default_direction_is_forward() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcdef");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('d'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 3);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_last_char_on_line() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcde");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('e'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 4);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_first_char_on_line() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcde");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('a'));
args.set("find_direction", ArgValue::String("backward".to_string()));
let mut state = TestState::with_buffer(Some(buffer_id));
if let Some(w) = state.windows.active_mut() {
w.cursor = Position::new(0, 4).into();
}
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 0);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_with_count() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcabc");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('a'));
args.set("count", reovim_driver_command::ArgValue::Count(1));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 3);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_till_backward() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('h'));
args.set("find_direction", ArgValue::String("backward".to_string()));
args.set("find_inclusive", ArgValue::Bool(false));
let mut state = TestState::with_buffer(Some(buffer_id));
if let Some(w) = state.windows.active_mut() {
w.cursor = Position::new(0, 5).into();
}
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 1);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_forward_direction_string() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcdef");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('f'));
args.set("find_direction", ArgValue::String("forward".to_string()));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 5);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_space_character() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char(' '));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 5); }
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_inclusive_default() {
let ctx = create_test_context();
let buffer = Buffer::from_string("abcdef");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('c'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 2); }
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_on_empty_line() {
let ctx = create_test_context();
let buffer = Buffer::from_string("");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('x'));
let mut state = TestState::with_buffer(Some(buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert_eq!(result, CommandResult::Success);
let window = runtime.windows().active().unwrap();
assert_eq!(window.cursor.column, 0);
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_no_active_window() {
let ctx = create_test_context();
let buffer = Buffer::from_string("hello world");
let buffer_id = ctx.buffers.register(buffer);
let mut args = CommandContext::new();
args.set_buffer_id(buffer_id);
args.set("find_char", ArgValue::Char('o'));
let home_mode = ModeId::new(ModuleId::new("test"), "normal");
let mut session = Session::new(ClientId::new(1), home_mode.clone());
let mut mode_stack = ModeStack::new(home_mode);
let mut windows = WindowLayout::empty(); let mut extensions = ExtensionMap::new();
let mut compositor = None;
let mut tabs = reovim_driver_session::TabPageSet::new();
let mut registers = RegisterBank::new();
let mut clipboard_history = HistoryRing::new();
let mut local_marks = MarkBank::new();
let mut jumplist = Jumplist::new();
let mut active_buffer = None;
let mut terminal_size = (80u16, 24u16);
let mut runtime = SessionRuntime::new(
&mut session,
reovim_driver_session::ClientContext {
mode_stack: &mut mode_stack,
windows: &mut windows,
extensions: &mut extensions,
compositor: &mut compositor,
tabs: &mut tabs,
registers: &mut registers,
clipboard_history: &mut clipboard_history,
local_marks: &mut local_marks,
jumplist: &mut jumplist,
active_buffer: &mut active_buffer,
terminal_size: &mut terminal_size,
},
&ctx,
&StubExecutor,
);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert!(matches!(result, CommandResult::Error(_)));
}
#[cfg_attr(coverage_nightly, coverage(off))]
#[test]
fn test_execute_find_char_buffer_not_found() {
let ctx = create_test_context();
let fake_buffer_id = BufferId::from_raw(999);
let mut args = CommandContext::new();
args.set_buffer_id(fake_buffer_id);
args.set("find_char", ArgValue::Char('o'));
let mut state = TestState::with_buffer(Some(fake_buffer_id));
let mut runtime = state.runtime(&ctx);
let result = ExecuteFindChar.execute(&mut runtime, &args);
assert!(matches!(result, CommandResult::Error(_)));
}