use gpui::{AppContext, Model, ModelContext}; use language::Buffer; use std::ops::Range; // TODO: Find a better home for `Direction`. // // This should live in an ancestor crate of `editor` and `inline_completion`, // but at time of writing there isn't an obvious spot. #[derive(Copy, Clone, PartialEq, Eq)] pub enum Direction { Prev, Next, } #[derive(Clone)] pub struct InlineCompletion { pub edits: Vec<(Range, String)>, } pub trait InlineCompletionProvider: 'static + Sized { fn name() -> &'static str; fn display_name() -> &'static str; fn show_completions_in_menu() -> bool; fn is_enabled( &self, buffer: &Model, cursor_position: language::Anchor, cx: &AppContext, ) -> bool; fn refresh( &mut self, buffer: Model, cursor_position: language::Anchor, debounce: bool, cx: &mut ModelContext, ); fn cycle( &mut self, buffer: Model, cursor_position: language::Anchor, direction: Direction, cx: &mut ModelContext, ); fn accept(&mut self, cx: &mut ModelContext); fn discard(&mut self, cx: &mut ModelContext); fn suggest( &mut self, buffer: &Model, cursor_position: language::Anchor, cx: &mut ModelContext, ) -> Option; } pub trait InlineCompletionProviderHandle { fn name(&self) -> &'static str; fn display_name(&self) -> &'static str; fn is_enabled( &self, buffer: &Model, cursor_position: language::Anchor, cx: &AppContext, ) -> bool; fn show_completions_in_menu(&self) -> bool; fn refresh( &self, buffer: Model, cursor_position: language::Anchor, debounce: bool, cx: &mut AppContext, ); fn cycle( &self, buffer: Model, cursor_position: language::Anchor, direction: Direction, cx: &mut AppContext, ); fn accept(&self, cx: &mut AppContext); fn discard(&self, cx: &mut AppContext); fn suggest( &self, buffer: &Model, cursor_position: language::Anchor, cx: &mut AppContext, ) -> Option; } impl InlineCompletionProviderHandle for Model where T: InlineCompletionProvider, { fn name(&self) -> &'static str { T::name() } fn display_name(&self) -> &'static str { T::display_name() } fn show_completions_in_menu(&self) -> bool { T::show_completions_in_menu() } fn is_enabled( &self, buffer: &Model, cursor_position: language::Anchor, cx: &AppContext, ) -> bool { self.read(cx).is_enabled(buffer, cursor_position, cx) } fn refresh( &self, buffer: Model, cursor_position: language::Anchor, debounce: bool, cx: &mut AppContext, ) { self.update(cx, |this, cx| { this.refresh(buffer, cursor_position, debounce, cx) }) } fn cycle( &self, buffer: Model, cursor_position: language::Anchor, direction: Direction, cx: &mut AppContext, ) { self.update(cx, |this, cx| { this.cycle(buffer, cursor_position, direction, cx) }) } fn accept(&self, cx: &mut AppContext) { self.update(cx, |this, cx| this.accept(cx)) } fn discard(&self, cx: &mut AppContext) { self.update(cx, |this, cx| this.discard(cx)) } fn suggest( &self, buffer: &Model, cursor_position: language::Anchor, cx: &mut AppContext, ) -> Option { self.update(cx, |this, cx| this.suggest(buffer, cursor_position, cx)) } }