Skip to content

Commit 871041c

Browse files
hacknusemilk
andauthored
Added an example to save plot to image (#2769)
* implement save_plot * fix for check.sh * clippy * add save_plot to Cargo.lock * adapted for PR #2676 (removes unsafe code) * add some comments * implemented the comments from emilk * update comments in code * rustfmt * remove picked_path * add more comments * removed unused import * use `inner.response.rect` as the plot position * remove plot_location from MyApp members * sort entries * Update examples/save_plot/src/main.rs Co-authored-by: Emil Ernerfeldt <emil.ernerfeldt@gmail.com> * use env_logger instead of tracing subscriber * use env_logger instead of tracing subscriber and combine if let --------- Co-authored-by: Emil Ernerfeldt <emil.ernerfeldt@gmail.com>
1 parent 01b1b2d commit 871041c

File tree

4 files changed

+130
-0
lines changed

4 files changed

+130
-0
lines changed

Cargo.lock

+10
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

examples/save_plot/Cargo.toml

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
[package]
2+
name = "save_plot"
3+
version = "0.1.0"
4+
authors = ["hacknus <l_stoeckli@bluewin.ch>"]
5+
license = "MIT OR Apache-2.0"
6+
edition = "2021"
7+
rust-version = "1.65"
8+
publish = false
9+
10+
[dependencies]
11+
eframe = { path = "../../crates/eframe", features = [
12+
"__screenshot", # __screenshot is so we can dump a screenshot using EFRAME_SCREENSHOT_TO
13+
] }
14+
image = { version = "0.24", default-features = false, features = ["png"] }
15+
rfd = "0.11.0"
16+
env_logger = "0.10"

examples/save_plot/README.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
This example shows that you can save a plot in egui as a png.
2+
3+
```sh
4+
cargo run -p save_plot
5+
```

examples/save_plot/src/main.rs

+99
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] // hide console window on Windows in release
2+
3+
use eframe::egui;
4+
use eframe::egui::plot::{Legend, Line, Plot, PlotPoints};
5+
use eframe::egui::ColorImage;
6+
7+
fn main() -> Result<(), eframe::Error> {
8+
env_logger::init(); // Log to stderr (if you run with `RUST_LOG=debug`).
9+
10+
let options = eframe::NativeOptions {
11+
initial_window_size: Some(egui::vec2(350.0, 400.0)),
12+
..Default::default()
13+
};
14+
eframe::run_native(
15+
"My egui App with a plot",
16+
options,
17+
Box::new(|_cc| Box::new(MyApp::default())),
18+
)
19+
}
20+
21+
#[derive(Default)]
22+
struct MyApp {
23+
screenshot: Option<ColorImage>,
24+
}
25+
26+
impl eframe::App for MyApp {
27+
fn update(&mut self, ctx: &egui::Context, frame: &mut eframe::Frame) {
28+
let mut plot_rect = None;
29+
egui::CentralPanel::default().show(ctx, |ui| {
30+
// these are just some dummy variables for the example,
31+
// such that the plot is not at position (0,0)
32+
let height = 200.0;
33+
let border_x = 11.0;
34+
let border_y = 18.0;
35+
let width = 300.0;
36+
37+
ui.heading("My egui Application");
38+
39+
// add some whitespace in y direction
40+
ui.add_space(border_y);
41+
42+
if ui.button("Save Plot").clicked() {
43+
frame.request_screenshot();
44+
}
45+
46+
// add some whitespace in y direction
47+
ui.add_space(border_y);
48+
49+
ui.horizontal(|ui| {
50+
// add some whitespace in x direction
51+
ui.add_space(border_x);
52+
53+
let my_plot = Plot::new("My Plot")
54+
.height(height)
55+
.width(width)
56+
.legend(Legend::default());
57+
58+
// let's create a dummy line in the plot
59+
let graph: Vec<[f64; 2]> = vec![[0.0, 1.0], [2.0, 3.0], [3.0, 2.0]];
60+
let inner = my_plot.show(ui, |plot_ui| {
61+
plot_ui.line(Line::new(PlotPoints::from(graph)).name("curve"));
62+
});
63+
// Remember the position of the plot
64+
plot_rect = Some(inner.response.rect);
65+
});
66+
67+
// add some whitespace in y direction
68+
ui.add_space(border_y);
69+
});
70+
71+
if let (Some(screenshot), Some(plot_location)) = (self.screenshot.take(), plot_rect) {
72+
if let Some(mut path) = rfd::FileDialog::new().save_file() {
73+
path.set_extension("png");
74+
75+
// for a full size application, we should put this in a different thread,
76+
// so that the GUI doesn't lag during saving
77+
78+
let pixels_per_point = frame.info().native_pixels_per_point;
79+
let plot = screenshot.region(&plot_location, pixels_per_point);
80+
// save the plot to png
81+
image::save_buffer(
82+
&path,
83+
plot.as_raw(),
84+
plot.width() as u32,
85+
plot.height() as u32,
86+
image::ColorType::Rgba8,
87+
)
88+
.unwrap();
89+
}
90+
}
91+
}
92+
93+
fn post_rendering(&mut self, _screen_size_px: [u32; 2], frame: &eframe::Frame) {
94+
// this is inspired by the Egui screenshot example
95+
if let Some(screenshot) = frame.screenshot() {
96+
self.screenshot = Some(screenshot);
97+
}
98+
}
99+
}

0 commit comments

Comments
 (0)