| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +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. | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |  * Contributor(s): none yet. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:29:51 +00:00
										 |  |  | /** \file blender/editors/metaball/mball_edit.c
 | 
					
						
							|  |  |  |  *  \ingroup edmeta | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | #include <math.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_blenlib.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | #include "BLI_rand.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-04 06:05:48 +00:00
										 |  |  | #include "DNA_defs.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | #include "DNA_meta_types.h"
 | 
					
						
							|  |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_scene_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "RNA_define.h"
 | 
					
						
							|  |  |  | #include "RNA_access.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-01 08:51:12 +00:00
										 |  |  | #include "RNA_enum_types.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_depsgraph.h"
 | 
					
						
							|  |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2010-09-07 05:47:34 +00:00
										 |  |  | #include "BKE_mball.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-14 17:55:27 +00:00
										 |  |  | #include "ED_mball.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | #include "ED_screen.h"
 | 
					
						
							|  |  |  | #include "ED_view3d.h"
 | 
					
						
							|  |  |  | #include "ED_transform.h"
 | 
					
						
							|  |  |  | #include "ED_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "WM_api.h"
 | 
					
						
							|  |  |  | #include "WM_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-14 17:55:27 +00:00
										 |  |  | #include "mball_intern.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | /* This function is used to free all MetaElems from MetaBall */ | 
					
						
							|  |  |  | void free_editMball(Object *obedit) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-05 12:52:47 +00:00
										 |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mb->editelems= NULL; | 
					
						
							|  |  |  | 	mb->lastelem= NULL; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This function is called, when MetaBall Object is
 | 
					
						
							|  |  |  |  * switched from object mode to edit mode */ | 
					
						
							|  |  |  | void make_editMball(Object *obedit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml;/*, *newml;*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ml= mb->elems.first; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	while(ml) { | 
					
						
							|  |  |  | 		if(ml->flag & SELECT) mb->lastelem = ml; | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mb->editelems = &mb->elems; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This function is called, when MetaBall Object switched from
 | 
					
						
							|  |  |  |  * edit mode to object mode. List od MetaElements is copied | 
					
						
							| 
									
										
										
										
											2010-04-25 15:24:18 +00:00
										 |  |  |  * from object->data->edit_elems to object->data->elems. */ | 
					
						
							| 
									
										
										
										
											2010-10-16 14:32:17 +00:00
										 |  |  | void load_editMball(Object *UNUSED(obedit)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add metaelem primitive to metaball object (which is in edit mode) */ | 
					
						
							| 
									
										
										
										
											2010-10-16 14:32:17 +00:00
										 |  |  | MetaElem *add_metaball_primitive(bContext *C, float mat[4][4], int type, int UNUSED(newname)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mball = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Deselect all existing metaelems */ | 
					
						
							|  |  |  | 	ml= mball->editelems->first; | 
					
						
							|  |  |  | 	while(ml) { | 
					
						
							|  |  |  | 		ml->flag &= ~SELECT; | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-09-07 05:47:34 +00:00
										 |  |  | 	ml= add_metaball_element(mball, type); | 
					
						
							|  |  |  | 	copy_v3_v3(&ml->x, mat[3]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ml->flag |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	mball->lastelem= ml; | 
					
						
							|  |  |  | 	return ml; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Select/Deselect operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Select or deselect all MetaElements */ | 
					
						
							| 
									
										
										
										
											2011-10-28 03:02:09 +00:00
										 |  |  | static int mball_select_all_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	//Scene *scene= CTX_data_scene(C);
 | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml; | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | 	int action = RNA_enum_get(op->ptr, "action"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 	if(ml) { | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | 		if (action == SEL_TOGGLE) { | 
					
						
							|  |  |  | 			action = SEL_SELECT; | 
					
						
							|  |  |  | 			while(ml) { | 
					
						
							|  |  |  | 				if(ml->flag & SELECT) { | 
					
						
							|  |  |  | 					action = SEL_DESELECT; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				ml= ml->next; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ml= mb->editelems->first; | 
					
						
							|  |  |  | 		while(ml) { | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | 			switch (action) { | 
					
						
							|  |  |  | 			case SEL_SELECT: | 
					
						
							|  |  |  | 				ml->flag |= SELECT; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case SEL_DESELECT: | 
					
						
							|  |  |  | 				ml->flag &= ~SELECT; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case SEL_INVERT: | 
					
						
							|  |  |  | 				ml->flag ^= SELECT; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 			ml= ml->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_SELECT, mb); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | void MBALL_OT_select_all(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2010-07-21 22:57:48 +00:00
										 |  |  | 	ot->name= "Select or Deselect All"; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Change selection of all meta elements"; | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_select_all"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							| 
									
										
										
										
											2011-10-28 03:02:09 +00:00
										 |  |  | 	ot->exec= mball_select_all_exec; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							| 
									
										
										
										
											2009-11-29 22:16:29 +00:00
										 |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	WM_operator_properties_select_all(ot); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Select random operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Random metaball selection */ | 
					
						
							|  |  |  | static int select_random_metaelems_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml; | 
					
						
							|  |  |  | 	float percent= RNA_float_get(op->ptr, "percent"); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-03-27 14:59:55 +00:00
										 |  |  | 	if(percent == 0.0f) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 	BLI_srand( BLI_rand() );	/* Random seed */ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* Stupid version of random selection. Should be improved. */ | 
					
						
							|  |  |  | 	while(ml) { | 
					
						
							|  |  |  | 		if(BLI_frand() < percent) | 
					
						
							|  |  |  | 			ml->flag |= SELECT; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			ml->flag &= ~SELECT; | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_SELECT, mb); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MBALL_OT_select_random_metaelems(struct wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Random..."; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Randomly select metaelements"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_select_random_metaelems"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							|  |  |  | 	ot->exec= select_random_metaelems_exec; | 
					
						
							|  |  |  | 	ot->invoke= WM_operator_props_popup; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* properties */ | 
					
						
							| 
									
										
										
										
											2011-09-19 12:26:20 +00:00
										 |  |  | 	RNA_def_float_percentage(ot->srna, "percent", 0.5f, 0.0f, 1.0f, "Percent", "Percentage of metaelems to select randomly", 0.0001f, 1.0f); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Duplicate operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Duplicate selected MetaElements */ | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int duplicate_metaelems_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml, *newml; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	ml= mb->editelems->last; | 
					
						
							|  |  |  | 	if(ml) { | 
					
						
							|  |  |  | 		while(ml) { | 
					
						
							|  |  |  | 			if(ml->flag & SELECT) { | 
					
						
							|  |  |  | 				newml= MEM_dupallocN(ml); | 
					
						
							|  |  |  | 				BLI_addtail(mb->editelems, newml); | 
					
						
							|  |  |  | 				mb->lastelem= newml; | 
					
						
							|  |  |  | 				ml->flag &= ~SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ml= ml->prev; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_DATA, mb); | 
					
						
							| 
									
										
										
										
											2011-01-03 12:41:16 +00:00
										 |  |  | 		DAG_id_tag_update(obedit->data, 0); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int duplicate_metaelems_invoke(bContext *C, wmOperator *op, wmEvent *UNUSED(event)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int retv= duplicate_metaelems_exec(C, op); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (retv == OPERATOR_FINISHED) { | 
					
						
							| 
									
										
										
										
											2011-03-22 12:53:36 +00:00
										 |  |  | 		RNA_enum_set(op->ptr, "mode", TFM_TRANSLATION); | 
					
						
							| 
									
										
										
										
											2009-12-10 10:36:32 +00:00
										 |  |  | 		WM_operator_name_call(C, "TRANSFORM_OT_transform", WM_OP_INVOKE_REGION_WIN, op->ptr); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return retv; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MBALL_OT_duplicate_metaelems(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2011-04-19 11:17:29 +00:00
										 |  |  | 	ot->name= "Duplicate Metaelements"; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Delete selected metaelement(s)"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_duplicate_metaelems"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							|  |  |  | 	ot->exec= duplicate_metaelems_exec; | 
					
						
							|  |  |  | 	ot->invoke= duplicate_metaelems_invoke; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* to give to transform */ | 
					
						
							| 
									
										
										
										
											2011-04-01 08:51:12 +00:00
										 |  |  | 	RNA_def_enum(ot->srna, "mode", transform_mode_types, TFM_TRANSLATION, "Mode", ""); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Delete operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Delete all selected MetaElems (not MetaBall) */ | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int delete_metaelems_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb= (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml, *next; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 	if(ml) { | 
					
						
							|  |  |  | 		while(ml) { | 
					
						
							|  |  |  | 			next= ml->next; | 
					
						
							|  |  |  | 			if(ml->flag & SELECT) { | 
					
						
							|  |  |  | 				if(mb->lastelem==ml) mb->lastelem= NULL; | 
					
						
							|  |  |  | 				BLI_remlink(mb->editelems, ml); | 
					
						
							|  |  |  | 				MEM_freeN(ml); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ml= next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_DATA, mb); | 
					
						
							| 
									
										
										
										
											2011-01-03 12:41:16 +00:00
										 |  |  | 		DAG_id_tag_update(obedit->data, 0); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MBALL_OT_delete_metaelems(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Delete"; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Delete selected metaelement(s)"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_delete_metaelems"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							|  |  |  | 	ot->exec= delete_metaelems_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Hide operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Hide selected MetaElems */ | 
					
						
							|  |  |  | static int hide_metaelems_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb= (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml; | 
					
						
							| 
									
										
										
										
											2011-02-24 16:04:36 +00:00
										 |  |  | 	const int invert= RNA_boolean_get(op->ptr, "unselected") ? SELECT : 0; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ml) { | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 		while(ml) { | 
					
						
							| 
									
										
										
										
											2011-02-24 16:04:36 +00:00
										 |  |  | 			if((ml->flag & SELECT) != invert) | 
					
						
							|  |  |  | 				ml->flag |= MB_HIDE; | 
					
						
							|  |  |  | 			ml= ml->next; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_DATA, mb); | 
					
						
							| 
									
										
										
										
											2011-01-03 12:41:16 +00:00
										 |  |  | 		DAG_id_tag_update(obedit->data, 0); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MBALL_OT_hide_metaelems(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Hide"; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Hide (un)selected metaelement(s)"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_hide_metaelems"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							|  |  |  | 	ot->exec= hide_metaelems_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* props */ | 
					
						
							| 
									
										
										
										
											2011-09-19 12:26:20 +00:00
										 |  |  | 	RNA_def_boolean(ot->srna, "unselected", 0, "Unselected", "Hide unselected rather than selected"); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************** Unhide operator *****************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Unhide all edited MetaElems */ | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | static int reveal_metaelems_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	MetaBall *mb= (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ml) { | 
					
						
							|  |  |  | 		while(ml) { | 
					
						
							|  |  |  | 			ml->flag &= ~MB_HIDE; | 
					
						
							|  |  |  | 			ml= ml->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_DATA, mb); | 
					
						
							| 
									
										
										
										
											2011-01-03 12:41:16 +00:00
										 |  |  | 		DAG_id_tag_update(obedit->data, 0); | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MBALL_OT_reveal_metaelems(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Reveal"; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 	ot->description= "Reveal all hidden metaelements"; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 	ot->idname= "MBALL_OT_reveal_metaelems"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* callback functions */ | 
					
						
							|  |  |  | 	ot->exec= reveal_metaelems_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmball; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Select MetaElement with mouse click (user can select radius circle or
 | 
					
						
							|  |  |  |  * stiffness circle) */ | 
					
						
							| 
									
										
										
										
											2011-05-12 16:47:36 +00:00
										 |  |  | int mouse_mball(bContext *C, const int mval[2], int extend) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	static MetaElem *startelem=NULL; | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	ViewContext vc; | 
					
						
							|  |  |  | 	MetaBall *mb = (MetaBall*)obedit->data; | 
					
						
							|  |  |  | 	MetaElem *ml, *act=NULL; | 
					
						
							|  |  |  | 	int a, hits; | 
					
						
							|  |  |  | 	unsigned int buffer[4*MAXPICKBUF]; | 
					
						
							|  |  |  | 	rcti rect; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	view3d_set_viewcontext(C, &vc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rect.xmin= mval[0]-12; | 
					
						
							|  |  |  | 	rect.xmax= mval[0]+12; | 
					
						
							|  |  |  | 	rect.ymin= mval[1]-12; | 
					
						
							|  |  |  | 	rect.ymax= mval[1]+12; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	hits= view3d_opengl_select(&vc, buffer, MAXPICKBUF, &rect); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* does startelem exist? */ | 
					
						
							|  |  |  | 	ml= mb->editelems->first; | 
					
						
							|  |  |  | 	while(ml) { | 
					
						
							|  |  |  | 		if(ml==startelem) break; | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ml==NULL) startelem= mb->editelems->first; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(hits>0) { | 
					
						
							|  |  |  | 		ml= startelem; | 
					
						
							|  |  |  | 		while(ml) { | 
					
						
							|  |  |  | 			for(a=0; a<hits; a++) { | 
					
						
							|  |  |  | 				/* index converted for gl stuff */ | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 				if(ml->selcol1==buffer[ 4 * a + 3 ]) { | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 					ml->flag |= MB_SCALE_RAD; | 
					
						
							|  |  |  | 					act= ml; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 				if(ml->selcol2==buffer[ 4 * a + 3 ]) { | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 					ml->flag &= ~MB_SCALE_RAD; | 
					
						
							|  |  |  | 					act= ml; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(act) break; | 
					
						
							|  |  |  | 			ml= ml->next; | 
					
						
							|  |  |  | 			if(ml==NULL) ml= mb->editelems->first; | 
					
						
							|  |  |  | 			if(ml==startelem) break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2010-07-20 10:41:08 +00:00
										 |  |  | 		/* When some metaelem was found, then it is necessary to select or
 | 
					
						
							| 
									
										
										
										
											2012-03-04 04:35:12 +00:00
										 |  |  | 		 * deselect it. */ | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		if(act) { | 
					
						
							|  |  |  | 			if(extend==0) { | 
					
						
							|  |  |  | 				/* Deselect all existing metaelems */ | 
					
						
							|  |  |  | 				ml= mb->editelems->first; | 
					
						
							|  |  |  | 				while(ml) { | 
					
						
							|  |  |  | 					ml->flag &= ~SELECT; | 
					
						
							|  |  |  | 					ml= ml->next; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				/* Select only metaelem clicked on */ | 
					
						
							|  |  |  | 				act->flag |= SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				if(act->flag & SELECT) | 
					
						
							|  |  |  | 					act->flag &= ~SELECT; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					act->flag |= SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			mb->lastelem= act; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | 			WM_event_add_notifier(C, NC_GEOM|ND_SELECT, mb); | 
					
						
							| 
									
										
										
										
											2009-11-24 04:59:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			return 1; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-24 04:59:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*  ************* undo for MetaBalls ************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* free all MetaElems from ListBase */ | 
					
						
							|  |  |  | static void freeMetaElemlist(ListBase *lb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MetaElem *ml, *next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(lb==NULL) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ml= lb->first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 	while(ml) { | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		next= ml->next; | 
					
						
							|  |  |  | 		BLI_remlink(lb, ml); | 
					
						
							|  |  |  | 		MEM_freeN(ml); | 
					
						
							|  |  |  | 		ml= next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lb->first= lb->last= NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 14:32:55 +00:00
										 |  |  | static void undoMball_to_editMball(void *lbu, void *lbe, void *UNUSED(obe)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	ListBase *lb= lbu; | 
					
						
							|  |  |  | 	ListBase *editelems= lbe; | 
					
						
							|  |  |  | 	MetaElem *ml, *newml; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	freeMetaElemlist(editelems); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* copy 'undo' MetaElems to 'edit' MetaElems */ | 
					
						
							|  |  |  | 	ml= lb->first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 	while(ml) { | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		newml= MEM_dupallocN(ml); | 
					
						
							|  |  |  | 		BLI_addtail(editelems, newml); | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 14:32:55 +00:00
										 |  |  | static void *editMball_to_undoMball(void *lbe, void *UNUSED(obe)) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	ListBase *editelems= lbe; | 
					
						
							|  |  |  | 	ListBase *lb; | 
					
						
							|  |  |  | 	MetaElem *ml, *newml; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* allocate memory for undo ListBase */ | 
					
						
							|  |  |  | 	lb= MEM_callocN(sizeof(ListBase), "listbase undo"); | 
					
						
							|  |  |  | 	lb->first= lb->last= NULL; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* copy contents of current ListBase to the undo ListBase */ | 
					
						
							|  |  |  | 	ml= editelems->first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:23:42 +00:00
										 |  |  | 	while(ml) { | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | 		newml= MEM_dupallocN(ml); | 
					
						
							|  |  |  | 		BLI_addtail(lb, newml); | 
					
						
							|  |  |  | 		ml= ml->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return lb; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* free undo ListBase of MetaElems */ | 
					
						
							|  |  |  | static void free_undoMball(void *lbv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ListBase *lb= lbv; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	freeMetaElemlist(lb); | 
					
						
							|  |  |  | 	MEM_freeN(lb); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-14 17:55:27 +00:00
										 |  |  | static ListBase *metaball_get_editelems(Object *ob) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if(ob && ob->type==OB_MBALL) { | 
					
						
							|  |  |  | 		struct MetaBall *mb= (struct MetaBall*)ob->data; | 
					
						
							|  |  |  | 		return mb->editelems; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void *get_data(bContext *C) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	return metaball_get_editelems(obedit); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* this is undo system for MetaBalls */ | 
					
						
							| 
									
										
										
										
											2010-11-17 09:45:45 +00:00
										 |  |  | void undo_push_mball(bContext *C, const char *name) | 
					
						
							| 
									
										
										
										
											2009-07-29 12:35:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	undo_editmode_push(C, name, get_data, free_undoMball, undoMball_to_editMball, editMball_to_undoMball, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |