mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-22 00:57:17 +01:00
282 lines
8.3 KiB
C++
282 lines
8.3 KiB
C++
/*************************************************************************/
|
|
/* mm_material.cpp */
|
|
/*************************************************************************/
|
|
/* This file is part of: */
|
|
/* PANDEMONIUM ENGINE */
|
|
/* https://github.com/Relintai/pandemonium_engine */
|
|
/*************************************************************************/
|
|
/* Copyright (c) 2022-present Péter Magyar. */
|
|
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
|
|
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
/* a copy of this software and associated documentation files (the */
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
/* the following conditions: */
|
|
/* */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/*************************************************************************/
|
|
|
|
#include "mm_material.h"
|
|
|
|
#include "core/os/thread_pool.h"
|
|
#include "core/os/thread_pool_job.h"
|
|
#include "mm_node.h"
|
|
#include "mm_node_universal_property.h"
|
|
|
|
Vector2 MMMaterial::get_image_size() {
|
|
return image_size;
|
|
}
|
|
|
|
void MMMaterial::set_image_size(const Vector2 &val) {
|
|
image_size = val;
|
|
}
|
|
|
|
Vector<Variant> MMMaterial::get_nodes() {
|
|
Vector<Variant> r;
|
|
for (int i = 0; i < nodes.size(); i++) {
|
|
r.push_back(nodes[i].get_ref_ptr());
|
|
}
|
|
return r;
|
|
}
|
|
|
|
void MMMaterial::set_nodes(const Vector<Variant> &val) {
|
|
nodes.clear();
|
|
for (int i = 0; i < val.size(); i++) {
|
|
Ref<MMNode> e = Ref<MMNode>(val[i]);
|
|
nodes.push_back(e);
|
|
}
|
|
}
|
|
|
|
bool MMMaterial::get_initialized() const {
|
|
return initialized;
|
|
}
|
|
|
|
void MMMaterial::set_initialized(const bool val) {
|
|
initialized = val;
|
|
}
|
|
|
|
bool MMMaterial::get_rendering() const {
|
|
return rendering;
|
|
}
|
|
|
|
void MMMaterial::set_rendering(const bool val) {
|
|
rendering = val;
|
|
}
|
|
|
|
bool MMMaterial::get_queued_render() const {
|
|
return queued_render;
|
|
}
|
|
|
|
void MMMaterial::set_queued_render(const bool val) {
|
|
queued_render = val;
|
|
}
|
|
|
|
void MMMaterial::initialize() {
|
|
if (!initialized) {
|
|
initialized = true;
|
|
job->setup(this, "_thread_func");
|
|
|
|
for (int i = 0; i < nodes.size(); ++i) {
|
|
Ref<MMNode> n = nodes[i];
|
|
|
|
if (!n.is_valid()) {
|
|
continue;
|
|
}
|
|
|
|
n->init_properties();
|
|
n->connect("changed", this, "on_node_changed");
|
|
}
|
|
}
|
|
}
|
|
|
|
void MMMaterial::add_node(const Ref<MMNode> &node) {
|
|
ERR_FAIL_COND(!node.is_valid());
|
|
|
|
nodes.push_back(node);
|
|
Ref<MMNode> n = node;
|
|
n->connect("changed", this, "on_node_changed");
|
|
emit_changed();
|
|
}
|
|
|
|
void MMMaterial::remove_node(const Ref<MMNode> &node) {
|
|
if (!node.is_valid()) {
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < node->output_properties.size(); ++i) {
|
|
Ref<MMNodeUniversalProperty> op = node->output_properties[i];
|
|
|
|
for (int j = 0; j < nodes.size(); ++j) {
|
|
Ref<MMNode> n = nodes[j];
|
|
|
|
if (!n.is_valid()) {
|
|
continue;
|
|
}
|
|
|
|
for (int k = 0; k < n->input_properties.size(); ++k) {
|
|
Ref<MMNodeUniversalProperty> ip = n->input_properties[k];
|
|
|
|
if (ip->get_input_property() == op) {
|
|
ip->set_input_property(Ref<MMNodeUniversalProperty>());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
nodes.erase(node);
|
|
|
|
Ref<MMNode> nn = node;
|
|
|
|
nn->disconnect("changed", this, "on_node_changed");
|
|
emit_changed();
|
|
}
|
|
|
|
void MMMaterial::render() {
|
|
initialize();
|
|
|
|
if (rendering) {
|
|
queued_render = true;
|
|
return;
|
|
}
|
|
|
|
if (USE_THREADS) {
|
|
render_threaded();
|
|
} else {
|
|
render_non_threaded();
|
|
}
|
|
}
|
|
|
|
void MMMaterial::render_non_threaded() {
|
|
bool did_render = true;
|
|
Ref<MMMaterial> self = Ref<MMMaterial>(this);
|
|
|
|
while (did_render) {
|
|
did_render = false;
|
|
|
|
for (int i = 0; i < nodes.size(); ++i) {
|
|
Ref<MMNode> n = nodes[i];
|
|
|
|
if (n.is_valid() && n->render(self)) {
|
|
did_render = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void MMMaterial::render_threaded() {
|
|
job->set_cancelled(false);
|
|
|
|
if (!ThreadPool::get_singleton()->has_job(job)) {
|
|
ThreadPool::get_singleton()->add_job(job);
|
|
}
|
|
}
|
|
|
|
void MMMaterial::_thread_func() {
|
|
if (job->get_cancelled()) {
|
|
rendering = false;
|
|
return;
|
|
}
|
|
|
|
rendering = true;
|
|
job->set_cancelled(false);
|
|
bool did_render = true;
|
|
Ref<MMMaterial> self = Ref<MMMaterial>(this);
|
|
|
|
while (did_render) {
|
|
did_render = false;
|
|
|
|
for (int i = 0; i < nodes.size(); ++i) {
|
|
Ref<MMNode> n = nodes[i];
|
|
|
|
if (n.is_valid() && n->render(self)) {
|
|
did_render = true;
|
|
}
|
|
|
|
if (job->get_cancelled()) {
|
|
rendering = false;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
rendering = false;
|
|
|
|
if (queued_render) {
|
|
queued_render = false;
|
|
_thread_func();
|
|
}
|
|
}
|
|
|
|
void MMMaterial::cancel_render_and_wait() {
|
|
if (rendering) {
|
|
ThreadPool::get_singleton()->cancel_job_wait(job);
|
|
job->set_cancelled(false);
|
|
}
|
|
}
|
|
|
|
void MMMaterial::on_node_changed() {
|
|
emit_changed();
|
|
call_deferred("render");
|
|
}
|
|
|
|
MMMaterial::MMMaterial() {
|
|
USE_THREADS = true;
|
|
image_size = Vector2(128, 128);
|
|
initialized = false;
|
|
rendering = false;
|
|
queued_render = false;
|
|
job.instance();
|
|
}
|
|
|
|
MMMaterial::~MMMaterial() {
|
|
}
|
|
|
|
void MMMaterial::_bind_methods() {
|
|
ClassDB::bind_method(D_METHOD("get_image_size"), &MMMaterial::get_image_size);
|
|
ClassDB::bind_method(D_METHOD("set_image_size", "value"), &MMMaterial::set_image_size);
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "image_size"), "set_image_size", "get_image_size");
|
|
|
|
ClassDB::bind_method(D_METHOD("get_nodes"), &MMMaterial::get_nodes);
|
|
ClassDB::bind_method(D_METHOD("set_nodes", "value"), &MMMaterial::set_nodes);
|
|
ADD_PROPERTY(PropertyInfo(Variant::ARRAY, "nodes", PROPERTY_HINT_NONE, "23/20:MMNode", PROPERTY_USAGE_DEFAULT, "MMNode"), "set_nodes", "get_nodes");
|
|
|
|
ClassDB::bind_method(D_METHOD("get_initialized"), &MMMaterial::get_initialized);
|
|
ClassDB::bind_method(D_METHOD("set_initialized", "value"), &MMMaterial::set_initialized);
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "initialized", PROPERTY_HINT_NONE, "", 0), "set_initialized", "get_initialized");
|
|
|
|
ClassDB::bind_method(D_METHOD("get_rendering"), &MMMaterial::get_rendering);
|
|
ClassDB::bind_method(D_METHOD("set_rendering", "value"), &MMMaterial::set_rendering);
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "rendering", PROPERTY_HINT_NONE, "", 0), "set_rendering", "get_rendering");
|
|
|
|
ClassDB::bind_method(D_METHOD("get_queued_render"), &MMMaterial::get_queued_render);
|
|
ClassDB::bind_method(D_METHOD("set_queued_render", "value"), &MMMaterial::set_queued_render);
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "queued_render", PROPERTY_HINT_NONE, "", 0), "set_queued_render", "get_queued_render");
|
|
|
|
ClassDB::bind_method(D_METHOD("initialize"), &MMMaterial::initialize);
|
|
|
|
ClassDB::bind_method(D_METHOD("add_node", "node"), &MMMaterial::add_node);
|
|
ClassDB::bind_method(D_METHOD("remove_node", "node"), &MMMaterial::remove_node);
|
|
|
|
ClassDB::bind_method(D_METHOD("render"), &MMMaterial::render);
|
|
ClassDB::bind_method(D_METHOD("render_non_threaded"), &MMMaterial::render_non_threaded);
|
|
ClassDB::bind_method(D_METHOD("render_threaded"), &MMMaterial::render_threaded);
|
|
|
|
ClassDB::bind_method(D_METHOD("_thread_func"), &MMMaterial::_thread_func);
|
|
|
|
ClassDB::bind_method(D_METHOD("cancel_render_and_wait"), &MMMaterial::cancel_render_and_wait);
|
|
ClassDB::bind_method(D_METHOD("on_node_changed"), &MMMaterial::on_node_changed);
|
|
}
|