| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +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-05-28 16:39:05 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Martin Poirier | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * graph.c: Common graph interface and methods | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:37:56 +00:00
										 |  |  | /** \file blender/blenlib/intern/graph.c
 | 
					
						
							|  |  |  |  *  \ingroup bli | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 11:27:47 +00:00
										 |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | #include <math.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 11:27:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | #include "BLI_graph.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:18:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | static void testRadialSymmetry(BGraph *graph, BNode *root_node, RadialArc *ring, int total, float axis[3], float limit, int group); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void handleAxialSymmetry(BGraph *graph, BNode *root_node, int depth, float axis[3], float limit); | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | static void testAxialSymmetry(BGraph *graph, BNode *root_node, BNode *node1, BNode *node2, BArc *arc1, BArc *arc2, float axis[3], float limit, int group); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | static void flagAxialSymmetry(BNode *root_node, BNode *end_node, BArc *arc, int group); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | void BLI_freeNode(BGraph *graph, BNode *node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->arcs) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		MEM_freeN(node->arcs); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (graph->free_node) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		graph->free_node(node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-05 22:01:46 +00:00
										 |  |  | void BLI_removeNode(BGraph *graph, BNode *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_freeNode(graph, node); | 
					
						
							|  |  |  | 	BLI_freelinkN(&graph->nodes, node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | BNode *BLI_otherNode(BArc *arc, BNode *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (arc->head == node) ? arc->tail : arc->head; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | void BLI_removeArc(BGraph *graph, BArc *arc) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (graph->free_arc) { | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 		graph->free_arc(arc); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_freelinkN(&graph->arcs, arc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | void BLI_flagNodes(BGraph *graph, int flag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		node->flag = flag; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_flagArcs(BGraph *graph, int flag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BArc *arc; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (arc = graph->arcs.first; arc; arc = arc->next) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		arc->flag = flag; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void addArcToNodeAdjacencyList(BNode *node, BArc *arc) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 	node->arcs[node->flag] = arc; | 
					
						
							|  |  |  | 	node->flag++; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-20 21:34:49 +00:00
										 |  |  | void BLI_buildAdjacencyList(BGraph *graph) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 	BArc *arc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->arcs != NULL) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			MEM_freeN(node->arcs); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 		node->arcs = MEM_callocN((node->degree) * sizeof(BArc *), "adjacency list"); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* temporary use to indicate the first index available in the lists */ | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 		node->flag = 0; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	for (arc = graph->arcs.first; arc; arc = arc->next) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		addArcToNodeAdjacencyList(arc->head, arc); | 
					
						
							|  |  |  | 		addArcToNodeAdjacencyList(arc->tail, arc); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->degree != node->flag) { | 
					
						
							| 
									
										
										
										
											2010-12-03 01:52:28 +00:00
										 |  |  | 			printf("error in node [%p]. Added only %i arcs out of %i\n", (void *)node, node->flag, node->degree); | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void BLI_rebuildAdjacencyListForNode(BGraph *graph, BNode *node) | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BArc *arc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->arcs != NULL) { | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | 		MEM_freeN(node->arcs); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	node->arcs = MEM_callocN((node->degree) * sizeof(BArc *), "adjacency list"); | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* temporary use to indicate the first index available in the lists */ | 
					
						
							|  |  |  | 	node->flag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	for (arc = graph->arcs.first; arc; arc = arc->next) { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (arc->head == node) { | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | 			addArcToNodeAdjacencyList(arc->head, arc); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		else if (arc->tail == node) { | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | 			addArcToNodeAdjacencyList(arc->tail, arc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->degree != node->flag) { | 
					
						
							| 
									
										
										
										
											2010-12-03 01:52:28 +00:00
										 |  |  | 		printf("error in node [%p]. Added only %i arcs out of %i\n", (void *)node, node->flag, node->degree); | 
					
						
							| 
									
										
										
										
											2008-07-15 21:07:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-20 21:34:49 +00:00
										 |  |  | void BLI_freeAdjacencyList(BGraph *graph) | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->arcs != NULL) { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 			MEM_freeN(node->arcs); | 
					
						
							|  |  |  | 			node->arcs = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool BLI_hasAdjacencyList(BGraph *graph) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->arcs == NULL) { | 
					
						
							| 
									
										
										
										
											2014-12-01 14:33:38 +01:00
										 |  |  | 			return false; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2014-12-01 14:33:38 +01:00
										 |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | void BLI_replaceNodeInArc(BGraph *graph, BArc *arc, BNode *node_src, BNode *node_replaced) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (arc->head == node_replaced) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		arc->head = node_src; | 
					
						
							|  |  |  | 		node_src->degree++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (arc->tail == node_replaced) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		arc->tail = node_src; | 
					
						
							|  |  |  | 		node_src->degree++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (arc->head == arc->tail) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		node_src->degree -= 2; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		graph->free_arc(arc); | 
					
						
							|  |  |  | 		BLI_freelinkN(&graph->arcs, arc); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node_replaced->degree == 0) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		BLI_removeNode(graph, node_replaced); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | void BLI_replaceNode(BGraph *graph, BNode *node_src, BNode *node_replaced) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BArc *arc, *next_arc; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (arc = graph->arcs.first; arc; arc = next_arc) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		next_arc = arc->next; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (arc->head == node_replaced) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			arc->head = node_src; | 
					
						
							| 
									
										
										
										
											2008-08-20 21:34:49 +00:00
										 |  |  | 			node_replaced->degree--; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			node_src->degree++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (arc->tail == node_replaced) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			arc->tail = node_src; | 
					
						
							| 
									
										
										
										
											2008-08-20 21:34:49 +00:00
										 |  |  | 			node_replaced->degree--; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			node_src->degree++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (arc->head == arc->tail) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			node_src->degree -= 2; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			graph->free_arc(arc); | 
					
						
							|  |  |  | 			BLI_freelinkN(&graph->arcs, arc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node_replaced->degree == 0) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		BLI_removeNode(graph, node_replaced); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_removeDoubleNodes(BGraph *graph, float limit) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 	const float limit_sq = limit * limit; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	BNode *node_src, *node_replaced; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node_src = graph->nodes.first; node_src; node_src = node_src->next) { | 
					
						
							|  |  |  | 		for (node_replaced = graph->nodes.first; node_replaced; node_replaced = node_replaced->next) { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 			if (node_replaced != node_src && len_squared_v3v3(node_replaced->p, node_src->p) <= limit_sq) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				BLI_replaceNode(graph, node_src, node_replaced); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 22:13:38 +00:00
										 |  |  | BNode *BLI_FindNodeByPosition(BGraph *graph, const float p[3], const float limit) | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 	const float limit_sq = limit * limit; | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 	BNode *closest_node = NULL, *node; | 
					
						
							| 
									
										
										
										
											2009-01-23 20:36:47 +00:00
										 |  |  | 	float min_distance = 0.0f; | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 		float distance = len_squared_v3v3(p, node->p); | 
					
						
							|  |  |  | 		if (distance <= limit_sq && (closest_node == NULL || distance < min_distance)) { | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 			closest_node = node; | 
					
						
							|  |  |  | 			min_distance = distance; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return closest_node; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | /************************************* SUBGRAPH DETECTION **********************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-13 10:52:18 +00:00
										 |  |  | static void flagSubgraph(BNode *node, int subgraph) | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->subgraph_index == 0) { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 		BArc *arc; | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 		node->subgraph_index = subgraph; | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 			arc = node->arcs[i]; | 
					
						
							|  |  |  | 			flagSubgraph(BLI_otherNode(arc, node), subgraph); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int BLI_FlagSubgraphs(BGraph *graph) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 	int subgraph = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (BLI_hasAdjacencyList(graph) == 0) { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 		BLI_buildAdjacencyList(graph); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 		node->subgraph_index = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->subgraph_index == 0) { | 
					
						
							| 
									
										
										
										
											2008-06-26 18:15:45 +00:00
										 |  |  | 			subgraph++; | 
					
						
							|  |  |  | 			flagSubgraph(node, subgraph); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return subgraph; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-29 20:20:24 +00:00
										 |  |  | void BLI_ReflagSubgraph(BGraph *graph, int old_subgraph, int new_subgraph) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							|  |  |  | 		if (node->flag == old_subgraph) { | 
					
						
							| 
									
										
										
										
											2008-07-29 20:20:24 +00:00
										 |  |  | 			node->flag = new_subgraph; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | /*************************************** CYCLE DETECTION ***********************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | static bool detectCycle(BNode *node, BArc *src_arc) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	bool value = false; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->flag == 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* mark node as visited */ | 
					
						
							|  |  |  | 		node->flag = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (i = 0; i < node->degree && value == 0; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			BArc *arc = node->arcs[i]; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* don't go back on the source arc */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (arc != src_arc) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				value = detectCycle(BLI_otherNode(arc, node), arc); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2014-12-01 14:33:38 +01:00
										 |  |  | 		value = true; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool BLI_isGraphCyclic(BGraph *graph) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	bool value = false; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* NEED TO CHECK IF ADJACENCY LIST EXIST */ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* Mark all nodes as not visited */ | 
					
						
							|  |  |  | 	BLI_flagNodes(graph, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	/* detectCycles in subgraphs */ | 
					
						
							| 
									
										
										
										
											2014-12-01 14:33:38 +01:00
										 |  |  | 	for (node = graph->nodes.first; node && value == false; node = node->next) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		/* only for nodes in subgraphs that haven't been visited yet */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (node->flag == 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			value = value || detectCycle(node, NULL); | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | BArc *BLI_findConnectedArc(BGraph *graph, BArc *arc, BNode *v) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 03:41:12 +00:00
										 |  |  | 	BArc *nextArc; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (nextArc = graph->arcs.first; nextArc; nextArc = nextArc->next) { | 
					
						
							|  |  |  | 		if (arc != nextArc && (nextArc->head == v || nextArc->tail == v)) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return nextArc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*********************************** GRAPH AS TREE FUNCTIONS *******************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-13 10:52:18 +00:00
										 |  |  | static int subtreeShape(BNode *node, BArc *rootArc, int include_root) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int depth = 0; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 	node->flag = 1; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (include_root) { | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 		BNode *newNode = BLI_otherNode(rootArc, node); | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 		return subtreeShape(newNode, rootArc, 0); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 		/* Base case, no arcs leading away */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (node->arcs == NULL || *(node->arcs) == NULL) { | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 			int i; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 				BArc *arc = node->arcs[i]; | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 				BNode *newNode = BLI_otherNode(arc, node); | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 				/* stop immediate and cyclic backtracking */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				if (arc != rootArc && newNode->flag == 0) { | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 					depth += subtreeShape(newNode, arc, 0); | 
					
						
							| 
									
										
										
										
											2008-07-07 20:31:53 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-07-08 16:22:18 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-07-15 19:38:48 +00:00
										 |  |  | 		return SHAPE_RADIX * depth + 1; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | int BLI_subtreeShape(BGraph *graph, BNode *node, BArc *rootArc, int include_root) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_flagNodes(graph, 0); | 
					
						
							|  |  |  | 	return subtreeShape(node, rootArc, include_root); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | float BLI_subtreeLength(BNode *node) | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float length = 0; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 	node->flag = 0; /* flag node as visited */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | 		BArc *arc = node->arcs[i]; | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 		BNode *other_node = BLI_otherNode(arc, node); | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (other_node->flag != 0) { | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 			float subgraph_length = arc->length + BLI_subtreeLength(other_node);  | 
					
						
							|  |  |  | 			length = MAX2(length, subgraph_length); | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return length; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_calcGraphLength(BGraph *graph) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 	float length = 0; | 
					
						
							|  |  |  | 	int nb_subgraphs; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	nb_subgraphs = BLI_FlagSubgraphs(graph); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 1; i <= nb_subgraphs; i++) { | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 		BNode *node; | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (node = graph->nodes.first; node; node = node->next) { | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 			/* start on an external node  of the subgraph */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (node->subgraph_index == i && node->degree == 1) { | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 				float subgraph_length = BLI_subtreeLength(node); | 
					
						
							|  |  |  | 				length = MAX2(length, subgraph_length); | 
					
						
							|  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-08-04 19:12:42 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	graph->length = length; | 
					
						
							| 
									
										
										
										
											2008-07-10 18:48:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | /********************************* SYMMETRY DETECTION **************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 16:52:06 +00:00
										 |  |  | static void markdownSymmetryArc(BGraph *graph, BArc *arc, BNode *node, int level, float limit); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void BLI_mirrorAlongAxis(float v[3], float center[3], float axis[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float dv[3], pv[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	sub_v3_v3v3(dv, v, center); | 
					
						
							|  |  |  | 	project_v3_v3v3(pv, dv, axis); | 
					
						
							|  |  |  | 	mul_v3_fl(pv, -2); | 
					
						
							| 
									
										
										
										
											2010-04-21 12:27:48 +00:00
										 |  |  | 	add_v3_v3(v, pv); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | static void testRadialSymmetry(BGraph *graph, BNode *root_node, RadialArc *ring, int total, float axis[3], float limit, int group) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 	const float limit_sq = limit * limit; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	int symmetric = 1; | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* sort ring by angle */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < total - 1; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		float minAngle = FLT_MAX; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		int minIndex = -1; | 
					
						
							|  |  |  | 		int j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (j = i + 1; j < total; j++) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 			float angle = dot_v3v3(ring[i].n, ring[j].n); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/* map negative values to 1..2 */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (angle < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				angle = 1 - angle; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (angle < minAngle) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				minIndex = j; | 
					
						
							|  |  |  | 				minAngle = angle; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* swap if needed */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (minIndex != i + 1) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			RadialArc tmp; | 
					
						
							|  |  |  | 			tmp = ring[i + 1]; | 
					
						
							|  |  |  | 			ring[i + 1] = ring[minIndex]; | 
					
						
							|  |  |  | 			ring[minIndex] = tmp; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < total && symmetric; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		BNode *node1, *node2; | 
					
						
							|  |  |  | 		float tangent[3]; | 
					
						
							|  |  |  | 		float normal[3]; | 
					
						
							|  |  |  | 		float p[3]; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		int j = (i + 1) % total; /* next arc in the circular list */ | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		add_v3_v3v3(tangent, ring[i].n, ring[j].n); | 
					
						
							|  |  |  | 		cross_v3_v3v3(normal, tangent, axis); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		node1 = BLI_otherNode(ring[i].arc, root_node); | 
					
						
							|  |  |  | 		node2 = BLI_otherNode(ring[j].arc, root_node); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-12 04:14:12 +00:00
										 |  |  | 		copy_v3_v3(p, node2->p); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		BLI_mirrorAlongAxis(p, root_node->p, normal); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* check if it's within limit before continuing */ | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 		if (len_squared_v3v3(node1->p, p) > limit_sq) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			symmetric = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (symmetric) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		/* mark node as symmetric physically */ | 
					
						
							| 
									
										
										
										
											2011-09-12 04:14:12 +00:00
										 |  |  | 		copy_v3_v3(root_node->symmetry_axis, axis); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		root_node->symmetry_flag |= SYM_PHYSICAL; | 
					
						
							|  |  |  | 		root_node->symmetry_flag |= SYM_RADIAL; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		/* FLAG SYMMETRY GROUP */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (i = 0; i < total; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			ring[i].arc->symmetry_group = group; | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 			ring[i].arc->symmetry_flag = SYM_SIDE_RADIAL + i; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (graph->radial_symmetry) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			graph->radial_symmetry(root_node, ring, total); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void handleRadialSymmetry(BGraph *graph, BNode *root_node, int depth, float axis[3], float limit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RadialArc *ring = NULL; | 
					
						
							|  |  |  | 	RadialArc *unit; | 
					
						
							|  |  |  | 	int total = 0; | 
					
						
							|  |  |  | 	int group; | 
					
						
							|  |  |  | 	int first; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* mark topological symmetry */ | 
					
						
							|  |  |  | 	root_node->symmetry_flag |= SYM_TOPOLOGICAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* total the number of arcs in the symmetry ring */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < root_node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		BArc *connectedArc = root_node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* depth is store as a negative in flag. symmetry level is positive */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level == -depth) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			total++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ring = MEM_callocN(sizeof(RadialArc) * total, "radial symmetry ring"); | 
					
						
							|  |  |  | 	unit = ring; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* fill in the ring */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (unit = ring, i = 0; i < root_node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		BArc *connectedArc = root_node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* depth is store as a negative in flag. symmetry level is positive */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level == -depth) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			BNode *otherNode = BLI_otherNode(connectedArc, root_node); | 
					
						
							|  |  |  | 			float vec[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			unit->arc = connectedArc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* project the node to node vector on the symmetry plane */ | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 			sub_v3_v3v3(unit->n, otherNode->p, root_node->p); | 
					
						
							|  |  |  | 			project_v3_v3v3(vec, unit->n, axis); | 
					
						
							|  |  |  | 			sub_v3_v3v3(unit->n, unit->n, vec); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 			normalize_v3(unit->n); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			unit++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* sort ring by arc length
 | 
					
						
							|  |  |  | 	 * using a rather bogus insertion sort | 
					
						
							| 
									
										
										
										
											2013-06-25 22:58:23 +00:00
										 |  |  | 	 * but rings will never get too big to matter | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	 * */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < total; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		int j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (j = i - 1; j >= 0; j--) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			BArc *arc1, *arc2; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			arc1 = ring[j].arc; | 
					
						
							|  |  |  | 			arc2 = ring[j + 1].arc; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (arc1->length > arc2->length) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				/* swap with smaller */ | 
					
						
							|  |  |  | 				RadialArc tmp; | 
					
						
							| 
									
										
										
										
											2008-05-30 18:13:43 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				tmp = ring[j + 1]; | 
					
						
							|  |  |  | 				ring[j + 1] = ring[j]; | 
					
						
							|  |  |  | 				ring[j] = tmp; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	/* Dispatch to specific symmetry tests */ | 
					
						
							|  |  |  | 	first = 0; | 
					
						
							|  |  |  | 	group = 0; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 1; i < total; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		int dispatch = 0; | 
					
						
							|  |  |  | 		int last = i - 1; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (fabsf(ring[first].arc->length - ring[i].arc->length) > limit) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			dispatch = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* if not dispatching already and on last arc
 | 
					
						
							|  |  |  | 		 * Dispatch using current arc as last | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (dispatch == 0 && i == total - 1) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			last = i; | 
					
						
							|  |  |  | 			dispatch = 1; | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (dispatch) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			int sub_total = last - first + 1;  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			group += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (sub_total == 1) { | 
					
						
							| 
									
										
										
										
											2008-09-15 21:48:05 +00:00
										 |  |  | 				group -= 1; /* not really a group so decrement */ | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				/* NOTHING TO DO */ | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else if (sub_total == 2) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				BArc *arc1, *arc2; | 
					
						
							|  |  |  | 				BNode *node1, *node2; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				arc1 = ring[first].arc; | 
					
						
							|  |  |  | 				arc2 = ring[last].arc; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				node1 = BLI_otherNode(arc1, root_node); | 
					
						
							|  |  |  | 				node2 = BLI_otherNode(arc2, root_node); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				testAxialSymmetry(graph, root_node, node1, node2, arc1, arc2, axis, limit, group); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else if (sub_total != total) /* allocate a new sub ring if needed */ { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				RadialArc *sub_ring = MEM_callocN(sizeof(RadialArc) * sub_total, "radial symmetry ring"); | 
					
						
							|  |  |  | 				int sub_i; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				/* fill in the sub ring */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				for (sub_i = 0; sub_i < sub_total; sub_i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 					sub_ring[sub_i] = ring[first + sub_i]; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				testRadialSymmetry(graph, root_node, sub_ring, sub_total, axis, limit, group); | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 				MEM_freeN(sub_ring); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else if (sub_total == total) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				testRadialSymmetry(graph, root_node, ring, total, axis, limit, group); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			first = i; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	MEM_freeN(ring); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | static void flagAxialSymmetry(BNode *root_node, BNode *end_node, BArc *arc, int group) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	arc->symmetry_group = group; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	sub_v3_v3v3(vec, end_node->p, root_node->p); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (dot_v3v3(vec, root_node->symmetry_axis) < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		arc->symmetry_flag |= SYM_SIDE_NEGATIVE; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		arc->symmetry_flag |= SYM_SIDE_POSITIVE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | static void testAxialSymmetry(BGraph *graph, BNode *root_node, BNode *node1, BNode *node2, BArc *arc1, BArc *arc2, float axis[3], float limit, int group) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 	const float limit_sq = limit * limit; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	float nor[3], vec[3], p[3]; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	sub_v3_v3v3(p, node1->p, root_node->p); | 
					
						
							|  |  |  | 	cross_v3_v3v3(nor, p, axis); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	sub_v3_v3v3(p, root_node->p, node2->p); | 
					
						
							|  |  |  | 	cross_v3_v3v3(vec, p, axis); | 
					
						
							| 
									
										
										
										
											2010-04-21 12:27:48 +00:00
										 |  |  | 	add_v3_v3(vec, nor); | 
					
						
							| 
									
										
										
										
											2008-07-24 19:12:30 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	cross_v3_v3v3(nor, vec, axis); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2014-03-01 14:20:54 +11:00
										 |  |  | 	if (fabsf(nor[0]) > fabsf(nor[1]) && fabsf(nor[0]) > fabsf(nor[2]) && nor[0] < 0) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		negate_v3(nor); | 
					
						
							| 
									
										
										
										
											2008-07-24 19:12:30 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-03-01 14:20:54 +11:00
										 |  |  | 	else if (fabsf(nor[1]) > fabsf(nor[0]) && fabsf(nor[1]) > fabsf(nor[2]) && nor[1] < 0) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		negate_v3(nor); | 
					
						
							| 
									
										
										
										
											2008-07-24 19:12:30 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-03-01 14:20:54 +11:00
										 |  |  | 	else if (fabsf(nor[2]) > fabsf(nor[1]) && fabsf(nor[2]) > fabsf(nor[0]) && nor[2] < 0) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		negate_v3(nor); | 
					
						
							| 
									
										
										
										
											2008-07-24 19:12:30 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	/* mirror node2 along axis */ | 
					
						
							| 
									
										
										
										
											2011-11-06 16:38:21 +00:00
										 |  |  | 	copy_v3_v3(p, node2->p); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	BLI_mirrorAlongAxis(p, root_node->p, nor); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* check if it's within limit before continuing */ | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 	if (len_squared_v3v3(node1->p, p) <= limit_sq) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		/* mark node as symmetric physically */ | 
					
						
							| 
									
										
										
										
											2011-11-06 16:38:21 +00:00
										 |  |  | 		copy_v3_v3(root_node->symmetry_axis, nor); | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		root_node->symmetry_flag |= SYM_PHYSICAL; | 
					
						
							|  |  |  | 		root_node->symmetry_flag |= SYM_AXIAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* flag side on arcs */ | 
					
						
							|  |  |  | 		flagAxialSymmetry(root_node, node1, arc1, group); | 
					
						
							|  |  |  | 		flagAxialSymmetry(root_node, node2, arc2, group); | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (graph->axial_symmetry) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 			graph->axial_symmetry(root_node, node1, node2, arc1, arc2); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2008-07-02 21:36:45 +00:00
										 |  |  | 		/* NOT SYMMETRIC */ | 
					
						
							| 
									
										
										
										
											2008-05-30 18:13:43 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void handleAxialSymmetry(BGraph *graph, BNode *root_node, int depth, float axis[3], float limit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BArc *arc1 = NULL, *arc2 = NULL; | 
					
						
							|  |  |  | 	BNode *node1 = NULL, *node2 = NULL; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* mark topological symmetry */ | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	root_node->symmetry_flag |= SYM_TOPOLOGICAL; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < root_node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		BArc *connectedArc = root_node->arcs[i]; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* depth is store as a negative in flag. symmetry level is positive */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level == -depth) { | 
					
						
							|  |  |  | 			if (arc1 == NULL) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				arc1 = connectedArc; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				node1 = BLI_otherNode(arc1, root_node); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				arc2 = connectedArc; | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 				node2 = BLI_otherNode(arc2, root_node); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				break; /* Can stop now, the two arcs have been found */ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* shouldn't happen, but just to be sure */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node1 == NULL || node2 == NULL) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 	testAxialSymmetry(graph, root_node, node1, node2, arc1, arc2, axis, limit, 1); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void markdownSecondarySymmetry(BGraph *graph, BNode *node, int depth, int level, float limit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float axis[3] = {0, 0, 0}; | 
					
						
							|  |  |  | 	int count = 0; | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	/* count the number of branches in this symmetry group
 | 
					
						
							| 
									
										
										
										
											2012-03-08 04:12:11 +00:00
										 |  |  | 	 * and determinate the axis of symmetry | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		BArc *connectedArc = node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* depth is store as a negative in flag. symmetry level is positive */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level == -depth) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			count++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		/* If arc is on the axis */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		else if (connectedArc->symmetry_level == level) { | 
					
						
							| 
									
										
										
										
											2010-04-21 12:27:48 +00:00
										 |  |  | 			add_v3_v3(axis, connectedArc->head->p); | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 			sub_v3_v3v3(axis, axis, connectedArc->tail->p); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	normalize_v3(axis); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Split between axial and radial symmetry */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (count == 2) { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		handleAxialSymmetry(graph, node, depth, axis, limit); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2008-05-30 17:42:02 +00:00
										 |  |  | 		handleRadialSymmetry(graph, node, depth, axis, limit); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-08-19 22:16:01 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	/* markdown secondary symetries */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		BArc *connectedArc = node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level == -depth) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			/* markdown symmetry for branches corresponding to the depth */ | 
					
						
							|  |  |  | 			markdownSymmetryArc(graph, connectedArc, node, level + 1, limit); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 16:52:06 +00:00
										 |  |  | static void markdownSymmetryArc(BGraph *graph, BArc *arc, BNode *node, int level, float limit) | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	/* if arc is null, we start straight from a node */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (arc) { | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 		arc->symmetry_level = level; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		node = BLI_otherNode(arc, node); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		BArc *connectedArc = node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc != arc) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			BNode *connectedNode = BLI_otherNode(connectedArc, node); | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* symmetry level is positive value, negative values is subtree depth */ | 
					
						
							| 
									
										
										
										
											2008-08-18 22:22:56 +00:00
										 |  |  | 			connectedArc->symmetry_level = -BLI_subtreeShape(graph, connectedNode, connectedArc, 0); | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	arc = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		int issymmetryAxis = 0; | 
					
						
							|  |  |  | 		BArc *connectedArc = node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* only arcs not already marked as symetric */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			int j; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* true by default */ | 
					
						
							|  |  |  | 			issymmetryAxis = 1; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			for (j = 0; j < node->degree; j++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				BArc *otherArc = node->arcs[j]; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				/* different arc, same depth */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				if (otherArc != connectedArc && otherArc->symmetry_level == connectedArc->symmetry_level) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 					/* not on the symmetry axis */ | 
					
						
							|  |  |  | 					issymmetryAxis = 0; | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* arc could be on the symmetry axis */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (issymmetryAxis == 1) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			/* no arc as been marked previously, keep this one */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			if (arc == NULL) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				arc = connectedArc; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else if (connectedArc->symmetry_level < arc->symmetry_level) { | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 				/* go with more complex subtree as symmetry arc */ | 
					
						
							|  |  |  | 				arc = connectedArc; | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* go down the arc continuing the symmetry axis */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (arc) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		markdownSymmetryArc(graph, arc, node, level, limit); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* secondary symmetry */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < node->degree; i++) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		BArc *connectedArc = node->arcs[i]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* only arcs not already marked as symetric and is not the next arc on the symmetry axis */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (connectedArc->symmetry_level < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 			/* subtree depth is store as a negative value in the symmetry */ | 
					
						
							|  |  |  | 			markdownSecondarySymmetry(graph, node, -connectedArc->symmetry_level, level, limit); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_markdownSymmetry(BGraph *graph, BNode *root_node, float limit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BNode *node; | 
					
						
							|  |  |  | 	BArc *arc; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (root_node == NULL) { | 
					
						
							| 
									
										
										
										
											2008-11-05 20:15:32 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (BLI_isGraphCyclic(graph)) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* mark down all arcs as non-symetric */ | 
					
						
							|  |  |  | 	BLI_flagArcs(graph, 0); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* mark down all nodes as not on the symmetry axis */ | 
					
						
							|  |  |  | 	BLI_flagNodes(graph, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	node = root_node; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 	/* sanity check REMOVE ME */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (node->degree > 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 		arc = node->arcs[0]; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		if (node->degree == 1) { | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 			markdownSymmetryArc(graph, arc, node, 1, limit); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2008-06-23 21:47:31 +00:00
										 |  |  | 			markdownSymmetryArc(graph, NULL, node, 1, limit); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* mark down non-symetric arcs */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (arc = graph->arcs.first; arc; arc = arc->next) { | 
					
						
							|  |  |  | 			if (arc->symmetry_level < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				arc->symmetry_level = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 				/* mark down nodes with the lowest level symmetry axis */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				if (arc->head->symmetry_level == 0 || arc->head->symmetry_level > arc->symmetry_level) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 					arc->head->symmetry_level = arc->symmetry_level; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				if (arc->tail->symmetry_level == 0 || arc->tail->symmetry_level > arc->symmetry_level) { | 
					
						
							| 
									
										
										
										
											2008-05-28 16:39:05 +00:00
										 |  |  | 					arc->tail->symmetry_level = arc->symmetry_level; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_head(void *arg) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->head(iter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_tail(void *arg) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->tail(iter);  | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_peek(void *arg, int n) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (iter->index + n < 0) { | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 		return iter->head(iter); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else if (iter->index + n >= iter->length) { | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 		return iter->tail(iter); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 		return iter->peek(iter, n); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_next(void *arg) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->next(iter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_nextN(void *arg, int n) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->nextN(iter, n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | void *IT_previous(void *arg) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->previous(iter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | int   IT_stopped(void *arg) | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:13:06 +00:00
										 |  |  | 	BArcIterator *iter = (BArcIterator *)arg; | 
					
						
							| 
									
										
										
										
											2008-11-29 20:37:10 +00:00
										 |  |  | 	return iter->stopped(iter); | 
					
						
							|  |  |  | } |