From 8cfa56b643de04c02b080ca7e8562baf38d6e774 Mon Sep 17 00:00:00 2001 From: Nik Revenco <154856872+NikitaRevenco@users.noreply.github.com> Date: Tue, 18 Mar 2025 16:59:01 +0000 Subject: [PATCH] feat: implement basic virtual text (end of line blame) --- helix-term/src/handlers/blame.rs | 36 +++++++++++++++++++++----------- helix-vcs/src/git.rs | 2 +- helix-view/src/document.rs | 2 ++ helix-view/src/view.rs | 8 +++++++ 4 files changed, 35 insertions(+), 13 deletions(-) diff --git a/helix-term/src/handlers/blame.rs b/helix-term/src/handlers/blame.rs index 50f1e842..e733f953 100644 --- a/helix-term/src/handlers/blame.rs +++ b/helix-term/src/handlers/blame.rs @@ -1,4 +1,4 @@ -#![allow(dead_code, unused_variables)] +use helix_core::text_annotations::InlineAnnotation; use helix_event::{register_hook, send_blocking}; use helix_view::{ handlers::{BlameEvent, Handlers}, @@ -14,14 +14,15 @@ impl helix_event::AsyncHook for BlameHandler { fn handle_event( &mut self, - event: Self::Event, - timeout: Option, + _event: Self::Event, + _timeout: Option, ) -> Option { self.finish_debounce(); None } fn finish_debounce(&mut self) { + // TODO: this blocks on the main thread. Figure out how not to do that job::dispatch_blocking(move |editor, _| { request_git_blame(editor); }) @@ -40,22 +41,33 @@ pub(super) fn register_hooks(handlers: &Handlers) { } fn request_git_blame(editor: &mut Editor) { - let (view, doc) = current_ref!(editor); + let blame_enabled = editor.config().vcs.blame; + let (view, doc) = current!(editor); let text = doc.text(); let selection = doc.selection(view.id); let Some(file) = doc.path() else { return; }; - let Ok(cursor_line) = TryInto::::try_into( - text.char_to_line(selection.primary().cursor(doc.text().slice(..))), - ) else { + if !blame_enabled { + return; + } + + let cursor_lin = text.char_to_line(selection.primary().cursor(doc.text().slice(..))); + let Ok(cursor_line) = TryInto::::try_into(cursor_lin) else { return; }; - let output = editor.diff_providers.blame_line(file, cursor_line); + // gix-blame expects a 1-based line + let Ok(output) = editor.diff_providers.blame_line(file, cursor_line + 1) else { + return; + }; - match output { - Ok(blame) => editor.set_status(blame.to_string()), - Err(err) => editor.set_error(err.to_string()), - } + doc.blame = Some(vec![InlineAnnotation::new( + text.try_line_to_char(cursor_lin + 1) + .unwrap_or(text.len_chars()) + // to get the last position in the current line + - 1, + output.to_string(), + )]); + log::error!("{:?}", doc.blame); } diff --git a/helix-vcs/src/git.rs b/helix-vcs/src/git.rs index dcc30fbf..1ea004d8 100644 --- a/helix-vcs/src/git.rs +++ b/helix-vcs/src/git.rs @@ -137,7 +137,7 @@ impl fmt::Display for BlameInformation { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!( f, - "{} - {} - {} - {}", + " {} - {} - {} - {}", self.author_name, self.commit_date, self.commit_message, self.commit_hash ) } diff --git a/helix-view/src/document.rs b/helix-view/src/document.rs index 06a708f0..46d84b7e 100644 --- a/helix-view/src/document.rs +++ b/helix-view/src/document.rs @@ -143,6 +143,7 @@ pub struct Document { /// /// To know if they're up-to-date, check the `id` field in `DocumentInlayHints`. pub(crate) inlay_hints: HashMap, + pub blame: Option>, pub(crate) jump_labels: HashMap>, /// Set to `true` when the document is updated, reset to `false` on the next inlay hints /// update from the LSP @@ -698,6 +699,7 @@ impl Document { focused_at: std::time::Instant::now(), readonly: false, jump_labels: HashMap::new(), + blame: None, } } diff --git a/helix-view/src/view.rs b/helix-view/src/view.rs index a229f01e..a45f74a7 100644 --- a/helix-view/src/view.rs +++ b/helix-view/src/view.rs @@ -452,6 +452,14 @@ impl View { text_annotations.add_overlay(labels, style); } + if let Some(blame_annotation) = &doc.blame { + let annotation_style = theme + .and_then(|t| t.find_scope_index("ui.virtual.inlay-hint.type")) + .map(Highlight); + + text_annotations.add_inline_annotations(blame_annotation, annotation_style); + } + if let Some(DocumentInlayHints { id: _, type_inlay_hints,