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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
use prelude::*;
use core::{Texture, Color, TextureFilter, Rect, Point2};
use backends::backend;

pub const NO_FRAME_PREPARED: &'static str = "Failed to get frame: None prepared.";

/// A target for rendering.
pub trait AsRenderTarget {
    /// Returns a RenderTarget representing a texture or a frame.
    fn as_render_target(self: &Self) -> RenderTarget;
}

/// An opaque type representing rendering targets like Display or Texture.
#[derive(Clone)]
pub struct RenderTarget(pub(crate) RenderTargetInner);

impl RenderTarget {
    /// Creates a new frame rendertarget.
    pub(crate) fn frame(frame: &Rc<RefCell<Option<backend::Frame>>>) -> RenderTarget {
        RenderTarget(RenderTargetInner::Frame(frame.clone()))
    }
    /// Creates a new texture rendertarget.
    pub(crate) fn texture(texture: &Texture) -> RenderTarget{
        RenderTarget(RenderTargetInner::Texture(texture.clone()))
    }
    /// Creates a null rendertarget.
    pub fn none() -> RenderTarget{
        RenderTarget(RenderTargetInner::None)
    }
}

impl AsRenderTarget for RenderTarget {
    fn as_render_target(self: &Self) -> RenderTarget {
        self.clone()
    }
}

impl Debug for RenderTarget {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let inner = match &self.0 {
            &RenderTargetInner::None => "None".to_string(),
            &RenderTargetInner::Frame(_) => "Frame".to_string(),
            &RenderTargetInner::Texture(ref texture) => format!("{:?}", texture)
        };
        write!(f, "RenderTarget {{ {:?} }}", inner)
    }
}

/// An enum of render target type instances.
#[derive(Clone)]
pub enum RenderTargetInner {
    None,
    Frame(Rc<RefCell<Option<backend::Frame>>>),
    Texture(Texture),
}

impl RenderTargetInner {
    /// Clears the target.
    pub fn clear(self: &Self, color: Color) {
        match *self {
            RenderTargetInner::Frame(ref display) => {
                let mut frame = display.borrow_mut();
                frame.as_mut().expect(NO_FRAME_PREPARED).clear(color);
            },
            RenderTargetInner::Texture(ref texture) => {
                texture.clear(color);
            }
            RenderTargetInner::None => { }
        }
    }
    /// Returns the dimensions of the target.
    pub fn dimensions(self: &Self) -> Point2<u32> {
        match *self {
            RenderTargetInner::Frame(ref display) => {
                let mut frame = display.borrow_mut();
                frame.as_mut().expect(NO_FRAME_PREPARED).dimensions()
            },
            RenderTargetInner::Texture(ref texture) => {
                texture.dimensions()
            }
            RenderTargetInner::None => {
                (0, 0)
            }
        }
    }
    /// Blits a source rect to a rect on the target.
    pub fn blit_rect(self: &Self, source: &RenderTarget, source_rect: Rect<i32>, target_rect: Rect<i32>, filter: TextureFilter) {
        match *self {
            RenderTargetInner::Frame(ref target_display) => {
                match source.0 {
                    RenderTargetInner::Frame(_) => {
                        let mut frame = target_display.borrow_mut();
                        frame.as_mut().expect(NO_FRAME_PREPARED).copy_rect(source_rect, target_rect, filter);
                    },
                    RenderTargetInner::Texture(ref src_texture) => {
                        let mut frame = target_display.borrow_mut();
                        frame.as_mut().expect(NO_FRAME_PREPARED).copy_rect_from_texture(src_texture, source_rect, target_rect, filter);
                    }
                    RenderTargetInner::None => { }
                }
            },
            RenderTargetInner::Texture(ref target_texture) => {
                match source.0 {
                    RenderTargetInner::Frame(ref src_display) => {
                        let mut frame = src_display.borrow_mut();
                        target_texture.handle.copy_rect_from_frame(frame.as_mut().expect(NO_FRAME_PREPARED), source_rect, target_rect, filter);
                    },
                    RenderTargetInner::Texture(ref src_texture) => {
                        target_texture.handle.copy_rect_from(src_texture, source_rect, target_rect, filter);
                    }
                    RenderTargetInner::None => { }
                }
            }
            RenderTargetInner::None => { }
        }
    }
    /// Blits to the target.
    pub fn blit(self: &Self, source: &RenderTarget, filter: TextureFilter) {
        match *self {
            RenderTargetInner::Frame(ref target_display) => {
                match source.0 {
                    RenderTargetInner::Frame(_) => { /* blitting entire frame to entire frame makes no sense */ },
                    RenderTargetInner::Texture(ref src_texture) => {
                        let mut frame = target_display.borrow_mut();
                        frame.as_mut().expect(NO_FRAME_PREPARED).copy_from_texture(src_texture, filter);
                    }
                    RenderTargetInner::None => { }
                }
            },
            RenderTargetInner::Texture(ref target_texture) => {
                match source.0 {
                    RenderTargetInner::Frame(ref src_display) => {
                        let mut frame = src_display.borrow_mut();
                        target_texture.handle.copy_from_frame(frame.as_mut().expect(NO_FRAME_PREPARED), filter);
                    },
                    RenderTargetInner::Texture(ref src_texture) => {
                        target_texture.handle.copy_from(src_texture, filter);
                    }
                    RenderTargetInner::None => { }
                }
            }
            RenderTargetInner::None => { }
        }
    }
}