2011-02-23 10:52:22 +00:00
|
|
|
/*
|
2009-02-14 13:07: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-02-14 13:07: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/space_view3d/view3d_snap.c
|
|
|
|
* \ingroup spview3d
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
#include <math.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "DNA_armature_types.h"
|
|
|
|
#include "DNA_curve_types.h"
|
|
|
|
#include "DNA_lattice_types.h"
|
|
|
|
#include "DNA_meta_types.h"
|
|
|
|
#include "DNA_scene_types.h"
|
2010-08-04 04:01:27 +00:00
|
|
|
#include "DNA_object_types.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
2009-11-10 20:43:45 +00:00
|
|
|
#include "BLI_math.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
#include "BLI_editVert.h"
|
|
|
|
#include "BLI_linklist.h"
|
2011-01-07 18:36:47 +00:00
|
|
|
#include "BLI_utildefines.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
#include "BKE_armature.h"
|
2010-08-01 12:47:49 +00:00
|
|
|
#include "BKE_context.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
#include "BKE_curve.h"
|
|
|
|
#include "BKE_depsgraph.h"
|
|
|
|
#include "BKE_lattice.h"
|
2010-08-01 12:47:49 +00:00
|
|
|
#include "BKE_main.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
#include "BKE_object.h"
|
2011-11-07 12:55:18 +00:00
|
|
|
#include "BKE_tracking.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
#include "WM_api.h"
|
|
|
|
#include "WM_types.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "ED_armature.h"
|
|
|
|
#include "ED_mesh.h"
|
2011-10-10 12:56:21 +00:00
|
|
|
#include "ED_keyframing.h"
|
2009-02-14 13:07:09 +00:00
|
|
|
#include "ED_screen.h"
|
2011-09-14 00:37:27 +00:00
|
|
|
#include "ED_curve.h" /* for curve_editnurbs */
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
#include "view3d_intern.h"
|
|
|
|
|
2011-05-10 14:48:06 +00:00
|
|
|
extern float originmat[3][3]; /* XXX object.c */
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
/* ************************************************** */
|
|
|
|
/* ********************* old transform stuff ******** */
|
|
|
|
/* *********** will get replaced with new transform * */
|
|
|
|
/* ************************************************** */
|
|
|
|
|
|
|
|
typedef struct TransVert {
|
|
|
|
float *loc;
|
|
|
|
float oldloc[3], fac;
|
|
|
|
float *val, oldval;
|
|
|
|
int flag;
|
|
|
|
float *nor;
|
|
|
|
} TransVert;
|
|
|
|
|
|
|
|
static TransVert *transvmain=NULL;
|
|
|
|
static int tottrans= 0;
|
|
|
|
|
|
|
|
/* copied from editobject.c, now uses (almost) proper depgraph */
|
2010-10-15 12:29:02 +00:00
|
|
|
static void special_transvert_update(Object *obedit)
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
if(obedit) {
|
|
|
|
|
2011-01-03 12:41:16 +00:00
|
|
|
DAG_id_tag_update(obedit->data, 0);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
if(obedit->type==OB_MESH) {
|
|
|
|
Mesh *me= obedit->data;
|
|
|
|
recalc_editnormals(me->edit_mesh); // does face centers too
|
|
|
|
}
|
|
|
|
else if (ELEM(obedit->type, OB_CURVE, OB_SURF)) {
|
|
|
|
Curve *cu= obedit->data;
|
2011-09-14 00:37:27 +00:00
|
|
|
ListBase *nurbs= curve_editnurbs(cu);
|
2010-07-25 11:57:36 +00:00
|
|
|
Nurb *nu= nurbs->first;
|
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
while(nu) {
|
2010-09-25 06:45:28 +00:00
|
|
|
/* keep handles' vectors unchanged */
|
|
|
|
if(nu->bezt) {
|
|
|
|
int a= nu->pntsu;
|
|
|
|
TransVert *tv= transvmain;
|
|
|
|
BezTriple *bezt= nu->bezt;
|
|
|
|
|
|
|
|
while(a--) {
|
|
|
|
if(bezt->f1 & SELECT) tv++;
|
|
|
|
|
|
|
|
if(bezt->f2 & SELECT) {
|
|
|
|
float v[3];
|
|
|
|
|
|
|
|
if(bezt->f1 & SELECT) {
|
|
|
|
sub_v3_v3v3(v, (tv-1)->oldloc, tv->oldloc);
|
|
|
|
add_v3_v3v3(bezt->vec[0], bezt->vec[1], v);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(bezt->f3 & SELECT) {
|
|
|
|
sub_v3_v3v3(v, (tv+1)->oldloc, tv->oldloc);
|
|
|
|
add_v3_v3v3(bezt->vec[2], bezt->vec[1], v);
|
|
|
|
}
|
|
|
|
|
|
|
|
tv++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(bezt->f3 & SELECT) tv++;
|
|
|
|
|
|
|
|
bezt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
test2DNurb(nu);
|
|
|
|
testhandlesNurb(nu); /* test for bezier too */
|
|
|
|
nu= nu->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(obedit->type==OB_ARMATURE){
|
|
|
|
bArmature *arm= obedit->data;
|
|
|
|
EditBone *ebo;
|
|
|
|
TransVert *tv= transvmain;
|
|
|
|
int a=0;
|
|
|
|
|
|
|
|
/* Ensure all bone tails are correctly adjusted */
|
|
|
|
for (ebo= arm->edbo->first; ebo; ebo=ebo->next) {
|
|
|
|
/* adjust tip if both ends selected */
|
|
|
|
if ((ebo->flag & BONE_ROOTSEL) && (ebo->flag & BONE_TIPSEL)) {
|
|
|
|
if (tv) {
|
|
|
|
float diffvec[3];
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
sub_v3_v3v3(diffvec, tv->loc, tv->oldloc);
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(ebo->tail, diffvec);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
a++;
|
|
|
|
if (a<tottrans) tv++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure all bones are correctly adjusted */
|
|
|
|
for (ebo= arm->edbo->first; ebo; ebo=ebo->next) {
|
|
|
|
if ((ebo->flag & BONE_CONNECTED) && ebo->parent){
|
|
|
|
/* If this bone has a parent tip that has been moved */
|
|
|
|
if (ebo->parent->flag & BONE_TIPSEL){
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(ebo->head, ebo->parent->tail);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
/* If this bone has a parent tip that has NOT been moved */
|
|
|
|
else{
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(ebo->parent->tail, ebo->head);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(arm->flag & ARM_MIRROR_EDIT)
|
|
|
|
transform_armature_mirror_update(obedit);
|
|
|
|
}
|
|
|
|
else if(obedit->type==OB_LATTICE) {
|
|
|
|
Lattice *lt= obedit->data;
|
|
|
|
|
2010-08-10 06:36:42 +00:00
|
|
|
if(lt->editlatt->latt->flag & LT_OUTSIDE)
|
|
|
|
outside_lattice(lt->editlatt->latt);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copied from editobject.c, needs to be replaced with new transform code still */
|
2010-09-25 06:45:28 +00:00
|
|
|
/* mode flags: */
|
|
|
|
#define TM_ALL_JOINTS 1 /* all joints (for bones only) */
|
|
|
|
#define TM_SKIP_HANDLES 2 /* skip handles when control point is selected (for curves only) */
|
2009-02-14 13:07:09 +00:00
|
|
|
static void make_trans_verts(Object *obedit, float *min, float *max, int mode)
|
|
|
|
{
|
|
|
|
Nurb *nu;
|
|
|
|
BezTriple *bezt;
|
|
|
|
BPoint *bp;
|
|
|
|
TransVert *tv=NULL;
|
|
|
|
MetaElem *ml;
|
|
|
|
EditVert *eve;
|
|
|
|
EditBone *ebo;
|
|
|
|
float total, center[3], centroid[3];
|
|
|
|
int a;
|
|
|
|
|
|
|
|
tottrans= 0; // global!
|
|
|
|
|
|
|
|
INIT_MINMAX(min, max);
|
|
|
|
centroid[0]=centroid[1]=centroid[2]= 0.0;
|
|
|
|
|
|
|
|
if(obedit->type==OB_MESH) {
|
|
|
|
Mesh *me= obedit->data;
|
|
|
|
EditMesh *em= me->edit_mesh;
|
|
|
|
|
|
|
|
// transform now requires awareness for select mode, so we tag the f1 flags in verts
|
|
|
|
tottrans= 0;
|
|
|
|
if(em->selectmode & SCE_SELECT_VERTEX) {
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) {
|
|
|
|
if(eve->h==0 && (eve->f & SELECT)) {
|
|
|
|
eve->f1= SELECT;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
else eve->f1= 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(em->selectmode & SCE_SELECT_EDGE) {
|
|
|
|
EditEdge *eed;
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) eve->f1= 0;
|
|
|
|
for(eed= em->edges.first; eed; eed= eed->next) {
|
|
|
|
if(eed->h==0 && (eed->f & SELECT)) eed->v1->f1= eed->v2->f1= SELECT;
|
|
|
|
}
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) if(eve->f1) tottrans++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
EditFace *efa;
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) eve->f1= 0;
|
|
|
|
for(efa= em->faces.first; efa; efa= efa->next) {
|
|
|
|
if(efa->h==0 && (efa->f & SELECT)) {
|
|
|
|
efa->v1->f1= efa->v2->f1= efa->v3->f1= SELECT;
|
|
|
|
if(efa->v4) efa->v4->f1= SELECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) if(eve->f1) tottrans++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* and now make transverts */
|
|
|
|
if(tottrans) {
|
|
|
|
tv=transvmain= MEM_callocN(tottrans*sizeof(TransVert), "maketransverts");
|
|
|
|
|
|
|
|
for(eve= em->verts.first; eve; eve= eve->next) {
|
|
|
|
if(eve->f1) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, eve->co);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= eve->co;
|
2011-03-27 15:57:27 +00:00
|
|
|
if(eve->no[0] != 0.0f || eve->no[1] != 0.0f ||eve->no[2] != 0.0f)
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->nor= eve->no; // note this is a hackish signal (ton)
|
|
|
|
tv->flag= eve->f1 & SELECT;
|
|
|
|
tv++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (obedit->type==OB_ARMATURE){
|
|
|
|
bArmature *arm= obedit->data;
|
|
|
|
int totmalloc= BLI_countlist(arm->edbo);
|
2009-11-02 17:15:14 +00:00
|
|
|
|
2010-03-22 09:30:00 +00:00
|
|
|
totmalloc *= 2; /* probably overkill but bones can have 2 trans verts each */
|
2009-11-02 17:15:14 +00:00
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
tv=transvmain= MEM_callocN(totmalloc*sizeof(TransVert), "maketransverts armature");
|
|
|
|
|
|
|
|
for (ebo= arm->edbo->first; ebo; ebo=ebo->next){
|
|
|
|
if(ebo->layer & arm->layer) {
|
|
|
|
short tipsel= (ebo->flag & BONE_TIPSEL);
|
|
|
|
short rootsel= (ebo->flag & BONE_ROOTSEL);
|
|
|
|
short rootok= (!(ebo->parent && (ebo->flag & BONE_CONNECTED) && ebo->parent->flag & BONE_TIPSEL));
|
|
|
|
|
|
|
|
if ((tipsel && rootsel) || (rootsel)) {
|
2010-09-25 06:45:28 +00:00
|
|
|
/* Don't add the tip (unless mode & TM_ALL_JOINTS, for getting all joints),
|
2009-02-14 13:07:09 +00:00
|
|
|
* otherwise we get zero-length bones as tips will snap to the same
|
|
|
|
* location as heads.
|
|
|
|
*/
|
|
|
|
if (rootok) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, ebo->head);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= ebo->head;
|
|
|
|
tv->nor= NULL;
|
|
|
|
tv->flag= 1;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
|
2010-09-25 06:45:28 +00:00
|
|
|
if ((mode & TM_ALL_JOINTS) && (tipsel)) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, ebo->tail);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= ebo->tail;
|
|
|
|
tv->nor= NULL;
|
|
|
|
tv->flag= 1;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (tipsel) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, ebo->tail);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= ebo->tail;
|
|
|
|
tv->nor= NULL;
|
|
|
|
tv->flag= 1;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (ELEM(obedit->type, OB_CURVE, OB_SURF)) {
|
|
|
|
Curve *cu= obedit->data;
|
|
|
|
int totmalloc= 0;
|
2011-09-14 00:37:27 +00:00
|
|
|
ListBase *nurbs= curve_editnurbs(cu);
|
2010-07-25 11:57:36 +00:00
|
|
|
|
|
|
|
for(nu= nurbs->first; nu; nu= nu->next) {
|
2009-09-08 00:23:33 +00:00
|
|
|
if(nu->type == CU_BEZIER)
|
2009-02-14 13:07:09 +00:00
|
|
|
totmalloc += 3*nu->pntsu;
|
|
|
|
else
|
|
|
|
totmalloc += nu->pntsu*nu->pntsv;
|
|
|
|
}
|
|
|
|
tv=transvmain= MEM_callocN(totmalloc*sizeof(TransVert), "maketransverts curve");
|
|
|
|
|
2010-07-25 11:57:36 +00:00
|
|
|
nu= nurbs->first;
|
2009-02-14 13:07:09 +00:00
|
|
|
while(nu) {
|
2009-09-08 00:23:33 +00:00
|
|
|
if(nu->type == CU_BEZIER) {
|
2009-02-14 13:07:09 +00:00
|
|
|
a= nu->pntsu;
|
|
|
|
bezt= nu->bezt;
|
|
|
|
while(a--) {
|
|
|
|
if(bezt->hide==0) {
|
2010-09-25 06:45:28 +00:00
|
|
|
int skip_handle= 0;
|
|
|
|
if(bezt->f2 & SELECT)
|
|
|
|
skip_handle= mode & TM_SKIP_HANDLES;
|
|
|
|
|
|
|
|
if((bezt->f1 & SELECT) && !skip_handle) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, bezt->vec[0]);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= bezt->vec[0];
|
|
|
|
tv->flag= bezt->f1 & SELECT;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
2010-09-25 06:45:28 +00:00
|
|
|
if(bezt->f2 & SELECT) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, bezt->vec[1]);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= bezt->vec[1];
|
|
|
|
tv->val= &(bezt->alfa);
|
|
|
|
tv->oldval= bezt->alfa;
|
|
|
|
tv->flag= bezt->f2 & SELECT;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
2010-09-25 06:45:28 +00:00
|
|
|
if((bezt->f3 & SELECT) && !skip_handle) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, bezt->vec[2]);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= bezt->vec[2];
|
|
|
|
tv->flag= bezt->f3 & SELECT;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bezt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
a= nu->pntsu*nu->pntsv;
|
|
|
|
bp= nu->bp;
|
|
|
|
while(a--) {
|
|
|
|
if(bp->hide==0) {
|
2010-09-25 06:45:28 +00:00
|
|
|
if(bp->f1 & SELECT) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->oldloc, bp->vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= bp->vec;
|
|
|
|
tv->val= &(bp->alfa);
|
|
|
|
tv->oldval= bp->alfa;
|
|
|
|
tv->flag= bp->f1 & SELECT;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bp++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nu= nu->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(obedit->type==OB_MBALL) {
|
|
|
|
MetaBall *mb= obedit->data;
|
|
|
|
int totmalloc= BLI_countlist(mb->editelems);
|
|
|
|
|
|
|
|
tv=transvmain= MEM_callocN(totmalloc*sizeof(TransVert), "maketransverts mball");
|
|
|
|
|
|
|
|
ml= mb->editelems->first;
|
|
|
|
while(ml) {
|
|
|
|
if(ml->flag & SELECT) {
|
|
|
|
tv->loc= &ml->x;
|
2010-07-31 10:58:10 +00:00
|
|
|
copy_v3_v3(tv->oldloc, tv->loc);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->val= &(ml->rad);
|
|
|
|
tv->oldval= ml->rad;
|
|
|
|
tv->flag= 1;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
ml= ml->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(obedit->type==OB_LATTICE) {
|
|
|
|
Lattice *lt= obedit->data;
|
|
|
|
|
2010-08-10 06:36:42 +00:00
|
|
|
bp= lt->editlatt->latt->def;
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2010-08-10 06:36:42 +00:00
|
|
|
a= lt->editlatt->latt->pntsu*lt->editlatt->latt->pntsv*lt->editlatt->latt->pntsw;
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2010-09-25 06:45:28 +00:00
|
|
|
tv=transvmain= MEM_callocN(a*sizeof(TransVert), "maketransverts latt");
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
while(a--) {
|
2010-09-25 06:45:28 +00:00
|
|
|
if(bp->f1 & SELECT) {
|
2009-02-14 13:07:09 +00:00
|
|
|
if(bp->hide==0) {
|
2010-07-31 10:58:10 +00:00
|
|
|
copy_v3_v3(tv->oldloc, bp->vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
tv->loc= bp->vec;
|
|
|
|
tv->flag= bp->f1 & SELECT;
|
|
|
|
tv++;
|
|
|
|
tottrans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bp++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-25 06:45:28 +00:00
|
|
|
if(!tottrans && transvmain) {
|
|
|
|
/* prevent memory leak. happens for curves/latticies due to */
|
|
|
|
/* difficult condition of adding points to trans data */
|
|
|
|
MEM_freeN(transvmain);
|
|
|
|
transvmain= NULL;
|
|
|
|
}
|
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
/* cent etc */
|
|
|
|
tv= transvmain;
|
|
|
|
total= 0.0;
|
|
|
|
for(a=0; a<tottrans; a++, tv++) {
|
|
|
|
if(tv->flag & SELECT) {
|
2010-07-31 10:58:10 +00:00
|
|
|
add_v3_v3(centroid, tv->oldloc);
|
2011-03-27 15:57:27 +00:00
|
|
|
total += 1.0f;
|
2009-02-14 13:07:09 +00:00
|
|
|
DO_MINMAX(tv->oldloc, min, max);
|
|
|
|
}
|
|
|
|
}
|
2011-03-27 15:57:27 +00:00
|
|
|
if(total != 0.0f) {
|
2010-07-31 10:58:10 +00:00
|
|
|
mul_v3_fl(centroid, 1.0f/total);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
|
2010-07-31 10:58:10 +00:00
|
|
|
mid_v3_v3v3(center, min, max);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* *********************** operators ******************** */
|
|
|
|
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_sel_to_grid(bContext *C, wmOperator *UNUSED(op))
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
2010-08-01 12:47:49 +00:00
|
|
|
Main *bmain= CTX_data_main(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
Object *obedit= CTX_data_edit_object(C);
|
|
|
|
Scene *scene= CTX_data_scene(C);
|
2009-12-27 18:09:17 +00:00
|
|
|
RegionView3D *rv3d= CTX_wm_region_data(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
TransVert *tv;
|
|
|
|
float gridf, imat[3][3], bmat[3][3], vec[3];
|
|
|
|
int a;
|
|
|
|
|
2009-12-27 18:09:17 +00:00
|
|
|
gridf= rv3d->gridview;
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
if(obedit) {
|
|
|
|
tottrans= 0;
|
|
|
|
|
|
|
|
if ELEM6(obedit->type, OB_ARMATURE, OB_LATTICE, OB_MESH, OB_SURF, OB_CURVE, OB_MBALL)
|
|
|
|
make_trans_verts(obedit, bmat[0], bmat[1], 0);
|
|
|
|
if(tottrans==0) return OPERATOR_CANCELLED;
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
copy_m3_m4(bmat, obedit->obmat);
|
|
|
|
invert_m3_m3(imat, bmat);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
tv= transvmain;
|
|
|
|
for(a=0; a<tottrans; a++, tv++) {
|
|
|
|
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(vec, tv->loc);
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m3_v3(bmat, vec);
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(vec, obedit->obmat[3]);
|
2011-03-27 15:57:27 +00:00
|
|
|
vec[0]= gridf*floorf(0.5f+ vec[0]/gridf);
|
|
|
|
vec[1]= gridf*floorf(0.5f+ vec[1]/gridf);
|
|
|
|
vec[2]= gridf*floorf(0.5f+ vec[2]/gridf);
|
2010-04-23 23:57:00 +00:00
|
|
|
sub_v3_v3(vec, obedit->obmat[3]);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m3_v3(imat, vec);
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(tv->loc, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
|
2010-10-15 12:29:02 +00:00
|
|
|
special_transvert_update(obedit);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
MEM_freeN(transvmain);
|
|
|
|
transvmain= NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
2011-10-23 05:08:02 +00:00
|
|
|
struct KeyingSet *ks = ANIM_get_keyingset_for_autokeying(scene, ANIM_KS_LOCATION_ID);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2009-10-29 09:14:20 +00:00
|
|
|
CTX_DATA_BEGIN(C, Object*, ob, selected_editable_objects) {
|
2009-08-16 03:24:23 +00:00
|
|
|
if(ob->mode & OB_MODE_POSE) {
|
2009-02-14 13:07:09 +00:00
|
|
|
bPoseChannel *pchan;
|
|
|
|
bArmature *arm= ob->data;
|
|
|
|
|
|
|
|
for (pchan= ob->pose->chanbase.first; pchan; pchan= pchan->next) {
|
|
|
|
if(pchan->bone->flag & BONE_SELECTED) {
|
|
|
|
if(pchan->bone->layer & arm->layer) {
|
|
|
|
if((pchan->bone->flag & BONE_CONNECTED)==0) {
|
|
|
|
float vecN[3], nLoc[3];
|
|
|
|
|
|
|
|
/* get nearest grid point to snap to */
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(nLoc, pchan->pose_mat[3]);
|
2011-03-27 15:57:27 +00:00
|
|
|
vec[0]= gridf * (float)(floor(0.5f+ nLoc[0]/gridf));
|
|
|
|
vec[1]= gridf * (float)(floor(0.5f+ nLoc[1]/gridf));
|
|
|
|
vec[2]= gridf * (float)(floor(0.5f+ nLoc[2]/gridf));
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
/* get bone-space location of grid point */
|
|
|
|
armature_loc_pose_to_bone(pchan, vec, vecN);
|
|
|
|
|
|
|
|
/* adjust location */
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCX)==0)
|
|
|
|
pchan->loc[0]= vecN[0];
|
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCY)==0)
|
|
|
|
pchan->loc[0]= vecN[1];
|
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCZ)==0)
|
|
|
|
pchan->loc[0]= vecN[2];
|
2011-10-10 12:56:21 +00:00
|
|
|
|
|
|
|
/* auto-keyframing */
|
|
|
|
ED_autokeyframe_pchan(C, scene, ob, pchan, ks);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
/* if the bone has a parent and is connected to the parent,
|
|
|
|
* don't do anything - will break chain unless we do auto-ik.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ob->pose->flag |= (POSE_LOCKED|POSE_DO_UNLOCK);
|
|
|
|
|
2010-12-05 18:59:23 +00:00
|
|
|
DAG_id_tag_update(&ob->id, OB_RECALC_DATA);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ob->recalc |= OB_RECALC_OB;
|
|
|
|
|
2011-03-27 15:57:27 +00:00
|
|
|
vec[0]= -ob->obmat[3][0]+gridf*floorf(0.5f+ ob->obmat[3][0]/gridf);
|
|
|
|
vec[1]= -ob->obmat[3][1]+gridf*floorf(0.5f+ ob->obmat[3][1]/gridf);
|
|
|
|
vec[2]= -ob->obmat[3][2]+gridf*floorf(0.5f+ ob->obmat[3][2]/gridf);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
if(ob->parent) {
|
|
|
|
where_is_object(scene, ob);
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
invert_m3_m3(imat, originmat);
|
|
|
|
mul_m3_v3(imat, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCX)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[0]+= vec[0];
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCY)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[1]+= vec[1];
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCZ)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[2]+= vec[2];
|
2010-12-21 18:55:49 +00:00
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
/* auto-keyframing */
|
2011-10-10 12:56:21 +00:00
|
|
|
ED_autokeyframe_object(C, scene, ob, ks);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
CTX_DATA_END;
|
|
|
|
}
|
2009-11-24 11:48:16 +00:00
|
|
|
|
2010-08-01 12:47:49 +00:00
|
|
|
DAG_ids_flush_update(bmain, 0);
|
2009-02-14 13:07:09 +00:00
|
|
|
WM_event_add_notifier(C, NC_OBJECT|ND_TRANSFORM, NULL);
|
|
|
|
|
|
|
|
return OPERATOR_FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_selected_to_grid(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Selection to Grid";
|
2010-02-10 21:15:44 +00:00
|
|
|
ot->description= "Snap selected item(s) to nearest grid node";
|
2009-02-14 13:07:09 +00:00
|
|
|
ot->idname= "VIEW3D_OT_snap_selected_to_grid";
|
|
|
|
|
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_sel_to_grid;
|
2010-09-27 10:44:46 +00:00
|
|
|
ot->poll= ED_operator_region_view3d_active;
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* *************************************************** */
|
|
|
|
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_sel_to_curs(bContext *C, wmOperator *UNUSED(op))
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
2010-08-01 12:47:49 +00:00
|
|
|
Main *bmain= CTX_data_main(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
Object *obedit= CTX_data_edit_object(C);
|
|
|
|
Scene *scene= CTX_data_scene(C);
|
|
|
|
View3D *v3d= CTX_wm_view3d(C);
|
|
|
|
TransVert *tv;
|
|
|
|
float *curs, imat[3][3], bmat[3][3], vec[3];
|
|
|
|
int a;
|
|
|
|
|
|
|
|
curs= give_cursor(scene, v3d);
|
|
|
|
|
|
|
|
if(obedit) {
|
|
|
|
tottrans= 0;
|
|
|
|
|
|
|
|
if ELEM6(obedit->type, OB_ARMATURE, OB_LATTICE, OB_MESH, OB_SURF, OB_CURVE, OB_MBALL)
|
|
|
|
make_trans_verts(obedit, bmat[0], bmat[1], 0);
|
|
|
|
if(tottrans==0) return OPERATOR_CANCELLED;
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
copy_m3_m4(bmat, obedit->obmat);
|
|
|
|
invert_m3_m3(imat, bmat);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
tv= transvmain;
|
|
|
|
for(a=0; a<tottrans; a++, tv++) {
|
2010-10-15 12:29:02 +00:00
|
|
|
sub_v3_v3v3(vec, curs, obedit->obmat[3]);
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m3_v3(imat, vec);
|
2010-10-15 12:29:02 +00:00
|
|
|
copy_v3_v3(tv->loc, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
|
2010-10-15 12:29:02 +00:00
|
|
|
special_transvert_update(obedit);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
MEM_freeN(transvmain);
|
|
|
|
transvmain= NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
2011-10-23 05:08:02 +00:00
|
|
|
struct KeyingSet *ks = ANIM_get_keyingset_for_autokeying(scene, ANIM_KS_LOCATION_ID);
|
2011-10-10 12:56:21 +00:00
|
|
|
|
2009-10-29 09:14:20 +00:00
|
|
|
CTX_DATA_BEGIN(C, Object*, ob, selected_editable_objects) {
|
2009-08-16 03:24:23 +00:00
|
|
|
if(ob->mode & OB_MODE_POSE) {
|
2009-02-14 13:07:09 +00:00
|
|
|
bPoseChannel *pchan;
|
|
|
|
bArmature *arm= ob->data;
|
|
|
|
float cursp[3];
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
invert_m4_m4(ob->imat, ob->obmat);
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(cursp, curs);
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m4_v3(ob->imat, cursp);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
for (pchan = ob->pose->chanbase.first; pchan; pchan=pchan->next) {
|
|
|
|
if(pchan->bone->flag & BONE_SELECTED) {
|
|
|
|
if(pchan->bone->layer & arm->layer) {
|
|
|
|
if((pchan->bone->flag & BONE_CONNECTED)==0) {
|
|
|
|
float curspn[3];
|
2011-12-07 15:21:38 +00:00
|
|
|
float inv_restmat[4][4];
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2011-12-07 15:21:38 +00:00
|
|
|
/* get location of cursor in *rest* bone-space */
|
|
|
|
copy_v3_v3(curspn, cursp);
|
|
|
|
invert_m4_m4(inv_restmat, pchan->bone->arm_mat);
|
|
|
|
mul_m4_v3(inv_restmat, curspn);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2010-12-21 18:55:49 +00:00
|
|
|
/* copy new position */
|
2011-12-07 15:21:38 +00:00
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCX)==0)
|
2010-12-21 18:55:49 +00:00
|
|
|
pchan->loc[0]= curspn[0];
|
2011-12-07 15:21:38 +00:00
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCY)==0)
|
2010-12-21 18:55:49 +00:00
|
|
|
pchan->loc[1]= curspn[1];
|
2011-12-07 15:21:38 +00:00
|
|
|
if ((pchan->protectflag & OB_LOCK_LOCZ)==0)
|
2010-12-21 18:55:49 +00:00
|
|
|
pchan->loc[2]= curspn[2];
|
2011-10-10 12:56:21 +00:00
|
|
|
|
|
|
|
/* auto-keyframing */
|
|
|
|
ED_autokeyframe_pchan(C, scene, ob, pchan, ks);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
/* if the bone has a parent and is connected to the parent,
|
|
|
|
* don't do anything - will break chain unless we do auto-ik.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ob->pose->flag |= (POSE_LOCKED|POSE_DO_UNLOCK);
|
|
|
|
|
2010-12-05 18:59:23 +00:00
|
|
|
DAG_id_tag_update(&ob->id, OB_RECALC_DATA);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ob->recalc |= OB_RECALC_OB;
|
|
|
|
|
|
|
|
vec[0]= -ob->obmat[3][0] + curs[0];
|
|
|
|
vec[1]= -ob->obmat[3][1] + curs[1];
|
|
|
|
vec[2]= -ob->obmat[3][2] + curs[2];
|
|
|
|
|
|
|
|
if(ob->parent) {
|
|
|
|
where_is_object(scene, ob);
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
invert_m3_m3(imat, originmat);
|
|
|
|
mul_m3_v3(imat, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCX)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[0]+= vec[0];
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCY)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[1]+= vec[1];
|
2010-12-21 18:55:49 +00:00
|
|
|
if ((ob->protectflag & OB_LOCK_LOCZ)==0)
|
2009-02-14 13:07:09 +00:00
|
|
|
ob->loc[2]+= vec[2];
|
2011-10-10 12:56:21 +00:00
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
/* auto-keyframing */
|
2011-10-10 12:56:21 +00:00
|
|
|
ED_autokeyframe_object(C, scene, ob, ks);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
CTX_DATA_END;
|
|
|
|
}
|
2009-11-24 11:48:16 +00:00
|
|
|
|
2010-08-01 12:47:49 +00:00
|
|
|
DAG_ids_flush_update(bmain, 0);
|
2009-02-14 13:07:09 +00:00
|
|
|
WM_event_add_notifier(C, NC_OBJECT|ND_TRANSFORM, NULL);
|
|
|
|
|
|
|
|
return OPERATOR_FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_selected_to_cursor(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Selection to Cursor";
|
2010-02-10 21:15:44 +00:00
|
|
|
ot->description= "Snap selected item(s) to cursor";
|
2009-02-14 13:07:09 +00:00
|
|
|
ot->idname= "VIEW3D_OT_snap_selected_to_cursor";
|
|
|
|
|
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_sel_to_curs;
|
|
|
|
ot->poll= ED_operator_view3d_active;
|
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* *************************************************** */
|
|
|
|
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_curs_to_grid(bContext *C, wmOperator *UNUSED(op))
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
|
|
|
Scene *scene= CTX_data_scene(C);
|
2009-12-27 18:09:17 +00:00
|
|
|
RegionView3D *rv3d= CTX_wm_region_data(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
View3D *v3d= CTX_wm_view3d(C);
|
|
|
|
float gridf, *curs;
|
|
|
|
|
2009-12-27 18:09:17 +00:00
|
|
|
gridf= rv3d->gridview;
|
2009-02-14 13:07:09 +00:00
|
|
|
curs= give_cursor(scene, v3d);
|
|
|
|
|
2011-03-27 15:57:27 +00:00
|
|
|
curs[0]= gridf*floorf(0.5f+curs[0]/gridf);
|
|
|
|
curs[1]= gridf*floorf(0.5f+curs[1]/gridf);
|
|
|
|
curs[2]= gridf*floorf(0.5f+curs[2]/gridf);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2010-01-25 19:42:33 +00:00
|
|
|
WM_event_add_notifier(C, NC_SPACE|ND_SPACE_VIEW3D, v3d); // hrm
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
return OPERATOR_FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_cursor_to_grid(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Cursor to Grid";
|
2010-02-10 21:15:44 +00:00
|
|
|
ot->description= "Snap cursor to nearest grid node";
|
2009-02-14 13:07:09 +00:00
|
|
|
ot->idname= "VIEW3D_OT_snap_cursor_to_grid";
|
|
|
|
|
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_curs_to_grid;
|
2010-09-27 10:44:46 +00:00
|
|
|
ot->poll= ED_operator_region_view3d_active;
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* **************************************************** */
|
|
|
|
|
2011-11-07 12:55:18 +00:00
|
|
|
static void bundle_midpoint(Scene *scene, Object *ob, float vec[3])
|
|
|
|
{
|
|
|
|
MovieTrackingTrack *track;
|
|
|
|
MovieClip *clip= object_get_movieclip(scene, ob, 0);
|
|
|
|
int ok= 0;
|
|
|
|
float min[3], max[3], mat[4][4], pos[3];
|
|
|
|
|
|
|
|
if(!clip)
|
|
|
|
return;
|
|
|
|
|
|
|
|
BKE_get_tracking_mat(scene, ob, mat);
|
|
|
|
|
|
|
|
INIT_MINMAX(min, max);
|
|
|
|
|
|
|
|
track= clip->tracking.tracks.first;
|
|
|
|
while(track) {
|
|
|
|
int selected= (track->flag&SELECT) || (track->pat_flag&SELECT) || (track->search_flag&SELECT);
|
|
|
|
if((track->flag&TRACK_HAS_BUNDLE) && selected) {
|
|
|
|
ok= 1;
|
|
|
|
mul_v3_m4v3(pos, mat, track->bundle_pos);
|
|
|
|
DO_MINMAX(pos, min, max);
|
|
|
|
}
|
|
|
|
|
|
|
|
track= track->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ok) {
|
2011-11-08 09:02:47 +00:00
|
|
|
mid_v3_v3v3(vec, min, max);
|
2011-11-07 12:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_curs_to_sel(bContext *C, wmOperator *UNUSED(op))
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
|
|
|
Object *obedit= CTX_data_edit_object(C);
|
|
|
|
Scene *scene= CTX_data_scene(C);
|
|
|
|
View3D *v3d= CTX_wm_view3d(C);
|
|
|
|
TransVert *tv;
|
|
|
|
float *curs, bmat[3][3], vec[3], min[3], max[3], centroid[3];
|
|
|
|
int count, a;
|
|
|
|
|
|
|
|
curs= give_cursor(scene, v3d);
|
|
|
|
|
|
|
|
count= 0;
|
|
|
|
INIT_MINMAX(min, max);
|
|
|
|
centroid[0]= centroid[1]= centroid[2]= 0.0;
|
|
|
|
|
|
|
|
if(obedit) {
|
|
|
|
tottrans=0;
|
|
|
|
|
|
|
|
if ELEM6(obedit->type, OB_ARMATURE, OB_LATTICE, OB_MESH, OB_SURF, OB_CURVE, OB_MBALL)
|
2010-09-25 06:45:28 +00:00
|
|
|
make_trans_verts(obedit, bmat[0], bmat[1], TM_ALL_JOINTS|TM_SKIP_HANDLES);
|
2009-02-14 13:07:09 +00:00
|
|
|
if(tottrans==0) return OPERATOR_CANCELLED;
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
copy_m3_m4(bmat, obedit->obmat);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
tv= transvmain;
|
|
|
|
for(a=0; a<tottrans; a++, tv++) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(vec, tv->loc);
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m3_v3(bmat, vec);
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(vec, obedit->obmat[3]);
|
|
|
|
add_v3_v3(centroid, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
DO_MINMAX(vec, min, max);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(v3d->around==V3D_CENTROID) {
|
2011-03-27 15:57:27 +00:00
|
|
|
mul_v3_fl(centroid, 1.0f/(float)tottrans);
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(curs, centroid);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-04-19 05:34:05 +00:00
|
|
|
mid_v3_v3v3(curs, min, max);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
MEM_freeN(transvmain);
|
|
|
|
transvmain= NULL;
|
|
|
|
}
|
|
|
|
else {
|
2010-12-03 12:30:59 +00:00
|
|
|
Object *obact= CTX_data_active_object(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
2010-12-03 12:30:59 +00:00
|
|
|
if(obact && (obact->mode & OB_MODE_POSE)) {
|
|
|
|
bArmature *arm= obact->data;
|
2009-02-14 13:07:09 +00:00
|
|
|
bPoseChannel *pchan;
|
2010-12-03 12:30:59 +00:00
|
|
|
for (pchan = obact->pose->chanbase.first; pchan; pchan=pchan->next) {
|
2009-02-14 13:07:09 +00:00
|
|
|
if(arm->layer & pchan->bone->layer) {
|
|
|
|
if(pchan->bone->flag & BONE_SELECTED) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(vec, pchan->pose_head);
|
2010-12-03 12:30:59 +00:00
|
|
|
mul_m4_v3(obact->obmat, vec);
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(centroid, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
DO_MINMAX(vec, min, max);
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2010-06-02 13:41:05 +00:00
|
|
|
CTX_DATA_BEGIN(C, Object*, ob, selected_objects) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(vec, ob->obmat[3]);
|
2011-11-07 12:55:18 +00:00
|
|
|
|
|
|
|
/* special case for camera -- snap to bundles */
|
|
|
|
if(ob->type==OB_CAMERA) {
|
|
|
|
/* snap to bundles should happen only when bundles are visible */
|
|
|
|
if(v3d->flag2&V3D_SHOW_RECONSTRUCTION) {
|
|
|
|
bundle_midpoint(scene, ob, vec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(centroid, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
DO_MINMAX(vec, min, max);
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
CTX_DATA_END;
|
|
|
|
}
|
|
|
|
if(count) {
|
|
|
|
if(v3d->around==V3D_CENTROID) {
|
2011-03-27 15:57:27 +00:00
|
|
|
mul_v3_fl(centroid, 1.0f/(float)count);
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(curs, centroid);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-04-19 05:34:05 +00:00
|
|
|
mid_v3_v3v3(curs, min, max);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-01-25 19:42:33 +00:00
|
|
|
WM_event_add_notifier(C, NC_SPACE|ND_SPACE_VIEW3D, v3d);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
return OPERATOR_FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_cursor_to_selected(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Cursor to Selected";
|
2010-02-10 21:15:44 +00:00
|
|
|
ot->description= "Snap cursor to center of selected item(s)";
|
2009-02-14 13:07:09 +00:00
|
|
|
ot->idname= "VIEW3D_OT_snap_cursor_to_selected";
|
|
|
|
|
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_curs_to_sel;
|
|
|
|
ot->poll= ED_operator_view3d_active;
|
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ********************************************** */
|
|
|
|
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_curs_to_active(bContext *C, wmOperator *UNUSED(op))
|
2009-02-14 13:07:09 +00:00
|
|
|
{
|
|
|
|
Object *obedit= CTX_data_edit_object(C);
|
2009-10-29 09:14:20 +00:00
|
|
|
Object *obact= CTX_data_active_object(C);
|
2009-02-14 13:07:09 +00:00
|
|
|
Scene *scene= CTX_data_scene(C);
|
|
|
|
View3D *v3d= CTX_wm_view3d(C);
|
|
|
|
float *curs;
|
|
|
|
|
|
|
|
curs = give_cursor(scene, v3d);
|
|
|
|
|
|
|
|
if (obedit) {
|
|
|
|
if (obedit->type == OB_MESH) {
|
|
|
|
/* check active */
|
|
|
|
Mesh *me= obedit->data;
|
|
|
|
EditSelection ese;
|
|
|
|
|
|
|
|
if (EM_get_actSelection(me->edit_mesh, &ese)) {
|
|
|
|
EM_editselection_center(curs, &ese);
|
|
|
|
}
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m4_v3(obedit->obmat, curs);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2009-10-29 09:14:20 +00:00
|
|
|
if (obact) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(curs, obact->obmat[3]);
|
2009-02-14 13:07:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-25 19:42:33 +00:00
|
|
|
WM_event_add_notifier(C, NC_SPACE|ND_SPACE_VIEW3D, v3d);
|
2009-02-14 13:07:09 +00:00
|
|
|
return OPERATOR_FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_cursor_to_active(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Cursor to Active";
|
2010-02-10 21:15:44 +00:00
|
|
|
ot->description= "Snap cursor to active item";
|
2009-02-14 13:07:09 +00:00
|
|
|
ot->idname= "VIEW3D_OT_snap_cursor_to_active";
|
|
|
|
|
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_curs_to_active;
|
|
|
|
ot->poll= ED_operator_view3d_active;
|
|
|
|
|
|
|
|
/* flags */
|
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
|
2010-01-13 22:17:56 +00:00
|
|
|
/* **************************************************** */
|
|
|
|
/*New Code - Snap Cursor to Center -*/
|
2010-10-15 01:36:14 +00:00
|
|
|
static int snap_curs_to_center(bContext *C, wmOperator *UNUSED(op))
|
2010-01-13 22:17:56 +00:00
|
|
|
{
|
2010-03-22 09:30:00 +00:00
|
|
|
Scene *scene= CTX_data_scene(C);
|
|
|
|
View3D *v3d= CTX_wm_view3d(C);
|
|
|
|
float *curs;
|
|
|
|
curs= give_cursor(scene, v3d);
|
|
|
|
|
|
|
|
curs[0]= 0.0;
|
|
|
|
curs[1]= 0.0;
|
|
|
|
curs[2]= 0.0;
|
2010-01-13 22:17:56 +00:00
|
|
|
|
2010-03-22 09:30:00 +00:00
|
|
|
WM_event_add_notifier(C, NC_SPACE|ND_SPACE_VIEW3D, v3d);
|
2010-01-13 22:17:56 +00:00
|
|
|
|
2010-03-22 09:30:00 +00:00
|
|
|
return OPERATOR_FINISHED;
|
2010-01-13 22:17:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VIEW3D_OT_snap_cursor_to_center(wmOperatorType *ot)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* identifiers */
|
|
|
|
ot->name= "Snap Cursor to Center";
|
|
|
|
ot->description= "Snap cursor to the Center";
|
|
|
|
ot->idname= "VIEW3D_OT_snap_cursor_to_center";
|
|
|
|
|
2010-03-22 09:30:00 +00:00
|
|
|
/* api callbacks */
|
|
|
|
ot->exec= snap_curs_to_center;
|
2010-10-22 03:56:50 +00:00
|
|
|
ot->poll= ED_operator_view3d_active;
|
2010-01-13 22:17:56 +00:00
|
|
|
|
2010-03-22 09:30:00 +00:00
|
|
|
/* flags */
|
2011-11-11 13:09:14 +00:00
|
|
|
ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
|
2010-01-13 22:17:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* **************************************************** */
|
|
|
|
|
|
|
|
|
2009-02-14 13:07:09 +00:00
|
|
|
int minmax_verts(Object *obedit, float *min, float *max)
|
|
|
|
{
|
|
|
|
TransVert *tv;
|
|
|
|
float centroid[3], vec[3], bmat[3][3];
|
|
|
|
int a;
|
|
|
|
|
|
|
|
tottrans=0;
|
|
|
|
if ELEM5(obedit->type, OB_ARMATURE, OB_LATTICE, OB_MESH, OB_SURF, OB_CURVE)
|
2010-09-25 06:45:28 +00:00
|
|
|
make_trans_verts(obedit, bmat[0], bmat[1], TM_ALL_JOINTS);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
if(tottrans==0) return 0;
|
|
|
|
|
2009-11-10 20:43:45 +00:00
|
|
|
copy_m3_m4(bmat, obedit->obmat);
|
2009-02-14 13:07:09 +00:00
|
|
|
|
|
|
|
tv= transvmain;
|
|
|
|
for(a=0; a<tottrans; a++, tv++) {
|
2011-10-28 12:40:15 +00:00
|
|
|
copy_v3_v3(vec, tv->loc);
|
2009-11-10 20:43:45 +00:00
|
|
|
mul_m3_v3(bmat, vec);
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(vec, obedit->obmat[3]);
|
|
|
|
add_v3_v3(centroid, vec);
|
2009-02-14 13:07:09 +00:00
|
|
|
DO_MINMAX(vec, min, max);
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(transvmain);
|
|
|
|
transvmain= NULL;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|