reflexo-vec2dom 0.7.0

Export a frame of Typst document into dom element.
use reflexo::error::prelude::*;
use reflexo::vector::ir::{self, Module, Page, Point, Scalar};
use reflexo::vector::vm::RenderVm;
use reflexo_vec2canvas::{BBoxAt, CanvasElem, CanvasNode, CanvasOp, CanvasTask, ExportFeature};

use crate::dom::*;

/// The default feature set which is used for exporting full-fledged svg.
pub struct DefaultExportFeature;

impl ExportFeature for DefaultExportFeature {
    const ENABLE_TRACING: bool = false;
}

type Vec2Canvas = CanvasTask<DefaultExportFeature>;

#[derive(Default)]
pub struct CanvasBackend {
    vec2canvas: Vec2Canvas,

    pub pixel_per_pt: f32,
}

impl CanvasBackend {
    pub fn reset(&mut self) {
        self.pixel_per_pt = 3.;
    }

    pub fn render_page(&mut self, module: &Module, page: &Page) -> Result<CanvasNode> {
        // todo: incremental
        let mut ct = self.vec2canvas.fork_canvas_render_task(module);

        Ok(ct.render_item(&page.content))
    }
}

impl TypstPageElem {
    pub fn attach_canvas(&mut self, g: CanvasNode) {
        self.g.attach_canvas(g)
    }

    pub fn mark_painted(&mut self) {
        self.g.mark_painted();
    }

    pub fn get_damage_rect(&mut self, ts: tiny_skia::Transform) -> Option<ir::Rect> {
        self.g.get_damage_rect(ts, true)
    }

    pub async fn repaint_canvas(
        &mut self,
        ts: tiny_skia::Transform,
        panel: &web_sys::CanvasRenderingContext2d,
    ) {
        self.g.repaint_canvas(ts, panel, true).await;
    }
}

impl TypstElem {
    pub fn attach_canvas(&mut self, g: CanvasNode) {
        use TypstDomExtra::*;

        self.canvas = Some(g.clone());

        match &mut self.extra {
            Group(gr) => {
                let CanvasElem::Group(c) = g.as_ref() else {
                    panic!("Invalid group canvas: {}", self.f.as_svg_id("g"));
                };

                let this = gr.children.iter_mut().map(|t| &mut t.1);
                let this_c = c.inner.iter().map(|t| t.1.clone());

                for (elem, c_elem) in this.zip(this_c) {
                    elem.attach_canvas(c_elem);
                }
            }
            Item(ch) => {
                let mut g = g.as_ref();
                if let CanvasElem::Clip(c) = g {
                    g = &c.inner;
                };
                let CanvasElem::Group(c) = g else {
                    panic!("Invalid item canvas: {}", self.f.as_svg_id("g"));
                };
                if c.inner.len() != 1 {
                    panic!("Invalid item canvas length: {}", self.f.as_svg_id("g"));
                }

                ch.child.attach_canvas(c.inner[0].1.clone());
            }
            Label(ch) => {
                ch.child.attach_canvas(g);
            }
            RawHtml(..) | Link(..) | Image(..) | Text(..) | Path(..) | ContentHint(..) => {}
        };
    }

    fn mark_painted(&mut self) {
        use TypstDomExtra::*;

        match &mut self.extra {
            Group(gr) => {
                for (_, child) in gr.children.iter_mut() {
                    child.mark_painted();
                }
            }
            Item(ch) => {
                ch.child.mark_painted();
            }
            _ => {
                self.is_canvas_painted = true;
            }
        }
    }

