TDME2 1.9.121
TextureReader.cpp
Go to the documentation of this file.
2
3#include <stdlib.h>
4
5#include <string>
6#include <vector>
7
8#include <tdme/tdme.h>
10
13#include <tdme/math/Vector2.h>
20
21#include <ext/libpng/png.h>
22
23using std::string;
24using std::to_string;
25using std::vector;
26
30
39
40vector<string> TextureReader::extensions = {"png"};
41Mutex* TextureReader::textureCacheMutex = new Mutex("texturecache-mutex");
42map<string, Texture*>* TextureReader::textureCache = new map<string, Texture*>();
43
44const vector<string>& TextureReader::getTextureExtensions() {
45 return extensions;
46}
47
48Texture* TextureReader::read(const string& pathName, const string& fileName, bool useCache, bool powerOfTwo, const string& idPrefix)
49{
50 Texture* texture = nullptr;
51
52 // make canonical
53 auto canonicalFilePath = FileSystem::getInstance()->getCanonicalPath(pathName, fileName);
54 auto canonicalPathName = FileSystem::getInstance()->getPathName(canonicalFilePath);
55 auto canonicalFileName = FileSystem::getInstance()->getFileName(canonicalFilePath);
56
57 // do cache look up
58 if (useCache == true) {
60 auto textureCacheIt = textureCache->find(idPrefix + canonicalPathName + "/" + canonicalFileName);
61 if (textureCacheIt != textureCache->end()) {
62 texture = textureCacheIt->second;
63 }
64 }
65
66 // have texture?
67 if (texture == nullptr) {
68 // nope try to load
69 try {
70 if (StringTools::endsWith(StringTools::toLowerCase(canonicalFileName), ".png") == true) {
71
72 // create PNG input stream
73 vector<uint8_t> data;
74 FileSystem::getInstance()->getContent(pathName, fileName, data);
75
76 texture = TextureReader::readPNG(canonicalFilePath, data, powerOfTwo, idPrefix);
77 if (texture != nullptr && useCache == true) {
78 (*textureCache)[texture->getId()] = texture;
79 }
80 }
81 } catch (Exception& exception) {
82 Console::println("TextureReader::loadTexture(): Could not load texture: " + canonicalPathName + "/" + canonicalFileName + ": " + (exception.what()));
83 }
84 }
85
86 // done
87 if (texture != nullptr) texture->acquireReference();
88
89 //
90 if (useCache == true) textureCacheMutex->unlock();
91
92 // done
93 return texture;
94}
95
96Texture* TextureReader::read2(const string& texturePathName, const string& textureFileName, const string& transparencyTexturePathName, const string& transparencyTextureFileName, bool useCache, bool powerOfTwo, const string& idPrefix) {
97 // make canonical
98 auto canonicalFile = FileSystem::getInstance()->getCanonicalPath(texturePathName, textureFileName);
99 auto canonicalPathName = FileSystem::getInstance()->getPathName(canonicalFile);
100 auto canonicalFileName = FileSystem::getInstance()->getFileName(canonicalFile);
101 auto cacheId = idPrefix + canonicalPathName + "/" + canonicalFileName + "/transparency";
102
103 // do cache look up
104 if (useCache == true) {
106 auto textureCacheIt = textureCache->find(cacheId);
107 if (textureCacheIt != textureCache->end()) {
108 auto texture = textureCacheIt->second;
109 texture->acquireReference();
111 return texture;
112 }
113 }
114
115 // load diffuse texture
116 auto texture = TextureReader::read(texturePathName, textureFileName, false, powerOfTwo, idPrefix);
117 if (texture == nullptr) {
118 if (useCache == true) textureCacheMutex->unlock();
119 return nullptr;
120 }
121 // additional transparency texture
122 auto transparencyTexture = TextureReader::read(transparencyTexturePathName, transparencyTextureFileName, false, powerOfTwo, idPrefix);
123 // do we have one?
124 if (transparencyTexture == nullptr) {
125 Console::println("TextureReader::read(): transparency texture: failed: " + texturePathName + "/" + textureFileName + ";" + transparencyTexturePathName + "/" + transparencyTextureFileName);
126 if (useCache == true) {
127 (*textureCache)[cacheId] = texture;
129 }
130 return texture;
131 }
132 // same dimensions and supported pixel depth?
133 if (texture->getTextureWidth() != transparencyTexture->getTextureWidth() || texture->getTextureHeight() != transparencyTexture->getTextureHeight()) {
134 Console::println("TextureReader::read(): texture does not match transparency texture dimension: " + texturePathName + "/" + textureFileName + ";" + transparencyTexturePathName + "/" + transparencyTextureFileName);
135 if (useCache == true) {
136 (*textureCache)[cacheId] = texture;
138 }
139 return texture;
140 }
141 // yep, combine diffuse map + diffuse transparency map
142 auto textureWidth = texture->getTextureWidth();
143 auto textureHeight = texture->getTextureHeight();
144 ByteBuffer* textureByteBuffer = new ByteBuffer(textureWidth * textureHeight * 4);
145 auto textureWithTransparency = new Texture(
146 idPrefix + texture->getId() + "/transparency",
147 32,
148 texture->getWidth(),
149 texture->getHeight(),
150 texture->getTextureWidth(),
151 texture->getTextureHeight(),
152 textureByteBuffer
153 );
154 auto diffuseTextureBytesPerPixel = texture->getDepth() / 8;
155 auto transparencyTextureBytesPerPixel = transparencyTexture->getDepth() / 8;
156 for (auto y = 0; y < textureHeight; y++) {
157 for (auto x = 0; x < textureWidth; x++) {
158 auto transparencyTextureRed = transparencyTexture->getTextureData()->get((y * textureWidth * transparencyTextureBytesPerPixel) + (x * transparencyTextureBytesPerPixel) + 0);
159 auto transparencyTextureGreen = transparencyTexture->getTextureData()->get((y * textureWidth * transparencyTextureBytesPerPixel) + (x * transparencyTextureBytesPerPixel) + 1);
160 auto transparencyTextureBlue = transparencyTexture->getTextureData()->get((y * textureWidth * transparencyTextureBytesPerPixel) + (x * transparencyTextureBytesPerPixel) + 2);
161 textureByteBuffer->put(texture->getTextureData()->get((y * textureWidth * diffuseTextureBytesPerPixel) + (x * diffuseTextureBytesPerPixel) + 0));
162 textureByteBuffer->put(texture->getTextureData()->get((y * textureWidth * diffuseTextureBytesPerPixel) + (x * diffuseTextureBytesPerPixel) + 1));
163 textureByteBuffer->put(texture->getTextureData()->get((y * textureWidth * diffuseTextureBytesPerPixel) + (x * diffuseTextureBytesPerPixel) + 2));
164 textureByteBuffer->put((uint8_t)((transparencyTextureRed + transparencyTextureGreen + transparencyTextureBlue) * 0.3333f));
165 }
166 }
167 textureWithTransparency->acquireReference();
168 if (useCache == true) {
169 (*textureCache)[cacheId] = textureWithTransparency;
171 }
172 texture->releaseReference();
173 transparencyTexture->releaseReference();
174 return textureWithTransparency;
175}
176
177void TextureReader::readPNGDataFromMemory(png_structp png_ptr, png_bytep outBytes, png_size_t outBytesToRead) {
178 png_voidp io_ptr = png_get_io_ptr(png_ptr);
179 if (io_ptr == nullptr) return;
180
181 PNGInputStream* pngInputStream = static_cast<PNGInputStream*>(io_ptr);
182 pngInputStream->readBytes((int8_t*)outBytes, outBytesToRead);
183}
184
185Texture* TextureReader::readPNG(const string& textureId, const vector<uint8_t>& data, bool powerOfTwo, const string& idPrefix) {
186 // see: http://devcry.heiho.net/html/2015/20150517-libpng.html
187
188 // create PNG input stream
189 PNGInputStream pngInputStream(&data);
190
191 // check that the PNG signature is in the file header
192 unsigned char sig[8];
193 pngInputStream.readBytes((int8_t*)sig, sizeof(sig));
194 if (png_sig_cmp(sig, 0, 8)) {
195 return nullptr;
196 }
197
198 // create two data structures: 'png_struct' and 'png_info'
199 png_structp png = png_create_read_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
200 if (png == nullptr) {
201 return nullptr;
202 }
203 png_infop info = png_create_info_struct(png);
204 if (info == nullptr) {
205 png_destroy_read_struct(&png, nullptr, nullptr);
206 return nullptr;
207 }
208
209 // set up custom read function
210 png_set_read_fn(png, &pngInputStream, TextureReader::readPNGDataFromMemory);
211
212 // set libpng error handling mechanism
213 if (setjmp(png_jmpbuf(png))) {
214 png_destroy_read_struct(&png, &info, nullptr);
215 return nullptr;
216 }
217
218 // tell libpng we already read the signature
219 png_set_sig_bytes(png, sizeof(sig));
220
221 // get image information
222 png_read_info(png, info);
223
224 // dimensions
225 auto width = png_get_image_width(png, info);
226 auto height = png_get_image_height(png, info);
227
228 // set one byte per channel
229 if (png_get_bit_depth(png, info) < 8) png_set_packing(png);
230 if (png_get_bit_depth(png, info) == 16) png_set_strip_16(png);
231
232 // determine bytes per pixel
233 auto bytesPerPixel = -1;
234 switch(png_get_color_type(png, info)) {
235 case PNG_COLOR_TYPE_GRAY:
236 {
237 bytesPerPixel = 3;
238 png_set_gray_to_rgb(png);
239 break;
240 }
241 case PNG_COLOR_TYPE_GRAY_ALPHA:
242 {
243 bytesPerPixel = 4;
244 png_set_gray_to_rgb(png);
245 break;
246 }
247 case PNG_COLOR_TYPE_PALETTE:
248 {
249 // if transparency, convert it to alpha
250 // does not seem to work though
251 bool alpha = false;
252 if (png_get_valid(png, info, PNG_INFO_tRNS)) {
253 alpha = true;
254 png_set_tRNS_to_alpha(png);
255 }
256 bytesPerPixel = alpha?4:3;
257 png_set_expand(png);
258 break;
259 }
260 case PNG_COLOR_TYPE_RGB:
261 {
262 bytesPerPixel = 3;
263 break;
264 }
265 case PNG_COLOR_TYPE_RGBA:
266 {
267 bytesPerPixel = 4;
268 break;
269 }
270 default:
271 {
272 png_destroy_read_struct(&png, &info, nullptr);
273 return nullptr;
274 }
275 }
276
277 // ... what ever :DDD
278 png_set_interlace_handling(png);
279 png_read_update_info(png, info);
280
281 // allocate pixel buffer
282 ByteBuffer* pixelByteBuffer = ByteBuffer::allocate(width * height * bytesPerPixel);
283
284 // setup array with row pointers into pixel buffer
285 png_bytep* rows = new png_bytep[height];
286 uint8_t* p = (uint8_t*)pixelByteBuffer->getBuffer();
287 for(auto i = 0; i < height; i++) {
288 rows[i] = p;
289 p += width * bytesPerPixel;
290 }
291
292 // read all rows (data goes into 'pixels' buffer)
293 // Note that any decoding errors will jump to the
294 // setjmp point and eventually return nullptr
295 png_read_image(png, rows);
296 png_read_end(png, nullptr);
297
298 //
299 delete [] rows;
300
301 // done
302 png_destroy_read_struct(&png, &info, nullptr);
303
304 // make width, height a power of 2
305 auto textureWidth = width;
306 auto textureHeight = height;
307 if (powerOfTwo == true) {
308 textureWidth = 1;
309 while (textureWidth < width) textureWidth*= 2;
310 textureHeight = 1;
311 while (textureHeight < height) textureHeight*= 2;
312 if (textureWidth != width || textureHeight != height) {
313 Console::println("TextureReader::loadPNG(): " + idPrefix + textureId + ": scaling to fit power of 2: " + to_string(width) + "x" + to_string(height) + " --> " + to_string(textureWidth) + "x" + to_string(textureHeight));
314 ByteBuffer* pixelByteBufferScaled = ByteBuffer::allocate(textureWidth * textureHeight * bytesPerPixel);
315 auto textureYIncrement = (float)textureHeight / (float)height;
316 auto textureYPixelRest = 0.0f;
317 auto textureY = 0;
318 for (auto y = 0; y < height; y++) {
319 for (auto i = 0; i < (int)textureYIncrement + (int)textureYPixelRest; i++) {
320 scaleTextureLine(pixelByteBuffer, pixelByteBufferScaled, width, textureWidth, bytesPerPixel, y);
321 textureY++;
322 }
323 textureYPixelRest-= (int)textureYPixelRest;
324 textureYPixelRest+= textureYIncrement - (int)textureYIncrement;
325
326 }
327 while (textureY < textureHeight) {
328 scaleTextureLine(pixelByteBuffer, pixelByteBufferScaled, width, textureWidth, bytesPerPixel, height - 1);
329 textureY++;
330 }
331 delete pixelByteBuffer;
332 pixelByteBuffer = pixelByteBufferScaled;
333 }
334 }
335
336 // thats it
337 auto texture = new Texture(
338 idPrefix + textureId,
339 bytesPerPixel * 8,
340 width,
341 height,
342 textureWidth,
343 textureHeight,
344 pixelByteBuffer
345 );
346 texture->acquireReference();
347 return texture;
348}
349
350void TextureReader::scaleTextureLine(ByteBuffer* pixelByteBuffer, ByteBuffer* pixelByteBufferScaled, int width, int textureWidth, int bytesPerPixel, int y) {
351 auto textureXIncrement = (float)textureWidth / (float)width;
352 auto textureXPixelRest = 0.0f;
353 auto textureX = 0;
354 for (auto x = 0; x < width; x++) {
355 for (auto i = 0; i < (int)textureXIncrement + (int)textureXPixelRest; i++) {
356 for (auto bytePerPixel = 0; bytePerPixel < bytesPerPixel; bytePerPixel++) {
357 pixelByteBufferScaled->put(pixelByteBuffer->get((y * width * bytesPerPixel) + (x * bytesPerPixel) + bytePerPixel));
358 }
359 textureX++;
360 }
361 textureXPixelRest-= (int)textureXPixelRest;
362 textureXPixelRest+= textureXIncrement - (int)textureXIncrement;
363 }
364 {
365 auto x = width - 1;
366 while (textureX < textureWidth) {
367 for (auto bytePerPixel = 0; bytePerPixel < bytesPerPixel; bytePerPixel++) {
368 pixelByteBufferScaled->put(pixelByteBuffer->get((y * width * bytesPerPixel) + (x * bytesPerPixel) + bytePerPixel));
369 }
370 textureX++;
371 }
372 }
373}
374
377 auto textureCacheIt = textureCache->find(texture->getId());
378 if (textureCacheIt != textureCache->end()) textureCache->erase(textureCacheIt);
380}
381
382Texture* TextureReader::rotate(Texture* texture, float rotation) {
383 auto textureWidth = texture->getTextureWidth();
384 auto textureHeight = texture->getTextureHeight();
385 auto textureBytesPerPixel = texture->getDepth() / 8;
386 auto textureWidthRotated = -1;
387 auto textureHeightRotated = -1;
388 {
389 auto rotationsMatrix = Matrix2D3x3::rotateAroundPoint(Vector2(textureWidth / 2.0f, textureHeight / 2.0f), rotation);
390 Vector2 leftTop(0.0f, 0.0f);
391 Vector2 rightTop(textureWidth, 0.0f);
392 Vector2 leftBottom(0.0f, textureHeight);
393 Vector2 rightBottom(textureWidth, textureHeight);
394 auto leftTopRotated = rotationsMatrix.multiply(leftTop);
395 auto rightTopRotated = rotationsMatrix.multiply(rightTop);
396 auto leftBottomRotated = rotationsMatrix.multiply(leftBottom);
397 auto rightBottomRotated = rotationsMatrix.multiply(rightBottom);
398 auto textureWidthTransformedMin = Math::min(leftTopRotated.getX(),
399 Math::min(rightTopRotated.getX(),
400 Math::min(leftBottomRotated.getX(),
401 rightBottomRotated.getX())));
402 auto textureWidthTransformedMax = Math::max(leftTopRotated.getX(),
403 Math::max(rightTopRotated.getX(),
404 Math::max(leftBottomRotated.getX(),
405 rightBottomRotated.getX())));
406 auto textureHeightTransformedMin = Math::min(leftTopRotated.getY(),
407 Math::min(rightTopRotated.getY(),
408 Math::min(leftBottomRotated.getY(),
409 rightBottomRotated.getY())));
410 auto textureHeightTransformedMax = Math::max(leftTopRotated.getY(),
411 Math::max(rightTopRotated.getY(),
412 Math::max(leftBottomRotated.getY(),
413 rightBottomRotated.getY())));
414 textureWidthRotated = static_cast<int>(textureWidthTransformedMax
415 - textureWidthTransformedMin);
416 textureHeightRotated = static_cast<int>(textureHeightTransformedMax
417 - textureHeightTransformedMin);
418 }
419 auto rotatedTextureByteBuffer = new ByteBuffer(textureWidthRotated * textureHeightRotated * 4);
420 auto rotatedTexture = new Texture(
421 texture->getId() + ":r=" + to_string(rotation),
422 32,
423 textureWidthRotated,
424 textureHeightRotated,
425 textureWidthRotated,
426 textureHeightRotated,
427 rotatedTextureByteBuffer
428 );
429 rotatedTexture->acquireReference();
430 auto rotationsMatrix = Matrix2D3x3::rotateAroundPoint(Vector2(textureWidth / 2.0f, textureHeight / 2.0f), rotation);
431 for (auto y = 0; y < textureHeightRotated; y++) {
432 for (auto x = 0; x < textureWidthRotated; x++) {
433 auto originalTexturePoint = rotationsMatrix.multiply(
434 Vector2(
435 x - (textureWidthRotated - textureWidth) / 2.0f,
436 y - (textureHeightRotated - textureHeight) / 2.0f
437 )
438 );
439 auto red = 0;
440 auto green = 0;
441 auto blue = 0;
442 auto alpha = 0;
443 auto originalX = static_cast<int>(originalTexturePoint.getX());
444 auto originalY = static_cast<int>(originalTexturePoint.getY());
445 if (originalX >= 0 && originalX < textureWidth && originalY >= 0 && originalY < textureHeight) {
446 red = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 0);
447 green = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 1);
448 blue = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 2);
449 alpha = textureBytesPerPixel == 4?texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 3):-1;
450 }
451 rotatedTextureByteBuffer->put(red);
452 rotatedTextureByteBuffer->put(green);
453 rotatedTextureByteBuffer->put(blue);
454 rotatedTextureByteBuffer->put(alpha);
455 }
456 }
457 // TODO: should be improved
458 auto filteredTextureByteBuffer = new ByteBuffer(textureWidthRotated * textureHeightRotated * 4);
459 auto filteredTexture = new Texture(
460 texture->getId() + ":r=" + to_string(rotation) + ":bf",
461 32,
462 textureWidthRotated,
463 textureHeightRotated,
464 textureWidthRotated,
465 textureHeightRotated,
466 filteredTextureByteBuffer
467 );
468 filteredTexture->acquireReference();
469 for (auto y = 0; y < textureHeightRotated; y++) {
470 for (auto x = 0; x < textureWidthRotated; x++) {
471 auto samples = 0;
472 auto red = 0;
473 auto green = 0;
474 auto blue = 0;
475 auto alpha = 0;
476 for (auto _y = -1; _y <= 1; _y++)
477 for (auto _x = -1; _x <= 1; _x++)
478 if ((Math::abs(_x) == 1 && Math::abs(_y) == 1) == false &&
479 (x + _x >= 0 && x + _x < textureWidthRotated && y + _y >= 0 && y + _y < textureHeightRotated)) {
480 auto pixelOffset = (y + _y) * textureWidthRotated * 4 + (x + _x) * 4;
481 red+= rotatedTexture->getTextureData()->get(pixelOffset + 0);
482 green+= rotatedTexture->getTextureData()->get(pixelOffset + 1);
483 blue+= rotatedTexture->getTextureData()->get(pixelOffset + 2);
484 alpha+= rotatedTexture->getTextureData()->get(pixelOffset + 3);
485 samples++;
486 }
487 filteredTextureByteBuffer->put(red / samples);
488 filteredTextureByteBuffer->put(green / samples);
489 filteredTextureByteBuffer->put(blue / samples);
490 filteredTextureByteBuffer->put(alpha / samples);
491 }
492 }
493 rotatedTexture->releaseReference();
494 return filteredTexture;
495}
496
497Texture* TextureReader::scale(Texture* texture, int width, int height) {
498 auto textureWidth = texture->getTextureWidth();
499 auto textureHeight = texture->getTextureHeight();
500 auto textureBytesPerPixel = texture->getDepth() / 8;
501 auto textureWidthScaled = width;
502 auto textureHeightScaled = height;
503 auto scaledTextureByteBuffer = new ByteBuffer(textureWidthScaled * textureHeightScaled * 4);
504 auto scaledTexture = new Texture(
505 texture->getId() + ":s=" + to_string(textureWidthScaled) + "x" + to_string(textureHeightScaled),
506 32,
507 textureWidthScaled,
508 textureHeightScaled,
509 textureWidthScaled,
510 textureHeightScaled,
511 scaledTextureByteBuffer
512 );
513 scaledTexture->acquireReference();
514 auto scaleX = static_cast<float>(textureWidth) / static_cast<float>(textureWidthScaled);
515 auto scaleY = static_cast<float>(textureHeight) / static_cast<float>(textureHeightScaled);
516 for (auto y = 0; y < textureHeightScaled; y++) {
517 for (auto x = 0; x < textureWidthScaled; x++) {
518 auto originalTexturePoint = Vector2(static_cast<float>(x) * scaleX, static_cast<float>(y) * scaleY);
519 auto red = 0;
520 auto green = 0;
521 auto blue = 0;
522 auto alpha = 0;
523 auto originalX = static_cast<int>(originalTexturePoint.getX());
524 auto originalY = static_cast<int>(originalTexturePoint.getY());
525 if (originalX >= 0 && originalX < textureWidth && originalY >= 0 && originalY < textureHeight) {
526 red = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 0);
527 green = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 1);
528 blue = texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 2);
529 alpha = textureBytesPerPixel == 4?texture->getTextureData()->get((originalY * textureWidth * textureBytesPerPixel) + (originalX * textureBytesPerPixel) + 3):-1;
530 }
531 scaledTextureByteBuffer->put(red);
532 scaledTextureByteBuffer->put(green);
533 scaledTextureByteBuffer->put(blue);
534 scaledTextureByteBuffer->put(alpha);
535 }
536 }
537 // no bilinear filtering if texture got smaller
538 if (textureWidthScaled < textureWidth) return scaledTexture;
539 // otherwise do bilinear filtering
540 // TODO: should be improved
541 auto filteredTextureByteBuffer = new ByteBuffer(textureWidthScaled * textureHeightScaled * 4);
542 auto filteredTexture = new Texture(
543 texture->getId() + ":s=" + to_string(textureWidthScaled) + "x" + to_string(textureHeightScaled) + ":bf",
544 32,
545 textureWidthScaled,
546 textureHeightScaled,
547 textureWidthScaled,
548 textureHeightScaled,
549 filteredTextureByteBuffer
550 );
551 filteredTexture->acquireReference();
552 for (auto y = 0; y < filteredTexture->getTextureHeight(); y++) {
553 for (auto x = 0; x < filteredTexture->getTextureWidth(); x++) {
554 auto samples = 0;
555 auto red = 0;
556 auto green = 0;
557 auto blue = 0;
558 auto alpha = 0;
559 for (auto _y = -1; _y <= 1; _y++)
560 for (auto _x = -1; _x <= 1; _x++)
561 if ((Math::abs(_x) == 1 && Math::abs(_y) == 1) == false &&
562 (x + _x >= 0 && x + _x < textureWidthScaled && y + _y >= 0 && y + _y < textureHeightScaled)) {
563 auto pixelOffset = (y + _y) * textureWidthScaled * 4 + (x + _x) * 4;
564 red+= scaledTexture->getTextureData()->get(pixelOffset + 0);
565 green+= scaledTexture->getTextureData()->get(pixelOffset + 1);
566 blue+= scaledTexture->getTextureData()->get(pixelOffset + 2);
567 alpha+= scaledTexture->getTextureData()->get(pixelOffset + 3);
568 samples++;
569 }
570 filteredTextureByteBuffer->put(red / samples);
571 filteredTextureByteBuffer->put(green / samples);
572 filteredTextureByteBuffer->put(blue / samples);
573 filteredTextureByteBuffer->put(alpha / samples);
574 }
575 }
576 scaledTexture->releaseReference();
577 return filteredTexture;
578}
579
581 // TODO: should be improved
582 auto filteredTextureByteBuffer = new ByteBuffer(texture->getTextureWidth() * texture->getTextureHeight() * 4);
583 auto filteredTexture = new Texture(
584 texture->getId() + ":smooth",
585 32,
586 texture->getTextureWidth(),
587 texture->getTextureHeight(),
588 texture->getTextureWidth(),
589 texture->getTextureHeight(),
590 filteredTextureByteBuffer
591 );
592 filteredTexture->acquireReference();
593 for (auto y = 0; y < filteredTexture->getTextureHeight(); y++) {
594 for (auto x = 0; x < filteredTexture->getTextureWidth(); x++) {
595 auto samples = 0;
596 auto red = 0;
597 auto green = 0;
598 auto blue = 0;
599 auto alpha = 0;
600 for (auto _y = -1; _y <= 1; _y++)
601 for (auto _x = -1; _x <= 1; _x++)
602 if ((Math::abs(_x) == 1 && Math::abs(_y) == 1) == false &&
603 (x + _x >= 0 && x + _x < texture->getTextureWidth() && y + _y >= 0 && y + _y < texture->getTextureHeight())) {
604 auto pixelOffset = (y + _y) * texture->getTextureWidth() * 4 + (x + _x) * 4;
605 red+= texture->getTextureData()->get(pixelOffset + 0);
606 green+= texture->getTextureData()->get(pixelOffset + 1);
607 blue+= texture->getTextureData()->get(pixelOffset + 2);
608 alpha+= texture->getTextureData()->get(pixelOffset + 3);
609 samples++;
610 }
611 filteredTextureByteBuffer->put(red / samples);
612 filteredTextureByteBuffer->put(green / samples);
613 filteredTextureByteBuffer->put(blue / samples);
614 filteredTextureByteBuffer->put(alpha / samples);
615 }
616 }
617 return filteredTexture;
618}
void readBytes(int8_t *outBytes, int32_t outBytesToRead)
Read bytes.
static void removeFromCache(Texture *texture)
Remove texture from cache.
static Texture * readPNG(const string &textureId, const vector< uint8_t > &data, bool powerOfTwo=true, const string &idPrefix=string())
Read PNG.
static void scaleTextureLine(ByteBuffer *pixelByteBuffer, ByteBuffer *pixelByteBufferScaled, int width, int textureWidth, int bytesPerPixel, int y)
Scales a texture line.
static Texture * smooth(Texture *texture)
Smooth texture.
static Texture * scale(Texture *texture, int width, int height)
Scale texture.
static void readPNGDataFromMemory(png_structp png_ptr, png_bytep outBytes, png_size_t outBytesToRead)
Read PNG data from memory.
static STATIC_DLL_IMPEXT Mutex * textureCacheMutex
static Texture * read(const string &pathName, const string &fileName, bool useCache=true, bool powerOfTwo=true, const string &idPrefix=string())
Reads a texture.
static STATIC_DLL_IMPEXT map< string, Texture * > * textureCache
static Texture * read2(const string &texturePathName, const string &textureFileName, const string &transparencyTexturePathName, const string &transparencyTextureFileName, bool useCache=true, bool powerOfTwo=true, const string &idPrefix=string())
Reads a texture with additional transparency texture.
static Texture * rotate(Texture *texture, float rotation)
Rotate texture around center.
static STATIC_DLL_IMPEXT vector< string > extensions
const string & getId() const
Definition: Texture.h:60
3x3 2D Matrix class
Definition: Matrix2D3x3.h:22
2D vector 2 class
Definition: Vector2.h:19
File system singleton class.
Definition: FileSystem.h:14
Mutex implementation.
Definition: Mutex.h:27
void unlock()
Unlocks this mutex.
Definition: Mutex.cpp:48
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Definition: Mutex.cpp:39
Buffer * put(uint8_t value)
Put value into buffer.
Definition: Buffer.h:110
uint8_t * getBuffer()
Definition: Buffer.h:131
uint8_t get(int32_t position)
Definition: Buffer.h:102
Byte buffer class.
Definition: ByteBuffer.h:24
Console class.
Definition: Console.h:26
void acquireReference()
acquires a reference, incrementing the counter
Definition: Reference.cpp:16
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19