summaryrefslogtreecommitdiffstats
path: root/examples
diff options
context:
space:
mode:
authorLibravatar Héctor Ramón Jiménez <hector@hecrj.dev>2025-02-04 07:53:56 +0100
committerLibravatar Héctor Ramón Jiménez <hector@hecrj.dev>2025-02-04 07:53:56 +0100
commit565599876172b3f56d86b119ae453b5bcd8949e1 (patch)
tree5353cb71aa595fc5ef3445e104b4776b8cc61ccb /examples
parentc02ae0c4a430994247e6fbc4318ac344ab89123c (diff)
downloadiced-565599876172b3f56d86b119ae453b5bcd8949e1.tar.gz
iced-565599876172b3f56d86b119ae453b5bcd8949e1.tar.bz2
iced-565599876172b3f56d86b119ae453b5bcd8949e1.zip
Draft `Viewer` trait for `markdown`
Diffstat (limited to 'examples')
-rw-r--r--examples/changelog/src/main.rs26
-rw-r--r--examples/markdown/Cargo.toml8
-rw-r--r--examples/markdown/src/main.rs139
3 files changed, 151 insertions, 22 deletions
diff --git a/examples/changelog/src/main.rs b/examples/changelog/src/main.rs
index f889e757..a6528ce9 100644
--- a/examples/changelog/src/main.rs
+++ b/examples/changelog/src/main.rs
@@ -267,25 +267,21 @@ impl Generator {
} => {
let details = {
let title = rich_text![
- span(&pull_request.title).size(24).link(
- Message::OpenPullRequest(pull_request.id)
- ),
+ span(&pull_request.title)
+ .size(24)
+ .link(pull_request.id),
span(format!(" by {}", pull_request.author))
.font(Font {
style: font::Style::Italic,
..Font::default()
}),
]
+ .on_link_clicked(Message::OpenPullRequest)
.font(Font::MONOSPACE);
- let description = markdown::view(
- description,
- markdown::Settings::default(),
- markdown::Style::from_palette(
- self.theme().palette(),
- ),
- )
- .map(Message::UrlClicked);
+ let description =
+ markdown::view(&self.theme(), description)
+ .map(Message::UrlClicked);
let labels =
row(pull_request.labels.iter().map(|label| {
@@ -349,11 +345,11 @@ impl Generator {
container(
scrollable(
markdown::view(
- preview,
- markdown::Settings::with_text_size(12),
- markdown::Style::from_palette(
- self.theme().palette(),
+ markdown::Settings::with_text_size(
+ 12,
+ &self.theme(),
),
+ preview,
)
.map(Message::UrlClicked),
)
diff --git a/examples/markdown/Cargo.toml b/examples/markdown/Cargo.toml
index fa6ced74..4711b1c4 100644
--- a/examples/markdown/Cargo.toml
+++ b/examples/markdown/Cargo.toml
@@ -7,6 +7,12 @@ publish = false
[dependencies]
iced.workspace = true
-iced.features = ["markdown", "highlighter", "tokio", "debug"]
+iced.features = ["markdown", "highlighter", "image", "tokio", "debug"]
+
+reqwest.version = "0.12"
+reqwest.features = ["json"]
+
+image.workspace = true
+tokio.workspace = true
open = "5.3"
diff --git a/examples/markdown/src/main.rs b/examples/markdown/src/main.rs
index ba93ee18..29625d79 100644
--- a/examples/markdown/src/main.rs
+++ b/examples/markdown/src/main.rs
@@ -1,10 +1,17 @@
use iced::highlighter;
use iced::time::{self, milliseconds};
use iced::widget::{
- self, hover, markdown, right, row, scrollable, text_editor, toggler,
+ self, center_x, horizontal_space, hover, image, markdown, pop, right, row,
+ scrollable, text_editor, toggler,
};
use iced::{Element, Fill, Font, Subscription, Task, Theme};
+use tokio::task;
+
+use std::collections::HashMap;
+use std::io;
+use std::sync::Arc;
+
pub fn main() -> iced::Result {
iced::application("Markdown - Iced", Markdown::update, Markdown::view)
.subscription(Markdown::subscription)
@@ -14,6 +21,7 @@ pub fn main() -> iced::Result {
struct Markdown {
content: text_editor::Content,
+ images: HashMap<markdown::Url, Image>,
mode: Mode,
theme: Theme,
}
@@ -26,10 +34,19 @@ enum Mode {
},
}
+enum Image {
+ Loading,
+ Ready(image::Handle),
+ #[allow(dead_code)]
+ Errored(Error),
+}
+
#[derive(Debug, Clone)]
enum Message {
Edit(text_editor::Action),
LinkClicked(markdown::Url),
+ ImageShown(markdown::Url),
+ ImageDownloaded(markdown::Url, Result<image::Handle, Error>),
ToggleStream(bool),
NextToken,
}
@@ -43,6 +60,7 @@ impl Markdown {
(
Self {
content: text_editor::Content::with_text(INITIAL_CONTENT),
+ images: HashMap::new(),
mode: Mode::Preview(markdown::parse(INITIAL_CONTENT).collect()),
theme,
},
@@ -70,6 +88,25 @@ impl Markdown {
Task::none()
}
+ Message::ImageShown(url) => {
+ if self.images.contains_key(&url) {
+ return Task::none();
+ }
+
+ let _ = self.images.insert(url.clone(), Image::Loading);
+
+ Task::perform(download_image(url.clone()), move |result| {
+ Message::ImageDownloaded(url.clone(), result)
+ })
+ }
+ Message::ImageDownloaded(url, result) => {
+ let _ = self.images.insert(
+ url,
+ result.map(Image::Ready).unwrap_or_else(Image::Errored),
+ );
+
+ Task::none()
+ }
Message::ToggleStream(enable_stream) => {
if enable_stream {
self.mode = Mode::Stream {
@@ -126,12 +163,13 @@ impl Markdown {
Mode::Stream { parsed, .. } => parsed.items(),
};
- let preview = markdown(
+ let preview = markdown::view_with(
+ &MarkdownViewer {
+ images: &self.images,
+ },
+ &self.theme,
items,
- markdown::Settings::default(),
- markdown::Style::from_palette(self.theme.palette()),
- )
- .map(Message::LinkClicked);
+ );
row![
editor,
@@ -167,3 +205,92 @@ impl Markdown {
}
}
}
+
+struct MarkdownViewer<'a> {
+ images: &'a HashMap<markdown::Url, Image>,
+}
+
+impl<'a> markdown::Viewer<'a, Message> for MarkdownViewer<'a> {
+ fn on_link_clicked(url: markdown::Url) -> Message {
+ Message::LinkClicked(url)
+ }
+
+ fn image(
+ &self,
+ _settings: markdown::Settings,
+ _title: &markdown::Text,
+ url: &'a markdown::Url,
+ ) -> Element<'a, Message> {
+ if let Some(Image::Ready(handle)) = self.images.get(url) {
+ center_x(image(handle)).into()
+ } else {
+ pop(horizontal_space().width(0))
+ .key(url.as_str())
+ .on_show(|_size| Message::ImageShown(url.clone()))
+ .into()
+ }
+ }
+}
+
+async fn download_image(url: markdown::Url) -> Result<image::Handle, Error> {
+ use std::io;
+ use tokio::task;
+
+ let client = reqwest::Client::new();
+
+ let bytes = client
+ .get(url)
+ .send()
+ .await?
+ .error_for_status()?
+ .bytes()
+ .await?;
+
+ let image = task::spawn_blocking(move || {
+ Ok::<_, Error>(
+ ::image::ImageReader::new(io::Cursor::new(bytes))
+ .with_guessed_format()?
+ .decode()?
+ .to_rgba8(),
+ )
+ })
+ .await??;
+
+ Ok(image::Handle::from_rgba(
+ image.width(),
+ image.height(),
+ image.into_raw(),
+ ))
+}
+
+#[derive(Debug, Clone)]
+pub enum Error {
+ RequestFailed(Arc<reqwest::Error>),
+ IOFailed(Arc<io::Error>),
+ JoinFailed(Arc<task::JoinError>),
+ ImageDecodingFailed(Arc<::image::ImageError>),
+}
+
+impl From<reqwest::Error> for Error {
+ fn from(error: reqwest::Error) -> Self {
+ Self::RequestFailed(Arc::new(error))
+ }
+}
+
+impl From<io::Error> for Error {
+ fn from(error: io::Error) -> Self {
+ Self::IOFailed(Arc::new(error))
+ }
+}
+
+impl From<task::JoinError> for Error {
+ fn from(error: task::JoinError) -> Self {
+ Self::JoinFailed(Arc::new(error))
+ }
+}
+
+impl From<::image::ImageError> for Error {
+ fn from(error: ::image::ImageError) -> Self {
+ Self::ImageDecodingFailed(Arc::new(error))
+ }
+}