diff --git a/helix-term/src/application.rs b/helix-term/src/application.rs index e1a622f9d06b5..6e15932e2802e 100644 --- a/helix-term/src/application.rs +++ b/helix-term/src/application.rs @@ -197,7 +197,7 @@ impl Application { // NOTE: this isn't necessarily true anymore. If // `--vsplit` or `--hsplit` are used, the file which is // opened last is focused on. - let view_id = editor.tree.focus; + let view_id = editor.tabs.curr_tree().focus; let doc = doc_mut!(editor, &doc_id); let pos = Selection::point(pos_at_coords(doc.text().slice(..), pos, true)); doc.set_selection(view_id, pos); @@ -383,7 +383,7 @@ impl Application { // reset view position in case softwrap was enabled/disabled let scrolloff = self.editor.config().scrolloff; - for (view, _) in self.editor.tree.views_mut() { + for (view, _) in self.editor.tabs.curr_tree_mut().views_mut() { let doc = &self.editor.documents[&view.doc]; view.ensure_cursor_in_view(doc, scrolloff) } diff --git a/helix-term/src/commands.rs b/helix-term/src/commands.rs index 1bd736523edcf..e68e61561a7a4 100644 --- a/helix-term/src/commands.rs +++ b/helix-term/src/commands.rs @@ -40,7 +40,7 @@ use helix_view::{ keyboard::KeyCode, tree, view::View, - Document, DocumentId, Editor, ViewId, + Document, DocumentId, Editor, TabId, ViewId, }; use anyhow::{anyhow, bail, ensure, Context as _}; @@ -307,6 +307,7 @@ impl MappableCommand { file_picker_in_current_buffer_directory, "Open file picker at current buffers's directory", file_picker_in_current_directory, "Open file picker at current working directory", code_action, "Perform code action", + tab_picker, "Open tab picker", buffer_picker, "Open buffer picker", jumplist_picker, "Open jumplist picker", symbol_picker, "Open symbol picker", @@ -2523,6 +2524,50 @@ fn file_picker_in_current_directory(cx: &mut Context) { cx.push_layer(Box::new(overlaid(picker))); } +fn tab_picker(cx: &mut Context) { + let current = cx.editor.tabs.focus; + + #[derive(Debug)] + struct TabMeta { + idx: usize, + id: TabId, + name: String, + is_current: bool, + } + + impl ui::menu::Item for TabMeta { + type Data = (); + + fn format(&self, _data: &Self::Data) -> Row { + let mut flags = String::new(); + if self.is_current { + flags.push('*'); + } + + Row::new([(self.idx + 1).to_string(), self.name.clone(), flags]) + } + } + + let opts = cx + .editor + .tabs + .iter_tabs() + .enumerate() + .map(|(idx, (id, tab))| TabMeta { + idx, + id, + name: tab.name.clone(), + is_current: id == current, + }) + .collect(); + + let picker = Picker::new(opts, (), |cx, meta, _action| { + cx.editor.tabs.focus = meta.id; + }); + + cx.push_layer(Box::new(overlaid(picker))); +} + fn buffer_picker(cx: &mut Context) { let current = view!(cx.editor).doc; @@ -2628,7 +2673,7 @@ fn jumplist_picker(cx: &mut Context) { } } - for (view, _) in cx.editor.tree.views_mut() { + for (view, _) in cx.editor.tabs.curr_tree_mut().views_mut() { for doc_id in view.jumps.iter().map(|e| e.0).collect::>().iter() { let doc = doc_mut!(cx.editor, doc_id); view.sync_changes(doc); @@ -2656,7 +2701,8 @@ fn jumplist_picker(cx: &mut Context) { let picker = Picker::new( cx.editor - .tree + .tabs + .curr_tree() .views() .flat_map(|(view, _)| { view.jumps @@ -2742,7 +2788,7 @@ pub fn command_palette(cx: &mut Context) { command.execute(&mut ctx); - if ctx.editor.tree.contains(focus) { + if ctx.editor.tabs.curr_tree().contains(focus) { let config = ctx.editor.config(); let mode = ctx.editor.mode(); let view = view_mut!(ctx.editor, focus); @@ -2869,7 +2915,7 @@ async fn make_format_callback( let format = format.await; let call: job::Callback = Callback::Editor(Box::new(move |editor| { - if !editor.documents.contains_key(&doc_id) || !editor.tree.contains(view_id) { + if !editor.documents.contains_key(&doc_id) || !editor.tabs.curr_tree().contains(view_id) { return; } @@ -4787,7 +4833,7 @@ fn vsplit_new(cx: &mut Context) { } fn wclose(cx: &mut Context) { - if cx.editor.tree.views().count() == 1 { + if cx.editor.tabs.curr_tree().views().count() == 1 { if let Err(err) = typed::buffers_remaining_impl(cx.editor) { cx.editor.set_error(err.to_string()); return; @@ -4801,7 +4847,8 @@ fn wclose(cx: &mut Context) { fn wonly(cx: &mut Context) { let views = cx .editor - .tree + .tabs + .curr_tree() .views() .map(|(v, focus)| (v.id, focus)) .collect::>(); diff --git a/helix-term/src/commands/lsp.rs b/helix-term/src/commands/lsp.rs index 55153648a37b7..492547a8adfbb 100644 --- a/helix-term/src/commands/lsp.rs +++ b/helix-term/src/commands/lsp.rs @@ -1511,7 +1511,7 @@ pub fn compute_inlay_hints_for_all_views(editor: &mut Editor, jobs: &mut crate:: return; } - for (view, _) in editor.tree.views() { + for (view, _) in editor.tabs.curr_tree().views() { let doc = match editor.documents.get(&view.doc) { Some(doc) => doc, None => continue, @@ -1576,7 +1576,9 @@ fn compute_inlay_hints_for_view( language_server.text_document_range_inlay_hints(doc.identifier(), range, None)?, move |editor, _compositor, response: Option>| { // The config was modified or the window was closed while the request was in flight - if !editor.config().lsp.display_inlay_hints || editor.tree.try_get(view_id).is_none() { + if !editor.config().lsp.display_inlay_hints + || editor.tabs.curr_tree().try_get(view_id).is_none() + { return; } diff --git a/helix-term/src/commands/typed.rs b/helix-term/src/commands/typed.rs index 5198e5bdf5cf1..08f6219023e53 100644 --- a/helix-term/src/commands/typed.rs +++ b/helix-term/src/commands/typed.rs @@ -73,7 +73,7 @@ fn quit(cx: &mut compositor::Context, args: &[Cow], event: PromptEvent) -> ensure!(args.is_empty(), ":quit takes no arguments"); // last view and we have unsaved changes - if cx.editor.tree.views().count() == 1 { + if cx.editor.tabs.curr_tree().views().count() == 1 { buffers_remaining_impl(cx.editor)? } @@ -770,7 +770,13 @@ fn quit_all_impl(cx: &mut compositor::Context, force: bool) -> anyhow::Result<() } // close all views - let views: Vec<_> = cx.editor.tree.views().map(|(view, _)| view.id).collect(); + let views: Vec<_> = cx + .editor + .tabs + .curr_tree() + .views() + .map(|(view, _)| view.id) + .collect(); for view_id in views { cx.editor.close(view_id); } @@ -1516,6 +1522,43 @@ fn tree_sitter_scopes( Ok(()) } +fn tabnext( + cx: &mut compositor::Context, + _args: &[Cow], + event: PromptEvent, +) -> anyhow::Result<()> { + if event != PromptEvent::Validate { + return Ok(()); + } + cx.editor.tabs.focus_next(); + Ok(()) +} + +fn tabnew( + cx: &mut compositor::Context, + _args: &[Cow], + event: PromptEvent, +) -> anyhow::Result<()> { + if event != PromptEvent::Validate { + return Ok(()); + } + + // let id = view!(cx.editor).doc; + + // if args.is_empty() { + cx.editor.tabs.new_tab(); + cx.editor.new_file(Action::VerticalSplit); + // cx.editor.switch(id, Action::NewTab); + // } else { + // for arg in args { + // cx.editor + // .open(&PathBuf::from(arg.as_ref()), Action::VerticalSplit)?; + // } + // } + + Ok(()) +} + fn vsplit( cx: &mut compositor::Context, args: &[Cow], @@ -2675,6 +2718,21 @@ pub const TYPABLE_COMMAND_LIST: &[TypableCommand] = &[ fun: debug_eval, signature: CommandSignature::none(), }, + TypableCommand { + name: "tab-new", + aliases: &[], + doc: "Open a new tab.", + fun: tabnew, + signature: CommandSignature::none(), + // signature: CommandSignature::all(completers::filename) + }, + TypableCommand { + name: "tab-next", + aliases: &[], + doc: "Focus next tab", + fun: tabnext, + signature: CommandSignature::none(), + }, TypableCommand { name: "vsplit", aliases: &["vs"], diff --git a/helix-term/src/keymap/default.rs b/helix-term/src/keymap/default.rs index c84c616c6c0be..6ff6be2a3280c 100644 --- a/helix-term/src/keymap/default.rs +++ b/helix-term/src/keymap/default.rs @@ -214,6 +214,7 @@ pub fn default() -> HashMap { "space" => { "Space" "f" => file_picker, "F" => file_picker_in_current_directory, + "t" => tab_picker, "b" => buffer_picker, "j" => jumplist_picker, "s" => symbol_picker, diff --git a/helix-term/src/ui/editor.rs b/helix-term/src/ui/editor.rs index 5b5cda9350430..358577fd3b612 100644 --- a/helix-term/src/ui/editor.rs +++ b/helix-term/src/ui/editor.rs @@ -513,6 +513,49 @@ impl EditorView { Vec::new() } + /// Render tabline at the top + pub fn render_tabline(editor: &Editor, viewport: Rect, surface: &mut Surface) { + surface.clear_with( + viewport, + editor + .theme + .try_get("ui.bufferline.background") + .unwrap_or_else(|| editor.theme.get("ui.statusline")), + ); + + let bufferline_active = editor + .theme + .try_get("ui.bufferline.active") + .unwrap_or_else(|| editor.theme.get("ui.statusline.active")); + + let bufferline_inactive = editor + .theme + .try_get("ui.bufferline") + .unwrap_or_else(|| editor.theme.get("ui.statusline.inactive")); + + let mut x = viewport.x; + + let current_tab = editor.tabs.focus; + for (id, tab) in editor.tabs.iter_tabs() { + let style = if current_tab == id { + bufferline_active + } else { + bufferline_inactive + }; + + let text = format!(" {} ", tab.name); + let used_width = viewport.x.saturating_sub(x); + let rem_width = surface.area.width.saturating_sub(used_width); + x = surface + .set_stringn(x, viewport.y, text, rem_width as usize, style) + .0; + + if x >= surface.area.right() { + break; + } + } + } + /// Render bufferline at the top pub fn render_bufferline(editor: &Editor, viewport: Rect, surface: &mut Surface) { let scratch = PathBuf::from(SCRATCH_BUFFER_NAME); // default filename to use for scratch buffer @@ -1044,7 +1087,7 @@ impl EditorView { } = *event; let pos_and_view = |editor: &Editor, row, column, ignore_virtual_text| { - editor.tree.views().find_map(|(view, _focus)| { + editor.tabs.curr_tree().views().find_map(|(view, _focus)| { view.pos_at_screen_coords( &editor.documents[&view.doc], row, @@ -1056,7 +1099,7 @@ impl EditorView { }; let gutter_coords_and_view = |editor: &Editor, row, column| { - editor.tree.views().find_map(|(view, _focus)| { + editor.tabs.curr_tree().views().find_map(|(view, _focus)| { view.gutter_coords_at_screen_coords(row, column) .map(|coords| (coords, view.id)) }) @@ -1122,7 +1165,7 @@ impl EditorView { } MouseEventKind::ScrollUp | MouseEventKind::ScrollDown => { - let current_view = cxt.editor.tree.focus; + let current_view = cxt.editor.tabs.curr_tree().focus; let direction = match event.kind { MouseEventKind::ScrollUp => Direction::Backward, @@ -1131,14 +1174,14 @@ impl EditorView { }; match pos_and_view(cxt.editor, row, column, false) { - Some((_, view_id)) => cxt.editor.tree.focus = view_id, + Some((_, view_id)) => cxt.editor.tabs.curr_tree_mut().focus = view_id, None => return EventResult::Ignored(None), } let offset = config.scroll_lines.unsigned_abs(); commands::scroll(cxt, offset, direction); - cxt.editor.tree.focus = current_view; + cxt.editor.tabs.curr_tree_mut().focus = current_view; cxt.editor.ensure_cursor_in_view(current_view); EventResult::Consumed(None) @@ -1349,7 +1392,7 @@ impl Component for EditorView { } // if the focused view still exists and wasn't closed - if cx.editor.tree.contains(focus) { + if cx.editor.tabs.curr_tree().contains(focus) { let config = cx.editor.config(); let mode = cx.editor.mode(); let view = view_mut!(cx.editor, focus); @@ -1404,10 +1447,10 @@ impl Component for EditorView { cx.editor.resize(editor_area); if use_bufferline { - Self::render_bufferline(cx.editor, area.with_height(1), surface); + Self::render_tabline(cx.editor, area.with_height(1), surface); } - for (view, is_focused) in cx.editor.tree.views() { + for (view, is_focused) in cx.editor.tabs.curr_tree().views() { let doc = cx.editor.document(view.doc).unwrap(); self.render_view(cx.editor, doc, view, area, surface, is_focused); } diff --git a/helix-view/src/editor.rs b/helix-view/src/editor.rs index 61d148d32c562..c097c5aea71d1 100644 --- a/helix-view/src/editor.rs +++ b/helix-view/src/editor.rs @@ -5,8 +5,9 @@ use crate::{ graphics::{CursorKind, Rect}, info::Info, input::KeyEvent, + tabs::Tabs, theme::{self, Theme}, - tree::{self, Tree}, + tree::{self}, view::ViewPosition, Align, Document, DocumentId, View, ViewId, }; @@ -702,7 +703,7 @@ pub struct WhitespaceCharacters { impl Default for WhitespaceCharacters { fn default() -> Self { Self { - space: '·', // U+00B7 + space: '·', // U+00B7 nbsp: '⍽', // U+237D tab: '→', // U+2192 newline: '⏎', // U+23CE @@ -862,7 +863,7 @@ use futures_util::stream::{Flatten, Once}; pub struct Editor { /// Current editing mode. pub mode: Mode, - pub tree: Tree, + pub tabs: Tabs, pub next_document_id: DocumentId, pub documents: BTreeMap, @@ -1010,7 +1011,7 @@ impl Editor { Self { mode: Mode::Normal, - tree: Tree::new(area), + tabs: Tabs::new(area), next_document_id: DocumentId::default(), documents: BTreeMap::new(), saves: HashMap::new(), @@ -1241,7 +1242,7 @@ impl Editor { } } - for (view, _) in self.tree.views_mut() { + for (view, _) in self.tabs.curr_tree_mut().views_mut() { let doc = doc_mut!(self, &view.doc); view.sync_changes(doc); view.gutters = config.gutters.clone(); @@ -1250,7 +1251,7 @@ impl Editor { } fn replace_document_in_view(&mut self, current_view: ViewId, doc_id: DocumentId) { - let view = self.tree.get_mut(current_view); + let view = self.tabs.curr_tree_mut().get_mut(current_view); view.doc = doc_id; view.offset = ViewPosition::default(); @@ -1275,6 +1276,7 @@ impl Editor { match action { Action::Replace => { let (view, doc) = current_ref!(self); + let view_id = view.id; // If the current view is an empty scratch buffer and is not displayed in any other views, delete it. // Boolean value is determined before the call to `view_mut` because the operation requires a borrow // of `self.tree`, which is mutably borrowed when `view_mut` is called. @@ -1285,9 +1287,10 @@ impl Editor { && id != doc.id // Ensure the buffer is not displayed in any other splits. && !self - .tree + .tabs + .curr_tree_mut() .traverse() - .any(|(_, v)| v.doc == doc.id && v.id != view.id); + .any(|(_, v)| v.doc == doc.id && v.id != view_id); let (view, doc) = current!(self); let view_id = view.id; @@ -1302,7 +1305,7 @@ impl Editor { self.documents.remove(&id); // Remove the scratch buffer from any jumplists - for (view, _) in self.tree.views_mut() { + for (view, _) in self.tabs.curr_tree_mut().views_mut() { view.remove_document(&id); } } else { @@ -1333,13 +1336,15 @@ impl Editor { } Action::HorizontalSplit | Action::VerticalSplit => { // copy the current view, unless there is no view yet + let focus = self.tabs.curr_tree().focus; let view = self - .tree - .try_get(self.tree.focus) + .tabs + .curr_tree_mut() + .try_get(focus) .filter(|v| id == v.doc) // Different Document .cloned() .unwrap_or_else(|| View::new(id, self.config().gutters.clone())); - let view_id = self.tree.split( + let view_id = self.tabs.curr_tree_mut().split( view, match action { Action::HorizontalSplit => Layout::Horizontal, @@ -1434,15 +1439,17 @@ impl Editor { Ok(id) } + // TODO(nrabulinski): Closing last view in a tab should move focus to previous tab pub fn close(&mut self, id: ViewId) { // Remove selections for the closed view on all documents. for doc in self.documents_mut() { doc.remove_view(id); } - self.tree.remove(id); + self.tabs.curr_tree_mut().remove(id); self._refresh(); } + // TODO(nrabulinski): Closing last view in a tab should move focus to previous tab pub fn close_document(&mut self, doc_id: DocumentId, force: bool) -> Result<(), CloseError> { let doc = match self.documents.get_mut(&doc_id) { Some(doc) => doc, @@ -1466,7 +1473,8 @@ impl Editor { } let actions: Vec = self - .tree + .tabs + .curr_tree_mut() .views_mut() .filter_map(|(view, _focus)| { view.remove_document(&doc_id); @@ -1501,7 +1509,7 @@ impl Editor { // If the document we removed was visible in all views, we will have no more views. We don't // want to close the editor just for a simple buffer close, so we need to create a new view // containing either an existing document, or a brand new document. - if self.tree.views().next().is_none() { + if self.tabs.curr_tree().views().next().is_none() { let doc_id = self .documents .iter() @@ -1509,7 +1517,7 @@ impl Editor { .next() .unwrap_or_else(|| self.new_document(Document::default(self.config.clone()))); let view = View::new(doc_id, self.config().gutters.clone()); - let view_id = self.tree.insert(view); + let view_id = self.tabs.curr_tree_mut().insert(view); let doc = doc_mut!(self, &doc_id); doc.ensure_view_init(view_id); doc.mark_as_focused(); @@ -1547,13 +1555,17 @@ impl Editor { } pub fn resize(&mut self, area: Rect) { - if self.tree.resize(area) { + if self + .tabs + .iter_tabs_mut() + .fold(false, |acc, (_, tab)| acc | tab.tree.resize(area)) + { self._refresh(); - }; + } } pub fn focus(&mut self, view_id: ViewId) { - let prev_id = std::mem::replace(&mut self.tree.focus, view_id); + let prev_id = std::mem::replace(&mut self.tabs.curr_tree_mut().focus, view_id); // if leaving the view: mode should reset and the cursor should be // within view @@ -1562,7 +1574,7 @@ impl Editor { self.ensure_cursor_in_view(view_id); // Update jumplist selections with new document changes. - for (view, _focused) in self.tree.views_mut() { + for (view, _focused) in self.tabs.curr_tree_mut().views_mut() { let doc = doc_mut!(self, &view.doc); view.sync_changes(doc); } @@ -1574,35 +1586,39 @@ impl Editor { } pub fn focus_next(&mut self) { - self.focus(self.tree.next()); + self.focus(self.tabs.curr_tree().next()); } pub fn focus_prev(&mut self) { - self.focus(self.tree.prev()); + self.focus(self.tabs.curr_tree().prev()); } pub fn focus_direction(&mut self, direction: tree::Direction) { - let current_view = self.tree.focus; - if let Some(id) = self.tree.find_split_in_direction(current_view, direction) { + let current_view = self.tabs.curr_tree().focus; + if let Some(id) = self + .tabs + .curr_tree_mut() + .find_split_in_direction(current_view, direction) + { self.focus(id) } } pub fn swap_split_in_direction(&mut self, direction: tree::Direction) { - self.tree.swap_split_in_direction(direction); + self.tabs.curr_tree_mut().swap_split_in_direction(direction); } pub fn transpose_view(&mut self) { - self.tree.transpose(); + self.tabs.curr_tree_mut().transpose(); } pub fn should_close(&self) -> bool { - self.tree.is_empty() + self.tabs.curr_tree().is_empty() } pub fn ensure_cursor_in_view(&mut self, id: ViewId) { let config = self.config(); - let view = self.tree.get_mut(id); + let view = self.tabs.curr_tree_mut().get_mut(id); let doc = &self.documents[&view.doc]; view.ensure_cursor_in_view(doc, config.scrolloff) } diff --git a/helix-view/src/lib.rs b/helix-view/src/lib.rs index c3f67345b3618..afe81bd1b141d 100644 --- a/helix-view/src/lib.rs +++ b/helix-view/src/lib.rs @@ -15,6 +15,7 @@ pub mod base64; pub mod info; pub mod input; pub mod keyboard; +pub mod tabs; pub mod theme; pub mod tree; pub mod view; @@ -40,6 +41,7 @@ impl std::fmt::Display for DocumentId { slotmap::new_key_type! { pub struct ViewId; + pub struct TabId; } pub enum Align { diff --git a/helix-view/src/macros.rs b/helix-view/src/macros.rs index ee9cd41117950..3570b5618057f 100644 --- a/helix-view/src/macros.rs +++ b/helix-view/src/macros.rs @@ -22,7 +22,8 @@ macro_rules! current { #[macro_export] macro_rules! current_ref { ($editor:expr) => {{ - let view = $editor.tree.get($editor.tree.focus); + let tree = $editor.tabs.curr_tree(); + let view = tree.get(tree.focus); let doc = &$editor.documents[&view.doc]; (view, doc) }}; @@ -45,10 +46,12 @@ macro_rules! doc_mut { #[macro_export] macro_rules! view_mut { ($editor:expr, $id:expr) => {{ - $editor.tree.get_mut($id) + let tree = $editor.tabs.curr_tree_mut(); + tree.get_mut($id) }}; ($editor:expr) => {{ - $editor.tree.get_mut($editor.tree.focus) + let tree = $editor.tabs.curr_tree_mut(); + tree.get_mut(tree.focus) }}; } @@ -57,10 +60,12 @@ macro_rules! view_mut { #[macro_export] macro_rules! view { ($editor:expr, $id:expr) => {{ - $editor.tree.get($id) + let tree = $editor.tabs.curr_tree(); + tree.get($id) }}; ($editor:expr) => {{ - $editor.tree.get($editor.tree.focus) + let tree = $editor.tabs.curr_tree(); + tree.get(tree.focus) }}; } diff --git a/helix-view/src/tabs.rs b/helix-view/src/tabs.rs new file mode 100644 index 0000000000000..302565be2b017 --- /dev/null +++ b/helix-view/src/tabs.rs @@ -0,0 +1,70 @@ +use slotmap::HopSlotMap; + +use crate::{graphics::Rect, tree::Tree, TabId}; + +#[derive(Debug)] +pub struct Tab { + pub name: String, + pub tree: Tree, +} + +#[derive(Debug)] +pub struct Tabs { + pub focus: TabId, + tabs: HopSlotMap, +} + +impl Tabs { + #[inline] + pub fn new(area: Rect) -> Self { + let mut tabs = HopSlotMap::with_key(); + let tab = Tab { + name: "Tab 0".to_string(), + tree: Tree::new(area), + }; + let focus = tabs.insert(tab); + Self { focus, tabs } + } + + #[inline] + pub fn curr_tree_mut(&mut self) -> &mut Tree { + &mut self.tabs.get_mut(self.focus).unwrap().tree + } + + #[inline] + pub fn curr_tree(&self) -> &Tree { + &self.tabs.get(self.focus).unwrap().tree + } + + #[inline] + pub fn iter_tabs_mut(&mut self) -> impl Iterator { + self.tabs.iter_mut() + } + + #[inline] + pub fn iter_tabs(&self) -> impl Iterator { + self.tabs.iter() + } + + #[inline] + pub fn new_tab(&mut self) -> TabId { + let area = self.curr_tree().area(); + let new_tab = Tab { + name: format!("Tab {}", self.tabs.len()), + tree: Tree::new(area), + }; + let new_focus = self.tabs.insert(new_tab); + self.focus = new_focus; + new_focus + } + + #[inline] + pub fn focus_next(&mut self) -> TabId { + let curr = self.focus; + let mut iter = self.tabs.keys().skip_while(|id| *id != curr); + iter.next(); + let id = iter.next().or_else(|| self.tabs.keys().next()).unwrap(); + self.focus = id; + id + } +}