mirror of
https://github.com/zed-industries/zed.git
synced 2026-06-01 03:14:56 +07:00
Adds opt-in rendering for Markdown frontmatter metadata blocks in Markdown Preview and agent markdown. - Simple `key: value` metadata blocks now render as a two-column table, while more complex metadata falls back to a code-style block. - Metadata block content and key/value rows are parsed in the parser step, and the request layout simply takes over rendering. <img width="1288" height="436" alt="image" src="https://github.com/user-attachments/assets/b35b949a-8bc4-47db-82ef-ed835e9ac06f" /> Release Notes: - Added support for rendering Markdown frontmatter metadata blocks in Markdown Preview and Agent Panel.
4676 lines
172 KiB
Rust
4676 lines
172 KiB
Rust
pub mod html;
|
|
mod mermaid;
|
|
pub mod parser;
|
|
mod path_range;
|
|
|
|
use base64::Engine as _;
|
|
use futures::FutureExt as _;
|
|
use gpui::EdgesRefinement;
|
|
use gpui::HitboxBehavior;
|
|
use gpui::UnderlineStyle;
|
|
use language::LanguageName;
|
|
|
|
use log::Level;
|
|
use mermaid::{
|
|
MermaidState, ParsedMarkdownMermaidDiagram, extract_mermaid_diagrams, render_mermaid_diagram,
|
|
};
|
|
pub use path_range::{LineCol, PathWithRange};
|
|
use settings::Settings as _;
|
|
use theme_settings::ThemeSettings;
|
|
|
|
use std::borrow::Cow;
|
|
use std::collections::BTreeMap;
|
|
use std::iter;
|
|
use std::mem;
|
|
use std::ops::Range;
|
|
use std::path::Path;
|
|
use std::rc::Rc;
|
|
use std::sync::Arc;
|
|
use std::time::Duration;
|
|
|
|
use collections::{HashMap, HashSet};
|
|
use gpui::{
|
|
AnyElement, App, BorderStyle, Bounds, ClipboardItem, CursorStyle, DispatchPhase, Edges, Entity,
|
|
FocusHandle, Focusable, FontStyle, FontWeight, GlobalElementId, Hitbox, Hsla, Image,
|
|
ImageFormat, ImageSource, KeyContext, Length, MouseButton, MouseDownEvent, MouseEvent,
|
|
MouseMoveEvent, MouseUpEvent, Point, ScrollHandle, Stateful, StrikethroughStyle,
|
|
StyleRefinement, StyledImage, StyledText, Subscription, Task, TextAlign, TextLayout, TextRun,
|
|
TextStyle, TextStyleRefinement, actions, img, point, quad,
|
|
};
|
|
use language::{CharClassifier, Language, LanguageRegistry, Rope};
|
|
use parser::CodeBlockMetadata;
|
|
use parser::{
|
|
MarkdownEvent, MarkdownTag, MarkdownTagEnd, ParsedMetadataBlock, parse_links_only,
|
|
parse_markdown_with_options,
|
|
};
|
|
use pulldown_cmark::{Alignment, BlockQuoteKind};
|
|
use sum_tree::TreeMap;
|
|
use theme::SyntaxTheme;
|
|
use ui::{Checkbox, CopyButton, ScrollAxes, Scrollbars, Tooltip, WithScrollbar, prelude::*};
|
|
use util::ResultExt;
|
|
|
|
use crate::parser::CodeBlockKind;
|
|
|
|
/// A callback function that can be used to customize the style of links based on the destination URL.
|
|
/// If the callback returns `None`, the default link style will be used.
|
|
type LinkStyleCallback = Rc<dyn Fn(&str, &App) -> Option<TextStyleRefinement>>;
|
|
pub type CodeSpanLinkCallback = Arc<dyn Fn(&str, &App) -> Option<SharedString> + 'static>;
|
|
type SourceClickCallback = Box<dyn Fn(usize, usize, &mut Window, &mut App) -> bool>;
|
|
type CheckboxToggleCallback = Rc<dyn Fn(Range<usize>, bool, &mut Window, &mut App)>;
|
|
|
|
#[derive(Clone, Copy, Default)]
|
|
pub struct BlockQuoteKindColors {
|
|
pub note: Hsla,
|
|
pub tip: Hsla,
|
|
pub important: Hsla,
|
|
pub warning: Hsla,
|
|
pub caution: Hsla,
|
|
}
|
|
|
|
impl BlockQuoteKindColors {
|
|
fn for_kind(&self, kind: Option<BlockQuoteKind>, default: Hsla) -> Hsla {
|
|
match kind {
|
|
Some(BlockQuoteKind::Note) => self.note,
|
|
Some(BlockQuoteKind::Tip) => self.tip,
|
|
Some(BlockQuoteKind::Important) => self.important,
|
|
Some(BlockQuoteKind::Warning) => self.warning,
|
|
Some(BlockQuoteKind::Caution) => self.caution,
|
|
None => default,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Default)]
|
|
pub struct HeadingLevelStyles {
|
|
pub h1: Option<TextStyleRefinement>,
|
|
pub h2: Option<TextStyleRefinement>,
|
|
pub h3: Option<TextStyleRefinement>,
|
|
pub h4: Option<TextStyleRefinement>,
|
|
pub h5: Option<TextStyleRefinement>,
|
|
pub h6: Option<TextStyleRefinement>,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct MarkdownStyle {
|
|
pub base_text_style: TextStyle,
|
|
pub container_style: StyleRefinement,
|
|
pub code_block: StyleRefinement,
|
|
pub code_block_overflow_x_scroll: bool,
|
|
pub inline_code: TextStyleRefinement,
|
|
pub block_quote: TextStyleRefinement,
|
|
pub link: TextStyleRefinement,
|
|
pub link_callback: Option<LinkStyleCallback>,
|
|
pub rule_color: Hsla,
|
|
pub block_quote_border_color: Hsla,
|
|
pub block_quote_kind_colors: BlockQuoteKindColors,
|
|
pub syntax: Arc<SyntaxTheme>,
|
|
pub selection_background_color: Hsla,
|
|
pub heading: StyleRefinement,
|
|
pub heading_level_styles: Option<HeadingLevelStyles>,
|
|
pub height_is_multiple_of_line_height: bool,
|
|
pub prevent_mouse_interaction: bool,
|
|
pub table_columns_min_size: bool,
|
|
}
|
|
|
|
impl Default for MarkdownStyle {
|
|
fn default() -> Self {
|
|
Self {
|
|
base_text_style: Default::default(),
|
|
container_style: Default::default(),
|
|
code_block: Default::default(),
|
|
code_block_overflow_x_scroll: false,
|
|
inline_code: Default::default(),
|
|
block_quote: Default::default(),
|
|
link: Default::default(),
|
|
link_callback: None,
|
|
rule_color: Default::default(),
|
|
block_quote_border_color: Default::default(),
|
|
block_quote_kind_colors: Default::default(),
|
|
syntax: Arc::new(SyntaxTheme::default()),
|
|
selection_background_color: Default::default(),
|
|
heading: Default::default(),
|
|
heading_level_styles: None,
|
|
height_is_multiple_of_line_height: false,
|
|
prevent_mouse_interaction: false,
|
|
table_columns_min_size: false,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy)]
|
|
pub enum MarkdownFont {
|
|
Agent,
|
|
Editor,
|
|
Preview,
|
|
}
|
|
|
|
impl MarkdownStyle {
|
|
pub fn themed(font: MarkdownFont, window: &Window, cx: &App) -> Self {
|
|
let colors = cx.theme().colors();
|
|
let syntax = cx.theme().syntax().clone();
|
|
Self::themed_with_overrides(font, colors, &syntax, window, cx)
|
|
}
|
|
|
|
/// Like [`Self::themed`], but takes explicit [`ThemeColors`] and
|
|
/// [`SyntaxTheme`] so callers (e.g. the markdown preview) can render the
|
|
/// markdown using a theme other than the active editor theme.
|
|
pub fn themed_with_overrides(
|
|
font: MarkdownFont,
|
|
colors: &theme::ThemeColors,
|
|
syntax: &Arc<SyntaxTheme>,
|
|
window: &Window,
|
|
cx: &App,
|
|
) -> Self {
|
|
let theme_settings = ThemeSettings::get_global(cx);
|
|
let is_preview = matches!(font, MarkdownFont::Preview);
|
|
|
|
let buffer_font_weight = theme_settings.buffer_font.weight;
|
|
let (buffer_font_size, ui_font_size) = match font {
|
|
MarkdownFont::Agent => (
|
|
theme_settings.agent_buffer_font_size(cx),
|
|
theme_settings.agent_ui_font_size(cx),
|
|
),
|
|
MarkdownFont::Editor | MarkdownFont::Preview => (
|
|
theme_settings.buffer_font_size(cx),
|
|
theme_settings.ui_font_size(cx),
|
|
),
|
|
};
|
|
|
|
let body_font_family = if is_preview {
|
|
theme_settings.markdown_preview_font_family().clone()
|
|
} else {
|
|
theme_settings.ui_font.family.clone()
|
|
};
|
|
let code_font_family = if is_preview {
|
|
theme_settings.markdown_preview_code_font_family().clone()
|
|
} else {
|
|
theme_settings.buffer_font.family.clone()
|
|
};
|
|
|
|
let text_color = colors.text;
|
|
|
|
let mut text_style = window.text_style();
|
|
let line_height = buffer_font_size * 1.75;
|
|
|
|
text_style.refine(&TextStyleRefinement {
|
|
font_family: Some(body_font_family),
|
|
font_fallbacks: theme_settings.ui_font.fallbacks.clone(),
|
|
font_features: Some(theme_settings.ui_font.features.clone()),
|
|
font_size: Some(ui_font_size.into()),
|
|
line_height: Some(line_height.into()),
|
|
color: Some(text_color),
|
|
..Default::default()
|
|
});
|
|
|
|
MarkdownStyle {
|
|
base_text_style: text_style.clone(),
|
|
syntax: syntax.clone(),
|
|
selection_background_color: colors.element_selection_background,
|
|
rule_color: colors.border,
|
|
block_quote_border_color: colors.border,
|
|
block_quote_kind_colors: {
|
|
let status = cx.theme().status();
|
|
BlockQuoteKindColors {
|
|
note: status.info,
|
|
tip: status.success,
|
|
important: status.info,
|
|
warning: status.warning,
|
|
caution: status.error,
|
|
}
|
|
},
|
|
code_block_overflow_x_scroll: true,
|
|
code_block: StyleRefinement {
|
|
padding: EdgesRefinement {
|
|
top: Some(DefiniteLength::Absolute(AbsoluteLength::Pixels(px(8.)))),
|
|
left: Some(DefiniteLength::Absolute(AbsoluteLength::Pixels(px(8.)))),
|
|
right: Some(DefiniteLength::Absolute(AbsoluteLength::Pixels(px(8.)))),
|
|
bottom: Some(DefiniteLength::Absolute(AbsoluteLength::Pixels(px(8.)))),
|
|
},
|
|
margin: EdgesRefinement {
|
|
top: Some(Length::Definite(px(8.).into())),
|
|
left: Some(Length::Definite(px(0.).into())),
|
|
right: Some(Length::Definite(px(0.).into())),
|
|
bottom: Some(Length::Definite(px(12.).into())),
|
|
},
|
|
border_style: Some(BorderStyle::Solid),
|
|
border_widths: EdgesRefinement {
|
|
top: Some(AbsoluteLength::Pixels(px(1.))),
|
|
left: Some(AbsoluteLength::Pixels(px(1.))),
|
|
right: Some(AbsoluteLength::Pixels(px(1.))),
|
|
bottom: Some(AbsoluteLength::Pixels(px(1.))),
|
|
},
|
|
border_color: Some(colors.border_variant),
|
|
background: Some(colors.editor_background.into()),
|
|
text: TextStyleRefinement {
|
|
font_family: Some(code_font_family.clone()),
|
|
font_fallbacks: theme_settings.buffer_font.fallbacks.clone(),
|
|
font_features: Some(theme_settings.buffer_font.features.clone()),
|
|
font_size: Some(buffer_font_size.into()),
|
|
font_weight: Some(buffer_font_weight),
|
|
..Default::default()
|
|
},
|
|
..Default::default()
|
|
},
|
|
inline_code: TextStyleRefinement {
|
|
font_family: Some(code_font_family),
|
|
font_fallbacks: theme_settings.buffer_font.fallbacks.clone(),
|
|
font_features: Some(theme_settings.buffer_font.features.clone()),
|
|
font_size: Some(buffer_font_size.into()),
|
|
font_weight: Some(buffer_font_weight),
|
|
background_color: Some(colors.editor_foreground.opacity(0.08)),
|
|
..Default::default()
|
|
},
|
|
link: TextStyleRefinement {
|
|
background_color: Some(colors.editor_foreground.opacity(0.025)),
|
|
color: Some(colors.text_accent),
|
|
underline: Some(UnderlineStyle {
|
|
color: Some(colors.text_accent.opacity(0.5)),
|
|
thickness: px(1.),
|
|
..Default::default()
|
|
}),
|
|
..Default::default()
|
|
},
|
|
heading_level_styles: matches!(font, MarkdownFont::Agent).then_some(
|
|
HeadingLevelStyles {
|
|
h1: Some(TextStyleRefinement {
|
|
font_size: Some(rems(1.15).into()),
|
|
..Default::default()
|
|
}),
|
|
h2: Some(TextStyleRefinement {
|
|
font_size: Some(rems(1.1).into()),
|
|
..Default::default()
|
|
}),
|
|
h3: Some(TextStyleRefinement {
|
|
font_size: Some(rems(1.05).into()),
|
|
..Default::default()
|
|
}),
|
|
h4: Some(TextStyleRefinement {
|
|
font_size: Some(rems(1.).into()),
|
|
..Default::default()
|
|
}),
|
|
h5: Some(TextStyleRefinement {
|
|
font_size: Some(rems(0.95).into()),
|
|
..Default::default()
|
|
}),
|
|
h6: Some(TextStyleRefinement {
|
|
font_size: Some(rems(0.875).into()),
|
|
..Default::default()
|
|
}),
|
|
},
|
|
),
|
|
..Default::default()
|
|
}
|
|
}
|
|
|
|
pub fn with_buffer_font(mut self, cx: &App) -> Self {
|
|
let theme_settings = ThemeSettings::get_global(cx);
|
|
self.base_text_style.font_family = theme_settings.buffer_font.family.clone();
|
|
self.base_text_style.font_fallbacks = theme_settings.buffer_font.fallbacks.clone();
|
|
self.base_text_style.font_features = theme_settings.buffer_font.features.clone();
|
|
self.base_text_style.font_weight = theme_settings.buffer_font.weight;
|
|
self
|
|
}
|
|
|
|
pub fn with_muted_text(mut self, cx: &App) -> Self {
|
|
let colors = cx.theme().colors();
|
|
self.base_text_style.color = colors.text_muted;
|
|
self
|
|
}
|
|
}
|
|
|
|
pub struct Markdown {
|
|
source: SharedString,
|
|
selection: Selection,
|
|
pressed_link: Option<RenderedLink>,
|
|
pressed_footnote_ref: Option<RenderedFootnoteRef>,
|
|
autoscroll_request: Option<usize>,
|
|
active_root_block: Option<usize>,
|
|
parsed_markdown: ParsedMarkdown,
|
|
images_by_source_offset: HashMap<usize, Arc<Image>>,
|
|
should_reparse: bool,
|
|
pending_parse: Option<Task<()>>,
|
|
focus_handle: FocusHandle,
|
|
language_registry: Option<Arc<LanguageRegistry>>,
|
|
fallback_code_block_language: Option<LanguageName>,
|
|
options: MarkdownOptions,
|
|
mermaid_state: MermaidState,
|
|
_mermaid_theme_subscription: Option<Subscription>,
|
|
mermaid_showing_code: HashSet<usize>,
|
|
copied_code_blocks: HashSet<ElementId>,
|
|
wrapped_code_blocks: HashSet<usize>,
|
|
code_block_scroll_handles: BTreeMap<usize, ScrollHandle>,
|
|
context_menu_link: Option<SharedString>,
|
|
context_menu_selected_text: Option<String>,
|
|
search_highlights: Vec<Range<usize>>,
|
|
active_search_highlight: Option<usize>,
|
|
}
|
|
|
|
#[derive(Clone, Copy, Default)]
|
|
pub struct MarkdownOptions {
|
|
pub parse_links_only: bool,
|
|
pub parse_html: bool,
|
|
pub render_mermaid_diagrams: bool,
|
|
pub parse_heading_slugs: bool,
|
|
pub render_metadata_blocks: bool,
|
|
}
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
pub enum CopyButtonVisibility {
|
|
Hidden,
|
|
AlwaysVisible,
|
|
VisibleOnHover,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
|
pub enum WrapButtonVisibility {
|
|
Hidden,
|
|
AlwaysVisible,
|
|
VisibleOnHover,
|
|
}
|
|
|
|
pub enum CodeBlockRenderer {
|
|
Default {
|
|
copy_button_visibility: CopyButtonVisibility,
|
|
wrap_button_visibility: WrapButtonVisibility,
|
|
border: bool,
|
|
},
|
|
Custom {
|
|
render: CodeBlockRenderFn,
|
|
/// A function that can modify the parent container after the code block
|
|
/// content has been appended as a child element.
|
|
transform: Option<CodeBlockTransformFn>,
|
|
},
|
|
}
|
|
|
|
pub type CodeBlockRenderFn = Arc<
|
|
dyn Fn(
|
|
&CodeBlockKind,
|
|
&ParsedMarkdown,
|
|
Range<usize>,
|
|
CodeBlockMetadata,
|
|
&mut Window,
|
|
&App,
|
|
) -> Div,
|
|
>;
|
|
|
|
pub type CodeBlockTransformFn =
|
|
Arc<dyn Fn(AnyDiv, Range<usize>, CodeBlockMetadata, &mut Window, &App) -> AnyDiv>;
|
|
|
|
actions!(
|
|
markdown,
|
|
[
|
|
/// Copies the selected text to the clipboard.
|
|
Copy,
|
|
/// Copies the selected text as markdown to the clipboard.
|
|
CopyAsMarkdown
|
|
]
|
|
);
|
|
|
|
enum EscapeAction {
|
|
PassThrough,
|
|
Nbsp(usize),
|
|
DoubleNewline,
|
|
PrefixBackslash,
|
|
}
|
|
|
|
impl EscapeAction {
|
|
fn output_len(&self, c: char) -> usize {
|
|
match self {
|
|
Self::PassThrough => c.len_utf8(),
|
|
Self::Nbsp(count) => count * '\u{00A0}'.len_utf8(),
|
|
Self::DoubleNewline => 2,
|
|
Self::PrefixBackslash => '\\'.len_utf8() + c.len_utf8(),
|
|
}
|
|
}
|
|
|
|
fn write_to(&self, c: char, output: &mut String) {
|
|
match self {
|
|
Self::PassThrough => output.push(c),
|
|
Self::Nbsp(count) => {
|
|
for _ in 0..*count {
|
|
output.push('\u{00A0}');
|
|
}
|
|
}
|
|
Self::DoubleNewline => {
|
|
output.push('\n');
|
|
output.push('\n');
|
|
}
|
|
Self::PrefixBackslash => {
|
|
// '\\' is a single backslash in Rust, e.g. '|' -> '\|'
|
|
output.push('\\');
|
|
output.push(c);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
struct MarkdownEscaper {
|
|
in_leading_whitespace: bool,
|
|
}
|
|
|
|
impl MarkdownEscaper {
|
|
const TAB_SIZE: usize = 4;
|
|
|
|
fn new() -> Self {
|
|
Self {
|
|
in_leading_whitespace: true,
|
|
}
|
|
}
|
|
|
|
fn next(&mut self, c: char) -> EscapeAction {
|
|
let action = if self.in_leading_whitespace && c == '\t' {
|
|
EscapeAction::Nbsp(Self::TAB_SIZE)
|
|
} else if self.in_leading_whitespace && c == ' ' {
|
|
EscapeAction::Nbsp(1)
|
|
} else if c == '\n' {
|
|
EscapeAction::DoubleNewline
|
|
} else if c.is_ascii_punctuation() {
|
|
EscapeAction::PrefixBackslash
|
|
} else {
|
|
EscapeAction::PassThrough
|
|
};
|
|
|
|
self.in_leading_whitespace =
|
|
c == '\n' || (self.in_leading_whitespace && (c == ' ' || c == '\t'));
|
|
action
|
|
}
|
|
}
|
|
|
|
impl Markdown {
|
|
pub fn new(
|
|
source: SharedString,
|
|
language_registry: Option<Arc<LanguageRegistry>>,
|
|
fallback_code_block_language: Option<LanguageName>,
|
|
cx: &mut Context<Self>,
|
|
) -> Self {
|
|
Self::new_with_options(
|
|
source,
|
|
language_registry,
|
|
fallback_code_block_language,
|
|
MarkdownOptions::default(),
|
|
cx,
|
|
)
|
|
}
|
|
|
|
pub fn new_with_options(
|
|
source: SharedString,
|
|
language_registry: Option<Arc<LanguageRegistry>>,
|
|
fallback_code_block_language: Option<LanguageName>,
|
|
options: MarkdownOptions,
|
|
cx: &mut Context<Self>,
|
|
) -> Self {
|
|
let focus_handle = cx.focus_handle();
|
|
|
|
let theme_subscription = if options.render_mermaid_diagrams {
|
|
Some(
|
|
cx.observe_global::<theme::GlobalTheme>(|this: &mut Self, cx| {
|
|
this.invalidate_mermaid_cache(cx);
|
|
}),
|
|
)
|
|
} else {
|
|
None
|
|
};
|
|
let mut this = Self {
|
|
source,
|
|
selection: Selection::default(),
|
|
pressed_link: None,
|
|
pressed_footnote_ref: None,
|
|
autoscroll_request: None,
|
|
active_root_block: None,
|
|
should_reparse: false,
|
|
images_by_source_offset: Default::default(),
|
|
parsed_markdown: ParsedMarkdown::default(),
|
|
pending_parse: None,
|
|
focus_handle,
|
|
language_registry,
|
|
fallback_code_block_language,
|
|
options,
|
|
mermaid_state: MermaidState::default(),
|
|
_mermaid_theme_subscription: theme_subscription,
|
|
mermaid_showing_code: HashSet::default(),
|
|
copied_code_blocks: HashSet::default(),
|
|
wrapped_code_blocks: HashSet::default(),
|
|
code_block_scroll_handles: BTreeMap::default(),
|
|
context_menu_link: None,
|
|
context_menu_selected_text: None,
|
|
search_highlights: Vec::new(),
|
|
active_search_highlight: None,
|
|
};
|
|
this.parse(cx);
|
|
this
|
|
}
|
|
|
|
pub fn new_text(source: SharedString, cx: &mut Context<Self>) -> Self {
|
|
Self::new_with_options(
|
|
source,
|
|
None,
|
|
None,
|
|
MarkdownOptions {
|
|
parse_links_only: true,
|
|
..Default::default()
|
|
},
|
|
cx,
|
|
)
|
|
}
|
|
|
|
fn is_code_block_wrapped(&self, id: usize) -> bool {
|
|
self.wrapped_code_blocks.contains(&id)
|
|
}
|
|
|
|
fn toggle_code_block_wrap(&mut self, id: usize) {
|
|
if !self.wrapped_code_blocks.remove(&id) {
|
|
self.wrapped_code_blocks.insert(id);
|
|
}
|
|
}
|
|
|
|
fn code_block_scroll_handle(&mut self, id: usize) -> ScrollHandle {
|
|
self.code_block_scroll_handles
|
|
.entry(id)
|
|
.or_insert_with(ScrollHandle::new)
|
|
.clone()
|
|
}
|
|
|
|
fn retain_code_block_scroll_handles(&mut self, ids: &HashSet<usize>) {
|
|
self.code_block_scroll_handles
|
|
.retain(|id, _| ids.contains(id));
|
|
}
|
|
|
|
pub fn invalidate_mermaid_cache(&mut self, cx: &mut Context<Self>) {
|
|
if !self.options.render_mermaid_diagrams || self.parsed_markdown.mermaid_diagrams.is_empty()
|
|
{
|
|
return;
|
|
}
|
|
|
|
self.mermaid_state.clear();
|
|
self.mermaid_state.update(&self.parsed_markdown, cx);
|
|
cx.notify();
|
|
}
|
|
|
|
pub(crate) fn is_mermaid_showing_code(&self, source_offset: usize) -> bool {
|
|
self.mermaid_showing_code.contains(&source_offset)
|
|
}
|
|
|
|
pub(crate) fn toggle_mermaid_tab(&mut self, source_offset: usize) {
|
|
if !self.mermaid_showing_code.remove(&source_offset) {
|
|
self.mermaid_showing_code.insert(source_offset);
|
|
}
|
|
}
|
|
|
|
fn clear_code_block_scroll_handles(&mut self) {
|
|
self.code_block_scroll_handles.clear();
|
|
}
|
|
|
|
fn autoscroll_code_block(&self, source_index: usize, cursor_position: Point<Pixels>) {
|
|
let Some((_, scroll_handle)) = self
|
|
.code_block_scroll_handles
|
|
.range(..=source_index)
|
|
.next_back()
|
|
else {
|
|
return;
|
|
};
|
|
|
|
let bounds = scroll_handle.bounds();
|
|
if cursor_position.y < bounds.top() || cursor_position.y > bounds.bottom() {
|
|
return;
|
|
}
|
|
|
|
let horizontal_delta = if cursor_position.x < bounds.left() {
|
|
bounds.left() - cursor_position.x
|
|
} else if cursor_position.x > bounds.right() {
|
|
bounds.right() - cursor_position.x
|
|
} else {
|
|
return;
|
|
};
|
|
|
|
let offset = scroll_handle.offset();
|
|
scroll_handle.set_offset(point(offset.x + horizontal_delta, offset.y));
|
|
}
|
|
|
|
pub fn is_parsing(&self) -> bool {
|
|
self.pending_parse.is_some()
|
|
}
|
|
|
|
pub fn scroll_to_heading(&mut self, slug: &str, cx: &mut Context<Self>) -> Option<usize> {
|
|
if let Some(source_index) = self.parsed_markdown.heading_slugs.get(slug).copied() {
|
|
self.autoscroll_request = Some(source_index);
|
|
cx.notify();
|
|
Some(source_index)
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
pub fn source(&self) -> &str {
|
|
&self.source
|
|
}
|
|
|
|
pub fn first_code_block_language(&self) -> Option<Arc<Language>> {
|
|
self.parsed_markdown.events.iter().find_map(|(_, event)| {
|
|
let MarkdownEvent::Start(MarkdownTag::CodeBlock { kind, .. }) = event else {
|
|
return None;
|
|
};
|
|
|
|
match kind {
|
|
CodeBlockKind::FencedLang(language) => self
|
|
.parsed_markdown
|
|
.languages_by_name
|
|
.get(language)
|
|
.cloned(),
|
|
CodeBlockKind::FencedSrc(path_range) => self
|
|
.parsed_markdown
|
|
.languages_by_path
|
|
.get(&path_range.path)
|
|
.cloned(),
|
|
CodeBlockKind::Fenced | CodeBlockKind::Indented => None,
|
|
}
|
|
})
|
|
}
|
|
|
|
pub fn append(&mut self, text: &str, cx: &mut Context<Self>) {
|
|
self.source = SharedString::new(self.source.to_string() + text);
|
|
self.parse(cx);
|
|
}
|
|
|
|
pub fn replace(&mut self, source: impl Into<SharedString>, cx: &mut Context<Self>) {
|
|
self.source = source.into();
|
|
self.parse(cx);
|
|
}
|
|
|
|
pub fn request_autoscroll_to_source_index(
|
|
&mut self,
|
|
source_index: usize,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.autoscroll_request = Some(source_index);
|
|
cx.refresh_windows();
|
|
}
|
|
|
|
fn footnote_definition_content_start(&self, label: &SharedString) -> Option<usize> {
|
|
self.parsed_markdown
|
|
.footnote_definitions
|
|
.get(label)
|
|
.copied()
|
|
}
|
|
|
|
pub fn set_active_root_for_source_index(
|
|
&mut self,
|
|
source_index: Option<usize>,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let active_root_block =
|
|
source_index.and_then(|index| self.parsed_markdown.root_block_for_source_index(index));
|
|
if self.active_root_block == active_root_block {
|
|
return;
|
|
}
|
|
|
|
self.active_root_block = active_root_block;
|
|
cx.notify();
|
|
}
|
|
|
|
pub fn reset(&mut self, source: SharedString, cx: &mut Context<Self>) {
|
|
if source == self.source() {
|
|
return;
|
|
}
|
|
self.source = source;
|
|
self.selection = Selection::default();
|
|
self.autoscroll_request = None;
|
|
self.pending_parse = None;
|
|
self.should_reparse = false;
|
|
self.search_highlights.clear();
|
|
self.active_search_highlight = None;
|
|
// Don't clear parsed_markdown here - keep existing content visible until new parse completes
|
|
self.parse(cx);
|
|
}
|
|
|
|
#[cfg(any(test, feature = "test-support"))]
|
|
pub fn parsed_markdown(&self) -> &ParsedMarkdown {
|
|
&self.parsed_markdown
|
|
}
|
|
|
|
pub fn escape(s: &str) -> Cow<'_, str> {
|
|
let output_len: usize = {
|
|
let mut escaper = MarkdownEscaper::new();
|
|
s.chars().map(|c| escaper.next(c).output_len(c)).sum()
|
|
};
|
|
|
|
if output_len == s.len() {
|
|
return s.into();
|
|
}
|
|
|
|
let mut escaper = MarkdownEscaper::new();
|
|
let mut output = String::with_capacity(output_len);
|
|
for c in s.chars() {
|
|
escaper.next(c).write_to(c, &mut output);
|
|
}
|
|
output.into()
|
|
}
|
|
|
|
pub fn selected_text(&self) -> Option<String> {
|
|
if self.selection.end <= self.selection.start {
|
|
None
|
|
} else {
|
|
Some(self.source[self.selection.start..self.selection.end].to_string())
|
|
}
|
|
}
|
|
|
|
pub fn set_search_highlights(
|
|
&mut self,
|
|
highlights: Vec<Range<usize>>,
|
|
active: Option<usize>,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.search_highlights = highlights;
|
|
self.active_search_highlight = active;
|
|
cx.notify();
|
|
}
|
|
|
|
pub fn clear_search_highlights(&mut self, cx: &mut Context<Self>) {
|
|
if !self.search_highlights.is_empty() || self.active_search_highlight.is_some() {
|
|
self.search_highlights.clear();
|
|
self.active_search_highlight = None;
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
pub fn set_active_search_highlight(&mut self, active: Option<usize>, cx: &mut Context<Self>) {
|
|
if self.active_search_highlight != active {
|
|
self.active_search_highlight = active;
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
pub fn search_highlights(&self) -> &[Range<usize>] {
|
|
&self.search_highlights
|
|
}
|
|
|
|
pub fn active_search_highlight(&self) -> Option<usize> {
|
|
self.active_search_highlight
|
|
}
|
|
|
|
fn copy(&self, text: &RenderedText, _: &mut Window, cx: &mut Context<Self>) {
|
|
if self.selection.end <= self.selection.start {
|
|
return;
|
|
}
|
|
let text = text.text_for_range(self.selection.start..self.selection.end);
|
|
cx.write_to_clipboard(ClipboardItem::new_string(text));
|
|
}
|
|
|
|
fn copy_as_markdown(&mut self, _: &mut Window, cx: &mut Context<Self>) {
|
|
if let Some(text) = self.context_menu_selected_text.take() {
|
|
cx.write_to_clipboard(ClipboardItem::new_string(text));
|
|
return;
|
|
}
|
|
if self.selection.end <= self.selection.start {
|
|
return;
|
|
}
|
|
let text = self.source[self.selection.start..self.selection.end].to_string();
|
|
cx.write_to_clipboard(ClipboardItem::new_string(text));
|
|
}
|
|
|
|
fn capture_for_context_menu(&mut self, link: Option<SharedString>) {
|
|
self.context_menu_selected_text = self.selected_text();
|
|
self.context_menu_link = link;
|
|
}
|
|
|
|
/// Returns the URL of the link that was most recently right-clicked, if any.
|
|
/// This is set during a right-click mouse-down event and can be read by parent
|
|
/// views to include a "Copy Link" item in their context menus.
|
|
pub fn context_menu_link(&self) -> Option<&SharedString> {
|
|
self.context_menu_link.as_ref()
|
|
}
|
|
|
|
fn parse(&mut self, cx: &mut Context<Self>) {
|
|
if self.source.is_empty() {
|
|
self.should_reparse = false;
|
|
self.pending_parse.take();
|
|
self.parsed_markdown = ParsedMarkdown {
|
|
source: self.source.clone(),
|
|
..Default::default()
|
|
};
|
|
self.active_root_block = None;
|
|
self.images_by_source_offset.clear();
|
|
self.mermaid_state.clear();
|
|
cx.notify();
|
|
cx.refresh_windows();
|
|
return;
|
|
}
|
|
|
|
if self.pending_parse.is_some() {
|
|
self.should_reparse = true;
|
|
return;
|
|
}
|
|
self.should_reparse = false;
|
|
self.pending_parse = Some(self.start_background_parse(cx));
|
|
}
|
|
|
|
fn start_background_parse(&self, cx: &Context<Self>) -> Task<()> {
|
|
let source = self.source.clone();
|
|
let should_parse_links_only = self.options.parse_links_only;
|
|
let should_parse_html = self.options.parse_html;
|
|
let should_render_mermaid_diagrams = self.options.render_mermaid_diagrams;
|
|
let should_parse_heading_slugs = self.options.parse_heading_slugs;
|
|
let should_parse_metadata_blocks = self.options.render_metadata_blocks;
|
|
let language_registry = self.language_registry.clone();
|
|
let fallback = self.fallback_code_block_language.clone();
|
|
|
|
let parsed = cx.background_spawn(async move {
|
|
if should_parse_links_only {
|
|
return (
|
|
ParsedMarkdown {
|
|
events: Arc::from(parse_links_only(source.as_ref())),
|
|
source,
|
|
languages_by_name: TreeMap::default(),
|
|
languages_by_path: TreeMap::default(),
|
|
root_block_starts: Arc::default(),
|
|
html_blocks: BTreeMap::default(),
|
|
metadata_blocks: BTreeMap::default(),
|
|
mermaid_diagrams: BTreeMap::default(),
|
|
heading_slugs: HashMap::default(),
|
|
footnote_definitions: HashMap::default(),
|
|
},
|
|
Default::default(),
|
|
);
|
|
}
|
|
|
|
let parsed = parse_markdown_with_options(
|
|
&source,
|
|
should_parse_html,
|
|
should_parse_heading_slugs,
|
|
should_parse_metadata_blocks,
|
|
);
|
|
let events = parsed.events;
|
|
let language_names = parsed.language_names;
|
|
let paths = parsed.language_paths;
|
|
let root_block_starts = parsed.root_block_starts;
|
|
let html_blocks = parsed.html_blocks;
|
|
let metadata_blocks = parsed.metadata_blocks;
|
|
let heading_slugs = parsed.heading_slugs;
|
|
let footnote_definitions = parsed.footnote_definitions;
|
|
let mermaid_diagrams = if should_render_mermaid_diagrams {
|
|
extract_mermaid_diagrams(&source, &events)
|
|
} else {
|
|
BTreeMap::default()
|
|
};
|
|
let mut images_by_source_offset = HashMap::default();
|
|
let mut languages_by_name = TreeMap::default();
|
|
let mut languages_by_path = TreeMap::default();
|
|
if let Some(registry) = language_registry.as_ref() {
|
|
for name in language_names {
|
|
let language = if !name.is_empty() {
|
|
registry.language_for_name_or_extension(&name).left_future()
|
|
} else if let Some(fallback) = &fallback {
|
|
registry.language_for_name(fallback.as_ref()).right_future()
|
|
} else {
|
|
continue;
|
|
};
|
|
if let Ok(language) = language.await {
|
|
languages_by_name.insert(name, language);
|
|
}
|
|
}
|
|
|
|
for path in paths {
|
|
if let Ok(language) = registry
|
|
.load_language_for_file_path(Path::new(path.as_ref()))
|
|
.await
|
|
{
|
|
languages_by_path.insert(path, language);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (range, event) in &events {
|
|
if let MarkdownEvent::Start(MarkdownTag::Image { dest_url, .. }) = event
|
|
&& let Some(data_url) = dest_url.strip_prefix("data:")
|
|
{
|
|
let Some((mime_info, data)) = data_url.split_once(',') else {
|
|
continue;
|
|
};
|
|
let Some((mime_type, encoding)) = mime_info.split_once(';') else {
|
|
continue;
|
|
};
|
|
let Some(format) = ImageFormat::from_mime_type(mime_type) else {
|
|
continue;
|
|
};
|
|
let is_base64 = encoding == "base64";
|
|
if is_base64
|
|
&& let Some(bytes) = base64::prelude::BASE64_STANDARD
|
|
.decode(data)
|
|
.log_with_level(Level::Debug)
|
|
{
|
|
let image = Arc::new(Image::from_bytes(format, bytes));
|
|
images_by_source_offset.insert(range.start, image);
|
|
}
|
|
}
|
|
}
|
|
|
|
(
|
|
ParsedMarkdown {
|
|
source,
|
|
events: Arc::from(events),
|
|
languages_by_name,
|
|
languages_by_path,
|
|
root_block_starts: Arc::from(root_block_starts),
|
|
html_blocks,
|
|
metadata_blocks,
|
|
mermaid_diagrams,
|
|
heading_slugs,
|
|
footnote_definitions,
|
|
},
|
|
images_by_source_offset,
|
|
)
|
|
});
|
|
|
|
cx.spawn(async move |this, cx| {
|
|
let (parsed, images_by_source_offset) = parsed.await;
|
|
|
|
this.update(cx, |this, cx| {
|
|
this.parsed_markdown = parsed;
|
|
this.images_by_source_offset = images_by_source_offset;
|
|
if this.active_root_block.is_some_and(|block_index| {
|
|
block_index >= this.parsed_markdown.root_block_starts.len()
|
|
}) {
|
|
this.active_root_block = None;
|
|
}
|
|
if this.options.render_mermaid_diagrams {
|
|
let parsed_markdown = this.parsed_markdown.clone();
|
|
this.mermaid_state.update(&parsed_markdown, cx);
|
|
this.mermaid_showing_code
|
|
.retain(|offset| parsed_markdown.mermaid_diagrams.contains_key(offset));
|
|
} else {
|
|
this.mermaid_state.clear();
|
|
this.mermaid_showing_code.clear();
|
|
}
|
|
this.pending_parse.take();
|
|
if this.should_reparse {
|
|
this.parse(cx);
|
|
}
|
|
cx.notify();
|
|
cx.refresh_windows();
|
|
})
|
|
.ok();
|
|
})
|
|
}
|
|
}
|
|
|
|
impl Focusable for Markdown {
|
|
fn focus_handle(&self, _cx: &App) -> FocusHandle {
|
|
self.focus_handle.clone()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Default, Clone)]
|
|
enum SelectMode {
|
|
#[default]
|
|
Character,
|
|
Word(Range<usize>),
|
|
Line(Range<usize>),
|
|
All,
|
|
}
|
|
|
|
#[derive(Clone, Default)]
|
|
struct Selection {
|
|
start: usize,
|
|
end: usize,
|
|
reversed: bool,
|
|
pending: bool,
|
|
mode: SelectMode,
|
|
}
|
|
|
|
impl Selection {
|
|
fn set_head(&mut self, head: usize, rendered_text: &RenderedText) {
|
|
match &self.mode {
|
|
SelectMode::Character => {
|
|
if head < self.tail() {
|
|
if !self.reversed {
|
|
self.end = self.start;
|
|
self.reversed = true;
|
|
}
|
|
self.start = head;
|
|
} else {
|
|
if self.reversed {
|
|
self.start = self.end;
|
|
self.reversed = false;
|
|
}
|
|
self.end = head;
|
|
}
|
|
}
|
|
SelectMode::Word(original_range) | SelectMode::Line(original_range) => {
|
|
let head_range = if matches!(self.mode, SelectMode::Word(_)) {
|
|
rendered_text.surrounding_word_range(head)
|
|
} else {
|
|
rendered_text.surrounding_line_range(head)
|
|
};
|
|
|
|
if head < original_range.start {
|
|
self.start = head_range.start;
|
|
self.end = original_range.end;
|
|
self.reversed = true;
|
|
} else if head >= original_range.end {
|
|
self.start = original_range.start;
|
|
self.end = head_range.end;
|
|
self.reversed = false;
|
|
} else {
|
|
self.start = original_range.start;
|
|
self.end = original_range.end;
|
|
self.reversed = false;
|
|
}
|
|
}
|
|
SelectMode::All => {
|
|
self.start = 0;
|
|
self.end = rendered_text
|
|
.lines
|
|
.last()
|
|
.map(|line| line.source_end)
|
|
.unwrap_or(0);
|
|
self.reversed = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
fn tail(&self) -> usize {
|
|
if self.reversed { self.end } else { self.start }
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, Default)]
|
|
pub struct ParsedMarkdown {
|
|
pub source: SharedString,
|
|
pub events: Arc<[(Range<usize>, MarkdownEvent)]>,
|
|
pub languages_by_name: TreeMap<SharedString, Arc<Language>>,
|
|
pub languages_by_path: TreeMap<Arc<str>, Arc<Language>>,
|
|
pub root_block_starts: Arc<[usize]>,
|
|
pub(crate) html_blocks: BTreeMap<usize, html::html_parser::ParsedHtmlBlock>,
|
|
pub(crate) metadata_blocks: BTreeMap<usize, ParsedMetadataBlock>,
|
|
pub(crate) mermaid_diagrams: BTreeMap<usize, ParsedMarkdownMermaidDiagram>,
|
|
pub heading_slugs: HashMap<SharedString, usize>,
|
|
pub footnote_definitions: HashMap<SharedString, usize>,
|
|
}
|
|
|
|
impl ParsedMarkdown {
|
|
pub fn source(&self) -> &SharedString {
|
|
&self.source
|
|
}
|
|
|
|
pub fn events(&self) -> &Arc<[(Range<usize>, MarkdownEvent)]> {
|
|
&self.events
|
|
}
|
|
|
|
pub fn root_block_starts(&self) -> &Arc<[usize]> {
|
|
&self.root_block_starts
|
|
}
|
|
|
|
pub fn root_block_for_source_index(&self, source_index: usize) -> Option<usize> {
|
|
if self.root_block_starts.is_empty() {
|
|
return None;
|
|
}
|
|
|
|
let partition = self
|
|
.root_block_starts
|
|
.partition_point(|block_start| *block_start <= source_index);
|
|
|
|
Some(partition.saturating_sub(1))
|
|
}
|
|
}
|
|
|
|
pub enum AutoscrollBehavior {
|
|
/// Propagate the request up the element tree for the nearest
|
|
/// scrollable ancestor (e.g. `List`) to handle.
|
|
Propagate,
|
|
/// Directly control a specific scroll handle.
|
|
Controlled(ScrollHandle),
|
|
}
|
|
|
|
pub struct MarkdownElement {
|
|
markdown: Entity<Markdown>,
|
|
style: MarkdownStyle,
|
|
code_block_renderer: CodeBlockRenderer,
|
|
on_url_click: Option<Box<dyn Fn(SharedString, &mut Window, &mut App)>>,
|
|
code_span_link: Option<CodeSpanLinkCallback>,
|
|
on_source_click: Option<SourceClickCallback>,
|
|
on_checkbox_toggle: Option<CheckboxToggleCallback>,
|
|
image_resolver: Option<Box<dyn Fn(&str) -> Option<ImageSource>>>,
|
|
show_root_block_markers: bool,
|
|
autoscroll: AutoscrollBehavior,
|
|
}
|
|
|
|
impl MarkdownElement {
|
|
pub fn new(markdown: Entity<Markdown>, style: MarkdownStyle) -> Self {
|
|
Self {
|
|
markdown,
|
|
style,
|
|
code_block_renderer: CodeBlockRenderer::Default {
|
|
copy_button_visibility: CopyButtonVisibility::VisibleOnHover,
|
|
wrap_button_visibility: WrapButtonVisibility::Hidden,
|
|
border: false,
|
|
},
|
|
on_url_click: None,
|
|
code_span_link: None,
|
|
on_source_click: None,
|
|
on_checkbox_toggle: None,
|
|
image_resolver: None,
|
|
show_root_block_markers: false,
|
|
autoscroll: AutoscrollBehavior::Propagate,
|
|
}
|
|
}
|
|
|
|
#[cfg(any(test, feature = "test-support"))]
|
|
pub fn rendered_text(
|
|
markdown: Entity<Markdown>,
|
|
cx: &mut gpui::VisualTestContext,
|
|
style: impl FnOnce(&Window, &App) -> MarkdownStyle,
|
|
) -> String {
|
|
use gpui::size;
|
|
|
|
let (text, _) = cx.draw(
|
|
Default::default(),
|
|
size(px(600.0), px(600.0)),
|
|
|window, cx| Self::new(markdown, style(window, cx)),
|
|
);
|
|
text.text
|
|
.lines
|
|
.iter()
|
|
.map(|line| line.layout.wrapped_text())
|
|
.collect::<Vec<_>>()
|
|
.join("\n")
|
|
}
|
|
|
|
pub fn code_block_renderer(mut self, variant: CodeBlockRenderer) -> Self {
|
|
self.code_block_renderer = variant;
|
|
self
|
|
}
|
|
|
|
pub fn on_url_click(
|
|
mut self,
|
|
handler: impl Fn(SharedString, &mut Window, &mut App) + 'static,
|
|
) -> Self {
|
|
self.on_url_click = Some(Box::new(handler));
|
|
self
|
|
}
|
|
|
|
pub fn on_code_span_link(
|
|
mut self,
|
|
callback: impl Fn(&str, &App) -> Option<SharedString> + 'static,
|
|
) -> Self {
|
|
self.code_span_link = Some(Arc::new(callback));
|
|
self
|
|
}
|
|
|
|
pub fn on_source_click(
|
|
mut self,
|
|
handler: impl Fn(usize, usize, &mut Window, &mut App) -> bool + 'static,
|
|
) -> Self {
|
|
self.on_source_click = Some(Box::new(handler));
|
|
self
|
|
}
|
|
|
|
pub fn on_checkbox_toggle(
|
|
mut self,
|
|
handler: impl Fn(Range<usize>, bool, &mut Window, &mut App) + 'static,
|
|
) -> Self {
|
|
self.on_checkbox_toggle = Some(Rc::new(handler));
|
|
self
|
|
}
|
|
|
|
pub fn image_resolver(
|
|
mut self,
|
|
resolver: impl Fn(&str) -> Option<ImageSource> + 'static,
|
|
) -> Self {
|
|
self.image_resolver = Some(Box::new(resolver));
|
|
self
|
|
}
|
|
|
|
pub fn show_root_block_markers(mut self) -> Self {
|
|
self.show_root_block_markers = true;
|
|
self
|
|
}
|
|
|
|
pub fn scroll_handle(mut self, scroll_handle: ScrollHandle) -> Self {
|
|
self.autoscroll = AutoscrollBehavior::Controlled(scroll_handle);
|
|
self
|
|
}
|
|
|
|
fn push_markdown_code_span(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
text: &str,
|
|
range: Range<usize>,
|
|
cx: &App,
|
|
) {
|
|
let link_url = if builder.code_block_stack.is_empty()
|
|
&& builder.link_depth == 0
|
|
&& !self.style.prevent_mouse_interaction
|
|
{
|
|
self.code_span_link
|
|
.as_ref()
|
|
.and_then(|callback| callback(text, cx))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
if let Some(url) = link_url {
|
|
builder.push_link(url.clone(), range.clone());
|
|
let link_style = self
|
|
.style
|
|
.link_callback
|
|
.as_ref()
|
|
.and_then(|callback| callback(url.as_ref(), cx))
|
|
.unwrap_or_else(|| self.style.link.clone());
|
|
builder.push_text_style(self.style.inline_code.clone());
|
|
builder.push_text_style(link_style);
|
|
builder.push_text(text, range);
|
|
builder.pop_text_style();
|
|
builder.pop_text_style();
|
|
} else {
|
|
builder.push_text_style(self.style.inline_code.clone());
|
|
builder.push_text(text, range);
|
|
builder.pop_text_style();
|
|
}
|
|
}
|
|
|
|
fn push_markdown_image(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
range: &Range<usize>,
|
|
source: ImageSource,
|
|
dest_url: SharedString,
|
|
alt_text: Option<SharedString>,
|
|
width: Option<DefiniteLength>,
|
|
height: Option<DefiniteLength>,
|
|
) {
|
|
builder.modify_current_div(|el| el.flex().flex_row().flex_wrap().items_start());
|
|
|
|
let image_element = div().min_w_0().child(
|
|
img(source)
|
|
.id(("markdown-image", range.start))
|
|
.min_w_0()
|
|
.max_w_full()
|
|
.rounded_md()
|
|
.when_some(height, |this, height| this.h(height))
|
|
.when_some(width, |this, width| this.w(width))
|
|
.with_fallback(move || image_fallback_element(dest_url.clone(), alt_text.clone())),
|
|
);
|
|
|
|
builder.push_image_child(image_element);
|
|
}
|
|
|
|
fn push_markdown_paragraph(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
range: &Range<usize>,
|
|
markdown_end: usize,
|
|
text_align_override: Option<TextAlign>,
|
|
) {
|
|
let align = text_align_override.unwrap_or(self.style.base_text_style.text_align);
|
|
let mut paragraph = div().when(!self.style.height_is_multiple_of_line_height, |el| {
|
|
el.mb_2().line_height(rems(1.3))
|
|
});
|
|
|
|
paragraph = match align {
|
|
TextAlign::Center => paragraph.text_center(),
|
|
TextAlign::Left => paragraph.text_left(),
|
|
TextAlign::Right => paragraph.text_right(),
|
|
};
|
|
|
|
builder.push_text_style(TextStyleRefinement {
|
|
text_align: Some(align),
|
|
..Default::default()
|
|
});
|
|
builder.push_div(paragraph, range, markdown_end);
|
|
}
|
|
|
|
fn pop_markdown_paragraph(&self, builder: &mut MarkdownElementBuilder) {
|
|
builder.pop_div();
|
|
builder.pop_text_style();
|
|
}
|
|
|
|
fn push_markdown_heading(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
level: pulldown_cmark::HeadingLevel,
|
|
range: &Range<usize>,
|
|
markdown_end: usize,
|
|
text_align_override: Option<TextAlign>,
|
|
) {
|
|
let align = text_align_override.unwrap_or(self.style.base_text_style.text_align);
|
|
let mut heading = div().mt_4().mb_2();
|
|
heading = apply_heading_style(heading, level, self.style.heading_level_styles.as_ref());
|
|
|
|
heading = match align {
|
|
TextAlign::Center => heading.text_center(),
|
|
TextAlign::Left => heading.text_left(),
|
|
TextAlign::Right => heading.text_right(),
|
|
};
|
|
|
|
let mut heading_style = self.style.heading.clone();
|
|
let heading_text_style = heading_style.text_style().clone();
|
|
heading.style().refine(&heading_style);
|
|
|
|
builder.push_text_style(TextStyleRefinement {
|
|
text_align: Some(align),
|
|
..heading_text_style
|
|
});
|
|
builder.push_div(heading, range, markdown_end);
|
|
}
|
|
|
|
fn pop_markdown_heading(&self, builder: &mut MarkdownElementBuilder) {
|
|
builder.pop_div();
|
|
builder.pop_text_style();
|
|
}
|
|
|
|
fn push_markdown_block_quote(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
kind: Option<pulldown_cmark::BlockQuoteKind>,
|
|
range: &Range<usize>,
|
|
markdown_end: usize,
|
|
) {
|
|
let border_color = self
|
|
.style
|
|
.block_quote_kind_colors
|
|
.for_kind(kind, self.style.block_quote_border_color);
|
|
|
|
let header = kind.map(|kind| {
|
|
let (icon_name, label) = match kind {
|
|
BlockQuoteKind::Note => (IconName::Info, "Note"),
|
|
BlockQuoteKind::Tip => (IconName::Sparkle, "Tip"),
|
|
BlockQuoteKind::Important => (IconName::Chat, "Important"),
|
|
BlockQuoteKind::Warning => (IconName::Warning, "Warning"),
|
|
BlockQuoteKind::Caution => (IconName::Stop, "Caution"),
|
|
};
|
|
h_flex()
|
|
.gap_1()
|
|
.items_center()
|
|
.mb_1()
|
|
.child(
|
|
Icon::new(icon_name)
|
|
.size(IconSize::Small)
|
|
.color(Color::Custom(border_color)),
|
|
)
|
|
.child(
|
|
Label::new(label)
|
|
.color(Color::Custom(border_color))
|
|
.weight(FontWeight::BOLD),
|
|
)
|
|
.into_any_element()
|
|
});
|
|
|
|
let block_div = div().pl_4().mb_2().border_l_4().border_color(border_color);
|
|
let block_div = match header {
|
|
Some(header) => block_div.child(header),
|
|
None => block_div,
|
|
};
|
|
|
|
builder.push_text_style(self.style.block_quote.clone());
|
|
builder.push_div(block_div, range, markdown_end);
|
|
}
|
|
|
|
fn pop_markdown_block_quote(&self, builder: &mut MarkdownElementBuilder) {
|
|
builder.pop_div();
|
|
builder.pop_text_style();
|
|
}
|
|
|
|
fn push_metadata_block(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
source: &str,
|
|
metadata_block: &ParsedMetadataBlock,
|
|
markdown_end: usize,
|
|
cx: &App,
|
|
) {
|
|
let content_range = &metadata_block.content_range;
|
|
if let Some(rows) = metadata_block.rows.as_deref() {
|
|
builder.push_div(
|
|
div()
|
|
.grid()
|
|
.grid_cols(2)
|
|
.w_full()
|
|
.mb_2()
|
|
.border_1()
|
|
.border_color(cx.theme().colors().border)
|
|
.rounded_sm()
|
|
.overflow_hidden(),
|
|
content_range,
|
|
markdown_end,
|
|
);
|
|
|
|
for (row_index, row) in rows.iter().enumerate() {
|
|
self.push_metadata_cell(
|
|
builder,
|
|
source,
|
|
row.key.clone(),
|
|
content_range,
|
|
markdown_end,
|
|
MetadataCellStyle {
|
|
row_index,
|
|
is_key: true,
|
|
},
|
|
cx,
|
|
);
|
|
self.push_metadata_cell(
|
|
builder,
|
|
source,
|
|
row.value.clone(),
|
|
content_range,
|
|
markdown_end,
|
|
MetadataCellStyle {
|
|
row_index,
|
|
is_key: false,
|
|
},
|
|
cx,
|
|
);
|
|
}
|
|
|
|
builder.pop_div();
|
|
} else {
|
|
let mut metadata_block = div().w_full().rounded_md();
|
|
metadata_block.style().refine(&self.style.code_block);
|
|
builder.push_text_style(self.style.code_block.text.to_owned());
|
|
builder.push_code_block(None);
|
|
builder.push_div(metadata_block, content_range, markdown_end);
|
|
builder.push_text(&source[content_range.clone()], content_range.clone());
|
|
builder.trim_trailing_newline();
|
|
builder.pop_div();
|
|
builder.pop_code_block();
|
|
builder.pop_text_style();
|
|
}
|
|
}
|
|
|
|
fn push_metadata_cell(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
source: &str,
|
|
text_range: Range<usize>,
|
|
block_range: &Range<usize>,
|
|
markdown_end: usize,
|
|
cell_style: MetadataCellStyle,
|
|
cx: &App,
|
|
) {
|
|
builder.push_div(
|
|
div()
|
|
.flex()
|
|
.flex_col()
|
|
.min_w_0()
|
|
.px_2()
|
|
.py_1()
|
|
.border_color(cx.theme().colors().border)
|
|
.when(cell_style.row_index > 0, |this| this.border_t_1())
|
|
.when(!cell_style.is_key, |this| this.border_l_1())
|
|
.when(cell_style.is_key, |this| {
|
|
this.bg(cx.theme().colors().panel_background)
|
|
}),
|
|
block_range,
|
|
markdown_end,
|
|
);
|
|
|
|
let text_style = if cell_style.is_key {
|
|
TextStyleRefinement {
|
|
color: Some(cx.theme().colors().text_muted),
|
|
font_weight: Some(FontWeight::SEMIBOLD),
|
|
..Default::default()
|
|
}
|
|
} else {
|
|
TextStyleRefinement::default()
|
|
};
|
|
builder.push_text_style(text_style);
|
|
builder.push_text(&source[text_range.clone()], text_range);
|
|
builder.pop_text_style();
|
|
builder.pop_div();
|
|
}
|
|
|
|
fn push_markdown_list_item(
|
|
&self,
|
|
builder: &mut MarkdownElementBuilder,
|
|
bullet: AnyElement,
|
|
range: &Range<usize>,
|
|
markdown_end: usize,
|
|
) {
|
|
builder.push_div(
|
|
div()
|
|
.when(!self.style.height_is_multiple_of_line_height, |el| {
|
|
el.mb_1().gap_1().line_height(rems(1.3))
|
|
})
|
|
.h_flex()
|
|
.items_start()
|
|
.child(bullet),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
// Without `w_0`, text doesn't wrap to the width of the container.
|
|
builder.push_div(div().flex_1().w_0(), range, markdown_end);
|
|
}
|
|
|
|
fn pop_markdown_list_item(&self, builder: &mut MarkdownElementBuilder) {
|
|
builder.pop_div();
|
|
builder.pop_div();
|
|
}
|
|
|
|
fn paint_highlight_range(
|
|
start: usize,
|
|
end: usize,
|
|
color: Hsla,
|
|
rendered_text: &RenderedText,
|
|
window: &mut Window,
|
|
) {
|
|
for bounds in rendered_text.bounds_for_source_range(start..end) {
|
|
window.paint_quad(quad(
|
|
bounds,
|
|
Pixels::ZERO,
|
|
color,
|
|
Edges::default(),
|
|
Hsla::transparent_black(),
|
|
BorderStyle::default(),
|
|
));
|
|
}
|
|
}
|
|
|
|
fn paint_selection(&self, rendered_text: &RenderedText, window: &mut Window, cx: &mut App) {
|
|
let selection = self.markdown.read(cx).selection.clone();
|
|
Self::paint_highlight_range(
|
|
selection.start,
|
|
selection.end,
|
|
self.style.selection_background_color,
|
|
rendered_text,
|
|
window,
|
|
);
|
|
}
|
|
|
|
fn paint_search_highlights(
|
|
&self,
|
|
rendered_text: &RenderedText,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) {
|
|
let markdown = self.markdown.read(cx);
|
|
let active_index = markdown.active_search_highlight;
|
|
let colors = cx.theme().colors();
|
|
|
|
for (i, highlight_range) in markdown.search_highlights.iter().enumerate() {
|
|
let color = if Some(i) == active_index {
|
|
colors.search_active_match_background
|
|
} else {
|
|
colors.search_match_background
|
|
};
|
|
Self::paint_highlight_range(
|
|
highlight_range.start,
|
|
highlight_range.end,
|
|
color,
|
|
rendered_text,
|
|
window,
|
|
);
|
|
}
|
|
}
|
|
|
|
fn paint_mouse_listeners(
|
|
&mut self,
|
|
hitbox: &Hitbox,
|
|
rendered_text: &RenderedText,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) {
|
|
if self.style.prevent_mouse_interaction {
|
|
return;
|
|
}
|
|
|
|
let is_hovering_clickable = hitbox.is_hovered(window)
|
|
&& !self.markdown.read(cx).selection.pending
|
|
&& rendered_text
|
|
.source_index_for_position(window.mouse_position())
|
|
.ok()
|
|
.is_some_and(|source_index| {
|
|
rendered_text.link_for_source_index(source_index).is_some()
|
|
|| rendered_text
|
|
.footnote_ref_for_source_index(source_index)
|
|
.is_some()
|
|
});
|
|
|
|
if is_hovering_clickable {
|
|
window.set_cursor_style(CursorStyle::PointingHand, hitbox);
|
|
} else {
|
|
window.set_cursor_style(CursorStyle::IBeam, hitbox);
|
|
}
|
|
|
|
let on_open_url = self.on_url_click.take();
|
|
let on_source_click = self.on_source_click.take();
|
|
|
|
self.on_mouse_event(window, cx, {
|
|
let hitbox = hitbox.clone();
|
|
let rendered_text = rendered_text.clone();
|
|
move |markdown, event: &MouseDownEvent, phase, window, _cx| {
|
|
if phase.capture()
|
|
&& event.button == MouseButton::Right
|
|
&& hitbox.is_hovered(window)
|
|
{
|
|
let link = rendered_text
|
|
.source_index_for_position(event.position)
|
|
.ok()
|
|
.and_then(|ix| rendered_text.link_for_source_index(ix))
|
|
.map(|link| link.destination_url.clone());
|
|
markdown.capture_for_context_menu(link);
|
|
}
|
|
}
|
|
});
|
|
|
|
self.on_mouse_event(window, cx, {
|
|
let rendered_text = rendered_text.clone();
|
|
let hitbox = hitbox.clone();
|
|
move |markdown, event: &MouseDownEvent, phase, window, cx| {
|
|
if hitbox.is_hovered(window) {
|
|
if phase.bubble() && event.button != MouseButton::Right {
|
|
let position_result =
|
|
rendered_text.source_index_for_position(event.position);
|
|
|
|
if let Ok(source_index) = position_result {
|
|
if let Some(footnote_ref) =
|
|
rendered_text.footnote_ref_for_source_index(source_index)
|
|
{
|
|
markdown.pressed_footnote_ref = Some(footnote_ref.clone());
|
|
} else if let Some(link) =
|
|
rendered_text.link_for_source_index(source_index)
|
|
{
|
|
markdown.pressed_link = Some(link.clone());
|
|
}
|
|
}
|
|
|
|
if markdown.pressed_footnote_ref.is_none()
|
|
&& markdown.pressed_link.is_none()
|
|
{
|
|
let source_index = match position_result {
|
|
Ok(ix) | Err(ix) => ix,
|
|
};
|
|
if let Some(handler) = on_source_click.as_ref() {
|
|
let blocked = handler(source_index, event.click_count, window, cx);
|
|
if blocked {
|
|
markdown.selection = Selection::default();
|
|
markdown.pressed_link = None;
|
|
window.prevent_default();
|
|
cx.notify();
|
|
return;
|
|
}
|
|
}
|
|
let (range, mode, reversed) = match event.click_count {
|
|
1 if event.modifiers.shift => {
|
|
let tail = markdown.selection.tail();
|
|
let reversed = source_index < tail;
|
|
let range = if reversed {
|
|
source_index..tail
|
|
} else {
|
|
tail..source_index
|
|
};
|
|
(range, SelectMode::Character, reversed)
|
|
}
|
|
1 => {
|
|
let range = source_index..source_index;
|
|
(range, SelectMode::Character, false)
|
|
}
|
|
2 => {
|
|
let range = rendered_text.surrounding_word_range(source_index);
|
|
(range.clone(), SelectMode::Word(range), false)
|
|
}
|
|
3 => {
|
|
let range = rendered_text.surrounding_line_range(source_index);
|
|
(range.clone(), SelectMode::Line(range), false)
|
|
}
|
|
_ => {
|
|
let range = 0..rendered_text
|
|
.lines
|
|
.last()
|
|
.map(|line| line.source_end)
|
|
.unwrap_or(0);
|
|
(range, SelectMode::All, false)
|
|
}
|
|
};
|
|
markdown.selection = Selection {
|
|
start: range.start,
|
|
end: range.end,
|
|
reversed,
|
|
pending: true,
|
|
mode,
|
|
};
|
|
window.focus(&markdown.focus_handle, cx);
|
|
}
|
|
|
|
window.prevent_default();
|
|
cx.notify();
|
|
}
|
|
} else if phase.capture() && event.button == MouseButton::Left {
|
|
markdown.selection = Selection::default();
|
|
markdown.pressed_link = None;
|
|
cx.notify();
|
|
}
|
|
}
|
|
});
|
|
self.on_mouse_event(window, cx, {
|
|
let rendered_text = rendered_text.clone();
|
|
let hitbox = hitbox.clone();
|
|
let was_hovering_clickable = is_hovering_clickable;
|
|
move |markdown, event: &MouseMoveEvent, phase, window, cx| {
|
|
if phase.capture() {
|
|
return;
|
|
}
|
|
|
|
if markdown.selection.pending {
|
|
let source_index = match rendered_text.source_index_for_position(event.position)
|
|
{
|
|
Ok(ix) | Err(ix) => ix,
|
|
};
|
|
markdown.selection.set_head(source_index, &rendered_text);
|
|
markdown.autoscroll_code_block(source_index, event.position);
|
|
markdown.autoscroll_request = Some(source_index);
|
|
cx.notify();
|
|
} else {
|
|
let is_hovering_clickable = hitbox.is_hovered(window)
|
|
&& rendered_text
|
|
.source_index_for_position(event.position)
|
|
.ok()
|
|
.is_some_and(|source_index| {
|
|
rendered_text.link_for_source_index(source_index).is_some()
|
|
|| rendered_text
|
|
.footnote_ref_for_source_index(source_index)
|
|
.is_some()
|
|
});
|
|
if is_hovering_clickable != was_hovering_clickable {
|
|
cx.notify();
|
|
}
|
|
}
|
|
}
|
|
});
|
|
self.on_mouse_event(window, cx, {
|
|
let rendered_text = rendered_text.clone();
|
|
move |markdown, event: &MouseUpEvent, phase, window, cx| {
|
|
if phase.bubble() {
|
|
let source_index = rendered_text.source_index_for_position(event.position).ok();
|
|
if let Some(pressed_footnote_ref) = markdown.pressed_footnote_ref.take()
|
|
&& source_index
|
|
.and_then(|ix| rendered_text.footnote_ref_for_source_index(ix))
|
|
== Some(&pressed_footnote_ref)
|
|
{
|
|
if let Some(source_index) =
|
|
markdown.footnote_definition_content_start(&pressed_footnote_ref.label)
|
|
{
|
|
markdown.autoscroll_request = Some(source_index);
|
|
cx.notify();
|
|
}
|
|
} else if let Some(pressed_link) = markdown.pressed_link.take()
|
|
&& source_index.and_then(|ix| rendered_text.link_for_source_index(ix))
|
|
== Some(&pressed_link)
|
|
{
|
|
if let Some(open_url) = on_open_url.as_ref() {
|
|
open_url(pressed_link.destination_url, window, cx);
|
|
} else {
|
|
cx.open_url(&pressed_link.destination_url);
|
|
}
|
|
}
|
|
} else if markdown.selection.pending {
|
|
markdown.selection.pending = false;
|
|
#[cfg(any(target_os = "linux", target_os = "freebsd"))]
|
|
{
|
|
let text = rendered_text
|
|
.text_for_range(markdown.selection.start..markdown.selection.end);
|
|
cx.write_to_primary(ClipboardItem::new_string(text))
|
|
}
|
|
cx.notify();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
fn autoscroll(
|
|
&self,
|
|
rendered_text: &RenderedText,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) -> Option<()> {
|
|
let autoscroll_index = self
|
|
.markdown
|
|
.update(cx, |markdown, _| markdown.autoscroll_request.take())?;
|
|
let (position, line_height) = rendered_text.position_for_source_index(autoscroll_index)?;
|
|
|
|
match &self.autoscroll {
|
|
AutoscrollBehavior::Controlled(scroll_handle) => {
|
|
let viewport = scroll_handle.bounds();
|
|
let margin = line_height * 3.;
|
|
let top_goal = viewport.top() + margin;
|
|
let bottom_goal = viewport.bottom() - margin;
|
|
let current_offset = scroll_handle.offset();
|
|
|
|
let new_offset_y = if position.y < top_goal {
|
|
current_offset.y + (top_goal - position.y)
|
|
} else if position.y + line_height > bottom_goal {
|
|
current_offset.y + (bottom_goal - (position.y + line_height))
|
|
} else {
|
|
current_offset.y
|
|
};
|
|
|
|
scroll_handle.set_offset(point(
|
|
current_offset.x,
|
|
new_offset_y.clamp(-scroll_handle.max_offset().y, Pixels::ZERO),
|
|
));
|
|
}
|
|
AutoscrollBehavior::Propagate => {
|
|
let text_style = self.style.base_text_style.clone();
|
|
let font_id = window.text_system().resolve_font(&text_style.font());
|
|
let font_size = text_style.font_size.to_pixels(window.rem_size());
|
|
let em_width = window.text_system().em_width(font_id, font_size).unwrap();
|
|
window.request_autoscroll(Bounds::from_corners(
|
|
point(position.x - 3. * em_width, position.y - 3. * line_height),
|
|
point(position.x + 3. * em_width, position.y + 3. * line_height),
|
|
));
|
|
}
|
|
}
|
|
Some(())
|
|
}
|
|
|
|
fn on_mouse_event<T: MouseEvent>(
|
|
&self,
|
|
window: &mut Window,
|
|
_cx: &mut App,
|
|
mut f: impl 'static
|
|
+ FnMut(&mut Markdown, &T, DispatchPhase, &mut Window, &mut Context<Markdown>),
|
|
) {
|
|
window.on_mouse_event({
|
|
let markdown = self.markdown.downgrade();
|
|
move |event, phase, window, cx| {
|
|
markdown
|
|
.update(cx, |markdown, cx| f(markdown, event, phase, window, cx))
|
|
.log_err();
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
impl Styled for MarkdownElement {
|
|
fn style(&mut self) -> &mut StyleRefinement {
|
|
&mut self.style.container_style
|
|
}
|
|
}
|
|
|
|
impl Element for MarkdownElement {
|
|
type RequestLayoutState = RenderedMarkdown;
|
|
type PrepaintState = Hitbox;
|
|
|
|
fn id(&self) -> Option<ElementId> {
|
|
None
|
|
}
|
|
|
|
fn source_location(&self) -> Option<&'static core::panic::Location<'static>> {
|
|
None
|
|
}
|
|
|
|
fn request_layout(
|
|
&mut self,
|
|
_id: Option<&GlobalElementId>,
|
|
_inspector_id: Option<&gpui::InspectorElementId>,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) -> (gpui::LayoutId, Self::RequestLayoutState) {
|
|
let mut builder = MarkdownElementBuilder::new(
|
|
&self.style.container_style,
|
|
self.style.base_text_style.clone(),
|
|
self.style.syntax.clone(),
|
|
);
|
|
let (parsed_markdown, images, active_root_block, render_mermaid_diagrams, mermaid_state) = {
|
|
let markdown = self.markdown.read(cx);
|
|
(
|
|
markdown.parsed_markdown.clone(),
|
|
markdown.images_by_source_offset.clone(),
|
|
markdown.active_root_block,
|
|
markdown.options.render_mermaid_diagrams,
|
|
markdown.mermaid_state.clone(),
|
|
)
|
|
};
|
|
let markdown_end = if let Some(last) = parsed_markdown.events.last() {
|
|
last.0.end
|
|
} else {
|
|
0
|
|
};
|
|
let mut code_block_ids = HashSet::default();
|
|
|
|
let mut current_img_block_range: Option<Range<usize>> = None;
|
|
let mut handled_html_block = false;
|
|
let mut rendered_mermaid_block = false;
|
|
let mut rendered_metadata_block = false;
|
|
for (index, (range, event)) in parsed_markdown.events.iter().enumerate() {
|
|
// Skip alt text for images that rendered
|
|
if let Some(current_img_block_range) = ¤t_img_block_range
|
|
&& current_img_block_range.end > range.end
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if handled_html_block {
|
|
if let MarkdownEvent::End(MarkdownTagEnd::HtmlBlock) = event {
|
|
handled_html_block = false;
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if rendered_mermaid_block {
|
|
if matches!(event, MarkdownEvent::End(MarkdownTagEnd::CodeBlock)) {
|
|
rendered_mermaid_block = false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if rendered_metadata_block {
|
|
if matches!(event, MarkdownEvent::End(MarkdownTagEnd::MetadataBlock(_))) {
|
|
rendered_metadata_block = false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
match event {
|
|
MarkdownEvent::RootStart => {
|
|
if self.show_root_block_markers {
|
|
builder.push_root_block(range, markdown_end);
|
|
}
|
|
}
|
|
MarkdownEvent::RootEnd(root_block_index) => {
|
|
if self.show_root_block_markers {
|
|
builder.pop_root_block(
|
|
active_root_block == Some(*root_block_index),
|
|
cx.theme().colors().border,
|
|
cx.theme().colors().border_variant,
|
|
);
|
|
}
|
|
}
|
|
MarkdownEvent::Start(tag) => {
|
|
match tag {
|
|
MarkdownTag::Image { dest_url, .. } => {
|
|
let alt_text = collect_image_alt_text(
|
|
&parsed_markdown.events[index..],
|
|
&parsed_markdown.source,
|
|
);
|
|
if let Some(image) = images.get(&range.start) {
|
|
current_img_block_range = Some(range.clone());
|
|
self.push_markdown_image(
|
|
&mut builder,
|
|
range,
|
|
image.clone().into(),
|
|
dest_url.clone(),
|
|
alt_text,
|
|
None,
|
|
None,
|
|
);
|
|
} else if let Some(source) = self
|
|
.image_resolver
|
|
.as_ref()
|
|
.and_then(|resolve| resolve(dest_url.as_ref()))
|
|
{
|
|
current_img_block_range = Some(range.clone());
|
|
self.push_markdown_image(
|
|
&mut builder,
|
|
range,
|
|
source,
|
|
dest_url.clone(),
|
|
alt_text,
|
|
None,
|
|
None,
|
|
);
|
|
}
|
|
}
|
|
MarkdownTag::Paragraph => {
|
|
let text_align_override = builder
|
|
.table
|
|
.current_cell_alignment()
|
|
.and_then(alignment_to_text_align);
|
|
self.push_markdown_paragraph(
|
|
&mut builder,
|
|
range,
|
|
markdown_end,
|
|
text_align_override,
|
|
);
|
|
}
|
|
MarkdownTag::Heading { level, .. } => {
|
|
let text_align_override = builder
|
|
.table
|
|
.current_cell_alignment()
|
|
.and_then(alignment_to_text_align);
|
|
self.push_markdown_heading(
|
|
&mut builder,
|
|
*level,
|
|
range,
|
|
markdown_end,
|
|
text_align_override,
|
|
);
|
|
}
|
|
MarkdownTag::BlockQuote(kind) => {
|
|
self.push_markdown_block_quote(
|
|
&mut builder,
|
|
*kind,
|
|
range,
|
|
markdown_end,
|
|
);
|
|
}
|
|
MarkdownTag::CodeBlock { kind, .. } => {
|
|
if render_mermaid_diagrams
|
|
&& let Some(mermaid_diagram) =
|
|
parsed_markdown.mermaid_diagrams.get(&range.start)
|
|
{
|
|
let showing_code =
|
|
self.markdown.read(cx).is_mermaid_showing_code(range.start);
|
|
let copy_button_visibility = match &self.code_block_renderer {
|
|
CodeBlockRenderer::Default {
|
|
copy_button_visibility,
|
|
..
|
|
} => *copy_button_visibility,
|
|
_ => CopyButtonVisibility::VisibleOnHover,
|
|
};
|
|
builder.push_sourced_element(
|
|
mermaid_diagram.content_range.clone(),
|
|
render_mermaid_diagram(
|
|
mermaid_diagram,
|
|
&mermaid_state,
|
|
&self.style,
|
|
self.markdown.clone(),
|
|
range.start,
|
|
showing_code,
|
|
copy_button_visibility,
|
|
),
|
|
);
|
|
rendered_mermaid_block = true;
|
|
continue;
|
|
}
|
|
|
|
let language = match kind {
|
|
CodeBlockKind::Fenced => None,
|
|
CodeBlockKind::FencedLang(language) => {
|
|
parsed_markdown.languages_by_name.get(language).cloned()
|
|
}
|
|
CodeBlockKind::FencedSrc(path_range) => parsed_markdown
|
|
.languages_by_path
|
|
.get(&path_range.path)
|
|
.cloned(),
|
|
_ => None,
|
|
};
|
|
|
|
let is_indented = matches!(kind, CodeBlockKind::Indented);
|
|
let scroll_handle = if self.style.code_block_overflow_x_scroll {
|
|
code_block_ids.insert(range.start);
|
|
Some(self.markdown.update(cx, |markdown, _| {
|
|
markdown.code_block_scroll_handle(range.start)
|
|
}))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
match (&self.code_block_renderer, is_indented) {
|
|
(CodeBlockRenderer::Default { .. }, _) | (_, true) => {
|
|
// This is a parent container that we can position the copy button inside.
|
|
let parent_container =
|
|
div().group("code_block").relative().w_full();
|
|
|
|
let mut parent_container: AnyDiv = if let Some(scroll_handle) =
|
|
scroll_handle.as_ref()
|
|
{
|
|
let scrollbars = Scrollbars::new(ScrollAxes::Horizontal)
|
|
.id(("markdown-code-block-scrollbar", range.start))
|
|
.tracked_scroll_handle(scroll_handle)
|
|
.with_track_along(
|
|
ScrollAxes::Horizontal,
|
|
cx.theme().colors().editor_background,
|
|
)
|
|
.notify_content();
|
|
|
|
parent_container
|
|
.rounded_lg()
|
|
.custom_scrollbars(scrollbars, window, cx)
|
|
.into()
|
|
} else {
|
|
parent_container.into()
|
|
};
|
|
|
|
if let CodeBlockRenderer::Default { border: true, .. } =
|
|
&self.code_block_renderer
|
|
{
|
|
parent_container = parent_container
|
|
.rounded_md()
|
|
.border_1()
|
|
.border_color(cx.theme().colors().border_variant);
|
|
}
|
|
|
|
parent_container.style().refine(&self.style.code_block);
|
|
builder.push_div(parent_container, range, markdown_end);
|
|
|
|
let is_wrapped =
|
|
self.markdown.read(cx).is_code_block_wrapped(range.start);
|
|
|
|
let code_block = div()
|
|
.id(("code-block", range.start))
|
|
.rounded_lg()
|
|
.map(|mut code_block| {
|
|
if is_wrapped {
|
|
code_block.w_full()
|
|
} else if let Some(scroll_handle) =
|
|
scroll_handle.as_ref()
|
|
{
|
|
code_block.style().restrict_scroll_to_axis =
|
|
Some(true);
|
|
code_block
|
|
.flex()
|
|
.overflow_x_scroll()
|
|
.track_scroll(scroll_handle)
|
|
} else {
|
|
code_block.w_full()
|
|
}
|
|
});
|
|
|
|
builder.push_text_style(self.style.code_block.text.to_owned());
|
|
builder.push_code_block(language);
|
|
builder.push_div(code_block, range, markdown_end);
|
|
}
|
|
(CodeBlockRenderer::Custom { .. }, _) => {}
|
|
}
|
|
}
|
|
MarkdownTag::HtmlBlock => {
|
|
builder.push_div(div(), range, markdown_end);
|
|
if let Some(block) = parsed_markdown.html_blocks.get(&range.start) {
|
|
self.render_html_block(block, &mut builder, markdown_end, cx);
|
|
handled_html_block = true;
|
|
}
|
|
}
|
|
MarkdownTag::List(bullet_index) => {
|
|
builder.push_list(*bullet_index);
|
|
builder.push_div(div().pl_2p5(), range, markdown_end);
|
|
}
|
|
MarkdownTag::Item => {
|
|
let bullet =
|
|
if let Some((task_range, MarkdownEvent::TaskListMarker(checked))) =
|
|
parsed_markdown.events.get(index.saturating_add(1))
|
|
{
|
|
let source = &parsed_markdown.source()[range.clone()];
|
|
let checked = *checked;
|
|
let toggle_state = if checked {
|
|
ToggleState::Selected
|
|
} else {
|
|
ToggleState::Unselected
|
|
};
|
|
|
|
let checkbox = Checkbox::new(
|
|
ElementId::Name(source.to_string().into()),
|
|
toggle_state,
|
|
)
|
|
.fill();
|
|
|
|
if let Some(on_toggle) = self.on_checkbox_toggle.clone() {
|
|
let task_source_range = task_range.clone();
|
|
checkbox
|
|
.on_click(move |_state, window, cx| {
|
|
on_toggle(
|
|
task_source_range.clone(),
|
|
!checked,
|
|
window,
|
|
cx,
|
|
);
|
|
})
|
|
.into_any_element()
|
|
} else {
|
|
checkbox.visualization_only(true).into_any_element()
|
|
}
|
|
} else if let Some(bullet_index) = builder.next_bullet_index() {
|
|
div().child(format!("{}.", bullet_index)).into_any_element()
|
|
} else {
|
|
div().child("•").into_any_element()
|
|
};
|
|
self.push_markdown_list_item(&mut builder, bullet, range, markdown_end);
|
|
}
|
|
MarkdownTag::Emphasis => builder.push_text_style(TextStyleRefinement {
|
|
font_style: Some(FontStyle::Italic),
|
|
..Default::default()
|
|
}),
|
|
MarkdownTag::Strong => builder.push_text_style(TextStyleRefinement {
|
|
font_weight: Some(FontWeight::BOLD),
|
|
..Default::default()
|
|
}),
|
|
MarkdownTag::Strikethrough => {
|
|
builder.push_text_style(TextStyleRefinement {
|
|
strikethrough: Some(StrikethroughStyle {
|
|
thickness: px(1.),
|
|
color: None,
|
|
}),
|
|
..Default::default()
|
|
})
|
|
}
|
|
MarkdownTag::Link { dest_url, .. } => {
|
|
if builder.code_block_stack.is_empty() {
|
|
builder.link_depth += 1;
|
|
builder.push_link(dest_url.clone(), range.clone());
|
|
let style = self
|
|
.style
|
|
.link_callback
|
|
.as_ref()
|
|
.and_then(|callback| callback(dest_url, cx))
|
|
.unwrap_or_else(|| self.style.link.clone());
|
|
builder.push_text_style(style)
|
|
}
|
|
}
|
|
MarkdownTag::FootnoteDefinition(label) => {
|
|
if !builder.rendered_footnote_separator {
|
|
builder.rendered_footnote_separator = true;
|
|
builder.push_div(
|
|
div()
|
|
.border_t_1()
|
|
.mt_2()
|
|
.border_color(self.style.rule_color),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
builder.pop_div();
|
|
}
|
|
builder.push_div(
|
|
div()
|
|
.pt_1()
|
|
.mb_1()
|
|
.line_height(rems(1.3))
|
|
.text_size(rems(0.85))
|
|
.h_flex()
|
|
.items_start()
|
|
.gap_2()
|
|
.child(
|
|
div().text_size(rems(0.85)).child(format!("{}.", label)),
|
|
),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
builder.push_div(div().flex_1().w_0(), range, markdown_end);
|
|
}
|
|
MarkdownTag::MetadataBlock(_) => {
|
|
if let Some(metadata_block) =
|
|
parsed_markdown.metadata_blocks.get(&range.start)
|
|
{
|
|
self.push_metadata_block(
|
|
&mut builder,
|
|
&parsed_markdown.source,
|
|
metadata_block,
|
|
markdown_end,
|
|
cx,
|
|
);
|
|
rendered_metadata_block = true;
|
|
}
|
|
}
|
|
MarkdownTag::Table(alignments) => {
|
|
builder.table.start(alignments.clone());
|
|
|
|
let column_count = alignments.len();
|
|
builder.push_div(
|
|
div()
|
|
.id(("table", range.start))
|
|
.grid()
|
|
.grid_cols(column_count as u16)
|
|
.when(self.style.table_columns_min_size, |this| {
|
|
this.grid_cols_min_content(column_count as u16)
|
|
})
|
|
.when(!self.style.table_columns_min_size, |this| {
|
|
this.grid_cols(column_count as u16)
|
|
})
|
|
.w_full()
|
|
.mb_2()
|
|
.border(px(1.5))
|
|
.border_color(cx.theme().colors().border)
|
|
.rounded_sm()
|
|
.overflow_hidden(),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
}
|
|
MarkdownTag::TableHead => {
|
|
builder.table.start_head();
|
|
builder.push_text_style(TextStyleRefinement {
|
|
font_weight: Some(FontWeight::SEMIBOLD),
|
|
..Default::default()
|
|
});
|
|
}
|
|
MarkdownTag::TableRow => {
|
|
builder.table.start_row();
|
|
}
|
|
MarkdownTag::TableCell => {
|
|
let is_header = builder.table.in_head;
|
|
let row_index = builder.table.row_index;
|
|
let col_index = builder.table.col_index;
|
|
let alignment = builder.table.current_cell_alignment();
|
|
let text_align = alignment
|
|
.and_then(alignment_to_text_align)
|
|
.unwrap_or(self.style.base_text_style.text_align);
|
|
|
|
let mut cell_div = div()
|
|
.flex()
|
|
.flex_col()
|
|
.h_full()
|
|
.when(col_index > 0, |this| this.border_l_1())
|
|
.when(row_index > 0, |this| this.border_t_1())
|
|
.border_color(cx.theme().colors().border)
|
|
.px_1()
|
|
.py_0p5()
|
|
.when(is_header, |this| {
|
|
this.bg(cx.theme().colors().title_bar_background)
|
|
})
|
|
.when(!is_header && row_index % 2 == 1, |this| {
|
|
this.bg(cx.theme().colors().panel_background)
|
|
});
|
|
|
|
cell_div = match alignment {
|
|
Some(Alignment::Center) => cell_div.items_center(),
|
|
Some(Alignment::Right) => cell_div.items_end(),
|
|
_ => cell_div,
|
|
};
|
|
|
|
builder.push_text_style(TextStyleRefinement {
|
|
text_align: Some(text_align),
|
|
..Default::default()
|
|
});
|
|
builder.push_div(cell_div, range, markdown_end);
|
|
builder.push_div(
|
|
div()
|
|
.flex()
|
|
.flex_col()
|
|
.flex_1()
|
|
.w_full()
|
|
.justify_center()
|
|
.text_align(text_align),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
}
|
|
_ => log::debug!("unsupported markdown tag {:?}", tag),
|
|
}
|
|
}
|
|
MarkdownEvent::End(tag) => match tag {
|
|
MarkdownTagEnd::Image => {
|
|
current_img_block_range.take();
|
|
}
|
|
MarkdownTagEnd::Paragraph => {
|
|
self.pop_markdown_paragraph(&mut builder);
|
|
}
|
|
MarkdownTagEnd::Heading(_) => {
|
|
self.pop_markdown_heading(&mut builder);
|
|
}
|
|
MarkdownTagEnd::BlockQuote(_kind) => {
|
|
self.pop_markdown_block_quote(&mut builder);
|
|
}
|
|
MarkdownTagEnd::CodeBlock => {
|
|
builder.trim_trailing_newline();
|
|
|
|
builder.pop_div();
|
|
builder.pop_code_block();
|
|
builder.pop_text_style();
|
|
|
|
if let CodeBlockRenderer::Default {
|
|
copy_button_visibility,
|
|
wrap_button_visibility,
|
|
..
|
|
} = &self.code_block_renderer
|
|
&& (*copy_button_visibility != CopyButtonVisibility::Hidden
|
|
|| *wrap_button_visibility != WrapButtonVisibility::Hidden)
|
|
{
|
|
let copy_button_visibility = *copy_button_visibility;
|
|
let wrap_button_visibility = *wrap_button_visibility;
|
|
builder.modify_current_div(|el| {
|
|
let content_range = parser::extract_code_block_content_range(
|
|
&parsed_markdown.source()[range.clone()],
|
|
);
|
|
let content_range = content_range.start + range.start
|
|
..content_range.end + range.start;
|
|
|
|
let code = parsed_markdown.source()[content_range].to_string();
|
|
|
|
let any_hover = copy_button_visibility
|
|
== CopyButtonVisibility::VisibleOnHover
|
|
|| wrap_button_visibility
|
|
== WrapButtonVisibility::VisibleOnHover;
|
|
let any_always = copy_button_visibility
|
|
== CopyButtonVisibility::AlwaysVisible
|
|
|| wrap_button_visibility
|
|
== WrapButtonVisibility::AlwaysVisible;
|
|
let use_hover = any_hover && !any_always;
|
|
|
|
let mut button_row = h_flex()
|
|
.gap_0p5()
|
|
.absolute()
|
|
.bg(cx.theme().colors().editor_background)
|
|
.when_else(
|
|
use_hover,
|
|
|this| {
|
|
this.top_1().right_1().visible_on_hover("code_block")
|
|
},
|
|
|this| this.top_1p5().right_1p5(),
|
|
);
|
|
|
|
if wrap_button_visibility != WrapButtonVisibility::Hidden {
|
|
let is_wrapped =
|
|
self.markdown.read(cx).is_code_block_wrapped(range.start);
|
|
button_row = button_row.child(render_wrap_code_block_button(
|
|
range.start,
|
|
is_wrapped,
|
|
self.markdown.clone(),
|
|
));
|
|
}
|
|
|
|
if copy_button_visibility != CopyButtonVisibility::Hidden {
|
|
button_row = button_row.child(render_copy_code_block_button(
|
|
range.end,
|
|
code,
|
|
self.markdown.clone(),
|
|
));
|
|
}
|
|
|
|
el.child(button_row)
|
|
});
|
|
}
|
|
|
|
// Pop the parent container.
|
|
builder.pop_div();
|
|
}
|
|
MarkdownTagEnd::HtmlBlock => builder.pop_div(),
|
|
MarkdownTagEnd::List(_) => {
|
|
builder.pop_list();
|
|
builder.pop_div();
|
|
}
|
|
MarkdownTagEnd::Item => {
|
|
self.pop_markdown_list_item(&mut builder);
|
|
}
|
|
MarkdownTagEnd::Emphasis => builder.pop_text_style(),
|
|
MarkdownTagEnd::Strong => builder.pop_text_style(),
|
|
MarkdownTagEnd::Strikethrough => builder.pop_text_style(),
|
|
MarkdownTagEnd::Link => {
|
|
if builder.code_block_stack.is_empty() {
|
|
builder.link_depth = builder.link_depth.saturating_sub(1);
|
|
builder.pop_text_style()
|
|
}
|
|
}
|
|
MarkdownTagEnd::Table => {
|
|
builder.pop_div();
|
|
builder.table.end();
|
|
}
|
|
MarkdownTagEnd::TableHead => {
|
|
builder.pop_text_style();
|
|
builder.table.end_head();
|
|
}
|
|
MarkdownTagEnd::TableRow => {
|
|
builder.table.end_row();
|
|
}
|
|
MarkdownTagEnd::TableCell => {
|
|
builder.replace_pending_checkbox(self.on_checkbox_toggle.clone());
|
|
builder.pop_div();
|
|
builder.pop_div();
|
|
builder.pop_text_style();
|
|
builder.table.end_cell();
|
|
}
|
|
MarkdownTagEnd::FootnoteDefinition => {
|
|
builder.pop_div();
|
|
builder.pop_div();
|
|
}
|
|
MarkdownTagEnd::MetadataBlock(_) => {}
|
|
_ => log::debug!("unsupported markdown tag end: {:?}", tag),
|
|
},
|
|
MarkdownEvent::Text => {
|
|
builder.push_text(&parsed_markdown.source[range.clone()], range.clone());
|
|
}
|
|
MarkdownEvent::SubstitutedText(text) => {
|
|
builder.push_text(text, range.clone());
|
|
}
|
|
MarkdownEvent::Code => {
|
|
self.push_markdown_code_span(
|
|
&mut builder,
|
|
&parsed_markdown.source[range.clone()],
|
|
range.clone(),
|
|
cx,
|
|
);
|
|
}
|
|
MarkdownEvent::Html => {
|
|
let html = &parsed_markdown.source[range.clone()];
|
|
if html.starts_with("<!--") {
|
|
builder.html_comment = true;
|
|
}
|
|
if html.trim_end().ends_with("-->") {
|
|
builder.html_comment = false;
|
|
continue;
|
|
}
|
|
if builder.html_comment {
|
|
continue;
|
|
}
|
|
builder.push_text(html, range.clone());
|
|
}
|
|
MarkdownEvent::InlineHtml => {
|
|
let html = &parsed_markdown.source[range.clone()];
|
|
if let Some(code) = html
|
|
.strip_prefix("<code>")
|
|
.and_then(|html| html.strip_suffix("</code>"))
|
|
{
|
|
let code_start = range.start + "<code>".len();
|
|
self.push_markdown_code_span(
|
|
&mut builder,
|
|
code,
|
|
code_start..code_start + code.len(),
|
|
cx,
|
|
);
|
|
continue;
|
|
}
|
|
if html.starts_with("<code>") {
|
|
builder.push_text_style(self.style.inline_code.clone());
|
|
continue;
|
|
}
|
|
if html.trim_end().starts_with("</code>") {
|
|
builder.pop_text_style();
|
|
continue;
|
|
}
|
|
builder.push_text(&parsed_markdown.source[range.clone()], range.clone());
|
|
}
|
|
MarkdownEvent::Rule => {
|
|
builder.push_div(
|
|
div()
|
|
.border_b_1()
|
|
.my_2()
|
|
.border_color(self.style.rule_color),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
builder.pop_div()
|
|
}
|
|
MarkdownEvent::SoftBreak => builder.push_text(" ", range.clone()),
|
|
MarkdownEvent::HardBreak => builder.push_text("\n", range.clone()),
|
|
MarkdownEvent::TaskListMarker(_) => {
|
|
// handled inside the `MarkdownTag::Item` case
|
|
}
|
|
MarkdownEvent::FootnoteReference(label) => {
|
|
builder.push_footnote_ref(label.clone(), range.clone());
|
|
builder.push_text_style(self.style.link.clone());
|
|
builder.push_text(&format!("[{label}]"), range.clone());
|
|
builder.pop_text_style();
|
|
}
|
|
}
|
|
}
|
|
if self.style.code_block_overflow_x_scroll {
|
|
let code_block_ids = code_block_ids;
|
|
self.markdown.update(cx, move |markdown, _| {
|
|
markdown.retain_code_block_scroll_handles(&code_block_ids);
|
|
});
|
|
} else {
|
|
self.markdown
|
|
.update(cx, |markdown, _| markdown.clear_code_block_scroll_handles());
|
|
}
|
|
let mut rendered_markdown = builder.build();
|
|
let child_layout_id = rendered_markdown.element.request_layout(window, cx);
|
|
let layout_id = window.request_layout(gpui::Style::default(), [child_layout_id], cx);
|
|
(layout_id, rendered_markdown)
|
|
}
|
|
|
|
fn prepaint(
|
|
&mut self,
|
|
_id: Option<&GlobalElementId>,
|
|
_inspector_id: Option<&gpui::InspectorElementId>,
|
|
bounds: Bounds<Pixels>,
|
|
rendered_markdown: &mut Self::RequestLayoutState,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) -> Self::PrepaintState {
|
|
let focus_handle = self.markdown.read(cx).focus_handle.clone();
|
|
window.set_focus_handle(&focus_handle, cx);
|
|
window.set_view_id(self.markdown.entity_id());
|
|
|
|
let hitbox = window.insert_hitbox(bounds, HitboxBehavior::Normal);
|
|
rendered_markdown.element.prepaint(window, cx);
|
|
self.autoscroll(&rendered_markdown.text, window, cx);
|
|
hitbox
|
|
}
|
|
|
|
fn paint(
|
|
&mut self,
|
|
_id: Option<&GlobalElementId>,
|
|
_inspector_id: Option<&gpui::InspectorElementId>,
|
|
_bounds: Bounds<Pixels>,
|
|
rendered_markdown: &mut Self::RequestLayoutState,
|
|
hitbox: &mut Self::PrepaintState,
|
|
window: &mut Window,
|
|
cx: &mut App,
|
|
) {
|
|
let mut context = KeyContext::default();
|
|
context.add("Markdown");
|
|
window.set_key_context(context);
|
|
window.on_action(std::any::TypeId::of::<crate::Copy>(), {
|
|
let entity = self.markdown.clone();
|
|
let text = rendered_markdown.text.clone();
|
|
move |_, phase, window, cx| {
|
|
let text = text.clone();
|
|
if phase == DispatchPhase::Bubble {
|
|
entity.update(cx, move |this, cx| this.copy(&text, window, cx))
|
|
}
|
|
}
|
|
});
|
|
window.on_action(std::any::TypeId::of::<crate::CopyAsMarkdown>(), {
|
|
let entity = self.markdown.clone();
|
|
move |_, phase, window, cx| {
|
|
if phase == DispatchPhase::Bubble {
|
|
entity.update(cx, move |this, cx| this.copy_as_markdown(window, cx))
|
|
}
|
|
}
|
|
});
|
|
|
|
self.paint_mouse_listeners(hitbox, &rendered_markdown.text, window, cx);
|
|
rendered_markdown.element.paint(window, cx);
|
|
self.paint_search_highlights(&rendered_markdown.text, window, cx);
|
|
self.paint_selection(&rendered_markdown.text, window, cx);
|
|
}
|
|
}
|
|
|
|
fn collect_image_alt_text(
|
|
events_from_image_start: &[(Range<usize>, MarkdownEvent)],
|
|
source: &str,
|
|
) -> Option<SharedString> {
|
|
let mut alt_text = String::new();
|
|
for (range, event) in events_from_image_start.iter().skip(1) {
|
|
match event {
|
|
MarkdownEvent::End(MarkdownTagEnd::Image) => break,
|
|
MarkdownEvent::Text => alt_text.push_str(&source[range.clone()]),
|
|
_ => {}
|
|
}
|
|
}
|
|
if alt_text.is_empty() {
|
|
None
|
|
} else {
|
|
Some(alt_text.into())
|
|
}
|
|
}
|
|
|
|
fn image_fallback_element(dest_url: SharedString, alt_text: Option<SharedString>) -> AnyElement {
|
|
let link_label = alt_text
|
|
.filter(|alt| !alt.is_empty())
|
|
.unwrap_or_else(|| dest_url.clone());
|
|
|
|
let label = format!("Failed to Load: {link_label}");
|
|
|
|
div()
|
|
.id("image-fallback")
|
|
.cursor_pointer()
|
|
.min_w_0()
|
|
.child(Label::new(label).color(Color::Warning).underline())
|
|
.tooltip(Tooltip::text(
|
|
"Image failed to load. Open `zed: log` for more details.",
|
|
))
|
|
.on_click(move |_, _, cx| cx.open_url(&dest_url))
|
|
.into_any_element()
|
|
}
|
|
|
|
fn apply_heading_style(
|
|
mut heading: Div,
|
|
level: pulldown_cmark::HeadingLevel,
|
|
custom_styles: Option<&HeadingLevelStyles>,
|
|
) -> Div {
|
|
heading = match level {
|
|
pulldown_cmark::HeadingLevel::H1 => heading.text_3xl(),
|
|
pulldown_cmark::HeadingLevel::H2 => heading.text_2xl(),
|
|
pulldown_cmark::HeadingLevel::H3 => heading.text_xl(),
|
|
pulldown_cmark::HeadingLevel::H4 => heading.text_lg(),
|
|
pulldown_cmark::HeadingLevel::H5 => heading.text_base(),
|
|
pulldown_cmark::HeadingLevel::H6 => heading.text_sm(),
|
|
};
|
|
|
|
if let Some(styles) = custom_styles {
|
|
let style_opt = match level {
|
|
pulldown_cmark::HeadingLevel::H1 => &styles.h1,
|
|
pulldown_cmark::HeadingLevel::H2 => &styles.h2,
|
|
pulldown_cmark::HeadingLevel::H3 => &styles.h3,
|
|
pulldown_cmark::HeadingLevel::H4 => &styles.h4,
|
|
pulldown_cmark::HeadingLevel::H5 => &styles.h5,
|
|
pulldown_cmark::HeadingLevel::H6 => &styles.h6,
|
|
};
|
|
|
|
if let Some(style) = style_opt {
|
|
heading.style().text = style.clone();
|
|
}
|
|
}
|
|
|
|
heading
|
|
}
|
|
|
|
fn render_wrap_code_block_button(
|
|
id: usize,
|
|
is_wrapped: bool,
|
|
markdown: Entity<Markdown>,
|
|
) -> impl IntoElement {
|
|
let (icon, tooltip) = if is_wrapped {
|
|
(IconName::TextUnwrap, "Unwrap Content")
|
|
} else {
|
|
(IconName::TextWrap, "Wrap Content")
|
|
};
|
|
|
|
IconButton::new(("wrap-code-block", id), icon)
|
|
.icon_size(IconSize::Small)
|
|
.icon_color(Color::Muted)
|
|
.tooltip(Tooltip::text(tooltip))
|
|
.on_click(move |_event, _window, cx| {
|
|
markdown.update(cx, |markdown, cx| {
|
|
markdown.toggle_code_block_wrap(id);
|
|
cx.notify();
|
|
});
|
|
})
|
|
}
|
|
|
|
fn render_copy_code_block_button(
|
|
id: usize,
|
|
code: String,
|
|
markdown: Entity<Markdown>,
|
|
) -> impl IntoElement {
|
|
let id = ElementId::named_usize("copy-markdown-code", id);
|
|
|
|
CopyButton::new(id.clone(), code.clone()).custom_on_click({
|
|
let markdown = markdown;
|
|
move |_window, cx| {
|
|
let id = id.clone();
|
|
markdown.update(cx, |this, cx| {
|
|
this.copied_code_blocks.insert(id.clone());
|
|
|
|
cx.write_to_clipboard(ClipboardItem::new_string(code.clone()));
|
|
|
|
cx.spawn(async move |this, cx| {
|
|
cx.background_executor().timer(Duration::from_secs(2)).await;
|
|
|
|
cx.update(|cx| {
|
|
this.update(cx, |this, cx| {
|
|
this.copied_code_blocks.remove(&id);
|
|
cx.notify();
|
|
})
|
|
})
|
|
.ok();
|
|
})
|
|
.detach();
|
|
});
|
|
}
|
|
})
|
|
}
|
|
|
|
impl IntoElement for MarkdownElement {
|
|
type Element = Self;
|
|
|
|
fn into_element(self) -> Self::Element {
|
|
self
|
|
}
|
|
}
|
|
|
|
pub enum AnyDiv {
|
|
Div(Div),
|
|
Stateful(Stateful<Div>),
|
|
}
|
|
|
|
impl AnyDiv {
|
|
fn into_any_element(self) -> AnyElement {
|
|
match self {
|
|
Self::Div(div) => div.into_any_element(),
|
|
Self::Stateful(div) => div.into_any_element(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<Div> for AnyDiv {
|
|
fn from(value: Div) -> Self {
|
|
Self::Div(value)
|
|
}
|
|
}
|
|
|
|
impl From<Stateful<Div>> for AnyDiv {
|
|
fn from(value: Stateful<Div>) -> Self {
|
|
Self::Stateful(value)
|
|
}
|
|
}
|
|
|
|
impl Styled for AnyDiv {
|
|
fn style(&mut self) -> &mut StyleRefinement {
|
|
match self {
|
|
Self::Div(div) => div.style(),
|
|
Self::Stateful(div) => div.style(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ParentElement for AnyDiv {
|
|
fn extend(&mut self, elements: impl IntoIterator<Item = AnyElement>) {
|
|
match self {
|
|
Self::Div(div) => div.extend(elements),
|
|
Self::Stateful(div) => div.extend(elements),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Default)]
|
|
struct TableState {
|
|
alignments: Vec<Alignment>,
|
|
in_head: bool,
|
|
row_index: usize,
|
|
col_index: usize,
|
|
}
|
|
|
|
impl TableState {
|
|
fn start(&mut self, alignments: Vec<Alignment>) {
|
|
self.alignments = alignments;
|
|
self.in_head = false;
|
|
self.row_index = 0;
|
|
self.col_index = 0;
|
|
}
|
|
|
|
fn end(&mut self) {
|
|
self.alignments.clear();
|
|
self.in_head = false;
|
|
self.row_index = 0;
|
|
self.col_index = 0;
|
|
}
|
|
|
|
fn start_head(&mut self) {
|
|
self.in_head = true;
|
|
}
|
|
|
|
fn end_head(&mut self) {
|
|
self.in_head = false;
|
|
}
|
|
|
|
fn start_row(&mut self) {
|
|
self.col_index = 0;
|
|
}
|
|
|
|
fn end_row(&mut self) {
|
|
self.row_index += 1;
|
|
}
|
|
|
|
fn end_cell(&mut self) {
|
|
self.col_index += 1;
|
|
}
|
|
|
|
fn current_cell_alignment(&self) -> Option<Alignment> {
|
|
if self.alignments.is_empty() {
|
|
return None;
|
|
}
|
|
if self.in_head {
|
|
return Some(Alignment::Center);
|
|
}
|
|
self.alignments.get(self.col_index).copied()
|
|
}
|
|
}
|
|
|
|
fn alignment_to_text_align(alignment: Alignment) -> Option<TextAlign> {
|
|
match alignment {
|
|
Alignment::Left => Some(TextAlign::Left),
|
|
Alignment::Center => Some(TextAlign::Center),
|
|
Alignment::Right => Some(TextAlign::Right),
|
|
Alignment::None => None,
|
|
}
|
|
}
|
|
|
|
struct MetadataCellStyle {
|
|
row_index: usize,
|
|
is_key: bool,
|
|
}
|
|
|
|
struct MarkdownElementBuilder {
|
|
div_stack: Vec<AnyDiv>,
|
|
rendered_lines: Vec<RenderedLine>,
|
|
pending_line: PendingLine,
|
|
rendered_links: Vec<RenderedLink>,
|
|
rendered_footnote_refs: Vec<RenderedFootnoteRef>,
|
|
current_source_index: usize,
|
|
html_comment: bool,
|
|
rendered_footnote_separator: bool,
|
|
base_text_style: TextStyle,
|
|
text_style_stack: Vec<TextStyleRefinement>,
|
|
code_block_stack: Vec<Option<Arc<Language>>>,
|
|
link_depth: usize,
|
|
list_stack: Vec<ListStackEntry>,
|
|
table: TableState,
|
|
syntax_theme: Arc<SyntaxTheme>,
|
|
}
|
|
|
|
#[derive(Default)]
|
|
struct PendingLine {
|
|
text: String,
|
|
runs: Vec<TextRun>,
|
|
source_mappings: Vec<SourceMapping>,
|
|
}
|
|
|
|
struct ListStackEntry {
|
|
bullet_index: Option<u64>,
|
|
}
|
|
|
|
impl MarkdownElementBuilder {
|
|
fn new(
|
|
container_style: &StyleRefinement,
|
|
base_text_style: TextStyle,
|
|
syntax_theme: Arc<SyntaxTheme>,
|
|
) -> Self {
|
|
Self {
|
|
div_stack: vec![{
|
|
let mut base_div = div();
|
|
base_div.style().refine(container_style);
|
|
base_div.debug_selector(|| "inner".into()).into()
|
|
}],
|
|
rendered_lines: Vec::new(),
|
|
pending_line: PendingLine::default(),
|
|
rendered_links: Vec::new(),
|
|
rendered_footnote_refs: Vec::new(),
|
|
current_source_index: 0,
|
|
html_comment: false,
|
|
rendered_footnote_separator: false,
|
|
base_text_style,
|
|
text_style_stack: Vec::new(),
|
|
code_block_stack: Vec::new(),
|
|
link_depth: 0,
|
|
list_stack: Vec::new(),
|
|
table: TableState::default(),
|
|
syntax_theme,
|
|
}
|
|
}
|
|
|
|
fn push_text_style(&mut self, style: TextStyleRefinement) {
|
|
self.text_style_stack.push(style);
|
|
}
|
|
|
|
fn text_style(&self) -> TextStyle {
|
|
let mut style = self.base_text_style.clone();
|
|
for refinement in &self.text_style_stack {
|
|
style.refine(refinement);
|
|
}
|
|
style
|
|
}
|
|
|
|
fn pop_text_style(&mut self) {
|
|
self.text_style_stack.pop();
|
|
}
|
|
|
|
fn push_div(&mut self, div: impl Into<AnyDiv>, range: &Range<usize>, markdown_end: usize) {
|
|
let mut div = div.into();
|
|
self.flush_text();
|
|
|
|
if range.start == 0 {
|
|
// Remove the top margin on the first element.
|
|
div.style().refine(&StyleRefinement {
|
|
margin: gpui::EdgesRefinement {
|
|
top: Some(Length::Definite(px(0.).into())),
|
|
left: None,
|
|
right: None,
|
|
bottom: None,
|
|
},
|
|
..Default::default()
|
|
});
|
|
}
|
|
|
|
if range.end == markdown_end {
|
|
div.style().refine(&StyleRefinement {
|
|
margin: gpui::EdgesRefinement {
|
|
top: None,
|
|
left: None,
|
|
right: None,
|
|
bottom: Some(Length::Definite(rems(0.).into())),
|
|
},
|
|
..Default::default()
|
|
});
|
|
}
|
|
|
|
self.div_stack.push(div);
|
|
}
|
|
|
|
fn push_root_block(&mut self, range: &Range<usize>, markdown_end: usize) {
|
|
self.push_div(
|
|
div().group("markdown-root-block").relative(),
|
|
range,
|
|
markdown_end,
|
|
);
|
|
self.push_div(div().pl_4(), range, markdown_end);
|
|
}
|
|
|
|
fn push_image_child(&mut self, child: impl IntoElement) {
|
|
self.flush_text();
|
|
self.div_stack
|
|
.last_mut()
|
|
.unwrap()
|
|
.extend([child.into_any_element()]);
|
|
}
|
|
|
|
fn modify_current_div(&mut self, f: impl FnOnce(AnyDiv) -> AnyDiv) {
|
|
self.flush_text();
|
|
if let Some(div) = self.div_stack.pop() {
|
|
self.div_stack.push(f(div));
|
|
}
|
|
}
|
|
|
|
fn pop_root_block(
|
|
&mut self,
|
|
is_active: bool,
|
|
active_gutter_color: Hsla,
|
|
hovered_gutter_color: Hsla,
|
|
) {
|
|
self.pop_div();
|
|
self.modify_current_div(|el| {
|
|
el.child(
|
|
div()
|
|
.h_full()
|
|
.w(px(4.0))
|
|
.when(is_active, |this| this.bg(active_gutter_color))
|
|
.group_hover("markdown-root-block", |this| {
|
|
if is_active {
|
|
this
|
|
} else {
|
|
this.bg(hovered_gutter_color)
|
|
}
|
|
})
|
|
.rounded_xs()
|
|
.absolute()
|
|
.left_0()
|
|
.top_0(),
|
|
)
|
|
});
|
|
self.pop_div();
|
|
}
|
|
|
|
fn pop_div(&mut self) {
|
|
self.flush_text();
|
|
let div = self.div_stack.pop().unwrap().into_any_element();
|
|
self.div_stack.last_mut().unwrap().extend(iter::once(div));
|
|
}
|
|
|
|
fn push_sourced_element(&mut self, source_range: Range<usize>, element: impl Into<AnyElement>) {
|
|
self.flush_text();
|
|
let anchor = self.render_source_anchor(source_range);
|
|
self.div_stack.last_mut().unwrap().extend([{
|
|
div()
|
|
.relative()
|
|
.child(anchor)
|
|
.child(element.into())
|
|
.into_any_element()
|
|
}]);
|
|
}
|
|
|
|
fn push_list(&mut self, bullet_index: Option<u64>) {
|
|
self.list_stack.push(ListStackEntry { bullet_index });
|
|
}
|
|
|
|
fn next_bullet_index(&mut self) -> Option<u64> {
|
|
self.list_stack.last_mut().and_then(|entry| {
|
|
let item_index = entry.bullet_index.as_mut()?;
|
|
*item_index += 1;
|
|
Some(*item_index - 1)
|
|
})
|
|
}
|
|
|
|
fn pop_list(&mut self) {
|
|
self.list_stack.pop();
|
|
}
|
|
|
|
fn push_code_block(&mut self, language: Option<Arc<Language>>) {
|
|
self.code_block_stack.push(language);
|
|
}
|
|
|
|
fn pop_code_block(&mut self) {
|
|
self.code_block_stack.pop();
|
|
}
|
|
|
|
fn push_link(&mut self, destination_url: SharedString, source_range: Range<usize>) {
|
|
self.rendered_links.push(RenderedLink {
|
|
source_range,
|
|
destination_url,
|
|
});
|
|
}
|
|
|
|
fn push_footnote_ref(&mut self, label: SharedString, source_range: Range<usize>) {
|
|
self.rendered_footnote_refs.push(RenderedFootnoteRef {
|
|
source_range,
|
|
label,
|
|
});
|
|
}
|
|
|
|
fn push_text(&mut self, text: &str, source_range: Range<usize>) {
|
|
self.pending_line.source_mappings.push(SourceMapping {
|
|
rendered_index: self.pending_line.text.len(),
|
|
source_index: source_range.start,
|
|
});
|
|
self.pending_line.text.push_str(text);
|
|
self.current_source_index = source_range.end;
|
|
|
|
// Compute the base text style once
|
|
let text_style = self.text_style();
|
|
|
|
if let Some(Some(language)) = self.code_block_stack.last() {
|
|
let mut offset = 0;
|
|
for (range, highlight_id) in language.highlight_text(&Rope::from(text), 0..text.len()) {
|
|
if range.start > offset {
|
|
self.pending_line
|
|
.runs
|
|
.push(text_style.to_run(range.start - offset));
|
|
}
|
|
|
|
let run_len = range.len();
|
|
if let Some(highlight) = self.syntax_theme.get(highlight_id).cloned() {
|
|
self.pending_line
|
|
.runs
|
|
.push(text_style.clone().highlight(highlight).to_run(run_len));
|
|
} else {
|
|
self.pending_line.runs.push(text_style.to_run(run_len));
|
|
}
|
|
offset = range.end;
|
|
}
|
|
|
|
if offset < text.len() {
|
|
self.pending_line
|
|
.runs
|
|
.push(text_style.to_run(text.len() - offset));
|
|
}
|
|
} else {
|
|
self.pending_line.runs.push(text_style.to_run(text.len()));
|
|
}
|
|
}
|
|
|
|
fn trim_trailing_newline(&mut self) {
|
|
if self.pending_line.text.ends_with('\n') {
|
|
self.pending_line
|
|
.text
|
|
.truncate(self.pending_line.text.len() - 1);
|
|
self.pending_line.runs.last_mut().unwrap().len -= 1;
|
|
self.current_source_index -= 1;
|
|
}
|
|
}
|
|
|
|
fn replace_pending_checkbox(&mut self, on_toggle: Option<CheckboxToggleCallback>) {
|
|
let text = &self.pending_line.text;
|
|
let trimmed = text.trim();
|
|
if trimmed != "[x]" && trimmed != "[X]" && trimmed != "[ ]" {
|
|
return;
|
|
}
|
|
let checked = trimmed != "[ ]";
|
|
|
|
let leading_ws = text.len() - text.trim_start().len();
|
|
let marker_rendered = leading_ws..leading_ws + trimmed.len();
|
|
let marker_source = self
|
|
.source_range_for_rendered(&marker_rendered)
|
|
.expect("pending checkbox text must have source mappings");
|
|
|
|
self.pending_line = PendingLine::default();
|
|
|
|
let toggle_state = if checked {
|
|
ToggleState::Selected
|
|
} else {
|
|
ToggleState::Unselected
|
|
};
|
|
let checkbox = Checkbox::new(
|
|
ElementId::Name(
|
|
format!(
|
|
"table_checkbox_{}_{}",
|
|
marker_source.start, marker_source.end
|
|
)
|
|
.into(),
|
|
),
|
|
toggle_state,
|
|
)
|
|
.fill();
|
|
|
|
let checkbox = if let Some(on_toggle) = on_toggle {
|
|
checkbox
|
|
.on_click(move |_state, window, cx| {
|
|
on_toggle(marker_source.clone(), !checked, window, cx);
|
|
})
|
|
.into_any_element()
|
|
} else {
|
|
checkbox.visualization_only(true).into_any_element()
|
|
};
|
|
|
|
let mut checkbox_container = h_flex().w_full();
|
|
checkbox_container = match self.text_style().text_align {
|
|
TextAlign::Left => checkbox_container.justify_start(),
|
|
TextAlign::Center => checkbox_container.justify_center(),
|
|
TextAlign::Right => checkbox_container.justify_end(),
|
|
};
|
|
|
|
self.div_stack
|
|
.last_mut()
|
|
.unwrap()
|
|
.extend([checkbox_container.child(checkbox).into_any_element()]);
|
|
}
|
|
|
|
fn source_range_for_rendered(&self, rendered: &Range<usize>) -> Option<Range<usize>> {
|
|
source_range_for_rendered(&self.pending_line.source_mappings, rendered)
|
|
}
|
|
|
|
fn render_source_anchor(&mut self, source_range: Range<usize>) -> AnyElement {
|
|
let mut text_style = self.base_text_style.clone();
|
|
text_style.color = Hsla::transparent_black();
|
|
let text = "\u{200B}";
|
|
let styled_text = StyledText::new(text).with_runs(vec![text_style.to_run(text.len())]);
|
|
self.rendered_lines.push(RenderedLine {
|
|
layout: styled_text.layout().clone(),
|
|
source_mappings: vec![SourceMapping {
|
|
rendered_index: 0,
|
|
source_index: source_range.start,
|
|
}],
|
|
source_end: source_range.end,
|
|
language: None,
|
|
});
|
|
div()
|
|
.absolute()
|
|
.top_0()
|
|
.left_0()
|
|
.opacity(0.)
|
|
.child(styled_text)
|
|
.into_any_element()
|
|
}
|
|
|
|
fn flush_text(&mut self) {
|
|
let line = mem::take(&mut self.pending_line);
|
|
if line.text.is_empty() {
|
|
return;
|
|
}
|
|
|
|
let text = StyledText::new(line.text).with_runs(line.runs);
|
|
self.rendered_lines.push(RenderedLine {
|
|
layout: text.layout().clone(),
|
|
source_mappings: line.source_mappings,
|
|
source_end: self.current_source_index,
|
|
language: self.code_block_stack.last().cloned().flatten(),
|
|
});
|
|
self.div_stack.last_mut().unwrap().extend([text.into_any()]);
|
|
}
|
|
|
|
fn build(mut self) -> RenderedMarkdown {
|
|
debug_assert_eq!(self.div_stack.len(), 1);
|
|
self.flush_text();
|
|
RenderedMarkdown {
|
|
element: self.div_stack.pop().unwrap().into_any_element(),
|
|
text: RenderedText {
|
|
lines: self.rendered_lines.into(),
|
|
links: self.rendered_links.into(),
|
|
footnote_refs: self.rendered_footnote_refs.into(),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
struct RenderedLine {
|
|
layout: TextLayout,
|
|
source_mappings: Vec<SourceMapping>,
|
|
source_end: usize,
|
|
language: Option<Arc<Language>>,
|
|
}
|
|
|
|
impl RenderedLine {
|
|
fn rendered_index_for_source_index(&self, source_index: usize) -> usize {
|
|
if source_index >= self.source_end {
|
|
return self.layout.len();
|
|
}
|
|
|
|
let mapping = match self
|
|
.source_mappings
|
|
.binary_search_by_key(&source_index, |probe| probe.source_index)
|
|
{
|
|
Ok(ix) => &self.source_mappings[ix],
|
|
Err(ix) => &self.source_mappings[ix - 1],
|
|
};
|
|
(mapping.rendered_index + (source_index - mapping.source_index)).min(self.layout.len())
|
|
}
|
|
|
|
fn source_index_for_rendered_index(&self, rendered_index: usize) -> usize {
|
|
if rendered_index >= self.layout.len() {
|
|
return self.source_end;
|
|
}
|
|
|
|
let mapping = match self
|
|
.source_mappings
|
|
.binary_search_by_key(&rendered_index, |probe| probe.rendered_index)
|
|
{
|
|
Ok(ix) => &self.source_mappings[ix],
|
|
Err(ix) => &self.source_mappings[ix - 1],
|
|
};
|
|
mapping.source_index + (rendered_index - mapping.rendered_index)
|
|
}
|
|
|
|
/// Returns the source index for use as an exclusive range end at a word/selection boundary.
|
|
/// When the rendered index is exactly at the start of a segment with a gap from the previous
|
|
/// segment (e.g., after stripped markdown syntax like backticks), this returns the end of the
|
|
/// previous segment rather than the start of the current one.
|
|
fn source_index_for_exclusive_rendered_end(&self, rendered_index: usize) -> usize {
|
|
if rendered_index >= self.layout.len() {
|
|
return self.source_end;
|
|
}
|
|
|
|
let ix = match self
|
|
.source_mappings
|
|
.binary_search_by_key(&rendered_index, |probe| probe.rendered_index)
|
|
{
|
|
Ok(ix) => ix,
|
|
Err(ix) => {
|
|
return self.source_mappings[ix - 1].source_index
|
|
+ (rendered_index - self.source_mappings[ix - 1].rendered_index);
|
|
}
|
|
};
|
|
|
|
// Exact match at the start of a segment. Check if there's a gap from the previous segment.
|
|
if ix > 0 {
|
|
let prev_mapping = &self.source_mappings[ix - 1];
|
|
let mapping = &self.source_mappings[ix];
|
|
let prev_segment_len = mapping.rendered_index - prev_mapping.rendered_index;
|
|
let prev_source_end = prev_mapping.source_index + prev_segment_len;
|
|
if prev_source_end < mapping.source_index {
|
|
return prev_source_end;
|
|
}
|
|
}
|
|
|
|
self.source_mappings[ix].source_index
|
|
}
|
|
|
|
fn source_index_for_position(&self, position: Point<Pixels>) -> Result<usize, usize> {
|
|
let line_rendered_index;
|
|
let out_of_bounds;
|
|
match self.layout.index_for_position(position) {
|
|
Ok(ix) => {
|
|
line_rendered_index = ix;
|
|
out_of_bounds = false;
|
|
}
|
|
Err(ix) => {
|
|
line_rendered_index = ix;
|
|
out_of_bounds = true;
|
|
}
|
|
};
|
|
let source_index = self.source_index_for_rendered_index(line_rendered_index);
|
|
if out_of_bounds {
|
|
Err(source_index)
|
|
} else {
|
|
Ok(source_index)
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug, Default)]
|
|
struct SourceMapping {
|
|
rendered_index: usize,
|
|
source_index: usize,
|
|
}
|
|
|
|
fn source_range_for_rendered(
|
|
mappings: &[SourceMapping],
|
|
rendered: &Range<usize>,
|
|
) -> Option<Range<usize>> {
|
|
if rendered.start >= rendered.end {
|
|
return None;
|
|
}
|
|
let start = source_index_for_rendered(mappings, rendered.start)?;
|
|
let end = source_index_for_rendered(mappings, rendered.end - 1)? + 1;
|
|
Some(start..end)
|
|
}
|
|
|
|
fn source_index_for_rendered(mappings: &[SourceMapping], rendered_index: usize) -> Option<usize> {
|
|
let mut last: Option<&SourceMapping> = None;
|
|
for mapping in mappings {
|
|
if mapping.rendered_index <= rendered_index {
|
|
last = Some(mapping);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
last.map(|m| m.source_index + (rendered_index - m.rendered_index))
|
|
}
|
|
|
|
pub struct RenderedMarkdown {
|
|
element: AnyElement,
|
|
text: RenderedText,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct RenderedText {
|
|
lines: Rc<[RenderedLine]>,
|
|
links: Rc<[RenderedLink]>,
|
|
footnote_refs: Rc<[RenderedFootnoteRef]>,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Eq, PartialEq)]
|
|
struct RenderedLink {
|
|
source_range: Range<usize>,
|
|
destination_url: SharedString,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Eq, PartialEq)]
|
|
struct RenderedFootnoteRef {
|
|
source_range: Range<usize>,
|
|
label: SharedString,
|
|
}
|
|
|
|
impl RenderedText {
|
|
fn bounds_for_source_range(&self, range: Range<usize>) -> Vec<Bounds<Pixels>> {
|
|
let mut all_bounds = Vec::new();
|
|
|
|
for line in self.lines.iter() {
|
|
let line_source_start = line.source_mappings.first().unwrap().source_index;
|
|
if line_source_start >= range.end {
|
|
break;
|
|
}
|
|
if line.source_end <= range.start {
|
|
continue;
|
|
}
|
|
|
|
let layout = &line.layout;
|
|
let line_bounds = layout.bounds();
|
|
let line_height = layout.line_height();
|
|
|
|
let rendered_start =
|
|
line.rendered_index_for_source_index(range.start.max(line_source_start));
|
|
let rendered_end = line.rendered_index_for_source_index(range.end.min(line.source_end));
|
|
|
|
let mut wrapped_line_start = 0;
|
|
let mut row_top = line_bounds.top();
|
|
|
|
while wrapped_line_start < rendered_end {
|
|
let Some(wrapped_line) = layout.line_layout_for_index(wrapped_line_start) else {
|
|
break;
|
|
};
|
|
|
|
let unwrapped_layout = &wrapped_line.unwrapped_layout;
|
|
let wrapped_line_end = wrapped_line_start + wrapped_line.len();
|
|
|
|
let row_ends = wrapped_line
|
|
.wrap_boundaries()
|
|
.iter()
|
|
.map(|wrap_boundary| {
|
|
let glyph = &unwrapped_layout.runs[wrap_boundary.run_ix].glyphs
|
|
[wrap_boundary.glyph_ix];
|
|
(wrapped_line_start + glyph.index, glyph.position.x)
|
|
})
|
|
.chain([(wrapped_line_end, unwrapped_layout.width)]);
|
|
|
|
let mut row_start = wrapped_line_start;
|
|
let mut row_start_x = Pixels::ZERO;
|
|
|
|
for (row_end, row_end_x) in row_ends {
|
|
let selection_start = rendered_start.max(row_start);
|
|
let selection_end = rendered_end.min(row_end);
|
|
|
|
if selection_start < selection_end {
|
|
let x_for_index = |index| {
|
|
line_bounds.left()
|
|
+ unwrapped_layout.x_for_index(index - wrapped_line_start)
|
|
- row_start_x
|
|
};
|
|
all_bounds.push(Bounds::from_corners(
|
|
point(x_for_index(selection_start), row_top),
|
|
point(x_for_index(selection_end), row_top + line_height),
|
|
));
|
|
}
|
|
|
|
row_start = row_end;
|
|
row_start_x = row_end_x;
|
|
row_top += line_height;
|
|
}
|
|
|
|
wrapped_line_start = wrapped_line_end + 1;
|
|
}
|
|
}
|
|
|
|
all_bounds
|
|
}
|
|
|
|
fn source_index_for_position(&self, position: Point<Pixels>) -> Result<usize, usize> {
|
|
let mut lines = self.lines.iter().peekable();
|
|
let mut fallback_line: Option<&RenderedLine> = None;
|
|
|
|
while let Some(line) = lines.next() {
|
|
let line_bounds = line.layout.bounds();
|
|
|
|
// Exact match: position is within bounds (handles overlapping bounds like table columns)
|
|
if line_bounds.contains(&position) {
|
|
return line.source_index_for_position(position);
|
|
}
|
|
|
|
// Track fallback for Y-coordinate based matching
|
|
if position.y <= line_bounds.bottom() && fallback_line.is_none() {
|
|
fallback_line = Some(line);
|
|
}
|
|
|
|
// Handle gap between lines
|
|
if position.y > line_bounds.bottom() {
|
|
if let Some(next_line) = lines.peek()
|
|
&& position.y < next_line.layout.bounds().top()
|
|
{
|
|
return Err(line.source_end);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fall back to Y-coordinate matched line
|
|
if let Some(line) = fallback_line {
|
|
return line.source_index_for_position(position);
|
|
}
|
|
|
|
Err(self.lines.last().map_or(0, |line| line.source_end))
|
|
}
|
|
|
|
fn position_for_source_index(&self, source_index: usize) -> Option<(Point<Pixels>, Pixels)> {
|
|
for line in self.lines.iter() {
|
|
let line_source_start = line.source_mappings.first().unwrap().source_index;
|
|
if source_index < line_source_start {
|
|
break;
|
|
} else if source_index > line.source_end {
|
|
continue;
|
|
} else {
|
|
let line_height = line.layout.line_height();
|
|
let rendered_index_within_line = line.rendered_index_for_source_index(source_index);
|
|
let position = line.layout.position_for_index(rendered_index_within_line)?;
|
|
return Some((position, line_height));
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
fn surrounding_word_range(&self, source_index: usize) -> Range<usize> {
|
|
for line in self.lines.iter() {
|
|
if source_index > line.source_end {
|
|
continue;
|
|
}
|
|
|
|
let line_rendered_start = line.source_mappings.first().unwrap().rendered_index;
|
|
let rendered_index_in_line =
|
|
line.rendered_index_for_source_index(source_index) - line_rendered_start;
|
|
let text = line.layout.text();
|
|
|
|
let scope = line.language.as_ref().map(|l| l.default_scope());
|
|
let classifier = CharClassifier::new(scope);
|
|
|
|
let mut prev_chars = text[..rendered_index_in_line].chars().rev().peekable();
|
|
let mut next_chars = text[rendered_index_in_line..].chars().peekable();
|
|
|
|
let word_kind = std::cmp::max(
|
|
prev_chars.peek().map(|&c| classifier.kind(c)),
|
|
next_chars.peek().map(|&c| classifier.kind(c)),
|
|
);
|
|
|
|
let mut start = rendered_index_in_line;
|
|
for c in prev_chars {
|
|
if Some(classifier.kind(c)) == word_kind {
|
|
start -= c.len_utf8();
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
let mut end = rendered_index_in_line;
|
|
for c in next_chars {
|
|
if Some(classifier.kind(c)) == word_kind {
|
|
end += c.len_utf8();
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return line.source_index_for_rendered_index(line_rendered_start + start)
|
|
..line.source_index_for_exclusive_rendered_end(line_rendered_start + end);
|
|
}
|
|
|
|
source_index..source_index
|
|
}
|
|
|
|
fn surrounding_line_range(&self, source_index: usize) -> Range<usize> {
|
|
for line in self.lines.iter() {
|
|
if source_index > line.source_end {
|
|
continue;
|
|
}
|
|
let line_source_start = line.source_mappings.first().unwrap().source_index;
|
|
return line_source_start..line.source_end;
|
|
}
|
|
|
|
source_index..source_index
|
|
}
|
|
|
|
fn text_for_range(&self, range: Range<usize>) -> String {
|
|
let mut accumulator = String::new();
|
|
|
|
for line in self.lines.iter() {
|
|
if range.start > line.source_end {
|
|
continue;
|
|
}
|
|
let line_source_start = line.source_mappings.first().unwrap().source_index;
|
|
if range.end < line_source_start {
|
|
break;
|
|
}
|
|
|
|
let text = line.layout.text();
|
|
|
|
let start = if range.start < line_source_start {
|
|
0
|
|
} else {
|
|
line.rendered_index_for_source_index(range.start)
|
|
};
|
|
let end = if range.end > line.source_end {
|
|
line.rendered_index_for_source_index(line.source_end)
|
|
} else {
|
|
line.rendered_index_for_source_index(range.end)
|
|
}
|
|
.min(text.len());
|
|
|
|
accumulator.push_str(&text[start..end]);
|
|
accumulator.push('\n');
|
|
}
|
|
// Remove trailing newline
|
|
accumulator.pop();
|
|
accumulator
|
|
}
|
|
|
|
fn link_for_source_index(&self, source_index: usize) -> Option<&RenderedLink> {
|
|
self.links
|
|
.iter()
|
|
.find(|link| link.source_range.contains(&source_index))
|
|
}
|
|
|
|
fn footnote_ref_for_source_index(&self, source_index: usize) -> Option<&RenderedFootnoteRef> {
|
|
self.footnote_refs
|
|
.iter()
|
|
.find(|fref| fref.source_range.contains(&source_index))
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use gpui::{TestAppContext, size};
|
|
use language::{Language, LanguageConfig, LanguageMatcher};
|
|
use std::sync::{
|
|
Arc,
|
|
atomic::{AtomicUsize, Ordering},
|
|
};
|
|
|
|
fn ensure_theme_initialized(cx: &mut TestAppContext) {
|
|
cx.update(|cx| {
|
|
if !cx.has_global::<settings::SettingsStore>() {
|
|
settings::init(cx);
|
|
}
|
|
if !cx.has_global::<theme::GlobalTheme>() {
|
|
theme_settings::init(theme::LoadThemes::JustBase, cx);
|
|
}
|
|
});
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_mappings(cx: &mut TestAppContext) {
|
|
// Formatting.
|
|
assert_mappings(
|
|
&render_markdown("He*l*lo", cx),
|
|
vec![vec![(0, 0), (1, 1), (2, 3), (3, 5), (4, 6), (5, 7)]],
|
|
);
|
|
|
|
// Multiple lines.
|
|
assert_mappings(
|
|
&render_markdown("Hello\n\nWorld", cx),
|
|
vec![
|
|
vec![(0, 0), (1, 1), (2, 2), (3, 3), (4, 4), (5, 5)],
|
|
vec![(0, 7), (1, 8), (2, 9), (3, 10), (4, 11), (5, 12)],
|
|
],
|
|
);
|
|
|
|
// Multi-byte characters.
|
|
assert_mappings(
|
|
&render_markdown("αβγ\n\nδεζ", cx),
|
|
vec![
|
|
vec![(0, 0), (2, 2), (4, 4), (6, 6)],
|
|
vec![(0, 8), (2, 10), (4, 12), (6, 14)],
|
|
],
|
|
);
|
|
|
|
// Smart quotes.
|
|
assert_mappings(&render_markdown("\"", cx), vec![vec![(0, 0), (3, 1)]]);
|
|
assert_mappings(
|
|
&render_markdown("\"hey\"", cx),
|
|
vec![vec![(0, 0), (3, 1), (4, 2), (5, 3), (6, 4), (9, 5)]],
|
|
);
|
|
|
|
// HTML Comments are ignored
|
|
assert_mappings(
|
|
&render_markdown(
|
|
"<!--\nrdoc-file=string.c\n- str.intern -> symbol\n- str.to_sym -> symbol\n-->\nReturns",
|
|
cx,
|
|
),
|
|
vec![vec![
|
|
(0, 78),
|
|
(1, 79),
|
|
(2, 80),
|
|
(3, 81),
|
|
(4, 82),
|
|
(5, 83),
|
|
(6, 84),
|
|
]],
|
|
);
|
|
}
|
|
|
|
fn render_markdown(markdown: &str, cx: &mut TestAppContext) -> RenderedText {
|
|
render_markdown_with_language_registry(markdown, None, cx)
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_frontmatter_renders_without_delimiters(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown_with_options(
|
|
"---\ntitle: Post\n---\nBody",
|
|
None,
|
|
MarkdownOptions {
|
|
render_metadata_blocks: true,
|
|
..Default::default()
|
|
},
|
|
cx,
|
|
);
|
|
assert_eq!(rendered.text_for_range(0..24), "title\nPost\nBody");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_frontmatter_falls_back_to_code_block_for_nested_yaml(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown_with_options(
|
|
"---\ntags:\n - zed\n---\nBody",
|
|
None,
|
|
MarkdownOptions {
|
|
render_metadata_blocks: true,
|
|
..Default::default()
|
|
},
|
|
cx,
|
|
);
|
|
assert_eq!(rendered.text_for_range(0..26), "tags:\n - zed\nBody");
|
|
}
|
|
|
|
fn render_markdown_with_code_span_link(
|
|
markdown: &str,
|
|
callback: impl Fn(&str, &App) -> Option<SharedString> + 'static,
|
|
cx: &mut TestAppContext,
|
|
) -> RenderedText {
|
|
render_markdown_with_code_span_link_style(markdown, MarkdownStyle::default(), callback, cx)
|
|
}
|
|
|
|
fn render_markdown_with_code_span_link_style(
|
|
markdown: &str,
|
|
style: MarkdownStyle,
|
|
callback: impl Fn(&str, &App) -> Option<SharedString> + 'static,
|
|
cx: &mut TestAppContext,
|
|
) -> RenderedText {
|
|
struct TestWindow;
|
|
|
|
impl Render for TestWindow {
|
|
fn render(&mut self, _: &mut Window, _: &mut Context<Self>) -> impl IntoElement {
|
|
div()
|
|
}
|
|
}
|
|
|
|
ensure_theme_initialized(cx);
|
|
|
|
let (_, cx) = cx.add_window_view(|_, _| TestWindow);
|
|
let markdown = cx.new(|cx| Markdown::new(markdown.to_string().into(), None, None, cx));
|
|
cx.run_until_parked();
|
|
let (rendered, _) = cx.draw(
|
|
Default::default(),
|
|
size(px(600.0), px(600.0)),
|
|
|_window, _cx| {
|
|
MarkdownElement::new(markdown, style)
|
|
.on_code_span_link(callback)
|
|
.code_block_renderer(CodeBlockRenderer::Default {
|
|
copy_button_visibility: CopyButtonVisibility::Hidden,
|
|
wrap_button_visibility: WrapButtonVisibility::Hidden,
|
|
border: false,
|
|
})
|
|
},
|
|
);
|
|
rendered.text
|
|
}
|
|
|
|
fn render_markdown_with_language_registry(
|
|
markdown: &str,
|
|
language_registry: Option<Arc<LanguageRegistry>>,
|
|
cx: &mut TestAppContext,
|
|
) -> RenderedText {
|
|
render_markdown_with_options(markdown, language_registry, MarkdownOptions::default(), cx)
|
|
}
|
|
|
|
fn render_markdown_with_options(
|
|
markdown: &str,
|
|
language_registry: Option<Arc<LanguageRegistry>>,
|
|
options: MarkdownOptions,
|
|
cx: &mut TestAppContext,
|
|
) -> RenderedText {
|
|
struct TestWindow;
|
|
|
|
impl Render for TestWindow {
|
|
fn render(&mut self, _: &mut Window, _: &mut Context<Self>) -> impl IntoElement {
|
|
div()
|
|
}
|
|
}
|
|
|
|
ensure_theme_initialized(cx);
|
|
|
|
let (_, cx) = cx.add_window_view(|_, _| TestWindow);
|
|
let markdown = cx.new(|cx| {
|
|
Markdown::new_with_options(
|
|
markdown.to_string().into(),
|
|
language_registry,
|
|
None,
|
|
options,
|
|
cx,
|
|
)
|
|
});
|
|
cx.run_until_parked();
|
|
let (rendered, _) = cx.draw(
|
|
Default::default(),
|
|
size(px(600.0), px(600.0)),
|
|
|_window, _cx| {
|
|
MarkdownElement::new(markdown, MarkdownStyle::default()).code_block_renderer(
|
|
CodeBlockRenderer::Default {
|
|
copy_button_visibility: CopyButtonVisibility::Hidden,
|
|
wrap_button_visibility: WrapButtonVisibility::Hidden,
|
|
border: false,
|
|
},
|
|
)
|
|
},
|
|
);
|
|
rendered.text
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_surrounding_word_range(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("Hello world tesεζ", cx);
|
|
|
|
// Test word selection for "Hello"
|
|
let word_range = rendered.surrounding_word_range(2); // Simulate click on 'l' in "Hello"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "Hello");
|
|
|
|
// Test word selection for "world"
|
|
let word_range = rendered.surrounding_word_range(7); // Simulate click on 'o' in "world"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "world");
|
|
|
|
// Test word selection for "tesεζ"
|
|
let word_range = rendered.surrounding_word_range(14); // Simulate click on 's' in "tesεζ"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "tesεζ");
|
|
|
|
// Test word selection at word boundary (space)
|
|
let word_range = rendered.surrounding_word_range(5); // Simulate click on space between "Hello" and "world", expect highlighting word to the left
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "Hello");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_surrounding_line_range(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("First line\n\nSecond line\n\nThird lineεζ", cx);
|
|
|
|
// Test getting line range for first line
|
|
let line_range = rendered.surrounding_line_range(5); // Simulate click somewhere in first line
|
|
let selected_text = rendered.text_for_range(line_range);
|
|
assert_eq!(selected_text, "First line");
|
|
|
|
// Test getting line range for second line
|
|
let line_range = rendered.surrounding_line_range(13); // Simulate click at beginning in second line
|
|
let selected_text = rendered.text_for_range(line_range);
|
|
assert_eq!(selected_text, "Second line");
|
|
|
|
// Test getting line range for third line
|
|
let line_range = rendered.surrounding_line_range(37); // Simulate click at end of third line with multi-byte chars
|
|
let selected_text = rendered.text_for_range(line_range);
|
|
assert_eq!(selected_text, "Third lineεζ");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_selection_head_movement(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("Hello world test", cx);
|
|
|
|
let mut selection = Selection {
|
|
start: 5,
|
|
end: 5,
|
|
reversed: false,
|
|
pending: false,
|
|
mode: SelectMode::Character,
|
|
};
|
|
|
|
// Test forward selection
|
|
selection.set_head(10, &rendered);
|
|
assert_eq!(selection.start, 5);
|
|
assert_eq!(selection.end, 10);
|
|
assert!(!selection.reversed);
|
|
assert_eq!(selection.tail(), 5);
|
|
|
|
// Test backward selection
|
|
selection.set_head(2, &rendered);
|
|
assert_eq!(selection.start, 2);
|
|
assert_eq!(selection.end, 5);
|
|
assert!(selection.reversed);
|
|
assert_eq!(selection.tail(), 5);
|
|
|
|
// Test forward selection again from reversed state
|
|
selection.set_head(15, &rendered);
|
|
assert_eq!(selection.start, 5);
|
|
assert_eq!(selection.end, 15);
|
|
assert!(!selection.reversed);
|
|
assert_eq!(selection.tail(), 5);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_word_selection_drag(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("Hello world test", cx);
|
|
|
|
// Start with a simulated double-click on "world" (index 6-10)
|
|
let word_range = rendered.surrounding_word_range(7); // Click on 'o' in "world"
|
|
let mut selection = Selection {
|
|
start: word_range.start,
|
|
end: word_range.end,
|
|
reversed: false,
|
|
pending: true,
|
|
mode: SelectMode::Word(word_range),
|
|
};
|
|
|
|
// Drag forward to "test" - should expand selection to include "test"
|
|
selection.set_head(13, &rendered); // Index in "test"
|
|
assert_eq!(selection.start, 6); // Start of "world"
|
|
assert_eq!(selection.end, 16); // End of "test"
|
|
assert!(!selection.reversed);
|
|
let selected_text = rendered.text_for_range(selection.start..selection.end);
|
|
assert_eq!(selected_text, "world test");
|
|
|
|
// Drag backward to "Hello" - should expand selection to include "Hello"
|
|
selection.set_head(2, &rendered); // Index in "Hello"
|
|
assert_eq!(selection.start, 0); // Start of "Hello"
|
|
assert_eq!(selection.end, 11); // End of "world" (original selection)
|
|
assert!(selection.reversed);
|
|
let selected_text = rendered.text_for_range(selection.start..selection.end);
|
|
assert_eq!(selected_text, "Hello world");
|
|
|
|
// Drag back within original word - should revert to original selection
|
|
selection.set_head(8, &rendered); // Back within "world"
|
|
assert_eq!(selection.start, 6); // Start of "world"
|
|
assert_eq!(selection.end, 11); // End of "world"
|
|
assert!(!selection.reversed);
|
|
let selected_text = rendered.text_for_range(selection.start..selection.end);
|
|
assert_eq!(selected_text, "world");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_selection_with_markdown_formatting(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown(
|
|
"This is **bold** text, this is *italic* text, use `code` here",
|
|
cx,
|
|
);
|
|
let word_range = rendered.surrounding_word_range(10); // Inside "bold"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "bold");
|
|
|
|
let word_range = rendered.surrounding_word_range(32); // Inside "italic"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "italic");
|
|
|
|
let word_range = rendered.surrounding_word_range(51); // Inside "code"
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "code");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_table_column_selection(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("| a | b |\n|---|---|\n| c | d |", cx);
|
|
|
|
assert!(rendered.lines.len() >= 2);
|
|
let first_bounds = rendered.lines[0].layout.bounds();
|
|
let second_bounds = rendered.lines[1].layout.bounds();
|
|
|
|
let first_index = match rendered.source_index_for_position(first_bounds.center()) {
|
|
Ok(index) | Err(index) => index,
|
|
};
|
|
let second_index = match rendered.source_index_for_position(second_bounds.center()) {
|
|
Ok(index) | Err(index) => index,
|
|
};
|
|
|
|
let first_word = rendered.text_for_range(rendered.surrounding_word_range(first_index));
|
|
let second_word = rendered.text_for_range(rendered.surrounding_word_range(second_index));
|
|
|
|
assert_eq!(first_word, "a");
|
|
assert_eq!(second_word, "b");
|
|
}
|
|
|
|
#[test]
|
|
fn test_table_state_current_cell_alignment_centers_headers() {
|
|
let mut table = TableState::default();
|
|
table.start(vec![Alignment::Left, Alignment::Right, Alignment::None]);
|
|
|
|
table.start_head();
|
|
for _ in 0..3 {
|
|
assert_eq!(table.current_cell_alignment(), Some(Alignment::Center));
|
|
table.end_cell();
|
|
}
|
|
|
|
table.end_head();
|
|
table.start_row();
|
|
assert_eq!(table.current_cell_alignment(), Some(Alignment::Left));
|
|
table.end_cell();
|
|
assert_eq!(table.current_cell_alignment(), Some(Alignment::Right));
|
|
table.end_cell();
|
|
assert_eq!(table.current_cell_alignment(), Some(Alignment::None));
|
|
table.end_cell();
|
|
table.end_row();
|
|
|
|
table.end();
|
|
assert_eq!(table.current_cell_alignment(), None);
|
|
}
|
|
|
|
#[test]
|
|
fn test_table_state_current_cell_alignment_outside_table() {
|
|
let table = TableState::default();
|
|
assert_eq!(table.current_cell_alignment(), None);
|
|
}
|
|
|
|
#[test]
|
|
fn test_table_checkbox_detection() {
|
|
let md = "| Done |\n|------|\n| [x] |\n| [ ] |";
|
|
let events = crate::parser::parse_markdown_with_options(md, false, false, false).events;
|
|
|
|
let mut in_table = false;
|
|
let mut cell_texts: Vec<String> = Vec::new();
|
|
let mut current_cell = String::new();
|
|
|
|
for (range, event) in &events {
|
|
match event {
|
|
MarkdownEvent::Start(MarkdownTag::Table(_)) => in_table = true,
|
|
MarkdownEvent::End(MarkdownTagEnd::Table) => in_table = false,
|
|
MarkdownEvent::Start(MarkdownTag::TableCell) => current_cell.clear(),
|
|
MarkdownEvent::End(MarkdownTagEnd::TableCell) => {
|
|
if in_table {
|
|
cell_texts.push(current_cell.clone());
|
|
}
|
|
}
|
|
MarkdownEvent::Text if in_table => {
|
|
current_cell.push_str(&md[range.clone()]);
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
let checkbox_cells: Vec<&String> = cell_texts
|
|
.iter()
|
|
.filter(|t| {
|
|
let trimmed = t.trim();
|
|
trimmed == "[x]" || trimmed == "[X]" || trimmed == "[ ]"
|
|
})
|
|
.collect();
|
|
assert_eq!(
|
|
checkbox_cells.len(),
|
|
2,
|
|
"Expected 2 checkbox cells, got: {cell_texts:?}"
|
|
);
|
|
assert_eq!(checkbox_cells[0].trim(), "[x]");
|
|
assert_eq!(checkbox_cells[1].trim(), "[ ]");
|
|
}
|
|
|
|
#[test]
|
|
fn test_table_checkbox_marker_source_range() {
|
|
let md = "| Done |\n|------|\n| [x] |\n| [ ] |";
|
|
let events = crate::parser::parse_markdown_with_options(md, false, false, false).events;
|
|
|
|
let mut in_cell = false;
|
|
let mut pending_text = String::new();
|
|
let mut mappings: Vec<SourceMapping> = Vec::new();
|
|
let mut cell_ranges: Vec<Range<usize>> = Vec::new();
|
|
|
|
for (range, event) in &events {
|
|
match event {
|
|
MarkdownEvent::Start(MarkdownTag::TableCell) => {
|
|
in_cell = true;
|
|
pending_text.clear();
|
|
mappings.clear();
|
|
}
|
|
MarkdownEvent::End(MarkdownTagEnd::TableCell) => {
|
|
if in_cell {
|
|
let trimmed = pending_text.trim();
|
|
if trimmed == "[x]" || trimmed == "[X]" || trimmed == "[ ]" {
|
|
let leading = pending_text.len() - pending_text.trim_start().len();
|
|
let rendered = leading..leading + trimmed.len();
|
|
let marker_source = source_range_for_rendered(&mappings, &rendered)
|
|
.expect("marker source range");
|
|
cell_ranges.push(marker_source);
|
|
}
|
|
}
|
|
in_cell = false;
|
|
}
|
|
MarkdownEvent::Text if in_cell => {
|
|
mappings.push(SourceMapping {
|
|
rendered_index: pending_text.len(),
|
|
source_index: range.start,
|
|
});
|
|
pending_text.push_str(&md[range.clone()]);
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
assert_eq!(cell_ranges.len(), 2);
|
|
for marker_range in &cell_ranges {
|
|
let slice = &md[marker_range.clone()];
|
|
assert!(
|
|
slice == "[x]" || slice == "[X]" || slice == "[ ]",
|
|
"expected `[x]`/`[X]`/`[ ]`, got {slice:?} at {marker_range:?}"
|
|
);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_source_range_for_rendered_handles_split_chunks() {
|
|
let mappings = vec![
|
|
SourceMapping {
|
|
rendered_index: 0,
|
|
source_index: 20,
|
|
},
|
|
SourceMapping {
|
|
rendered_index: 1,
|
|
source_index: 21,
|
|
},
|
|
SourceMapping {
|
|
rendered_index: 2,
|
|
source_index: 22,
|
|
},
|
|
];
|
|
|
|
let range = source_range_for_rendered(&mappings, &(0..3)).unwrap();
|
|
assert_eq!(range, 20..23);
|
|
|
|
let range = source_range_for_rendered(&mappings, &(1..2)).unwrap();
|
|
assert_eq!(range, 21..22);
|
|
|
|
assert_eq!(source_range_for_rendered(&mappings, &(2..2)), None);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_inline_code_word_selection_excludes_backticks(cx: &mut TestAppContext) {
|
|
// Test that double-clicking on inline code selects just the code content,
|
|
// not the backticks. This verifies the fix for the bug where selecting
|
|
// inline code would include the trailing backtick.
|
|
let rendered = render_markdown("use `blah` here", cx);
|
|
|
|
// Source layout: "use `blah` here"
|
|
// 0123456789...
|
|
// The inline code "blah" is at source positions 5-8 (content range 5..9)
|
|
|
|
// Click inside "blah" - should select just "blah", not "blah`"
|
|
let word_range = rendered.surrounding_word_range(6); // 'l' in "blah"
|
|
|
|
// text_for_range extracts from the rendered text (without backticks), so it
|
|
// would return "blah" even with a wrong source range. We check it anyway.
|
|
let selected_text = rendered.text_for_range(word_range.clone());
|
|
assert_eq!(selected_text, "blah");
|
|
|
|
// The source range is what matters for copy_as_markdown and selected_text,
|
|
// which extract directly from the source. With the bug, this would be 5..10
|
|
// which includes the closing backtick at position 9.
|
|
assert_eq!(word_range, 5..9);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_surrounding_word_range_respects_word_characters(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("foo.bar() baz", cx);
|
|
|
|
// Double clicking on 'f' in "foo" - should select just "foo"
|
|
let word_range = rendered.surrounding_word_range(0);
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "foo");
|
|
|
|
// Double clicking on 'b' in "bar" - should select just "bar"
|
|
let word_range = rendered.surrounding_word_range(4);
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "bar");
|
|
|
|
// Double clicking on 'b' in "baz" - should select "baz"
|
|
let word_range = rendered.surrounding_word_range(10);
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "baz");
|
|
|
|
// Double clicking selects word characters in code blocks
|
|
let javascript_language = Arc::new(Language::new(
|
|
LanguageConfig {
|
|
name: "JavaScript".into(),
|
|
matcher: LanguageMatcher {
|
|
path_suffixes: vec!["js".to_string()],
|
|
..Default::default()
|
|
},
|
|
word_characters: ['$', '#'].into_iter().collect(),
|
|
..Default::default()
|
|
},
|
|
None,
|
|
));
|
|
|
|
let language_registry = Arc::new(LanguageRegistry::test(cx.executor()));
|
|
language_registry.add(javascript_language);
|
|
|
|
let rendered = render_markdown_with_language_registry(
|
|
"```javascript\n$foo #bar\n```",
|
|
Some(language_registry),
|
|
cx,
|
|
);
|
|
|
|
let word_range = rendered.surrounding_word_range(14);
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "$foo");
|
|
|
|
let word_range = rendered.surrounding_word_range(19);
|
|
let selected_text = rendered.text_for_range(word_range);
|
|
assert_eq!(selected_text, "#bar");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_all_selection(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("Hello world\n\nThis is a test\n\nwith multiple lines", cx);
|
|
|
|
let total_length = rendered
|
|
.lines
|
|
.last()
|
|
.map(|line| line.source_end)
|
|
.unwrap_or(0);
|
|
|
|
let mut selection = Selection {
|
|
start: 0,
|
|
end: total_length,
|
|
reversed: false,
|
|
pending: true,
|
|
mode: SelectMode::All,
|
|
};
|
|
|
|
selection.set_head(5, &rendered); // Try to set head in middle
|
|
assert_eq!(selection.start, 0);
|
|
assert_eq!(selection.end, total_length);
|
|
assert!(!selection.reversed);
|
|
|
|
selection.set_head(25, &rendered); // Try to set head near end
|
|
assert_eq!(selection.start, 0);
|
|
assert_eq!(selection.end, total_length);
|
|
assert!(!selection.reversed);
|
|
|
|
let selected_text = rendered.text_for_range(selection.start..selection.end);
|
|
assert_eq!(
|
|
selected_text,
|
|
"Hello world\nThis is a test\nwith multiple lines"
|
|
);
|
|
}
|
|
|
|
fn nbsp(n: usize) -> String {
|
|
"\u{00A0}".repeat(n)
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_plain_text() {
|
|
assert_eq!(Markdown::escape("hello world"), "hello world");
|
|
assert_eq!(Markdown::escape(""), "");
|
|
assert_eq!(Markdown::escape("café ☕ naïve"), "café ☕ naïve");
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_punctuation() {
|
|
assert_eq!(Markdown::escape("hello `world`"), r"hello \`world\`");
|
|
assert_eq!(Markdown::escape("a|b"), r"a\|b");
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_leading_spaces() {
|
|
assert_eq!(Markdown::escape(" hello"), [ (4), "hello"].concat());
|
|
assert_eq!(
|
|
Markdown::escape(" | { a: string }"),
|
|
[ (4), r"\| \{ a\: string \}"].concat()
|
|
);
|
|
assert_eq!(
|
|
Markdown::escape(" first\n second"),
|
|
[ (2), "first\n\n",  (2), "second"].concat()
|
|
);
|
|
assert_eq!(Markdown::escape("hello world"), "hello world");
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_leading_tabs() {
|
|
assert_eq!(Markdown::escape("\thello"), [ (4), "hello"].concat());
|
|
assert_eq!(
|
|
Markdown::escape("hello\n\t\tindented"),
|
|
["hello\n\n",  (8), "indented"].concat()
|
|
);
|
|
assert_eq!(
|
|
Markdown::escape(" \t hello"),
|
|
[ (1 + 4 + 1), "hello"].concat()
|
|
);
|
|
assert_eq!(Markdown::escape("hello\tworld"), "hello\tworld");
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_newlines() {
|
|
assert_eq!(Markdown::escape("a\nb"), "a\n\nb");
|
|
assert_eq!(Markdown::escape("a\n\nb"), "a\n\n\n\nb");
|
|
assert_eq!(Markdown::escape("\nhello"), "\n\nhello");
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_multiline_diagnostic() {
|
|
assert_eq!(
|
|
Markdown::escape(" | { a: string }\n | { b: number }"),
|
|
[
|
|
 (4),
|
|
r"\| \{ a\: string \}",
|
|
"\n\n",
|
|
 (4),
|
|
r"\| \{ b\: number \}",
|
|
]
|
|
.concat()
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_non_ascii() {
|
|
// Cyrillic characters should not have backslashes added before them,
|
|
// but ASCII punctuation should still be escaped.
|
|
assert_eq!(Markdown::escape("Привет, мир"), r"Привет\, мир");
|
|
// Test with markdown special characters mixed in
|
|
assert_eq!(Markdown::escape("Привет, *мир*"), r"Привет\, \*мир\*");
|
|
// Test with the exact example from the issue (single quotes are also ASCII punctuation)
|
|
assert_eq!(
|
|
Markdown::escape("Отсутствует пробел справа от ','"),
|
|
r"Отсутствует пробел справа от \'\,\'"
|
|
);
|
|
// Test more non-ASCII scripts
|
|
assert_eq!(
|
|
Markdown::escape("こんにちは *world*"),
|
|
r"こんにちは \*world\*"
|
|
);
|
|
assert_eq!(Markdown::escape("العربيّة [link]"), r"العربيّة \[link\]");
|
|
assert_eq!(Markdown::escape("Ελληνικά _text_"), r"Ελληνικά \_text\_");
|
|
assert_eq!(Markdown::escape("עברית `code`"), r"עברית \`code\`");
|
|
// Non-ASCII followed by ASCII punctuation
|
|
assert_eq!(Markdown::escape("Test: тест"), r"Test\: тест");
|
|
}
|
|
|
|
fn has_code_block(markdown: &str) -> bool {
|
|
let parsed_data = parse_markdown_with_options(markdown, false, false, false);
|
|
parsed_data
|
|
.events
|
|
.iter()
|
|
.any(|(_, event)| matches!(event, MarkdownEvent::Start(MarkdownTag::CodeBlock { .. })))
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_output_len_matches_precomputed() {
|
|
let cases = [
|
|
"",
|
|
"hello world",
|
|
"hello `world`",
|
|
" hello",
|
|
" | { a: string }",
|
|
"\thello",
|
|
"hello\n\t\tindented",
|
|
" \t hello",
|
|
"hello\tworld",
|
|
"a\nb",
|
|
"a\n\nb",
|
|
"\nhello",
|
|
" | { a: string }\n | { b: number }",
|
|
"café ☕ naïve",
|
|
];
|
|
for input in cases {
|
|
let mut escaper = MarkdownEscaper::new();
|
|
let precomputed: usize = input.chars().map(|c| escaper.next(c).output_len(c)).sum();
|
|
|
|
let mut escaper = MarkdownEscaper::new();
|
|
let mut output = String::new();
|
|
for c in input.chars() {
|
|
escaper.next(c).write_to(c, &mut output);
|
|
}
|
|
|
|
assert_eq!(precomputed, output.len(), "length mismatch for {:?}", input);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_escape_prevents_code_block() {
|
|
let diagnostic = " | { a: string }";
|
|
assert!(has_code_block(diagnostic));
|
|
assert!(!has_code_block(&Markdown::escape(diagnostic)));
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_link_detected_for_source_index(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("[Click here](https://example.com)", cx);
|
|
|
|
assert_eq!(rendered.links.len(), 1);
|
|
assert_eq!(rendered.links[0].destination_url, "https://example.com");
|
|
|
|
// Source index 1 ('C' in "Click") is inside the link's source range
|
|
let link = rendered.link_for_source_index(1);
|
|
assert!(link.is_some());
|
|
assert_eq!(link.unwrap().destination_url, "https://example.com");
|
|
|
|
// A source index past the end of the link range returns None
|
|
let past_end = rendered.links[0].source_range.end;
|
|
assert!(rendered.link_for_source_index(past_end).is_none());
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_link_for_source_index_ignores_plain_text(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("Hello world", cx);
|
|
|
|
assert!(rendered.links.is_empty());
|
|
assert!(rendered.link_for_source_index(0).is_none());
|
|
assert!(rendered.link_for_source_index(5).is_none());
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_code_span_link_detected_for_source_index(cx: &mut TestAppContext) {
|
|
let source = "see `foo.rs` for details";
|
|
let rendered = render_markdown_with_code_span_link(
|
|
source,
|
|
|text, _cx| (text == "foo.rs").then(|| "file:///tmp/foo.rs".into()),
|
|
cx,
|
|
);
|
|
|
|
assert_eq!(rendered.links.len(), 1);
|
|
assert_eq!(rendered.links[0].destination_url, "file:///tmp/foo.rs");
|
|
|
|
let code_index = source.find("foo.rs").unwrap();
|
|
let link = rendered.link_for_source_index(code_index);
|
|
assert!(link.is_some());
|
|
assert_eq!(link.unwrap().destination_url, "file:///tmp/foo.rs");
|
|
|
|
assert!(
|
|
rendered
|
|
.link_for_source_index(source.find("see").unwrap())
|
|
.is_none()
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_code_span_link_ignores_code_when_mouse_interaction_is_prevented(
|
|
cx: &mut TestAppContext,
|
|
) {
|
|
let callback_count = Arc::new(AtomicUsize::new(0));
|
|
let rendered = render_markdown_with_code_span_link_style(
|
|
"see `foo.rs` for details",
|
|
MarkdownStyle {
|
|
prevent_mouse_interaction: true,
|
|
..MarkdownStyle::default()
|
|
},
|
|
{
|
|
let callback_count = callback_count.clone();
|
|
move |text, _cx| {
|
|
callback_count.fetch_add(1, Ordering::Relaxed);
|
|
(text == "foo.rs").then(|| "file:///tmp/foo.rs".into())
|
|
}
|
|
},
|
|
cx,
|
|
);
|
|
|
|
assert!(rendered.links.is_empty());
|
|
assert_eq!(callback_count.load(Ordering::Relaxed), 0);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_code_span_link_ignores_code_without_callback(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("see `foo.rs` for details", cx);
|
|
|
|
assert!(rendered.links.is_empty());
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_code_span_link_ignores_code_inside_markdown_link(cx: &mut TestAppContext) {
|
|
let source = "see [`foo.rs`](https://example.com) for details";
|
|
let rendered = render_markdown_with_code_span_link(
|
|
source,
|
|
|text, _cx| (text == "foo.rs").then(|| "file:///tmp/foo.rs".into()),
|
|
cx,
|
|
);
|
|
|
|
assert_eq!(rendered.links.len(), 1);
|
|
assert_eq!(rendered.links[0].destination_url, "https://example.com");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_context_menu_link_initial_state(cx: &mut TestAppContext) {
|
|
struct TestWindow;
|
|
impl Render for TestWindow {
|
|
fn render(&mut self, _: &mut Window, _: &mut Context<Self>) -> impl IntoElement {
|
|
div()
|
|
}
|
|
}
|
|
|
|
ensure_theme_initialized(cx);
|
|
let (_, cx) = cx.add_window_view(|_, _| TestWindow);
|
|
let markdown =
|
|
cx.new(|cx| Markdown::new("Hello [world](https://example.com)".into(), None, None, cx));
|
|
cx.run_until_parked();
|
|
|
|
cx.update(|_window, cx| {
|
|
assert!(markdown.read(cx).context_menu_link().is_none());
|
|
});
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_capture_for_context_menu(cx: &mut TestAppContext) {
|
|
struct TestWindow;
|
|
impl Render for TestWindow {
|
|
fn render(&mut self, _: &mut Window, _: &mut Context<Self>) -> impl IntoElement {
|
|
div()
|
|
}
|
|
}
|
|
|
|
ensure_theme_initialized(cx);
|
|
let (_, cx) = cx.add_window_view(|_, _| TestWindow);
|
|
let markdown = cx.new(|cx| Markdown::new("text".into(), None, None, cx));
|
|
cx.run_until_parked();
|
|
|
|
// Simulates right-clicking on a link
|
|
let url: SharedString = "https://example.com".into();
|
|
markdown.update(cx, |md, _cx| {
|
|
md.capture_for_context_menu(Some(url.clone()));
|
|
});
|
|
cx.update(|_window, cx| {
|
|
assert_eq!(
|
|
markdown
|
|
.read(cx)
|
|
.context_menu_link()
|
|
.map(SharedString::as_ref),
|
|
Some("https://example.com")
|
|
);
|
|
});
|
|
|
|
// Simulates right-clicking on plain text — link is cleared
|
|
markdown.update(cx, |md, _cx| {
|
|
md.capture_for_context_menu(None);
|
|
});
|
|
cx.update(|_window, cx| {
|
|
assert!(markdown.read(cx).context_menu_link().is_none());
|
|
});
|
|
}
|
|
|
|
#[track_caller]
|
|
fn assert_mappings(rendered: &RenderedText, expected: Vec<Vec<(usize, usize)>>) {
|
|
assert_eq!(rendered.lines.len(), expected.len(), "line count mismatch");
|
|
for (line_ix, line_mappings) in expected.into_iter().enumerate() {
|
|
let line = &rendered.lines[line_ix];
|
|
|
|
assert!(
|
|
line.source_mappings.windows(2).all(|mappings| {
|
|
mappings[0].source_index < mappings[1].source_index
|
|
&& mappings[0].rendered_index < mappings[1].rendered_index
|
|
}),
|
|
"line {} has duplicate mappings: {:?}",
|
|
line_ix,
|
|
line.source_mappings
|
|
);
|
|
|
|
for (rendered_ix, source_ix) in line_mappings {
|
|
assert_eq!(
|
|
line.source_index_for_rendered_index(rendered_ix),
|
|
source_ix,
|
|
"line {}, rendered_ix {}",
|
|
line_ix,
|
|
rendered_ix
|
|
);
|
|
|
|
assert_eq!(
|
|
line.rendered_index_for_source_index(source_ix),
|
|
rendered_ix,
|
|
"line {}, source_ix {}",
|
|
line_ix,
|
|
source_ix
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_bounds_for_source_range_skips_gaps_between_rendered_lines(cx: &mut TestAppContext) {
|
|
let source = "First\n\nSecond";
|
|
let rendered = render_markdown(source, cx);
|
|
let highlight_bounds = rendered.bounds_for_source_range(0..source.len());
|
|
assert_eq!(highlight_bounds.len(), rendered.lines.len());
|
|
|
|
for (line, highlight_bounds) in rendered.lines.iter().zip(highlight_bounds.iter()) {
|
|
let line_bounds = line.layout.bounds();
|
|
assert_eq!(highlight_bounds.top(), line_bounds.top());
|
|
assert_eq!(
|
|
highlight_bounds.bottom(),
|
|
line_bounds.top() + line.layout.line_height()
|
|
);
|
|
}
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_bounds_for_source_range_returns_one_bound_per_soft_wrap_row(cx: &mut TestAppContext) {
|
|
let sentence = "Lorem ipsum dolor sit amet, consectetur adipiscing elit, \
|
|
sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.";
|
|
let source = [sentence, sentence, sentence, sentence].join(" ");
|
|
let rendered = render_markdown(&source, cx);
|
|
let line = &rendered.lines[0];
|
|
let line_bounds = line.layout.bounds();
|
|
let line_height = line.layout.line_height();
|
|
let wrapped_line = line.layout.line_layout_for_index(0).unwrap();
|
|
let visual_row_count = wrapped_line.wrap_boundaries().len() + 1;
|
|
|
|
let highlight_bounds = rendered.bounds_for_source_range(0..source.len());
|
|
assert_eq!(highlight_bounds.len(), visual_row_count);
|
|
|
|
let mut row_top = line_bounds.top();
|
|
for (row_index, row_bounds) in highlight_bounds.iter().enumerate() {
|
|
assert_eq!(row_bounds.top(), row_top);
|
|
assert_eq!(row_bounds.bottom(), row_top + line_height);
|
|
assert!(
|
|
row_bounds.size.width > Pixels::ZERO,
|
|
"row {row_index} should have a non-empty highlight"
|
|
);
|
|
row_top += line_height;
|
|
}
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_heading_font_sizes_are_distinct(cx: &mut TestAppContext) {
|
|
let rendered = render_markdown("# H1\n\n## H2\n\n### H3\n\nBody text", cx);
|
|
|
|
assert!(
|
|
rendered.lines.len() >= 4,
|
|
"expected at least 4 rendered lines, got {}",
|
|
rendered.lines.len()
|
|
);
|
|
|
|
let h1_line_height = rendered.lines[0].layout.line_height();
|
|
let h2_line_height = rendered.lines[1].layout.line_height();
|
|
let h3_line_height = rendered.lines[2].layout.line_height();
|
|
let body_line_height = rendered.lines[3].layout.line_height();
|
|
|
|
assert!(
|
|
h1_line_height > h2_line_height,
|
|
"H1 line height ({h1_line_height:?}) should be greater than H2 ({h2_line_height:?})"
|
|
);
|
|
assert!(
|
|
h2_line_height > h3_line_height,
|
|
"H2 line height ({h2_line_height:?}) should be greater than H3 ({h3_line_height:?})"
|
|
);
|
|
assert!(
|
|
h3_line_height > body_line_height,
|
|
"H3 line height ({h3_line_height:?}) should be greater than body text ({body_line_height:?})"
|
|
);
|
|
}
|
|
}
|