1use std::fmt::Debug;
6use std::hash::Hash;
7use std::io::Read;
8use std::ops::Add;
9use std::{error, fmt, u32};
10
11use anyhow::Error;
12use display_utils::PixelFormat;
13use euclid::default::{Point2D, Rect, Size2D, Transform2D, Vector2D};
14use euclid::{point2, size2};
15use fidl::endpoints::ClientEnd;
16use fidl_fuchsia_sysmem2::BufferCollectionTokenMarker;
17
18use crate::color::Color;
19use crate::drawing::DisplayRotation;
20use crate::{Point, ViewAssistantContext};
21
22pub mod forma;
23
24pub use self::forma::Forma;
25
26pub trait Backend: Copy + Debug + Default + Eq + Hash + Ord + Sized + 'static {
28 type Image: Copy + Debug + Eq + Hash + Ord;
30 type Context: Context<Self>;
32 type Path: Clone + Eq;
34 type PathBuilder: PathBuilder<Self>;
36 type Raster: Raster;
38 type RasterBuilder: RasterBuilder<Self>;
40 type Composition: Composition<Self>;
42
43 fn new_context(
45 token: ClientEnd<BufferCollectionTokenMarker>,
46 size: Size2D<u32>,
47 display_rotation: DisplayRotation,
48 ) -> Self::Context;
49}
50
51#[derive(Clone, Copy, Debug, PartialEq)]
53pub struct CopyRegion {
54 pub src_offset: Point2D<u32>,
56 pub dst_offset: Point2D<u32>,
58 pub extent: Size2D<u32>,
60}
61
62#[derive(Clone, Debug, Default, PartialEq)]
64pub struct RenderExt<B: Backend> {
65 pub pre_clear: Option<PreClear>,
67 pub pre_copy: Option<PreCopy<B>>,
69 pub post_copy: Option<PostCopy<B>>,
71}
72
73#[derive(Clone, Debug, PartialEq)]
75pub struct PreClear {
76 pub color: Color,
78}
79
80#[derive(Clone, Debug, PartialEq)]
82pub struct PreCopy<B: Backend> {
83 pub image: B::Image,
85 pub copy_region: CopyRegion,
87}
88
89#[derive(Clone, Debug, PartialEq)]
91pub struct PostCopy<B: Backend> {
92 pub image: B::Image,
94 pub copy_region: CopyRegion,
96}
97
98pub trait Context<B: Backend> {
100 fn pixel_format(&self) -> PixelFormat;
102 fn path_builder(&self) -> Option<B::PathBuilder>;
104 fn raster_builder(&self) -> Option<B::RasterBuilder>;
106 fn new_image(&mut self, size: Size2D<u32>) -> B::Image;
108 fn new_image_from_png<R: Read>(
110 &mut self,
111 reader: &mut png::Reader<R>,
112 ) -> Result<B::Image, Error>;
113 fn get_image(&mut self, image_index: u32) -> B::Image;
115 fn get_current_image(&mut self, context: &ViewAssistantContext) -> B::Image;
117 fn render(
119 &mut self,
120 composition: &mut B::Composition,
121 clip: Option<Rect<u32>>,
122 image: B::Image,
123 ext: &RenderExt<B>,
124 ) {
125 self.render_with_clip(
126 composition,
127 clip.unwrap_or_else(|| {
128 Rect::new(point2(u32::MIN, u32::MIN), size2(u32::MAX, u32::MAX))
129 }),
130 image,
131 ext,
132 );
133 }
134 fn render_with_clip(
136 &mut self,
137 composition: &mut B::Composition,
138 clip: Rect<u32>,
139 image: B::Image,
140 ext: &RenderExt<B>,
141 );
142}
143
144pub trait PathBuilder<B: Backend> {
146 fn move_to(&mut self, point: Point) -> &mut Self;
148 fn line_to(&mut self, point: Point) -> &mut Self;
150 fn quad_to(&mut self, p1: Point, p2: Point) -> &mut Self;
152 fn cubic_to(&mut self, p1: Point, p2: Point, p3: Point) -> &mut Self;
154 fn rat_quad_to(&mut self, p1: Point, p2: Point, w: f32) -> &mut Self;
157 fn rat_cubic_to(&mut self, p1: Point, p2: Point, p3: Point, w1: f32, w2: f32) -> &mut Self;
160 fn build(self) -> B::Path;
164}
165
166pub trait Raster: Add<Output = Self> + Clone + Eq {
168 fn translate(self, translation: Vector2D<i32>) -> Self;
170}
171
172pub trait RasterBuilder<B: Backend> {
174 fn add(&mut self, path: &B::Path, transform: Option<&Transform2D<f32>>) -> &mut Self {
176 #[allow(clippy::or_fun_call)] self.add_with_transform(path, transform.unwrap_or(&Transform2D::identity()))
178 }
179 fn add_with_transform(&mut self, path: &B::Path, transform: &Transform2D<f32>) -> &mut Self;
181 fn build(self) -> B::Raster;
185}
186
187#[derive(Clone, Copy, Debug, PartialEq)]
191pub enum FillRule {
192 NonZero,
194 EvenOdd,
196}
197
198#[derive(Clone, Copy, Debug)]
199pub enum GradientType {
200 Linear,
201 Radial,
202}
203
204#[derive(Clone, Debug)]
205pub struct Gradient {
206 pub r#type: GradientType,
207 pub start: Point,
208 pub end: Point,
209 pub stops: Vec<(Color, f32)>,
210}
211
212#[derive(Clone, Debug)]
214pub enum Fill {
215 Solid(Color),
217 Gradient(Gradient),
219}
220
221#[derive(Clone, Copy, Debug, PartialEq)]
223pub enum BlendMode {
224 Over,
226 Multiply,
228 Screen,
231 Overlay,
233 Darken,
235 Lighten,
237 ColorDodge,
240 ColorBurn,
243 HardLight,
246 SoftLight,
249 Difference,
251 Exclusion,
254 Hue,
256 Saturation,
259 Color,
262 Luminosity,
265}
266
267#[derive(Clone, Debug)]
269pub struct Style {
270 pub fill_rule: FillRule,
272 pub fill: Fill,
274 pub blend_mode: BlendMode,
276}
277
278#[derive(Debug, PartialEq)]
279pub enum OrderError {
280 ExceededLayerLimit(u32),
281}
282
283impl fmt::Display for OrderError {
284 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
285 write!(
286 f,
287 "exceeded layer limit ({})",
288 match self {
289 OrderError::ExceededLayerLimit(val) => val,
290 }
291 )
292 }
293}
294
295const CARNELIAN_MAX_ORDER: usize = (1 << 18) - 1;
296
297pub type Order = GenericOrder<CARNELIAN_MAX_ORDER>;
298
299type GuaranteedOrderType = u16;
300static_assertions::const_assert!((GuaranteedOrderType::MAX as u32) < (Order::MAX.as_u32()));
301
302impl error::Error for OrderError {}
303
304#[derive(Clone, Copy, Debug, Default, Eq, Hash, Ord, PartialEq, PartialOrd)]
305pub struct GenericOrder<const MAX: usize>(u32);
306
307impl<const MAX: usize> GenericOrder<MAX> {
308 pub const MAX: Self = Self(MAX as u32);
309
310 pub const fn as_u32(&self) -> u32 {
311 self.0
312 }
313
314 pub const fn new(order: u32) -> Result<Self, OrderError> {
315 if order > Self::MAX.as_u32() {
316 Err(OrderError::ExceededLayerLimit(Self::MAX.as_u32()))
317 } else {
318 Ok(Self(order))
319 }
320 }
321
322 pub const fn from_u16(order: GuaranteedOrderType) -> Self {
326 Self(order as u32)
327 }
328}
329
330impl<const MAX: usize> TryFrom<u32> for GenericOrder<MAX> {
331 type Error = OrderError;
332
333 fn try_from(order: u32) -> Result<Self, OrderError> {
334 Self::new(order)
335 }
336}
337
338impl<const MAX: usize> TryFrom<usize> for GenericOrder<MAX> {
339 type Error = OrderError;
340
341 fn try_from(order: usize) -> Result<Self, OrderError> {
342 u32::try_from(order)
343 .map_err(|_| OrderError::ExceededLayerLimit(Self::MAX.as_u32()))
344 .and_then(|x| Self::try_from(x))
345 }
346}
347
348#[derive(Clone, Debug)]
350pub struct Layer<B: Backend> {
351 pub raster: B::Raster,
353 pub clip: Option<B::Raster>,
355 pub style: Style, }
358
359pub trait Composition<B: Backend> {
361 fn new(background_color: Color) -> Self;
363 fn clear(&mut self);
365 fn insert(&mut self, order: Order, layer: Layer<B>);
367 fn remove(&mut self, order: Order);
369}
370
371#[cfg(test)]
372pub(crate) mod tests {
373 use super::*;
374
375 use fuchsia_async as fasync;
376
377 #[test]
378 fn generic_compile_test() {
379 fn _generic<B: Backend>(
380 token: ClientEnd<BufferCollectionTokenMarker>,
381 size: Size2D<u32>,
382 display_rotation: DisplayRotation,
383 view_context: &ViewAssistantContext,
384 ) {
385 let mut context = B::new_context(token, size, display_rotation);
386
387 let mut path_builder = context.path_builder().unwrap();
388 path_builder.move_to(point2(0.0, 0.0)).line_to(point2(1.0, 1.0));
389 let path = path_builder.build();
390
391 let mut raster_builder = context.raster_builder().unwrap();
392 raster_builder.add(&path, None);
393 let raster = raster_builder.build();
394
395 let src_image = context.new_image(size2(100, 100));
396 let dst_image = context.get_current_image(view_context);
397
398 let mut composition: B::Composition = Composition::new(Color::new());
399 composition.insert(
400 Order::default(),
401 Layer {
402 raster: raster.clone() + raster,
403 clip: None,
404 style: Style {
405 fill_rule: FillRule::NonZero,
406 fill: Fill::Solid(Color::white()),
407 blend_mode: BlendMode::Over,
408 },
409 },
410 );
411
412 context.render(
413 &mut composition,
414 None,
415 dst_image,
416 &RenderExt {
417 pre_copy: Some(PreCopy {
418 image: src_image,
419 copy_region: CopyRegion {
420 src_offset: Point2D::zero(),
421 dst_offset: Point2D::zero(),
422 extent: size2(100, 100),
423 },
424 }),
425 ..Default::default()
426 },
427 );
428 }
429 }
430
431 fn has_vk_instance() -> bool {
435 false
436 }
437
438 pub(crate) fn run(f: impl Fn()) {
439 if !has_vk_instance() {
440 return;
441 }
442
443 let mut executor = fasync::LocalExecutor::new();
444 let f = async { f() };
445 executor.run_singlethreaded(f);
446 }
447}