Compare commits

...

2 commits

Author SHA1 Message Date
Epicalert 3224199813 Use libpng for writing row by row 2024-09-15 06:39:53 +08:00
Epicalert e2b3daf3ee Change window title 2024-09-15 04:14:23 +08:00
4 changed files with 86 additions and 27 deletions

View file

@ -6,6 +6,7 @@ set(CMAKE_AUTOMOC ON)
set(CMAKE_CXX_STANDARD_REQUIRED ON) set(CMAKE_CXX_STANDARD_REQUIRED ON)
find_package(Qt6 6.2 REQUIRED COMPONENTS Quick QuickControls2) find_package(Qt6 6.2 REQUIRED COMPONENTS Quick QuickControls2)
find_package(PNG REQUIRED)
qt_add_executable(apptexcrafting qt_add_executable(apptexcrafting
@ -19,6 +20,7 @@ qt_add_qml_module(apptexcrafting
Main.qml Main.qml
TexSlot.qml TexSlot.qml
SOURCES texturecrafter.h texturecrafter.cpp SOURCES texturecrafter.h texturecrafter.cpp
RESOURCES
) )
# Qt for iOS sets MACOSX_BUNDLE_GUI_IDENTIFIER automatically since Qt 6.1. # Qt for iOS sets MACOSX_BUNDLE_GUI_IDENTIFIER automatically since Qt 6.1.
@ -34,6 +36,7 @@ set_target_properties(apptexcrafting PROPERTIES
target_link_libraries(apptexcrafting target_link_libraries(apptexcrafting
PRIVATE Qt6::Quick PRIVATE Qt6::Quick
PNG::PNG
) )
include(GNUInstallDirs) include(GNUInstallDirs)

View file

@ -8,7 +8,7 @@ ApplicationWindow {
width: 640 width: 640
height: 480 height: 480
visible: true visible: true
title: qsTr("Hello World") title: qsTr("TexCrafting")
TextureCrafter { TextureCrafter {
id: crafter id: crafter

View file

@ -4,6 +4,18 @@ TextureCrafter::TextureCrafter(QObject *parent)
: QObject{parent} : QObject{parent}
{} {}
QString TextureCrafter::getOutFilename() {
QString outFileName;
if (outDir.isValid()) {
outFileName = outDir.path();
outFileName.append("/out.png");
} else {
return QString();
}
return outFileName;
}
QUrl TextureCrafter::packChannels(QVector<QUrl> imagePaths) { QUrl TextureCrafter::packChannels(QVector<QUrl> imagePaths) {
QVector<QImage> sourceImages; QVector<QImage> sourceImages;
@ -26,44 +38,84 @@ QUrl TextureCrafter::packChannels(QVector<QUrl> imagePaths) {
sourceImages.append(newImage); sourceImages.append(newImage);
} }
QImage outImage(width, height, format); png_structp outPng;
png_infop outInfo;
FILE *outFile;
QString outFilename = getOutFilename();
unsigned char *rowBuffer;
// maybe abstract this?
outFile = fopen(outFilename.toLatin1().data(), "wb");
if (!outFile) {
return QUrl();
}
outPng = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
outInfo = png_create_info_struct(outPng);
rowBuffer = (unsigned char*)malloc(sizeof(char) * 3 * width);
png_init_io(outPng, outFile);
png_set_IHDR (outPng,
outInfo,
width,
height,
8, // TODO: support other bit depths someday
PNG_COLOR_TYPE_RGB,
PNG_INTERLACE_NONE,
PNG_COMPRESSION_TYPE_DEFAULT,
PNG_FILTER_TYPE_DEFAULT);
png_write_info(outPng, outInfo);
// do the channel packing!! // do the channel packing!!
// theres probably a faster way to do this but eh // theres probably a faster way to do this but eh
for (int i = 0; i < width; i++) { for (int i = 0; i < height; i++) {
for (int j = 0; j < height; j++) { memset(rowBuffer, 0, sizeof(char) * 3 * width);
int r, g, b; for (int j = 0; j < width; j++) {
unsigned char r, g, b;
// too eepy for dry // too eepy for dry
r = sourceImages.at(0).pixelColor(i, j).red(); r = sourceImages.at(0).pixelColor(j, i).red();
g = sourceImages.at(1).pixelColor(i, j).green(); g = sourceImages.at(1).pixelColor(j, i).green();
b = sourceImages.at(2).pixelColor(i, j).blue(); b = sourceImages.at(2).pixelColor(j, i).blue();
QColor newColor(r, g, b); rowBuffer[j*3+0] = r;
rowBuffer[j*3+1] = g;
outImage.setPixelColor(i, j, newColor); rowBuffer[j*3+2] = b;
}
} }
// write it to a temp file png_write_row(outPng, rowBuffer);
if (outDir.isValid()) {
QString outFileName = outDir.path();
outFileName.append("/out.png");
printf("wrote to %s\n", outFileName.toLatin1().data());
//TODO: use libpng and do progressive write to update the progress bar
if (outImage.save(outFileName)) {
printf("cool ^-^\n");
return QUrl::fromLocalFile(outFileName);
} else {
//TODO: return some error value to show the user
printf("that dream is fucked it is fucking fucked\n");
}
} else {
printf("chat its so over\n");
} }
png_write_end(outPng, NULL);
// good soldiers free their memory
free(rowBuffer);
png_destroy_write_struct(&outPng, &outInfo);
fclose(outFile);
return QUrl::fromLocalFile(outFilename);
// // write it to a temp file
// if (outDir.isValid()) {
// QString outFileName = outDir.path();
// outFileName.append("/out.png");
// printf("wrote to %s\n", outFileName.toLatin1().data());
// //TODO: use libpng and do progressive write to update the progress bar
// if (outImage.save(outFileName)) {
// printf("cool ^-^\n");
// return QUrl::fromLocalFile(outFileName);
// } else {
// //TODO: return some error value to show the user
// printf("that dream is fucked it is fucking fucked\n");
// }
// } else {
// printf("chat its so over\n");
// }
return QUrl(); return QUrl();
//do we need to close these??? idk lets hope qt handles it //do we need to close these??? idk lets hope qt handles it
} }

View file

@ -6,6 +6,8 @@
#include <QImage> #include <QImage>
#include <QTemporaryDir> #include <QTemporaryDir>
#include <png.h>
class TextureCrafter : public QObject class TextureCrafter : public QObject
{ {
Q_OBJECT Q_OBJECT
@ -17,6 +19,8 @@ public:
signals: signals:
private: private:
QString getOutFilename();
QTemporaryDir outDir; QTemporaryDir outDir;
}; };