2009-07-02 05:25:14 +00:00
|
|
|
/**
|
|
|
|
* $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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, 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 *****
|
2009-04-10 13:08:12 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <float.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
|
|
|
#include "BLI_arithb.h"
|
|
|
|
#include "BLI_dynstr.h"
|
|
|
|
|
|
|
|
#include "DNA_anim_types.h"
|
|
|
|
#include "DNA_action_types.h"
|
|
|
|
#include "DNA_armature_types.h"
|
|
|
|
#include "DNA_constraint_types.h"
|
|
|
|
#include "DNA_key_types.h"
|
|
|
|
#include "DNA_object_types.h"
|
|
|
|
#include "DNA_material_types.h"
|
|
|
|
#include "DNA_scene_types.h"
|
|
|
|
#include "DNA_userdef_types.h"
|
|
|
|
#include "DNA_windowmanager_types.h"
|
|
|
|
|
|
|
|
#include "BKE_animsys.h"
|
|
|
|
#include "BKE_action.h"
|
|
|
|
#include "BKE_constraint.h"
|
|
|
|
#include "BKE_fcurve.h"
|
|
|
|
#include "BKE_utildefines.h"
|
|
|
|
#include "BKE_context.h"
|
|
|
|
#include "BKE_report.h"
|
|
|
|
#include "BKE_key.h"
|
|
|
|
#include "BKE_material.h"
|
|
|
|
|
|
|
|
#include "ED_anim_api.h"
|
|
|
|
#include "ED_keyframing.h"
|
|
|
|
#include "ED_keyframes_edit.h"
|
|
|
|
#include "ED_screen.h"
|
|
|
|
#include "ED_util.h"
|
|
|
|
|
|
|
|
#include "UI_interface.h"
|
|
|
|
|
|
|
|
#include "WM_api.h"
|
|
|
|
#include "WM_types.h"
|
|
|
|
|
|
|
|
#include "RNA_access.h"
|
|
|
|
#include "RNA_define.h"
|
|
|
|
#include "RNA_types.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.
|
2009-09-25 04:51:04 +00:00
|
|
|
*
|
|
|
|
* - add: 0 - don't add anything if not found,
|
|
|
|
* 1 - add new Driver FCurve,
|
|
|
|
* -1 - add new Driver FCurve without driver stuff (for pasting)
|
2009-04-10 13:08:12 +00:00
|
|
|
*/
|
|
|
|
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;
|
|
|
|
|
2009-09-25 04:51:04 +00:00
|
|
|
/* 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);
|
|
|
|
}
|
2009-04-10 13:16:50 +00:00
|
|
|
|
2009-04-10 13:08:12 +00:00
|
|
|
/* 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
|
|
|
|
*/
|
2009-07-12 02:06:15 +00:00
|
|
|
short ANIM_add_driver (ID *id, const char rna_path[], int array_index, short flag, int type)
|
2009-04-10 13:08:12 +00:00
|
|
|
{
|
|
|
|
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)) {
|
2009-09-25 04:51:04 +00:00
|
|
|
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);
|
2009-04-10 13:08:12 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* create F-Curve with Driver */
|
|
|
|
fcu= verify_driver_fcurve(id, rna_path, array_index, 1);
|
2009-07-12 02:06:15 +00:00
|
|
|
|
2009-09-04 11:19:12 +00:00
|
|
|
if (fcu && fcu->driver) {
|
2009-07-12 02:06:15 +00:00
|
|
|
fcu->driver->type= type;
|
2009-09-04 11:19:12 +00:00
|
|
|
|
2009-07-12 02:06:15 +00:00
|
|
|
/* fill in current value for python */
|
2009-09-04 11:19:12 +00:00
|
|
|
if (type == DRIVER_TYPE_PYTHON) {
|
2009-07-12 02:06:15 +00:00
|
|
|
PropertyType proptype= RNA_property_type(prop);
|
Implemented dynamic and multidimensional array support in RNA.
Example code: http://www.pasteall.org/7332/c.
New API functions: http://www.pasteall.org/7330/c.
Maximum number of dimensions is currently limited to 3, but can be increased arbitrarily if needed.
What this means for ID property access:
* MeshFace.verts - dynamic array, size 3 or 4 depending on MFace.v4
* MeshTextureFace.uv - dynamic, 2-dimensional array, size depends on MFace.v4
* Object.matrix - 2-dimensional array
What this means for functions:
* more intuitive API possibility, for example:
Mesh.add_vertices([(x, y, z), (x, y, z), ...])
Mesh.add_faces([(1, 2, 3), (4, 5, 6), ...])
Python part is not complete yet, e.g. it is possible to:
MeshFace.verts = (1, 2, 3) # even if Mesh.verts is (1, 2, 3, 4) and vice-versa
MeshTextureFace.uv = [(0.0, 0.0)] * 4 # only if a corresponding MFace is a quad
but the following won't work:
MeshTextureFace.uv[3] = (0.0, 0.0) # setting uv[3] modifies MTFace.uv[1][0] instead of MTFace.uv[3]
2009-08-25 17:06:36 +00:00
|
|
|
int array= RNA_property_array_length(&ptr, prop);
|
2009-07-12 02:06:15 +00:00
|
|
|
char *expression= fcu->driver->expression;
|
|
|
|
int val, maxlen= sizeof(fcu->driver->expression);
|
|
|
|
float fval;
|
2009-09-04 11:19:12 +00:00
|
|
|
|
|
|
|
if (proptype == PROP_BOOLEAN) {
|
2009-09-25 04:51:04 +00:00
|
|
|
if (!array) val= RNA_property_boolean_get(&ptr, prop);
|
2009-07-12 02:06:15 +00:00
|
|
|
else val= RNA_property_boolean_get_index(&ptr, prop, array_index);
|
2009-09-04 11:19:12 +00:00
|
|
|
|
2009-07-12 02:06:15 +00:00
|
|
|
BLI_strncpy(expression, (val)? "True": "False", maxlen);
|
|
|
|
}
|
2009-09-04 11:19:12 +00:00
|
|
|
else if (proptype == PROP_INT) {
|
|
|
|
if (!array) val= RNA_property_int_get(&ptr, prop);
|
2009-07-12 02:06:15 +00:00
|
|
|
else val= RNA_property_int_get_index(&ptr, prop, array_index);
|
2009-09-04 11:19:12 +00:00
|
|
|
|
2009-07-12 02:06:15 +00:00
|
|
|
BLI_snprintf(expression, maxlen, "%d", val);
|
|
|
|
}
|
2009-09-04 11:19:12 +00:00
|
|
|
else if (proptype == PROP_FLOAT) {
|
|
|
|
if (!array) fval= RNA_property_float_get(&ptr, prop);
|
2009-07-12 02:06:15 +00:00
|
|
|
else fval= RNA_property_float_get_index(&ptr, prop, array_index);
|
2009-09-04 11:19:12 +00:00
|
|
|
|
2009-07-12 02:06:15 +00:00
|
|
|
BLI_snprintf(expression, maxlen, "%.3f", fval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* done */
|
|
|
|
return (fcu != NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
|
|
|
|
/* 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... */
|
|
|
|
fcu= verify_driver_fcurve(id, rna_path, array_index, 0);
|
|
|
|
adt= BKE_animdata_from_id(id);
|
|
|
|
|
|
|
|
/* only continue if we have an driver to remove */
|
|
|
|
if (adt && fcu) {
|
|
|
|
/* remove F-Curve from driver stack, then free it */
|
|
|
|
BLI_remlink(&adt->drivers, fcu);
|
|
|
|
free_fcurve(fcu);
|
|
|
|
|
|
|
|
/* done successfully */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* failed */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-09-25 04:51:04 +00:00
|
|
|
/* ************************************************** */
|
|
|
|
/* 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);
|
|
|
|
}
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* ************************************************** */
|
|
|
|
/* 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 a, index, length, all= RNA_boolean_get(op->ptr, "all");
|
|
|
|
|
2009-09-04 11:19:12 +00:00
|
|
|
/* try to create driver using property retrieved from UI */
|
2009-04-10 13:08:12 +00:00
|
|
|
memset(&ptr, 0, sizeof(PointerRNA));
|
|
|
|
uiAnimContextProperty(C, &ptr, &prop, &index);
|
|
|
|
|
|
|
|
if (ptr.data && prop && RNA_property_animateable(ptr.data, prop)) {
|
|
|
|
path= RNA_path_from_ID_to_property(&ptr, prop);
|
|
|
|
|
|
|
|
if (path) {
|
|
|
|
if (all) {
|
Implemented dynamic and multidimensional array support in RNA.
Example code: http://www.pasteall.org/7332/c.
New API functions: http://www.pasteall.org/7330/c.
Maximum number of dimensions is currently limited to 3, but can be increased arbitrarily if needed.
What this means for ID property access:
* MeshFace.verts - dynamic array, size 3 or 4 depending on MFace.v4
* MeshTextureFace.uv - dynamic, 2-dimensional array, size depends on MFace.v4
* Object.matrix - 2-dimensional array
What this means for functions:
* more intuitive API possibility, for example:
Mesh.add_vertices([(x, y, z), (x, y, z), ...])
Mesh.add_faces([(1, 2, 3), (4, 5, 6), ...])
Python part is not complete yet, e.g. it is possible to:
MeshFace.verts = (1, 2, 3) # even if Mesh.verts is (1, 2, 3, 4) and vice-versa
MeshTextureFace.uv = [(0.0, 0.0)] * 4 # only if a corresponding MFace is a quad
but the following won't work:
MeshTextureFace.uv[3] = (0.0, 0.0) # setting uv[3] modifies MTFace.uv[1][0] instead of MTFace.uv[3]
2009-08-25 17:06:36 +00:00
|
|
|
length= RNA_property_array_length(&ptr, prop);
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
if (length) index= 0;
|
|
|
|
else length= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
length= 1;
|
|
|
|
|
|
|
|
for (a=0; a<length; a++)
|
2009-07-12 02:06:15 +00:00
|
|
|
success+= ANIM_add_driver(ptr.id.data, path, index+a, 0, DRIVER_TYPE_PYTHON);
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
MEM_freeN(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (success) {
|
|
|
|
/* send updates */
|
|
|
|
ED_anim_dag_flush_update(C);
|
|
|
|
|
|
|
|
/* for now, only send ND_KEYS for KeyingSets */
|
|
|
|
WM_event_add_notifier(C, ND_KEYS, NULL); // XXX
|
|
|
|
}
|
|
|
|
|
|
|
|
return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ANIM_OT_add_driver_button (wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Add Driver";
|
|
|
|
ot->idname= "ANIM_OT_add_driver_button";
|
2009-09-25 04:51:04 +00:00
|
|
|
ot->description= "Add driver(s) for the property(s) connected represented by the highlighted button.";
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* callbacks */
|
|
|
|
ot->exec= add_driver_button_exec;
|
2009-09-25 04:51:04 +00:00
|
|
|
//op->poll= ??? // TODO: need to have some animateable property to do this
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
|
|
|
|
/* properties */
|
2009-09-04 07:26:32 +00:00
|
|
|
RNA_def_boolean(ot->srna, "all", 1, "All", "Create drivers for all elements of the array.");
|
2009-04-10 13:08:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 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 a, index, length, all= RNA_boolean_get(op->ptr, "all");
|
|
|
|
|
2009-09-04 11:19:12 +00:00
|
|
|
/* try to find driver using property retrieved from UI */
|
2009-04-10 13:08:12 +00:00
|
|
|
memset(&ptr, 0, sizeof(PointerRNA));
|
|
|
|
uiAnimContextProperty(C, &ptr, &prop, &index);
|
|
|
|
|
|
|
|
if (ptr.data && prop) {
|
|
|
|
path= RNA_path_from_ID_to_property(&ptr, prop);
|
|
|
|
|
|
|
|
if (path) {
|
|
|
|
if (all) {
|
Implemented dynamic and multidimensional array support in RNA.
Example code: http://www.pasteall.org/7332/c.
New API functions: http://www.pasteall.org/7330/c.
Maximum number of dimensions is currently limited to 3, but can be increased arbitrarily if needed.
What this means for ID property access:
* MeshFace.verts - dynamic array, size 3 or 4 depending on MFace.v4
* MeshTextureFace.uv - dynamic, 2-dimensional array, size depends on MFace.v4
* Object.matrix - 2-dimensional array
What this means for functions:
* more intuitive API possibility, for example:
Mesh.add_vertices([(x, y, z), (x, y, z), ...])
Mesh.add_faces([(1, 2, 3), (4, 5, 6), ...])
Python part is not complete yet, e.g. it is possible to:
MeshFace.verts = (1, 2, 3) # even if Mesh.verts is (1, 2, 3, 4) and vice-versa
MeshTextureFace.uv = [(0.0, 0.0)] * 4 # only if a corresponding MFace is a quad
but the following won't work:
MeshTextureFace.uv[3] = (0.0, 0.0) # setting uv[3] modifies MTFace.uv[1][0] instead of MTFace.uv[3]
2009-08-25 17:06:36 +00:00
|
|
|
length= RNA_property_array_length(&ptr, prop);
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
if(length) index= 0;
|
|
|
|
else length= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
length= 1;
|
|
|
|
|
|
|
|
for (a=0; a<length; a++)
|
|
|
|
success+= ANIM_remove_driver(ptr.id.data, path, index+a, 0);
|
|
|
|
|
|
|
|
MEM_freeN(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (success) {
|
|
|
|
/* send updates */
|
|
|
|
ED_anim_dag_flush_update(C);
|
|
|
|
|
|
|
|
/* for now, only send ND_KEYS for KeyingSets */
|
|
|
|
WM_event_add_notifier(C, ND_KEYS, NULL); // XXX
|
|
|
|
}
|
|
|
|
|
|
|
|
return (success)? OPERATOR_FINISHED: OPERATOR_CANCELLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ANIM_OT_remove_driver_button (wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Remove Driver";
|
|
|
|
ot->idname= "ANIM_OT_remove_driver_button";
|
2009-09-25 04:51:04 +00:00
|
|
|
ot->description= "Remove the driver(s) for the property(s) connected represented by the highlighted button.";
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* callbacks */
|
|
|
|
ot->exec= remove_driver_button_exec;
|
2009-09-25 04:51:04 +00:00
|
|
|
//op->poll= ??? // TODO: need to have some driver to be able to do this...
|
2009-04-10 13:08:12 +00:00
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
|
|
|
|
/* properties */
|
2009-09-04 07:26:32 +00:00
|
|
|
RNA_def_boolean(ot->srna, "all", 1, "All", "Delete drivers for all elements of the array.");
|
2009-04-10 13:08:12 +00:00
|
|
|
}
|
|
|
|
|
2009-09-25 04:51:04 +00:00
|
|
|
/* 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.data, 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.data, 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;
|
|
|
|
}
|
|
|
|
|
2009-10-07 13:22:56 +00:00
|
|
|
|
2009-10-08 11:29:27 +00:00
|
|
|
/* Copy to Clipboard Button Operator ------------------------ */
|
2009-10-07 13:22:56 +00:00
|
|
|
|
|
|
|
static int copy_clipboard_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);
|
|
|
|
|
2009-10-08 11:29:27 +00:00
|
|
|
if (ptr.data && prop) {
|
2009-10-07 13:22:56 +00:00
|
|
|
path= RNA_path_from_ID_to_property(&ptr, prop);
|
2009-10-08 11:29:27 +00:00
|
|
|
|
2009-10-07 13:22:56 +00:00
|
|
|
if (path) {
|
|
|
|
WM_clipboard_text_set(path, FALSE);
|
|
|
|
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_clipboard_button(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Copy Data Path";
|
|
|
|
ot->idname= "ANIM_OT_copy_clipboard_button";
|
2009-10-08 11:29:27 +00:00
|
|
|
ot->description= "Copy the RNA data path for this property to the clipboard.";
|
2009-10-07 13:22:56 +00:00
|
|
|
|
|
|
|
/* callbacks */
|
|
|
|
ot->exec= copy_clipboard_button_exec;
|
2009-10-08 11:29:27 +00:00
|
|
|
//op->poll= ??? // TODO: need to have some valid property before this can be done
|
2009-10-07 13:22:56 +00:00
|
|
|
|
|
|
|
/* flags */
|
2009-10-08 11:29:27 +00:00
|
|
|
ot->flag= OPTYPE_REGISTER;
|
2009-10-07 13:22:56 +00:00
|
|
|
}
|
|
|
|
|
2009-04-10 13:08:12 +00:00
|
|
|
/* ************************************************** */
|