
This PR removes the `full` style method and replaces it with `size_full`, as the two do the same thing. This is the generated code for `size_full`: ```rs #[doc = "Sets the width and height of the element.\n\n100%"] fn size_full(mut self) -> Self { let style = self.style(); style.size.width = Some((gpui::relative(1.)).into()); style.size.height = Some((gpui::relative(1.)).into()); self } ``` Release Notes: - N/A
221 lines
7.6 KiB
Rust
221 lines
7.6 KiB
Rust
use std::{ops::Range, path::PathBuf};
|
|
|
|
use editor::{Editor, EditorEvent};
|
|
use gpui::{
|
|
list, AnyElement, AppContext, EventEmitter, FocusHandle, FocusableView, InteractiveElement,
|
|
IntoElement, ListState, ParentElement, Render, Styled, View, ViewContext, WeakView,
|
|
};
|
|
use ui::prelude::*;
|
|
use workspace::item::Item;
|
|
use workspace::Workspace;
|
|
|
|
use crate::{
|
|
markdown_elements::ParsedMarkdown,
|
|
markdown_parser::parse_markdown,
|
|
markdown_renderer::{render_markdown_block, RenderContext},
|
|
OpenPreview,
|
|
};
|
|
|
|
pub struct MarkdownPreviewView {
|
|
workspace: WeakView<Workspace>,
|
|
focus_handle: FocusHandle,
|
|
contents: ParsedMarkdown,
|
|
selected_block: usize,
|
|
list_state: ListState,
|
|
}
|
|
|
|
impl MarkdownPreviewView {
|
|
pub fn register(workspace: &mut Workspace, _cx: &mut ViewContext<Workspace>) {
|
|
workspace.register_action(move |workspace, _: &OpenPreview, cx| {
|
|
if workspace.has_active_modal(cx) {
|
|
cx.propagate();
|
|
return;
|
|
}
|
|
|
|
if let Some(editor) = workspace.active_item_as::<Editor>(cx) {
|
|
let workspace_handle = workspace.weak_handle();
|
|
let view: View<MarkdownPreviewView> =
|
|
MarkdownPreviewView::new(editor, workspace_handle, cx);
|
|
workspace.split_item(workspace::SplitDirection::Right, Box::new(view.clone()), cx);
|
|
cx.notify();
|
|
}
|
|
});
|
|
}
|
|
|
|
pub fn new(
|
|
active_editor: View<Editor>,
|
|
workspace: WeakView<Workspace>,
|
|
cx: &mut ViewContext<Workspace>,
|
|
) -> View<Self> {
|
|
cx.new_view(|cx: &mut ViewContext<Self>| {
|
|
let view = cx.view().downgrade();
|
|
let editor = active_editor.read(cx);
|
|
|
|
let file_location = MarkdownPreviewView::get_folder_for_active_editor(editor, cx);
|
|
let contents = editor.buffer().read(cx).snapshot(cx).text();
|
|
let contents = parse_markdown(&contents, file_location);
|
|
|
|
cx.subscribe(&active_editor, |this, editor, event: &EditorEvent, cx| {
|
|
match event {
|
|
EditorEvent::Edited => {
|
|
let editor = editor.read(cx);
|
|
let contents = editor.buffer().read(cx).snapshot(cx).text();
|
|
let file_location =
|
|
MarkdownPreviewView::get_folder_for_active_editor(editor, cx);
|
|
this.contents = parse_markdown(&contents, file_location);
|
|
this.list_state.reset(this.contents.children.len());
|
|
cx.notify();
|
|
|
|
// TODO: This does not work as expected.
|
|
// The scroll request appears to be dropped
|
|
// after `.reset` is called.
|
|
this.list_state.scroll_to_reveal_item(this.selected_block);
|
|
cx.notify();
|
|
}
|
|
EditorEvent::SelectionsChanged { .. } => {
|
|
let editor = editor.read(cx);
|
|
let selection_range = editor.selections.last::<usize>(cx).range();
|
|
this.selected_block = this.get_block_index_under_cursor(selection_range);
|
|
this.list_state.scroll_to_reveal_item(this.selected_block);
|
|
cx.notify();
|
|
}
|
|
_ => {}
|
|
};
|
|
})
|
|
.detach();
|
|
|
|
let list_state = ListState::new(
|
|
contents.children.len(),
|
|
gpui::ListAlignment::Top,
|
|
px(1000.),
|
|
move |ix, cx| {
|
|
if let Some(view) = view.upgrade() {
|
|
view.update(cx, |view, cx| {
|
|
let mut render_cx =
|
|
RenderContext::new(Some(view.workspace.clone()), cx);
|
|
let block = view.contents.children.get(ix).unwrap();
|
|
let block = render_markdown_block(block, &mut render_cx);
|
|
let block = div().child(block).pl_4().pb_3();
|
|
|
|
if ix == view.selected_block {
|
|
let indicator = div()
|
|
.h_full()
|
|
.w(px(4.0))
|
|
.bg(cx.theme().colors().border)
|
|
.rounded_sm();
|
|
|
|
return div()
|
|
.relative()
|
|
.child(block)
|
|
.child(indicator.absolute().left_0().top_0())
|
|
.into_any();
|
|
}
|
|
|
|
block.into_any()
|
|
})
|
|
} else {
|
|
div().into_any()
|
|
}
|
|
},
|
|
);
|
|
|
|
Self {
|
|
selected_block: 0,
|
|
focus_handle: cx.focus_handle(),
|
|
workspace,
|
|
contents,
|
|
list_state,
|
|
}
|
|
})
|
|
}
|
|
|
|
/// The absolute path of the file that is currently being previewed.
|
|
fn get_folder_for_active_editor(
|
|
editor: &Editor,
|
|
cx: &ViewContext<MarkdownPreviewView>,
|
|
) -> Option<PathBuf> {
|
|
if let Some(file) = editor.file_at(0, cx) {
|
|
if let Some(file) = file.as_local() {
|
|
file.abs_path(cx).parent().map(|p| p.to_path_buf())
|
|
} else {
|
|
None
|
|
}
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn get_block_index_under_cursor(&self, selection_range: Range<usize>) -> usize {
|
|
let mut block_index = 0;
|
|
let cursor = selection_range.start;
|
|
|
|
for (i, block) in self.contents.children.iter().enumerate() {
|
|
let Range { start, end } = block.source_range();
|
|
if start <= cursor && end >= cursor {
|
|
block_index = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return block_index;
|
|
}
|
|
}
|
|
|
|
impl FocusableView for MarkdownPreviewView {
|
|
fn focus_handle(&self, _: &AppContext) -> gpui::FocusHandle {
|
|
self.focus_handle.clone()
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
pub enum PreviewEvent {}
|
|
|
|
impl EventEmitter<PreviewEvent> for MarkdownPreviewView {}
|
|
|
|
impl Item for MarkdownPreviewView {
|
|
type Event = PreviewEvent;
|
|
|
|
fn tab_content(
|
|
&self,
|
|
_detail: Option<usize>,
|
|
selected: bool,
|
|
_cx: &WindowContext,
|
|
) -> AnyElement {
|
|
h_flex()
|
|
.gap_2()
|
|
.child(Icon::new(IconName::FileDoc).color(if selected {
|
|
Color::Default
|
|
} else {
|
|
Color::Muted
|
|
}))
|
|
.child(Label::new("Markdown preview").color(if selected {
|
|
Color::Default
|
|
} else {
|
|
Color::Muted
|
|
}))
|
|
.into_any()
|
|
}
|
|
|
|
fn telemetry_event_text(&self) -> Option<&'static str> {
|
|
Some("markdown preview")
|
|
}
|
|
|
|
fn to_item_events(_event: &Self::Event, _f: impl FnMut(workspace::item::ItemEvent)) {}
|
|
}
|
|
|
|
impl Render for MarkdownPreviewView {
|
|
fn render(&mut self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
|
v_flex()
|
|
.id("MarkdownPreview")
|
|
.key_context("MarkdownPreview")
|
|
.track_focus(&self.focus_handle)
|
|
.size_full()
|
|
.bg(cx.theme().colors().editor_background)
|
|
.p_4()
|
|
.child(
|
|
div()
|
|
.flex_grow()
|
|
.map(|this| this.child(list(self.list_state.clone()).size_full())),
|
|
)
|
|
}
|
|
}
|