/* * ***** BEGIN GPL LICENSE BLOCK ***** * * 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. * * The Original Code is Copyright (C) 2005 Blender Foundation. * All rights reserved. * * The Original Code is: all of this file. * * Contributor(s): David Millan Escriva, Juho Vepsäläinen, Nathan Letwory * * ***** END GPL LICENSE BLOCK ***** */ /** \file blender/editors/space_node/node_add.c * \ingroup spnode */ #include #include "MEM_guardedalloc.h" #include "DNA_node_types.h" #include "BLI_listbase.h" #include "BLI_math.h" #include "BLF_translation.h" #include "BKE_context.h" #include "BKE_image.h" #include "BKE_library.h" #include "BKE_main.h" #include "BKE_node.h" #include "BKE_report.h" #include "ED_node.h" /* own include */ #include "ED_screen.h" #include "ED_render.h" #include "RNA_access.h" #include "RNA_define.h" #include "RNA_enum_types.h" #include "WM_api.h" #include "WM_types.h" #include "UI_view2d.h" #include "node_intern.h" /* own include */ /* can be called from menus too, but they should do own undopush and redraws */ bNode *node_add_node(SpaceNode *snode, Main *bmain, Scene *scene, bNodeTemplate *ntemp, float locx, float locy) { bNode *node = NULL, *gnode; node_deselect_all(snode); node = nodeAddNode(snode->edittree, ntemp); /* generics */ if (node) { node_select(node); gnode = node_tree_get_editgroup(snode->nodetree); // arbitrary y offset of 60 so its visible if (gnode) { nodeFromView(gnode, locx, locy + 60.0f, &node->locx, &node->locy); } else { node->locx = locx; node->locy = locy + 60.0f; } ntreeUpdateTree(snode->edittree); ED_node_set_active(bmain, snode->edittree, node); if (snode->nodetree->type == NTREE_COMPOSIT) { if (ELEM4(node->type, CMP_NODE_R_LAYERS, CMP_NODE_COMPOSITE, CMP_NODE_DEFOCUS, CMP_NODE_OUTPUT_FILE)) { node->id = &scene->id; } else if (ELEM3(node->type, CMP_NODE_MOVIECLIP, CMP_NODE_MOVIEDISTORTION, CMP_NODE_STABILIZE2D)) { node->id = (ID *)scene->clip; } ntreeCompositForceHidden(snode->edittree, scene); } if (node->id) id_us_plus(node->id); if (snode->flag & SNODE_USE_HIDDEN_PREVIEW) node->flag &= ~NODE_PREVIEW; snode_update(snode, node); } if (snode->nodetree->type == NTREE_TEXTURE) { ntreeTexCheckCyclics(snode->edittree); } return node; } /* ********************** Add reroute operator ***************** */ static int add_reroute_intersect_check(bNodeLink *link, float mcoords[][2], int tot, float result[2]) { float coord_array[NODE_LINK_RESOL + 1][2]; int i, b; if (node_link_bezier_points(NULL, NULL, link, coord_array, NODE_LINK_RESOL)) { for (i = 0; i < tot - 1; i++) for (b = 0; b < NODE_LINK_RESOL; b++) if (isect_line_line_v2(mcoords[i], mcoords[i + 1], coord_array[b], coord_array[b + 1]) > 0) { result[0] = (mcoords[i][0] + mcoords[i + 1][0]) / 2.0f; result[1] = (mcoords[i][1] + mcoords[i + 1][1]) / 2.0f; return 1; } } return 0; } typedef struct bNodeSocketLink { struct bNodeSocketLink *next, *prev; struct bNodeSocket *sock; struct bNodeLink *link; float point[2]; } bNodeSocketLink; static bNodeSocketLink *add_reroute_insert_socket_link(ListBase *lb, bNodeSocket *sock, bNodeLink *link, float point[2]) { bNodeSocketLink *socklink, *prev; socklink = MEM_callocN(sizeof(bNodeSocketLink), "socket link"); socklink->sock = sock; socklink->link = link; copy_v2_v2(socklink->point, point); for (prev = lb->last; prev; prev = prev->prev) { if (prev->sock == sock) break; } BLI_insertlinkafter(lb, prev, socklink); return socklink; } static bNodeSocketLink *add_reroute_do_socket_section(bContext *C, bNodeSocketLink *socklink, int in_out) { SpaceNode *snode = CTX_wm_space_node(C); bNodeTree *ntree = snode->edittree; bNode *reroute_node = NULL; bNodeSocket *cursock = socklink->sock; float insert_point[2]; int num_links; zero_v2(insert_point); num_links = 0; while (socklink && socklink->sock == cursock) { if (!(socklink->link->flag & NODE_LINK_TEST)) { socklink->link->flag |= NODE_LINK_TEST; /* create the reroute node for this cursock */ if (!reroute_node) { bNodeTemplate ntemp; ntemp.type = NODE_REROUTE; reroute_node = nodeAddNode(ntree, &ntemp); /* add a single link to/from the reroute node to replace multiple links */ if (in_out == SOCK_OUT) { nodeAddLink(ntree, socklink->link->fromnode, socklink->link->fromsock, reroute_node, reroute_node->inputs.first); } else { nodeAddLink(ntree, reroute_node, reroute_node->outputs.first, socklink->link->tonode, socklink->link->tosock); } } /* insert the reroute node into the link */ if (in_out == SOCK_OUT) { socklink->link->fromnode = reroute_node; socklink->link->fromsock = reroute_node->outputs.first; } else { socklink->link->tonode = reroute_node; socklink->link->tosock = reroute_node->inputs.first; } add_v2_v2(insert_point, socklink->point); ++num_links; } socklink = socklink->next; } if (num_links > 0) { bNode *gnode = node_tree_get_editgroup(snode->nodetree); /* average cut point from shared links */ mul_v2_fl(insert_point, 1.0f / num_links); if (gnode) { nodeFromView(gnode, insert_point[0], insert_point[1], &reroute_node->locx, &reroute_node->locy); } else { reroute_node->locx = insert_point[0]; reroute_node->locy = insert_point[1]; } } return socklink; } static int add_reroute_exec(bContext *C, wmOperator *op) { SpaceNode *snode = CTX_wm_space_node(C); ARegion *ar = CTX_wm_region(C); bNodeTree *ntree = snode->edittree; float mcoords[256][2]; int i = 0; /* Get the cut path */ RNA_BEGIN(op->ptr, itemptr, "path") { float loc[2]; RNA_float_get_array(&itemptr, "loc", loc); UI_view2d_region_to_view(&ar->v2d, (short)loc[0], (short)loc[1], &mcoords[i][0], &mcoords[i][1]); i++; if (i >= 256) break; } RNA_END; if (i > 1) { ListBase output_links, input_links; bNodeLink *link; bNodeSocketLink *socklink; float insert_point[2]; /* always first */ ED_preview_kill_jobs(C); node_deselect_all(snode); /* Find cut links and sort them by sockets */ output_links.first = output_links.last = NULL; input_links.first = input_links.last = NULL; for (link = ntree->links.first; link; link = link->next) { if (add_reroute_intersect_check(link, mcoords, i, insert_point)) { add_reroute_insert_socket_link(&output_links, link->fromsock, link, insert_point); add_reroute_insert_socket_link(&input_links, link->tosock, link, insert_point); /* Clear flag */ link->flag &= ~NODE_LINK_TEST; } } /* Create reroute nodes for intersected links. * Only one reroute if links share the same input/output socket. */ socklink = output_links.first; while (socklink) { socklink = add_reroute_do_socket_section(C, socklink, SOCK_OUT); } socklink = input_links.first; while (socklink) { socklink = add_reroute_do_socket_section(C, socklink, SOCK_IN); } BLI_freelistN(&output_links); BLI_freelistN(&input_links); /* always last */ ntreeUpdateTree(ntree); snode_notify(C, snode); snode_dag_update(C, snode); return OPERATOR_FINISHED; } return OPERATOR_CANCELLED | OPERATOR_PASS_THROUGH; } void NODE_OT_add_reroute(wmOperatorType *ot) { PropertyRNA *prop; ot->name = "Add reroute"; ot->idname = "NODE_OT_add_reroute"; ot->invoke = WM_gesture_lines_invoke; ot->modal = WM_gesture_lines_modal; ot->exec = add_reroute_exec; ot->cancel = WM_gesture_lines_cancel; ot->poll = ED_operator_node_active; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; prop = RNA_def_property(ot->srna, "path", PROP_COLLECTION, PROP_NONE); RNA_def_property_struct_runtime(prop, &RNA_OperatorMousePath); /* internal */ RNA_def_int(ot->srna, "cursor", BC_CROSSCURSOR, 0, INT_MAX, "Cursor", "", 0, INT_MAX); } /* ****************** Add File Node Operator ******************* */ static int node_add_file_exec(bContext *C, wmOperator *op) { Main *bmain = CTX_data_main(C); Scene *scene = CTX_data_scene(C); SpaceNode *snode = CTX_wm_space_node(C); bNode *node; Image *ima = NULL; bNodeTemplate ntemp; /* check input variables */ if (RNA_struct_property_is_set(op->ptr, "filepath")) { char path[FILE_MAX]; RNA_string_get(op->ptr, "filepath", path); errno = 0; ima = BKE_image_load_exists(path); if (!ima) { BKE_reportf(op->reports, RPT_ERROR, "Cannot read image '%s': %s", path, errno ? strerror(errno) : TIP_("unsupported format")); return OPERATOR_CANCELLED; } } else if (RNA_struct_property_is_set(op->ptr, "name")) { char name[MAX_ID_NAME - 2]; RNA_string_get(op->ptr, "name", name); ima = (Image *)BKE_libblock_find_name(ID_IM, name); if (!ima) { BKE_reportf(op->reports, RPT_ERROR, "Image '%s' not found", name); return OPERATOR_CANCELLED; } } node_deselect_all(snode); switch (snode->nodetree->type) { case NTREE_SHADER: ntemp.type = SH_NODE_TEX_IMAGE; break; case NTREE_TEXTURE: ntemp.type = TEX_NODE_IMAGE; break; case NTREE_COMPOSIT: ntemp.type = CMP_NODE_IMAGE; break; default: return OPERATOR_CANCELLED; } ED_preview_kill_jobs(C); node = node_add_node(snode, bmain, scene, &ntemp, snode->cursor[0], snode->cursor[1]); if (!node) { BKE_report(op->reports, RPT_WARNING, "Could not add an image node"); return OPERATOR_CANCELLED; } node->id = (ID *)ima; id_us_plus(node->id); BKE_image_signal(ima, NULL, IMA_SIGNAL_RELOAD); WM_event_add_notifier(C, NC_IMAGE | NA_EDITED, ima); snode_notify(C, snode); snode_dag_update(C, snode); return OPERATOR_FINISHED; } static int node_add_file_invoke(bContext *C, wmOperator *op, wmEvent *event) { ARegion *ar = CTX_wm_region(C); SpaceNode *snode = CTX_wm_space_node(C); /* convert mouse coordinates to v2d space */ UI_view2d_region_to_view(&ar->v2d, event->mval[0], event->mval[1], &snode->cursor[0], &snode->cursor[1]); if (RNA_struct_property_is_set(op->ptr, "filepath") || RNA_struct_property_is_set(op->ptr, "name")) return node_add_file_exec(C, op); else return WM_operator_filesel(C, op, event); } void NODE_OT_add_file(wmOperatorType *ot) { /* identifiers */ ot->name = "Add File Node"; ot->description = "Add a file node to the current node editor"; ot->idname = "NODE_OT_add_file"; /* callbacks */ ot->exec = node_add_file_exec; ot->invoke = node_add_file_invoke; ot->poll = ED_operator_node_active; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; WM_operator_properties_filesel(ot, FOLDERFILE | IMAGEFILE, FILE_SPECIAL, FILE_OPENFILE, WM_FILESEL_FILEPATH, FILE_DEFAULTDISPLAY); //XXX TODO, relative_path RNA_def_string(ot->srna, "name", "Image", MAX_ID_NAME - 2, "Name", "Datablock name to assign"); } /********************** New node tree operator *********************/ static int new_node_tree_exec(bContext *C, wmOperator *op) { SpaceNode *snode; bNodeTree *ntree; PointerRNA ptr, idptr; PropertyRNA *prop; int treetype; char treename[MAX_ID_NAME - 2] = "NodeTree"; /* retrieve state */ snode = CTX_wm_space_node(C); if (RNA_struct_property_is_set(op->ptr, "type")) treetype = RNA_enum_get(op->ptr, "type"); else treetype = snode->treetype; if (RNA_struct_property_is_set(op->ptr, "name")) RNA_string_get(op->ptr, "name", treename); ntree = ntreeAddTree(treename, treetype, 0); if (!ntree) return OPERATOR_CANCELLED; /* hook into UI */ uiIDContextProperty(C, &ptr, &prop); if (prop) { RNA_id_pointer_create(&ntree->id, &idptr); RNA_property_pointer_set(&ptr, prop, idptr); /* RNA_property_pointer_set increases the user count, * fixed here as the editor is the initial user. */ ntree->id.us++; RNA_property_update(C, &ptr, prop); } else if (snode) { Scene *scene = CTX_data_scene(C); snode->nodetree = ntree; ED_node_tree_update(snode, scene); } return OPERATOR_FINISHED; } void NODE_OT_new_node_tree(wmOperatorType *ot) { /* identifiers */ ot->name = "New Node Tree"; ot->idname = "NODE_OT_new_node_tree"; ot->description = "Create a new node tree"; /* api callbacks */ ot->exec = new_node_tree_exec; ot->poll = ED_operator_node_active; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; RNA_def_enum(ot->srna, "type", nodetree_type_items, NTREE_COMPOSIT, "Tree Type", ""); RNA_def_string(ot->srna, "name", "NodeTree", MAX_ID_NAME - 2, "Name", ""); }