egui/egui_demo_lib/src/apps/http_app.rs

320 lines
9.7 KiB
Rust
Raw Normal View History

use epi::http::Response;
use std::sync::mpsc::Receiver;
struct Resource {
/// HTTP response
response: Response,
/// If set, the response was an image.
image: Option<Image>,
/// If set, the response was text with some supported syntax highlighting (e.g. ".rs" or ".md").
colored_text: Option<ColoredText>,
}
2020-11-20 18:50:47 +00:00
impl Resource {
fn from_response(response: Response) -> Self {
let image = if response.header_content_type.starts_with("image/") {
Image::decode(&response.bytes)
} else {
None
};
let colored_text = syntax_highlighting(&response);
Self {
response,
image,
colored_text,
}
2020-11-20 18:50:47 +00:00
}
}
#[cfg_attr(feature = "persistence", derive(serde::Deserialize, serde::Serialize))]
2021-01-01 16:11:05 +00:00
pub struct HttpApp {
url: String,
2021-01-01 16:11:05 +00:00
#[cfg_attr(feature = "persistence", serde(skip))]
in_progress: Option<Receiver<Result<Response, String>>>,
2021-01-01 16:11:05 +00:00
#[cfg_attr(feature = "persistence", serde(skip))]
result: Option<Result<Resource, String>>,
2021-01-01 16:11:05 +00:00
#[cfg_attr(feature = "persistence", serde(skip))]
2020-11-20 18:50:47 +00:00
tex_mngr: TexMngr,
2020-11-17 22:24:14 +00:00
}
2021-01-01 16:11:05 +00:00
impl Default for HttpApp {
2020-11-17 22:24:14 +00:00
fn default() -> Self {
Self {
url: "https://raw.githubusercontent.com/emilk/egui/master/README.md".to_owned(),
in_progress: Default::default(),
result: Default::default(),
2020-11-20 18:50:47 +00:00
tex_mngr: Default::default(),
2020-11-17 22:24:14 +00:00
}
}
}
2021-01-01 16:11:05 +00:00
impl epi::App for HttpApp {
fn name(&self) -> &str {
2021-01-01 23:13:34 +00:00
"⬇ HTTP"
}
2020-11-17 22:24:14 +00:00
/// Called each time the UI needs repainting, which may be many times per second.
/// Put your widgets into a `SidePanel`, `TopBottomPanel`, `CentralPanel`, `Window` or `Area`.
fn update(&mut self, ctx: &egui::CtxRef, frame: &mut epi::Frame<'_>) {
if let Some(receiver) = &mut self.in_progress {
// Are we there yet?
if let Ok(result) = receiver.try_recv() {
self.in_progress = None;
self.result = Some(result.map(Resource::from_response));
}
}
2020-11-17 22:24:14 +00:00
egui::CentralPanel::default().show(ctx, |ui| {
ui.heading("HTTP Fetch Example");
ui.add(egui::github_link_file!(
"https://github.com/emilk/egui/blob/master/",
"(source code)"
));
2020-11-17 22:24:14 +00:00
if let Some(url) = ui_url(ui, frame, &mut self.url) {
2020-12-31 13:31:11 +00:00
let repaint_signal = frame.repaint_signal();
let (sender, receiver) = std::sync::mpsc::channel();
self.in_progress = Some(receiver);
2020-12-31 13:31:11 +00:00
frame.http_fetch(epi::http::Request::get(url), move |response| {
sender.send(response).ok();
repaint_signal.request_repaint();
});
}
2020-11-17 22:24:14 +00:00
ui.separator();
if self.in_progress.is_some() {
ui.label("Please wait...");
} else if let Some(result) = &self.result {
match result {
Ok(resource) => {
ui_resource(ui, frame, &mut self.tex_mngr, resource);
}
Err(error) => {
// This should only happen if the fetch API isn't available or something similar.
ui.add(egui::Label::new(error).text_color(egui::Color32::RED));
}
}
2020-11-17 22:24:14 +00:00
}
});
}
}
fn ui_url(ui: &mut egui::Ui, frame: &mut epi::Frame<'_>, url: &mut String) -> Option<String> {
let mut trigger_fetch = false;
ui.horizontal(|ui| {
ui.label("URL:");
trigger_fetch |= ui.text_edit_singleline(url).lost_focus();
trigger_fetch |= ui.button("GET").clicked();
2020-11-20 18:50:47 +00:00
});
if frame.is_web() {
ui.label("HINT: paste the url of this page into the field above!");
}
2020-11-20 18:50:47 +00:00
ui.horizontal(|ui| {
if ui.button("Source code for this example").clicked() {
*url = format!(
"https://raw.githubusercontent.com/emilk/egui/master/{}",
file!()
);
trigger_fetch = true;
}
if ui.button("Random image").clicked() {
let seed = ui.input().time;
let width = 640;
let height = 480;
*url = format!("https://picsum.photos/seed/{}/{}/{}", seed, width, height);
trigger_fetch = true;
}
});
if trigger_fetch {
Some(url.clone())
} else {
None
}
}
fn ui_resource(
2020-11-20 18:50:47 +00:00
ui: &mut egui::Ui,
2020-12-31 13:31:11 +00:00
frame: &mut epi::Frame<'_>,
2020-11-20 18:50:47 +00:00
tex_mngr: &mut TexMngr,
resource: &Resource,
) {
let Resource {
response,
image,
colored_text,
} = resource;
ui.monospace(format!("url: {}", response.url));
ui.monospace(format!(
"status: {} ({})",
response.status, response.status_text
));
ui.monospace(format!("Content-Type: {}", response.header_content_type));
ui.monospace(format!(
"Size: {:.1} kB",
response.bytes.len() as f32 / 1000.0
));
if let Some(text) = &response.text {
let tooltip = "Click to copy the response body";
if ui.button("📋").on_hover_text(tooltip).clicked() {
ui.output().copied_text = text.clone();
}
}
ui.separator();
egui::ScrollArea::auto_sized().show(ui, |ui| {
if let Some(image) = image {
2021-01-01 16:11:05 +00:00
if let Some(texture_id) = tex_mngr.texture(frame, &response.url, image) {
let size = egui::Vec2::new(image.size.0 as f32, image.size.1 as f32);
ui.image(texture_id, size);
}
} else if let Some(colored_text) = colored_text {
colored_text.ui(ui);
} else if let Some(text) = &response.text {
ui.monospace(text);
} else {
ui.monospace("[binary]");
}
});
}
// ----------------------------------------------------------------------------
// Syntax highlighting:
#[cfg(feature = "syntect")]
fn syntax_highlighting(response: &Response) -> Option<ColoredText> {
let text = response.text.as_ref()?;
let extension_and_rest: Vec<&str> = response.url.rsplitn(2, '.').collect();
let extension = extension_and_rest.get(0)?;
ColoredText::text_with_extension(text, extension)
}
/// Lines of text fragments
#[cfg(feature = "syntect")]
struct ColoredText(Vec<Vec<(syntect::highlighting::Style, String)>>);
#[cfg(feature = "syntect")]
impl ColoredText {
2021-01-01 16:11:05 +00:00
/// e.g. `text_with_extension("fn foo() {}", "rs")`
pub fn text_with_extension(text: &str, extension: &str) -> Option<ColoredText> {
use syntect::easy::HighlightLines;
use syntect::highlighting::ThemeSet;
use syntect::parsing::SyntaxSet;
use syntect::util::LinesWithEndings;
let ps = SyntaxSet::load_defaults_newlines(); // should be cached and reused
let ts = ThemeSet::load_defaults(); // should be cached and reused
let syntax = ps.find_syntax_by_extension(extension)?;
let mut h = HighlightLines::new(syntax, &ts.themes["base16-mocha.dark"]);
2021-01-01 16:11:05 +00:00
let lines = LinesWithEndings::from(text)
.map(|line| {
h.highlight(line, &ps)
.into_iter()
.map(|(style, range)| (style, range.trim_end_matches('\n').to_owned()))
.collect()
})
.collect();
Some(ColoredText(lines))
}
pub fn ui(&self, ui: &mut egui::Ui) {
for line in &self.0 {
2021-03-21 13:44:59 +00:00
ui.horizontal_wrapped(|ui| {
ui.spacing_mut().item_spacing = egui::Vec2::ZERO;
ui.set_row_height(ui.fonts()[egui::TextStyle::Body].row_height());
for (style, range) in line {
let fg = style.foreground;
2021-01-02 16:02:18 +00:00
let text_color = egui::Color32::from_rgb(fg.r, fg.g, fg.b);
ui.add(egui::Label::new(range).monospace().text_color(text_color));
}
});
}
}
}
2020-11-20 18:50:47 +00:00
#[cfg(not(feature = "syntect"))]
fn syntax_highlighting(_: &Response) -> Option<ColoredText> {
None
}
#[cfg(not(feature = "syntect"))]
struct ColoredText();
#[cfg(not(feature = "syntect"))]
impl ColoredText {
pub fn ui(&self, _ui: &mut egui::Ui) {}
}
2020-11-20 18:50:47 +00:00
// ----------------------------------------------------------------------------
// Texture/image handling is very manual at the moment.
/// Immediate mode texture manager that supports at most one texture at the time :)
2020-11-20 18:50:47 +00:00
#[derive(Default)]
struct TexMngr {
loaded_url: String,
texture_id: Option<egui::TextureId>,
}
impl TexMngr {
fn texture(
&mut self,
2020-12-31 13:31:11 +00:00
frame: &mut epi::Frame<'_>,
2020-11-20 18:50:47 +00:00
url: &str,
image: &Image,
) -> Option<egui::TextureId> {
if self.loaded_url != url {
if let Some(texture_id) = self.texture_id.take() {
frame.tex_allocator().free(texture_id);
}
self.texture_id = Some(
frame
.tex_allocator()
.alloc_srgba_premultiplied(image.size, &image.pixels),
);
2020-11-20 18:50:47 +00:00
self.loaded_url = url.to_owned();
}
self.texture_id
2020-11-20 18:50:47 +00:00
}
}
struct Image {
size: (usize, usize),
2021-01-02 16:02:18 +00:00
pixels: Vec<egui::Color32>,
2020-11-20 18:50:47 +00:00
}
impl Image {
fn decode(bytes: &[u8]) -> Option<Image> {
use image::GenericImageView;
2021-01-01 16:11:05 +00:00
let image = image::load_from_memory(bytes).ok()?;
2020-11-28 10:06:39 +00:00
let image_buffer = image.to_rgba8();
2020-11-20 18:50:47 +00:00
let size = (image.width() as usize, image.height() as usize);
let pixels = image_buffer.into_vec();
assert_eq!(size.0 * size.1 * 4, pixels.len());
let pixels = pixels
.chunks(4)
2021-01-02 16:02:18 +00:00
.map(|p| egui::Color32::from_rgba_unmultiplied(p[0], p[1], p[2], p[3]))
2020-11-20 18:50:47 +00:00
.collect();
Some(Image { size, pixels })
}
}