-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathgeneric-layer.c
263 lines (241 loc) · 6.36 KB
/
generic-layer.c
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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
/*
* SPDX-FileCopyrightText: 2021 Samuel Cuella <[email protected]>
*
* This file is part of SoFIS - an open source EFIS
*
* SPDX-License-Identifier: GPL-2.0-only
*/
#include <stdio.h>
#include <stdlib.h>
#include <SDL2/SDL.h>
#include <SDL2/SDL_image.h>
#include "generic-layer.h"
#include "SDL_gpu.h"
/**
* GenericLayer: A class to manage drawing areas (SDL_Surfaces)
* that will be drawn on (or loaded from file) and then be used as-is
* without further modification and optionaly turned into textures.
*
* This is intended to be used for anything that must be blit to the
* screen each frame.
*
*/
/**
* @brief Creates a new GenericLayer of given size.
*
* The GenericLayer returned must be freed by the calling code.
*
* @param width canvas width
* @param height canvas width
* @return a newly allocated GenericLayer on success, NULL on failure.
*
* @see generic_layer_free
*/
GenericLayer *generic_layer_new(int width, int height)
{
GenericLayer *self;
self = calloc(1, sizeof(GenericLayer));
if(self){
if(!generic_layer_init(self, width, height)){
generic_layer_dispose(self);
return NULL;
}
}
return self;
}
/**
* @brief Creates a new GenericLayer from an existing image. Loading is
* done through SDL_Image, refer to their documention for format support.
*
* The GenericLayer returned must be freed by the calling code.
*
* @param filaname The image to load
* @return a newly allocated GenericLayer on success, NULL on failure.
*
* @see generic_layer_free
*/
GenericLayer *generic_layer_new_from_file(const char *filename)
{
GenericLayer *self;
self = calloc(1, sizeof(GenericLayer));
if(self){
if(!generic_layer_init_from_file(self, filename)){
generic_layer_dispose(self);
return NULL;
}
}
return self;
}
/**
* @brief Creates the underlying canvas (SDL_Surface)
*
* After calling this function successfuly, the canvas (self->canvas)
* can be accessed and drawed on.
*
* @param self a GenericLayer
* @param width the width of the canvas
* @param height the height of the canvas
* @return true on success, false otherwise. If the function returns
* false, the canvas is not suitable for use/access.
*
*/
bool generic_layer_init(GenericLayer *self, int width, int height)
{
self->canvas = SDL_CreateRGBSurfaceWithFormat(
0,
width, height,
32, SDL_PIXELFORMAT_RGBA32
);
/*TODO: Is this useful? Enclosing object would have memeset'ed itself*/
#if USE_SDL_GPU
self->texture = NULL;
#endif
return self->canvas != NULL;
}
/**
* @brief Creates the underlying canvas (SDL_Surface) with the specified mask.
*
* After calling this function successfuly, the canvas (self->canvas) can be
* accessed and drawed on.
*
* @note All GenericLayer surfaces will/must have a depth of 32 bits per pixel.
*
* @param self a GenericLayer
* @param width the width of the canvas
* @param height the height of the canvas
* @param Rmask the red mask for the pixels
* @param Gmask the green mask for the pixels
* @param Bmask the blue mask for the pixels
* @param Amask the alpha mask for the pixels
* @return true on success, false otherwise. If the function returns
* false, the canvas is not suitable for use/access.
*/
bool generic_layer_init_with_masks(GenericLayer *self, int width, int height, Uint32 Rmask, Uint32 Gmask, Uint32 Bmask, Uint32 Amask)
{
self->canvas = SDL_CreateRGBSurface(
0,
width, height,
32,
Rmask, Gmask, Bmask, Amask
);
/*TODO: Is this useful? Enclosing object would have memeset'ed itself*/
#if USE_SDL_GPU
self->texture = NULL;
#endif
return self->canvas != NULL;
}
/**
* @brief Release the resources held by self.
*
* @param self a GenericLayer
*/
void generic_layer_dispose(GenericLayer *self)
{
if(self->canvas)
SDL_FreeSurface(self->canvas);
#if USE_SDL_GPU
if(self->texture)
GPU_FreeImage(self->texture);
#endif
}
/**
* @brief Release any resource held by and free
* the memory used by @p self.
*
* @param self a GenericLayer
*/
void generic_layer_free(GenericLayer *self)
{
generic_layer_dispose(self);
free(self);
}
/**
* @brief Prevent others to remove @p self from under
* your feets.
*
* Match call with generic_layer_unref when you're done.
*
* @param self a GenericLayer
*
* @see generic_layer_unref
*/
/*TODO: Use inderlying refcounts from SDL_Surface/GPU_Image?*/
/*TODO: inline*/
void generic_layer_ref(GenericLayer *self)
{
self->refcount++;
}
/**
* @brief Matching call to generic_layer_ref. Frees the memory
* when refcount reaches 0
*
* @param self a GenericLayer
*
* @see generic_layer_ref
*
*/
void generic_layer_unref(GenericLayer *self)
{
if(--self->refcount == 0){
generic_layer_free(self);
}
}
/**
* @brief Loads a file into a newly-created/uninited GenericLayer.
*
* Supported formats are those supported by SDL_Image which does the loading.
* @p self is assumed to be non-inited: No checks are made, no resources
* are freed.
*
* @param self a GenericLayer
* @param filename The file to read from.
* @return true on success, false otherwise. The error - as set by SDL_Image -
* can be retrieved through SDL_GetError.
*/
bool generic_layer_init_from_file(GenericLayer *self, const char *filename)
{
self->canvas = IMG_Load(filename);
#if USE_SDL_GPU
self->texture = NULL;
#endif
return self->canvas != NULL;
}
/**
* @brief Creates a texture from the canvas.
*
* @param self a GenericLayer
* @return true on success, false otherwise.
*
* @note If no texture support is built, this
* funtion will always return true.
* TODO: Get rid of the surface ?
*/
bool generic_layer_build_texture(GenericLayer *self)
{
#if USE_SDL_GPU
self->texture = GPU_CopyImageFromSurface(self->canvas);
return self->texture != NULL;
#else
return true;
#endif
}
/**
* @brief Updates the texture from the content of the canvas.
*
* Must be called when you want any modifications done to the
* surface done after the call to generic_layer_build_texture
* to be visible by the GPU.
*
* @param self a GenericLayer
*
* @see generic_layer_build_texture
*/
void generic_layer_update_texture(GenericLayer *self)
{
#if USE_SDL_GPU
if(self->texture)
GPU_UpdateImage(self->texture, NULL, self->canvas, NULL);
else
generic_layer_build_texture(self);
#endif
}