eg:
 bpy.context.object.driver_remove("location")
ANIM_remove_driver now accepts -1 as an index for removing all drivers from one path.
		
	
		
			
				
	
	
		
			558 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			558 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**
 | |
|  * $Id$
 | |
|  *
 | |
|  * ***** BEGIN GPL LICENSE BLOCK *****
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software Foundation,
 | |
|  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 | |
|  *
 | |
|  * The Original Code is Copyright (C) 2009 Blender Foundation, Joshua Leung
 | |
|  * All rights reserved.
 | |
|  *
 | |
|  * The Original Code is: all of this file.
 | |
|  *
 | |
|  * Contributor(s): Joshua Leung (full recode)
 | |
|  *
 | |
|  * ***** END GPL LICENSE BLOCK *****
 | |
|  */
 | |
|  
 | |
| #include <string.h>
 | |
| 
 | |
| #include "MEM_guardedalloc.h"
 | |
| 
 | |
| #include "BLI_blenlib.h"
 | |
| 
 | |
| #include "DNA_anim_types.h"
 | |
| 
 | |
| #include "BKE_animsys.h"
 | |
| #include "BKE_depsgraph.h"
 | |
| #include "BKE_fcurve.h"
 | |
| #include "BKE_context.h"
 | |
| 
 | |
| #include "UI_interface.h"
 | |
| 
 | |
| #include "WM_api.h"
 | |
| #include "WM_types.h"
 | |
| 
 | |
| #include "RNA_access.h"
 | |
| #include "RNA_define.h"
 | |
| 
 | |
| /* ************************************************** */
 | |
| /* Animation Data Validation */
 | |
| 
 | |
| /* Get (or add relevant data to be able to do so) F-Curve from the driver stack, 
 | |
|  * for the given Animation Data block. This assumes that all the destinations are valid.
 | |
|  *	
 | |
|  *	- add:	0 - don't add anything if not found, 
 | |
|  *			1 - add new Driver FCurve, 
 | |
|  *			-1 - add new Driver FCurve without driver stuff (for pasting)
 | |
|  */
 | |
| FCurve *verify_driver_fcurve (ID *id, const char rna_path[], const int array_index, short add)
 | |
| {
 | |
| 	AnimData *adt;
 | |
| 	FCurve *fcu;
 | |
| 	
 | |
| 	/* sanity checks */
 | |
| 	if ELEM(NULL, id, rna_path)
 | |
| 		return NULL;
 | |
| 	
 | |
| 	/* init animdata if none available yet */
 | |
| 	adt= BKE_animdata_from_id(id);
 | |
| 	if ((adt == NULL) && (add))
 | |
| 		adt= BKE_id_add_animdata(id);
 | |
| 	if (adt == NULL) { 
 | |
| 		/* if still none (as not allowed to add, or ID doesn't have animdata for some reason) */
 | |
| 		return NULL;
 | |
| 	}
 | |
| 		
 | |
| 	/* try to find f-curve matching for this setting 
 | |
| 	 *	- add if not found and allowed to add one
 | |
| 	 *		TODO: add auto-grouping support? how this works will need to be resolved
 | |
| 	 */
 | |
| 	fcu= list_find_fcurve(&adt->drivers, rna_path, array_index);
 | |
| 	
 | |
| 	if ((fcu == NULL) && (add)) {
 | |
| 		/* use default settings to make a F-Curve */
 | |
| 		fcu= MEM_callocN(sizeof(FCurve), "FCurve");
 | |
| 		
 | |
| 		fcu->flag = (FCURVE_VISIBLE|FCURVE_AUTO_HANDLES|FCURVE_SELECTED);
 | |
| 		
 | |
| 		/* store path - make copy, and store that */
 | |
| 		fcu->rna_path= BLI_strdupn(rna_path, strlen(rna_path));
 | |
| 		fcu->array_index= array_index;
 | |
| 		
 | |
| 		/* if add is negative, don't init this data yet, since it will be filled in by the pasted driver */
 | |
| 		if (add > 0) {
 | |
| 			/* add some new driver data */
 | |
| 			fcu->driver= MEM_callocN(sizeof(ChannelDriver), "ChannelDriver");
 | |
| 			
 | |
| 			/* add simple generator modifier for driver so that there is some visible representation */
 | |
| 			add_fmodifier(&fcu->modifiers, FMODIFIER_TYPE_GENERATOR);
 | |
| 		}
 | |
| 		
 | |
| 		/* just add F-Curve to end of driver list */
 | |
| 		BLI_addtail(&adt->drivers, fcu);
 | |
| 	}
 | |
| 	
 | |
| 	/* return the F-Curve */
 | |
| 	return fcu;
 | |
| }
 | |
