summaryrefslogtreecommitdiffstats
path: root/wgpu/src/primitive/pipeline.rs
blob: 302e38f68fc9d56656a32f0fec9803dd22d8ccea (plain) (blame)
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
106
107
108
109
110
111
112
113
114
115
116
//! Draw primitives using custom pipelines.
use crate::core::{Rectangle, Size};

use std::any::{Any, TypeId};
use std::collections::HashMap;
use std::fmt::Debug;
use std::sync::Arc;

#[derive(Clone, Debug)]
/// A custom primitive which can be used to render primitives associated with a custom pipeline.
pub struct Pipeline {
    /// The bounds of the [`Pipeline`].
    pub bounds: Rectangle,

    /// The [`Primitive`] to render.
    pub primitive: Arc<dyn Primitive>,
}

impl Pipeline {
    /// Creates a new [`Pipeline`] with the given [`Primitive`].
    pub fn new(bounds: Rectangle, primitive: impl Primitive) -> Self {
        Pipeline {
            bounds,
            primitive: Arc::new(primitive),
        }
    }
}

impl PartialEq for Pipeline {
    fn eq(&self, other: &Self) -> bool {
        self.primitive.type_id() == other.primitive.type_id()
    }
}

/// A set of methods which allows a [`Primitive`] to be rendered.
pub trait Primitive: Debug + Send + Sync + 'static {
    /// Processes the [`Primitive`], allowing for GPU buffer allocation.
    fn prepare(
        &self,
        format: wgpu::TextureFormat,
        device: &wgpu::Device,
        queue: &wgpu::Queue,
        bounds: Rectangle,
        target_size: Size<u32>,
        scale_factor: f32,
        storage: &mut Storage,
    );

    /// Renders the [`Primitive`].
    fn render(
        &self,
        storage: &Storage,
        target: &wgpu::TextureView,
        target_size: Size<u32>,
        viewport: Rectangle<u32>,
        encoder: &mut wgpu::CommandEncoder,
    );
}

/// A renderer than can draw custom pipeline primitives.
pub trait Renderer: crate::core::Renderer {
    /// Draws a custom pipeline primitive.
    fn draw_pipeline_primitive(
        &mut self,
        bounds: Rectangle,
        primitive: impl Primitive,
    );
}

impl<Theme> Renderer for crate::Renderer<Theme> {
    fn draw_pipeline_primitive(
        &mut self,
        bounds: Rectangle,
        primitive: impl Primitive,
    ) {
        self.draw_primitive(super::Primitive::Custom(super::Custom::Pipeline(
            Pipeline::new(bounds, primitive),
        )));
    }
}

/// Stores custom, user-provided pipelines.
#[derive(Default, Debug)]
pub struct Storage {
    pipelines: HashMap<TypeId, Box<dyn Any>>,
}

impl Storage {
    /// Returns `true` if `Storage` contains a pipeline with type `T`.
    pub fn has<T: 'static>(&self) -> bool {
        self.pipelines.get(&TypeId::of::<T>()).is_some()
    }

    /// Inserts the pipeline `T` in to [`Storage`].
    pub fn store<T: 'static>(&mut self, pipeline: T) {
        let _ = self.pipelines.insert(TypeId::of::<T>(), Box::new(pipeline));
    }

    /// Returns a reference to pipeline with type `T` if it exists in [`Storage`].
    pub fn get<T: 'static>(&self) -> Option<&T> {
        self.pipelines.get(&TypeId::of::<T>()).map(|pipeline| {
            pipeline
                .downcast_ref::<T>()
                .expect("Pipeline with this type does not exist in Storage.")
        })
    }

    /// Returns a mutable reference to pipeline `T` if it exists in [`Storage`].
    pub fn get_mut<T: 'static>(&mut self) -> Option<&mut T> {
        self.pipelines.get_mut(&TypeId::of::<T>()).map(|pipeline| {
            pipeline
                .downcast_mut::<T>()
                .expect("Pipeline with this type does not exist in Storage.")
        })
    }
}