| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  |  * ***** 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, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2008 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Contributor(s): Blender Foundation, Nathan Letwory | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:29:51 +00:00
										 |  |  | /** \file blender/editors/space_node/node_select.c
 | 
					
						
							|  |  |  |  *  \ingroup spnode | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-02 03:51:42 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | #include "DNA_node_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | #include "BLI_rect.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | #include "BLI_lasso.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | #include "BLI_string.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | #include "BLI_string_utf8.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-24 21:57:13 +00:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							|  |  |  | #include "BKE_main.h"
 | 
					
						
							|  |  |  | #include "BKE_node.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-02 23:03:16 +00:00
										 |  |  | #include "ED_node.h"  /* own include */
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | #include "ED_screen.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "RNA_access.h"
 | 
					
						
							|  |  |  | #include "RNA_define.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "WM_api.h"
 | 
					
						
							|  |  |  | #include "WM_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | #include "UI_interface.h"
 | 
					
						
							|  |  |  | #include "UI_resources.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | #include "UI_view2d.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-02 23:03:16 +00:00
										 |  |  | #include "node_intern.h"  /* own include */
 | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ****** helpers ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-01 12:38:03 +00:00
										 |  |  | static bNode *node_under_mouse_select(bNodeTree *ntree, int mx, int my) | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = ntree->nodes.last; node; node = node->prev) { | 
					
						
							| 
									
										
										
										
											2012-06-01 12:38:03 +00:00
										 |  |  | 		if (node->typeinfo->select_area_func) { | 
					
						
							|  |  |  | 			if (node->typeinfo->select_area_func(node, mx, my)) | 
					
						
							|  |  |  | 				return node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bNode *node_under_mouse_tweak(bNodeTree *ntree, int mx, int my) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = ntree->nodes.last; node; node = node->prev) { | 
					
						
							| 
									
										
										
										
											2012-06-01 12:38:03 +00:00
										 |  |  | 		if (node->typeinfo->tweak_area_func) { | 
					
						
							|  |  |  | 			if (node->typeinfo->tweak_area_func(node, mx, my)) | 
					
						
							|  |  |  | 				return node; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | static void node_toggle(bNode *node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 	nodeSetSelected(node, !(node->flag & SELECT)); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 18:10:57 +00:00
										 |  |  | void node_socket_select(bNode *node, bNodeSocket *sock) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	sock->flag |= SELECT; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* select node too */ | 
					
						
							|  |  |  | 	if (node) | 
					
						
							|  |  |  | 		node->flag |= SELECT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | void node_socket_deselect(bNode *node, bNodeSocket *sock, const bool deselect_node) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	sock->flag &= ~SELECT; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (node && deselect_node) { | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 		bool sel = 0; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* if no selected sockets remain, also deselect the node */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		for (sock = node->inputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			if (sock->flag & SELECT) { | 
					
						
							|  |  |  | 				sel = 1; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		for (sock = node->outputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			if (sock->flag & SELECT) { | 
					
						
							|  |  |  | 				sel = 1; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (!sel) | 
					
						
							|  |  |  | 			node->flag &= ~SELECT; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void node_socket_toggle(bNode *node, bNodeSocket *sock, int deselect_node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (sock->flag & SELECT) | 
					
						
							|  |  |  | 		node_socket_deselect(node, sock, deselect_node); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		node_socket_select(node, sock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | /* no undo here! */ | 
					
						
							|  |  |  | void node_deselect_all(SpaceNode *snode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 		nodeSetSelected(node, false); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | void node_deselect_all_input_sockets(SpaceNode *snode, const bool deselect_nodes) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	bNodeSocket *sock; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* XXX not calling node_socket_deselect here each time, because this does iteration
 | 
					
						
							|  |  |  | 	 * over all node sockets internally to check if the node stays selected. | 
					
						
							|  |  |  | 	 * We can do that more efficiently here. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		int sel = 0; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		for (sock = node->inputs.first; sock; sock = sock->next) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			sock->flag &= ~SELECT; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* if no selected sockets remain, also deselect the node */ | 
					
						
							|  |  |  | 		if (deselect_nodes) { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 			for (sock = node->outputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 				if (sock->flag & SELECT) { | 
					
						
							|  |  |  | 					sel = 1; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if (!sel) | 
					
						
							|  |  |  | 				node->flag &= ~SELECT; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | void node_deselect_all_output_sockets(SpaceNode *snode, const bool deselect_nodes) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	bNodeSocket *sock; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* XXX not calling node_socket_deselect here each time, because this does iteration
 | 
					
						
							|  |  |  | 	 * over all node sockets internally to check if the node stays selected. | 
					
						
							|  |  |  | 	 * We can do that more efficiently here. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 		bool sel = false; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		for (sock = node->outputs.first; sock; sock = sock->next) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			sock->flag &= ~SELECT; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* if no selected sockets remain, also deselect the node */ | 
					
						
							|  |  |  | 		if (deselect_nodes) { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 			for (sock = node->inputs.first; sock; sock = sock->next) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 				if (sock->flag & SELECT) { | 
					
						
							|  |  |  | 					sel = 1; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if (!sel) | 
					
						
							|  |  |  | 				node->flag &= ~SELECT; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | /* Return true if we need redraw, otherwise false. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool node_select_grouped_type(SpaceNode *snode, bNode *node_act) | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	bNode *node; | 
					
						
							| 
									
										
										
										
											2014-07-05 17:58:21 +02:00
										 |  |  | 	bool changed = false; | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if ((node->flag & SELECT) == 0) { | 
					
						
							|  |  |  | 			if (node->type == node_act->type) { | 
					
						
							|  |  |  | 				nodeSetSelected(node, true); | 
					
						
							|  |  |  | 				changed = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	return changed; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | static bool node_select_grouped_color(SpaceNode *snode, bNode *node_act) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	bool changed = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if ((node->flag & SELECT) == 0) { | 
					
						
							|  |  |  | 			if (compare_v3v3(node->color, node_act->color, 0.005f)) { | 
					
						
							|  |  |  | 				nodeSetSelected(node, true); | 
					
						
							|  |  |  | 				changed = true; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return changed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool node_select_grouped_name(SpaceNode *snode, bNode *node_act, const bool from_right) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	bool changed = false; | 
					
						
							|  |  |  | 	const unsigned int delims[] = {'.', '-', '_', '\0'}; | 
					
						
							|  |  |  | 	size_t index_act, index_curr; | 
					
						
							|  |  |  | 	char *sep, *suf_act, *suf_curr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	index_act = BLI_str_partition_ex_utf8(node_act->name, delims, &sep, &suf_act, from_right); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (index_act > 0) { | 
					
						
							|  |  |  | 		for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 			if ((node->flag & SELECT) == 0) { | 
					
						
							|  |  |  | 				index_curr = BLI_str_partition_ex_utf8(node->name, delims, &sep, &suf_curr, from_right); | 
					
						
							|  |  |  | 				if ((from_right && STREQ(suf_act, suf_curr)) || | 
					
						
							|  |  |  | 				    (!from_right && (index_act == index_curr) && STREQLEN(node_act->name, node->name, index_act))) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					nodeSetSelected(node, true); | 
					
						
							|  |  |  | 					changed = true; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return changed; | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | enum { | 
					
						
							|  |  |  | 	NODE_SELECT_GROUPED_TYPE   = 0, | 
					
						
							|  |  |  | 	NODE_SELECT_GROUPED_COLOR  = 1, | 
					
						
							|  |  |  | 	NODE_SELECT_GROUPED_PREFIX = 2, | 
					
						
							|  |  |  | 	NODE_SELECT_GROUPED_SUFIX  = 3, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_select_grouped_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNode *node_act = nodeGetActive(snode->edittree); | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	bool changed = false; | 
					
						
							|  |  |  | 	const bool extend = RNA_boolean_get(op->ptr, "extend"); | 
					
						
							|  |  |  | 	const int type = RNA_enum_get(op->ptr, "type"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!extend) { | 
					
						
							|  |  |  | 		for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 			nodeSetSelected(node, false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	nodeSetSelected(node_act, true); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	switch (type) { | 
					
						
							|  |  |  | 		case NODE_SELECT_GROUPED_TYPE: | 
					
						
							|  |  |  | 			changed = node_select_grouped_type(snode, node_act); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case NODE_SELECT_GROUPED_COLOR: | 
					
						
							|  |  |  | 			changed = node_select_grouped_color(snode, node_act); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case NODE_SELECT_GROUPED_PREFIX: | 
					
						
							|  |  |  | 			changed = node_select_grouped_name(snode, node_act, false); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case NODE_SELECT_GROUPED_SUFIX: | 
					
						
							|  |  |  | 			changed = node_select_grouped_name(snode, node_act, true); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	if (changed) { | 
					
						
							|  |  |  | 		ED_node_sort(snode->edittree); | 
					
						
							|  |  |  | 		WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							|  |  |  | 		return OPERATOR_FINISHED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | void NODE_OT_select_grouped(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static EnumPropertyItem prop_select_grouped_types[] = { | 
					
						
							|  |  |  | 		{NODE_SELECT_GROUPED_TYPE, "TYPE", 0, "Type", ""}, | 
					
						
							|  |  |  | 		{NODE_SELECT_GROUPED_COLOR, "COLOR", 0, "Color", ""}, | 
					
						
							|  |  |  | 		{NODE_SELECT_GROUPED_PREFIX, "PREFIX", 0, "Prefix", ""}, | 
					
						
							|  |  |  | 		{NODE_SELECT_GROUPED_SUFIX, "SUFFIX", 0, "Suffix", ""}, | 
					
						
							|  |  |  | 		{0, NULL, 0, NULL, NULL} | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Select Grouped"; | 
					
						
							|  |  |  | 	ot->description = "Select nodes with similar properties"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select_grouped"; | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 14:17:54 +02:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->invoke = WM_menu_invoke; | 
					
						
							|  |  |  | 	ot->exec = node_select_grouped_exec; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* properties */ | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "extend", false, "Extend", "Extend selection instead of deselecting everything first"); | 
					
						
							|  |  |  | 	ot->prop = RNA_def_enum(ot->srna, "type", prop_select_grouped_types, 0, "Type", ""); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void node_select_single(bContext *C, bNode *node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	bNode *tnode; | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (tnode = snode->edittree->nodes.first; tnode; tnode = tnode->next) | 
					
						
							|  |  |  | 		if (tnode != node) | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			nodeSetSelected(tnode, false); | 
					
						
							|  |  |  | 	nodeSetSelected(node, true); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	ED_node_set_active(bmain, snode->edittree, node); | 
					
						
							| 
									
										
										
										
											2013-04-24 16:36:50 +00:00
										 |  |  | 	ED_node_set_active_viewer_key(snode); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 	ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2011-12-18 12:51:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-20 21:04:41 +00:00
										 |  |  | /* ****** Click Select ****** */ | 
					
						
							|  |  |  |   | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | static int node_mouse_select(Main *bmain, SpaceNode *snode, ARegion *ar, const int mval[2], short extend) | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	bNode *node, *tnode; | 
					
						
							|  |  |  | 	bNodeSocket *sock, *tsock; | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 	float cursor[2]; | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 	int selected = 0; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	/* get mouse coordinates in view2d space */ | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 	UI_view2d_region_to_view(&ar->v2d, mval[0], mval[1], &cursor[0], &cursor[1]); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 	if (extend) { | 
					
						
							|  |  |  | 		/* first do socket selection, these generally overlap with nodes.
 | 
					
						
							|  |  |  | 		 * socket selection only in extend mode. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 		if (node_find_indicated_socket(snode, &node, &sock, cursor, SOCK_IN)) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			node_socket_toggle(node, sock, 1); | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 			selected = 1; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 		else if (node_find_indicated_socket(snode, &node, &sock, cursor, SOCK_OUT)) { | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			if (sock->flag & SELECT) { | 
					
						
							|  |  |  | 				node_socket_deselect(node, sock, 1); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 				/* only allow one selected output per node, for sensible linking.
 | 
					
						
							| 
									
										
										
										
											2012-04-22 11:54:53 +00:00
										 |  |  | 				 * allows selecting outputs from different nodes though. | 
					
						
							|  |  |  | 				 */ | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 				if (node) { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 					for (tsock = node->outputs.first; tsock; tsock = tsock->next) | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 						node_socket_deselect(node, tsock, 1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				node_socket_select(node, sock); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 			selected = 1; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 			/* find the closest visible node */ | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 			node = node_under_mouse_select(snode->edittree, cursor[0], cursor[1]); | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			if (node) { | 
					
						
							| 
									
										
										
										
											2012-08-02 16:01:05 +00:00
										 |  |  | 				if ((node->flag & SELECT) && (node->flag & NODE_ACTIVE) == 0) { | 
					
						
							|  |  |  | 					/* if node is selected but not active make it active
 | 
					
						
							|  |  |  | 					 * before it'll be desleected | 
					
						
							|  |  |  | 					 */ | 
					
						
							|  |  |  | 					ED_node_set_active(bmain, snode->edittree, node); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					node_toggle(node); | 
					
						
							|  |  |  | 					ED_node_set_active(bmain, snode->edittree, node); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 				selected = 1; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-08-11 22:12:32 +00:00
										 |  |  | 	else {  /* extend == 0 */ | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		/* find the closest visible node */ | 
					
						
							| 
									
										
										
										
											2013-09-05 13:03:03 +00:00
										 |  |  | 		node = node_under_mouse_select(snode->edittree, cursor[0], cursor[1]); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		if (node) { | 
					
						
							| 
									
										
										
										
											2013-03-18 18:25:05 +00:00
										 |  |  | 			for (tnode = snode->edittree->nodes.first; tnode; tnode = tnode->next) { | 
					
						
							|  |  |  | 				nodeSetSelected(tnode, false); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			nodeSetSelected(node, true); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 			ED_node_set_active(bmain, snode->edittree, node); | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 			selected = 1; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* update node order */ | 
					
						
							| 
									
										
										
										
											2013-04-24 16:36:50 +00:00
										 |  |  | 	if (selected) { | 
					
						
							|  |  |  | 		ED_node_set_active_viewer_key(snode); | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 		ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2013-04-24 16:36:50 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-14 17:52:50 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return selected; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_select_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	ARegion *ar = CTX_wm_region(C); | 
					
						
							| 
									
										
										
										
											2011-05-12 16:47:36 +00:00
										 |  |  | 	int mval[2]; | 
					
						
							| 
									
										
										
										
											2009-01-02 23:58:03 +00:00
										 |  |  | 	short extend; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	/* get settings from RNA properties for operator */ | 
					
						
							|  |  |  | 	mval[0] = RNA_int_get(op->ptr, "mouse_x"); | 
					
						
							|  |  |  | 	mval[1] = RNA_int_get(op->ptr, "mouse_y"); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	extend = RNA_boolean_get(op->ptr, "extend"); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* perform the select */ | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	if (node_mouse_select(bmain, snode, ar, mval, extend)) { | 
					
						
							|  |  |  | 		/* send notifiers */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* allow tweak event to work too */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		return OPERATOR_FINISHED | OPERATOR_PASS_THROUGH; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		/* allow tweak event to work too */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		return OPERATOR_CANCELLED | OPERATOR_PASS_THROUGH; | 
					
						
							| 
									
										
										
										
											2012-03-09 10:16:41 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-01-12 00:00:59 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-13 09:03:46 +00:00
										 |  |  | static int node_select_invoke(bContext *C, wmOperator *op, const wmEvent *event) | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-05-20 07:40:05 +00:00
										 |  |  | 	RNA_int_set(op->ptr, "mouse_x", event->mval[0]); | 
					
						
							|  |  |  | 	RNA_int_set(op->ptr, "mouse_y", event->mval[1]); | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	return node_select_exec(C, op); | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-28 20:49:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | void NODE_OT_select(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->name = "Select"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select"; | 
					
						
							|  |  |  | 	ot->description = "Select the node under the cursor"; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->invoke = node_select_invoke; | 
					
						
							| 
									
										
										
										
											2013-07-12 08:31:39 +00:00
										 |  |  | 	ot->exec = node_select_exec; | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	/* properties */ | 
					
						
							| 
									
										
										
										
											2009-01-16 23:53:11 +00:00
										 |  |  | 	RNA_def_int(ot->srna, "mouse_x", 0, INT_MIN, INT_MAX, "Mouse X", "", INT_MIN, INT_MAX); | 
					
						
							|  |  |  | 	RNA_def_int(ot->srna, "mouse_y", 0, INT_MIN, INT_MAX, "Mouse Y", "", INT_MIN, INT_MAX); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "extend", 0, "Extend", ""); | 
					
						
							| 
									
										
										
										
											2008-12-28 00:08:34 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ****** Border Select ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_borderselect_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	ARegion *ar = CTX_wm_region(C); | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	rctf rectf; | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	int gesture_mode = RNA_int_get(op->ptr, "gesture_mode"); | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | 	const bool extend = RNA_boolean_get(op->ptr, "extend"); | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2014-04-21 16:47:16 +10:00
										 |  |  | 	WM_operator_properties_border_to_rctf(op, &rectf); | 
					
						
							|  |  |  | 	UI_view2d_region_to_view_rctf(&ar->v2d, &rectf, &rectf); | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2012-07-15 00:29:56 +00:00
										 |  |  | 		if (BLI_rctf_isect(&rectf, &node->totr, NULL)) { | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 			nodeSetSelected(node, (gesture_mode == GESTURE_MODAL_SELECT)); | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		else if (!extend) { | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			nodeSetSelected(node, false); | 
					
						
							| 
									
										
										
										
											2011-11-16 19:22:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 	ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2009-01-07 16:49:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-13 09:03:46 +00:00
										 |  |  | static int node_border_select_invoke(bContext *C, wmOperator *op, const wmEvent *event) | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	const bool tweak = RNA_boolean_get(op->ptr, "tweak"); | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (tweak) { | 
					
						
							|  |  |  | 		/* prevent initiating the border select if the mouse is over a node */ | 
					
						
							|  |  |  | 		/* this allows border select on empty space, but drag-translate on nodes */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 		SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 		ARegion *ar = CTX_wm_region(C); | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 		float mx, my; | 
					
						
							| 
									
										
										
										
											2011-05-20 07:40:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		UI_view2d_region_to_view(&ar->v2d, event->mval[0], event->mval[1], &mx, &my); | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-01 12:38:03 +00:00
										 |  |  | 		if (node_under_mouse_tweak(snode->edittree, mx, my)) | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 			return OPERATOR_CANCELLED | OPERATOR_PASS_THROUGH; | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return WM_border_select_invoke(C, op, event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-29 02:15:13 +00:00
										 |  |  | void NODE_OT_select_border(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->name = "Border Select"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select_border"; | 
					
						
							|  |  |  | 	ot->description = "Use box selection to select nodes"; | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->invoke = node_border_select_invoke; | 
					
						
							|  |  |  | 	ot->exec = node_borderselect_exec; | 
					
						
							|  |  |  | 	ot->modal = WM_border_select_modal; | 
					
						
							|  |  |  | 	ot->cancel = WM_border_select_cancel; | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-22 07:26:09 +00:00
										 |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-01-05 23:53:04 +00:00
										 |  |  | 	/* rna */ | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 	WM_operator_properties_gesture_border(ot, true); | 
					
						
							| 
									
										
										
										
											2010-01-06 03:00:19 +00:00
										 |  |  | 	RNA_def_boolean(ot->srna, "tweak", 0, "Tweak", "Only activate when mouse is not over a node - useful for tweak gesture"); | 
					
						
							| 
									
										
										
										
											2009-01-16 23:53:11 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-06 19:21:42 +00:00
										 |  |  | /* ****** Circle Select ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_circleselect_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	ARegion *ar = CTX_wm_region(C); | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int x, y, radius, gesture_mode; | 
					
						
							|  |  |  | 	float offset[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	float zoom  = (float)(BLI_rcti_size_x(&ar->winrct)) / (float)(BLI_rctf_size_x(&ar->v2d.cur)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gesture_mode = RNA_int_get(op->ptr, "gesture_mode"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* get operator properties */ | 
					
						
							|  |  |  | 	x = RNA_int_get(op->ptr, "x"); | 
					
						
							|  |  |  | 	y = RNA_int_get(op->ptr, "y"); | 
					
						
							|  |  |  | 	radius = RNA_int_get(op->ptr, "radius"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	UI_view2d_region_to_view(&ar->v2d, x, y, &offset[0], &offset[1]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (BLI_rctf_isect_circle(&node->totr, offset, radius / zoom)) { | 
					
						
							|  |  |  | 			nodeSetSelected(node, (gesture_mode == GESTURE_MODAL_SELECT)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_select_circle(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Circle Select"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select_circle"; | 
					
						
							|  |  |  | 	ot->description = "Use circle selection to select nodes"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->invoke = WM_gesture_circle_invoke; | 
					
						
							|  |  |  | 	ot->exec = node_circleselect_exec; | 
					
						
							|  |  |  | 	ot->modal = WM_gesture_circle_modal; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* rna */ | 
					
						
							|  |  |  | 	RNA_def_int(ot->srna, "x", 0, INT_MIN, INT_MAX, "X", "", INT_MIN, INT_MAX); | 
					
						
							|  |  |  | 	RNA_def_int(ot->srna, "y", 0, INT_MIN, INT_MAX, "Y", "", INT_MIN, INT_MAX); | 
					
						
							| 
									
										
										
										
											2014-06-22 13:02:26 +10:00
										 |  |  | 	RNA_def_int(ot->srna, "radius", 1, 1, INT_MAX, "Radius", "", 1, INT_MAX); | 
					
						
							| 
									
										
										
										
											2013-11-06 19:21:42 +00:00
										 |  |  | 	RNA_def_int(ot->srna, "gesture_mode", 0, INT_MIN, INT_MAX, "Gesture Mode", "", INT_MIN, INT_MAX); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | /* ****** Lasso Select ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | static bool do_lasso_select_node(bContext *C, const int mcords[][2], short moves, short select) | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ARegion *ar = CTX_wm_region(C); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rcti rect; | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | 	bool changed = false; | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* get rectangle from operator */ | 
					
						
							|  |  |  | 	BLI_lasso_boundbox(&rect, mcords, moves); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* do actual selection */ | 
					
						
							|  |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		int screen_co[2]; | 
					
						
							| 
									
										
										
										
											2012-09-15 11:48:20 +00:00
										 |  |  | 		const float cent[2] = {BLI_rctf_cent_x(&node->totr), | 
					
						
							|  |  |  | 		                       BLI_rctf_cent_y(&node->totr)}; | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* marker in screen coords */ | 
					
						
							| 
									
										
										
										
											2014-04-21 18:11:47 +10:00
										 |  |  | 		if (UI_view2d_view_to_region_clip(&ar->v2d, cent[0], cent[1], &screen_co[0], &screen_co[1]) && | 
					
						
							|  |  |  | 		    BLI_rcti_isect_pt(&rect, screen_co[0], screen_co[1]) && | 
					
						
							| 
									
										
										
										
											2014-01-12 22:05:24 +11:00
										 |  |  | 		    BLI_lasso_is_point_inside(mcords, moves, screen_co[0], screen_co[1], INT_MAX)) | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 			nodeSetSelected(node, select); | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | 			changed = true; | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | 	if (changed) { | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 06:39:14 +11:00
										 |  |  | 	return changed; | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_lasso_select_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int mcords_tot; | 
					
						
							| 
									
										
										
										
											2012-10-05 15:44:11 +00:00
										 |  |  | 	const int (*mcords)[2] = WM_gesture_lasso_path_to_array(C, op, &mcords_tot); | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mcords) { | 
					
						
							|  |  |  | 		short select; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		select = !RNA_boolean_get(op->ptr, "deselect"); | 
					
						
							|  |  |  | 		do_lasso_select_node(C, mcords, mcords_tot, select); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-14 07:40:16 +00:00
										 |  |  | 		MEM_freeN((void *)mcords); | 
					
						
							| 
									
										
										
										
											2012-08-22 13:34:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return OPERATOR_FINISHED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return OPERATOR_PASS_THROUGH; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_select_lasso(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Lasso Select"; | 
					
						
							|  |  |  | 	ot->description = "Select nodes using lasso selection"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select_lasso"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->invoke = WM_gesture_lasso_invoke; | 
					
						
							|  |  |  | 	ot->modal = WM_gesture_lasso_modal; | 
					
						
							|  |  |  | 	ot->exec = node_lasso_select_exec; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	ot->cancel = WM_gesture_lasso_cancel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag = OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* properties */ | 
					
						
							|  |  |  | 	RNA_def_collection_runtime(ot->srna, "path", &RNA_OperatorMousePath, "Path", ""); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "deselect", 0, "Deselect", "Deselect rather than select items"); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "extend", 1, "Extend", "Extend selection instead of deselecting everything first"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | /* ****** Select/Deselect All ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | static int node_select_all_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 	ListBase *node_lb = &snode->edittree->nodes; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	bNode *node; | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 	int action = RNA_enum_get(op->ptr, "action"); | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 	if (action == SEL_TOGGLE) { | 
					
						
							|  |  |  | 		if (ED_node_select_check(node_lb)) | 
					
						
							|  |  |  | 			action = SEL_DESELECT; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			action = SEL_SELECT; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (node = node_lb->first; node; node = node->next) { | 
					
						
							|  |  |  | 		switch (action) { | 
					
						
							|  |  |  | 			case SEL_SELECT: | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 				nodeSetSelected(node, true); | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			case SEL_DESELECT: | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 				nodeSetSelected(node, false); | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			case SEL_INVERT: | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 				nodeSetSelected(node, !(node->flag & SELECT)); | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 	ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_select_all(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2012-03-17 19:14:08 +00:00
										 |  |  | 	ot->name = "(De)select All"; | 
					
						
							| 
									
										
										
										
											2010-02-10 21:15:44 +00:00
										 |  |  | 	ot->description = "(De)select all nodes"; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	ot->idname = "NODE_OT_select_all"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec = node_select_all_exec; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2012-08-01 13:28:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	WM_operator_properties_select_all(ot); | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ****** Select Linked To ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int node_select_linked_to_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNodeLink *link; | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 		node->flag &= ~NODE_TEST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (link = snode->edittree->links.first; link; link = link->next) { | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 		if (nodeLinkIsHidden(link)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2011-02-21 13:47:49 +00:00
										 |  |  | 		if (link->fromnode && link->tonode && (link->fromnode->flag & NODE_SELECT)) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 			link->tonode->flag |= NODE_TEST; | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 		if (node->flag & NODE_TEST) | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			nodeSetSelected(node, true); | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 	ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_select_linked_to(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Select Linked To"; | 
					
						
							| 
									
										
										
										
											2010-02-10 21:15:44 +00:00
										 |  |  | 	ot->description = "Select nodes linked to the selected ones"; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	ot->idname = "NODE_OT_select_linked_to"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec = node_select_linked_to_exec; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ****** Select Linked From ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int node_select_linked_from_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNodeLink *link; | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 		node->flag &= ~NODE_TEST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (link = snode->edittree->links.first; link; link = link->next) { | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 		if (nodeLinkIsHidden(link)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (link->fromnode && link->tonode && (link->tonode->flag & NODE_SELECT)) | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 			link->fromnode->flag |= NODE_TEST; | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (node->flag & NODE_TEST) | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			nodeSetSelected(node, true); | 
					
						
							| 
									
										
										
										
											2009-11-20 04:19:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-22 14:13:33 +00:00
										 |  |  | 	ED_node_sort(snode->edittree); | 
					
						
							| 
									
										
										
										
											2011-09-05 21:01:50 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_NODE | NA_SELECTED, NULL); | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_select_linked_from(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Select Linked From"; | 
					
						
							| 
									
										
										
										
											2010-02-10 21:15:44 +00:00
										 |  |  | 	ot->description = "Select nodes linked from the selected ones"; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | 	ot->idname = "NODE_OT_select_linked_from"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec = node_select_linked_from_exec; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2012-08-04 12:54:27 +00:00
										 |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-09-14 08:47:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 18:28:25 +00:00
										 |  |  | static int node_select_same_type_step_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	ARegion *ar = CTX_wm_region(C); | 
					
						
							|  |  |  | 	bNode **node_array; | 
					
						
							|  |  |  | 	bNode *active = nodeGetActive(snode->edittree); | 
					
						
							|  |  |  | 	int totnodes; | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	const bool revert = RNA_boolean_get(op->ptr, "prev"); | 
					
						
							|  |  |  | 	const bool same_type = 1; | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	ntreeGetDependencyList(snode->edittree, &node_array, &totnodes); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (totnodes > 1) { | 
					
						
							|  |  |  | 		int a; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		for (a = 0; a < totnodes; a++) { | 
					
						
							|  |  |  | 			if (node_array[a] == active) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (same_type) { | 
					
						
							|  |  |  | 			bNode *node = NULL; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			while (node == NULL) { | 
					
						
							|  |  |  | 				if (revert) a--; | 
					
						
							|  |  |  | 				else a++; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (a < 0 || a >= totnodes) | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				node = node_array[a]; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (node->type == active->type) | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				else node = NULL; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (node) | 
					
						
							|  |  |  | 				active = node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			if (revert) { | 
					
						
							|  |  |  | 				if (a == 0) | 
					
						
							|  |  |  | 					active = node_array[totnodes - 1]; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					active = node_array[a - 1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				if (a == totnodes - 1) | 
					
						
							|  |  |  | 					active = node_array[0]; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					active = node_array[a + 1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		node_select_single(C, active); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* is note outside view? */ | 
					
						
							|  |  |  | 		if (active->totr.xmax < ar->v2d.cur.xmin || active->totr.xmin > ar->v2d.cur.xmax || | 
					
						
							| 
									
										
										
										
											2013-03-28 19:33:14 +00:00
										 |  |  | 		    active->totr.ymax < ar->v2d.cur.ymin || active->totr.ymin > ar->v2d.cur.ymax) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-09-16 04:04:44 +00:00
										 |  |  | 			const int smooth_viewtx = WM_operator_smooth_viewtx_get(op); | 
					
						
							|  |  |  | 			space_node_view_flag(C, snode, ar, NODE_SELECT, smooth_viewtx); | 
					
						
							| 
									
										
										
										
											2013-03-28 19:33:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (node_array) | 
					
						
							|  |  |  | 		MEM_freeN(node_array); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 18:28:25 +00:00
										 |  |  | void NODE_OT_select_same_type_step(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2013-03-27 18:28:25 +00:00
										 |  |  | 	ot->name = "Activate Same Type Next/Prev"; | 
					
						
							|  |  |  | 	ot->description = "Activate and view same node type, step by step"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_select_same_type_step"; | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2013-03-27 18:28:25 +00:00
										 |  |  | 	ot->exec = node_select_same_type_step_exec; | 
					
						
							| 
									
										
										
										
											2013-03-27 17:22:12 +00:00
										 |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "prev", 0, "Previous", ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | /* *************** find a node **************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* generic  search invoke */ | 
					
						
							|  |  |  | static void node_find_cb(const struct bContext *C, void *UNUSED(arg), const char *str, uiSearchItems *items) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	for (node = snode->edittree->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (BLI_strcasestr(node->name, str) || BLI_strcasestr(node->label, str)) { | 
					
						
							|  |  |  | 			char name[256]; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if (node->label[0]) | 
					
						
							|  |  |  | 				BLI_snprintf(name, 256, "%s (%s)", node->name, node->label); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				BLI_strncpy(name, node->name, 256); | 
					
						
							| 
									
										
										
										
											2013-04-04 02:05:11 +00:00
										 |  |  | 			if (false == uiSearchItemAdd(items, name, node, 0)) | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void node_find_call_cb(struct bContext *C, void *UNUSED(arg1), void *arg2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	SpaceNode *snode = CTX_wm_space_node(C); | 
					
						
							|  |  |  | 	bNode *active = arg2; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (active) { | 
					
						
							|  |  |  | 		ARegion *ar = CTX_wm_region(C); | 
					
						
							|  |  |  | 		node_select_single(C, active); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* is note outside view? */ | 
					
						
							|  |  |  | 		if (active->totr.xmax < ar->v2d.cur.xmin || active->totr.xmin > ar->v2d.cur.xmax || | 
					
						
							|  |  |  | 		    active->totr.ymax < ar->v2d.cur.ymin || active->totr.ymin > ar->v2d.cur.ymax) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2013-09-16 04:04:44 +00:00
										 |  |  | 			space_node_view_flag(C, snode, ar, NODE_SELECT, U.smooth_viewtx); | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uiBlock *node_find_menu(bContext *C, ARegion *ar, void *arg_op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static char search[256] = ""; | 
					
						
							|  |  |  | 	wmEvent event; | 
					
						
							|  |  |  | 	wmWindow *win = CTX_wm_window(C); | 
					
						
							|  |  |  | 	uiBlock *block; | 
					
						
							|  |  |  | 	uiBut *but; | 
					
						
							|  |  |  | 	wmOperator *op = (wmOperator *)arg_op; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	block = uiBeginBlock(C, ar, "_popup", UI_EMBOSS); | 
					
						
							|  |  |  | 	uiBlockSetFlag(block, UI_BLOCK_LOOP | UI_BLOCK_MOVEMOUSE_QUIT | UI_BLOCK_SEARCH_MENU); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	but = uiDefSearchBut(block, search, 0, ICON_VIEWZOOM, sizeof(search), 10, 10, 9 * UI_UNIT_X, UI_UNIT_Y, 0, 0, ""); | 
					
						
							|  |  |  | 	uiButSetSearchFunc(but, node_find_cb, op->type, node_find_call_cb, NULL); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* fake button, it holds space for search items */ | 
					
						
							|  |  |  | 	uiDefBut(block, LABEL, 0, "", 10, 10 - uiSearchBoxHeight(), uiSearchBoxWidth(), uiSearchBoxHeight(), NULL, 0, 0, 0, 0, NULL); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	uiPopupBoundsBlock(block, 6, 0, -UI_UNIT_Y); /* move it downwards, mouse over button */ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	//	uiButActiveOnly(C, ar, block, but); XXX using this here makes Blender hang - investigate
 | 
					
						
							| 
									
										
										
										
											2013-07-29 08:29:04 +00:00
										 |  |  | 	wm_event_init_from_window(win, &event); | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | 	event.type = EVT_BUT_OPEN; | 
					
						
							|  |  |  | 	event.val = KM_PRESS; | 
					
						
							|  |  |  | 	event.customdata = but; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 	event.customdatafree = false; | 
					
						
							| 
									
										
										
										
											2013-04-01 15:07:22 +00:00
										 |  |  | 	wm_event_add(win, &event); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return block; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int node_find_node_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	uiPupBlock(C, node_find_menu, op); | 
					
						
							|  |  |  | 	return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NODE_OT_find_node(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name = "Find Node"; | 
					
						
							|  |  |  | 	ot->description = "Search for named node and allow to select and activate it"; | 
					
						
							|  |  |  | 	ot->idname = "NODE_OT_find_node"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->invoke = node_find_node_invoke; | 
					
						
							|  |  |  | 	ot->poll = ED_operator_node_active; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "prev", 0, "Previous", ""); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 |