2020-11-21 04:26:02 +11:00
|
|
|
use std::{borrow::Cow, ops::RangeBounds};
|
2020-04-23 03:28:38 +10:00
|
|
|
|
2020-06-11 07:10:28 +10:00
|
|
|
use piet_gpu_types::encoder::{Encode, Encoder};
|
2020-05-12 13:01:06 +10:00
|
|
|
|
2020-11-20 06:53:59 +11:00
|
|
|
use piet_gpu_types::scene::{
|
2021-03-19 23:49:47 +11:00
|
|
|
Clip, CubicSeg, Element, FillColor, SetFillMode, LineSeg, QuadSeg, SetLineWidth, Transform,
|
2020-11-20 06:53:59 +11:00
|
|
|
};
|
2020-04-23 03:28:38 +10:00
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
use piet::{
|
2020-11-20 06:53:59 +11:00
|
|
|
kurbo::{Affine, Insets, PathEl, Point, Rect, Shape, Size},
|
2020-11-15 03:22:56 +11:00
|
|
|
HitTestPosition, TextAttribute, TextStorage,
|
|
|
|
};
|
2020-04-23 03:28:38 +10:00
|
|
|
|
|
|
|
use piet::{
|
2020-11-15 03:22:56 +11:00
|
|
|
Color, Error, FixedGradient, FontFamily, HitTestPoint, ImageFormat, InterpolationMode,
|
|
|
|
IntoBrush, LineMetric, RenderContext, StrokeStyle, Text, TextLayout, TextLayoutBuilder,
|
2020-04-23 03:28:38 +10:00
|
|
|
};
|
|
|
|
|
|
|
|
pub struct PietGpuImage;
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct PietGpuTextLayout;
|
|
|
|
|
|
|
|
pub struct PietGpuTextLayoutBuilder;
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
#[derive(Clone)]
|
2020-04-23 03:28:38 +10:00
|
|
|
pub struct PietGpuText;
|
|
|
|
|
|
|
|
pub struct PietGpuRenderContext {
|
|
|
|
encoder: Encoder,
|
2020-05-12 13:01:06 +10:00
|
|
|
elements: Vec<Element>,
|
2020-04-23 03:28:38 +10:00
|
|
|
// Will probably need direct accesss to hal Device to create images etc.
|
|
|
|
inner_text: PietGpuText,
|
2020-05-12 13:01:06 +10:00
|
|
|
stroke_width: f32,
|
2021-03-19 23:49:47 +11:00
|
|
|
fill_mode: FillMode,
|
2020-06-03 10:10:20 +10:00
|
|
|
// We're tallying these cpu-side for expedience, but will probably
|
|
|
|
// move this to some kind of readback from element processing.
|
2020-11-20 06:53:59 +11:00
|
|
|
/// The count of elements that make it through to coarse rasterization.
|
2020-06-03 10:10:20 +10:00
|
|
|
path_count: usize,
|
2020-11-20 06:53:59 +11:00
|
|
|
/// The count of path segment elements.
|
2020-06-03 10:10:20 +10:00
|
|
|
pathseg_count: usize,
|
2021-03-15 22:28:04 +11:00
|
|
|
/// The count of transform elements.
|
|
|
|
trans_count: usize,
|
2020-11-20 06:53:59 +11:00
|
|
|
|
|
|
|
cur_transform: Affine,
|
|
|
|
state_stack: Vec<State>,
|
|
|
|
clip_stack: Vec<ClipElement>,
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub enum PietGpuBrush {
|
|
|
|
Solid(u32),
|
|
|
|
Gradient,
|
|
|
|
}
|
|
|
|
|
2020-11-20 06:53:59 +11:00
|
|
|
#[derive(Default)]
|
|
|
|
struct State {
|
|
|
|
/// The transform relative to the parent state.
|
2020-11-21 04:26:02 +11:00
|
|
|
rel_transform: Affine,
|
|
|
|
/// The transform at the parent state.
|
|
|
|
///
|
|
|
|
/// This invariant should hold: transform * rel_transform = cur_transform
|
2020-11-20 06:53:59 +11:00
|
|
|
transform: Affine,
|
|
|
|
n_clip: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ClipElement {
|
|
|
|
/// Index of BeginClip element in element vec, for bbox fixup.
|
|
|
|
begin_ix: usize,
|
|
|
|
bbox: Option<Rect>,
|
|
|
|
}
|
|
|
|
|
2021-03-19 23:49:47 +11:00
|
|
|
#[derive(Clone,Copy,PartialEq)]
|
2021-03-17 21:08:28 +11:00
|
|
|
enum FillMode {
|
|
|
|
// Fill path according to the non-zero winding rule.
|
|
|
|
Nonzero = 0,
|
|
|
|
// Fill stroked path.
|
|
|
|
Stroke = 1,
|
|
|
|
}
|
|
|
|
|
2020-05-03 02:50:36 +10:00
|
|
|
const TOLERANCE: f64 = 0.25;
|
2020-04-25 06:06:47 +10:00
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
impl PietGpuRenderContext {
|
|
|
|
pub fn new() -> PietGpuRenderContext {
|
2020-05-12 13:01:06 +10:00
|
|
|
let encoder = Encoder::new();
|
|
|
|
let elements = Vec::new();
|
2020-04-23 03:28:38 +10:00
|
|
|
let inner_text = PietGpuText;
|
2020-05-12 13:01:06 +10:00
|
|
|
let stroke_width = 0.0;
|
2020-04-23 03:28:38 +10:00
|
|
|
PietGpuRenderContext {
|
|
|
|
encoder,
|
2020-05-12 13:01:06 +10:00
|
|
|
elements,
|
2020-04-23 03:28:38 +10:00
|
|
|
inner_text,
|
2020-05-12 13:01:06 +10:00
|
|
|
stroke_width,
|
2021-03-19 23:49:47 +11:00
|
|
|
fill_mode: FillMode::Nonzero,
|
2020-06-03 10:10:20 +10:00
|
|
|
path_count: 0,
|
|
|
|
pathseg_count: 0,
|
2021-03-15 22:28:04 +11:00
|
|
|
trans_count: 0,
|
2020-11-20 06:53:59 +11:00
|
|
|
cur_transform: Affine::default(),
|
|
|
|
state_stack: Vec::new(),
|
|
|
|
clip_stack: Vec::new(),
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_scene_buf(&mut self) -> &[u8] {
|
2020-05-12 13:01:06 +10:00
|
|
|
self.elements.encode(&mut self.encoder);
|
2020-04-23 03:28:38 +10:00
|
|
|
self.encoder.buf()
|
|
|
|
}
|
2020-06-03 10:10:20 +10:00
|
|
|
|
|
|
|
pub fn path_count(&self) -> usize {
|
|
|
|
self.path_count
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn pathseg_count(&self) -> usize {
|
|
|
|
self.pathseg_count
|
|
|
|
}
|
2021-03-15 22:28:04 +11:00
|
|
|
|
|
|
|
pub fn trans_count(&self) -> usize {
|
|
|
|
self.trans_count
|
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
2021-03-19 23:49:47 +11:00
|
|
|
fn set_fill_mode(ctx: &mut PietGpuRenderContext, fill_mode: FillMode) {
|
|
|
|
if ctx.fill_mode != fill_mode {
|
|
|
|
ctx.elements
|
|
|
|
.push(Element::SetFillMode(SetFillMode { fill_mode: fill_mode as u32 }));
|
|
|
|
ctx.fill_mode = fill_mode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
impl RenderContext for PietGpuRenderContext {
|
|
|
|
type Brush = PietGpuBrush;
|
|
|
|
type Image = PietGpuImage;
|
|
|
|
type Text = PietGpuText;
|
|
|
|
type TextLayout = PietGpuTextLayout;
|
|
|
|
|
|
|
|
fn status(&mut self) -> Result<(), Error> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn solid_brush(&mut self, color: Color) -> Self::Brush {
|
|
|
|
PietGpuBrush::Solid(color.as_rgba_u32())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gradient(&mut self, _gradient: impl Into<FixedGradient>) -> Result<Self::Brush, Error> {
|
|
|
|
Ok(Self::Brush::Gradient)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear(&mut self, _color: Color) {}
|
|
|
|
|
2020-04-25 06:06:47 +10:00
|
|
|
fn stroke(&mut self, shape: impl Shape, brush: &impl IntoBrush<Self>, width: f64) {
|
2020-11-20 06:53:59 +11:00
|
|
|
let width_f32 = width as f32;
|
|
|
|
if self.stroke_width != width_f32 {
|
2020-05-12 13:01:06 +10:00
|
|
|
self.elements
|
2020-11-20 06:53:59 +11:00
|
|
|
.push(Element::SetLineWidth(SetLineWidth { width: width_f32 }));
|
|
|
|
self.stroke_width = width_f32;
|
2020-05-12 13:01:06 +10:00
|
|
|
}
|
2021-03-19 23:49:47 +11:00
|
|
|
set_fill_mode(self, FillMode::Stroke);
|
2020-05-12 13:01:06 +10:00
|
|
|
let brush = brush.make_brush(self, || shape.bounding_box()).into_owned();
|
2020-04-25 06:06:47 +10:00
|
|
|
match brush {
|
|
|
|
PietGpuBrush::Solid(rgba_color) => {
|
2020-11-20 06:53:59 +11:00
|
|
|
// Note: the bbox contribution of stroke becomes more complicated with miter joins.
|
|
|
|
self.accumulate_bbox(|| shape.bounding_box() + Insets::uniform(width * 0.5));
|
|
|
|
let path = shape.path_elements(TOLERANCE);
|
|
|
|
self.encode_path(path, false);
|
2021-03-17 21:08:28 +11:00
|
|
|
let stroke = FillColor { rgba_color };
|
2021-03-19 23:49:47 +11:00
|
|
|
self.elements.push(Element::FillColor(stroke));
|
2020-06-03 10:10:20 +10:00
|
|
|
self.path_count += 1;
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
|
|
|
|
fn stroke_styled(
|
|
|
|
&mut self,
|
|
|
|
_shape: impl Shape,
|
|
|
|
_brush: &impl IntoBrush<Self>,
|
|
|
|
_width: f64,
|
|
|
|
_style: &StrokeStyle,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fill(&mut self, shape: impl Shape, brush: &impl IntoBrush<Self>) {
|
2020-04-25 06:06:47 +10:00
|
|
|
let brush = brush.make_brush(self, || shape.bounding_box()).into_owned();
|
2020-11-20 06:53:59 +11:00
|
|
|
if let PietGpuBrush::Solid(rgba_color) = brush {
|
|
|
|
// Note: we might get a good speedup from using an approximate bounding box.
|
|
|
|
// Perhaps that should be added to kurbo.
|
|
|
|
self.accumulate_bbox(|| shape.bounding_box());
|
|
|
|
let path = shape.path_elements(TOLERANCE);
|
2021-03-19 23:49:47 +11:00
|
|
|
set_fill_mode(self, FillMode::Nonzero);
|
2020-11-20 06:53:59 +11:00
|
|
|
self.encode_path(path, true);
|
2021-03-17 21:08:28 +11:00
|
|
|
let fill = FillColor { rgba_color };
|
2021-03-19 23:49:47 +11:00
|
|
|
self.elements.push(Element::FillColor(fill));
|
2020-11-20 06:53:59 +11:00
|
|
|
self.path_count += 1;
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fill_even_odd(&mut self, _shape: impl Shape, _brush: &impl IntoBrush<Self>) {}
|
|
|
|
|
2020-11-20 06:53:59 +11:00
|
|
|
fn clip(&mut self, shape: impl Shape) {
|
2021-03-19 23:49:47 +11:00
|
|
|
set_fill_mode(self, FillMode::Nonzero);
|
2020-11-20 06:53:59 +11:00
|
|
|
let path = shape.path_elements(TOLERANCE);
|
|
|
|
self.encode_path(path, true);
|
2020-11-21 04:26:02 +11:00
|
|
|
let begin_ix = self.elements.len();
|
|
|
|
self.elements.push(Element::BeginClip(Clip {
|
2020-11-20 06:53:59 +11:00
|
|
|
bbox: Default::default(),
|
|
|
|
}));
|
|
|
|
self.clip_stack.push(ClipElement {
|
|
|
|
bbox: None,
|
|
|
|
begin_ix,
|
|
|
|
});
|
|
|
|
self.path_count += 1;
|
2020-11-21 04:26:02 +11:00
|
|
|
if let Some(tos) = self.state_stack.last_mut() {
|
|
|
|
tos.n_clip += 1;
|
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
|
|
|
|
fn text(&mut self) -> &mut Self::Text {
|
|
|
|
&mut self.inner_text
|
|
|
|
}
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn draw_text(&mut self, _layout: &Self::TextLayout, _pos: impl Into<Point>) {}
|
2020-04-23 03:28:38 +10:00
|
|
|
|
|
|
|
fn save(&mut self) -> Result<(), Error> {
|
2020-11-21 04:26:02 +11:00
|
|
|
self.state_stack.push(State {
|
|
|
|
rel_transform: Affine::default(),
|
|
|
|
transform: self.cur_transform,
|
|
|
|
n_clip: 0,
|
|
|
|
});
|
2020-04-23 03:28:38 +10:00
|
|
|
Ok(())
|
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
fn restore(&mut self) -> Result<(), Error> {
|
2020-11-20 06:53:59 +11:00
|
|
|
if let Some(state) = self.state_stack.pop() {
|
2020-11-21 04:26:02 +11:00
|
|
|
if state.rel_transform != Affine::default() {
|
|
|
|
let a_inv = state.rel_transform.inverse();
|
2020-11-20 06:53:59 +11:00
|
|
|
self.elements
|
|
|
|
.push(Element::Transform(to_scene_transform(a_inv)));
|
2021-03-15 22:28:04 +11:00
|
|
|
self.trans_count += 1;
|
2020-11-20 06:53:59 +11:00
|
|
|
}
|
2020-11-21 04:26:02 +11:00
|
|
|
self.cur_transform = state.transform;
|
2020-11-20 06:53:59 +11:00
|
|
|
for _ in 0..state.n_clip {
|
|
|
|
self.pop_clip();
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(Error::StackUnbalance)
|
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
fn finish(&mut self) -> Result<(), Error> {
|
2020-11-21 04:26:02 +11:00
|
|
|
for _ in 0..self.clip_stack.len() {
|
|
|
|
self.pop_clip();
|
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
Ok(())
|
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
|
|
|
|
fn transform(&mut self, transform: Affine) {
|
|
|
|
self.elements
|
|
|
|
.push(Element::Transform(to_scene_transform(transform)));
|
2021-03-15 22:28:04 +11:00
|
|
|
self.trans_count += 1;
|
2020-11-20 06:53:59 +11:00
|
|
|
if let Some(tos) = self.state_stack.last_mut() {
|
2020-11-21 04:26:02 +11:00
|
|
|
tos.rel_transform *= transform;
|
2020-11-20 06:53:59 +11:00
|
|
|
}
|
|
|
|
self.cur_transform *= transform;
|
|
|
|
}
|
2020-04-23 03:28:38 +10:00
|
|
|
|
|
|
|
fn make_image(
|
|
|
|
&mut self,
|
|
|
|
_width: usize,
|
|
|
|
_height: usize,
|
|
|
|
_buf: &[u8],
|
|
|
|
_format: ImageFormat,
|
|
|
|
) -> Result<Self::Image, Error> {
|
|
|
|
Ok(PietGpuImage)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn draw_image(
|
|
|
|
&mut self,
|
|
|
|
_image: &Self::Image,
|
|
|
|
_rect: impl Into<Rect>,
|
|
|
|
_interp: InterpolationMode,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
fn draw_image_area(
|
|
|
|
&mut self,
|
|
|
|
_image: &Self::Image,
|
|
|
|
_src_rect: impl Into<Rect>,
|
|
|
|
_dst_rect: impl Into<Rect>,
|
|
|
|
_interp: InterpolationMode,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
fn blurred_rect(&mut self, _rect: Rect, _blur_radius: f64, _brush: &impl IntoBrush<Self>) {}
|
|
|
|
|
|
|
|
fn current_transform(&self) -> Affine {
|
2020-11-20 06:53:59 +11:00
|
|
|
self.cur_transform
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_save(&mut self, f: impl FnOnce(&mut Self) -> Result<(), Error>) -> Result<(), Error> {
|
|
|
|
self.save()?;
|
|
|
|
// Always try to restore the stack, even if `f` errored.
|
|
|
|
f(self).and(self.restore())
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-12 13:01:06 +10:00
|
|
|
impl PietGpuRenderContext {
|
2021-03-19 23:49:47 +11:00
|
|
|
fn encode_line_seg(&mut self, seg: LineSeg) {
|
|
|
|
self.elements.push(Element::Line(seg));
|
2020-06-03 10:10:20 +10:00
|
|
|
self.pathseg_count += 1;
|
2020-05-21 00:38:52 +10:00
|
|
|
}
|
|
|
|
|
2021-03-19 23:49:47 +11:00
|
|
|
fn encode_quad_seg(&mut self, seg: QuadSeg) {
|
|
|
|
self.elements.push(Element::Quad(seg));
|
2020-06-10 09:01:47 +10:00
|
|
|
self.pathseg_count += 1;
|
|
|
|
}
|
|
|
|
|
2021-03-19 23:49:47 +11:00
|
|
|
fn encode_cubic_seg(&mut self, seg: CubicSeg) {
|
|
|
|
self.elements.push(Element::Cubic(seg));
|
2020-06-10 09:01:47 +10:00
|
|
|
self.pathseg_count += 1;
|
|
|
|
}
|
|
|
|
|
2020-05-21 00:38:52 +10:00
|
|
|
fn encode_path(&mut self, path: impl Iterator<Item = PathEl>, is_fill: bool) {
|
2021-03-17 02:51:13 +11:00
|
|
|
if is_fill {
|
|
|
|
self.encode_path_inner(path.flat_map(|el| {
|
|
|
|
match el {
|
|
|
|
PathEl::MoveTo(..) => {
|
|
|
|
Some(PathEl::ClosePath)
|
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
}.into_iter().chain(Some(el))
|
2021-03-19 23:49:47 +11:00
|
|
|
}).chain(Some(PathEl::ClosePath)))
|
2021-03-17 02:51:13 +11:00
|
|
|
} else {
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_path_inner(path)
|
2021-03-17 02:51:13 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-19 23:49:47 +11:00
|
|
|
fn encode_path_inner(&mut self, path: impl Iterator<Item = PathEl>) {
|
2020-06-10 10:20:58 +10:00
|
|
|
let flatten = false;
|
2020-05-12 13:01:06 +10:00
|
|
|
if flatten {
|
|
|
|
let mut start_pt = None;
|
|
|
|
let mut last_pt = None;
|
|
|
|
piet::kurbo::flatten(path, TOLERANCE, |el| {
|
|
|
|
match el {
|
|
|
|
PathEl::MoveTo(p) => {
|
|
|
|
let scene_pt = to_f32_2(p);
|
2020-05-21 09:36:09 +10:00
|
|
|
start_pt = Some(scene_pt);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_pt);
|
|
|
|
}
|
|
|
|
PathEl::LineTo(p) => {
|
|
|
|
let scene_pt = to_f32_2(p);
|
|
|
|
let seg = LineSeg {
|
|
|
|
p0: last_pt.unwrap(),
|
|
|
|
p1: scene_pt,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_line_seg(seg);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_pt);
|
|
|
|
}
|
|
|
|
PathEl::ClosePath => {
|
|
|
|
if let (Some(start), Some(last)) = (start_pt.take(), last_pt.take()) {
|
2020-05-21 09:36:09 +10:00
|
|
|
if last != start {
|
|
|
|
let seg = LineSeg {
|
|
|
|
p0: last,
|
|
|
|
p1: start,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_line_seg(seg);
|
2020-05-21 09:36:09 +10:00
|
|
|
}
|
2020-05-12 13:01:06 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => (),
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
2020-05-12 13:01:06 +10:00
|
|
|
//println!("{:?}", el);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
let mut start_pt = None;
|
|
|
|
let mut last_pt = None;
|
|
|
|
for el in path {
|
|
|
|
match el {
|
|
|
|
PathEl::MoveTo(p) => {
|
|
|
|
let scene_pt = to_f32_2(p);
|
2020-05-21 09:36:09 +10:00
|
|
|
start_pt = Some(scene_pt);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_pt);
|
|
|
|
}
|
|
|
|
PathEl::LineTo(p) => {
|
|
|
|
let scene_pt = to_f32_2(p);
|
|
|
|
let seg = LineSeg {
|
|
|
|
p0: last_pt.unwrap(),
|
|
|
|
p1: scene_pt,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_line_seg(seg);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_pt);
|
|
|
|
}
|
|
|
|
PathEl::QuadTo(p1, p2) => {
|
|
|
|
let scene_p1 = to_f32_2(p1);
|
|
|
|
let scene_p2 = to_f32_2(p2);
|
|
|
|
let seg = QuadSeg {
|
|
|
|
p0: last_pt.unwrap(),
|
|
|
|
p1: scene_p1,
|
|
|
|
p2: scene_p2,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_quad_seg(seg);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_p2);
|
|
|
|
}
|
|
|
|
PathEl::CurveTo(p1, p2, p3) => {
|
|
|
|
let scene_p1 = to_f32_2(p1);
|
|
|
|
let scene_p2 = to_f32_2(p2);
|
|
|
|
let scene_p3 = to_f32_2(p3);
|
|
|
|
let seg = CubicSeg {
|
|
|
|
p0: last_pt.unwrap(),
|
|
|
|
p1: scene_p1,
|
|
|
|
p2: scene_p2,
|
|
|
|
p3: scene_p3,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_cubic_seg(seg);
|
2020-05-12 13:01:06 +10:00
|
|
|
last_pt = Some(scene_p3);
|
|
|
|
}
|
|
|
|
PathEl::ClosePath => {
|
|
|
|
if let (Some(start), Some(last)) = (start_pt.take(), last_pt.take()) {
|
2020-05-21 09:36:09 +10:00
|
|
|
if last != start {
|
|
|
|
let seg = LineSeg {
|
|
|
|
p0: last,
|
|
|
|
p1: start,
|
|
|
|
};
|
2021-03-19 23:49:47 +11:00
|
|
|
self.encode_line_seg(seg);
|
2020-05-21 09:36:09 +10:00
|
|
|
}
|
2020-05-12 13:01:06 +10:00
|
|
|
}
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
|
|
|
}
|
2020-05-12 13:01:06 +10:00
|
|
|
//println!("{:?}", el);
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
|
|
|
}
|
2020-05-12 13:01:06 +10:00
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
|
|
|
|
fn pop_clip(&mut self) {
|
|
|
|
let tos = self.clip_stack.pop().unwrap();
|
2020-11-21 04:26:02 +11:00
|
|
|
let bbox = tos.bbox.unwrap_or_default();
|
|
|
|
let bbox_f32_4 = rect_to_f32_4(bbox);
|
|
|
|
self.elements
|
|
|
|
.push(Element::EndClip(Clip { bbox: bbox_f32_4 }));
|
2020-11-20 06:53:59 +11:00
|
|
|
self.path_count += 1;
|
2020-11-21 04:26:02 +11:00
|
|
|
if let Element::BeginClip(begin_clip) = &mut self.elements[tos.begin_ix] {
|
|
|
|
begin_clip.bbox = bbox_f32_4;
|
|
|
|
} else {
|
|
|
|
unreachable!("expected BeginClip, not found");
|
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
if let Some(bbox) = tos.bbox {
|
2020-11-21 04:26:02 +11:00
|
|
|
self.union_bbox(bbox);
|
2020-11-20 06:53:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 04:26:02 +11:00
|
|
|
/// Accumulate a bbox.
|
|
|
|
///
|
|
|
|
/// The bbox is given lazily as a closure, relative to the current transform.
|
|
|
|
/// It's lazy because we don't need to compute it unless we're inside a clip.
|
2020-11-20 06:53:59 +11:00
|
|
|
fn accumulate_bbox(&mut self, f: impl FnOnce() -> Rect) {
|
2020-11-21 04:26:02 +11:00
|
|
|
if !self.clip_stack.is_empty() {
|
2020-11-20 06:53:59 +11:00
|
|
|
let bbox = f();
|
2020-11-21 04:26:02 +11:00
|
|
|
let bbox = self.cur_transform.transform_rect_bbox(bbox);
|
|
|
|
self.union_bbox(bbox);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Accumulate an absolute bbox.
|
|
|
|
///
|
|
|
|
/// The bbox is given already transformed into surface coordinates.
|
|
|
|
fn union_bbox(&mut self, bbox: Rect) {
|
|
|
|
if let Some(tos) = self.clip_stack.last_mut() {
|
2020-11-20 06:53:59 +11:00
|
|
|
tos.bbox = if let Some(old_bbox) = tos.bbox {
|
|
|
|
Some(old_bbox.union(bbox))
|
|
|
|
} else {
|
|
|
|
Some(bbox)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
impl Text for PietGpuText {
|
|
|
|
type TextLayout = PietGpuTextLayout;
|
|
|
|
type TextLayoutBuilder = PietGpuTextLayoutBuilder;
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn load_font(&mut self, _data: &[u8]) -> Result<FontFamily, Error> {
|
|
|
|
Ok(FontFamily::default())
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn new_text_layout(&mut self, _text: impl TextStorage) -> Self::TextLayoutBuilder {
|
|
|
|
PietGpuTextLayoutBuilder
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn font_family(&mut self, _family_name: &str) -> Option<FontFamily> {
|
|
|
|
Some(FontFamily::default())
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TextLayoutBuilder for PietGpuTextLayoutBuilder {
|
|
|
|
type Out = PietGpuTextLayout;
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn max_width(self, _width: f64) -> Self {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn alignment(self, _alignment: piet::TextAlignment) -> Self {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn default_attribute(self, _attribute: impl Into<TextAttribute>) -> Self {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn range_attribute(
|
|
|
|
self,
|
|
|
|
_range: impl RangeBounds<usize>,
|
|
|
|
_attribute: impl Into<TextAttribute>,
|
|
|
|
) -> Self {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-04-23 03:28:38 +10:00
|
|
|
fn build(self) -> Result<Self::Out, Error> {
|
2020-11-15 03:22:56 +11:00
|
|
|
Ok(PietGpuTextLayout)
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TextLayout for PietGpuTextLayout {
|
2020-11-15 03:22:56 +11:00
|
|
|
fn size(&self) -> Size {
|
|
|
|
Size::ZERO
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn image_bounds(&self) -> Rect {
|
|
|
|
Rect::ZERO
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn line_text(&self, _line_number: usize) -> Option<&str> {
|
2020-11-15 03:22:56 +11:00
|
|
|
None
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn line_metric(&self, _line_number: usize) -> Option<LineMetric> {
|
2020-11-15 03:22:56 +11:00
|
|
|
None
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn line_count(&self) -> usize {
|
2020-11-15 03:22:56 +11:00
|
|
|
0
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn hit_test_point(&self, _point: Point) -> HitTestPoint {
|
2020-11-15 03:22:56 +11:00
|
|
|
HitTestPoint::default()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn hit_test_text_position(&self, _text_position: usize) -> HitTestPosition {
|
|
|
|
HitTestPosition::default()
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
|
2020-11-15 03:22:56 +11:00
|
|
|
fn text(&self) -> &str {
|
|
|
|
""
|
2020-04-23 03:28:38 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoBrush<PietGpuRenderContext> for PietGpuBrush {
|
|
|
|
fn make_brush<'b>(
|
|
|
|
&'b self,
|
|
|
|
_piet: &mut PietGpuRenderContext,
|
|
|
|
_bbox: impl FnOnce() -> Rect,
|
|
|
|
) -> std::borrow::Cow<'b, PietGpuBrush> {
|
|
|
|
Cow::Borrowed(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-12 13:01:06 +10:00
|
|
|
fn to_f32_2(point: Point) -> [f32; 2] {
|
|
|
|
[point.x as f32, point.y as f32]
|
2020-04-25 06:06:47 +10:00
|
|
|
}
|
2020-11-20 06:53:59 +11:00
|
|
|
|
|
|
|
fn rect_to_f32_4(rect: Rect) -> [f32; 4] {
|
2020-11-21 04:26:02 +11:00
|
|
|
[
|
|
|
|
rect.x0 as f32,
|
|
|
|
rect.y0 as f32,
|
|
|
|
rect.x1 as f32,
|
|
|
|
rect.y1 as f32,
|
|
|
|
]
|
2020-11-20 06:53:59 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn to_scene_transform(transform: Affine) -> Transform {
|
|
|
|
let c = transform.as_coeffs();
|
|
|
|
Transform {
|
|
|
|
mat: [c[0] as f32, c[1] as f32, c[2] as f32, c[3] as f32],
|
|
|
|
translate: [c[4] as f32, c[5] as f32],
|
|
|
|
}
|
|
|
|
}
|