use std::ops::RangeInclusive; use helix_core::diagnostic::Severity; use helix_term::application::Application; use super::*; #[tokio::test(flavor = "multi_thread")] async fn test_write_quit_fail() -> anyhow::Result<()> { let file = helpers::new_readonly_tempfile()?; let mut app = helpers::AppBuilder::new() .with_file(file.path(), None) .build()?; test_key_sequence( &mut app, Some("ihello:wq"), Some(&|app| { let mut docs: Vec<_> = app.editor.documents().collect(); assert_eq!(1, docs.len()); let doc = docs.pop().unwrap(); assert_eq!(Some(file.path()), doc.path().map(PathBuf::as_path)); assert_eq!(&Severity::Error, app.editor.get_status().unwrap().1); }), false, ) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_buffer_close_concurrent() -> anyhow::Result<()> { test_key_sequences( &mut helpers::AppBuilder::new().build()?, vec![ ( None, Some(&|app| { assert_eq!(1, app.editor.documents().count()); assert!(!app.editor.is_err()); }), ), ( Some("ihello:new"), Some(&|app| { assert_eq!(2, app.editor.documents().count()); assert!(!app.editor.is_err()); }), ), ( Some(":bufferclose"), Some(&|app| { assert_eq!(1, app.editor.documents().count()); assert!(!app.editor.is_err()); }), ), ], false, ) .await?; // verify if writes are queued up, it finishes them before closing the buffer let mut file = tempfile::NamedTempFile::new()?; let mut command = String::new(); const RANGE: RangeInclusive = 1..=1000; for i in RANGE { let cmd = format!("%c{}:w!", i); command.push_str(&cmd); } command.push_str(":bufferclose"); let mut app = helpers::AppBuilder::new() .with_file(file.path(), None) .build()?; test_key_sequence( &mut app, Some(&command), Some(&|app| { assert!(!app.editor.is_err(), "error: {:?}", app.editor.get_status()); let doc = app.editor.document_by_path(file.path()); assert!(doc.is_none(), "found doc: {:?}", doc); }), false, ) .await?; helpers::assert_file_has_content(file.as_file_mut(), &RANGE.end().to_string())?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_selection_duplication() -> anyhow::Result<()> { // Forward test(( platform_line(indoc! {"\ #[lo|]#rem ipsum dolor "}) .as_str(), "CC", platform_line(indoc! {"\ #(lo|)#rem #(ip|)#sum #[do|]#lor "}) .as_str(), )) .await?; // Backward test(( platform_line(indoc! {"\ #[|lo]#rem ipsum dolor "}) .as_str(), "CC", platform_line(indoc! {"\ #(|lo)#rem #(|ip)#sum #[|do]#lor "}) .as_str(), )) .await?; // Copy the selection to previous line, skipping the first line in the file test(( platform_line(indoc! {"\ test #[testitem|]# "}) .as_str(), "", platform_line(indoc! {"\ test #[testitem|]# "}) .as_str(), )) .await?; // Copy the selection to previous line, including the first line in the file test(( platform_line(indoc! {"\ test #[test|]# "}) .as_str(), "", platform_line(indoc! {"\ #[test|]# #(test|)# "}) .as_str(), )) .await?; // Copy the selection to next line, skipping the last line in the file test(( platform_line(indoc! {"\ #[testitem|]# test "}) .as_str(), "C", platform_line(indoc! {"\ #[testitem|]# test "}) .as_str(), )) .await?; // Copy the selection to next line, including the last line in the file test(( platform_line(indoc! {"\ #[test|]# test "}) .as_str(), "C", platform_line(indoc! {"\ #(test|)# #[test|]# "}) .as_str(), )) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_goto_file_impl() -> anyhow::Result<()> { let file = tempfile::NamedTempFile::new()?; fn match_paths(app: &Application, matches: Vec<&str>) -> usize { app.editor .documents() .filter_map(|d| d.path()?.file_name()) .filter(|n| matches.iter().any(|m| *m == n.to_string_lossy())) .count() } // Single selection test_key_sequence( &mut AppBuilder::new().with_file(file.path(), None).build()?, Some("ione.js%gf"), Some(&|app| { assert_eq!(1, match_paths(app, vec!["one.js"])); }), false, ) .await?; // Multiple selection test_key_sequence( &mut AppBuilder::new().with_file(file.path(), None).build()?, Some("ione.jstwo.js%gf"), Some(&|app| { assert_eq!(2, match_paths(app, vec!["one.js", "two.js"])); }), false, ) .await?; // Cursor on first quote test_key_sequence( &mut AppBuilder::new().with_file(file.path(), None).build()?, Some("iimport 'one.js'B;gf"), Some(&|app| { assert_eq!(1, match_paths(app, vec!["one.js"])); }), false, ) .await?; // Cursor on last quote test_key_sequence( &mut AppBuilder::new().with_file(file.path(), None).build()?, Some("iimport 'one.js'bgf"), Some(&|app| { assert_eq!(1, match_paths(app, vec!["one.js"])); }), false, ) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_search_selection() -> anyhow::Result<()> { // Single selection with a length of 1: search for the whole word test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some("ifoobar::baz3bl*"), // 3b places the cursor on the first letter of 'foobar', then move one to the right for good measure Some(&|app| { assert!( r#"register '/' set to 'foobar'"# == app.editor.get_status().unwrap().0 && Some(&"foobar".to_string()) == app.editor.registers.first('/') ); }), false, ) .await?; // Single selection with a length greather than 1: only search for the selection test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some("ifoobar::baz3blvll*"), // 3b places the cursor on the first letter of 'foobar', then move one to the right for good measure, then select two more chars for a total of three Some(&|app| { assert!( r#"register '/' set to 'oob'"# == app.editor.get_status().unwrap().0 && Some(&"oob".to_string()) == app.editor.registers.first('/') ); }), false, ) .await?; // Multiple selection of length 1 each : should still only search for the selection test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some("ifoobar::bazbar::cruxk3blC*"), // k3b places the cursor on the first letter of 'foobar', then move one to the right for good measure, then adds a cursor on the line below Some(&|app| { assert!( // The selections don't seem to be ordered, so we have to test for the two possible orders. (r#"register '/' set to 'o|a'"# == app.editor.get_status().unwrap().0 || r#"register '/' set to 'a|o'"# == app.editor.get_status().unwrap().0) && (Some(&"o|a".to_string()) == app.editor.registers.first('/') || Some(&"a|o".to_string()) == app.editor.registers.first('/')) ); }), false, ) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_multi_selection_paste() -> anyhow::Result<()> { test(( platform_line(indoc! {"\ #[|lorem]# #(|ipsum)# #(|dolor)# "}) .as_str(), "yp", platform_line(indoc! {"\ lorem#[|lorem]# ipsum#(|ipsum)# dolor#(|dolor)# "}) .as_str(), )) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_multi_selection_shell_commands() -> anyhow::Result<()> { // pipe test(( platform_line(indoc! {"\ #[|lorem]# #(|ipsum)# #(|dolor)# "}) .as_str(), "|echo foo", platform_line(indoc! {"\ #[|foo ]# #(|foo )# #(|foo )# "}) .as_str(), )) .await?; // insert-output test(( platform_line(indoc! {"\ #[|lorem]# #(|ipsum)# #(|dolor)# "}) .as_str(), "!echo foo", platform_line(indoc! {"\ #[|foo ]#lorem #(|foo )#ipsum #(|foo )#dolor "}) .as_str(), )) .await?; // append-output test(( platform_line(indoc! {"\ #[|lorem]# #(|ipsum)# #(|dolor)# "}) .as_str(), "echo foo", platform_line(indoc! {"\ lorem#[|foo ]# ipsum#(|foo )# dolor#(|foo )# "}) .as_str(), )) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_undo_redo() -> anyhow::Result<()> { // A jumplist selection is created at a point which is undone. // // * 2[ Add two newlines at line start. We're now on line 3. // * Save the selection on line 3 in the jumplist. // * u Undo the two newlines. We're now on line 1. // * Jump forward an back again in the jumplist. This would panic // if the jumplist were not being updated correctly. test(("#[|]#", "2[u", "#[|]#")).await?; // A jumplist selection is passed through an edit and then an undo and then a redo. // // * [ Add a newline at line start. We're now on line 2. // * Save the selection on line 2 in the jumplist. // * kd Delete line 1. The jumplist selection should be adjusted to the new line 1. // * uU Undo and redo the `kd` edit. // * Jump back in the jumplist. This would panic if the jumplist were not being // updated correctly. // * Jump forward to line 1. test(("#[|]#", "[kduU", "#[|]#")).await?; // In this case we 'redo' manually to ensure that the transactions are composing correctly. test(("#[|]#", "[u[u", "#[|]#")).await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_extend_line() -> anyhow::Result<()> { // extend with line selected then count test(( platform_line(indoc! {"\ #[l|]#orem ipsum dolor "}) .as_str(), "x2x", platform_line(indoc! {"\ #[lorem ipsum dolor |]# "}) .as_str(), )) .await?; // extend with count on partial selection test(( platform_line(indoc! {"\ #[l|]#orem ipsum "}) .as_str(), "2x", platform_line(indoc! {"\ #[lorem ipsum |]# "}) .as_str(), )) .await?; Ok(()) } #[tokio::test(flavor = "multi_thread")] async fn test_character_info() -> anyhow::Result<()> { // UTF-8, single byte test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some("ihh:char"), Some(&|app| { assert_eq!( r#""h" (U+0068) Dec 104 Hex 68"#, app.editor.get_status().unwrap().0 ); }), false, ) .await?; // UTF-8, multi-byte test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some("iëh:char"), Some(&|app| { assert_eq!( r#""ë" (U+0065 U+0308) Hex 65 + cc 88"#, app.editor.get_status().unwrap().0 ); }), false, ) .await?; // Multiple characters displayed as one, escaped characters test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some(":lineending crlf:char"), Some(&|app| { assert_eq!( r#""\r\n" (U+000d U+000a) Hex 0d + 0a"#, app.editor.get_status().unwrap().0 ); }), false, ) .await?; // Non-UTF-8 test_key_sequence( &mut helpers::AppBuilder::new().build()?, Some(":encoding asciiihh:char"), Some(&|app| { assert_eq!(r#""h" Dec 104 Hex 68"#, app.editor.get_status().unwrap().0); }), false, ) .await?; Ok(()) }