mod completion; mod document; pub(crate) mod editor; mod info; pub mod lsp; mod markdown; pub mod menu; pub mod overlay; pub mod picker; pub mod popup; pub mod prompt; mod spinner; mod statusline; mod text; mod text_decorations; use crate::compositor::Compositor; use crate::filter_picker_entry; use crate::job::{self, Callback}; pub use completion::Completion; pub use editor::EditorView; use helix_stdx::rope; use helix_view::icons::ICONS; use helix_view::theme::Style; pub use markdown::Markdown; pub use menu::Menu; pub use picker::{Column as PickerColumn, FileLocation, Picker}; pub use popup::Popup; pub use prompt::{Prompt, PromptEvent}; pub use spinner::{ProgressSpinners, Spinner}; pub use text::Text; use helix_view::Editor; use tui::text::{Span, Spans}; use std::path::Path; use std::{error::Error, path::PathBuf}; struct Utf8PathBuf { path: String, is_dir: bool, } impl AsRef for Utf8PathBuf { fn as_ref(&self) -> &str { &self.path } } pub fn prompt( cx: &mut crate::commands::Context, prompt: std::borrow::Cow<'static, str>, history_register: Option, completion_fn: impl FnMut(&Editor, &str) -> Vec + 'static, callback_fn: impl FnMut(&mut crate::compositor::Context, &str, PromptEvent) + 'static, ) { let mut prompt = Prompt::new(prompt, history_register, completion_fn, callback_fn); // Calculate the initial completion prompt.recalculate_completion(cx.editor); cx.push_layer(Box::new(prompt)); } pub fn prompt_with_input( cx: &mut crate::commands::Context, prompt: std::borrow::Cow<'static, str>, input: String, history_register: Option, completion_fn: impl FnMut(&Editor, &str) -> Vec + 'static, callback_fn: impl FnMut(&mut crate::compositor::Context, &str, PromptEvent) + 'static, ) { let prompt = Prompt::new(prompt, history_register, completion_fn, callback_fn) .with_line(input, cx.editor); cx.push_layer(Box::new(prompt)); } pub fn regex_prompt( cx: &mut crate::commands::Context, prompt: std::borrow::Cow<'static, str>, history_register: Option, completion_fn: impl FnMut(&Editor, &str) -> Vec + 'static, fun: impl Fn(&mut crate::compositor::Context, rope::Regex, PromptEvent) + 'static, ) { raw_regex_prompt( cx, prompt, history_register, completion_fn, move |cx, regex, _, event| fun(cx, regex, event), ); } pub fn raw_regex_prompt( cx: &mut crate::commands::Context, prompt: std::borrow::Cow<'static, str>, history_register: Option, completion_fn: impl FnMut(&Editor, &str) -> Vec + 'static, fun: impl Fn(&mut crate::compositor::Context, rope::Regex, &str, PromptEvent) + 'static, ) { let (view, doc) = current!(cx.editor); let doc_id = view.doc; let snapshot = doc.selection(view.id).clone(); let offset_snapshot = doc.view_offset(view.id); let config = cx.editor.config(); let mut prompt = Prompt::new( prompt, history_register, completion_fn, move |cx: &mut crate::compositor::Context, input: &str, event: PromptEvent| { match event { PromptEvent::Abort => { let (view, doc) = current!(cx.editor); doc.set_selection(view.id, snapshot.clone()); doc.set_view_offset(view.id, offset_snapshot); } PromptEvent::Update | PromptEvent::Validate => { // skip empty input if input.is_empty() { return; } let case_insensitive = if config.search.smart_case { !input.chars().any(char::is_uppercase) } else { false }; match rope::RegexBuilder::new() .syntax( rope::Config::new() .case_insensitive(case_insensitive) .multi_line(true), ) .build(input) { Ok(regex) => { let (view, doc) = current!(cx.editor); // revert state to what it was before the last update doc.set_selection(view.id, snapshot.clone()); if event == PromptEvent::Validate { // Equivalent to push_jump to store selection just before jump view.jumps.push((doc_id, snapshot.clone())); } fun(cx, regex, input, event); let (view, doc) = current!(cx.editor); view.ensure_cursor_in_view(doc, config.scrolloff); } Err(err) => { let (view, doc) = current!(cx.editor); doc.set_selection(view.id, snapshot.clone()); doc.set_view_offset(view.id, offset_snapshot); if event == PromptEvent::Validate { let callback = async move { let call: job::Callback = Callback::EditorCompositor(Box::new( move |_editor: &mut Editor, compositor: &mut Compositor| { let contents = Text::new(format!("{}", err)); let size = compositor.size(); let popup = Popup::new("invalid-regex", contents) .position(Some(helix_core::Position::new( size.height as usize - 2, // 2 = statusline + commandline 0, ))) .auto_close(true); compositor.replace_or_push("invalid-regex", popup); }, )); Ok(call) }; cx.jobs.callback(callback); } } } } } }, ) .with_language("regex", std::sync::Arc::clone(&cx.editor.syn_loader)); // Calculate initial completion prompt.recalculate_completion(cx.editor); // prompt cx.push_layer(Box::new(prompt)); } #[derive(Debug)] pub struct FilePickerData { root: PathBuf, directory_style: Style, } type FilePicker = Picker; pub fn file_picker(editor: &Editor, root: PathBuf) -> FilePicker { use ignore::{types::TypesBuilder, WalkBuilder}; use std::time::Instant; let config = editor.config(); let data = FilePickerData { root: root.clone(), directory_style: editor.theme.get("ui.text.directory"), }; let now = Instant::now(); let dedup_symlinks = config.file_picker.deduplicate_links; let absolute_root = root.canonicalize().unwrap_or_else(|_| root.clone()); let mut walk_builder = WalkBuilder::new(&root); walk_builder .hidden(config.file_picker.hidden) .parents(config.file_picker.parents) .ignore(config.file_picker.ignore) .follow_links(config.file_picker.follow_symlinks) .git_ignore(config.file_picker.git_ignore) .git_global(config.file_picker.git_global) .git_exclude(config.file_picker.git_exclude) .sort_by_file_name(|name1, name2| name1.cmp(name2)) .max_depth(config.file_picker.max_depth) .filter_entry(move |entry| filter_picker_entry(entry, &absolute_root, dedup_symlinks)); walk_builder.add_custom_ignore_filename(helix_loader::config_dir().join("ignore")); walk_builder.add_custom_ignore_filename(".helix/ignore"); // We want to exclude files that the editor can't handle yet let mut type_builder = TypesBuilder::new(); type_builder .add( "compressed", "*.{zip,gz,bz2,zst,lzo,sz,tgz,tbz2,lz,lz4,lzma,lzo,z,Z,xz,7z,rar,cab}", ) .expect("Invalid type definition"); type_builder.negate("all"); let excluded_types = type_builder .build() .expect("failed to build excluded_types"); walk_builder.types(excluded_types); let mut files = walk_builder.build().filter_map(|entry| { let entry = entry.ok()?; if !entry.file_type()?.is_file() { return None; } Some(entry.into_path()) }); log::debug!("file_picker init {:?}", Instant::now().duration_since(now)); let columns = [PickerColumn::new( "path", |item: &PathBuf, data: &FilePickerData| { let path = item.strip_prefix(&data.root).unwrap_or(item); let mut spans = Vec::with_capacity(4); let icons = ICONS.load(); if let Some(icon) = icons.mime().get(Some(&path.to_path_buf()), None) { if let Some(color) = icon.color() { spans.push(Span::styled( format!("{} ", icon.glyph()), Style::default().fg(color), )); } else { spans.push(Span::raw(format!("{} ", icon.glyph()))); } } if let Some(dirs) = path.parent().filter(|p| !p.as_os_str().is_empty()) { spans.extend([ Span::styled(dirs.to_string_lossy(), data.directory_style), Span::styled(std::path::MAIN_SEPARATOR_STR, data.directory_style), ]); } let filename = path .file_name() .expect("normalized paths can't end in `..`") .to_string_lossy(); spans.push(Span::raw(filename)); Spans::from(spans).into() }, )]; let picker = Picker::new(columns, 0, [], data, move |cx, path: &PathBuf, action| { if let Err(e) = cx.editor.open(path, action) { let err = if let Some(err) = e.source() { format!("{}", err) } else { format!("unable to open \"{}\"", path.display()) }; cx.editor.set_error(err); } }) .with_preview(|_editor, path| Some((path.as_path().into(), None))); let injector = picker.injector(); let timeout = std::time::Instant::now() + std::time::Duration::from_millis(30); let mut hit_timeout = false; for file in &mut files { if injector.push(file).is_err() { break; } if std::time::Instant::now() >= timeout { hit_timeout = true; break; } } if hit_timeout { std::thread::spawn(move || { for file in files { if injector.push(file).is_err() { break; } } }); } picker } type FileExplorer = Picker<(PathBuf, bool), (PathBuf, Style)>; pub fn file_explorer(root: PathBuf, editor: &Editor) -> Result { let directory_style = editor.theme.get("ui.text.directory"); let directory_content = directory_content(&root)?; let columns = [PickerColumn::new( "path", |(path, is_dir): &(PathBuf, bool), (root, directory_style): &(PathBuf, Style)| { let name = path.strip_prefix(root).unwrap_or(path).to_string_lossy(); let icons = ICONS.load(); if *is_dir { if let Some(icon) = icons.mime().directory() { Span::styled(format!("{icon} {name}/"), *directory_style).into() } else { Span::styled(format!("{name}/"), *directory_style).into() } } else if let Some(icon) = icons.mime().get(Some(path), None).cloned() { let mut spans = Vec::with_capacity(2); if let Some(color) = icon.color() { let icon = Span::styled(format!("{} ", icon.glyph()), Style::default().fg(color)); let filename = Span::raw(name); spans.push(icon); spans.push(filename); } else { spans.push(Span::raw(format!("{} ", icon.glyph()))); spans.push(Span::raw(name)); } Spans::from(spans).into() } else { name.into() } }, )]; let picker = Picker::new( columns, 0, directory_content, (root, directory_style), move |cx, (path, is_dir): &(PathBuf, bool), action| { if *is_dir { let new_root = helix_stdx::path::normalize(path); let callback = Box::pin(async move { let call: Callback = Callback::EditorCompositor(Box::new(move |editor, compositor| { if let Ok(picker) = file_explorer(new_root, editor) { compositor.push(Box::new(overlay::overlaid(picker))); } })); Ok(call) }); cx.jobs.callback(callback); } else if let Err(e) = cx.editor.open(path, action) { let err = if let Some(err) = e.source() { format!("{}", err) } else { format!("unable to open \"{}\"", path.display()) }; cx.editor.set_error(err); } }, ) .with_preview(|_editor, (path, _is_dir)| Some((path.as_path().into(), None))); Ok(picker) } fn directory_content(path: &Path) -> Result, std::io::Error> { let mut content: Vec<_> = std::fs::read_dir(path)? .flatten() .map(|entry| { ( entry.path(), entry.file_type().is_ok_and(|file_type| file_type.is_dir()), ) }) .collect(); content.sort_by(|(path1, is_dir1), (path2, is_dir2)| (!is_dir1, path1).cmp(&(!is_dir2, path2))); if path.parent().is_some() { content.insert(0, (path.join(".."), true)); } Ok(content) } pub mod completers { use super::Utf8PathBuf; use crate::ui::prompt::Completion; use helix_core::fuzzy::fuzzy_match; use helix_core::syntax::LanguageServerFeature; use helix_view::document::SCRATCH_BUFFER_NAME; use helix_view::theme; use helix_view::{editor::Config, Editor}; use once_cell::sync::Lazy; use std::borrow::Cow; use tui::text::Span; pub type Completer = fn(&Editor, &str) -> Vec; pub fn none(_editor: &Editor, _input: &str) -> Vec { Vec::new() } pub fn buffer(editor: &Editor, input: &str) -> Vec { let names = editor.documents.values().map(|doc| { doc.relative_path() .map(|p| p.display().to_string().into()) .unwrap_or_else(|| Cow::from(SCRATCH_BUFFER_NAME)) }); fuzzy_match(input, names, true) .into_iter() .map(|(name, _)| ((0..), name.into())) .collect() } pub fn theme(_editor: &Editor, input: &str) -> Vec { let mut names = theme::Loader::read_names(&helix_loader::config_dir().join("themes")); for rt_dir in helix_loader::runtime_dirs() { names.extend(theme::Loader::read_names(&rt_dir.join("themes"))); } names.push("default".into()); names.push("base16_default".into()); names.sort(); names.dedup(); fuzzy_match(input, names, false) .into_iter() .map(|(name, _)| ((0..), name.into())) .collect() } /// Recursive function to get all keys from this value and add them to vec fn get_keys(value: &serde_json::Value, vec: &mut Vec, scope: Option<&str>) { if let Some(map) = value.as_object() { for (key, value) in map.iter() { let key = match scope { Some(scope) => format!("{}.{}", scope, key), None => key.clone(), }; get_keys(value, vec, Some(&key)); if !value.is_object() { vec.push(key); } } } } /// Completes names of language servers which are running for the current document. pub fn active_language_servers(editor: &Editor, input: &str) -> Vec { let language_servers = doc!(editor).language_servers().map(|ls| ls.name()); fuzzy_match(input, language_servers, false) .into_iter() .map(|(name, _)| ((0..), Span::raw(name.to_string()))) .collect() } /// Completes names of language servers which are configured for the language of the current /// document. pub fn configured_language_servers(editor: &Editor, input: &str) -> Vec { let language_servers = doc!(editor) .language_config() .into_iter() .flat_map(|config| &config.language_servers) .map(|ls| ls.name.as_str()); fuzzy_match(input, language_servers, false) .into_iter() .map(|(name, _)| ((0..), Span::raw(name.to_string()))) .collect() } pub fn setting(_editor: &Editor, input: &str) -> Vec { static KEYS: Lazy> = Lazy::new(|| { let mut keys = Vec::new(); let json = serde_json::json!(Config::default()); get_keys(&json, &mut keys, None); keys }); fuzzy_match(input, &*KEYS, false) .into_iter() .map(|(name, _)| ((0..), Span::raw(name))) .collect() } pub fn filename(editor: &Editor, input: &str) -> Vec { filename_with_git_ignore(editor, input, true) } pub fn filename_with_git_ignore( editor: &Editor, input: &str, git_ignore: bool, ) -> Vec { filename_impl(editor, input, git_ignore, |entry| { let is_dir = entry.file_type().is_some_and(|entry| entry.is_dir()); if is_dir { FileMatch::AcceptIncomplete } else { FileMatch::Accept } }) } pub fn language(editor: &Editor, input: &str) -> Vec { let text: String = "text".into(); let loader = editor.syn_loader.load(); let language_ids = loader .language_configs() .map(|config| &config.language_id) .chain(std::iter::once(&text)); fuzzy_match(input, language_ids, false) .into_iter() .map(|(name, _)| ((0..), name.to_owned().into())) .collect() } pub fn lsp_workspace_command(editor: &Editor, input: &str) -> Vec { let commands = doc!(editor) .language_servers_with_feature(LanguageServerFeature::WorkspaceCommand) .flat_map(|ls| { ls.capabilities() .execute_command_provider .iter() .flat_map(|options| options.commands.iter()) }); fuzzy_match(input, commands, false) .into_iter() .map(|(name, _)| ((0..), name.to_owned().into())) .collect() } pub fn directory(editor: &Editor, input: &str) -> Vec { directory_with_git_ignore(editor, input, true) } pub fn directory_with_git_ignore( editor: &Editor, input: &str, git_ignore: bool, ) -> Vec { filename_impl(editor, input, git_ignore, |entry| { let is_dir = entry.file_type().is_some_and(|entry| entry.is_dir()); if is_dir { FileMatch::Accept } else { FileMatch::Reject } }) } #[derive(Copy, Clone, PartialEq, Eq)] enum FileMatch { /// Entry should be ignored Reject, /// Entry is usable but can't be the end (for instance if the entry is a directory and we /// try to match a file) AcceptIncomplete, /// Entry is usable and can be the end of the match Accept, } // TODO: we could return an iter/lazy thing so it can fetch as many as it needs. fn filename_impl( editor: &Editor, input: &str, git_ignore: bool, filter_fn: F, ) -> Vec where F: Fn(&ignore::DirEntry) -> FileMatch, { // Rust's filename handling is really annoying. use ignore::WalkBuilder; use std::path::Path; let is_tilde = input == "~"; let path = helix_stdx::path::expand_tilde(Path::new(input)); let (dir, file_name) = if input.ends_with(std::path::MAIN_SEPARATOR) { (path, None) } else { let is_period = (input.ends_with((format!("{}.", std::path::MAIN_SEPARATOR)).as_str()) && input.len() > 2) || input == "."; let file_name = if is_period { Some(String::from(".")) } else { path.file_name() .and_then(|file| file.to_str().map(|path| path.to_owned())) }; let path = if is_period { path } else { match path.parent() { Some(path) if !path.as_os_str().is_empty() => Cow::Borrowed(path), // Path::new("h")'s parent is Some("")... _ => Cow::Owned(helix_stdx::env::current_working_dir()), } }; (path, file_name) }; let end = input.len()..; let files = WalkBuilder::new(&dir) .hidden(false) .follow_links(false) // We're scanning over depth 1 .git_ignore(git_ignore) .max_depth(Some(1)) .build() .filter_map(|file| { file.ok().and_then(|entry| { let fmatch = filter_fn(&entry); if fmatch == FileMatch::Reject { return None; } let is_dir = entry.file_type().is_some_and(|entry| entry.is_dir()); let path = entry.path(); let mut path = if is_tilde { // if it's a single tilde an absolute path is displayed so that when `TAB` is pressed on // one of the directories the tilde will be replaced with a valid path not with a relative // home directory name. // ~ -> -> /home/user // ~/ -> -> ~/first_entry path.to_path_buf() } else { path.strip_prefix(&dir).unwrap_or(path).to_path_buf() }; if fmatch == FileMatch::AcceptIncomplete { path.push(""); } let path = path.into_os_string().into_string().ok()?; Some(Utf8PathBuf { path, is_dir }) }) }) // TODO: unwrap or skip .filter(|path| !path.path.is_empty()); let directory_color = editor.theme.get("ui.text.directory"); let style_from_file = |file: Utf8PathBuf| { if file.is_dir { Span::styled(file.path, directory_color) } else { Span::raw(file.path) } }; // if empty, return a list of dirs and files in current dir if let Some(file_name) = file_name { let range = (input.len().saturating_sub(file_name.len()))..; fuzzy_match(&file_name, files, true) .into_iter() .map(|(name, _)| (range.clone(), style_from_file(name))) .collect() // TODO: complete to longest common match } else { let mut files: Vec<_> = files .map(|file| (end.clone(), style_from_file(file))) .collect(); files.sort_unstable_by(|(_, path1), (_, path2)| path1.content.cmp(&path2.content)); files } } pub fn register(editor: &Editor, input: &str) -> Vec { let iter = editor .registers .iter_preview() // Exclude special registers that shouldn't be written to .filter(|(ch, _)| !matches!(ch, '%' | '#' | '.')) .map(|(ch, _)| ch.to_string()); fuzzy_match(input, iter, false) .into_iter() .map(|(name, _)| ((0..), name.into())) .collect() } }