use crate::Facet;
#[derive(Debug, Clone)]
pub struct RenderReport {
pub title: String,
pub state: serde_json::Value,
pub vertices: usize,
}
impl RenderReport {
pub fn drew(&self) -> bool {
self.vertices > 0
}
}
#[allow(deprecated)] fn capture(ctx: &egui::Context, facet: &mut dyn Facet, size: (f32, f32)) -> RenderReport {
let title = facet.title().to_string();
crate::trace::emit_in(
"facet.render",
&serde_json::json!({ "title": title, "size": [size.0, size.1] }),
);
let input = egui::RawInput {
screen_rect: Some(egui::Rect::from_min_size(egui::pos2(0.0, 0.0), egui::vec2(size.0, size.1))),
..Default::default()
};
let output = ctx.run(input, |ctx| {
egui::CentralPanel::default().show(ctx, |ui| facet.ui(ui));
});
let prims = ctx.tessellate(output.shapes, output.pixels_per_point);
let vertices = prims
.iter()
.map(|p| match &p.primitive {
egui::epaint::Primitive::Mesh(m) => m.vertices.len(),
_ => 0,
})
.sum();
let report = RenderReport { title, state: facet.state_json(), vertices };
log(&report);
trail(Kind::Render, format!("{} size={}x{} → {} verts", report.title, size.0 as i32, size.1 as i32, vertices));
dump_state(&report);
crate::trace::emit_out(
"facet.render",
&serde_json::json!({
"title": report.title,
"vertices": report.vertices,
"drew": report.drew(),
"state": report.state,
}),
);
report
}
pub fn render_sized(facet: &mut dyn Facet, size: (f32, f32)) -> RenderReport {
capture(&egui::Context::default(), facet, size)
}
pub fn headless_render(facet: &mut dyn Facet) -> RenderReport {
render_sized(facet, (800.0, 600.0))
}
pub fn render_themed(facet: &mut dyn Facet, theme: crate::Theme) -> RenderReport {
let ctx = egui::Context::default();
crate::set_theme(&ctx, theme);
capture(&ctx, facet, (800.0, 600.0))
}
pub fn log(r: &RenderReport) {
let full = r.state.to_string();
let shown: String = if full.chars().count() > 160 {
full.chars().take(159).chain(std::iter::once('…')).collect()
} else {
full
};
eprintln!("facett: {:<14} {:>7} verts · {}", r.title, r.vertices, shown);
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub enum Kind {
Render,
State,
Case,
}
impl Kind {
pub fn tag(self) -> &'static str {
match self {
Kind::Render => "RENDER",
Kind::State => "STATE",
Kind::Case => "CASE",
}
}
}
fn next_seq() -> u64 {
use std::sync::atomic::{AtomicU64, Ordering};
static SEQ: AtomicU64 = AtomicU64::new(0);
SEQ.fetch_add(1, Ordering::Relaxed) + 1
}
fn now_stamp() -> String {
let now = std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.unwrap_or_default();
let total_ms = now.as_millis();
let ms = (total_ms % 1000) as u64;
let secs = (total_ms / 1000) as u64;
let h = (secs / 3600) % 24;
let m = (secs / 60) % 60;
let s = secs % 60;
format!("{h:02}:{m:02}:{s:02}.{ms:03}")
}
pub fn trail(kind: Kind, detail: impl AsRef<str>) {
let stamp = now_stamp();
let seq = next_seq();
let detail = detail.as_ref();
let line = format!("facett ACTION {stamp} {seq:>5} [{}] {detail}", kind.tag());
eprintln!("{line}");
if let Ok(path) = std::env::var("FACETT_TRAIL") {
use std::io::Write;
if let Ok(mut f) = std::fs::OpenOptions::new().create(true).append(true).open(path) {
let _ = writeln!(f, "{line}");
}
}
}
pub fn dump_state(r: &RenderReport) {
eprintln!("facett STATE {} = {}", r.title, r.state);
trail(Kind::State, format!("{} state={}", r.title, r.state));
}
pub fn case_summary(component: &str, axis: &str, r: &RenderReport) {
eprintln!(
"facett CASE {:<14} {:<16} → {:>8} verts drew={} state={}",
component,
axis,
r.vertices,
r.drew(),
r.state,
);
trail(Kind::Case, format!("{component} {axis} verts={} drew={}", r.vertices, r.drew()));
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Scene, hash_color};
struct Tiny(Scene);
impl Facet for Tiny {
fn title(&self) -> &str {
"tiny"
}
fn ui(&mut self, ui: &mut egui::Ui) {
crate::draw(ui, &self.0, crate::Layout::Circular, "empty");
}
fn state_json(&self) -> serde_json::Value {
serde_json::json!({ "nodes": self.0.nodes.len() })
}
}
#[test]
fn now_stamp_is_hms_millis_shaped() {
let s = now_stamp();
assert_eq!(s.len(), 12, "stamp `{s}` should be HH:MM:SS.mmm");
assert_eq!(s.matches(':').count(), 2, "stamp `{s}` needs two colons");
assert_eq!(s.matches('.').count(), 1, "stamp `{s}` needs one dot");
}
#[test]
fn seq_is_monotonic() {
let a = next_seq();
let b = next_seq();
assert!(b > a, "seq must strictly increase: {a} then {b}");
}
#[test]
fn kind_tags_are_distinct() {
let tags = [Kind::Render.tag(), Kind::State.tag(), Kind::Case.tag()];
for (i, t) in tags.iter().enumerate() {
assert!(!t.is_empty());
assert!(!tags[..i].contains(t), "duplicate tag {t}");
}
}
#[test]
fn headless_render_captures_state_and_draws() {
let mut scene = Scene::new();
let a = scene.node("a", hash_color("a"));
let b = scene.node("b", hash_color("b"));
scene.edge(a, b);
let mut t = Tiny(scene);
let r = headless_render(&mut t);
assert_eq!(r.title, "tiny");
assert_eq!(r.state["nodes"], 2);
assert!(r.drew(), "a 2-node graph should tessellate to vertices");
}
}