-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathbitmap.cpp
394 lines (335 loc) Β· 12.6 KB
/
bitmap.cpp
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
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
// Copyright (C) 2020-2021 Sami VΓ€isΓ€nen
// Copyright (C) 2020-2021 Ensisoft http://www.ensisoft.com
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
#include "config.h"
#include "warnpush.h"
# include <stb/stb_image_write.h>
#include "warnpop.h"
#include <algorithm>
#include <fstream>
#include <cmath>
#include <set>
#include "base/hash.h"
#include "base/math.h"
#include "data/writer.h"
#include "data/reader.h"
#include "graphics/bitmap.h"
namespace {
void PremultiplyPixel_lRGB(const gfx::Pixel_RGBA& src, gfx::Pixel_RGBA* dst)
{
gfx::Pixel_RGBAf norm;
norm = gfx::Pixel_to_floats(src);
norm = gfx::RGBA_premul_alpha(norm);
*dst = gfx::Pixel_to_uints(norm);
}
void PremultiplyPixel_sRGB(const gfx::Pixel_RGBA& src, gfx::Pixel_RGBA* dst)
{
gfx::Pixel_RGBAf norm;
norm = gfx::Pixel_to_floats(src);
norm = gfx::sRGB_decode(norm);
norm = gfx::RGBA_premul_alpha(norm);
norm = gfx::sRGB_encode(norm);
*dst = gfx::Pixel_to_uints(norm);
}
template<typename T_u8, typename T_float>
std::unique_ptr<gfx::Bitmap<T_u8>> ConvertToLinear(const gfx::IBitmapReadView& src)
{
ASSERT(src.IsValid());
const auto width = src.GetWidth();
const auto height = src.GetHeight();
using Bitmap = gfx::Bitmap<T_u8>;
auto ret = std::make_unique<Bitmap>(width, height);
auto dst = ret->GetWriteView();
for (unsigned row=0; row<height; ++row)
{
for (unsigned col=0; col<width; ++col)
{
T_u8 value;
T_float norm;
src.ReadPixel(row, col, &value);
norm = gfx::Pixel_to_floats(value);
norm = gfx::sRGB_decode(norm);
value = gfx::Pixel_to_uints(norm);
dst->WritePixel(row, col, value);
}
}
return ret;
}
template<typename T_u8, typename T_float, bool support_srgb, bool alpha_channel>
std::unique_ptr<gfx::Bitmap<T_u8>> BoxFilter(const gfx::IBitmapReadView& src, bool srgb, bool premul_alpha)
{
ASSERT(src.IsValid());
const auto src_width = src.GetWidth();
const auto src_height = src.GetHeight();
if (src_width == 1 && src_height == 1)
return nullptr;
const auto dst_width = std::max(1u, src_width / 2);
const auto dst_height = std::max(1u, src_height / 2);
const auto src_height_max = src_height & ~0x1;
const auto src_width_max = src_width & ~0x1;
using Bitmap = gfx::Bitmap<T_u8>;
auto ret = std::make_unique<Bitmap>(dst_width, dst_height);
auto dst = ret->GetWriteView();
for (unsigned dst_row=0, src_row=0; src_row<src_height_max; src_row+=2, dst_row++)
{
for (unsigned dst_col=0, src_col=0; src_col<src_width_max; src_col+=2, dst_col++)
{
// read 2x2 pixels from the source image
T_u8 values[4];
src.ReadPixel(std::min(src_row+0, src_height-1), std::min(src_col+0, src_width-1), &values[0]);
src.ReadPixel(std::min(src_row+0, src_height-1), std::min(src_col+1, src_width-1), &values[1]);
src.ReadPixel(std::min(src_row+1, src_height-1), std::min(src_col+0, src_width-1), &values[2]);
src.ReadPixel(std::min(src_row+1, src_height-1), std::min(src_col+1, src_width-1), &values[3]);
// normalize the values for better precision. alternative would
// be to use bit shift (right shift by two) on the integer type
// but that would lose some precision.
T_float values_norm[4];
values_norm[0] = gfx::Pixel_to_floats(values[0]);
values_norm[1] = gfx::Pixel_to_floats(values[1]);
values_norm[2] = gfx::Pixel_to_floats(values[2]);
values_norm[3] = gfx::Pixel_to_floats(values[3]);
// if we're dealing with sRGB color space then the values need first
// be converted into linear before averaging. this is only supported
// for RGBA and RGB formats.
if constexpr (support_srgb)
{
if (srgb)
{
values_norm[0] = gfx::sRGB_decode(values_norm[0]);
values_norm[1] = gfx::sRGB_decode(values_norm[1]);
values_norm[2] = gfx::sRGB_decode(values_norm[2]);
values_norm[3] = gfx::sRGB_decode(values_norm[3]);
}
}
// premultiply alpha into RGB
if constexpr (alpha_channel)
{
if (premul_alpha)
{
for (auto& rgba : values_norm)
{
rgba = gfx::RGBA_premul_alpha(rgba);
}
}
}
// compute the average of the original 4 pixels.
T_float value = values_norm[0] * 0.25f +
values_norm[1] * 0.25f +
values_norm[2] * 0.25f +
values_norm[3] * 0.25f;
// encode linear in sRGB if needed.
if constexpr (support_srgb)
{
if (srgb)
value = gfx::sRGB_encode(value);
}
// write the new pixel value out.
dst->WritePixel(dst_row, dst_col, gfx::Pixel_to_uints(value));
}
}
return ret;
}
} // namespace
namespace gfx
{
void WritePPM(const IBitmapReadView& bmp, const std::string& filename)
{
static_assert(sizeof(Pixel_RGB) == 3,
"Padding bytes found. Cannot copy Pixel_RGB data as a byte stream.");
std::ofstream out(filename, std::ios::binary);
if (!out.is_open())
throw std::runtime_error("failed to open file: " + filename);
const auto width = bmp.GetWidth();
const auto height = bmp.GetHeight();
const auto depth = bmp.GetDepthBits() / 8;
const auto bytes = width * height * sizeof(Pixel_RGB);
std::vector<Pixel_RGB> tmp;
tmp.reserve(width * height);
for (unsigned row=0; row<height; ++row)
{
for (unsigned col=0; col<width; ++col)
{
Pixel_RGB value;
bmp.ReadPixel(row, col, &value);
tmp.push_back(value);
}
}
out << "P6 " << width << " " << height << " 255\n";
out.write((const char*)&tmp[0], bytes);
out.close();
}
void WritePPM(const IBitmap& bmp, const std::string& filename)
{
auto view = bmp.GetReadView();
WritePPM(*view, filename);
}
void WritePNG(const IBitmapReadView& bmp, const std::string& filename)
{
const auto w = bmp.GetWidth();
const auto h = bmp.GetHeight();
const auto d = bmp.GetDepthBits() / 8;
if (!stbi_write_png(filename.c_str(), w, h, d, bmp.GetReadPtr(), d * w))
throw std::runtime_error(std::string("failed to write png: " + filename));
}
void WritePNG(const IBitmap& bmp, const std::string& filename)
{
auto view = bmp.GetReadView();
WritePNG(*view, filename);
}
std::unique_ptr<IBitmap> GenerateNextMipmap(const IBitmapReadView& src, bool srgb)
{
const auto premul_alpha = false;
if (src.GetDepthBits() == 32) {
return ::BoxFilter<Pixel_RGBA, Pixel_RGBAf, true, true>(src, srgb, premul_alpha);
} else if (src.GetDepthBits() == 24) {
return ::BoxFilter<Pixel_RGB, Pixel_RGBf, true, false> (src, srgb, false);
} else if (src.GetDepthBits() == 8)
return ::BoxFilter<Pixel_A, Pixel_Af, false, false>(src, false, false);
return nullptr;
}
std::unique_ptr<IBitmap> GenerateNextMipmap(const IBitmap& src, bool srgb)
{
auto view = src.GetReadView();
return GenerateNextMipmap(*view, srgb);
}
std::unique_ptr<IBitmap> ConvertToLinear(const IBitmapReadView& src)
{
if (src.GetDepthBits() == 32)
return ::ConvertToLinear < Pixel_RGBA, Pixel_RGBAf > (src);
else if (src.GetDepthBits() == 24)
return ::ConvertToLinear < Pixel_RGB, Pixel_RGBf > (src);
return nullptr;
}
std::unique_ptr<IBitmap> ConvertToLinear(const IBitmap& src)
{
auto view = src.GetReadView();
return ConvertToLinear(*view);
}
void PremultiplyAlpha(const BitmapWriteView<Pixel_RGBA>& dst,
const BitmapReadView<Pixel_RGBA>& src, bool srgb)
{
auto conversion = srgb ? &PremultiplyPixel_sRGB
: &PremultiplyPixel_lRGB;
ConvertBitmap(dst, src, conversion);
}
Bitmap<Pixel_RGBA> PremultiplyAlpha(const BitmapReadView<Pixel_RGBA>& src, bool srgb)
{
const auto src_width = src.GetWidth();
const auto src_height = src.GetHeight();
Bitmap<Pixel_RGBA> ret;
ret.Resize(src_width, src_height);
PremultiplyAlpha(ret.GetPixelWriteView(), src, srgb);
return ret;
}
Bitmap<Pixel_RGBA> PremultiplyAlpha(const Bitmap<Pixel_RGBA>& src, bool srgb)
{
return PremultiplyAlpha(src.GetPixelReadView(), srgb);
}
URect FindImageRectangle(const IBitmapReadView& img, const IPoint& start)
{
URect ret;
if (img.GetDepthBits() != 32)
return ret;
using BitmapReadViewRGBA = BitmapReadView<Pixel_RGBA>;
const auto* rgba_view = dynamic_cast<const BitmapReadViewRGBA*>(&img);
ASSERT(rgba_view);
// explorer all adjacent pixels and their adjacent pixels etc
// with a breadth first search until the alpha value is 0.0.
struct Compare {
bool operator ()(const IPoint& lhs, const IPoint& rhs) const
{
if (lhs.GetX() < rhs.GetX())
return true;
else if (lhs.GetX() == rhs.GetX())
if (lhs.GetY() < rhs.GetY())
return true;
return false;
}
};
auto ReadPixel = [](const BitmapReadViewRGBA& img, const IPoint& point, Pixel_RGBA* pixel) {
const auto w = img.GetWidth();
const auto h = img.GetHeight();
if (point.GetX() >= w || point.GetX() < 0)
return false;
else if (point.GetY() >= h || point.GetY() < 0)
return false;
img.ReadPixel(point.GetY(), point.GetX(), pixel);
return true;
};
Pixel_RGBA pixel;
std::set<IPoint, Compare> pixels_visited;
std::set<IPoint, Compare> pixels_to_explore;
pixels_to_explore.insert(start);
int min_x = std::numeric_limits<int>::max();
int min_y = std::numeric_limits<int>::max();
int max_x = 0;
int max_y = 0;
struct PixelOffset {
int x, y;
};
static PixelOffset neighbors[] = {
{-1, 0}, // left
{-1, -1}, // left up
{0, -1}, // up
{1, -1}, // up right
{1, 0}, // right
{1, 1}, // right down
{0, 1}, // down
{-1, 1} // down left
};
while (!pixels_to_explore.empty())
{
auto beg = pixels_to_explore.begin();
auto this_pixel = *beg;
pixels_to_explore.erase(beg);
if (!ReadPixel(*rgba_view, this_pixel, &pixel))
continue;
pixels_visited.insert(this_pixel);
if (pixel.a == 0)
continue;
min_x = std::min(min_x, this_pixel.GetX());
max_x = std::max(max_x, this_pixel.GetX());
min_y = std::min(min_y, this_pixel.GetY());
max_y = std::max(max_y, this_pixel.GetY());
for (size_t i=0; i<8; ++i)
{
const auto neighbor = neighbors[i];
const auto neighbor_pixel = this_pixel.TranslateCopy(neighbor.x, neighbor.y);
if (auto it = pixels_visited.find(neighbor_pixel); it != pixels_visited.end())
continue;
if (!ReadPixel(*rgba_view, neighbor_pixel, &pixel) || pixel.a == 0)
continue;
pixels_to_explore.insert(neighbor_pixel);
}
}
if (min_x == std::numeric_limits<int>::max() || min_y == std::numeric_limits<int>::max())
return ret;
const auto found_width = max_x - min_x + 1;
const auto found_height = max_y - min_y + 1;
ASSERT(min_x >= 0 && (min_x + found_width <= img.GetWidth()));
ASSERT(min_y >= 0 && (min_y + found_height <= img.GetHeight()));
ret.SetX(min_x);
ret.SetY(min_y);
ret.SetWidth(found_width);
ret.SetHeight(found_height);
return ret;
}
void SetAlphaToOne(RgbaBitmap& bitmap)
{
const auto count = bitmap.GetWidth() * bitmap.GetHeight();
for (unsigned index=0; index<count; ++index)
bitmap.GetPixel(index).a = 255;
}
} // namespace