Added cropping support to texture tool
This commit is contained in:
@ -46,6 +46,10 @@ function(tool_texture target)
|
|||||||
set(WRAP_X "")
|
set(WRAP_X "")
|
||||||
set(WRAP_Y "")
|
set(WRAP_Y "")
|
||||||
set(SCALE "")
|
set(SCALE "")
|
||||||
|
set(CROP_START_X "")
|
||||||
|
set(CROP_START_Y "")
|
||||||
|
set(CROP_END_X "")
|
||||||
|
set(CROP_END_Y "")
|
||||||
|
|
||||||
# Parse Args
|
# Parse Args
|
||||||
foreach(_PAIR IN LISTS ARGN)
|
foreach(_PAIR IN LISTS ARGN)
|
||||||
@ -75,6 +79,10 @@ function(tool_texture target)
|
|||||||
--filterMin="${FILTER_MIN}"
|
--filterMin="${FILTER_MIN}"
|
||||||
--filterMag="${FILTER_MIN}"
|
--filterMag="${FILTER_MIN}"
|
||||||
--scale="${SCALE}"
|
--scale="${SCALE}"
|
||||||
|
--cropStartX="${CROP_START_X}"
|
||||||
|
--cropStartY="${CROP_START_Y}"
|
||||||
|
--cropEndX="${CROP_END_X}"
|
||||||
|
--cropEndY="${CROP_END_Y}"
|
||||||
COMMENT "Generating texture ${target} from ${FILE}"
|
COMMENT "Generating texture ${target} from ${FILE}"
|
||||||
DEPENDS ${DEPS}
|
DEPENDS ${DEPS}
|
||||||
)
|
)
|
||||||
|
@ -21,7 +21,11 @@ std::map<std::string, std::string> TextureTool::getOptionalFlags() {
|
|||||||
{ "scaleWrapX", "clamp" },
|
{ "scaleWrapX", "clamp" },
|
||||||
{ "scaleWrapY", "clamp" },
|
{ "scaleWrapY", "clamp" },
|
||||||
{ "scaleFilterX", "nearest" },
|
{ "scaleFilterX", "nearest" },
|
||||||
{ "scaleFilterY", "nearest" }
|
{ "scaleFilterY", "nearest" },
|
||||||
|
{ "cropStartX", "" },
|
||||||
|
{ "cropStartY", "" },
|
||||||
|
{ "cropEndX", "" },
|
||||||
|
{ "cropEndY", "" }
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -36,41 +40,84 @@ int32_t TextureTool::start() {
|
|||||||
std::cout << "Failed to open input file " << in.filename << std::endl;
|
std::cout << "Failed to open input file " << in.filename << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int w, h, channels;
|
int32_t originalWidth, originalHeight, channels;
|
||||||
auto imageRaw = stbi_load_from_file(in.file, &w, &h, &channels, STBI_rgb_alpha);
|
auto bufferCurrent = stbi_load_from_file(
|
||||||
if(imageRaw == NULL) {
|
in.file,
|
||||||
|
&originalWidth,
|
||||||
|
&originalHeight,
|
||||||
|
&channels,
|
||||||
|
STBI_rgb_alpha
|
||||||
|
);
|
||||||
|
if(bufferCurrent == NULL) {
|
||||||
std::cout << "Failed to load input texture!" << std::endl;
|
std::cout << "Failed to load input texture!" << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
in.close();
|
in.close();
|
||||||
|
|
||||||
// Buffer to output
|
// Create a temporary buffer to hold pixels.
|
||||||
size_t len = STBI_rgb_alpha * w * h;
|
size_t len = STBI_rgb_alpha * originalWidth * originalHeight;
|
||||||
uint8_t *dataImage = (uint8_t*)malloc(sizeof(uint8_t) * len);
|
uint8_t *bufferTemporary = (uint8_t*)malloc(sizeof(uint8_t) * len);
|
||||||
|
int32_t currentWidth = originalWidth;
|
||||||
|
int32_t currentHeight = originalHeight;
|
||||||
|
|
||||||
float_t scale = 1;
|
// Crop
|
||||||
if(!flags["scale"].empty()) {
|
int32_t cropStartX = 0;
|
||||||
scale = std::stof(flags["scale"]);
|
int32_t cropStartY = 0;
|
||||||
|
int32_t cropEndX = 0;
|
||||||
|
int32_t cropEndY = 0;
|
||||||
|
|
||||||
|
if(!flags["cropStartX"].empty()) cropStartX = std::stoi(flags["cropStartX"]);
|
||||||
|
if(!flags["cropStartY"].empty()) cropStartY = std::stoi(flags["cropStartY"]);
|
||||||
|
if(!flags["cropEndX"].empty()) cropEndX = std::stoi(flags["cropEndX"]);
|
||||||
|
if(!flags["cropEndY"].empty()) cropEndY = std::stoi(flags["cropEndY"]);
|
||||||
|
|
||||||
|
if(cropStartX > 0 || cropStartY > 0 || cropEndX > 0 || cropEndY > 0) {
|
||||||
|
int32_t cropWidth = originalWidth - cropStartX - cropEndX;
|
||||||
|
int32_t cropHeight = originalHeight - cropStartY - cropEndY;
|
||||||
|
|
||||||
|
float_t s0, t0, s1, t1;
|
||||||
|
s0 = (float_t)cropStartX / (float_t)originalWidth;
|
||||||
|
t0 = (float_t)cropStartY / (float_t)originalHeight;
|
||||||
|
s1 = 1.0f - ((float_t)cropEndX / (float_t)originalWidth);
|
||||||
|
t1 = 1.0f - ((float_t)cropEndY / (float_t)originalHeight);
|
||||||
|
|
||||||
|
stbir_resize_region(
|
||||||
|
bufferCurrent, currentWidth, currentHeight, 0,
|
||||||
|
bufferTemporary, cropWidth, cropHeight, 0,
|
||||||
|
STBIR_TYPE_UINT8,
|
||||||
|
STBI_rgb_alpha, -1, 0,
|
||||||
|
STBIR_EDGE_CLAMP, STBIR_EDGE_CLAMP,
|
||||||
|
STBIR_FILTER_BOX, STBIR_FILTER_BOX,
|
||||||
|
STBIR_COLORSPACE_LINEAR, NULL,
|
||||||
|
s0, t0, s1, t1
|
||||||
|
);
|
||||||
|
memcpy(bufferCurrent, bufferTemporary, sizeof(uint8_t) * len);
|
||||||
|
|
||||||
|
currentWidth = cropWidth;
|
||||||
|
currentHeight = cropHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(scale != 1) {
|
// Scale
|
||||||
|
if(!flags["scale"].empty()) {
|
||||||
|
float_t scale = std::stof(flags["scale"]);
|
||||||
|
int32_t scaleWidth = currentWidth * scale;
|
||||||
|
int32_t scaleHeight = currentHeight * scale;
|
||||||
|
|
||||||
stbir_resize_uint8_generic(
|
stbir_resize_uint8_generic(
|
||||||
imageRaw, w, h, 0,
|
bufferCurrent, currentWidth, currentHeight, 0,
|
||||||
dataImage, w * scale, h * scale, 0,
|
bufferTemporary, scaleWidth, scaleHeight, 0,
|
||||||
STBI_rgb_alpha, -1, 0,
|
STBI_rgb_alpha, -1, 0,
|
||||||
STBIR_EDGE_CLAMP, STBIR_FILTER_TRIANGLE, STBIR_COLORSPACE_LINEAR,
|
STBIR_EDGE_CLAMP, STBIR_FILTER_TRIANGLE, STBIR_COLORSPACE_LINEAR,
|
||||||
NULL
|
NULL
|
||||||
);
|
);
|
||||||
w = w * scale;
|
memcpy(bufferCurrent, bufferTemporary, sizeof(uint8_t) * len);
|
||||||
h = h * scale;
|
|
||||||
printf("Changing size to %ix%i\n", w, h);
|
currentWidth = scaleWidth;
|
||||||
} else {
|
currentHeight = scaleHeight;
|
||||||
memcpy(dataImage, imageRaw, len);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
stbi_image_free(imageRaw);
|
// Wrapping Settings
|
||||||
|
|
||||||
std::function<int32_t(std::string)> wrapFromString = [&](std::string wr) {
|
std::function<int32_t(std::string)> wrapFromString = [&](std::string wr) {
|
||||||
if(wr == "repeat") return 0;
|
if(wr == "repeat") return 0;
|
||||||
if(wr == "mirror") return 1;
|
if(wr == "mirror") return 1;
|
||||||
@ -94,8 +141,8 @@ int32_t TextureTool::start() {
|
|||||||
// Write info
|
// Write info
|
||||||
char headerBuffer[256];
|
char headerBuffer[256];
|
||||||
size_t headerBufferLength = sprintf((char *)headerBuffer, "DT_2.00|%i|%i|%i|%i|%i|%i|%i|",
|
size_t headerBufferLength = sprintf((char *)headerBuffer, "DT_2.00|%i|%i|%i|%i|%i|%i|%i|",
|
||||||
w,
|
currentWidth,
|
||||||
h,
|
currentHeight,
|
||||||
4, // RGBA,
|
4, // RGBA,
|
||||||
wrapX, // WRAPX
|
wrapX, // WRAPX
|
||||||
wrapY, // WRAPY
|
wrapY, // WRAPY
|
||||||
@ -119,11 +166,12 @@ int32_t TextureTool::start() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Write texture
|
// Write texture
|
||||||
if(!out.writeRaw((char *)dataImage, sizeof(uint8_t) * len)) {
|
if(!out.writeRaw((char*)bufferCurrent, sizeof(uint8_t) * len)) {
|
||||||
std::cout << "Failed to write texture data for " << out.filename << std::endl;
|
std::cout << "Failed to write texture data for " << out.filename << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
free(dataImage);
|
free(bufferCurrent);
|
||||||
|
free(bufferTemporary);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
Reference in New Issue
Block a user