1 module gfm.sdl2.surface;
2 
3 import derelict.sdl2.sdl;
4 
5 import gfm.sdl2.sdl;
6 
7 /// SDL Surface wrapper.
8 /// A SDL2Surface might own the SDL_Surface* handle or not.
9 final class SDL2Surface
10 {
11     public
12     {
13         /// Whether a SDL Surface is owned by the wrapper or borrowed.
14         enum Owned
15         {
16             NO,  // Not owned.
17             YES  // Owned.
18         }
19 
20         /// Create from an existing SDL_Surface* handle.
21         this(SDL2 sdl2, SDL_Surface* surface, Owned owned)
22         {
23             assert(surface !is null);
24             _sdl2 = sdl2;
25             _surface = surface;
26             _handleOwned = owned;
27         }
28 
29         /// Create a new RGBA surface. Both pixels data and handle are owned.
30         /// See_also: $(LINK http://wiki.libsdl.org/SDL_CreateRGBSurface)
31         /// Throws: $(D SDL2Exception) on error.
32         this(SDL2 sdl2, int width, int height, int depth,
33              uint Rmask, uint Gmask, uint Bmask, uint Amask)
34         {
35             _sdl2 = sdl2;
36             _surface = SDL_CreateRGBSurface(0, width, height, depth, Rmask, Gmask, Bmask, Amask);
37             if (_surface is null)
38                 _sdl2.throwSDL2Exception("SDL_CreateRGBSurface");
39             _handleOwned = Owned.YES;
40         }
41 
42         /// Create surface from RGBA data. Pixels data is <b>not</b> and not owned.
43         /// See_also: clone, $(WEB wiki.libsdl.org/SDL_CreateRGBSurfaceFrom,SDL_CreateRGBSurfaceFrom)
44         /// Throws: $(D SDL2Exception) on error.
45         this(SDL2 sdl2, void* pixels, int width, int height, int depth, int pitch,
46              uint Rmask, uint Gmask, uint Bmask, uint Amask)
47         {
48             _sdl2 = sdl2;
49             _surface = SDL_CreateRGBSurfaceFrom(pixels, width, height, depth, pitch, Rmask, Gmask, Bmask, Amask);
50             if (_surface is null)
51                 _sdl2.throwSDL2Exception("SDL_CreateRGBSurfaceFrom");
52             _handleOwned = Owned.YES;
53         }
54 
55         /// Releases the SDL resource.
56         ~this()
57         {
58             if (_surface !is null)
59             {
60                 debug ensureNotInGC("SDL2Surface");
61                 if (_handleOwned == Owned.YES)
62                     SDL_FreeSurface(_surface);
63                 _surface = null;
64             }
65         }
66         deprecated("Use .destroy instead") void close(){}
67 
68         /// Converts the surface to another format.
69         /// See_also: $(LINK http://wiki.libsdl.org/SDL_ConvertSurface)
70         /// Returns: A new surface.
71         SDL2Surface convert(const(SDL_PixelFormat)* newFormat)
72         {
73             SDL_Surface* surface = SDL_ConvertSurface(_surface, newFormat, 0);
74             if (surface is null)
75                 _sdl2.throwSDL2Exception("SDL_ConvertSurface");
76             assert(surface != _surface); // should not be the same handle
77             return new SDL2Surface(_sdl2, surface, Owned.YES);
78         }
79 
80         /// Returns: A copy of the surface, useful for taking ownership of not-owned pixel data.
81         /// See_also: $WEB(wiki.libsdl.org/SDL_CreateRGBSurfaceFrom,SDL_CreateRGBSurfaceFrom)
82         SDL2Surface clone()
83         {
84             return convert(pixelFormat());
85         }
86 
87         /// Returns: Width of the surface in pixels.
88         @property int width() const
89         {
90             return _surface.w;
91         }
92 
93         /// Returns: Height of the surface in pixels.
94         @property int height() const
95         {
96             return _surface.h;
97         }
98 
99         /// Returns: Pointer to surface data.
100         /// You must lock the surface before accessng it.
101         ubyte* pixels()
102         {
103             return cast(ubyte*) _surface.pixels;
104         }
105 
106         /// Get the surface pitch (number of bytes between lines).
107         size_t pitch()
108         {
109             return _surface.pitch;
110         }
111 
112         /// Lock the surface, allow to use pixels().
113         /// Throws: $(D SDL2Exception) on error.
114         void lock()
115         {
116             if (SDL_LockSurface(_surface) != 0)
117                 _sdl2.throwSDL2Exception("SDL_LockSurface");
118         }
119 
120         /// Unlock the surface.
121         void unlock()
122         {
123             SDL_UnlockSurface(_surface);
124         }
125 
126         /// Returns: SDL handle.
127         SDL_Surface* handle()
128         {
129             return _surface;
130         }
131 
132         /// Returns: SDL_PixelFormat which describe the surface.
133         SDL_PixelFormat* pixelFormat()
134         {
135             return _surface.format;
136         }
137 
138         ///
139         struct RGBA
140         {
141             ubyte r, g, b, a;
142         }
143 
144         /// Get a surface pixel color.
145         /// Bugs: must be locked when using this method. Slow!
146         RGBA getRGBA(int x, int y)
147         {
148             // crash if out of image
149             if (x < 0 || x >= width())
150                 assert(0);
151 
152             if (y < 0 || y >= height())
153                 assert(0);
154 
155             SDL_PixelFormat* fmt = _surface.format;
156 
157             ubyte* pixels = cast(ubyte*)_surface.pixels;
158             int pitch = _surface.pitch;
159 
160             uint* pixel = cast(uint*)(pixels + y * pitch + x * fmt.BytesPerPixel);
161             ubyte r, g, b, a;
162             SDL_GetRGBA(*pixel, fmt, &r, &g, &b, &a);
163             return RGBA(r, g, b, a);
164         }
165     }
166 
167     package
168     {
169         SDL2 _sdl2;
170         SDL_Surface* _surface;
171         Owned _handleOwned;
172     }
173 }