| 
 | |
| /* ************************************************** */
 | |
| /* Driver Management API */
 | |
| 
 | |
| /* Main Driver Management API calls:
 | |
|  * 	Add a new driver for the specified property on the given ID block
 | |
|  */
 | |
| short ANIM_add_driver (ID *id, const char rna_path[], int array_index, short flag, int type)
 | |
| {	
 | |
| 	PointerRNA id_ptr, ptr;
 | |
| 	PropertyRNA *prop;
 | |
| 	FCurve *fcu;
 | |
| 	int array_index_max;
 | |
| 	int done = 0;
 | |
| 	
 | |
| 	/* validate pointer first - exit if failure */
 | |
| 	RNA_id_pointer_create(id, &id_ptr);
 | |
| 	if ((RNA_path_resolve(&id_ptr, rna_path, &ptr, &prop) == 0) || (prop == NULL)) {
 | |
| 		printf("Add Driver: Could not add Driver, as RNA Path is invalid for the given ID (ID = %s, Path = %s)\n", id->name, rna_path);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	
 | |
| 	/* key entire array convenience method */
 | |
| 	if (array_index == -1) {
 | |
| 		array_index_max= RNA_property_array_length(&ptr, prop);
 | |
| 		array_index= 0;
 | |
| 	}
 | |
| 	else
 | |
| 		array_index_max= array_index;
 | |
| 	
 | |
| 	/* maximum index should be greater than the start index */
 | |
| 	if (array_index == array_index_max)
 | |
| 		array_index_max += 1;
 | |
| 	
 | |
| 	/* will only loop once unless the array index was -1 */
 | |
| 	for (; array_index < array_index_max; array_index++) {
 | |
| 		/* create F-Curve with Driver */
 | |
| 		fcu= verify_driver_fcurve(id, rna_path, array_index, 1);
 | |
| 		
 | |
| 		if (fcu && fcu->driver) {
 | |
| 			ChannelDriver *driver= fcu->driver;
 | |
| 			
 | |
| 			/* set the type of the driver */
 | |
| 			driver->type= type;
 | |
| 			
 | |
| 			/* fill in current value for python */
 | |
| 			if (type == DRIVER_TYPE_PYTHON) {
 | |
| 				PropertyType proptype= RNA_property_type(prop);
 | |
| 				int array= RNA_property_array_length(&ptr, prop);
 | |
| 				char *expression= driver->expression;
 | |
| 				int val, maxlen= sizeof(driver->expression);
 | |
| 				float fval;
 | |
| 				
 | |
| 				if (proptype == PROP_BOOLEAN) {
 | |
| 					if (!array) val= RNA_property_boolean_get(&ptr, prop);
 | |
| 					else val= RNA_property_boolean_get_index(&ptr, prop, array_index);
 | |
| 					
 | |
| 					BLI_strncpy(expression, (val)? "True": "False", maxlen);
 | |
| 				}
 | |
| 				else if (proptype == PROP_INT) {
 | |
| 					if (!array) val= RNA_property_int_get(&ptr, prop);
 | |
| 					else val= RNA_property_int_get_index(&ptr, prop, array_index);
 | |
| 					
 | |
| 					BLI_snprintf(expression, maxlen, "%d", val);
 | |
| 				}
 | |
| 				else if (proptype == PROP_FLOAT) {
 | |
| 					if (!array) fval= RNA_property_float_get(&ptr, prop);
 | |
| 					else fval= RNA_property_float_get_index(&ptr, prop, array_index);
 | |
| 					
 | |
| 					BLI_snprintf(expression, maxlen, "%.3f", fval);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		/* set the done status */
 | |
| 		done += (fcu != NULL);
 | |
| 	}
 | |
| 	
 | |
| 	/* done */
 | |
| 	return done;
 | |
| }
 | |
| 
 | |
| /* Main Driver Management API calls:
 | |
|  * 	Remove the driver for the specified property on the given ID block (if available)
 | |
|  */
 | |
| short ANIM_remove_driver (struct ID *id, const char rna_path[], int array_index, short flag)
 | |
| {
 | |
| 	AnimData *adt;
 | |
| 	FCurve *fcu;
 | |
| 	int success= 0;
 | |
| 	
 | |
| 	/* get F-Curve
 | |
| 	 * Note: here is one of the places where we don't want new F-Curve + Driver added!
 | |
| 	 * 		so 'add' var must be 0
 | |
| 	 */
 | |
| 	/* we don't check the validity of the path here yet, but it should be ok... */
 | |
| 	adt= BKE_animdata_from_id(id);
 | |
| 	
 | |
| 	if(adt) {
 | |
| 		if(array_index == -1) {
 | |
| 			FCurve *fcu_iter= adt->drivers.first;
 | |
| 			while((fcu= iter_step_fcurve(fcu_iter, rna_path))) {
 | |
| 				/* store the next fcurve for looping  */
 | |
| 				fcu_iter= fcu->next;
 | |
| 				
 | |
| 				/* remove F-Curve from driver stack, then free it */
 | |
| 				BLI_remlink(&adt->drivers, fcu);
 | |
| 				free_fcurve(fcu);
 | |
| 				
 | |
| 				/* done successfully */
 | |
| 				success |= 1;
 | |
| 			}
 | |
| 		}
 | |
| 		else {
 | |
| 			fcu= verify_driver_fcurve(id, rna_path, array_index, 0);
 | |
| 			if(fcu) {
 | |
| 				BLI_remlink(&adt->drivers, fcu);
 | |
| 				free_fcurve(fcu);
 | |
| 				
 | |
| 				success = 1;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return success;
 | |
| }
 | |
| 
 | |
| /* ************************************************** */
 | |
| /* Driver Management API - Copy/Paste Drivers */
 | |
| 
 | |
| /* Copy/Paste Buffer for Driver Data... */
 | |
| static FCurve *channeldriver_copypaste_buf = NULL;
 | |
| 
 | |
| /* This function frees any MEM_calloc'ed copy/paste buffer data */
 | |
| // XXX find some header to put this in!
 | |
| void free_anim_drivers_copybuf (void)
 | |
| {
 | |
| 	/* free the buffer F-Curve if it exists, as if it were just another F-Curve */
 | |
| 	if (channeldriver_copypaste_buf)
 | |
| 		free_fcurve(channeldriver_copypaste_buf);
 | |
| 	channeldriver_copypaste_buf= NULL;
 | |
| }
 | |
| 
 | |
| /* Checks if there is a driver in the copy/paste buffer */
 | |
| short ANIM_driver_can_paste (void)
 | |
| {
 | |
| 	return (channeldriver_copypaste_buf != NULL);
 | |
| }
 | |
| 
 | |
| /* ------------------- */
 | |
| 
 | |
| /* Main Driver Management API calls:
 | |
|  * 	Make a copy of the driver for the specified property on the given ID block
 | |
|  */
 | |
| short ANIM_copy_driver (ID *id, const char rna_path[], int array_index, short flag)
 | |
| {
 | |
| 	PointerRNA id_ptr, ptr;
 | |
| 	PropertyRNA *prop;
 | |
| 	FCurve *fcu;
 | |
| 	
 | |
| 	/* validate pointer first - exit if failure */
 | |
| 	RNA_id_pointer_create(id, &id_ptr);
 | |
| 	if ((RNA_path_resolve(&id_ptr, rna_path, &ptr, &prop) == 0) || (prop == NULL)) {
 | |
| 		printf("Copy Driver: Could not find Driver, as RNA Path is invalid for the given ID (ID = %s, Path = %s)\n", id->name, rna_path);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	
 | |
| 	/* try to get F-Curve with Driver */
 | |
| 	fcu= verify_driver_fcurve(id, rna_path, array_index, 0);
 | |
| 	
 | |
| 	/* clear copy/paste buffer first (for consistency with other copy/paste buffers) */
 | |
| 	free_anim_drivers_copybuf();
 | |
| 	
 | |
| 	/* copy this to the copy/paste buf if it exists */
 | |
| 	if (fcu && fcu->driver) {
 | |
| 		/* make copies of some info such as the rna_path, then clear this info from the F-Curve temporarily
 | |
| 		 * so that we don't end up wasting memory storing the path which won't get used ever...
 | |
| 		 */
 | |
| 		char *tmp_path = fcu->rna_path;
 | |
| 		fcu->rna_path= NULL;
 | |
| 		
 | |
| 		/* make a copy of the F-Curve with */
 | |
| 		channeldriver_copypaste_buf= copy_fcurve(fcu);
 | |
| 		
 | |
| 		/* restore the path */
 | |
| 		fcu->rna_path= tmp_path;
 | |
| 		
 | |
| 		/* copied... */
 | |
| 		return 1;
 | |
| 	}
 | |
| 	
 | |
| 	/* done */
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /* Main Driver Management API calls:
 | |
|  * 	Add a new driver for the specified property on the given ID block or replace an existing one
 | |
|  *	with the driver + driver-curve data from the buffer 
 | |
|  */
 | |
| short ANIM_paste_driver (ID *id, const char rna_path[], int array_index, short flag)
 | |
| {	
 | |
| 	PointerRNA id_ptr, ptr;
 | |
| 	PropertyRNA *prop;
 | |
| 	FCurve *fcu;
 | |
| 	
 | |
| 	/* validate pointer first - exit if failure */
 | |
| 	RNA_id_pointer_create(id, &id_ptr);
 | |
| 	if ((RNA_path_resolve(&id_ptr, rna_path, &ptr, &prop) == 0) || (prop == NULL)) {
 | |
| 		printf("Paste Driver: Could not add Driver, as RNA Path is invalid for the given ID (ID = %s, Path = %s)\n", id->name, rna_path);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	
 | |
| 	/* if the buffer is empty, cannot paste... */
 | |
| 	if (channeldriver_copypaste_buf == NULL) {
 | |
| 		printf("Paste Driver: No Driver to paste. \n");
 | |
| 		return 0;
 | |
| 	}
 | |
| 	
 | |
| 	/* create Driver F-Curve, but without data which will be copied across... */
 | |
| 	fcu= verify_driver_fcurve(id, rna_path, array_index, -1);
 | |
| 
 | |
| 	if (fcu) {
 | |
| 		/* copy across the curve data from the buffer curve 
 | |
| 		 * NOTE: this step needs care to not miss new settings
 | |
| 		 */
 | |
| 			/* keyframes/samples */
 | |
| 		fcu->bezt= MEM_dupallocN(channeldriver_copypaste_buf->bezt);
 | |
| 		fcu->fpt= MEM_dupallocN(channeldriver_copypaste_buf->fpt);
 | |
| 		fcu->totvert= channeldriver_copypaste_buf->totvert;
 | |
| 		
 | |
| 			/* modifiers */
 | |
| 		copy_fmodifiers(&fcu->modifiers, &channeldriver_copypaste_buf->modifiers);
 | |
| 		
 | |
| 			/* flags - on a per-relevant-flag basis */
 | |
| 		if (channeldriver_copypaste_buf->flag & FCURVE_AUTO_HANDLES)
 | |
| 			fcu->flag |= FCURVE_AUTO_HANDLES;
 | |
| 		else
 | |
| 			fcu->flag &= ~FCURVE_AUTO_HANDLES;
 | |
| 			/* extrapolation mode */
 | |
| 		fcu->extend= channeldriver_copypaste_buf->extend;
 | |
| 			
 | |
| 			/* the 'juicy' stuff - the driver */
 | |
| 		fcu->driver= fcurve_copy_driver(channeldriver_copypaste_buf->driver);
 | |
| 	}
 | |
| 	
 | |
| 	/* done */
 | |
| 	return (fcu != NULL);
 | |
| }
 | |
| 
 | |
| /* ************************************************** */
 | |
| /* UI-Button Interface */
 | |
| 
 | |
| /* Add Driver Button Operator ------------------------ */
 | |
| 
 | |
| static int add_driver_button_exec (bContext *C, wmOperator *op)
 | |
| {
 | |
| 	PointerRNA ptr;
 | |
| 	PropertyRNA *prop= NULL;
 | |
| 	char *path;
 | |
| 	short success= 0;
 | |
| 	int index, all= RNA_boolean_get(op->ptr, "all");
 | |
| 	
 | |
| 	/* try to create driver using property retrieved from UI */
 | |
| 	memset(&ptr, 0, sizeof(PointerRNA));
 | |
| 	uiAnimContextProperty(C, &ptr, &prop, &index);
 | |
| 
 | |
| 	if (all)
 | |
| 		index= -1;
 | |
| 
 | |
| 	if (ptr.data && prop && RNA_property_animateable(&ptr, prop)) {
 | |
| 		path= RNA_path_from_ID_to_property(&ptr, prop);
 | |
| 		
 | |
| 		if (path) {			
 | |
| 			success+= ANIM_add_driver(ptr.id.data, path, index, 0, DRIVER_TYPE_PYTHON);
 | |
| 			
 | |
| 			MEM_freeN(path);
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	if (success) {
 | |
| 		/* send updates */
 | |
| 		DAG_ids_flush_update(0);
 | |
| 		
 | |
| 		WM_event_add_notifier(C, NC_ANIMATION|ND_FCURVES_ORDER, NULL); // XXX
 | |
| 	}
 | |
| 	
 | |
| 	return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
 | |
| }
 | |
| 
 | |
| void ANIM_OT_driver_button_add (wmOperatorType *ot)
 | |
| {
 | |
| 	/* identifiers */
 | |
| 	ot->name= "Add Driver";
 | |
| 	ot->idname= "ANIM_OT_driver_button_add";
 | |
| 	ot->description= "Add driver(s) for the property(s) connected represented by the highlighted button";
 | |
| 	
 | |
| 	/* callbacks */
 | |
| 	ot->exec= add_driver_button_exec; 
 | |
| 	//op->poll= ??? // TODO: need to have some animateable property to do this
 | |
| 	
 | |
| 	/* flags */
 | |
| 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
 | |
| 
 | |
| 	/* properties */
 | |
| 	RNA_def_boolean(ot->srna, "all", 1, "All", "Create drivers for all elements of the array.");
 | |
| }
 | |
| 
 | |
| /* Remove Driver Button Operator ------------------------ */
 | |
| 
 | |
| static int remove_driver_button_exec (bContext *C, wmOperator *op)
 | |
| {
 | |
| 	PointerRNA ptr;
 | |
| 	PropertyRNA *prop= NULL;
 | |
| 	char *path;
 | |
| 	short success= 0;
 | |
| 	int index, all= RNA_boolean_get(op->ptr, "all");
 | |
| 	
 | |
| 	/* try to find driver using property retrieved from UI */
 | |
| 	memset(&ptr, 0, sizeof(PointerRNA));
 | |
| 	uiAnimContextProperty(C, &ptr, &prop, &index);
 | |
| 	
 | |
| 	if (all)
 | |
| 		index= -1;
 | |
| 
 | |
| 	if (ptr.data && prop) {
 | |
| 		path= RNA_path_from_ID_to_property(&ptr, prop);
 | |
| 		success= ANIM_remove_driver(ptr.id.data, path, index, 0);
 | |
| 		MEM_freeN(path);
 | |
| 	}
 | |
| 	
 | |
| 	if (success) {
 | |
| 		/* send updates */
 | |
| 		DAG_ids_flush_update(0);
 | |
| 		
 | |
| 		WM_event_add_notifier(C, NC_ANIMATION|ND_FCURVES_ORDER, NULL);  // XXX
 | |
| 	}
 | |
| 	
 | |
| 	return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
 | |
| }
 | |
| 
 | |
| void ANIM_OT_driver_button_remove (wmOperatorType *ot)
 | |
| {
 | |
| 	/* identifiers */
 | |
| 	ot->name= "Remove Driver";
 | |
| 	ot->idname= "ANIM_OT_driver_button_remove";
 | |
| 	ot->description= "Remove the driver(s) for the property(s) connected represented by the highlighted button";
 | |
| 	
 | |
| 	/* callbacks */
 | |
| 	ot->exec= remove_driver_button_exec; 
 | |
| 	//op->poll= ??? // TODO: need to have some driver to be able to do this...
 | |
| 	
 | |
| 	/* flags */
 | |
| 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
 | |
| 
 | |
| 	/* properties */
 | |
| 	RNA_def_boolean(ot->srna, "all", 1, "All", "Delete drivers for all elements of the array.");
 | |
| }
 | |
| 
 | |
| /* Copy Driver Button Operator ------------------------ */
 | |
| 
 | |
| static int copy_driver_button_exec (bContext *C, wmOperator *op)
 | |
| {
 | |
| 	PointerRNA ptr;
 | |
| 	PropertyRNA *prop= NULL;
 | |
| 	char *path;
 | |
| 	short success= 0;
 | |
| 	int index;
 | |
| 	
 | |
| 	/* try to create driver using property retrieved from UI */
 | |
| 	memset(&ptr, 0, sizeof(PointerRNA));
 | |
| 	uiAnimContextProperty(C, &ptr, &prop, &index);
 | |
| 	
 | |
| 	if (ptr.data && prop && RNA_property_animateable(&ptr, prop)) {
 | |
| 		path= RNA_path_from_ID_to_property(&ptr, prop);
 | |
| 		
 | |
| 		if (path) {
 | |
| 			/* only copy the driver for the button that this was involved for */
 | |
| 			success= ANIM_copy_driver(ptr.id.data, path, index, 0);
 | |
| 			
 | |
| 			MEM_freeN(path);
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	/* since we're just copying, we don't really need to do anything else...*/
 | |
| 	return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
 | |
| }
 | |
| 
 | |
| void ANIM_OT_copy_driver_button (wmOperatorType *ot)
 | |
| {
 | |
| 	/* identifiers */
 | |
| 	ot->name= "Copy Driver";
 | |
| 	ot->idname= "ANIM_OT_copy_driver_button";
 | |
| 	ot->description= "Copy the driver for the highlighted button";
 | |
| 	
 | |
| 	/* callbacks */
 | |
| 	ot->exec= copy_driver_button_exec; 
 | |
| 	//op->poll= ??? // TODO: need to have some driver to be able to do this...
 | |
| 	
 | |
| 	/* flags */
 | |
| 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
 | |
| }
 | |
| 
 | |
| /* Paste Driver Button Operator ------------------------ */
 | |
| 
 | |
| static int paste_driver_button_exec (bContext *C, wmOperator *op)
 | |
| {
 | |
| 	PointerRNA ptr;
 | |
| 	PropertyRNA *prop= NULL;
 | |
| 	char *path;
 | |
| 	short success= 0;
 | |
| 	int index;
 | |
| 	
 | |
| 	/* try to create driver using property retrieved from UI */
 | |
| 	memset(&ptr, 0, sizeof(PointerRNA));
 | |
| 	uiAnimContextProperty(C, &ptr, &prop, &index);
 | |
| 	
 | |
| 	if (ptr.data && prop && RNA_property_animateable(&ptr, prop)) {
 | |
| 		path= RNA_path_from_ID_to_property(&ptr, prop);
 | |
| 		
 | |
| 		if (path) {
 | |
| 			/* only copy the driver for the button that this was involved for */
 | |
| 			success= ANIM_paste_driver(ptr.id.data, path, index, 0);
 | |
| 			
 | |
| 			MEM_freeN(path);
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	/* since we're just copying, we don't really need to do anything else...*/
 | |
| 	return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
 | |
| }
 | |
| 
 | |
| void ANIM_OT_paste_driver_button (wmOperatorType *ot)
 | |
| {
 | |
| 	/* identifiers */
 | |
| 	ot->name= "Paste Driver";
 | |
| 	ot->idname= "ANIM_OT_paste_driver_button";
 | |
| 	ot->description= "Paste the driver in the copy/paste buffer for the highlighted button";
 | |
| 	
 | |
| 	/* callbacks */
 | |
| 	ot->exec= paste_driver_button_exec; 
 | |
| 	//op->poll= ??? // TODO: need to have some driver to be able to do this...
 | |
| 	
 | |
| 	/* flags */
 | |
| 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
 | |
| }
 | |
| 
 | |
| /* ************************************************** */
 |