2019-04-20 20:25:23 +02:00
|
|
|
/*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** \file
|
|
|
|
* \ingroup blenloader
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* allow readfile to use deprecated functionality */
|
|
|
|
#define DNA_DEPRECATED_ALLOW
|
|
|
|
|
|
|
|
#include <float.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "BLI_math.h"
|
|
|
|
#include "BLI_string.h"
|
2019-08-18 11:16:04 +02:00
|
|
|
#include "BLI_listbase.h"
|
2019-04-20 20:25:23 +02:00
|
|
|
#include "BLI_utildefines.h"
|
|
|
|
|
|
|
|
#include "DNA_color_types.h"
|
|
|
|
#include "DNA_light_types.h"
|
|
|
|
#include "DNA_node_types.h"
|
|
|
|
#include "DNA_particle_types.h"
|
2019-05-17 16:57:31 +02:00
|
|
|
#include "DNA_camera_types.h"
|
2019-09-04 23:17:13 +02:00
|
|
|
#include "DNA_anim_types.h"
|
2019-04-20 20:25:23 +02:00
|
|
|
|
|
|
|
#include "BKE_colortools.h"
|
2019-09-04 23:17:13 +02:00
|
|
|
#include "BKE_animsys.h"
|
2019-04-20 20:25:23 +02:00
|
|
|
#include "BKE_idprop.h"
|
|
|
|
#include "BKE_main.h"
|
|
|
|
#include "BKE_node.h"
|
|
|
|
|
2019-09-04 23:17:13 +02:00
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
2019-04-23 13:56:30 +02:00
|
|
|
#include "IMB_colormanagement.h"
|
|
|
|
|
2019-04-20 20:25:23 +02:00
|
|
|
#include "BLO_readfile.h"
|
|
|
|
#include "readfile.h"
|
|
|
|
|
Shading: Add more operators to Vector Math node.
Add Multiply, Divide, Project, Reflect, Distance, Length, Scale, Snap,
Floor, Ceil, Modulo, Fraction, Absolute, Minimum, and Maximum operators
to the Vector Math node. The Value output has been removed from operators
whose output is a vector, and the other way around. All of those removals
has been handled properly in versioning code.
The patch doesn't include tests for the new operators. Tests will be added
in a later patch.
Reviewers: brecht, JacquesLucke
Differential Revision: https://developer.blender.org/D5523
2019-08-21 19:36:33 +02:00
|
|
|
static bool socket_is_used(bNodeSocket *sock)
|
|
|
|
{
|
|
|
|
return sock->flag & SOCK_IN_USE;
|
|
|
|
}
|
|
|
|
|
2019-04-20 20:25:23 +02:00
|
|
|
static float *cycles_node_socket_float_value(bNodeSocket *socket)
|
|
|
|
{
|
|
|
|
bNodeSocketValueFloat *socket_data = socket->default_value;
|
|
|
|
return &socket_data->value;
|
|
|
|
}
|
|
|
|
|
2019-05-12 13:41:23 +02:00
|
|
|
static float *cycles_node_socket_rgba_value(bNodeSocket *socket)
|
|
|
|
{
|
|
|
|
bNodeSocketValueRGBA *socket_data = socket->default_value;
|
|
|
|
return socket_data->value;
|
|
|
|
}
|
|
|
|
|
Shading: Add more operators to Vector Math node.
Add Multiply, Divide, Project, Reflect, Distance, Length, Scale, Snap,
Floor, Ceil, Modulo, Fraction, Absolute, Minimum, and Maximum operators
to the Vector Math node. The Value output has been removed from operators
whose output is a vector, and the other way around. All of those removals
has been handled properly in versioning code.
The patch doesn't include tests for the new operators. Tests will be added
in a later patch.
Reviewers: brecht, JacquesLucke
Differential Revision: https://developer.blender.org/D5523
2019-08-21 19:36:33 +02:00
|
|
|
static float *cycles_node_socket_vector_value(bNodeSocket *socket)
|
|
|
|
{
|
|
|
|
bNodeSocketValueVector *socket_data = socket->default_value;
|
|
|
|
return socket_data->value;
|
|
|
|
}
|
|
|
|
|
2019-04-20 20:25:23 +02:00
|
|
|
static IDProperty *cycles_properties_from_ID(ID *id)
|
|
|
|
{
|
|
|
|
IDProperty *idprop = IDP_GetProperties(id, false);
|
|
|
|
return (idprop) ? IDP_GetPropertyTypeFromGroup(idprop, "cycles", IDP_GROUP) : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float cycles_property_float(IDProperty *idprop, const char *name, float default_value)
|
|
|
|
{
|
|
|
|
IDProperty *prop = IDP_GetPropertyTypeFromGroup(idprop, name, IDP_FLOAT);
|
|
|
|
return (prop) ? IDP_Float(prop) : default_value;
|
|
|
|
}
|
|
|
|
|
2019-05-17 16:57:31 +02:00
|
|
|
static float cycles_property_int(IDProperty *idprop, const char *name, int default_value)
|
|
|
|
{
|
|
|
|
IDProperty *prop = IDP_GetPropertyTypeFromGroup(idprop, name, IDP_INT);
|
|
|
|
return (prop) ? IDP_Int(prop) : default_value;
|
|
|
|
}
|
|
|
|
|
2019-04-20 20:25:23 +02:00
|
|
|
static bool cycles_property_boolean(IDProperty *idprop, const char *name, bool default_value)
|
|
|
|
{
|
|
|
|
IDProperty *prop = IDP_GetPropertyTypeFromGroup(idprop, name, IDP_INT);
|
|
|
|
return (prop) ? IDP_Int(prop) : default_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void displacement_node_insert(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
/* Iterate backwards from end so we don't encounter newly added links. */
|
|
|
|
bNodeLink *prevlink;
|
|
|
|
for (bNodeLink *link = ntree->links.last; link; link = prevlink) {
|
|
|
|
prevlink = link->prev;
|
|
|
|
|
|
|
|
/* Detect link to replace. */
|
|
|
|
bNode *fromnode = link->fromnode;
|
|
|
|
bNodeSocket *fromsock = link->fromsock;
|
|
|
|
bNode *tonode = link->tonode;
|
|
|
|
bNodeSocket *tosock = link->tosock;
|
|
|
|
|
|
|
|
if (!(tonode->type == SH_NODE_OUTPUT_MATERIAL && fromnode->type != SH_NODE_DISPLACEMENT &&
|
|
|
|
STREQ(tosock->identifier, "Displacement"))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace link with displacement node. */
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
|
|
|
|
/* Add displacement node. */
|
|
|
|
bNode *node = nodeAddStaticNode(NULL, ntree, SH_NODE_DISPLACEMENT);
|
|
|
|
node->locx = 0.5f * (fromnode->locx + tonode->locx);
|
|
|
|
node->locy = 0.5f * (fromnode->locy + tonode->locy);
|
|
|
|
|
|
|
|
bNodeSocket *scale_socket = nodeFindSocket(node, SOCK_IN, "Scale");
|
|
|
|
bNodeSocket *midlevel_socket = nodeFindSocket(node, SOCK_IN, "Midlevel");
|
|
|
|
bNodeSocket *height_socket = nodeFindSocket(node, SOCK_IN, "Height");
|
|
|
|
bNodeSocket *displacement_socket = nodeFindSocket(node, SOCK_OUT, "Displacement");
|
|
|
|
|
|
|
|
/* Set default values for compatibility. */
|
|
|
|
*cycles_node_socket_float_value(scale_socket) = 0.1f;
|
|
|
|
*cycles_node_socket_float_value(midlevel_socket) = 0.0f;
|
|
|
|
|
|
|
|
/* Link to input and material output node. */
|
|
|
|
nodeAddLink(ntree, fromnode, fromsock, node, height_socket);
|
|
|
|
nodeAddLink(ntree, node, displacement_socket, tonode, tosock);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void displacement_principled_nodes(bNode *node)
|
|
|
|
{
|
|
|
|
if (node->type == SH_NODE_DISPLACEMENT) {
|
|
|
|
if (node->custom1 != SHD_SPACE_WORLD) {
|
|
|
|
node->custom1 = SHD_SPACE_OBJECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (node->type == SH_NODE_BSDF_PRINCIPLED) {
|
|
|
|
if (node->custom2 != SHD_SUBSURFACE_RANDOM_WALK) {
|
|
|
|
node->custom2 = SHD_SUBSURFACE_BURLEY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool node_has_roughness(bNode *node)
|
|
|
|
{
|
|
|
|
return ELEM(node->type,
|
|
|
|
SH_NODE_BSDF_ANISOTROPIC,
|
|
|
|
SH_NODE_BSDF_GLASS,
|
|
|
|
SH_NODE_BSDF_GLOSSY,
|
|
|
|
SH_NODE_BSDF_REFRACTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void square_roughness_node_insert(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
/* Update default values */
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node_has_roughness(node)) {
|
|
|
|
bNodeSocket *roughness_input = nodeFindSocket(node, SOCK_IN, "Roughness");
|
|
|
|
float *roughness_value = cycles_node_socket_float_value(roughness_input);
|
|
|
|
*roughness_value = sqrtf(max_ff(*roughness_value, 0.0f));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Iterate backwards from end so we don't encounter newly added links. */
|
|
|
|
bNodeLink *prevlink;
|
|
|
|
for (bNodeLink *link = ntree->links.last; link; link = prevlink) {
|
|
|
|
prevlink = link->prev;
|
|
|
|
|
|
|
|
/* Detect link to replace. */
|
|
|
|
bNode *fromnode = link->fromnode;
|
|
|
|
bNodeSocket *fromsock = link->fromsock;
|
|
|
|
bNode *tonode = link->tonode;
|
|
|
|
bNodeSocket *tosock = link->tosock;
|
|
|
|
|
|
|
|
if (!(node_has_roughness(tonode) && STREQ(tosock->identifier, "Roughness"))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace links with sqrt node */
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
|
|
|
|
/* Add sqrt node. */
|
|
|
|
bNode *node = nodeAddStaticNode(NULL, ntree, SH_NODE_MATH);
|
2019-08-18 11:16:04 +02:00
|
|
|
node->custom1 = NODE_MATH_POWER;
|
2019-04-20 20:25:23 +02:00
|
|
|
node->locx = 0.5f * (fromnode->locx + tonode->locx);
|
|
|
|
node->locy = 0.5f * (fromnode->locy + tonode->locy);
|
|
|
|
|
|
|
|
/* Link to input and material output node. */
|
|
|
|
*cycles_node_socket_float_value(node->inputs.last) = 0.5f;
|
|
|
|
nodeAddLink(ntree, fromnode, fromsock, node, node->inputs.first);
|
|
|
|
nodeAddLink(ntree, node, node->outputs.first, tonode, tosock);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mapping_node_order_flip(bNode *node)
|
|
|
|
{
|
|
|
|
/* Flip euler order of mapping shader node */
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_MAPPING && node->storage) {
|
2019-04-20 20:25:23 +02:00
|
|
|
TexMapping *texmap = node->storage;
|
|
|
|
|
|
|
|
float quat[4];
|
|
|
|
eulO_to_quat(quat, texmap->rot, EULER_ORDER_ZYX);
|
|
|
|
quat_to_eulO(texmap->rot, EULER_ORDER_XYZ, quat);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vector_curve_node_remap(bNode *node)
|
|
|
|
{
|
|
|
|
/* Remap values of vector curve node from normalized to absolute values */
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_CURVE_VEC && node->storage) {
|
2019-04-20 20:25:23 +02:00
|
|
|
CurveMapping *mapping = node->storage;
|
|
|
|
mapping->flag &= ~CUMA_DO_CLIP;
|
|
|
|
|
|
|
|
for (int curve_index = 0; curve_index < CM_TOT; curve_index++) {
|
|
|
|
CurveMap *cm = &mapping->cm[curve_index];
|
|
|
|
if (cm->curve) {
|
2019-06-25 11:22:23 +10:00
|
|
|
for (int i = 0; i < cm->totpoint; i++) {
|
2019-04-20 20:25:23 +02:00
|
|
|
cm->curve[i].x = (cm->curve[i].x * 2.0f) - 1.0f;
|
|
|
|
cm->curve[i].y = (cm->curve[i].y - 0.5f) * 2.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-07 03:21:55 +10:00
|
|
|
BKE_curvemapping_changed_all(mapping);
|
2019-04-20 20:25:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ambient_occlusion_node_relink(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
/* Set default values. */
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_AMBIENT_OCCLUSION) {
|
|
|
|
node->custom1 = 1; /* samples */
|
|
|
|
node->custom2 &= ~SHD_AO_LOCAL;
|
|
|
|
|
|
|
|
bNodeSocket *distance_socket = nodeFindSocket(node, SOCK_IN, "Distance");
|
|
|
|
*cycles_node_socket_float_value(distance_socket) = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Iterate backwards from end so we don't encounter newly added links. */
|
|
|
|
bNodeLink *prevlink;
|
|
|
|
for (bNodeLink *link = ntree->links.last; link; link = prevlink) {
|
|
|
|
prevlink = link->prev;
|
|
|
|
|
|
|
|
/* Detect link to replace. */
|
|
|
|
bNode *fromnode = link->fromnode;
|
|
|
|
bNode *tonode = link->tonode;
|
|
|
|
bNodeSocket *tosock = link->tosock;
|
|
|
|
|
|
|
|
if (!(fromnode->type == SH_NODE_AMBIENT_OCCLUSION)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace links with color socket. */
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
bNodeSocket *color_socket = nodeFindSocket(fromnode, SOCK_OUT, "Color");
|
|
|
|
nodeAddLink(ntree, fromnode, color_socket, tonode, tosock);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-23 13:56:30 +02:00
|
|
|
static void image_node_colorspace(bNode *node)
|
|
|
|
{
|
|
|
|
if (node->id == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int color_space;
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_IMAGE && node->storage) {
|
2019-04-23 13:56:30 +02:00
|
|
|
NodeTexImage *tex = node->storage;
|
|
|
|
color_space = tex->color_space;
|
|
|
|
}
|
2019-11-06 20:03:22 +01:00
|
|
|
else if (node->type == SH_NODE_TEX_ENVIRONMENT && node->storage) {
|
2019-04-23 13:56:30 +02:00
|
|
|
NodeTexEnvironment *tex = node->storage;
|
|
|
|
color_space = tex->color_space;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const int SHD_COLORSPACE_NONE = 0;
|
|
|
|
Image *image = (Image *)node->id;
|
|
|
|
if (color_space == SHD_COLORSPACE_NONE) {
|
|
|
|
STRNCPY(image->colorspace_settings.name,
|
|
|
|
IMB_colormanagement_role_colorspace_name_get(COLOR_ROLE_DATA));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-12 13:41:23 +02:00
|
|
|
static void light_emission_node_to_energy(Light *light, float *energy, float color[3])
|
|
|
|
{
|
|
|
|
*energy = 1.0;
|
|
|
|
copy_v3_fl(color, 1.0f);
|
|
|
|
|
|
|
|
/* If nodetree has animation or drivers, don't try to convert. */
|
|
|
|
bNodeTree *ntree = light->nodetree;
|
|
|
|
if (ntree == NULL || ntree->adt) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find emission node */
|
|
|
|
bNode *output_node = ntreeShaderOutputNode(ntree, SHD_OUTPUT_CYCLES);
|
|
|
|
if (output_node == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bNode *emission_node = NULL;
|
|
|
|
for (bNodeLink *link = ntree->links.first; link; link = link->next) {
|
|
|
|
if (link->tonode == output_node && link->fromnode->type == SH_NODE_EMISSION) {
|
|
|
|
emission_node = link->fromnode;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (emission_node == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Don't convert if anything is linked */
|
|
|
|
bNodeSocket *strength_socket = nodeFindSocket(emission_node, SOCK_IN, "Strength");
|
|
|
|
bNodeSocket *color_socket = nodeFindSocket(emission_node, SOCK_IN, "Color");
|
|
|
|
|
|
|
|
if ((strength_socket->flag & SOCK_IN_USE) || (color_socket->flag & SOCK_IN_USE)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
float *strength_value = cycles_node_socket_float_value(strength_socket);
|
|
|
|
float *color_value = cycles_node_socket_rgba_value(color_socket);
|
|
|
|
|
|
|
|
*energy = *strength_value;
|
|
|
|
copy_v3_v3(color, color_value);
|
|
|
|
|
|
|
|
*strength_value = 1.0f;
|
|
|
|
copy_v4_fl(color_value, 1.0f);
|
|
|
|
light->use_nodes = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void light_emission_unify(Light *light, const char *engine)
|
|
|
|
{
|
|
|
|
if (light->type != LA_SUN) {
|
|
|
|
light->energy *= 100.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Attempt to extract constant energy and color from nodes. */
|
|
|
|
bool use_nodes = light->use_nodes;
|
|
|
|
float energy, color[3];
|
|
|
|
light_emission_node_to_energy(light, &energy, color);
|
|
|
|
|
|
|
|
if (STREQ(engine, "CYCLES")) {
|
|
|
|
if (use_nodes) {
|
|
|
|
/* Energy extracted from nodes */
|
|
|
|
light->energy = energy;
|
|
|
|
copy_v3_v3(&light->r, color);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Default cycles multipliers if there are no nodes */
|
|
|
|
if (light->type == LA_SUN) {
|
|
|
|
light->energy = 1.0f;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
light->energy = 100.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Disable nodes if scene was configured for Eevee */
|
|
|
|
light->use_nodes = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-18 11:16:04 +02:00
|
|
|
/* The B input of the Math node is no longer used for single-operand operators.
|
|
|
|
* Previously, if the B input was linked and the A input was not, the B input
|
|
|
|
* was used as the input of the operator. To correct this, we move the link
|
|
|
|
* from B to A if B is linked and A is not.
|
|
|
|
*/
|
|
|
|
static void update_math_node_single_operand_operators(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_MATH) {
|
|
|
|
if (ELEM(node->custom1,
|
|
|
|
NODE_MATH_SQRT,
|
|
|
|
NODE_MATH_CEIL,
|
|
|
|
NODE_MATH_SINE,
|
|
|
|
NODE_MATH_ROUND,
|
|
|
|
NODE_MATH_FLOOR,
|
|
|
|
NODE_MATH_COSINE,
|
|
|
|
NODE_MATH_ARCSINE,
|
|
|
|
NODE_MATH_TANGENT,
|
|
|
|
NODE_MATH_ABSOLUTE,
|
|
|
|
NODE_MATH_FRACTION,
|
|
|
|
NODE_MATH_ARCCOSINE,
|
|
|
|
NODE_MATH_ARCTANGENT)) {
|
|
|
|
bNodeSocket *sockA = BLI_findlink(&node->inputs, 0);
|
|
|
|
bNodeSocket *sockB = BLI_findlink(&node->inputs, 1);
|
|
|
|
if (!sockA->link && sockB->link) {
|
|
|
|
nodeAddLink(ntree, sockB->link->fromnode, sockB->link->fromsock, node, sockA);
|
|
|
|
nodeRemLink(ntree, sockB->link);
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Shading: Add more operators to Vector Math node.
Add Multiply, Divide, Project, Reflect, Distance, Length, Scale, Snap,
Floor, Ceil, Modulo, Fraction, Absolute, Minimum, and Maximum operators
to the Vector Math node. The Value output has been removed from operators
whose output is a vector, and the other way around. All of those removals
has been handled properly in versioning code.
The patch doesn't include tests for the new operators. Tests will be added
in a later patch.
Reviewers: brecht, JacquesLucke
Differential Revision: https://developer.blender.org/D5523
2019-08-21 19:36:33 +02:00
|
|
|
/* The Value output of the Vector Math node is no longer available in the Add
|
|
|
|
* and Subtract operators. Previously, this Value output was computed from the
|
|
|
|
* Vector output V as follows:
|
|
|
|
*
|
|
|
|
* Value = (abs(V.x) + abs(V.y) + abs(V.z)) / 3
|
|
|
|
*
|
|
|
|
* Or more compactly using vector operators:
|
|
|
|
*
|
|
|
|
* Value = dot(abs(V), (1 / 3, 1 / 3, 1 / 3))
|
|
|
|
*
|
|
|
|
* To correct this, if the Value output was used, we are going to compute
|
|
|
|
* it using the second equation by adding an absolute and a dot node, and
|
|
|
|
* then connect them appropriately.
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_add_and_subtract_operators(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
bNodeSocket *sockOutValue = nodeFindSocket(node, SOCK_OUT, "Value");
|
|
|
|
if (socket_is_used(sockOutValue) &&
|
|
|
|
ELEM(node->custom1, NODE_VECTOR_MATH_ADD, NODE_VECTOR_MATH_SUBTRACT)) {
|
|
|
|
|
|
|
|
bNode *absNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
absNode->custom1 = NODE_VECTOR_MATH_ABSOLUTE;
|
|
|
|
absNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
absNode->locy = node->locy;
|
|
|
|
|
|
|
|
bNode *dotNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
dotNode->custom1 = NODE_VECTOR_MATH_DOT_PRODUCT;
|
|
|
|
dotNode->locx = absNode->locx + absNode->width + 20.0f;
|
|
|
|
dotNode->locy = absNode->locy;
|
|
|
|
bNodeSocket *sockDotB = BLI_findlink(&dotNode->inputs, 1);
|
|
|
|
bNodeSocket *sockDotOutValue = nodeFindSocket(dotNode, SOCK_OUT, "Value");
|
|
|
|
copy_v3_fl(cycles_node_socket_vector_value(sockDotB), 1 / 3.0f);
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutValue) {
|
|
|
|
nodeAddLink(ntree, dotNode, sockDotOutValue, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeSocket *sockAbsA = BLI_findlink(&absNode->inputs, 0);
|
|
|
|
bNodeSocket *sockDotA = BLI_findlink(&dotNode->inputs, 0);
|
|
|
|
bNodeSocket *sockOutVector = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
bNodeSocket *sockAbsOutVector = nodeFindSocket(absNode, SOCK_OUT, "Vector");
|
|
|
|
|
|
|
|
nodeAddLink(ntree, node, sockOutVector, absNode, sockAbsA);
|
|
|
|
nodeAddLink(ntree, absNode, sockAbsOutVector, dotNode, sockDotA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Vector output of the Vector Math node is no longer available in the Dot
|
|
|
|
* Product operator. Previously, this Vector was always zero initialized. To
|
|
|
|
* correct this, we zero out any socket the Vector Output was connected to.
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_dot_product_operator(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
bNodeSocket *sockOutVector = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
if (socket_is_used(sockOutVector) && node->custom1 == NODE_VECTOR_MATH_DOT_PRODUCT) {
|
|
|
|
LISTBASE_FOREACH_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutVector) {
|
|
|
|
switch (link->tosock->type) {
|
|
|
|
case SOCK_FLOAT:
|
|
|
|
*cycles_node_socket_float_value(link->tosock) = 0.0f;
|
|
|
|
break;
|
|
|
|
case SOCK_VECTOR:
|
|
|
|
copy_v3_fl(cycles_node_socket_vector_value(link->tosock), 0.0f);
|
|
|
|
break;
|
|
|
|
case SOCK_RGBA:
|
|
|
|
copy_v4_fl(cycles_node_socket_rgba_value(link->tosock), 0.0f);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Previously, the Vector output of the cross product operator was normalized.
|
|
|
|
* To correct this, a Normalize node is added to normalize the output if used.
|
|
|
|
* Moreover, the Value output was removed. This Value was equal to the length
|
|
|
|
* of the cross product. To correct this, a Length node is added if needed.
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_cross_product_operator(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
if (node->custom1 == NODE_VECTOR_MATH_CROSS_PRODUCT) {
|
|
|
|
bNodeSocket *sockOutVector = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
if (socket_is_used(sockOutVector)) {
|
|
|
|
bNode *normalizeNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
normalizeNode->custom1 = NODE_VECTOR_MATH_NORMALIZE;
|
|
|
|
normalizeNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
normalizeNode->locy = node->locy;
|
|
|
|
bNodeSocket *sockNormalizeOut = nodeFindSocket(normalizeNode, SOCK_OUT, "Vector");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutVector) {
|
|
|
|
nodeAddLink(ntree, normalizeNode, sockNormalizeOut, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockNormalizeA = BLI_findlink(&normalizeNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockOutVector, normalizeNode, sockNormalizeA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeSocket *sockOutValue = nodeFindSocket(node, SOCK_OUT, "Value");
|
|
|
|
if (socket_is_used(sockOutValue)) {
|
|
|
|
bNode *lengthNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
lengthNode->custom1 = NODE_VECTOR_MATH_LENGTH;
|
|
|
|
lengthNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
if (socket_is_used(sockOutVector)) {
|
|
|
|
lengthNode->locy = node->locy - lengthNode->height - 20.0f;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
lengthNode->locy = node->locy;
|
|
|
|
}
|
|
|
|
bNodeSocket *sockLengthOut = nodeFindSocket(lengthNode, SOCK_OUT, "Value");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutValue) {
|
|
|
|
nodeAddLink(ntree, lengthNode, sockLengthOut, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockLengthA = BLI_findlink(&lengthNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockOutVector, lengthNode, sockLengthA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Value output of the Vector Math node is no longer available in the
|
|
|
|
* Normalize operator. This Value output was equal to the length of the
|
|
|
|
* the input vector A. To correct this, we either add a Length node or
|
|
|
|
* convert the Normalize node into a Length node, depending on if the
|
|
|
|
* Vector output is needed.
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_normalize_operator(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
bNodeSocket *sockOutValue = nodeFindSocket(node, SOCK_OUT, "Value");
|
|
|
|
if (node->custom1 == NODE_VECTOR_MATH_NORMALIZE && socket_is_used(sockOutValue)) {
|
|
|
|
bNodeSocket *sockOutVector = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
if (socket_is_used(sockOutVector)) {
|
|
|
|
bNode *lengthNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
lengthNode->custom1 = NODE_VECTOR_MATH_LENGTH;
|
|
|
|
lengthNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
lengthNode->locy = node->locy;
|
|
|
|
bNodeSocket *sockLengthValue = nodeFindSocket(lengthNode, SOCK_OUT, "Value");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutValue) {
|
|
|
|
nodeAddLink(ntree, lengthNode, sockLengthValue, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockA = BLI_findlink(&node->inputs, 0);
|
|
|
|
bNodeSocket *sockLengthA = BLI_findlink(&lengthNode->inputs, 0);
|
|
|
|
if (sockA->link) {
|
|
|
|
bNodeLink *link = sockA->link;
|
|
|
|
nodeAddLink(ntree, link->fromnode, link->fromsock, lengthNode, sockLengthA);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockLengthA),
|
|
|
|
cycles_node_socket_vector_value(sockA));
|
|
|
|
}
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
node->custom1 = NODE_VECTOR_MATH_LENGTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Vector Math operator types didn't have an enum, but rather, their
|
|
|
|
* values were hard coded into the code. After the enum was created and
|
|
|
|
* after more vector operators were added, the hard coded values needs
|
|
|
|
* to be remapped to their correct enum values. To fix this, we remap
|
|
|
|
* the values according to the following rules:
|
|
|
|
*
|
|
|
|
* Dot Product Operator : 3 -> 7
|
|
|
|
* Normalize Operator : 5 -> 11
|
|
|
|
*
|
|
|
|
* Additionally, since the Average operator was removed, it is assigned
|
|
|
|
* a value of -1 just to be identified later in the versioning code:
|
|
|
|
*
|
|
|
|
* Average Operator : 2 -> -1
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_operators_enum_mapping(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
switch (node->custom1) {
|
|
|
|
case 2:
|
|
|
|
node->custom1 = -1;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
node->custom1 = 7;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
node->custom1 = 11;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Average operator is no longer available in the Vector Math node.
|
|
|
|
* The Vector output was equal to the normalized sum of input vectors while
|
|
|
|
* the Value output was equal to the length of the sum of input vectors.
|
|
|
|
* To correct this, we convert the node into an Add node and add a length
|
|
|
|
* node or a normalize node if needed.
|
|
|
|
*/
|
|
|
|
static void update_vector_math_node_average_operator(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_VECTOR_MATH) {
|
|
|
|
/* See update_vector_math_node_operators_enum_mapping. */
|
|
|
|
if (node->custom1 == -1) {
|
|
|
|
node->custom1 = NODE_VECTOR_MATH_ADD;
|
|
|
|
bNodeSocket *sockOutVector = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
if (socket_is_used(sockOutVector)) {
|
|
|
|
bNode *normalizeNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
normalizeNode->custom1 = NODE_VECTOR_MATH_NORMALIZE;
|
|
|
|
normalizeNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
normalizeNode->locy = node->locy;
|
|
|
|
bNodeSocket *sockNormalizeOut = nodeFindSocket(normalizeNode, SOCK_OUT, "Vector");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutVector) {
|
|
|
|
nodeAddLink(ntree, normalizeNode, sockNormalizeOut, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockNormalizeA = BLI_findlink(&normalizeNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockOutVector, normalizeNode, sockNormalizeA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeSocket *sockOutValue = nodeFindSocket(node, SOCK_OUT, "Value");
|
|
|
|
if (socket_is_used(sockOutValue)) {
|
|
|
|
bNode *lengthNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
lengthNode->custom1 = NODE_VECTOR_MATH_LENGTH;
|
|
|
|
lengthNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
if (socket_is_used(sockOutVector)) {
|
|
|
|
lengthNode->locy = node->locy - lengthNode->height - 20.0f;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
lengthNode->locy = node->locy;
|
|
|
|
}
|
|
|
|
bNodeSocket *sockLengthOut = nodeFindSocket(lengthNode, SOCK_OUT, "Value");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockOutValue) {
|
|
|
|
nodeAddLink(ntree, lengthNode, sockLengthOut, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockLengthA = BLI_findlink(&lengthNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockOutVector, lengthNode, sockLengthA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-04 17:54:32 +02:00
|
|
|
/* The Noise node now have a dimension property. This property should be
|
|
|
|
* initialized to 3 by default.
|
|
|
|
*/
|
|
|
|
static void update_noise_node_dimensions(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_NOISE && node->storage) {
|
2019-09-04 17:54:32 +02:00
|
|
|
NodeTexNoise *tex = (NodeTexNoise *)node->storage;
|
|
|
|
tex->dimensions = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-26 13:45:40 +02:00
|
|
|
/* This structure is only used to pass data to
|
|
|
|
* update_mapping_node_fcurve_rna_path_callback.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
char *nodePath;
|
|
|
|
bNode *minimumNode;
|
|
|
|
bNode *maximumNode;
|
|
|
|
} MappingNodeFCurveCallbackData;
|
|
|
|
|
|
|
|
/* This callback function is used by update_mapping_node_inputs_and_properties.
|
|
|
|
* It is executed on every fcurve in the nodetree id updating its RNA paths. The
|
|
|
|
* paths needs to be updated because the node properties became inputs.
|
|
|
|
*
|
|
|
|
* nodes["Mapping"].translation --> nodes["Mapping"].inputs[1].default_value
|
|
|
|
* nodes["Mapping"].rotation --> nodes["Mapping"].inputs[2].default_value
|
|
|
|
* nodes["Mapping"].scale --> nodes["Mapping"].inputs[3].default_value
|
|
|
|
* nodes["Mapping"].max --> nodes["Maximum"].inputs[1].default_value
|
|
|
|
* nodes["Mapping"].min --> nodes["Minimum"].inputs[1].default_value
|
|
|
|
*
|
|
|
|
* The fcurve can be that of any node or property in the nodetree, so we only
|
|
|
|
* update if the rna path starts with the rna path of the mapping node and
|
|
|
|
* doesn't end with "default_value", that is, not the Vector input.
|
|
|
|
*/
|
|
|
|
static void update_mapping_node_fcurve_rna_path_callback(ID *UNUSED(id),
|
|
|
|
FCurve *fcurve,
|
|
|
|
void *_data)
|
|
|
|
{
|
|
|
|
MappingNodeFCurveCallbackData *data = (MappingNodeFCurveCallbackData *)_data;
|
|
|
|
if (!STRPREFIX(fcurve->rna_path, data->nodePath) ||
|
|
|
|
BLI_str_endswith(fcurve->rna_path, "default_value")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
char *old_fcurve_rna_path = fcurve->rna_path;
|
|
|
|
|
|
|
|
if (BLI_str_endswith(old_fcurve_rna_path, "translation")) {
|
|
|
|
fcurve->rna_path = BLI_sprintfN("%s.%s", data->nodePath, "inputs[1].default_value");
|
|
|
|
}
|
|
|
|
else if (BLI_str_endswith(old_fcurve_rna_path, "rotation")) {
|
|
|
|
fcurve->rna_path = BLI_sprintfN("%s.%s", data->nodePath, "inputs[2].default_value");
|
|
|
|
}
|
|
|
|
else if (BLI_str_endswith(old_fcurve_rna_path, "scale")) {
|
|
|
|
fcurve->rna_path = BLI_sprintfN("%s.%s", data->nodePath, "inputs[3].default_value");
|
|
|
|
}
|
|
|
|
else if (data->minimumNode && BLI_str_endswith(old_fcurve_rna_path, "max")) {
|
|
|
|
fcurve->rna_path = BLI_sprintfN(
|
|
|
|
"nodes[\"%s\"].%s", data->minimumNode->name, "inputs[1].default_value");
|
|
|
|
}
|
|
|
|
else if (data->maximumNode && BLI_str_endswith(old_fcurve_rna_path, "min")) {
|
|
|
|
fcurve->rna_path = BLI_sprintfN(
|
|
|
|
"nodes[\"%s\"].%s", data->maximumNode->name, "inputs[1].default_value");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fcurve->rna_path != old_fcurve_rna_path) {
|
|
|
|
MEM_freeN(old_fcurve_rna_path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-04 23:17:13 +02:00
|
|
|
/* The Mapping node has been rewritten to support dynamic inputs. Previously,
|
|
|
|
* the transformation information was stored in a TexMapping struct in the
|
|
|
|
* node->storage member of bNode. Currently, the transformation information
|
|
|
|
* is stored in input sockets. To correct this, we transfer the information
|
|
|
|
* from the TexMapping struct to the input sockets.
|
|
|
|
*
|
|
|
|
* Additionally, the Minimum and Maximum properties are no longer available
|
|
|
|
* in the node. To correct this, a Vector Minimum and/or a Vector Maximum
|
|
|
|
* nodes are added if needed.
|
|
|
|
*
|
|
|
|
* Finally, the TexMapping struct is freed and node->storage is set to NULL.
|
|
|
|
*
|
|
|
|
* Since the RNA paths of the properties changed, we also have to update the
|
|
|
|
* rna_path of the FCurves if they exist. To do that, we loop over FCurves
|
|
|
|
* and check if they control a property of the node, if they do, we update
|
2019-09-07 21:08:20 +10:00
|
|
|
* the path to be that of the corresponding socket in the node or the added
|
2019-09-04 23:17:13 +02:00
|
|
|
* minimum/maximum node.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void update_mapping_node_inputs_and_properties(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-09-09 11:11:16 +02:00
|
|
|
/* If node->storage is NULL, then conversion has already taken place.
|
|
|
|
* This can happen if a file with the new mapping node [saved from (2, 81, 8) or newer]
|
|
|
|
* is opened in a blender version prior to (2, 81, 8) and saved from there again. */
|
|
|
|
if (node->type == SH_NODE_MAPPING && node->storage) {
|
2019-09-04 23:17:13 +02:00
|
|
|
TexMapping *mapping = (TexMapping *)node->storage;
|
|
|
|
node->custom1 = mapping->type;
|
|
|
|
node->width = 140.0f;
|
|
|
|
|
|
|
|
bNodeSocket *sockLocation = nodeFindSocket(node, SOCK_IN, "Location");
|
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockLocation), mapping->loc);
|
|
|
|
bNodeSocket *sockRotation = nodeFindSocket(node, SOCK_IN, "Rotation");
|
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockRotation), mapping->rot);
|
|
|
|
bNodeSocket *sockScale = nodeFindSocket(node, SOCK_IN, "Scale");
|
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockScale), mapping->size);
|
|
|
|
|
|
|
|
bNode *maximumNode = NULL;
|
2019-09-17 21:54:09 +02:00
|
|
|
if (mapping->flag & TEXMAP_CLIP_MIN) {
|
2019-09-04 23:17:13 +02:00
|
|
|
maximumNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
maximumNode->custom1 = NODE_VECTOR_MATH_MAXIMUM;
|
2019-09-17 21:54:09 +02:00
|
|
|
if (mapping->flag & TEXMAP_CLIP_MAX) {
|
2019-09-04 23:17:13 +02:00
|
|
|
maximumNode->locx = node->locx + (node->width + 20.0f) * 2.0f;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
maximumNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
}
|
|
|
|
maximumNode->locy = node->locy;
|
|
|
|
bNodeSocket *sockMaximumB = BLI_findlink(&maximumNode->inputs, 1);
|
2019-09-17 21:54:09 +02:00
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockMaximumB), mapping->min);
|
2019-09-04 23:17:13 +02:00
|
|
|
bNodeSocket *sockMappingResult = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockMappingResult) {
|
|
|
|
bNodeSocket *sockMaximumResult = nodeFindSocket(maximumNode, SOCK_OUT, "Vector");
|
|
|
|
nodeAddLink(ntree, maximumNode, sockMaximumResult, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
2019-09-17 21:54:09 +02:00
|
|
|
if (!(mapping->flag & TEXMAP_CLIP_MAX)) {
|
2019-09-04 23:17:13 +02:00
|
|
|
bNodeSocket *sockMaximumA = BLI_findlink(&maximumNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockMappingResult, maximumNode, sockMaximumA);
|
|
|
|
}
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bNode *minimumNode = NULL;
|
2019-09-17 21:54:09 +02:00
|
|
|
if (mapping->flag & TEXMAP_CLIP_MAX) {
|
2019-09-04 23:17:13 +02:00
|
|
|
minimumNode = nodeAddStaticNode(NULL, ntree, SH_NODE_VECTOR_MATH);
|
|
|
|
minimumNode->custom1 = NODE_VECTOR_MATH_MINIMUM;
|
|
|
|
minimumNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
minimumNode->locy = node->locy;
|
|
|
|
bNodeSocket *sockMinimumB = BLI_findlink(&minimumNode->inputs, 1);
|
2019-09-17 21:54:09 +02:00
|
|
|
copy_v3_v3(cycles_node_socket_vector_value(sockMinimumB), mapping->max);
|
2019-09-04 23:17:13 +02:00
|
|
|
|
|
|
|
bNodeSocket *sockMinimumResult = nodeFindSocket(minimumNode, SOCK_OUT, "Vector");
|
|
|
|
bNodeSocket *sockMappingResult = nodeFindSocket(node, SOCK_OUT, "Vector");
|
|
|
|
|
|
|
|
if (maximumNode) {
|
|
|
|
bNodeSocket *sockMaximumA = BLI_findlink(&maximumNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, minimumNode, sockMinimumResult, maximumNode, sockMaximumA);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockMappingResult) {
|
|
|
|
nodeAddLink(ntree, minimumNode, sockMinimumResult, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bNodeSocket *sockMinimumA = BLI_findlink(&minimumNode->inputs, 0);
|
|
|
|
nodeAddLink(ntree, node, sockMappingResult, minimumNode, sockMinimumA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(node->storage);
|
|
|
|
node->storage = NULL;
|
|
|
|
|
2019-11-26 13:45:40 +02:00
|
|
|
char *nodePath = BLI_sprintfN("nodes[\"%s\"]", node->name);
|
|
|
|
MappingNodeFCurveCallbackData data = {nodePath, minimumNode, maximumNode};
|
|
|
|
BKE_fcurves_id_cb(&ntree->id, update_mapping_node_fcurve_rna_path_callback, &data);
|
|
|
|
MEM_freeN(nodePath);
|
2019-09-04 23:17:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-09 21:06:55 +02:00
|
|
|
/* The Musgrave node now has a dimension property. This property should
|
|
|
|
* be initialized to 3 by default.
|
|
|
|
*/
|
|
|
|
static void update_musgrave_node_dimensions(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_MUSGRAVE && node->storage) {
|
2019-09-09 21:06:55 +02:00
|
|
|
NodeTexMusgrave *tex = (NodeTexMusgrave *)node->storage;
|
|
|
|
tex->dimensions = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Color output of the Musgrave node has been removed. Previously, this
|
|
|
|
* output was just equal to the Fac output. To correct this, we move links
|
|
|
|
* from the Color output to the Fac output if they exist.
|
|
|
|
*/
|
|
|
|
static void update_musgrave_node_color_output(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
LISTBASE_FOREACH (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromnode && link->fromnode->type == SH_NODE_TEX_MUSGRAVE) {
|
|
|
|
if (link->fromsock->type == SOCK_RGBA) {
|
|
|
|
link->fromsock = link->fromsock->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-12 13:09:31 +02:00
|
|
|
/* The Voronoi node now have a dimension property. This property should be
|
|
|
|
* initialized to 3 by default.
|
|
|
|
*/
|
|
|
|
static void update_voronoi_node_dimensions(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_VORONOI && node->storage) {
|
2019-09-12 13:09:31 +02:00
|
|
|
NodeTexVoronoi *tex = (NodeTexVoronoi *)node->storage;
|
|
|
|
tex->dimensions = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The F3 and F4 features of the Voronoi node have been removed.
|
|
|
|
* To correct this, we set the feature type to be F2 if it is F3
|
|
|
|
* or F4. The SHD_VORONOI_F3 and SHD_VORONOI_F4 enum values were
|
|
|
|
* 2 and 3 respectively.
|
|
|
|
*/
|
|
|
|
static void update_voronoi_node_f3_and_f4(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_VORONOI && node->storage) {
|
2019-09-12 13:09:31 +02:00
|
|
|
NodeTexVoronoi *tex = (NodeTexVoronoi *)node->storage;
|
|
|
|
if (ELEM(tex->feature, 2, 3)) {
|
|
|
|
tex->feature = SHD_VORONOI_F2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Fac output of the Voronoi node has been removed. Previously, this
|
|
|
|
* output was the voronoi distance in the Intensity mode and the Cell ID
|
|
|
|
* in the Cell mode. To correct this, we update the identifier and name
|
|
|
|
* of the Fac socket such that it gets mapped to the Distance socket.
|
|
|
|
* This is supposed to work with update_voronoi_node_coloring.
|
|
|
|
*/
|
|
|
|
static void update_voronoi_node_fac_output(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_TEX_VORONOI) {
|
|
|
|
bNodeSocket *facOutput = BLI_findlink(&node->outputs, 1);
|
|
|
|
strcpy(facOutput->identifier, "Distance");
|
|
|
|
strcpy(facOutput->name, "Distance");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Crackle feature of the Voronoi node has been removed. Previously,
|
|
|
|
* this feature returned the F2 distance minus the F1 distance. The
|
|
|
|
* crackle feature had an enum value of 4. To fix this we do the
|
|
|
|
* following:
|
|
|
|
*
|
|
|
|
* 1. The node feature is set to F1.
|
|
|
|
* 2. A new Voronoi node is added and its feature is set to F2.
|
|
|
|
* 3. The properties, input values, and connections are copied
|
|
|
|
* from the node to the new Voronoi node so that they match
|
|
|
|
* exactly.
|
|
|
|
* 4. A Subtract node is added.
|
|
|
|
* 5. The outputs of the F1 and F2 voronoi are connected to
|
|
|
|
* the inputs of the subtract node.
|
|
|
|
* 6. The output of the subtract node is connected to the
|
|
|
|
* appropriate sockets.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void update_voronoi_node_crackle(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_VORONOI && node->storage) {
|
2019-09-12 13:09:31 +02:00
|
|
|
NodeTexVoronoi *tex = (NodeTexVoronoi *)node->storage;
|
|
|
|
bNodeSocket *sockDistance = nodeFindSocket(node, SOCK_OUT, "Distance");
|
|
|
|
bNodeSocket *sockColor = nodeFindSocket(node, SOCK_OUT, "Color");
|
|
|
|
if (tex->feature == 4 && (socket_is_used(sockDistance) || socket_is_used(sockColor))) {
|
|
|
|
tex->feature = SHD_VORONOI_F1;
|
|
|
|
|
|
|
|
bNode *voronoiNode = nodeAddStaticNode(NULL, ntree, SH_NODE_TEX_VORONOI);
|
|
|
|
NodeTexVoronoi *texVoronoi = (NodeTexVoronoi *)voronoiNode->storage;
|
|
|
|
texVoronoi->feature = SHD_VORONOI_F2;
|
|
|
|
texVoronoi->distance = tex->distance;
|
|
|
|
texVoronoi->dimensions = 3;
|
|
|
|
voronoiNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
voronoiNode->locy = node->locy;
|
|
|
|
|
|
|
|
bNodeSocket *sockVector = nodeFindSocket(node, SOCK_IN, "Vector");
|
|
|
|
bNodeSocket *sockScale = nodeFindSocket(node, SOCK_IN, "Scale");
|
|
|
|
bNodeSocket *sockExponent = nodeFindSocket(node, SOCK_IN, "Exponent");
|
|
|
|
bNodeSocket *sockVoronoiVector = nodeFindSocket(voronoiNode, SOCK_IN, "Vector");
|
|
|
|
bNodeSocket *sockVoronoiScale = nodeFindSocket(voronoiNode, SOCK_IN, "Scale");
|
|
|
|
bNodeSocket *sockVoronoiExponent = nodeFindSocket(voronoiNode, SOCK_IN, "Exponent");
|
|
|
|
if (sockVector->link) {
|
|
|
|
nodeAddLink(ntree,
|
|
|
|
sockVector->link->fromnode,
|
|
|
|
sockVector->link->fromsock,
|
|
|
|
voronoiNode,
|
|
|
|
sockVoronoiVector);
|
|
|
|
}
|
|
|
|
*cycles_node_socket_float_value(sockVoronoiScale) = *cycles_node_socket_float_value(
|
|
|
|
sockScale);
|
|
|
|
if (sockScale->link) {
|
|
|
|
nodeAddLink(ntree,
|
|
|
|
sockScale->link->fromnode,
|
|
|
|
sockScale->link->fromsock,
|
|
|
|
voronoiNode,
|
|
|
|
sockVoronoiScale);
|
|
|
|
}
|
|
|
|
*cycles_node_socket_float_value(sockVoronoiExponent) = *cycles_node_socket_float_value(
|
|
|
|
sockExponent);
|
|
|
|
if (sockExponent->link) {
|
|
|
|
nodeAddLink(ntree,
|
|
|
|
sockExponent->link->fromnode,
|
|
|
|
sockExponent->link->fromsock,
|
|
|
|
voronoiNode,
|
|
|
|
sockVoronoiExponent);
|
|
|
|
}
|
|
|
|
|
|
|
|
bNode *subtractNode = nodeAddStaticNode(NULL, ntree, SH_NODE_MATH);
|
|
|
|
subtractNode->custom1 = NODE_MATH_SUBTRACT;
|
|
|
|
subtractNode->locx = voronoiNode->locx + voronoiNode->width + 20.0f;
|
|
|
|
subtractNode->locy = voronoiNode->locy;
|
|
|
|
bNodeSocket *sockSubtractOutValue = nodeFindSocket(subtractNode, SOCK_OUT, "Value");
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromnode == node) {
|
|
|
|
nodeAddLink(ntree, subtractNode, sockSubtractOutValue, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeSocket *sockDistanceF1 = nodeFindSocket(node, SOCK_OUT, "Distance");
|
|
|
|
bNodeSocket *sockDistanceF2 = nodeFindSocket(voronoiNode, SOCK_OUT, "Distance");
|
|
|
|
bNodeSocket *sockSubtractA = BLI_findlink(&subtractNode->inputs, 0);
|
|
|
|
bNodeSocket *sockSubtractB = BLI_findlink(&subtractNode->inputs, 1);
|
|
|
|
|
|
|
|
nodeAddLink(ntree, node, sockDistanceF1, subtractNode, sockSubtractB);
|
|
|
|
nodeAddLink(ntree, voronoiNode, sockDistanceF2, subtractNode, sockSubtractA);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-19 13:18:52 +10:00
|
|
|
/**
|
|
|
|
* The coloring property of the Voronoi node was removed. Previously,
|
2019-09-12 13:09:31 +02:00
|
|
|
* if the coloring enum was set to Intensity (0), the voronoi distance
|
|
|
|
* was returned in all outputs, otherwise, the Cell ID was returned.
|
|
|
|
* Since we remapped the Fac output in update_voronoi_node_fac_output,
|
|
|
|
* then to fix this, we relink the Color output to the Distance
|
2019-09-19 13:18:52 +10:00
|
|
|
* output if coloring was set to 0, and the other way around otherwise.
|
2019-09-12 13:09:31 +02:00
|
|
|
*/
|
|
|
|
static void update_voronoi_node_coloring(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
bNode *node = link->fromnode;
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node && node->type == SH_NODE_TEX_VORONOI && node->storage) {
|
2019-09-12 13:09:31 +02:00
|
|
|
NodeTexVoronoi *tex = (NodeTexVoronoi *)node->storage;
|
|
|
|
if (tex->coloring == 0) {
|
|
|
|
bNodeSocket *sockColor = nodeFindSocket(node, SOCK_OUT, "Color");
|
|
|
|
if (link->fromsock == sockColor) {
|
|
|
|
bNodeSocket *sockDistance = nodeFindSocket(node, SOCK_OUT, "Distance");
|
|
|
|
nodeAddLink(ntree, node, sockDistance, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bNodeSocket *sockDistance = nodeFindSocket(node, SOCK_OUT, "Distance");
|
|
|
|
if (link->fromsock == sockDistance) {
|
|
|
|
bNodeSocket *sockColor = nodeFindSocket(node, SOCK_OUT, "Color");
|
|
|
|
nodeAddLink(ntree, node, sockColor, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Previously, the output euclidean distance was actually the squared
|
2019-11-25 00:55:11 +11:00
|
|
|
* euclidean distance. To fix this, we square the output distance
|
2019-09-12 13:09:31 +02:00
|
|
|
* socket if the distance metric is set to SHD_VORONOI_EUCLIDEAN.
|
|
|
|
*/
|
|
|
|
static void update_voronoi_node_square_distance(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-11-06 20:03:22 +01:00
|
|
|
if (node->type == SH_NODE_TEX_VORONOI && node->storage) {
|
2019-09-12 13:09:31 +02:00
|
|
|
NodeTexVoronoi *tex = (NodeTexVoronoi *)node->storage;
|
|
|
|
bNodeSocket *sockDistance = nodeFindSocket(node, SOCK_OUT, "Distance");
|
|
|
|
if (tex->distance == SHD_VORONOI_EUCLIDEAN &&
|
|
|
|
(tex->feature == SHD_VORONOI_F1 || tex->feature == SHD_VORONOI_F2) &&
|
|
|
|
socket_is_used(sockDistance)) {
|
|
|
|
bNode *multiplyNode = nodeAddStaticNode(NULL, ntree, SH_NODE_MATH);
|
|
|
|
multiplyNode->custom1 = NODE_MATH_MULTIPLY;
|
|
|
|
multiplyNode->locx = node->locx + node->width + 20.0f;
|
|
|
|
multiplyNode->locy = node->locy;
|
|
|
|
|
|
|
|
bNodeSocket *sockValue = nodeFindSocket(multiplyNode, SOCK_OUT, "Value");
|
|
|
|
LISTBASE_FOREACH_BACKWARD_MUTABLE (bNodeLink *, link, &ntree->links) {
|
|
|
|
if (link->fromsock == sockDistance) {
|
|
|
|
nodeAddLink(ntree, multiplyNode, sockValue, link->tonode, link->tosock);
|
|
|
|
nodeRemLink(ntree, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeSocket *sockMultiplyA = BLI_findlink(&multiplyNode->inputs, 0);
|
|
|
|
bNodeSocket *sockMultiplyB = BLI_findlink(&multiplyNode->inputs, 1);
|
|
|
|
|
|
|
|
nodeAddLink(ntree, node, sockDistance, multiplyNode, sockMultiplyA);
|
|
|
|
nodeAddLink(ntree, node, sockDistance, multiplyNode, sockMultiplyB);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-05 22:07:43 +01:00
|
|
|
/* Noise and Wave Texture nodes: Restore previous Distortion range.
|
|
|
|
* In 2.81 we used noise() for distortion, now we use snoise() which has twice the range.
|
|
|
|
* To fix this we halve distortion value, directly or by adding multiply node for used sockets.
|
|
|
|
*/
|
|
|
|
static void update_noise_and_wave_distortion(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bool need_update = false;
|
|
|
|
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
2019-12-13 09:02:21 +11:00
|
|
|
if (node->type == SH_NODE_TEX_NOISE || node->type == SH_NODE_TEX_WAVE) {
|
2019-12-05 22:07:43 +01:00
|
|
|
|
|
|
|
bNodeSocket *sockDistortion = nodeFindSocket(node, SOCK_IN, "Distortion");
|
|
|
|
float *distortion = cycles_node_socket_float_value(sockDistortion);
|
|
|
|
|
2020-01-06 11:52:35 +01:00
|
|
|
if (socket_is_used(sockDistortion) && sockDistortion->link != NULL) {
|
2019-12-05 22:07:43 +01:00
|
|
|
bNode *distortionInputNode = sockDistortion->link->fromnode;
|
|
|
|
bNodeSocket *distortionInputSock = sockDistortion->link->fromsock;
|
|
|
|
|
|
|
|
bNode *mulNode = nodeAddStaticNode(NULL, ntree, SH_NODE_MATH);
|
|
|
|
mulNode->custom1 = NODE_MATH_MULTIPLY;
|
|
|
|
mulNode->locx = node->locx;
|
|
|
|
mulNode->locy = node->locy - 240.0f;
|
|
|
|
mulNode->flag |= NODE_HIDDEN;
|
|
|
|
bNodeSocket *mulSockA = BLI_findlink(&mulNode->inputs, 0);
|
|
|
|
bNodeSocket *mulSockB = BLI_findlink(&mulNode->inputs, 1);
|
|
|
|
*cycles_node_socket_float_value(mulSockB) = 0.5f;
|
|
|
|
bNodeSocket *mulSockOut = nodeFindSocket(mulNode, SOCK_OUT, "Value");
|
|
|
|
|
|
|
|
nodeRemLink(ntree, sockDistortion->link);
|
|
|
|
nodeAddLink(ntree, distortionInputNode, distortionInputSock, mulNode, mulSockA);
|
|
|
|
nodeAddLink(ntree, mulNode, mulSockOut, node, sockDistortion);
|
|
|
|
|
|
|
|
need_update = true;
|
|
|
|
}
|
|
|
|
else if (*distortion != 0.0f) {
|
|
|
|
*distortion = *distortion * 0.5f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_update) {
|
|
|
|
ntreeUpdateTree(NULL, ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-03 22:06:52 +11:00
|
|
|
/**
|
|
|
|
* Wave Texture node: Restore previous texture directions and offset.
|
2020-02-17 12:31:38 +01:00
|
|
|
* 1. In 2.81, Wave texture had fixed diagonal direction (Bands) or
|
|
|
|
* mapping along distance (Rings). Now, directions are customizable
|
|
|
|
* properties, with X axis being new default. To fix this we set new
|
|
|
|
* direction options to Diagonal and Spherical.
|
2020-03-03 22:06:52 +11:00
|
|
|
* 2. Sine profile is now negatively offset by PI/2 to better match
|
2020-02-17 12:31:38 +01:00
|
|
|
* other profiles. To fix this we set new Phase Offset input to PI/2
|
|
|
|
* in nodes with Sine profile.
|
|
|
|
*/
|
|
|
|
static void update_wave_node_directions_and_offset(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
if (node->type == SH_NODE_TEX_WAVE) {
|
|
|
|
NodeTexWave *tex = (NodeTexWave *)node->storage;
|
|
|
|
tex->bands_direction = SHD_WAVE_BANDS_DIRECTION_DIAGONAL;
|
|
|
|
tex->rings_direction = SHD_WAVE_RINGS_DIRECTION_SPHERICAL;
|
|
|
|
|
|
|
|
if (tex->wave_profile == SHD_WAVE_PROFILE_SIN) {
|
|
|
|
bNodeSocket *sockPhaseOffset = nodeFindSocket(node, SOCK_IN, "Phase Offset");
|
|
|
|
*cycles_node_socket_float_value(sockPhaseOffset) = M_PI_2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-20 20:25:23 +02:00
|
|
|
void blo_do_versions_cycles(FileData *UNUSED(fd), Library *UNUSED(lib), Main *bmain)
|
|
|
|
{
|
|
|
|
/* Particle shape shared with Eevee. */
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 16)) {
|
|
|
|
for (ParticleSettings *part = bmain->particles.first; part; part = part->id.next) {
|
|
|
|
IDProperty *cpart = cycles_properties_from_ID(&part->id);
|
|
|
|
|
|
|
|
if (cpart) {
|
|
|
|
part->shape = cycles_property_float(cpart, "shape", 0.0);
|
|
|
|
part->rad_root = cycles_property_float(cpart, "root_width", 1.0);
|
|
|
|
part->rad_tip = cycles_property_float(cpart, "tip_width", 0.0);
|
|
|
|
part->rad_scale = cycles_property_float(cpart, "radius_scale", 0.01);
|
|
|
|
if (cycles_property_boolean(cpart, "use_closetip", true)) {
|
|
|
|
part->shape_flag |= PART_SHAPE_CLOSE_TIP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-05-16 17:03:16 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 68)) {
|
2019-05-17 16:57:31 +02:00
|
|
|
/* Unify Cycles and Eevee film transparency. */
|
2019-05-16 17:03:16 +02:00
|
|
|
for (Scene *scene = bmain->scenes.first; scene; scene = scene->id.next) {
|
|
|
|
if (STREQ(scene->r.engine, RE_engine_id_CYCLES)) {
|
|
|
|
IDProperty *cscene = cycles_properties_from_ID(&scene->id);
|
|
|
|
if (cscene) {
|
|
|
|
bool cycles_film_transparency = cycles_property_boolean(
|
|
|
|
cscene, "film_transparent", false);
|
|
|
|
scene->r.alphamode = cycles_film_transparency ? R_ALPHAPREMUL : R_ADDSKY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Shading: Add more operators to Vector Math node.
Add Multiply, Divide, Project, Reflect, Distance, Length, Scale, Snap,
Floor, Ceil, Modulo, Fraction, Absolute, Minimum, and Maximum operators
to the Vector Math node. The Value output has been removed from operators
whose output is a vector, and the other way around. All of those removals
has been handled properly in versioning code.
The patch doesn't include tests for the new operators. Tests will be added
in a later patch.
Reviewers: brecht, JacquesLucke
Differential Revision: https://developer.blender.org/D5523
2019-08-21 19:36:33 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 3)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_vector_math_node_operators_enum_mapping(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-09 21:06:55 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 10)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_musgrave_node_color_output(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-12 13:09:31 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 11)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_voronoi_node_f3_and_f4(ntree);
|
|
|
|
update_voronoi_node_fac_output(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-04-20 20:25:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void do_versions_after_linking_cycles(Main *bmain)
|
|
|
|
{
|
2019-05-16 05:17:22 +02:00
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 66)) {
|
2019-04-20 20:25:23 +02:00
|
|
|
/* Shader node tree changes. After lib linking so we have all the typeinfo
|
|
|
|
* pointers and updated sockets and we can use the high level node API to
|
|
|
|
* manipulate nodes. */
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type != NTREE_SHADER) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 273, 5)) {
|
|
|
|
/* Euler order was ZYX in previous versions. */
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
mapping_node_order_flip(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 276, 6)) {
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
vector_curve_node_remap(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 279, 2) ||
|
|
|
|
(MAIN_VERSION_ATLEAST(bmain, 280, 0) && !MAIN_VERSION_ATLEAST(bmain, 280, 4))) {
|
|
|
|
displacement_node_insert(ntree);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 279, 3)) {
|
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
displacement_principled_nodes(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 279, 4) ||
|
|
|
|
(MAIN_VERSION_ATLEAST(bmain, 280, 0) && !MAIN_VERSION_ATLEAST(bmain, 280, 5))) {
|
|
|
|
/* Switch to squared roughness convention */
|
|
|
|
square_roughness_node_insert(ntree);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 279, 5)) {
|
|
|
|
ambient_occlusion_node_relink(ntree);
|
|
|
|
}
|
2019-04-23 13:56:30 +02:00
|
|
|
|
2019-05-16 05:17:22 +02:00
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 66)) {
|
2019-04-23 13:56:30 +02:00
|
|
|
for (bNode *node = ntree->nodes.first; node; node = node->next) {
|
|
|
|
image_node_colorspace(node);
|
|
|
|
}
|
|
|
|
}
|
2019-04-20 20:25:23 +02:00
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-05-12 13:41:23 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 64)) {
|
|
|
|
/* Unfiy Cycles and Eevee settings. */
|
|
|
|
Scene *scene = bmain->scenes.first;
|
|
|
|
const char *engine = (scene) ? scene->r.engine : "CYCLES";
|
|
|
|
|
|
|
|
for (Light *light = bmain->lights.first; light; light = light->id.next) {
|
|
|
|
light_emission_unify(light, engine);
|
|
|
|
}
|
|
|
|
}
|
2019-05-17 22:53:39 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 280, 69)) {
|
|
|
|
/* Unify Cycles and Eevee depth of field. */
|
|
|
|
Scene *scene = bmain->scenes.first;
|
|
|
|
const char *engine = (scene) ? scene->r.engine : "CYCLES";
|
|
|
|
|
|
|
|
if (STREQ(engine, RE_engine_id_CYCLES)) {
|
|
|
|
for (Camera *camera = bmain->cameras.first; camera; camera = camera->id.next) {
|
|
|
|
IDProperty *ccamera = cycles_properties_from_ID(&camera->id);
|
|
|
|
if (ccamera) {
|
2019-06-04 17:14:09 +02:00
|
|
|
const bool is_fstop = cycles_property_int(ccamera, "aperture_type", 0) == 1;
|
2019-05-17 22:53:39 +02:00
|
|
|
|
|
|
|
camera->dof.aperture_fstop = cycles_property_float(ccamera, "aperture_fstop", 5.6f);
|
|
|
|
camera->dof.aperture_blades = cycles_property_int(ccamera, "aperture_blades", 0);
|
|
|
|
camera->dof.aperture_rotation = cycles_property_float(ccamera, "aperture_rotation", 0.0);
|
|
|
|
camera->dof.aperture_ratio = cycles_property_float(ccamera, "aperture_ratio", 1.0f);
|
|
|
|
camera->dof.flag |= CAM_DOF_ENABLED;
|
|
|
|
|
|
|
|
float aperture_size = cycles_property_float(ccamera, "aperture_size", 0.0f);
|
|
|
|
|
2019-06-04 17:14:09 +02:00
|
|
|
if (is_fstop) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (aperture_size > 0.0f) {
|
2019-05-17 22:53:39 +02:00
|
|
|
if (camera->type == CAM_ORTHO) {
|
|
|
|
camera->dof.aperture_fstop = 1.0f / (2.0f * aperture_size);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
camera->dof.aperture_fstop = (camera->lens * 1e-3f) / (2.0f * aperture_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No depth of field, set default settings. */
|
2019-05-20 16:14:02 +02:00
|
|
|
camera->dof.aperture_fstop = 2.8f;
|
2019-05-17 22:53:39 +02:00
|
|
|
camera->dof.aperture_blades = 0;
|
|
|
|
camera->dof.aperture_rotation = 0.0f;
|
|
|
|
camera->dof.aperture_ratio = 1.0f;
|
|
|
|
camera->dof.flag &= ~CAM_DOF_ENABLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-08-18 11:16:04 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 2)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_math_node_single_operand_operators(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
Shading: Add more operators to Vector Math node.
Add Multiply, Divide, Project, Reflect, Distance, Length, Scale, Snap,
Floor, Ceil, Modulo, Fraction, Absolute, Minimum, and Maximum operators
to the Vector Math node. The Value output has been removed from operators
whose output is a vector, and the other way around. All of those removals
has been handled properly in versioning code.
The patch doesn't include tests for the new operators. Tests will be added
in a later patch.
Reviewers: brecht, JacquesLucke
Differential Revision: https://developer.blender.org/D5523
2019-08-21 19:36:33 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 3)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_vector_math_node_add_and_subtract_operators(ntree);
|
|
|
|
update_vector_math_node_dot_product_operator(ntree);
|
|
|
|
update_vector_math_node_cross_product_operator(ntree);
|
|
|
|
update_vector_math_node_normalize_operator(ntree);
|
|
|
|
update_vector_math_node_average_operator(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-04 17:54:32 +02:00
|
|
|
|
2019-09-04 19:02:40 +02:00
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 7)) {
|
2019-09-04 17:54:32 +02:00
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_noise_node_dimensions(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-04 23:17:13 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 8)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_mapping_node_inputs_and_properties(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-09 21:06:55 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 10)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_musgrave_node_dimensions(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-09-12 13:09:31 +02:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 281, 11)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_voronoi_node_dimensions(ntree);
|
|
|
|
update_voronoi_node_crackle(ntree);
|
|
|
|
update_voronoi_node_coloring(ntree);
|
|
|
|
update_voronoi_node_square_distance(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-12-05 22:07:43 +01:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 282, 4)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_noise_and_wave_distortion(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2020-02-17 12:31:38 +01:00
|
|
|
|
|
|
|
if (!MAIN_VERSION_ATLEAST(bmain, 283, 4)) {
|
|
|
|
FOREACH_NODETREE_BEGIN (bmain, ntree, id) {
|
|
|
|
if (ntree->type == NTREE_SHADER) {
|
|
|
|
update_wave_node_directions_and_offset(ntree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FOREACH_NODETREE_END;
|
|
|
|
}
|
2019-04-20 20:25:23 +02:00
|
|
|
}
|