use futures_util::{stream::FuturesOrdered, FutureExt}; use helix_lsp::{ block_on, lsp::{ self, CodeAction, CodeActionOrCommand, CodeActionTriggerKind, DiagnosticSeverity, NumberOrString, }, util::{diagnostic_to_lsp_diagnostic, lsp_range_to_range, range_to_lsp_range}, Client, LanguageServerId, OffsetEncoding, }; use tokio_stream::StreamExt; use tui::{text::Span, widgets::Row}; use super::{align_view, push_jump, Align, Context, Editor}; use helix_core::{diagnostic::DiagnosticProvider, syntax::LanguageServerFeature, Selection, Uri}; use helix_stdx::path; use helix_view::{editor::Action, handlers::lsp::SignatureHelpInvoked, theme::Style}; use crate::{ compositor::{self, Compositor}, job::Callback, ui::{self, overlay::overlaid, FileLocation, Picker, Popup, PromptEvent}, }; use std::{cmp::Ordering, collections::HashSet, fmt::Display, future::Future, path::Path}; /// Gets the first language server that is attached to a document which supports a specific feature. /// If there is no configured language server that supports the feature, this displays a status message. /// Using this macro in a context where the editor automatically queries the LSP /// (instead of when the user explicitly does so via a keybind like `gd`) /// will spam the "No configured language server supports \" status message confusingly. #[macro_export] macro_rules! language_server_with_feature { ($editor:expr, $doc:expr, $feature:expr) => {{ let language_server = $doc.language_servers_with_feature($feature).next(); match language_server { Some(language_server) => language_server, None => { $editor.set_status(format!( "No configured language server supports {}", $feature )); return; } } }}; } /// A wrapper around `lsp::Location` that swaps out the LSP URI for `helix_core::Uri` and adds /// the server's offset encoding. #[derive(Debug, Clone, PartialEq, Eq)] struct Location { uri: Uri, range: lsp::Range, offset_encoding: OffsetEncoding, } fn lsp_location_to_location( location: lsp::Location, offset_encoding: OffsetEncoding, ) -> Option { let uri = match location.uri.try_into() { Ok(uri) => uri, Err(err) => { log::warn!("discarding invalid or unsupported URI: {err}"); return None; } }; Some(Location { uri, range: location.range, offset_encoding, }) } struct SymbolInformationItem { location: Location, symbol: lsp::SymbolInformation, } struct DiagnosticStyles { hint: Style, info: Style, warning: Style, error: Style, } struct PickerDiagnostic { location: Location, diag: lsp::Diagnostic, } fn location_to_file_location(location: &Location) -> Option { let path = location.uri.as_path()?; let line = Some(( location.range.start.line as usize, location.range.end.line as usize, )); Some((path.into(), line)) } fn jump_to_location(editor: &mut Editor, location: &Location, action: Action) { let (view, doc) = current!(editor); push_jump(view, doc); let Some(path) = location.uri.as_path() else { let err = format!("unable to convert URI to filepath: {:?}", location.uri); editor.set_error(err); return; }; jump_to_position( editor, path, location.range, location.offset_encoding, action, ); } fn jump_to_position( editor: &mut Editor, path: &Path, range: lsp::Range, offset_encoding: OffsetEncoding, action: Action, ) { let doc = match editor.open(path, action) { Ok(id) => doc_mut!(editor, &id), Err(err) => { let err = format!("failed to open path: {:?}: {:?}", path, err); editor.set_error(err); return; } }; let view = view_mut!(editor); // TODO: convert inside server let new_range = if let Some(new_range) = lsp_range_to_range(doc.text(), range, offset_encoding) { new_range } else { log::warn!("lsp position out of bounds - {:?}", range); return; }; // we flip the range so that the cursor sits on the start of the symbol // (for example start of the function). doc.set_selection(view.id, Selection::single(new_range.head, new_range.anchor)); if action.align_view(view, doc.id()) { align_view(doc, view, Align::Center); } } fn display_symbol_kind(kind: lsp::SymbolKind) -> &'static str { match kind { lsp::SymbolKind::FILE => "file", lsp::SymbolKind::MODULE => "module", lsp::SymbolKind::NAMESPACE => "namespace", lsp::SymbolKind::PACKAGE => "package", lsp::SymbolKind::CLASS => "class", lsp::SymbolKind::METHOD => "method", lsp::SymbolKind::PROPERTY => "property", lsp::SymbolKind::FIELD => "field", lsp::SymbolKind::CONSTRUCTOR => "construct", lsp::SymbolKind::ENUM => "enum", lsp::SymbolKind::INTERFACE => "interface", lsp::SymbolKind::FUNCTION => "function", lsp::SymbolKind::VARIABLE => "variable", lsp::SymbolKind::CONSTANT => "constant", lsp::SymbolKind::STRING => "string", lsp::SymbolKind::NUMBER => "number", lsp::SymbolKind::BOOLEAN => "boolean", lsp::SymbolKind::ARRAY => "array", lsp::SymbolKind::OBJECT => "object", lsp::SymbolKind::KEY => "key", lsp::SymbolKind::NULL => "null", lsp::SymbolKind::ENUM_MEMBER => "enummem", lsp::SymbolKind::STRUCT => "struct", lsp::SymbolKind::EVENT => "event", lsp::SymbolKind::OPERATOR => "operator", lsp::SymbolKind::TYPE_PARAMETER => "typeparam", _ => { log::warn!("Unknown symbol kind: {:?}", kind); "" } } } #[derive(Copy, Clone, PartialEq)] enum DiagnosticsFormat { ShowSourcePath, HideSourcePath, } type DiagnosticsPicker = Picker; fn diag_picker( cx: &Context, diagnostics: impl IntoIterator)>, format: DiagnosticsFormat, ) -> DiagnosticsPicker { // TODO: drop current_path comparison and instead use workspace: bool flag? // flatten the map to a vec of (url, diag) pairs let mut flat_diag = Vec::new(); for (uri, diags) in diagnostics { flat_diag.reserve(diags.len()); for (diag, provider) in diags { if let Some(ls) = provider .language_server_id() .and_then(|id| cx.editor.language_server_by_id(id)) { flat_diag.push(PickerDiagnostic { location: Location { uri: uri.clone(), range: diag.range, offset_encoding: ls.offset_encoding(), }, diag, }); } } } let styles = DiagnosticStyles { hint: cx.editor.theme.get("hint"), info: cx.editor.theme.get("info"), warning: cx.editor.theme.get("warning"), error: cx.editor.theme.get("error"), }; let mut columns = vec![ ui::PickerColumn::new( "severity", |item: &PickerDiagnostic, styles: &DiagnosticStyles| { match item.diag.severity { Some(DiagnosticSeverity::HINT) => Span::styled("HINT", styles.hint), Some(DiagnosticSeverity::INFORMATION) => Span::styled("INFO", styles.info), Some(DiagnosticSeverity::WARNING) => Span::styled("WARN", styles.warning), Some(DiagnosticSeverity::ERROR) => Span::styled("ERROR", styles.error), _ => Span::raw(""), } .into() }, ), ui::PickerColumn::new("code", |item: &PickerDiagnostic, _| { match item.diag.code.as_ref() { Some(NumberOrString::Number(n)) => n.to_string().into(), Some(NumberOrString::String(s)) => s.as_str().into(), None => "".into(), } }), ui::PickerColumn::new("message", |item: &PickerDiagnostic, _| { item.diag.message.as_str().into() }), ]; let mut primary_column = 2; // message if format == DiagnosticsFormat::ShowSourcePath { columns.insert( // between message code and message 2, ui::PickerColumn::new("path", |item: &PickerDiagnostic, _| { if let Some(path) = item.location.uri.as_path() { path::get_truncated_path(path) .to_string_lossy() .to_string() .into() } else { Default::default() } }), ); primary_column += 1; } Picker::new( columns, primary_column, flat_diag, styles, move |cx, diag, action| { jump_to_location(cx.editor, &diag.location, action); let (view, doc) = current!(cx.editor); view.diagnostics_handler .immediately_show_diagnostic(doc, view.id); }, ) .with_preview(move |_editor, diag| location_to_file_location(&diag.location)) .truncate_start(false) } pub fn symbol_picker(cx: &mut Context) { fn nested_to_flat( list: &mut Vec, file: &lsp::TextDocumentIdentifier, uri: &Uri, symbol: lsp::DocumentSymbol, offset_encoding: OffsetEncoding, ) { #[allow(deprecated)] list.push(SymbolInformationItem { symbol: lsp::SymbolInformation { name: symbol.name, kind: symbol.kind, tags: symbol.tags, deprecated: symbol.deprecated, location: lsp::Location::new(file.uri.clone(), symbol.selection_range), container_name: None, }, location: Location { uri: uri.clone(), range: symbol.selection_range, offset_encoding, }, }); for child in symbol.children.into_iter().flatten() { nested_to_flat(list, file, uri, child, offset_encoding); } } let doc = doc!(cx.editor); let mut seen_language_servers = HashSet::new(); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(LanguageServerFeature::DocumentSymbols) .filter(|ls| seen_language_servers.insert(ls.id())) .map(|language_server| { let request = language_server.document_symbols(doc.identifier()).unwrap(); let offset_encoding = language_server.offset_encoding(); let doc_id = doc.identifier(); let doc_uri = doc .uri() .expect("docs with active language servers must be backed by paths"); async move { let symbols = match request.await? { Some(symbols) => symbols, None => return anyhow::Ok(vec![]), }; // lsp has two ways to represent symbols (flat/nested) // convert the nested variant to flat, so that we have a homogeneous list let symbols = match symbols { lsp::DocumentSymbolResponse::Flat(symbols) => symbols .into_iter() .map(|symbol| SymbolInformationItem { location: Location { uri: doc_uri.clone(), range: symbol.location.range, offset_encoding, }, symbol, }) .collect(), lsp::DocumentSymbolResponse::Nested(symbols) => { let mut flat_symbols = Vec::new(); for symbol in symbols { nested_to_flat( &mut flat_symbols, &doc_id, &doc_uri, symbol, offset_encoding, ) } flat_symbols } }; Ok(symbols) } }) .collect(); if futures.is_empty() { cx.editor .set_error("No configured language server supports document symbols"); return; } cx.jobs.callback(async move { let mut symbols = Vec::new(); while let Some(response) = futures.next().await { match response { Ok(mut items) => symbols.append(&mut items), Err(err) => log::error!("Error requesting document symbols: {err}"), } } let call = move |_editor: &mut Editor, compositor: &mut Compositor| { let columns = [ ui::PickerColumn::new("kind", |item: &SymbolInformationItem, _| { display_symbol_kind(item.symbol.kind).into() }), // Some symbols in the document symbol picker may have a URI that isn't // the current file. It should be rare though, so we concatenate that // URI in with the symbol name in this picker. ui::PickerColumn::new("name", |item: &SymbolInformationItem, _| { item.symbol.name.as_str().into() }), ui::PickerColumn::new("container", |item: &SymbolInformationItem, _| { item.symbol .container_name .as_deref() .unwrap_or_default() .into() }), ]; let picker = Picker::new( columns, 1, // name column symbols, (), move |cx, item, action| { jump_to_location(cx.editor, &item.location, action); }, ) .with_preview(move |_editor, item| location_to_file_location(&item.location)) .truncate_start(false); compositor.push(Box::new(overlaid(picker))) }; Ok(Callback::EditorCompositor(Box::new(call))) }); } pub fn workspace_symbol_picker(cx: &mut Context) { use crate::ui::picker::Injector; let doc = doc!(cx.editor); if doc .language_servers_with_feature(LanguageServerFeature::WorkspaceSymbols) .count() == 0 { cx.editor .set_error("No configured language server supports workspace symbols"); return; } let get_symbols = |pattern: &str, editor: &mut Editor, _data, injector: &Injector<_, _>| { let doc = doc!(editor); let mut seen_language_servers = HashSet::new(); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(LanguageServerFeature::WorkspaceSymbols) .filter(|ls| seen_language_servers.insert(ls.id())) .map(|language_server| { let request = language_server .workspace_symbols(pattern.to_string()) .unwrap(); let offset_encoding = language_server.offset_encoding(); async move { let symbols = request .await? .and_then(|resp| match resp { lsp::WorkspaceSymbolResponse::Flat(symbols) => Some(symbols), lsp::WorkspaceSymbolResponse::Nested(_) => None, }) .unwrap_or_default(); let response: Vec<_> = symbols .into_iter() .filter_map(|symbol| { let uri = match Uri::try_from(&symbol.location.uri) { Ok(uri) => uri, Err(err) => { log::warn!("discarding symbol with invalid URI: {err}"); return None; } }; Some(SymbolInformationItem { location: Location { uri, range: symbol.location.range, offset_encoding, }, symbol, }) }) .collect(); anyhow::Ok(response) } }) .collect(); if futures.is_empty() { editor.set_error("No configured language server supports workspace symbols"); } let injector = injector.clone(); async move { while let Some(response) = futures.next().await { match response { Ok(items) => { for item in items { injector.push(item)?; } } Err(err) => log::error!("Error requesting workspace symbols: {err}"), } } Ok(()) } .boxed() }; let columns = [ ui::PickerColumn::new("kind", |item: &SymbolInformationItem, _| { display_symbol_kind(item.symbol.kind).into() }), ui::PickerColumn::new("name", |item: &SymbolInformationItem, _| { item.symbol.name.as_str().into() }) .without_filtering(), ui::PickerColumn::new("container", |item: &SymbolInformationItem, _| { item.symbol .container_name .as_deref() .unwrap_or_default() .into() }), ui::PickerColumn::new("path", |item: &SymbolInformationItem, _| { if let Some(path) = item.location.uri.as_path() { path::get_relative_path(path) .to_string_lossy() .to_string() .into() } else { item.symbol.location.uri.to_string().into() } }), ]; let picker = Picker::new( columns, 1, // name column [], (), move |cx, item, action| { jump_to_location(cx.editor, &item.location, action); }, ) .with_preview(|_editor, item| location_to_file_location(&item.location)) .with_dynamic_query(get_symbols, None) .truncate_start(false); cx.push_layer(Box::new(overlaid(picker))); } pub fn diagnostics_picker(cx: &mut Context) { let doc = doc!(cx.editor); if let Some(uri) = doc.uri() { let diagnostics = cx.editor.diagnostics.get(&uri).cloned().unwrap_or_default(); let picker = diag_picker(cx, [(uri, diagnostics)], DiagnosticsFormat::HideSourcePath); cx.push_layer(Box::new(overlaid(picker))); } } pub fn workspace_diagnostics_picker(cx: &mut Context) { // TODO not yet filtered by LanguageServerFeature, need to do something similar as Document::shown_diagnostics here for all open documents let diagnostics = cx.editor.diagnostics.clone(); let picker = diag_picker(cx, diagnostics, DiagnosticsFormat::ShowSourcePath); cx.push_layer(Box::new(overlaid(picker))); } struct CodeActionOrCommandItem { lsp_item: lsp::CodeActionOrCommand, language_server_id: LanguageServerId, } impl ui::menu::Item for CodeActionOrCommandItem { type Data = (); fn format(&self, _data: &Self::Data) -> Row { match &self.lsp_item { lsp::CodeActionOrCommand::CodeAction(action) => action.title.as_str().into(), lsp::CodeActionOrCommand::Command(command) => command.title.as_str().into(), } } } /// Determines the category of the `CodeAction` using the `CodeAction::kind` field. /// Returns a number that represent these categories. /// Categories with a lower number should be displayed first. /// /// /// While the `kind` field is defined as open ended in the LSP spec (any value may be used) /// in practice a closed set of common values (mostly suggested in the LSP spec) are used. /// VSCode displays each of these categories separately (separated by a heading in the codeactions picker) /// to make them easier to navigate. Helix does not display these headings to the user. /// However it does sort code actions by their categories to achieve the same order as the VScode picker, /// just without the headings. /// /// The order used here is modeled after the [vscode sourcecode](https://github.com/microsoft/vscode/blob/eaec601dd69aeb4abb63b9601a6f44308c8d8c6e/src/vs/editor/contrib/codeAction/browser/codeActionWidget.ts>) fn action_category(action: &CodeActionOrCommand) -> u32 { if let CodeActionOrCommand::CodeAction(CodeAction { kind: Some(kind), .. }) = action { let mut components = kind.as_str().split('.'); match components.next() { Some("quickfix") => 0, Some("refactor") => match components.next() { Some("extract") => 1, Some("inline") => 2, Some("rewrite") => 3, Some("move") => 4, Some("surround") => 5, _ => 7, }, Some("source") => 6, _ => 7, } } else { 7 } } fn action_preferred(action: &CodeActionOrCommand) -> bool { matches!( action, CodeActionOrCommand::CodeAction(CodeAction { is_preferred: Some(true), .. }) ) } fn action_fixes_diagnostics(action: &CodeActionOrCommand) -> bool { matches!( action, CodeActionOrCommand::CodeAction(CodeAction { diagnostics: Some(diagnostics), .. }) if !diagnostics.is_empty() ) } pub fn code_action(cx: &mut Context) { let (view, doc) = current!(cx.editor); let selection_range = doc.selection(view.id).primary(); let mut seen_language_servers = HashSet::new(); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(LanguageServerFeature::CodeAction) .filter(|ls| seen_language_servers.insert(ls.id())) // TODO this should probably already been filtered in something like "language_servers_with_feature" .filter_map(|language_server| { let offset_encoding = language_server.offset_encoding(); let language_server_id = language_server.id(); let range = range_to_lsp_range(doc.text(), selection_range, offset_encoding); // Filter and convert overlapping diagnostics let code_action_context = lsp::CodeActionContext { diagnostics: doc .diagnostics() .iter() .filter(|&diag| { selection_range .overlaps(&helix_core::Range::new(diag.range.start, diag.range.end)) }) .map(|diag| diagnostic_to_lsp_diagnostic(doc.text(), diag, offset_encoding)) .collect(), only: None, trigger_kind: Some(CodeActionTriggerKind::INVOKED), }; let code_action_request = language_server.code_actions(doc.identifier(), range, code_action_context)?; Some((code_action_request, language_server_id)) }) .map(|(request, ls_id)| async move { let Some(mut actions) = request.await? else { return anyhow::Ok(Vec::new()); }; // remove disabled code actions actions.retain(|action| { matches!( action, CodeActionOrCommand::Command(_) | CodeActionOrCommand::CodeAction(CodeAction { disabled: None, .. }) ) }); // Sort codeactions into a useful order. This behaviour is only partially described in the LSP spec. // Many details are modeled after vscode because language servers are usually tested against it. // VScode sorts the codeaction two times: // // First the codeactions that fix some diagnostics are moved to the front. // If both codeactions fix some diagnostics (or both fix none) the codeaction // that is marked with `is_preferred` is shown first. The codeactions are then shown in separate // submenus that only contain a certain category (see `action_category`) of actions. // // Below this done in in a single sorting step actions.sort_by(|action1, action2| { // sort actions by category let order = action_category(action1).cmp(&action_category(action2)); if order != Ordering::Equal { return order; } // within the categories sort by relevancy. // Modeled after the `codeActionsComparator` function in vscode: // https://github.com/microsoft/vscode/blob/eaec601dd69aeb4abb63b9601a6f44308c8d8c6e/src/vs/editor/contrib/codeAction/browser/codeAction.ts // if one code action fixes a diagnostic but the other one doesn't show it first let order = action_fixes_diagnostics(action1) .cmp(&action_fixes_diagnostics(action2)) .reverse(); if order != Ordering::Equal { return order; } // if one of the codeactions is marked as preferred show it first // otherwise keep the original LSP sorting action_preferred(action1) .cmp(&action_preferred(action2)) .reverse() }); Ok(actions .into_iter() .map(|lsp_item| CodeActionOrCommandItem { lsp_item, language_server_id: ls_id, }) .collect()) }) .collect(); if futures.is_empty() { cx.editor .set_error("No configured language server supports code actions"); return; } cx.jobs.callback(async move { let mut actions = Vec::new(); while let Some(output) = futures.next().await { match output { Ok(mut lsp_items) => actions.append(&mut lsp_items), Err(err) => log::error!("while gathering code actions: {err}"), } } let call = move |editor: &mut Editor, compositor: &mut Compositor| { if actions.is_empty() { editor.set_error("No code actions available"); return; } let mut picker = ui::Menu::new(actions, (), move |editor, action, event| { if event != PromptEvent::Validate { return; } // always present here let action = action.unwrap(); let Some(language_server) = editor.language_server_by_id(action.language_server_id) else { editor.set_error("Language Server disappeared"); return; }; let offset_encoding = language_server.offset_encoding(); match &action.lsp_item { lsp::CodeActionOrCommand::Command(command) => { log::debug!("code action command: {:?}", command); editor.execute_lsp_command(command.clone(), action.language_server_id); } lsp::CodeActionOrCommand::CodeAction(code_action) => { log::debug!("code action: {:?}", code_action); // we support lsp "codeAction/resolve" for `edit` and `command` fields let mut resolved_code_action = None; if code_action.edit.is_none() || code_action.command.is_none() { if let Some(future) = language_server.resolve_code_action(code_action) { if let Ok(code_action) = helix_lsp::block_on(future) { resolved_code_action = Some(code_action); } } } let resolved_code_action = resolved_code_action.as_ref().unwrap_or(code_action); if let Some(ref workspace_edit) = resolved_code_action.edit { let _ = editor.apply_workspace_edit(offset_encoding, workspace_edit); } // if code action provides both edit and command first the edit // should be applied and then the command if let Some(command) = &code_action.command { editor.execute_lsp_command(command.clone(), action.language_server_id); } } } }); picker.move_down(); // pre-select the first item let popup = Popup::new("code-action", picker).with_scrollbar(false); compositor.replace_or_push("code-action", popup); }; Ok(Callback::EditorCompositor(Box::new(call))) }); } #[derive(Debug)] pub struct ApplyEditError { pub kind: ApplyEditErrorKind, pub failed_change_idx: usize, } #[derive(Debug)] pub enum ApplyEditErrorKind { DocumentChanged, FileNotFound, UnknownURISchema, IoError(std::io::Error), // TODO: check edits before applying and propagate failure // InvalidEdit, } impl Display for ApplyEditErrorKind { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { ApplyEditErrorKind::DocumentChanged => f.write_str("document has changed"), ApplyEditErrorKind::FileNotFound => f.write_str("file not found"), ApplyEditErrorKind::UnknownURISchema => f.write_str("URI schema not supported"), ApplyEditErrorKind::IoError(err) => f.write_str(&format!("{err}")), } } } /// Precondition: `locations` should be non-empty. fn goto_impl(editor: &mut Editor, compositor: &mut Compositor, locations: Vec) { let cwdir = helix_stdx::env::current_working_dir(); match locations.as_slice() { [location] => { jump_to_location(editor, location, Action::Replace); } [] => unreachable!("`locations` should be non-empty for `goto_impl`"), _locations => { let columns = [ui::PickerColumn::new( "location", |item: &Location, cwdir: &std::path::PathBuf| { let path = if let Some(path) = item.uri.as_path() { path.strip_prefix(cwdir).unwrap_or(path).to_string_lossy() } else { item.uri.to_string().into() }; format!("{path}:{}", item.range.start.line + 1).into() }, )]; let picker = Picker::new(columns, 0, locations, cwdir, |cx, location, action| { jump_to_location(cx.editor, location, action) }) .with_preview(|_editor, location| location_to_file_location(location)); compositor.push(Box::new(overlaid(picker))); } } } fn goto_single_impl(cx: &mut Context, feature: LanguageServerFeature, request_provider: P) where P: Fn(&Client, lsp::Position, lsp::TextDocumentIdentifier) -> Option, F: Future>> + 'static + Send, { let (view, doc) = current_ref!(cx.editor); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(feature) .map(|language_server| { let offset_encoding = language_server.offset_encoding(); let pos = doc.position(view.id, offset_encoding); let future = request_provider(language_server, pos, doc.identifier()).unwrap(); async move { anyhow::Ok((future.await?, offset_encoding)) } }) .collect(); cx.jobs.callback(async move { let mut locations = Vec::new(); while let Some(response) = futures.next().await { match response { Ok((response, offset_encoding)) => match response { Some(lsp::GotoDefinitionResponse::Scalar(lsp_location)) => { locations.extend(lsp_location_to_location(lsp_location, offset_encoding)); } Some(lsp::GotoDefinitionResponse::Array(lsp_locations)) => { locations.extend(lsp_locations.into_iter().flat_map(|location| { lsp_location_to_location(location, offset_encoding) })); } Some(lsp::GotoDefinitionResponse::Link(lsp_locations)) => { locations.extend( lsp_locations .into_iter() .map(|location_link| { lsp::Location::new( location_link.target_uri, location_link.target_range, ) }) .flat_map(|location| { lsp_location_to_location(location, offset_encoding) }), ); } None => (), }, Err(err) => log::error!("Error requesting locations: {err}"), } } let call = move |editor: &mut Editor, compositor: &mut Compositor| { if locations.is_empty() { editor.set_error("No definition found."); } else { goto_impl(editor, compositor, locations); } }; Ok(Callback::EditorCompositor(Box::new(call))) }); } pub fn goto_declaration(cx: &mut Context) { goto_single_impl( cx, LanguageServerFeature::GotoDeclaration, |ls, pos, doc_id| ls.goto_declaration(doc_id, pos, None), ); } pub fn goto_definition(cx: &mut Context) { goto_single_impl( cx, LanguageServerFeature::GotoDefinition, |ls, pos, doc_id| ls.goto_definition(doc_id, pos, None), ); } pub fn goto_type_definition(cx: &mut Context) { goto_single_impl( cx, LanguageServerFeature::GotoTypeDefinition, |ls, pos, doc_id| ls.goto_type_definition(doc_id, pos, None), ); } pub fn goto_implementation(cx: &mut Context) { goto_single_impl( cx, LanguageServerFeature::GotoImplementation, |ls, pos, doc_id| ls.goto_implementation(doc_id, pos, None), ); } pub fn goto_reference(cx: &mut Context) { let config = cx.editor.config(); let (view, doc) = current_ref!(cx.editor); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(LanguageServerFeature::GotoReference) .map(|language_server| { let offset_encoding = language_server.offset_encoding(); let pos = doc.position(view.id, offset_encoding); let future = language_server .goto_reference( doc.identifier(), pos, config.lsp.goto_reference_include_declaration, None, ) .unwrap(); async move { anyhow::Ok((future.await?, offset_encoding)) } }) .collect(); cx.jobs.callback(async move { let mut locations = Vec::new(); while let Some(response) = futures.next().await { match response { Ok((lsp_locations, offset_encoding)) => locations.extend( lsp_locations .into_iter() .flatten() .flat_map(|location| lsp_location_to_location(location, offset_encoding)), ), Err(err) => log::error!("Error requesting references: {err}"), } } let call = move |editor: &mut Editor, compositor: &mut Compositor| { if locations.is_empty() { editor.set_error("No references found."); } else { goto_impl(editor, compositor, locations); } }; Ok(Callback::EditorCompositor(Box::new(call))) }); } pub fn signature_help(cx: &mut Context) { cx.editor .handlers .trigger_signature_help(SignatureHelpInvoked::Manual, cx.editor) } pub fn hover(cx: &mut Context) { use ui::lsp::hover::Hover; let (view, doc) = current!(cx.editor); if doc .language_servers_with_feature(LanguageServerFeature::Hover) .count() == 0 { cx.editor .set_error("No configured language server supports hover"); return; } let mut seen_language_servers = HashSet::new(); let mut futures: FuturesOrdered<_> = doc .language_servers_with_feature(LanguageServerFeature::Hover) .filter(|ls| seen_language_servers.insert(ls.id())) .map(|language_server| { let server_name = language_server.name().to_string(); // TODO: factor out a doc.position_identifier() that returns lsp::TextDocumentPositionIdentifier let pos = doc.position(view.id, language_server.offset_encoding()); let request = language_server .text_document_hover(doc.identifier(), pos, None) .unwrap(); async move { anyhow::Ok((server_name, request.await?)) } }) .collect(); cx.jobs.callback(async move { let mut hovers: Vec<(String, lsp::Hover)> = Vec::new(); while let Some(response) = futures.next().await { match response { Ok((server_name, Some(hover))) => hovers.push((server_name, hover)), Ok(_) => (), Err(err) => log::error!("Error requesting hover: {err}"), } } let call = move |editor: &mut Editor, compositor: &mut Compositor| { if hovers.is_empty() { editor.set_status("No hover results available."); return; } // create new popup let contents = Hover::new(hovers, editor.syn_loader.clone()); let popup = Popup::new(Hover::ID, contents).auto_close(true); compositor.replace_or_push(Hover::ID, popup); }; Ok(Callback::EditorCompositor(Box::new(call))) }); } pub fn rename_symbol(cx: &mut Context) { fn get_prefill_from_word_boundary(editor: &Editor) -> String { let (view, doc) = current_ref!(editor); let text = doc.text().slice(..); let primary_selection = doc.selection(view.id).primary(); if primary_selection.len() > 1 { primary_selection } else { use helix_core::textobject::{textobject_word, TextObject}; textobject_word(text, primary_selection, TextObject::Inside, 1, false) } .fragment(text) .into() } fn get_prefill_from_lsp_response( editor: &Editor, offset_encoding: OffsetEncoding, response: Option, ) -> Result { match response { Some(lsp::PrepareRenameResponse::Range(range)) => { let text = doc!(editor).text(); Ok(lsp_range_to_range(text, range, offset_encoding) .ok_or("lsp sent invalid selection range for rename")? .fragment(text.slice(..)) .into()) } Some(lsp::PrepareRenameResponse::RangeWithPlaceholder { placeholder, .. }) => { Ok(placeholder) } Some(lsp::PrepareRenameResponse::DefaultBehavior { .. }) => { Ok(get_prefill_from_word_boundary(editor)) } None => Err("lsp did not respond to prepare rename request"), } } fn create_rename_prompt( editor: &Editor, prefill: String, history_register: Option, language_server_id: Option, ) -> Box { let prompt = ui::Prompt::new( "rename-to:".into(), history_register, ui::completers::none, move |cx: &mut compositor::Context, input: &str, event: PromptEvent| { if event != PromptEvent::Validate { return; } let (view, doc) = current!(cx.editor); let Some(language_server) = doc .language_servers_with_feature(LanguageServerFeature::RenameSymbol) .find(|ls| language_server_id.map_or(true, |id| id == ls.id())) else { cx.editor .set_error("No configured language server supports symbol renaming"); return; }; let offset_encoding = language_server.offset_encoding(); let pos = doc.position(view.id, offset_encoding); let future = language_server .rename_symbol(doc.identifier(), pos, input.to_string()) .unwrap(); match block_on(future) { Ok(edits) => { let _ = cx .editor .apply_workspace_edit(offset_encoding, &edits.unwrap_or_default()); } Err(err) => cx.editor.set_error(err.to_string()), } }, ) .with_line(prefill, editor); Box::new(prompt) } let (view, doc) = current_ref!(cx.editor); let history_register = cx.register; if doc .language_servers_with_feature(LanguageServerFeature::RenameSymbol) .next() .is_none() { cx.editor .set_error("No configured language server supports symbol renaming"); return; } let language_server_with_prepare_rename_support = doc .language_servers_with_feature(LanguageServerFeature::RenameSymbol) .find(|ls| { matches!( ls.capabilities().rename_provider, Some(lsp::OneOf::Right(lsp::RenameOptions { prepare_provider: Some(true), .. })) ) }); if let Some(language_server) = language_server_with_prepare_rename_support { let ls_id = language_server.id(); let offset_encoding = language_server.offset_encoding(); let pos = doc.position(view.id, offset_encoding); let future = language_server .prepare_rename(doc.identifier(), pos) .unwrap(); cx.callback( future, move |editor, compositor, response: Option| { let prefill = match get_prefill_from_lsp_response(editor, offset_encoding, response) { Ok(p) => p, Err(e) => { editor.set_error(e); return; } }; let prompt = create_rename_prompt(editor, prefill, history_register, Some(ls_id)); compositor.push(prompt); }, ); } else { let prefill = get_prefill_from_word_boundary(cx.editor); let prompt = create_rename_prompt(cx.editor, prefill, history_register, None); cx.push_layer(prompt); } } pub fn select_references_to_symbol_under_cursor(cx: &mut Context) { let (view, doc) = current!(cx.editor); let language_server = language_server_with_feature!(cx.editor, doc, LanguageServerFeature::DocumentHighlight); let offset_encoding = language_server.offset_encoding(); let pos = doc.position(view.id, offset_encoding); let future = language_server .text_document_document_highlight(doc.identifier(), pos, None) .unwrap(); cx.callback( future, move |editor, _compositor, response: Option>| { let document_highlights = match response { Some(highlights) if !highlights.is_empty() => highlights, _ => return, }; let (view, doc) = current!(editor); let text = doc.text(); let pos = doc.selection(view.id).primary().cursor(text.slice(..)); // We must find the range that contains our primary cursor to prevent our primary cursor to move let mut primary_index = 0; let ranges = document_highlights .iter() .filter_map(|highlight| lsp_range_to_range(text, highlight.range, offset_encoding)) .enumerate() .map(|(i, range)| { if range.contains(pos) { primary_index = i; } range }) .collect(); let selection = Selection::new(ranges, primary_index); doc.set_selection(view.id, selection); }, ); }