use std::any::Any;
use time::Instant;
use piet_common::kurbo::{Line, Size};
use piet_common::{Color, FontFamily, Piet, RenderContext, Text, TextLayoutBuilder};
use druid_shell::{Application, KeyEvent, Region, WinHandler, WindowBuilder, WindowHandle};
const BG_COLOR: Color = Color::rgb8(0x27, 0x28, 0x22);
const FG_COLOR: Color = Color::rgb8(0xf0, 0xf0, 0xea);
const RED: Color = Color::rgb8(0xff, 0x80, 0x80);
const CYAN: Color = Color::rgb8(0x80, 0xff, 0xff);
struct PerfTest {
handle: WindowHandle,
size: Size,
start_time: Instant,
last_time: Instant,
red: bool,
}
impl WinHandler for PerfTest {
fn connect(&mut self, handle: &WindowHandle) {
self.handle = handle.clone();
}
fn prepare_paint(&mut self) {
self.handle.invalidate();
}
fn paint(&mut self, piet: &mut Piet, _: &Region) {
let rect = self.size.to_rect();
piet.fill(rect, &BG_COLOR);
piet.stroke(
Line::new((0.0, self.size.height), (self.size.width, 0.0)),
&FG_COLOR,
1.0,
);
let current_ns = (Instant::now() - self.start_time).whole_nanoseconds();
let th = ::std::f64::consts::PI * (current_ns as f64) * 2e-9;
let dx = 100.0 * th.sin();
let dy = 100.0 * th.cos();
piet.stroke(
Line::new((100.0, 100.0), (100.0 + dx, 100.0 - dy)),
&FG_COLOR,
1.0,
);
let now = Instant::now();
let msg = format!("{}ms", (now - self.last_time).whole_milliseconds());
self.last_time = now;
let layout = piet
.text()
.new_text_layout(msg)
.font(FontFamily::MONOSPACE, 15.0)
.text_color(FG_COLOR)
.build()
.unwrap();
piet.draw_text(&layout, (10.0, 210.0));
let msg = "VSYNC";
let color = if self.red { RED } else { CYAN };
let layout = piet
.text()
.new_text_layout(msg)
.text_color(color)
.font(FontFamily::MONOSPACE, 48.0)
.build()
.unwrap();
piet.draw_text(&layout, (10.0, 280.0));
self.red = !self.red;
let msg = "Hello DWrite! This is a somewhat longer string of text intended to provoke slightly longer draw times.";
let layout = piet
.text()
.new_text_layout(msg)
.font(FontFamily::MONOSPACE, 15.0)
.text_color(FG_COLOR)
.build()
.unwrap();
let dy = 15.0;
let x0 = 210.0;
let y0 = 10.0;
for i in 0..60 {
let y = y0 + (i as f64) * dy;
piet.draw_text(&layout, (x0, y));
}
self.handle.request_anim_frame();
}
fn command(&mut self, id: u32) {
match id {
0x100 => self.handle.close(),
_ => println!("unexpected id {id}"),
}
}
fn key_down(&mut self, event: KeyEvent) -> bool {
println!("keydown: {event:?}");
false
}
fn size(&mut self, size: Size) {
self.size = size;
}
fn request_close(&mut self) {
self.handle.close();
}
fn destroy(&mut self) {
Application::global().quit()
}
fn as_any(&mut self) -> &mut dyn Any {
self
}
}
fn main() {
tracing_subscriber::fmt().init();
let app = Application::new().unwrap();
let mut builder = WindowBuilder::new(app.clone());
let perf_test = PerfTest {
size: Size::ZERO,
handle: Default::default(),
start_time: time::Instant::now(),
last_time: time::Instant::now(),
red: true,
};
builder.set_handler(Box::new(perf_test));
builder.set_title("Performance tester");
let window = builder.build().unwrap();
window.show();
app.run(None);
}