skia_safe/svg/
canvas.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
use std::{fmt, ops::Deref, pin::Pin, ptr};

use skia_bindings::{self as sb, SkCanvas};

use crate::{interop::DynamicMemoryWStream, prelude::*, Data, Rect};

pub struct Canvas {
    canvas: *mut SkCanvas,
    stream: Pin<Box<DynamicMemoryWStream>>,
}

impl Drop for Canvas {
    fn drop(&mut self) {
        unsafe {
            sb::C_SkCanvas_delete(self.canvas);
        }
    }
}

impl Deref for Canvas {
    type Target = crate::Canvas;

    fn deref(&self) -> &Self::Target {
        crate::Canvas::borrow_from_native(unsafe { &*self.canvas })
    }
}

bitflags! {
    #[derive(Default, Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
    pub struct Flags : u32 {
        const CONVERT_TEXT_TO_PATHS = sb::SkSVGCanvas_kConvertTextToPaths_Flag as _;
        const NO_PRETTY_XML = sb::SkSVGCanvas_kNoPrettyXML_Flag as _;
        const RELATIVE_PATH_ENCODING = sb::SkSVGCanvas_kRelativePathEncoding_Flag as _;
    }
}

impl fmt::Debug for Canvas {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        f.debug_struct("Canvas")
            .field(
                "canvas",
                crate::Canvas::borrow_from_native(unsafe { &*self.canvas }),
            )
            .field("stream", &self.stream)
            .finish()
    }
}

impl Canvas {
    /// Creates a new SVG canvas.
    pub fn new(bounds: impl AsRef<Rect>, flags: impl Into<Option<Flags>>) -> Canvas {
        let bounds = bounds.as_ref();
        let flags = flags.into().unwrap_or_default();
        let mut stream = Box::pin(DynamicMemoryWStream::new());
        let canvas = unsafe {
            sb::C_SkSVGCanvas_Make(
                bounds.native(),
                &mut stream.native_mut()._base,
                flags.bits(),
            )
        };
        Canvas { canvas, stream }
    }

    /// Ends the Canvas drawing and returns the resulting SVG.
    /// TODO: rename to into_svg() or into_svg_data()?
    pub fn end(mut self) -> Data {
        // note: flushing canvas + XMLStreamWriter does not seem to work,
        // we have to delete the canvas and destruct the stream writer
        // to get all data out _and_ keep the referential integrity.
        unsafe {
            sb::C_SkCanvas_delete(self.canvas);
        }
        self.canvas = ptr::null_mut();
        self.stream.detach_as_data()
    }
}

#[cfg(test)]
mod tests {
    use super::Canvas;
    use crate::Rect;

    #[test]
    fn test_svg() {
        use crate::Paint;

        let canvas = Canvas::new(Rect::from_size((20, 20)), None);
        let paint = Paint::default();
        canvas.draw_circle((10, 10), 10.0, &paint);
        let data = canvas.end();
        let contents = String::from_utf8_lossy(data.as_bytes());
        dbg!(&contents);
        assert!(contents.contains(r#"<ellipse cx="10" cy="10" rx="10" ry="10"/>"#));
        assert!(contents.contains(r#"</svg>"#));
    }

    #[test]
    fn test_svg_without_ending() {
        use crate::Paint;
        let canvas = Canvas::new(Rect::from_size((20, 20)), None);
        let paint = Paint::default();
        canvas.draw_circle((10, 10), 10.0, &paint);
    }
}