diff options
author | 2025-02-04 20:58:06 +0100 | |
---|---|---|
committer | 2025-02-04 20:58:06 +0100 | |
commit | e8020f3eaf3baec2b41847f6250d8554136e8d89 (patch) | |
tree | bc077692333b71f48a7efe984db60f87f9555b4c | |
parent | 387abafa3abda3ba68eb7a2e4ce4240ad3bdda53 (diff) | |
download | iced-e8020f3eaf3baec2b41847f6250d8554136e8d89.tar.gz iced-e8020f3eaf3baec2b41847f6250d8554136e8d89.tar.bz2 iced-e8020f3eaf3baec2b41847f6250d8554136e8d89.zip |
Add `Copy` action to code blocks in `markdown` example
-rw-r--r-- | Cargo.lock | 8 | ||||
-rw-r--r-- | core/src/pixels.rs | 8 | ||||
-rw-r--r-- | examples/markdown/Cargo.toml | 5 | ||||
-rw-r--r-- | examples/markdown/build.rs | 5 | ||||
-rw-r--r-- | examples/markdown/fonts/markdown-icons.toml | 4 | ||||
-rw-r--r-- | examples/markdown/fonts/markdown-icons.ttf | bin | 0 -> 5856 bytes | |||
-rw-r--r-- | examples/markdown/src/icon.rs | 15 | ||||
-rw-r--r-- | examples/markdown/src/main.rs | 32 | ||||
-rw-r--r-- | widget/src/markdown.rs | 28 |
9 files changed, 93 insertions, 12 deletions
@@ -770,9 +770,9 @@ checksum = "37b2a672a2cb129a2e41c10b1224bb368f9f37a2b16b612598138befd7b37eb5" [[package]] name = "cc" -version = "1.2.11" +version = "1.2.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e4730490333d58093109dc02c23174c3f4d490998c3fed3cc8e82d57afedb9cf" +checksum = "755717a7de9ec452bf7f3f1a3099085deabd7f2962b861dae91ecd7a365903d2" dependencies = [ "jobserver", "libc", @@ -877,9 +877,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.27" +version = "4.5.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "769b0145982b4b48713e01ec42d61614425f27b7058bda7180a3a41f30104796" +checksum = "3e77c3243bd94243c03672cb5154667347c457ca271254724f9f393aee1c05ff" dependencies = [ "clap_builder", ] diff --git a/core/src/pixels.rs b/core/src/pixels.rs index a1ea0f15..7d6267cf 100644 --- a/core/src/pixels.rs +++ b/core/src/pixels.rs @@ -79,3 +79,11 @@ impl std::ops::Div<f32> for Pixels { Pixels(self.0 / rhs) } } + +impl std::ops::Div<u32> for Pixels { + type Output = Pixels; + + fn div(self, rhs: u32) -> Self { + Pixels(self.0 / rhs as f32) + } +} diff --git a/examples/markdown/Cargo.toml b/examples/markdown/Cargo.toml index 4711b1c4..7af1741b 100644 --- a/examples/markdown/Cargo.toml +++ b/examples/markdown/Cargo.toml @@ -16,3 +16,8 @@ image.workspace = true tokio.workspace = true open = "5.3" + +# Disabled to keep amount of build dependencies low +# This can be re-enabled on demand +# [build-dependencies] +# iced_fontello = "0.13" diff --git a/examples/markdown/build.rs b/examples/markdown/build.rs new file mode 100644 index 00000000..ecbb7666 --- /dev/null +++ b/examples/markdown/build.rs @@ -0,0 +1,5 @@ +pub fn main() { + // println!("cargo::rerun-if-changed=fonts/markdown-icons.toml"); + // iced_fontello::build("fonts/markdown-icons.toml") + // .expect("Build icons font"); +} diff --git a/examples/markdown/fonts/markdown-icons.toml b/examples/markdown/fonts/markdown-icons.toml new file mode 100644 index 00000000..60c91d17 --- /dev/null +++ b/examples/markdown/fonts/markdown-icons.toml @@ -0,0 +1,4 @@ +module = "icon" + +[glyphs] +copy = "fontawesome-docs" diff --git a/examples/markdown/fonts/markdown-icons.ttf b/examples/markdown/fonts/markdown-icons.ttf Binary files differnew file mode 100644 index 00000000..013f03a5 --- /dev/null +++ b/examples/markdown/fonts/markdown-icons.ttf diff --git a/examples/markdown/src/icon.rs b/examples/markdown/src/icon.rs new file mode 100644 index 00000000..cfe32541 --- /dev/null +++ b/examples/markdown/src/icon.rs @@ -0,0 +1,15 @@ +// Generated automatically by iced_fontello at build time. +// Do not edit manually. Source: ../fonts/markdown-icons.toml +// dcd2f0c969d603e2ee9237a4b70fa86b1a6e84d86f4689046d8fdd10440b06b9 +use iced::widget::{text, Text}; +use iced::Font; + +pub const FONT: &[u8] = include_bytes!("../fonts/markdown-icons.ttf"); + +pub fn copy<'a>() -> Text<'a> { + icon("\u{F0C5}") +} + +fn icon(codepoint: &str) -> Text<'_> { + text(codepoint).font(Font::with_name("markdown-icons")) +} diff --git a/examples/markdown/src/main.rs b/examples/markdown/src/main.rs index 84c20b7e..6a881288 100644 --- a/examples/markdown/src/main.rs +++ b/examples/markdown/src/main.rs @@ -1,10 +1,13 @@ +mod icon; + use iced::animation; +use iced::clipboard; use iced::highlighter; use iced::task; use iced::time::{self, milliseconds, Instant}; use iced::widget::{ - self, center_x, horizontal_space, hover, image, markdown, pop, right, row, - scrollable, text_editor, toggler, + self, button, center_x, horizontal_space, hover, image, markdown, pop, + right, row, scrollable, text_editor, toggler, }; use iced::window; use iced::{Animation, Element, Fill, Font, Subscription, Task, Theme}; @@ -15,6 +18,7 @@ use std::sync::Arc; pub fn main() -> iced::Result { iced::application("Markdown - Iced", Markdown::update, Markdown::view) + .font(icon::FONT) .subscription(Markdown::subscription) .theme(Markdown::theme) .run_with(Markdown::new) @@ -49,6 +53,7 @@ enum Image { #[derive(Debug, Clone)] enum Message { Edit(text_editor::Action), + Copy(String), LinkClicked(markdown::Url), ImageShown(markdown::Url), ImageDownloaded(markdown::Url, Result<image::Handle, Error>), @@ -91,6 +96,7 @@ impl Markdown { Task::none() } + Message::Copy(content) => clipboard::write(content), Message::LinkClicked(link) => { let _ = open::that_in_background(link.to_string()); @@ -141,6 +147,8 @@ impl Markdown { } Message::ToggleStream(enable_stream) => { if enable_stream { + self.content = markdown::Content::new(); + self.mode = Mode::Stream { pending: self.raw.text(), }; @@ -282,6 +290,26 @@ impl<'a> markdown::Viewer<'a, Message> for CustomViewer<'a> { .into() } } + + fn code_block( + &self, + settings: markdown::Settings, + code: &'a str, + lines: &'a [markdown::Text], + ) -> Element<'a, Message> { + let code_block = + markdown::code_block(settings, code, lines, Message::LinkClicked); + + hover( + code_block, + right( + button(icon::copy().size(12)) + .padding(settings.spacing / 2) + .on_press_with(|| Message::Copy(code.to_owned())) + .style(button::text), + ), + ) + } } async fn download_image(url: markdown::Url) -> Result<image::Handle, Error> { diff --git a/widget/src/markdown.rs b/widget/src/markdown.rs index 9ce5930f..3af301e9 100644 --- a/widget/src/markdown.rs +++ b/widget/src/markdown.rs @@ -55,6 +55,7 @@ use crate::{column, container, rich_text, row, scrollable, span, text}; use std::borrow::BorrowMut; use std::cell::{Cell, RefCell}; use std::collections::{HashMap, HashSet}; +use std::mem; use std::ops::Range; use std::rc::Rc; use std::sync::Arc; @@ -182,7 +183,12 @@ pub enum Item { /// A code block. /// /// You can enable the `highlighter` feature for syntax highlighting. - CodeBlock(Vec<Text>), + CodeBlock { + /// The raw code of the code block. + code: String, + /// The styled lines of text in the code block. + lines: Vec<Text>, + }, /// A list. List { /// The first number of the list, if it is ordered. @@ -457,7 +463,8 @@ fn parse_with<'a>( let broken_links = Rc::new(RefCell::new(HashSet::new())); let mut spans = Vec::new(); - let mut code = Vec::new(); + let mut code = String::new(); + let mut code_lines = Vec::new(); let mut strong = false; let mut emphasis = false; let mut strikethrough = false; @@ -726,7 +733,10 @@ fn parse_with<'a>( produce( state.borrow_mut(), &mut stack, - Item::CodeBlock(code.drain(..).collect()), + Item::CodeBlock { + code: mem::take(&mut code), + lines: code_lines.drain(..).collect(), + }, source, ) } @@ -743,8 +753,10 @@ fn parse_with<'a>( pulldown_cmark::Event::Text(text) if !metadata && !table => { #[cfg(feature = "highlighter")] if let Some(highlighter) = &mut highlighter { + code.push_str(&text); + for line in text.lines() { - code.push(Text::new( + code_lines.push(Text::new( highlighter.highlight_line(line).to_vec(), )); } @@ -1017,7 +1029,9 @@ where viewer.heading(settings, level, text, index) } Item::Paragraph(text) => viewer.paragraph(settings, text), - Item::CodeBlock(lines) => viewer.code_block(settings, lines), + Item::CodeBlock { code, lines } => { + viewer.code_block(settings, code, lines) + } Item::List { start: None, items } => { viewer.unordered_list(settings, items) } @@ -1157,6 +1171,7 @@ where /// Displays a code block using the default look. pub fn code_block<'a, Message, Theme, Renderer>( settings: Settings, + _code: &'a str, lines: &'a [Text], on_link_clicked: impl Fn(Url) -> Message + Clone + 'a, ) -> Element<'a, Message, Theme, Renderer> @@ -1251,9 +1266,10 @@ where fn code_block( &self, settings: Settings, + code: &'a str, lines: &'a [Text], ) -> Element<'a, Message, Theme, Renderer> { - code_block(settings, lines, Self::on_link_clicked) + code_block(settings, code, lines, Self::on_link_clicked) } /// Displays an unordered list. |