Awesome
<img src="Documents/Acid_03.png" alt="Acid" height="130px">Acid is an open-source, cross-platform game engine written in modern C++17 and structured to be fast, simple, and extremely modular.
Vulkan is the sole graphics API, Vulkan can be accessed in apps with the provided Acid rendering pipeline. Metal is supported through MoltenVK; eventually, DirectX will be supported in a similar way.
This project is being worked on part-time by a single developer, this is under heavy development, expect bugs, API changes, and plenty of missing features.
Features
- Multiplatform (Windows, Linux, MacOS, 32bit and 64bit)
- Multithreaded command buffers and thread safety
- On the fly GLSL to SPIR-V compilation and reflection
- Deferred physically based rendering (PBR)
- Networking (HTTP, FTP, UDP, TCP)
- Object serialization (JSON, XML)
- Resource management using serialization
- Event delegate callbacks with scoped functions
- Bullet physics
- Entity component system
- Particle effect systems
- File multi-path searching, and packaging
- UI constraints system, and MSDF font rendering
- Audio systems (flac, mp3, ogg, opus, wave)
- Shadow mapping
- Post effects pipeline (lensflare, glow, blur, SSAO, ...)
- Model file loading (obj, glTF 2.0)
- Animations loading (Collada)
- Image file loading (png, jpeg, dng, tiff, OpenEXR, bmp, dds, ppm, tga)
Code Snippets
// Imports a 2D texture using nearest filtering.
auto guiBlack = Image2d::Create("Guis/Black.png", VK_FILTER_NEAREST);
// Imports a 3D cubemap (face names defined in Cubemap.cpp).
auto skyboxSnowy = ImageCube::Create("Objects/SkyboxSnowy", ".png");
// Imports a OBJ model.
auto dragon = ObjModel::Create("Objects/Testing/ModelDragon.obj");
// Creates a sphere model with 20 latitude and longitude bands with a radius of 1.
auto sphere = SphereModel::Create(20, 20, 1.0f);
// Plays a 3D sound (sound buffer resource internally managed), at half volume.
Sound jump("Sounds/Jump.ogg", Audio::Type::Effect, false, true, 0.5f);
// Loads a entity from a prefab file.
auto playerObject = GetStructure()->CreateEntity("Objects/Player/Player.json");
playerObject->AddComponent<Transform>();
// Creates a entity in code.
auto sphere = GetStructure()->CreateEntity();
sphere->AddComponent<Transform>(Vector3f(6.7f, 6.7f, -8.0f), Vector3f(0.0f, Maths::Radians(180.0f), 0.0f), Vector3f(3.0f));
sphere->AddComponent<Mesh>(SphereModel::Create(20, 20, 1.0f), // This will used the sphere buffers created earlier.
std::make_unique<DefaultMaterial>(Colour::White, Image2d::Create("Objects/Testing/Albedo.png"), 0.0f, 0.5f,
Image2d::Create("Objects/Testing/Material.png"), Image2d::Create("Objects/Testing/Normal.png")));
sphere->AddComponent<Rigidbody>(std::make_unique<SphereCollider>(), 2.0f); // Will be created weighing 2 units.
// Vector maths.
Vector2f a(3.0f, -7.2f);
Vector2f b(-1.74f, 15.4f);
Vector2f c = a * b;
// Distance between the two points.
float distance = a.Distance(b);
// Right shift of the x and y bits by 1.
Vector2i rightShift = Vector2i(5, 9) >> 1;
// Split a string by spaces.
std::string stringSource = "Hello world!";
std::vector<std::string> stringSplit = String::Split(stringSource, ' ');
// Will run a lambda on window resize, and when this object is deleted the lambda is removed.
Window::Get()->OnSize().connect([](Vector2ui size) {
Log::Out("Hello world: ", size, '\n');
});
// Time addition.
Time dateTime = 4h + 2min + 11s + 9ms + 1us + 4ns;
// Calls the function once after 150 milliseconds.
Timers::Get()->Once([]() {
Log::Out("Timer Once After\n");
}, 150ms);
// Calls the function every 4 seconds.
Timers::Get()->Every([]() {
Log::Out("Timer Every Tick\n");
}, 4s);
// Calls the function every 7 seconds 3 times.
Timers::Get()->Repeat([]() {
static uint32_t i = 0;
Log::Out("Timer Repeat Tick #", i, '\n');
i++;
}, 7s, 3);
Screenshots
<img src="Documents/Screenshot1.png" alt="Acid" width="600px"> <img src="Documents/Screenshot2.png" alt="Acid" width="600px"> <img src="Documents/Screenshot3.png" alt="Acid" width="600px"> <img src="Documents/Screenshot4.png" alt="Acid" width="600px"> <img src="Documents/Screenshot5.png" alt="Acid" width="600px">Compiling
All platforms depend on CMake, 3.11.0 or higher, to generate IDE/make files.
CMake options (default ON):
BUILD_TESTS
ACID_INSTALL_EXAMPLES
ACID_INSTALL_RESOURCES
If you installed Acid using only system libs, then find_package(Acid)
will work from CMake. Versioning is also supported.
When using find_package(Acid)
the imported target Acid::Acid
will be created.
The ACID_RESOURCES_DIR
variable will also be available, which will point to the on-disk location of Acid/Resources
(if installed).
Python 3, Vulkan SDK, OpenAL, and OpenAL SDK are required to develop Acid.
Make sure you have environment variables VULKAN_SDK
and OPENALDIR
set to the paths you have Vulkan and OpenAL installed into.
Ensure you are using a compiler with full C++17 support, on Windows it is recommended that you use MSVC or MinGW w64.
If using Visual Studio it must be 2019 or later (see why 2015 is currently broken). Use the Visual Studio installer and select both "Desktop development with C++" and "Windows SDK" if they are not already installed. Then on Visual Studio Acid can be opened as a CMake workspace folder.
On Linux Acid requires xorg-dev
, libopenal1
, and libvulkan1
to be installed. Read about how to setup Vulkan on Linux so a Vulkan SDK is found.
Setup on MacOS is similar to the setup on Linux, a compiler that supports C++17 is required, such as XCode 10.0.
Contributing
You can contribute to Acid in any way you want, we are always looking for help. You can learn about Acids code style from the GUIDELINES.md.