    fn get_damage_rect(&mut self, ts: tiny_skia::Transform, visible: bool) -> Option<ir::Rect> {
        use TypstDomExtra::*;

        let visible = visible && self.is_svg_visible;
        match &mut self.extra {
            ContentHint(_) => None,
            Group(i) => {
                let g = self.canvas.as_deref().unwrap();

                let CanvasElem::Group(c) = g else {
                    panic!("Invalid group canvas: {}", self.f.as_svg_id("g"));
                };

                let ts = ts.pre_concat(*c.ts.as_ref());

                i.children
                    .iter_mut()
                    .map(|t| {
                        t.1.get_damage_rect(ts, visible).map(|r| {
                            let dp = t.0;
                            let dp = Point::new(
                                Scalar(dp.x.0 * ts.sx + dp.y.0 * ts.kx),
                                Scalar(dp.y.0 * ts.sy + dp.x.0 * ts.ky),
                            );
                            r.translate(dp)
                        })
                    })
                    .fold(None, |acc, r| match (acc, r) {
                        (Some(a), Some(b)) => Some(a.union(&b)),
                        (a, b) => a.or(b),
                    })
            }
            Item(i) => {
                let g = self.canvas.as_deref().unwrap();

                let (g, clip) = if let CanvasElem::Clip(clip) = g {
                    (clip.inner.as_ref(), Some(clip))
                } else {
                    (g, None)
                };

                let CanvasElem::Group(c) = g else {
                    panic!("Invalid group canvas: {}", self.f.as_svg_id("g"));
                };

                let ts = ts.pre_concat(*c.ts.as_ref());

                let child_bbox = i.child.get_damage_rect(ts, visible);

                let clip = clip.and_then(|c| c.clip_bbox_at(ts));
                match (clip, child_bbox) {
                    (Some(clip), Some(child_bbox)) => Some(clip.intersect(&child_bbox)),
                    (None, Some(child_bbox)) => Some(child_bbox),
                    _ => None,
                }
            }
            _ => {
                // let damaged = visible == self.is_canvas_painted;
                let damaged = visible;
                if !damaged {
                    return None;
                }

                // web_sys::console::log_1(
                //     &format!(
                //         "get_damage_rect_partial: {} vis:{visible} cvs:{} {ret:?}",
                //         self.f.as_svg_id("g"),
                //         self.is_canvas_painted,
                //     )
                //     .into(),
                // );

                self.canvas.as_ref().unwrap().bbox_at(ts)
            }
        }
    }

    #[async_recursion::async_recursion(?Send)]
    async fn repaint_canvas(
        &mut self,
        ts: tiny_skia::Transform,
        panel: &web_sys::CanvasRenderingContext2d,
        svg_visible: bool,
    ) {
        use TypstDomExtra::*;

        let svg_visible = svg_visible && self.is_svg_visible;
        match &mut self.extra {
            ContentHint(_) => {}
            Group(i) => {
                // if !visible {
                //     self.canvas.as_ref().unwrap().realize(ts, panel).await;
                //     return;
                // }

                let g = self.canvas.as_deref().unwrap();

                let CanvasElem::Group(c) = g else {
                    panic!("Invalid group canvas: {}", self.f.as_svg_id("g"));
                };

                let ts = ts.pre_concat(*c.ts.as_ref());

                for (p, child) in i.children.iter_mut() {
                    let ts = ts.pre_translate(p.x.0, p.y.0);
                    child.repaint_canvas(ts, panel, svg_visible).await;
                }
            }
            Item(i) => {
                // if !visible {
                //     self.canvas.as_ref().unwrap().realize(ts, panel).await;
                //     return;
                // }

                let g = self.canvas.as_deref().unwrap();

                let (g, _clip_guard) = if let CanvasElem::Clip(clip) = g {
                    (clip.inner.as_ref(), Some(clip.realize_with(ts, panel)))
                } else {
                    (g, None)
                };

                let CanvasElem::Group(c) = g else {
                    panic!("Invalid group canvas: {}", self.f.as_svg_id("g"));
                };

                let ts = ts.pre_concat(*c.ts.as_ref());

                // todo: intersect viewport
                // if let TransformItem::Clip(c) = g.trans {

                // }
                i.child.repaint_canvas(ts, panel, svg_visible).await;
            }
            _ => {
                // web_sys::console::log_2(
                //     &format!(
                //         "repaint_canvas_partial: {} {} {}",
                //         self.f.as_svg_id("g"),
                //         visible,
                //         self.is_svg_visible
                //     )
                //     .into(),
                //     &self.g,
                // );

                match (svg_visible, self.is_canvas_painted) {
                    (true, false) | (false, true) => {}
                    (true, true) => {
                        self.is_canvas_painted = false;
                    }
                    (false, false) => {
                        self.canvas.as_ref().unwrap().realize(ts, panel).await;
                        self.is_canvas_painted = true;
                        return;
                    }
                }
            }
        }
    }
}