diff --git a/.vscode/c_cpp_properties.json b/.vscode/c_cpp_properties.json new file mode 100644 index 0000000..1c3a862 --- /dev/null +++ b/.vscode/c_cpp_properties.json @@ -0,0 +1,18 @@ +{ + "configurations": [ + { + "name": "CMake", + "includePath": [ + "${workspaceFolder}/**" + ], + "defines": [ + "_DEBUG", + "UNICODE", + "_UNICODE" + ], + "intelliSenseMode": "${default}", + "configurationProvider": "ms-vscode.cmake-tools" + } + ], + "version": 4 +} \ No newline at end of file diff --git a/.vscode/launch.json b/.vscode/launch.json deleted file mode 100644 index 725841c..0000000 --- a/.vscode/launch.json +++ /dev/null @@ -1,15 +0,0 @@ -{ - // Use IntelliSense to learn about possible attributes. - // Hover to view descriptions of existing attributes. - // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 - "version": "0.2.0", - "configurations": [ - { - "type": "by-gdb", - "request": "launch", - "name": "Launch(gdb)", - "program": "${workspaceRoot}/build/Debug/LangtonsAnt.exe", - "cwd": "${workspaceRoot}" - } - ] -} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json index 0db5873..b4d8c35 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,3 +1,3 @@ { - "cmake.configureOnOpen": true + "C_Cpp.default.configurationProvider": "ms-vscode.cmake-tools" } \ No newline at end of file diff --git a/ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED.png b/ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_4320x4320.png similarity index 100% rename from ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED.png rename to ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_4320x4320.png diff --git a/ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_512x512.png b/ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_512x512.png new file mode 100644 index 0000000..046a1bd Binary files /dev/null and b/ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_512x512.png differ diff --git a/README.md b/README.md index 131a1cf..f0c1f41 100644 --- a/README.md +++ b/README.md @@ -1,13 +1,24 @@ +![Langton's ant originally on a 30720x17280 grid (resized to 4320x4320) with the LRRRRRLLR pattern after 1292334158 iterations](ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED_512x512.png) + This repository contains a C++ and web implementation of the [Langton's ant](https://wikipedia.org/wiki/Langton's_ant) universal Turing machine. The web version can be found [here](https://rafa-br34.github.io/LangtonsAnt) - - -![Langton's ant originally on a 30720x17280 grid (resized to 4320x4320) with the LRRRRRLLR pattern after 1292334158 iterations](ASSETS/LRRRRRLLR_30720x17280_1292334158_RESIZED.png) +## Table of contents +* [Implementation notes](#implementation-notes) +* [Usage](#usage) + * [Web version](#web-version) + * [C++ version](#c-version) + * [Arguments](#arguments) + * [`x` & `y`](#x---y) + * [`m`](#m) + * [`a`](#a) + * [`i`](#i) + * [`s`](#s) + * [`o`](#o) + * [`t`](#t) -# Usage -## Implementation notes +# Implementation notes Both the C++ ([Ant.h](https://github.com/rafa-br34/LangtonsAnt/blob/master/SOURCE/Types/Ant.h)) and JavaScript ([Ant.js](https://github.com/rafa-br34/LangtonsAnt/blob/master/WEBSITE/Scripts/Ant.js)) version share the same implementation concept. There's support for 8 possible operations, which are: - R45/L45 @@ -15,14 +26,15 @@ There's support for 8 possible operations, which are: - R135/L135 - C (continue) - U (180 degree turn) -## Web +# Usage +## Web version Placeholder -## C++ +## C++ version ### Arguments: #### `-x` & `-y`: Canvas width/height -### `-m`: +#### `-m`: Defines a state machine, can later be used when defining a ant with `-a`. Can be chained using `;` Example: `-m LRRRRRLLR;RLLLLLRRL`: Creates two state machines with index 0 and 1 @@ -51,9 +63,8 @@ Defines how to output images. - `f::`: Write image files. `%d` can be used for the image index and `%i` for the current iteration. - `s:::`: Write images to a pipe/stream. Supported formats: -- `idx`/`u8`: Outputs the raw buffer. +- `idx`: Outputs the raw buffer. - `rgb24`: Outputs the raw buffer as RGB24. - `png`: Outputs the raw buffer as PNG. -- `idx-gzip`/`u8-gzip`: Outputs a compressed version of the buffer. #### `-t`: Defines how many threads should be used, for now this argument only changes the amount of threads used when encoding images, *eventually* multithreading for the ants will also be added. \ No newline at end of file diff --git a/SOURCE/Encoding.h b/SOURCE/Encoding.h new file mode 100644 index 0000000..243b11e --- /dev/null +++ b/SOURCE/Encoding.h @@ -0,0 +1,260 @@ +#pragma once +#include "Common.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +#include "Types/SimulationState.h" +#include "Types/Vector.h" +#include "Types/Ant.h" + +namespace Encoding { + union RGBA32 { + uint32_t Color = 0; + + struct { + uint8_t R, G, B, A; + } RGBA; + + operator uint32_t() const { return Color; } + }; + + typedef RGBA32(* ColorDescriptor)(size_t); + typedef void(* EncoderCallback)(const uint8_t*, size_t, const Vector2&, unsigned int); + + class PaletteManager { + private: + std::vector m_Palette = {}; + ColorDescriptor m_LastProcedure = nullptr; + + public: + ColorDescriptor ColoringProcedure = [](size_t Index) -> RGBA32 { + auto I = uint32_t(Index); + XOR_SHIFT32(I); + I *= 0x9E3779B9; + return { ((I & 0x00FF0000) >> 16) | (I & 0x0000FF00) | ((I & 0x000000FF) << 16) | (I & 0xFF000000) }; + }; + + void ResizePalette(size_t StateCount) { + if (m_LastProcedure != ColoringProcedure) { // Reset the cache if the coloring procedure has changed + m_LastProcedure = ColoringProcedure; + m_Palette.clear(); + } + + size_t CacheSize = m_Palette.size(); + if (StateCount <= CacheSize) { + if (CacheSize - StateCount > CacheSize * 2) // Resize the cache if it's more than 2x bigger + m_Palette.resize(StateCount); + + return; + } + + m_Palette.reserve(StateCount); + for (size_t s = m_Palette.size(); s < StateCount; s++) + m_Palette.push_back(ColoringProcedure(s)); + } + + INLINE size_t GetSize() const { + return m_Palette.size(); + } + + INLINE RGBA32 GetColor(size_t Index) { + if (Index + 1 > m_Palette.size()) + ResizePalette(Index + 1); + + return m_Palette[Index]; + } + + INLINE RGBA32* GetData() { + return m_Palette.data(); + } + + INLINE RGBA32 operator[](size_t Index) { + return m_Palette[Index]; + } + }; + + class ThreadManager { + private: + std::condition_variable m_ThreadsSignal = {}; + std::atomic m_ThreadsActive = 0; // @todo std::atomic might not be required because of m_ThreadsLock and m_ThreadsSignal + std::mutex m_ThreadsLock = {}; + + public: + size_t ThreadCount = 0; + + void AcquireThread() { + std::unique_lock Lock(m_ThreadsLock); + while (ThreadCount > 0 && m_ThreadsActive >= ThreadCount) + m_ThreadsSignal.wait(Lock); + + m_ThreadsActive++; + DEBUG_PRINT("Acquired thread(%d active)\n", (int)m_ThreadsActive); + } + + void ReleaseThread() { + std::unique_lock Lock(m_ThreadsLock); + m_ThreadsActive--; + m_ThreadsSignal.notify_all(); + DEBUG_PRINT("Released thread(%d active)\n", (int)m_ThreadsActive); + } + + void WaitJobs() { + std::unique_lock Lock(m_ThreadsLock); + while (m_ThreadsActive > 0) + m_ThreadsSignal.wait(Lock); + } + + size_t ActiveThreads() const { return m_ThreadsActive; } + + ThreadManager() = default; + ~ThreadManager() { WaitJobs(); } + }; + + enum class PNGFormat : uint8_t { + PNG_PALETTE, + PNG_GRAYSCALE, + _PNG_END, + + RAW_U8, + _RAW_END + }; + + class EncoderState { + private: + lodepng::State m_SetupEncoderState(size_t StateCount, unsigned int BitDepth) { + lodepng::State State = {}; + + lodepng_state_init(&State); + + LodePNGColorType ColorType; + + switch (Format) { + case DataFormat::PNG_PALETTE: ColorType = LCT_PALETTE; break; + case DataFormat::PNG_GRAYSCALE: ColorType = LCT_GREY; break; + + default: { + DEBUG_PRINT("Unknown data format %d, ColorType will be set to LCT_GREY\n", (int)Format); + ColorType = LCT_GREY; + break; + } + } + + State.info_png.color.colortype = ColorType; + State.info_raw.colortype = ColorType; + + State.info_png.color.bitdepth = BitDepth; + State.info_raw.bitdepth = BitDepth; + + State.encoder.auto_convert = 0; + + if (Format != DataFormat::PNG_PALETTE) return State; + + // DEBUG_PRINT("Setting up palette with %d colors (%d bits per pixel):\n", (int)StateCount, (int)BitDepth); + Palette.ResizePalette(StateCount); + for (size_t i = 0; i < StateCount; i++) { + auto [R, G, B, A] = Palette[i].RGBA; + + // DEBUG_PRINT(" %lu: %06X\n", (long unsigned int)i, Color & 0x00FFFFFF); + lodepng_palette_add(&State.info_png.color, R, G, B, 0xFF); + lodepng_palette_add(&State.info_raw, R, G, B, 0xFF); + } + + return State; + } + + + template + void m_EncodeBuffer(const CellType* Grid, const Vector2& Size, size_t StateCount, Function Callback) { + if (DataFormat::_PNG_END > Format) { + std::vector Buffer = {}; + lodepng::State State = m_SetupEncoderState(StateCount, 8); + + unsigned int Result = lodepng::encode(Buffer, (const unsigned char*)Grid, (unsigned int)Size.X, (unsigned int)Size.Y, State); + + ASSERT_MSG(Result == 0, "lodepng::encode -> %d\n", Result); + + Callback(Buffer.data(), Buffer.size(), Vector2(Size.X, Size.Y), Result); + } + else if (DataFormat::_RAW_END > Format) { + Callback(Grid, (size_t)Size.X * (size_t)Size.Y, Vector2(Size.X, Size.Y), 0); + } + } + + + public: + PaletteManager Palette = {}; + ThreadManager Threads = {}; + + DataFormat Format = DataFormat::PNG_PALETTE; + + template + void EncodeSync(const SimulationState& State, Function Callback) { + m_EncodeBuffer(State.CanvasPointer, State.CanvasSize, State.PossibleStates, Callback); + } + + template + void EncodeSync(const SimulationState& State, const Vector2& SectionPosition, const Vector2& SectionSize, Function Callback) { + std::vector Section(SectionSize.X * SectionSize.Y); + + for (SizeType Y = 0; Y < SectionSize.Y; Y++) { + auto RowStart = State.CanvasPointer + FLATTEN_2D(SectionPosition.X, SectionPosition.Y + Y, State.CanvasSize.X); + + std::copy( + RowStart, + RowStart + SectionSize.X, + Section.begin() + SectionSize.Y * Y + ); + } + + return m_EncodeBuffer(Section.data(), SectionSize, State.PossibleStates, Callback); + } + + template + void EncodeAsync(const SimulationState& State, Function Callback) { + auto GridCopy = std::make_shared>(); + auto GridSize = State.CanvasSize; + + GridCopy->assign(State.CanvasPointer, State.CanvasPointer + GridSize.X * GridSize.Y); + + Threads.AcquireThread(); + std::thread([&, GridCopy, GridSize, States = State.PossibleStates, Callback]() { + m_EncodeBuffer(GridCopy->data(), GridSize, States, Callback); + Threads.ReleaseThread(); + }).detach(); + } + + template + void EncodeAsync(const SimulationState& State, const Vector2& SectionPosition, const Vector2& SectionSize, Function Callback) { + auto Section = std::make_shared>(SectionSize.X * SectionSize.Y); + + for (SizeType Y = 0; Y < SectionSize.Y; Y++) { + auto RowStart = State.CanvasPointer + FLATTEN_2D(SectionPosition.X, SectionPosition.Y + Y, State.CanvasSize.X); + + std::copy( + RowStart, + RowStart + SectionSize.X, + Section->begin() + SectionSize.Y * Y + ); + } + + Threads.AcquireThread(); + std::thread([&, Section, SectionSize, States = State.PossibleStates, Callback]() { + m_EncodeBuffer(Section->data(), SectionSize, States, Callback); + Threads.ReleaseThread(); + }).detach(); + } + + EncoderState() = default; + ~EncoderState() { Threads.WaitJobs(); } + }; +} diff --git a/SOURCE/Entry.cpp b/SOURCE/Entry.cpp index e52feaf..6138440 100644 --- a/SOURCE/Entry.cpp +++ b/SOURCE/Entry.cpp @@ -6,6 +6,7 @@ #include #include #include +#include #include #include @@ -13,10 +14,11 @@ #include #include "ConfigParser.h" +#include "Encoding.h" #include "Common.h" + #include "Types/SimulationState.h" -#include "Types/EncoderState.h" #include "Types/Vector.h" #include "Types/Ant.h" @@ -103,6 +105,7 @@ int main(int ArgCount, const char* ArgValues[]) { } */ + int main(int ArgCount, const char* ArgValues[]) { using enum DirectionEnum; std::vector StateMachine = { @@ -130,10 +133,13 @@ int main(int ArgCount, const char* ArgValues[]) { //std::cout << "State machine: " << StateMachineToString(StateMachine, "") << '\n'; - Vector2 CanvasSize(1920, 1920);//{ 30720, 17280 }; + Vector2 CanvasSize(30720, 30720);//{ 30720, 17280 }; - SimulationState Simulation; - EncoderState Encoder; + SimulationState Simulation = {}; + Encoding::PaletteManager Palette = {}; + Encoding::ThreadManager Threads = {}; + + Threads.ThreadCount = 50; Simulation.Resize(CanvasSize); @@ -157,33 +163,72 @@ int main(int ArgCount, const char* ArgValues[]) { // ffmpeg -r 60 -i "Frames/%d.png" -b:v 5M -c:v libx264 -preset veryslow -qp 0 -s 1920x1920 output.mp4 // ffmpeg -r 60 -i "Frames/%d.png" -b:v 5M -c:v libx264 -preset veryslow -qp 0 -s 1920x1920 -sws_flags neighbor output.mp4 // ffmpeg -r 30 -i "Frames/%d.png" -c:v libx264 -preset veryslow -qp 0 -s 7680x4320 output.mp4 - // ./LangtonsAnt | ~/ffmpeg/ffmpeg -y -f rawvideo -pix_fmt rgba -s 1920x1920 -r 30 -i - -c:v libx264 -preset veryslow output.h264 - // ./LangtonsAnt | ~/ffmpeg/ffmpeg -y -f rawvideo -pix_fmt rgba -s 1920x1920 -r 30 -i - -c:v libx264 -preset veryslow -s 1920x1920 output.h264 + // ./LangtonsAnt | ~/ffmpeg/ffmpeg -y -f rawvideo -pix_fmt rgb24 -s 30720x30720 -r 30 -i - -c:v libx264 -preset veryslow -s 7680x7680 output.h264 + // ./LangtonsAnt | ~/ffmpeg/ffmpeg -y -f rawvideo -pix_fmt rgb24 -s 1920x1920 -r 30 -i - -c:v libx264 -preset veryslow -s 1920x1920 output.h264 + // 1ull * 1000000000ull 1b // 1ull * 1000000ull 1m - size_t Iterations = 1ull * 1000000000ull; + size_t Iterations = 3ull * 1000000000ull; double FrameRate = 30.0; // Video frame rate double Time = 240.0; // Video time size_t Frames = size_t(Time * FrameRate); size_t CaptureDelta = size_t(double(Iterations) / double(Frames)); - size_t FrameSize = (size_t)CanvasSize.X * (size_t)CanvasSize.Y; - uint32_t* FrameBuffer = new uint32_t[FrameSize]; - Simulation.Reset(); - Encoder.Threads = 75; + Palette.ResizePalette(Simulation.PossibleStates); + + std::mutex Mutex = {}; + size_t CurrentFrame = 0; + for (size_t i = 0; i < Frames; i++) { Simulation.Simulate(CaptureDelta);//std::cout << "i:" << i << ' ' << Simulation.Simulate(CaptureDelta) << '/' << CaptureDelta << '\n'; - for (size_t p = 0; p < FrameSize; p++) { - FrameBuffer[p] = Encoder.GetColor(Simulation.CanvasPointer[p]) & 0x00FFFFFF; - } - fwrite(FrameBuffer, 1, FrameSize * sizeof(uint32_t), stdout); + auto GridCopy = std::make_shared>(); + + GridCopy->assign(Simulation.CanvasPointer, Simulation.CanvasPointer + CanvasSize.X * CanvasSize.Y); + + Threads.AcquireThread(); + std::thread([&, GridCopy, i]() { + uint8_t* Canvas = GridCopy->data(); + + uint8_t* Image = new uint8_t[Center.X * Center.Y * 3]; + + auto Start = std::chrono::high_resolution_clock::now(); + for (int X = 0; X < Center.X; X++) { + for (int Y = 0; Y < Center.Y; Y++) { + //* + auto C00 = Palette[Canvas[FLATTEN_2D(X * 2 + 0, Y * 2 + 0, CanvasSize.X)]].RGBA; + auto C01 = Palette[Canvas[FLATTEN_2D(X * 2 + 0, Y * 2 + 1, CanvasSize.X)]].RGBA; + auto C10 = Palette[Canvas[FLATTEN_2D(X * 2 + 1, Y * 2 + 0, CanvasSize.X)]].RGBA; + auto C11 = Palette[Canvas[FLATTEN_2D(X * 2 + 1, Y * 2 + 1, CanvasSize.X)]].RGBA; + + size_t Index = FLATTEN_2D(X, Y, Center.X) * 3; + + Image[Index + 0] = uint8_t(((float)C00.R + (float)C01.R + (float)C10.R + (float)C11.R) / 4.f); + Image[Index + 1] = uint8_t(((float)C00.G + (float)C01.G + (float)C10.G + (float)C11.G) / 4.f); + Image[Index + 2] = uint8_t(((float)C00.B + (float)C01.B + (float)C10.B + (float)C11.B) / 4.f); + //*/ + } + } + auto Elapsed = std::chrono::duration_cast(std::chrono::high_resolution_clock::now() - Start); + + while (CurrentFrame != i) + std::this_thread::sleep_for(std::chrono::milliseconds(100)); + + Mutex.lock(); + std::cerr << "Frame " << CurrentFrame << " Took " << Elapsed << " seconds\n"; + fwrite(Image, 1, Center.X * Center.Y * 3, stdout); + CurrentFrame++; + Mutex.unlock(); + + delete[] Image; + + Threads.ReleaseThread(); + }).detach(); + + //Encoder.EncodeAsync(Simulation, std::string("Frames/") + std::to_string(i) +".png"); } - - delete[] FrameBuffer; - Encoder.WaitJobs(); } diff --git a/SOURCE/Types/EncoderState.h b/SOURCE/Types/EncoderState.h deleted file mode 100644 index c3ae2b2..0000000 --- a/SOURCE/Types/EncoderState.h +++ /dev/null @@ -1,199 +0,0 @@ -#pragma once -#include "../Common.h" - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include - -#include "SimulationState.h" -#include "Vector.h" -#include "Ant.h" - - -typedef uint32_t(* ColorDescriptor)(size_t); - -class EncoderState { -private: - std::vector m_PaletteCache = {}; - - std::atomic m_ThreadsActive = 0; // @todo std::atomic might not be required - std::condition_variable m_ThreadsSignal = {}; - std::mutex m_ThreadsLock = {}; - - void m_AcquireThread() { - std::unique_lock Lock(m_ThreadsLock); - while (Threads > 0 && m_ThreadsActive >= Threads) - m_ThreadsSignal.wait(Lock); - - m_ThreadsActive++; - DEBUG_PRINT("Acquired thread(%d active)\n", (int)m_ThreadsActive); - } - - void m_ReleaseThread() { - std::unique_lock Lock(m_ThreadsLock); - m_ThreadsActive--; - m_ThreadsSignal.notify_all(); - DEBUG_PRINT("Released thread(%d active)\n", (int)m_ThreadsActive); - } - - ColorDescriptor m_LastProcedure = nullptr; - - void m_UpdatePaletteCache(size_t StateCount) { - if (m_LastProcedure != Coloring) { - m_PaletteCache.clear(); - m_LastProcedure = Coloring; - } - - size_t CacheSize = m_PaletteCache.size(); - if (StateCount <= CacheSize) { - if (CacheSize - StateCount > CacheSize * 2) - m_PaletteCache.resize(StateCount); - - return; - } - - m_PaletteCache.reserve(StateCount); - for (size_t s = m_PaletteCache.size(); s < StateCount; s++) - m_PaletteCache.push_back(Coloring(s)); - } - - template - lodepng::State m_SetupEncoderState(size_t StateCount) { - lodepng::State State = {}; - - lodepng_state_init(&State); - - State.info_raw.colortype = LCT_PALETTE; - State.info_raw.bitdepth = sizeof(CellType) * 8; - State.info_png.color.colortype = LCT_PALETTE; - State.info_png.color.bitdepth = sizeof(CellType) * 8; - State.encoder.auto_convert = 0; - - // DEBUG_PRINT("Setting up palette with %d colors (%d bit depth):\n", (int)StateCount, (int)Depth); - m_UpdatePaletteCache(StateCount); - for (size_t i = 0; i < StateCount; i++) { - uint32_t Color = m_PaletteCache[i]; - - // DEBUG_PRINT(" %lu: %06X\n", (long unsigned int)i, Color & 0x00FFFFFF); - lodepng_palette_add( - &State.info_png.color, - uint8_t((Color & 0xFF0000) >> 16), - uint8_t((Color & 0x00FF00) >> 8), - uint8_t((Color & 0x0000FF) >> 0), - 0xFF - ); - lodepng_palette_add( - &State.info_raw, - uint8_t((Color & 0xFF0000) >> 16), - uint8_t((Color & 0x00FF00) >> 8), - uint8_t((Color & 0x0000FF) >> 0), - 0xFF - ); - } - - return State; - } - - template - unsigned int m_EncodeBuffer(const CellType* Grid, const Vector2& Size, size_t StateCount, const std::string& Path) { - std::vector Buffer = {}; - unsigned int Result = 0; - lodepng::State State = m_SetupEncoderState(StateCount); - - Result = lodepng::encode(Buffer, (const unsigned char*)Grid, (unsigned int)Size.X, (unsigned int)Size.Y, State); - - if (Result) { DEBUG_PRINT("lodepng::encode -> %d\n", Result); return Result; } - - Result = lodepng::save_file(Buffer, Path); - if (Result) { DEBUG_PRINT("lodepng::save_file -> %d\n", Result); return Result; } - - return Result; - } - -public: - ColorDescriptor Coloring = [](size_t Index) -> uint32_t { auto I = uint32_t(Index); XOR_SHIFT32(I); return I * 0x9E3779B9; }; - size_t Threads = 1; - - INLINE uint32_t GetColor(size_t Index) { - if (Index + 1 > m_PaletteCache.size()) - m_UpdatePaletteCache(Index + 1); - - return m_PaletteCache[Index]; - } - - template - unsigned int EncodeSync(const SimulationState& State, std::string Path) { - return m_EncodeBuffer(State.CanvasPointer, State.CanvasSize, State.PossibleStates, Path); - } - - template - unsigned int EncodeSync(const SimulationState& State, const Vector2& SectionPosition, const Vector2& SectionSize, std::string OutputPath) { - std::vector Section(SectionSize.X * SectionSize.Y); - - for (SizeType Y = 0; Y < SectionSize.Y; Y++) { - auto RowStart = State.CanvasPointer + FLATTEN_2D(SectionPosition.X, SectionPosition.Y + Y, State.CanvasSize.X); - - std::copy( - RowStart, - RowStart + SectionSize.X, - Section.begin() + SectionSize.Y * Y - ); - } - - return m_EncodeBuffer(Section.data(), SectionSize, State.PossibleStates, OutputPath); - } - - template - void EncodeAsync(const SimulationState& State, std::string OutputPath) { - auto GridCopy = std::make_shared>(); - auto GridSize = State.CanvasSize; - - GridCopy->assign(State.CanvasPointer, State.CanvasPointer + GridSize.X * GridSize.Y); - - m_AcquireThread(); - std::thread([&, GridCopy, GridSize, States = State.PossibleStates, OutputPath]() { - m_EncodeBuffer(GridCopy->data(), GridSize, States, OutputPath); - m_ReleaseThread(); - }).detach(); - } - - template - void EncodeAsync(const SimulationState& State, const Vector2& SectionPosition, const Vector2& SectionSize, std::string OutputPath) { - auto Section = std::make_shared>(SectionSize.X * SectionSize.Y); - - for (SizeType Y = 0; Y < SectionSize.Y; Y++) { - auto RowStart = State.CanvasPointer + FLATTEN_2D(SectionPosition.X, SectionPosition.Y + Y, State.CanvasSize.X); - - std::copy( - RowStart, - RowStart + SectionSize.X, - Section->begin() + SectionSize.Y * Y - ); - } - - m_AcquireThread(); - std::thread([&, Section, SectionSize, States = State.PossibleStates, OutputPath]() { - m_EncodeBuffer(Section->data(), SectionSize, States, OutputPath); - m_ReleaseThread(); - }).detach(); - } - - void WaitJobs() { - std::unique_lock Lock(m_ThreadsLock); - while (m_ThreadsActive > 0) - m_ThreadsSignal.wait(Lock); - } - - EncoderState() = default; - ~EncoderState() { - WaitJobs(); - } -}; \ No newline at end of file diff --git a/WEBSITE/Scripts/Main.js b/WEBSITE/Scripts/Main.js index cf2faee..7495d5d 100644 --- a/WEBSITE/Scripts/Main.js +++ b/WEBSITE/Scripts/Main.js @@ -367,6 +367,9 @@ async function Main() { $("#Simulation_Step").on("click", () => { ReuploadTexture = true; SimulationObject.Update(IPF) }) $("#ResetCamera").on("click", () => { CameraPosition = { X: 0, Y: 0, Z: 0 }; Stats.Camera.html("0, 0, 0") }) $("#SaveImage").on("click", () => { alert("Not yet implemented") }) + + // Others + $("#GitHubLink").on("click", () => { window.open("https://github.com/rafa-br34/LangtonsAnt", "_blank") }) } diff --git a/WEBSITE/Styles/Main.css b/WEBSITE/Styles/Main.css index 968f7c9..b21e0a6 100644 --- a/WEBSITE/Styles/Main.css +++ b/WEBSITE/Styles/Main.css @@ -125,18 +125,20 @@ canvas { text-align: center; background: rgba(0.1, 0.1, 0.1, 0.4); border-bottom: outset 1px; - margin-bottom: 2px; border-top: var(--Text) 1px solid; + margin-bottom: 0px; margin-top: -1px; &.Spaced { - margin-top: 8px; + margin-top: 0px; } } .SettingsPanel .Entry { display: flex; margin-top: 1px; + min-height: 21px; + margin-bottom: 1px; } .SettingsPanel .Entry .Name { @@ -165,7 +167,8 @@ canvas { .List { border: var(--ForegroundActive) 1px solid; width: 100%; - height: fit-content; + max-height: calc(8 * 21px); + min-height: calc(8 * 21px); overflow-y: auto; } diff --git a/WEBSITE/index.html b/WEBSITE/index.html index e899374..4d27238 100644 --- a/WEBSITE/index.html +++ b/WEBSITE/index.html @@ -58,6 +58,7 @@
+
Grid Options
Size:
@@ -72,6 +73,7 @@
+
Simulation Options
@@ -90,6 +92,7 @@
+
Stats
Live Ants:
@@ -127,8 +130,19 @@
0
+ +
Others
+
+ + + +
+
Ant Settings
Position:
@@ -168,6 +182,7 @@
+
Ant List
@@ -176,9 +191,9 @@
-
+
-
+