diff --git a/.gitmodules b/.gitmodules
index f943cf3f..61e594dc 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -21,4 +21,4 @@
url = https://gitlab.freedesktop.org/freetype/freetype.git
[submodule "lib/libarchive"]
path = lib/libarchive
- url = https://github.com/libarchive/libarchive
+ url = https://github.com/libarchive/libarchive
\ No newline at end of file
diff --git a/assets/games/liminal/scenes/prologue/CMakeLists.txt b/assets/games/liminal/scenes/prologue/CMakeLists.txt
index 2696e504..6aa9bd4e 100644
--- a/assets/games/liminal/scenes/prologue/CMakeLists.txt
+++ b/assets/games/liminal/scenes/prologue/CMakeLists.txt
@@ -11,4 +11,5 @@ tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologue4.xml)
tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologue5.xml)
tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologueSportsField.xml)
tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologueToilets.xml)
-tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologue7.xml)
\ No newline at end of file
+tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologue7.xml)
+tool_vnscene(${CMAKE_CURRENT_LIST_DIR}/ScenePrologue9.xml)
\ No newline at end of file
diff --git a/assets/games/liminal/scenes/prologue/ScenePrologue9.xml b/assets/games/liminal/scenes/prologue/ScenePrologue9.xml
new file mode 100644
index 00000000..812190d9
--- /dev/null
+++ b/assets/games/liminal/scenes/prologue/ScenePrologue9.xml
@@ -0,0 +1,10 @@
+
+
+
+
+
+
+ There is a bucket.
+
+
+
\ No newline at end of file
diff --git a/cmake/targets/target-liminal-emscripten/CMakeLists.txt b/cmake/targets/target-liminal-emscripten/CMakeLists.txt
new file mode 100644
index 00000000..7bade086
--- /dev/null
+++ b/cmake/targets/target-liminal-emscripten/CMakeLists.txt
@@ -0,0 +1,13 @@
+# Copyright (c) 2023 Dominic Masters
+#
+# This software is released under the MIT License.
+# https://opensource.org/licenses/MIT
+
+set(DAWN_BUILDING dawnliminal CACHE INTERNAL ${DAWN_CACHE_TARGET})
+set(DAWN_BUILD_HOST_LIBS "" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+set(DAWN_TARGET_EMSCRIPTEN true CACHE INTERNAL ${DAWN_CACHE_TARGET})
+set(DAWN_TARGET_GLFW true CACHE INTERNAL ${DAWN_CACHE_TARGET})
+set(DAWN_TARGET_NAME "Liminal" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+set(DAWN_VISUAL_NOVEL true CACHE INTERNAL ${DAWN_CACHE_TARGET})
+
+set(DAWN_EMSCRIPTEN_FLAGS "" CACHE INTERNAL ${DAWN_CACHE_TARGET})
\ No newline at end of file
diff --git a/lib/CMakeLists.txt b/lib/CMakeLists.txt
index e805855a..d6e8ffaf 100644
--- a/lib/CMakeLists.txt
+++ b/lib/CMakeLists.txt
@@ -5,8 +5,12 @@
# GLFW
if(DAWN_TARGET_GLFW)
- add_subdirectory(glad)
- add_subdirectory(glfw)
+ if(${CMAKE_SYSTEM_NAME} MATCHES "Emscripten")
+ set(DAWN_EMSCRIPTEN_FLAGS "${DAWN_EMSCRIPTEN_FLAGS} -s USE_GLFW=3" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+ else()
+ add_subdirectory(glad)
+ add_subdirectory(glfw)
+ endif()
endif()
# SDL
@@ -23,7 +27,11 @@ add_library(stb INTERFACE)
target_include_directories(stb INTERFACE stb)
# FreeType
-add_subdirectory(freetype)
+if(${CMAKE_SYSTEM_NAME} MATCHES "Emscripten")
+ set(DAWN_EMSCRIPTEN_FLAGS "${DAWN_EMSCRIPTEN_FLAGS} -s USE_FREETYPE=1" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+else()
+ add_subdirectory(freetype)
+endif()
# LibArchive
add_subdirectory(libarchive)
@@ -36,4 +44,11 @@ if(DAWN_TARGET_OPENAL)
set(BUILD_TESTS OFF CACHE BOOL "Build tests" FORCE)
set(BUILD_EXAMPLES OFF CACHE BOOL "Build examples" FORCE)
add_subdirectory(AudioFile)
+endif()
+
+# Test
+if(DEFINED DAWN_EMSCRIPTEN_FLAGS)
+ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${DAWN_EMSCRIPTEN_FLAGS}" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+ set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${DAWN_EMSCRIPTEN_FLAGS}" CACHE INTERNAL ${DAWN_CACHE_TARGET})
+ set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} ${DAWN_EMSCRIPTEN_FLAGS}" CACHE INTERNAL ${DAWN_CACHE_TARGET})
endif()
\ No newline at end of file
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt
index 4e1b9a3a..a42e7d0a 100644
--- a/src/CMakeLists.txt
+++ b/src/CMakeLists.txt
@@ -37,6 +37,8 @@ if(DEFINED DAWN_TARGET_NAME)
add_subdirectory(dawnosx)
elseif(DAWN_TARGET_VITA)
add_subdirectory(dawnvita)
+ elseif(DAWN_TARGET_EMSCRIPTEN)
+ add_subdirectory(dawnemscripten)
else()
message(FATAL_ERROR "You need to define an entry target")
endif()
diff --git a/src/dawn/asset/assets/TextureAsset.cpp b/src/dawn/asset/assets/TextureAsset.cpp
index 69c17a40..eb8b7450 100644
--- a/src/dawn/asset/assets/TextureAsset.cpp
+++ b/src/dawn/asset/assets/TextureAsset.cpp
@@ -37,7 +37,6 @@ void TextureAsset::updateSync() {
void TextureAsset::updateAsync() {
if(this->state != 0x00) return;
this->state = 0x01;
- std::cout << "Update texture tool" << std::endl;
this->loader.open();
this->buffer = (uint8_t*)memoryAllocate(this->loader.getSize());
this->loader.read(this->buffer, this->loader.getSize());
diff --git a/src/dawnemscripten/CMakeLists.txt b/src/dawnemscripten/CMakeLists.txt
new file mode 100644
index 00000000..d67cd61c
--- /dev/null
+++ b/src/dawnemscripten/CMakeLists.txt
@@ -0,0 +1,22 @@
+# Copyright (c) 2022 Dominic Masters
+#
+# This software is released under the MIT License.
+# https://opensource.org/licenses/MIT
+
+# Includes
+target_include_directories(${DAWN_TARGET_NAME}
+ PUBLIC
+ ${CMAKE_CURRENT_LIST_DIR}
+)
+
+# Platform variables
+target_compile_definitions(${DAWN_TARGET_NAME}
+ PUBLIC
+ DAWN_ASSET_LOCATION="../../assets.tar"
+)
+
+# Subdirs
+add_subdirectory(host)
+
+# Ensures a .HTML file is generated.
+set(CMAKE_EXECUTABLE_SUFFIX ".html" CACHE INTERNAL ${DAWN_CACHE_TARGET})
\ No newline at end of file
diff --git a/src/dawnemscripten/host/CMakeLists.txt b/src/dawnemscripten/host/CMakeLists.txt
new file mode 100644
index 00000000..36af58d2
--- /dev/null
+++ b/src/dawnemscripten/host/CMakeLists.txt
@@ -0,0 +1,10 @@
+# Copyright (c) 2023 Dominic Masters
+#
+# This software is released under the MIT License.
+# https://opensource.org/licenses/MIT
+
+# Sources
+target_sources(${DAWN_TARGET_NAME}
+ PRIVATE
+ DawnEmscripten.cpp
+)
\ No newline at end of file
diff --git a/src/dawnemscripten/host/DawnEmscripten.cpp b/src/dawnemscripten/host/DawnEmscripten.cpp
new file mode 100644
index 00000000..7b04d464
--- /dev/null
+++ b/src/dawnemscripten/host/DawnEmscripten.cpp
@@ -0,0 +1,11 @@
+// Copyright (c) 2023 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+#include "host/DawnEmscripten.hpp"
+
+int main() {
+ printf("Hello Emscripten\n");
+ return 0;
+}
\ No newline at end of file
diff --git a/src/dawnemscripten/host/DawnEmscripten.hpp b/src/dawnemscripten/host/DawnEmscripten.hpp
new file mode 100644
index 00000000..94a0ba24
--- /dev/null
+++ b/src/dawnemscripten/host/DawnEmscripten.hpp
@@ -0,0 +1,9 @@
+// Copyright (c) 2023 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+#pragma once
+#include
+
+int main();
\ No newline at end of file
diff --git a/src/dawnliminal/CMakeLists.txt b/src/dawnliminal/CMakeLists.txt
index ef104a51..10bef676 100644
--- a/src/dawnliminal/CMakeLists.txt
+++ b/src/dawnliminal/CMakeLists.txt
@@ -15,6 +15,7 @@ target_include_directories(${DAWN_TARGET_NAME}
# Subdirs
add_subdirectory(game)
add_subdirectory(save)
+add_subdirectory(vnscenes)
# Assets
include("${DAWN_ASSETS_SOURCE_DIR}/games/liminal/CMakeLists.txt")
\ No newline at end of file
diff --git a/src/dawnliminal/vnscenes/CMakeLists.txt b/src/dawnliminal/vnscenes/CMakeLists.txt
new file mode 100644
index 00000000..1d7a7b0c
--- /dev/null
+++ b/src/dawnliminal/vnscenes/CMakeLists.txt
@@ -0,0 +1,10 @@
+# Copyright (c) 2023 Dominic Masters
+#
+# This software is released under the MIT License.
+# https://opensource.org/licenses/MIT
+
+# Sources
+target_sources(${DAWN_TARGET_NAME}
+ PRIVATE
+ ScenePrologue8.cpp
+)
\ No newline at end of file
diff --git a/src/dawnliminal/vnscenes/ScenePrologue8.cpp b/src/dawnliminal/vnscenes/ScenePrologue8.cpp
new file mode 100644
index 00000000..c25df7ce
--- /dev/null
+++ b/src/dawnliminal/vnscenes/ScenePrologue8.cpp
@@ -0,0 +1,117 @@
+// Copyright (c) 2023 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+#include "ScenePrologue8.hpp"
+
+using namespace Dawn;
+
+void ScenePrologue8CustomEventForChoiceSet::onStart() {
+ auto scene = dynamic_cast(this->manager->getScene());
+ assertNotNull(scene, "ScenePrologue8CustomEventForChoiceSet - Scene is null?");
+
+ this->choices = scene->remainingChoices;
+ this->key = "killer";
+ this->text = "It is...";
+
+ VNChoiceEvent::onStart();
+}
+
+void ScenePrologue8CustomEventForAfterChoice::onStart() {
+ auto prev = dynamic_cast(this->previous);
+ assertNotNull(prev, "ScenePrologue8CustomEventForAfterChoice - Previous is null?");
+ auto scene = dynamic_cast(this->manager->getScene());
+ assertNotNull(scene, "ScenePrologue8CustomEventForAfterChoice - Scene is null?");
+
+ auto key = prev->getChoiceKey();
+ scene->remainingChoices.erase(key);
+
+ this->next();
+}
+
+void ScenePrologue8CustomEventForAfterCharacterFollowup::onStart() {
+ auto scene = dynamic_cast(this->manager->getScene());
+ assertNotNull(scene, "ScenePrologue8CustomEventForAfterChoice - Scene is null?");
+
+ if(scene->remainingChoices.empty()) {
+ auto sceneChange = this->manager->createEvent>();
+ this->then(sceneChange);
+ } else {
+ auto choiceSet = this->manager->createEvent();
+ this->then(choiceSet);
+ }
+}
+
+ScenePrologue8::ScenePrologue8(DawnGame *game) : SceneMonologue(game) {
+}
+
+std::vector ScenePrologue8::getRequiredAssets() {
+ auto man = &this->game->assetManager;
+ std::vector assets = SceneMonologue::getRequiredAssets();
+ return assets;
+}
+
+void ScenePrologue8::stage() {
+ SceneMonologue::stage();
+
+ remainingChoices["ave"] = "Ave?";
+ remainingChoices["ronin"] = "Ronin?";
+ remainingChoices["craig"] = "Craig?";
+
+ auto man = &this->game->assetManager;
+ assertNotNull(vnManager, "VNSceneGenInit - VN Manager is null?");
+ VNEvent *previous = vnManager->createEvent();
+ auto eventStart = previous;
+
+ auto event0 = vnManager->createEvent();
+ event0->font = "{{ text }}";
+
+ auto eventChoice = vnManager->createEvent();
+ auto eventAfterChoice = vnManager->createEvent();
+ auto eventFollowUp = vnManager->createEvent();
+
+ // Ave
+ auto eventIfAve = vnManager->createEvent();
+ eventIfAve->key = "killer";
+ eventIfAve->value = "ave";
+
+ auto eventTestAve = vnManager->createEvent();
+ eventTestAve->text = "Ave?";
+ eventIfAve->ifTrue = eventTestAve;
+ eventIfAve->ifEnd = eventTestAve;
+
+
+ // Craig
+ auto eventIfCraig = vnManager->createEvent();
+ eventIfCraig->key = "killer";
+ eventIfCraig->value = "craig";
+
+ auto eventTestCraig = vnManager->createEvent();
+ eventTestCraig->text = "Craig?";
+ eventIfCraig->ifTrue = eventTestCraig;
+ eventIfCraig->ifEnd = eventTestCraig;
+
+
+ // Ronin
+ auto eventIfRonin = vnManager->createEvent();
+ eventIfRonin->key = "killer";
+ eventIfRonin->value = "ronin";
+
+ auto eventTestRonin = vnManager->createEvent();
+ eventTestRonin->text = "Ronin?";
+ eventIfRonin->ifTrue = eventTestRonin;
+ eventIfRonin->ifEnd = eventTestRonin;
+
+
+ // End
+ eventStart->then(event0);
+ event0->then(eventChoice);
+ eventChoice->then(eventAfterChoice);
+ eventAfterChoice->then(eventIfAve);
+ eventIfAve->then(eventIfCraig);
+ eventIfCraig->then(eventIfRonin);
+ eventIfRonin->then(eventFollowUp);
+
+ vnManager->setEvent(eventStart);
+}
\ No newline at end of file
diff --git a/src/dawnliminal/vnscenes/ScenePrologue8.hpp b/src/dawnliminal/vnscenes/ScenePrologue8.hpp
index e7ce4bac..b1d3cb9a 100644
--- a/src/dawnliminal/vnscenes/ScenePrologue8.hpp
+++ b/src/dawnliminal/vnscenes/ScenePrologue8.hpp
@@ -1,5 +1,9 @@
-#pragma once
+// Copyright (c) 2023 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+#pragma once
#include "scenes/SceneMonologue.hpp"
#include "games/vn/events/VNDummyEvent.hpp"
#include "games/vn/events/VNSetDefaultFontEvent.hpp"
@@ -13,70 +17,27 @@
namespace Dawn {
class ScenePrologue8CustomEventForChoiceSet : public VNChoiceEvent {
protected:
- void onStart() override {
- auto scene = dynamic_castthis->manager->getScene();
- assertNotNull(scene, "ScenePrologue8CustomEventForChoiceSet - Scene is null?");
-
- this->choices = scene->remainingChoices;
- this->key = "killer";
- this->text = "It is...";
-
- VNChoiceSetEvent::onStart();
- }
+ void onStart() override;
};
class ScenePrologue8CustomEventForAfterChoice : public VNEvent {
protected:
- void onStart() override {
- auto prev = dynamic_cast(this->previous);
- assertNotNull(prev, "ScenePrologue8CustomEventForAfterChoice - Previous is null?");
- auto scene = dynamic_castthis->manager->getScene();
- assertNotNull(scene, "ScenePrologue8CustomEventForAfterChoice - Scene is null?");
-
- auto key = prev->getChoiceKey();
- scene->remainingChoices.erase(key);
-
- this->next();
- }
+ void onStart() override;
+ };
+
+ class ScenePrologue8CustomEventForAfterCharacterFollowup : public VNEvent {
+ protected:
+ void onStart() override;
};
class ScenePrologue8 : public SceneMonologue {
public:
std::map remainingChoices;
- ScenePrologue8(DawnGame *game) : SceneMonologue(game) {
- }
+ ScenePrologue8(DawnGame *game);
- std::vector getRequiredAssets() override {
- auto man = &this->game->assetManager;
- std::vector assets = SceneMonologue::getRequiredAssets();
- return assets;
- }
+ std::vector getRequiredAssets() override;
- void stage() override {
- SceneMonologue::stage();
-
- remainingChoices["ave"] = "Ave?";
- remainingChoices["ronin"] = "Ronin?";
- remainingChoices["craig"] = "Craig?";
-
- auto man = &this->game->assetManager;
- assertNotNull(vnManager, "VNSceneGenInit - VN Manager is null?");
- VNEvent *previous = vnManager->createEvent();
- auto eventStart = previous;
-
- auto event0 = vnManager->createEvent();
- event0->font = "{{ text }}";
-
- auto event1 = vnManager->createEvent();
- auto event2 = vnManager->createEvent();
- auto sceneChange = vnManager->createEvent>();
-
- eventStart->then(event0);
- event0->then(event1);
- event1->then(event2);
- event2->then(sceneChange);
- vnManager->setEvent(eventStart);
- }
+ void stage() override;
};
-}
+}
\ No newline at end of file
diff --git a/src/dawnshared/util/memory.cpp b/src/dawnshared/util/memory.cpp
index 93b28940..93b6cb9e 100644
--- a/src/dawnshared/util/memory.cpp
+++ b/src/dawnshared/util/memory.cpp
@@ -6,14 +6,6 @@
#include "memory.hpp"
#include "assert/assert.hpp"
-void * operator new(size_t size) noexcept {
- return memoryAllocate(size);
-}
-
-void operator delete (void *p) noexcept {
- return memoryFree(p);
-}
-
void * memoryAllocate(const size_t size) {
assertTrue(size >= 0, "memoryAllocate: size must be greater than 0 or equal to.");
void *x = (void *)memoryCallMalloc(size);
diff --git a/src/dawntools/texturetool/CMakeLists.txt b/src/dawntools/texturetool/CMakeLists.txt
index 530a1924..ef87fe7d 100644
--- a/src/dawntools/texturetool/CMakeLists.txt
+++ b/src/dawntools/texturetool/CMakeLists.txt
@@ -3,40 +3,6 @@
# This software is released under the MIT License.
# https://opensource.org/licenses/MIT
-# Texture Build Tool
-project(texturetool VERSION 1.0)
-add_executable(texturetool)
-
-target_sources(texturetool
- PRIVATE
- ${DAWN_SHARED_SOURCES}
- ${DAWN_TOOL_SOURCES}
- TextureTool.cpp
- ../util/Image.cpp
-)
-
-target_include_directories(texturetool
- PUBLIC
- ${DAWN_SHARED_INCLUDES}
- ${DAWN_TOOL_INCLUDES}
- ${CMAKE_CURRENT_LIST_DIR}
-)
-
-# Definitions
-target_compile_definitions(texturetool
- PUBLIC
- ${DAWN_SHARED_DEFINITIONS}
- DAWN_TOOL_INSTANCE=TextureTool
- DAWN_TOOL_HEADER="TextureTool.hpp"
-)
-
-# Libraries
-target_link_libraries(texturetool
- PUBLIC
- ${DAWN_BUILD_HOST_LIBS}
- stb
-)
-
# Tool Function
function(tool_texture target)
# Defaults
@@ -64,28 +30,21 @@ function(tool_texture target)
if(NOT DEFINED FILE)
message(FATAL_ERROR "Missing FILE input")
endif()
-
- set(DEPS "")
- if(DAWN_BUILD_TOOLS)
- set(DEPS texturetool)
- endif()
-
+
add_custom_target(${target}
- COMMAND texturetool
+ COMMAND ${DAWN_TOOLS_DIR}/texturetool/texturetool.py
--input="${FILE}"
- --output="${DAWN_ASSETS_BUILD_DIR}/${target}"
- --wrapX="${WRAP_X}"
- --wrapY="${WRAP_Y}"
- --filterMin="${FILTER_MIN}"
- --filterMag="${FILTER_MIN}"
+ --output="${DAWN_ASSETS_BUILD_DIR}/${target}.texture"
+ --wrap-x="${WRAP_X}"
+ --wrap-y="${WRAP_Y}"
+ --filter-min="${FILTER_MIN}"
+ --filter-mag="${FILTER_MIN}"
--scale="${SCALE}"
- --cropStartX="${CROP_START_X}"
- --cropStartY="${CROP_START_Y}"
- --cropEndX="${CROP_END_X}"
- --cropEndY="${CROP_END_Y}"
- --preview="${DAWN_BUILD_DIR}/preview/${target}"
+ --crop-start-x="${CROP_START_X}"
+ --crop-start-y="${CROP_START_Y}"
+ --crop-end-x="${CROP_END_X}"
+ --crop-end-y="${CROP_END_Y}"
COMMENT "Generating texture ${target} from ${FILE}"
- DEPENDS ${DEPS}
)
add_dependencies(dawnassets ${target})
endfunction()
\ No newline at end of file
diff --git a/src/dawntools/texturetool/TextureTool.cpp b/src/dawntools/texturetool/TextureTool.cpp
deleted file mode 100644
index 97c11567..00000000
--- a/src/dawntools/texturetool/TextureTool.cpp
+++ /dev/null
@@ -1,193 +0,0 @@
-// Copyright (c) 2023 Dominic Masters
-//
-// This software is released under the MIT License.
-// https://opensource.org/licenses/MIT
-
-#include "TextureTool.hpp"
-
-using namespace Dawn;
-
-std::vector TextureTool::getRequiredFlags() {
- return std::vector{ "input", "output" };
-}
-
-std::map TextureTool::getOptionalFlags() {
- return {
- { "wrapX", "clamp" },
- { "wrapY", "clamp" },
- { "filterMin", "linear" },
- { "filterMax", "linear" },
- { "scale", "" },
- { "scaleWrapX", "clamp" },
- { "scaleWrapY", "clamp" },
- { "scaleFilterX", "nearest" },
- { "scaleFilterY", "nearest" },
- { "cropStartX", "" },
- { "cropStartY", "" },
- { "cropEndX", "" },
- { "cropEndY", "" },
- { "preview", "" }
- };
-}
-
-int32_t TextureTool::start() {
- // Finished with XML data, now we can write data out.
- File fileOut(flags["output"] + ".texture");
- if(fileOut.exists()) return 0;
-
- // Load input file
- File in(flags["input"]);
- if(!in.open(FILE_MODE_READ)) {
- std::cout << "Failed to open input file " << in.filename << std::endl;
- return 1;
- }
-
- int32_t originalWidth, originalHeight, channels;
- auto bufferCurrent = stbi_load_from_file(
- in.file,
- &originalWidth,
- &originalHeight,
- &channels,
- STBI_rgb_alpha
- );
- if(bufferCurrent == NULL) {
- std::cout << "Failed to load input texture!" << std::endl;
- return 1;
- }
- in.close();
-
- // Create a temporary buffer to hold pixels.
- size_t len = STBI_rgb_alpha * originalWidth * originalHeight;
- uint8_t *bufferTemporary = (uint8_t*)malloc(sizeof(uint8_t) * len);
- int32_t currentWidth = originalWidth;
- int32_t currentHeight = originalHeight;
-
- // Crop
- int32_t cropStartX = 0;
- 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 = (cropEndX == 0 ? originalWidth : cropEndX) - cropStartX;
- int32_t cropHeight = (cropEndY == 0 ? originalHeight : cropEndY) - cropStartY;
-
- float_t s0, t0, s1, t1;
- s0 = (float_t)cropStartX / (float_t)originalWidth;
- t0 = (float_t)cropStartY / (float_t)originalHeight;
- s1 = ((float_t)(cropEndX == 0 ? originalWidth : cropEndX) / (float_t)originalWidth);
- t1 = ((float_t)(cropEndY == 0 ? originalHeight : 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_DEFAULT, STBIR_FILTER_DEFAULT,
- STBIR_COLORSPACE_LINEAR, NULL,
- s0, t0, s1, t1
- );
- memcpy(bufferCurrent, bufferTemporary, sizeof(uint8_t) * len);
-
- currentWidth = cropWidth;
- currentHeight = cropHeight;
- }
-
- // 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(
- bufferCurrent, currentWidth, currentHeight, 0,
- bufferTemporary, scaleWidth, scaleHeight, 0,
- STBI_rgb_alpha, -1, 0,
- STBIR_EDGE_CLAMP, STBIR_FILTER_DEFAULT, STBIR_COLORSPACE_LINEAR,
- NULL
- );
- memcpy(bufferCurrent, bufferTemporary, sizeof(uint8_t) * len);
-
- currentWidth = scaleWidth;
- currentHeight = scaleHeight;
- }
-
- // Wrapping Settings
- std::function wrapFromString = [&](std::string wr) {
- if(wr == "repeat") return 0;
- if(wr == "mirror") return 1;
- if(wr == "clamp") return 2;
- if(wr == "border") return 3;
- return -1;
- };
-
- int32_t wrapX = wrapFromString(flags["wrapX"]);
- if(wrapX == -1) {
- std::cout << "Invalid wrapX value " << flags["wrapX"] << std::endl;
- return 1;
- }
-
- int32_t wrapY = wrapFromString(flags["wrapY"]);
- if(wrapY == -1) {
- std::cout << "Invalid wrapY value " << flags["wrapY"] << std::endl;
- return 1;
- }
-
- // Write info
- char headerBuffer[256];
- size_t headerBufferLength = sprintf((char *)headerBuffer, "DT_2.00|%i|%i|%i|%i|%i|%i|%i|",
- currentWidth,
- currentHeight,
- 4, // RGBA,
- wrapX, // WRAPX
- wrapY, // WRAPY
- flags["filterMin"] == "nearest" ? 0 : 1,
- flags["filterMag"] == "nearest" ? 0 : 1
- );
-
- // Open and create output
- File out(flags["output"] + ".texture");
- if(!out.mkdirp()) {
- std::cout << "Failed to make output dir " << out.filename << std::endl;
- return 1;
- }
- if(!out.open(FILE_MODE_WRITE)) {
- std::cout << "Failed to open texture file for writing " << out.filename << std::endl;
- return 1;
- }
- if(!out.writeRaw(headerBuffer, headerBufferLength)) {
- std::cout << "Failed to write texture header for " << out.filename << std::endl;
- return 1;
- }
-
- // Write preview
- File preview(flags["preview"] + ".png");
- if(!preview.mkdirp()) {
- std::cout << "Failed to make preview dir " << preview.filename << std::endl;
- return 1;
- }
- stbi_write_png(
- preview.filename.c_str(),
- currentWidth,
- currentHeight,
- STBI_rgb_alpha,
- bufferCurrent,
- 0
- );
-
- // Write texture
- if(!out.writeRaw((char*)bufferCurrent, sizeof(uint8_t) * len)) {
- std::cout << "Failed to write texture data for " << out.filename << std::endl;
- return 1;
- }
- free(bufferCurrent);
- free(bufferTemporary);
-
- return 0;
-}
\ No newline at end of file
diff --git a/src/dawntools/texturetool/TextureTool.hpp b/src/dawntools/texturetool/TextureTool.hpp
deleted file mode 100644
index 1e7114e7..00000000
--- a/src/dawntools/texturetool/TextureTool.hpp
+++ /dev/null
@@ -1,20 +0,0 @@
-// Copyright (c) 2023 Dominic Masters
-//
-// This software is released under the MIT License.
-// https://opensource.org/licenses/MIT
-
-#pragma once
-#include "util/DawnTool.hpp"
-#include "util/File.hpp"
-#include "util/Image.hpp"
-
-namespace Dawn {
- class TextureTool : public DawnTool {
- protected:
- std::vector getRequiredFlags() override;
- std::map getOptionalFlags() override;
-
- public:
- int32_t start();
- };
-}
\ No newline at end of file
diff --git a/src/dawntools/texturetool/texturetool.py b/src/dawntools/texturetool/texturetool.py
new file mode 100755
index 00000000..c96fcb67
--- /dev/null
+++ b/src/dawntools/texturetool/texturetool.py
@@ -0,0 +1,113 @@
+#!/usr/bin/env python
+# Copyright (c) 2023 Dominic Masters
+#
+# This software is released under the MIT License.
+# https://opensource.org/licenses/MIT
+
+from PIL import Image
+import argparse
+import os
+
+# Args
+parser = argparse.ArgumentParser(description='Converts image textures to internal game data format.')
+parser.add_argument('-i', '--input');
+parser.add_argument('-o', '--output');
+parser.add_argument('-s', '--scale');
+parser.add_argument('-sf', '--scale-filter');
+parser.add_argument('-wx', '--wrap-x');
+parser.add_argument('-wy', '--wrap-y');
+parser.add_argument('-fi', '--filter-min')
+parser.add_argument('-fg', '--filter-mag')
+parser.add_argument('-csx', '--crop-start-x');
+parser.add_argument('-csy', '--crop-start-y');
+parser.add_argument('-cex', '--crop-end-x');
+parser.add_argument('-cey', '--crop-end-y');
+args = parser.parse_args()
+
+# Ensure input exists
+if not os.path.exists(args.input):
+ print(f"Input file '{args.input}' does not exist.")
+ exit(1)
+
+# Open image
+img = Image.open(args.input)
+
+# Normalize the image
+hasAlpha = img.mode == 'RGBA' or img.mode == 'LA'
+
+# Convert the image to RGB or RGBA mode based on alpha channel
+if hasAlpha:
+ img = img.convert('RGBA')
+else:
+ img = img.convert('RGB')
+
+ # Apply cropping
+crop_box = [
+ int(args.crop_start_x) if args.crop_start_x not in (None, "") else 0,
+ int(args.crop_start_y) if args.crop_start_y not in (None, "") else 0,
+ int(args.crop_end_x) if args.crop_end_x not in (None, "") else img.width,
+ int(args.crop_end_y) if args.crop_end_y not in (None, "") else img.height
+]
+img = img.crop(crop_box)
+
+# Apply scaling
+if args.scale not in (None, ""):
+ scale = float(args.scale)
+ newSize = (int(img.width * scale), int(img.height * scale))
+ if args.scale_filter == 'NEAREST':
+ img = img.resize(newSize, Image.NEAREST)
+ elif args.scale_filter == 'BILINEAR':
+ img = img.resize(newSize, Image.BILINEAR)
+ elif args.scale_filter == 'BICUBIC':
+ img = img.resize(newSize, Image.BICUBIC)
+ elif args.scale_filter == 'LANCZOS':
+ img = img.resize(newSize, Image.LANCZOS)
+ else:
+ img = img.resize(newSize)
+
+
+# Filter
+if args.filter_min.lower() == 'NEAREST':
+ filterMin = 0
+else:
+ filterMin = 1
+
+if args.filter_mag.lower() == 'NEAREST':
+ filterMag = 0
+else:
+ filterMag = 1
+
+# Wrap
+if args.wrap_x.lower() == 'repeat':
+ wrapX = 0
+elif args.wrap_x.lower() == 'mirror':
+ wrapX = 1
+elif args.wrap_x.lower() == 'clamp':
+ wrapX = 2
+elif args.wrap_x.lower() == 'border':
+ wrapX = 3
+else:
+ wrapX = 2
+
+if args.wrap_y.lower() == 'repeat':
+ wrapY = 0
+elif args.wrap_y.lower() == 'mirror':
+ wrapY = 1
+elif args.wrap_y.lower() == 'clamp':
+ wrapY = 2
+elif args.wrap_y.lower() == 'border':
+ wrapY = 3
+else:
+ wrapY = 2
+
+# Get raw pixel data
+buffer = img.tobytes()
+
+# Create the output directory if it doesn't exist
+os.makedirs(os.path.dirname(args.output), exist_ok=True)
+
+# Write the image metadata and pixel data to the output file
+with open(args.output, 'wb') as f:
+ header = f"DT_2.00|{img.width}|{img.height}|{4 if hasAlpha else 3}|{wrapX}|{wrapY}|{filterMin}|{filterMag}|"
+ f.write(header.encode())
+ f.write(buffer)
\ No newline at end of file