21#include <ext/libpng/png.h>
40vector<string> TextureReader::extensions = {
"png"};
41Mutex* TextureReader::textureCacheMutex =
new Mutex(
"texturecache-mutex");
42map<string, Texture*>* TextureReader::textureCache =
new map<string, Texture*>();
44const vector<string>& TextureReader::getTextureExtensions() {
53 auto canonicalFilePath = FileSystem::getInstance()->getCanonicalPath(pathName, fileName);
54 auto canonicalPathName = FileSystem::getInstance()->getPathName(canonicalFilePath);
55 auto canonicalFileName = FileSystem::getInstance()->getFileName(canonicalFilePath);
58 if (useCache ==
true) {
60 auto textureCacheIt =
textureCache->find(idPrefix + canonicalPathName +
"/" + canonicalFileName);
62 texture = textureCacheIt->second;
67 if (texture ==
nullptr) {
70 if (StringTools::endsWith(StringTools::toLowerCase(canonicalFileName),
".png") ==
true) {
74 FileSystem::getInstance()->getContent(pathName, fileName, data);
77 if (texture !=
nullptr && useCache ==
true) {
78 (*textureCache)[texture->
getId()] = texture;
82 Console::println(
"TextureReader::loadTexture(): Could not load texture: " + canonicalPathName +
"/" + canonicalFileName +
": " + (exception.what()));
96Texture*
TextureReader::read2(
const string& texturePathName,
const string& textureFileName,
const string& transparencyTexturePathName,
const string& transparencyTextureFileName,
bool useCache,
bool powerOfTwo,
const string& idPrefix) {
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";
104 if (useCache ==
true) {
108 auto texture = textureCacheIt->second;
109 texture->acquireReference();
116 auto texture =
TextureReader::read(texturePathName, textureFileName,
false, powerOfTwo, idPrefix);
117 if (texture ==
nullptr) {
122 auto transparencyTexture =
TextureReader::read(transparencyTexturePathName, transparencyTextureFileName,
false, powerOfTwo, idPrefix);
124 if (transparencyTexture ==
nullptr) {
125 Console::println(
"TextureReader::read(): transparency texture: failed: " + texturePathName +
"/" + textureFileName +
";" + transparencyTexturePathName +
"/" + transparencyTextureFileName);
126 if (useCache ==
true) {
127 (*textureCache)[cacheId] = texture;
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;
142 auto textureWidth = texture->getTextureWidth();
143 auto textureHeight = texture->getTextureHeight();
145 auto textureWithTransparency =
new Texture(
146 idPrefix + texture->getId() +
"/transparency",
149 texture->getHeight(),
150 texture->getTextureWidth(),
151 texture->getTextureHeight(),
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));
167 textureWithTransparency->acquireReference();
168 if (useCache ==
true) {
169 (*textureCache)[cacheId] = textureWithTransparency;
172 texture->releaseReference();
173 transparencyTexture->releaseReference();
174 return textureWithTransparency;
178 png_voidp io_ptr = png_get_io_ptr(png_ptr);
179 if (io_ptr ==
nullptr)
return;
182 pngInputStream->
readBytes((int8_t*)outBytes, outBytesToRead);
192 unsigned char sig[8];
193 pngInputStream.
readBytes((int8_t*)sig,
sizeof(sig));
194 if (png_sig_cmp(sig, 0, 8)) {
199 png_structp png = png_create_read_struct(PNG_LIBPNG_VER_STRING,
nullptr,
nullptr,
nullptr);
200 if (png ==
nullptr) {
203 png_infop info = png_create_info_struct(png);
204 if (info ==
nullptr) {
205 png_destroy_read_struct(&png,
nullptr,
nullptr);
213 if (setjmp(png_jmpbuf(png))) {
214 png_destroy_read_struct(&png, &info,
nullptr);
219 png_set_sig_bytes(png,
sizeof(sig));
222 png_read_info(png, info);
225 auto width = png_get_image_width(png, info);
226 auto height = png_get_image_height(png, info);
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);
233 auto bytesPerPixel = -1;
234 switch(png_get_color_type(png, info)) {
235 case PNG_COLOR_TYPE_GRAY:
238 png_set_gray_to_rgb(png);
241 case PNG_COLOR_TYPE_GRAY_ALPHA:
244 png_set_gray_to_rgb(png);
247 case PNG_COLOR_TYPE_PALETTE:
252 if (png_get_valid(png, info, PNG_INFO_tRNS)) {
254 png_set_tRNS_to_alpha(png);
256 bytesPerPixel = alpha?4:3;
260 case PNG_COLOR_TYPE_RGB:
265 case PNG_COLOR_TYPE_RGBA:
272 png_destroy_read_struct(&png, &info,
nullptr);
278 png_set_interlace_handling(png);
279 png_read_update_info(png, info);
282 ByteBuffer* pixelByteBuffer = ByteBuffer::allocate(width * height * bytesPerPixel);
285 png_bytep* rows =
new png_bytep[height];
286 uint8_t* p = (uint8_t*)pixelByteBuffer->
getBuffer();
287 for(
auto i = 0; i < height; i++) {
289 p += width * bytesPerPixel;
295 png_read_image(png, rows);
296 png_read_end(png,
nullptr);
302 png_destroy_read_struct(&png, &info,
nullptr);
305 auto textureWidth = width;
306 auto textureHeight = height;
307 if (powerOfTwo ==
true) {
309 while (textureWidth < width) textureWidth*= 2;
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;
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);
323 textureYPixelRest-= (int)textureYPixelRest;
324 textureYPixelRest+= textureYIncrement - (int)textureYIncrement;
327 while (textureY < textureHeight) {
328 scaleTextureLine(pixelByteBuffer, pixelByteBufferScaled, width, textureWidth, bytesPerPixel, height - 1);
331 delete pixelByteBuffer;
332 pixelByteBuffer = pixelByteBufferScaled;
338 idPrefix + textureId,
346 texture->acquireReference();
351 auto textureXIncrement = (float)textureWidth / (
float)width;
352 auto textureXPixelRest = 0.0f;
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));
361 textureXPixelRest-= (int)textureXPixelRest;
362 textureXPixelRest+= textureXIncrement - (int)textureXIncrement;
366 while (textureX < textureWidth) {
367 for (
auto bytePerPixel = 0; bytePerPixel < bytesPerPixel; bytePerPixel++) {
368 pixelByteBufferScaled->
put(pixelByteBuffer->
get((y * width * bytesPerPixel) + (x * bytesPerPixel) + bytePerPixel));
385 auto textureBytesPerPixel = texture->
getDepth() / 8;
386 auto textureWidthRotated = -1;
387 auto textureHeightRotated = -1;
389 auto rotationsMatrix = Matrix2D3x3::rotateAroundPoint(
Vector2(textureWidth / 2.0f, textureHeight / 2.0f), rotation);
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);
419 auto rotatedTextureByteBuffer =
new ByteBuffer(textureWidthRotated * textureHeightRotated * 4);
420 auto rotatedTexture =
new Texture(
421 texture->
getId() +
":r=" + to_string(rotation),
424 textureHeightRotated,
426 textureHeightRotated,
427 rotatedTextureByteBuffer
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(
435 x - (textureWidthRotated - textureWidth) / 2.0f,
436 y - (textureHeightRotated - textureHeight) / 2.0f
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;
451 rotatedTextureByteBuffer->put(red);
452 rotatedTextureByteBuffer->put(green);
453 rotatedTextureByteBuffer->put(blue);
454 rotatedTextureByteBuffer->put(alpha);
458 auto filteredTextureByteBuffer =
new ByteBuffer(textureWidthRotated * textureHeightRotated * 4);
459 auto filteredTexture =
new Texture(
460 texture->
getId() +
":r=" + to_string(rotation) +
":bf",
463 textureHeightRotated,
465 textureHeightRotated,
466 filteredTextureByteBuffer
468 filteredTexture->acquireReference();
469 for (
auto y = 0; y < textureHeightRotated; y++) {
470 for (
auto x = 0; x < textureWidthRotated; x++) {
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);
487 filteredTextureByteBuffer->put(red / samples);
488 filteredTextureByteBuffer->put(green / samples);
489 filteredTextureByteBuffer->put(blue / samples);
490 filteredTextureByteBuffer->put(alpha / samples);
493 rotatedTexture->releaseReference();
494 return filteredTexture;
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),
511 scaledTextureByteBuffer
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);
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;
531 scaledTextureByteBuffer->put(red);
532 scaledTextureByteBuffer->put(green);
533 scaledTextureByteBuffer->put(blue);
534 scaledTextureByteBuffer->put(alpha);
538 if (textureWidthScaled < textureWidth)
return scaledTexture;
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",
549 filteredTextureByteBuffer
551 filteredTexture->acquireReference();
552 for (
auto y = 0; y < filteredTexture->getTextureHeight(); y++) {
553 for (
auto x = 0; x < filteredTexture->getTextureWidth(); x++) {
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);
570 filteredTextureByteBuffer->put(red / samples);
571 filteredTextureByteBuffer->put(green / samples);
572 filteredTextureByteBuffer->put(blue / samples);
573 filteredTextureByteBuffer->put(alpha / samples);
576 scaledTexture->releaseReference();
577 return filteredTexture;
583 auto filteredTexture =
new Texture(
584 texture->
getId() +
":smooth",
590 filteredTextureByteBuffer
592 filteredTexture->acquireReference();
593 for (
auto y = 0; y < filteredTexture->getTextureHeight(); y++) {
594 for (
auto x = 0; x < filteredTexture->getTextureWidth(); x++) {
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;
611 filteredTextureByteBuffer->put(red / samples);
612 filteredTextureByteBuffer->put(green / samples);
613 filteredTextureByteBuffer->put(blue / samples);
614 filteredTextureByteBuffer->put(alpha / samples);
617 return filteredTexture;
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
ByteBuffer * getTextureData()
int32_t getTextureHeight() const
int32_t getTextureWidth() const
File system singleton class.
void unlock()
Unlocks this mutex.
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Buffer * put(uint8_t value)
Put value into buffer.
uint8_t get(int32_t position)
void acquireReference()
acquires a reference, incrementing the counter
std::exception Exception
Exception base class.