2002-10-12 11:37:38 +00:00
|
|
|
/**
|
2003-04-26 11:56:44 +00:00
|
|
|
* lattice.c
|
|
|
|
|
*
|
|
|
|
|
*
|
2002-10-12 11:37:38 +00:00
|
|
|
* $Id$
|
|
|
|
|
*
|
|
|
|
|
* ***** BEGIN GPL/BL DUAL 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. The Blender
|
|
|
|
|
* Foundation also sells licenses for use in proprietary software under
|
|
|
|
|
* the Blender License. See http://www.blender.org/BL/ for information
|
|
|
|
|
* about this.
|
|
|
|
|
*
|
|
|
|
|
* 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) 2001-2002 by NaN Holding BV.
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* The Original Code is: all of this file.
|
|
|
|
|
*
|
|
|
|
|
* Contributor(s): none yet.
|
|
|
|
|
*
|
|
|
|
|
* ***** END GPL/BL DUAL LICENSE BLOCK *****
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
|
|
|
|
#include "BLI_arithb.h"
|
|
|
|
|
|
|
|
|
|
#include "DNA_armature_types.h"
|
|
|
|
|
#include "DNA_mesh_types.h"
|
2004-03-20 22:55:42 +00:00
|
|
|
#include "DNA_meshdata_types.h"
|
2005-08-15 10:30:53 +00:00
|
|
|
#include "DNA_modifier_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "DNA_object_types.h"
|
|
|
|
|
#include "DNA_scene_types.h"
|
|
|
|
|
#include "DNA_lattice_types.h"
|
|
|
|
|
#include "DNA_curve_types.h"
|
|
|
|
|
#include "DNA_key_types.h"
|
|
|
|
|
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "BKE_anim.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_armature.h"
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "BKE_curve.h"
|
|
|
|
|
#include "BKE_deform.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_displist.h"
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "BKE_global.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_key.h"
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "BKE_lattice.h"
|
|
|
|
|
#include "BKE_library.h"
|
|
|
|
|
#include "BKE_main.h"
|
2005-08-15 10:30:53 +00:00
|
|
|
#include "BKE_modifier.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_object.h"
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "BKE_screen.h"
|
|
|
|
|
#include "BKE_utildefines.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-07-21 21:19:38 +00:00
|
|
|
#include "BIF_editdeform.h"
|
|
|
|
|
|
2002-11-25 12:02:15 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2004-09-14 19:03:11 +00:00
|
|
|
#include "blendef.h"
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
Lattice *editLatt=0;
|
|
|
|
|
static Lattice *deformLatt=0;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
static float *latticedata=0, latmat[4][4];
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
void calc_lat_fudu(int flag, int res, float *fu, float *du)
|
|
|
|
|
{
|
|
|
|
|
if(res==1) {
|
|
|
|
|
*fu= 0.0;
|
|
|
|
|
*du= 0.0;
|
|
|
|
|
}
|
|
|
|
|
else if(flag & LT_GRID) {
|
|
|
|
|
*fu= -0.5f*(res-1);
|
|
|
|
|
*du= 1.0f;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
*fu= -1.0f;
|
|
|
|
|
*du= 2.0f/(res-1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void resizelattice(Lattice *lt, int uNew, int vNew, int wNew, Object *ltOb)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
BPoint *bp;
|
2005-08-15 10:30:53 +00:00
|
|
|
int i, u, v, w;
|
|
|
|
|
float fu, fv, fw, uc, vc, wc, du=0.0, dv=0.0, dw=0.0;
|
|
|
|
|
float *co, (*vertexCos)[3] = NULL;
|
|
|
|
|
|
|
|
|
|
while(uNew*vNew*wNew > 32000) {
|
|
|
|
|
if( uNew>=vNew && uNew>=wNew) uNew--;
|
|
|
|
|
else if( vNew>=uNew && vNew>=wNew) vNew--;
|
|
|
|
|
else wNew--;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vertexCos = MEM_mallocN(sizeof(*vertexCos)*uNew*vNew*wNew, "tmp_vcos");
|
|
|
|
|
|
|
|
|
|
calc_lat_fudu(lt->flag, uNew, &fu, &du);
|
|
|
|
|
calc_lat_fudu(lt->flag, vNew, &fv, &dv);
|
|
|
|
|
calc_lat_fudu(lt->flag, wNew, &fw, &dw);
|
|
|
|
|
|
|
|
|
|
/* If old size is different then resolution changed in interface,
|
|
|
|
|
* try to do clever reinit of points. Pretty simply idea, we just
|
|
|
|
|
* deform new verts by old lattice, but scaling them to match old
|
|
|
|
|
* size first.
|
|
|
|
|
*/
|
|
|
|
|
if (ltOb) {
|
|
|
|
|
if (uNew!=1 && lt->pntsu!=1) {
|
|
|
|
|
fu = lt->fu;
|
|
|
|
|
du = (lt->pntsu-1)*lt->du/(uNew-1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (vNew!=1 && lt->pntsv!=1) {
|
|
|
|
|
fv = lt->fv;
|
|
|
|
|
dv = (lt->pntsv-1)*lt->dv/(vNew-1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (wNew!=1 && lt->pntsw!=1) {
|
|
|
|
|
fw = lt->fw;
|
|
|
|
|
dw = (lt->pntsw-1)*lt->dw/(wNew-1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
co = vertexCos[0];
|
|
|
|
|
for(w=0,wc=fw; w<wNew; w++,wc+=dw) {
|
|
|
|
|
for(v=0,vc=fv; v<vNew; v++,vc+=dv) {
|
|
|
|
|
for(u=0,uc=fu; u<uNew; u++,co+=3,uc+=du) {
|
|
|
|
|
co[0] = uc;
|
|
|
|
|
co[1] = vc;
|
|
|
|
|
co[2] = wc;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
if (ltOb) {
|
|
|
|
|
float mat[4][4];
|
|
|
|
|
int typeu = lt->typeu, typev = lt->typev, typew = lt->typew;
|
|
|
|
|
|
|
|
|
|
/* works best if we force to linear type (endpoints match) */
|
|
|
|
|
lt->typeu = lt->typev = lt->typew = KEY_LINEAR;
|
|
|
|
|
|
|
|
|
|
/* prevent using deformed locations */
|
|
|
|
|
freedisplist(<Ob->disp);
|
|
|
|
|
|
|
|
|
|
Mat4CpyMat4(mat, ltOb->obmat);
|
|
|
|
|
Mat4One(ltOb->obmat);
|
|
|
|
|
lattice_deform_verts(ltOb, NULL, vertexCos, uNew*vNew*wNew);
|
|
|
|
|
Mat4CpyMat4(ltOb->obmat, mat);
|
|
|
|
|
|
|
|
|
|
lt->typeu = typeu;
|
|
|
|
|
lt->typev = typev;
|
|
|
|
|
lt->typew = typew;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
lt->fu = fu;
|
|
|
|
|
lt->fv = fv;
|
|
|
|
|
lt->fw = fw;
|
|
|
|
|
lt->du = du;
|
|
|
|
|
lt->dv = dv;
|
|
|
|
|
lt->dw = dw;
|
|
|
|
|
|
|
|
|
|
lt->pntsu = uNew;
|
|
|
|
|
lt->pntsv = vNew;
|
|
|
|
|
lt->pntsw = wNew;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
MEM_freeN(lt->def);
|
|
|
|
|
lt->def= MEM_callocN(lt->pntsu*lt->pntsv*lt->pntsw*sizeof(BPoint), "lattice bp");
|
|
|
|
|
|
|
|
|
|
bp= lt->def;
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
for (i=0; i<lt->pntsu*lt->pntsv*lt->pntsw; i++,bp++) {
|
|
|
|
|
VECCOPY(bp->vec, vertexCos[i]);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2005-08-15 10:30:53 +00:00
|
|
|
|
|
|
|
|
MEM_freeN(vertexCos);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Lattice *add_lattice()
|
|
|
|
|
{
|
|
|
|
|
Lattice *lt;
|
|
|
|
|
|
|
|
|
|
lt= alloc_libblock(&G.main->latt, ID_LT, "Lattice");
|
|
|
|
|
|
|
|
|
|
lt->flag= LT_GRID;
|
|
|
|
|
|
|
|
|
|
lt->typeu= lt->typev= lt->typew= KEY_BSPLINE;
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
lt->def= MEM_callocN(sizeof(BPoint), "lattvert"); /* temporary */
|
|
|
|
|
resizelattice(lt, 2, 2, 2, NULL); /* creates a uniform lattice */
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
return lt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Lattice *copy_lattice(Lattice *lt)
|
|
|
|
|
{
|
|
|
|
|
Lattice *ltn;
|
|
|
|
|
|
|
|
|
|
ltn= copy_libblock(lt);
|
|
|
|
|
ltn->def= MEM_dupallocN(lt->def);
|
|
|
|
|
|
|
|
|
|
id_us_plus((ID *)ltn->ipo);
|
|
|
|
|
|
|
|
|
|
ltn->key= copy_key(ltn->key);
|
|
|
|
|
if(ltn->key) ltn->key->from= (ID *)ltn;
|
|
|
|
|
|
|
|
|
|
return ltn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void free_lattice(Lattice *lt)
|
|
|
|
|
{
|
|
|
|
|
if(lt->def) MEM_freeN(lt->def);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void make_local_lattice(Lattice *lt)
|
|
|
|
|
{
|
|
|
|
|
Object *ob;
|
|
|
|
|
Lattice *ltn;
|
|
|
|
|
int local=0, lib=0;
|
2003-04-26 11:56:44 +00:00
|
|
|
|
|
|
|
|
/* - only lib users: do nothing
|
|
|
|
|
* - only local users: set flag
|
|
|
|
|
* - mixed: make copy
|
2002-10-12 11:37:38 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if(lt->id.lib==0) return;
|
|
|
|
|
if(lt->id.us==1) {
|
|
|
|
|
lt->id.lib= 0;
|
|
|
|
|
lt->id.flag= LIB_LOCAL;
|
|
|
|
|
new_id(0, (ID *)lt, 0);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ob= G.main->object.first;
|
|
|
|
|
while(ob) {
|
|
|
|
|
if(ob->data==lt) {
|
|
|
|
|
if(ob->id.lib) lib= 1;
|
|
|
|
|
else local= 1;
|
|
|
|
|
}
|
|
|
|
|
ob= ob->id.next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(local && lib==0) {
|
|
|
|
|
lt->id.lib= 0;
|
|
|
|
|
lt->id.flag= LIB_LOCAL;
|
|
|
|
|
new_id(0, (ID *)lt, 0);
|
|
|
|
|
}
|
|
|
|
|
else if(local && lib) {
|
|
|
|
|
ltn= copy_lattice(lt);
|
|
|
|
|
ltn->id.us= 0;
|
|
|
|
|
|
|
|
|
|
ob= G.main->object.first;
|
|
|
|
|
while(ob) {
|
|
|
|
|
if(ob->data==lt) {
|
|
|
|
|
|
|
|
|
|
if(ob->id.lib==0) {
|
|
|
|
|
ob->data= ltn;
|
|
|
|
|
ltn->id.us++;
|
|
|
|
|
lt->id.us--;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ob= ob->id.next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void init_latt_deform(Object *oblatt, Object *ob)
|
|
|
|
|
{
|
2005-08-15 10:30:53 +00:00
|
|
|
/* we make an array with all differences */
|
|
|
|
|
Lattice *lt = deformLatt = (oblatt==G.obedit)?editLatt:oblatt->data;
|
|
|
|
|
BPoint *bp = lt->def;
|
|
|
|
|
DispList *dl = find_displist(&oblatt->disp, DL_VERTS);
|
|
|
|
|
float *co = dl?dl->verts:NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
float *fp, imat[4][4];
|
2005-08-15 10:30:53 +00:00
|
|
|
float fu, fv, fw;
|
2002-10-12 11:37:38 +00:00
|
|
|
int u, v, w;
|
2005-08-15 10:30:53 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
fp= latticedata= MEM_mallocN(sizeof(float)*3*deformLatt->pntsu*deformLatt->pntsv*deformLatt->pntsw, "latticedata");
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
/* for example with a particle system: ob==0 */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(ob==0) {
|
2003-04-26 11:56:44 +00:00
|
|
|
/* in deformspace, calc matrix */
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4Invert(latmat, oblatt->obmat);
|
|
|
|
|
|
2003-04-26 11:56:44 +00:00
|
|
|
/* back: put in deform array */
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4Invert(imat, latmat);
|
|
|
|
|
}
|
|
|
|
|
else {
|
2003-04-26 11:56:44 +00:00
|
|
|
/* in deformspace, calc matrix */
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4Invert(imat, oblatt->obmat);
|
|
|
|
|
Mat4MulMat4(latmat, ob->obmat, imat);
|
|
|
|
|
|
2003-04-26 11:56:44 +00:00
|
|
|
/* back: put in deform array */
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4Invert(imat, latmat);
|
|
|
|
|
}
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
for(w=0,fw=lt->fw; w<lt->pntsw; w++,fw+=lt->dw) {
|
|
|
|
|
for(v=0,fv=lt->fv; v<lt->pntsv; v++, fv+=lt->dv) {
|
|
|
|
|
for(u=0,fu=lt->fu; u<lt->pntsu; u++, bp++, co+=3, fp+=3, fu+=lt->du) {
|
|
|
|
|
if (dl) {
|
|
|
|
|
fp[0] = co[0] - fu;
|
|
|
|
|
fp[1] = co[1] - fv;
|
|
|
|
|
fp[2] = co[2] - fw;
|
|
|
|
|
} else {
|
|
|
|
|
fp[0] = bp->vec[0] - fu;
|
|
|
|
|
fp[1] = bp->vec[1] - fv;
|
|
|
|
|
fp[2] = bp->vec[2] - fw;
|
|
|
|
|
}
|
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4Mul3Vecfl(imat, fp);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void calc_latt_deform(float *co)
|
|
|
|
|
{
|
|
|
|
|
Lattice *lt;
|
2005-08-15 10:30:53 +00:00
|
|
|
float u, v, w, tu[4], tv[4], tw[4];
|
2002-10-12 11:37:38 +00:00
|
|
|
float *fpw, *fpv, *fpu, vec[3];
|
2003-02-13 16:56:42 +00:00
|
|
|
int ui, vi, wi, uu, vv, ww;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
if(latticedata==0) return;
|
|
|
|
|
|
2003-04-26 11:56:44 +00:00
|
|
|
lt= deformLatt; /* just for shorter notation! */
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2003-04-26 11:56:44 +00:00
|
|
|
/* co is in local coords, treat with latmat */
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
VECCOPY(vec, co);
|
|
|
|
|
Mat4MulVecfl(latmat, vec);
|
|
|
|
|
|
|
|
|
|
/* u v w coords */
|
|
|
|
|
|
|
|
|
|
if(lt->pntsu>1) {
|
2005-08-15 10:30:53 +00:00
|
|
|
u= (vec[0]-lt->fu)/lt->du;
|
2002-10-12 11:37:38 +00:00
|
|
|
ui= (int)floor(u);
|
|
|
|
|
u -= ui;
|
|
|
|
|
set_four_ipo(u, tu, lt->typeu);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
tu[0]= tu[2]= tu[3]= 0.0; tu[1]= 1.0;
|
|
|
|
|
ui= 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(lt->pntsv>1) {
|
2005-08-15 10:30:53 +00:00
|
|
|
v= (vec[1]-lt->fv)/lt->dv;
|
2002-10-12 11:37:38 +00:00
|
|
|
vi= (int)floor(v);
|
|
|
|
|
v -= vi;
|
|
|
|
|
set_four_ipo(v, tv, lt->typev);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
tv[0]= tv[2]= tv[3]= 0.0; tv[1]= 1.0;
|
|
|
|
|
vi= 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(lt->pntsw>1) {
|
2005-08-15 10:30:53 +00:00
|
|
|
w= (vec[2]-lt->fw)/lt->dw;
|
2002-10-12 11:37:38 +00:00
|
|
|
wi= (int)floor(w);
|
|
|
|
|
w -= wi;
|
|
|
|
|
set_four_ipo(w, tw, lt->typew);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
tw[0]= tw[2]= tw[3]= 0.0; tw[1]= 1.0;
|
|
|
|
|
wi= 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-13 16:56:42 +00:00
|
|
|
for(ww= wi-1; ww<=wi+2; ww++) {
|
2002-10-12 11:37:38 +00:00
|
|
|
w= tw[ww-wi+1];
|
|
|
|
|
|
|
|
|
|
if(w!=0.0) {
|
|
|
|
|
if(ww>0) {
|
|
|
|
|
if(ww<lt->pntsw) fpw= latticedata + 3*ww*lt->pntsu*lt->pntsv;
|
|
|
|
|
else fpw= latticedata + 3*(lt->pntsw-1)*lt->pntsu*lt->pntsv;
|
|
|
|
|
}
|
|
|
|
|
else fpw= latticedata;
|
|
|
|
|
|
2003-02-13 16:56:42 +00:00
|
|
|
for(vv= vi-1; vv<=vi+2; vv++) {
|
2002-10-12 11:37:38 +00:00
|
|
|
v= w*tv[vv-vi+1];
|
|
|
|
|
|
|
|
|
|
if(v!=0.0) {
|
|
|
|
|
if(vv>0) {
|
|
|
|
|
if(vv<lt->pntsv) fpv= fpw + 3*vv*lt->pntsu;
|
|
|
|
|
else fpv= fpw + 3*(lt->pntsv-1)*lt->pntsu;
|
|
|
|
|
}
|
|
|
|
|
else fpv= fpw;
|
|
|
|
|
|
|
|
|
|
for(uu= ui-1; uu<=ui+2; uu++) {
|
|
|
|
|
u= v*tu[uu-ui+1];
|
|
|
|
|
|
|
|
|
|
if(u!=0.0) {
|
|
|
|
|
if(uu>0) {
|
|
|
|
|
if(uu<lt->pntsu) fpu= fpv + 3*uu;
|
|
|
|
|
else fpu= fpv + 3*(lt->pntsu-1);
|
|
|
|
|
}
|
|
|
|
|
else fpu= fpv;
|
|
|
|
|
|
|
|
|
|
co[0]+= u*fpu[0];
|
|
|
|
|
co[1]+= u*fpu[1];
|
|
|
|
|
co[2]+= u*fpu[2];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void end_latt_deform()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
MEM_freeN(latticedata);
|
|
|
|
|
latticedata= 0;
|
|
|
|
|
}
|
|
|
|
|
|
2004-09-14 19:03:11 +00:00
|
|
|
/* calculations is in local space of deformed object
|
|
|
|
|
so we store in latmat transform from path coord inside object
|
|
|
|
|
*/
|
|
|
|
|
typedef struct {
|
|
|
|
|
float dmin[3], dmax[3], dsize, dloc[3];
|
|
|
|
|
float curvespace[4][4], objectspace[4][4];
|
|
|
|
|
} CurveDeform;
|
|
|
|
|
|
|
|
|
|
static void init_curve_deform(Object *par, Object *ob, CurveDeform *cd)
|
|
|
|
|
{
|
|
|
|
|
Mat4Invert(ob->imat, ob->obmat);
|
|
|
|
|
Mat4MulMat4(cd->objectspace, par->obmat, ob->imat);
|
|
|
|
|
Mat4Invert(cd->curvespace, cd->objectspace);
|
|
|
|
|
|
|
|
|
|
// offset vector for 'no smear'
|
|
|
|
|
Mat4Invert(par->imat, par->obmat);
|
|
|
|
|
VecMat4MulVecfl(cd->dloc, par->imat, ob->obmat[3]);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2004-11-21 10:42:42 +00:00
|
|
|
/* this makes sure we can extend for non-cyclic. *vec needs 4 items! */
|
2004-09-14 19:03:11 +00:00
|
|
|
static int where_on_path_deform(Object *ob, float ctime, float *vec, float *dir) /* returns OK */
|
|
|
|
|
{
|
|
|
|
|
Curve *cu= ob->data;
|
|
|
|
|
BevList *bl;
|
|
|
|
|
float ctime1;
|
|
|
|
|
int cycl=0;
|
|
|
|
|
|
|
|
|
|
/* test for cyclic */
|
|
|
|
|
bl= cu->bev.first;
|
|
|
|
|
if(bl && bl->poly> -1) cycl= 1;
|
|
|
|
|
|
|
|
|
|
if(cycl==0) {
|
|
|
|
|
ctime1= CLAMPIS(ctime, 0.0, 1.0);
|
|
|
|
|
}
|
|
|
|
|
else ctime1= ctime;
|
2004-11-21 10:42:42 +00:00
|
|
|
|
|
|
|
|
/* vec needs 4 items */
|
2004-09-14 19:03:11 +00:00
|
|
|
if(where_on_path(ob, ctime1, vec, dir)) {
|
|
|
|
|
|
|
|
|
|
if(cycl==0) {
|
|
|
|
|
Path *path= cu->path;
|
|
|
|
|
float dvec[3];
|
|
|
|
|
|
|
|
|
|
if(ctime < 0.0) {
|
|
|
|
|
VecSubf(dvec, path->data+4, path->data);
|
|
|
|
|
VecMulf(dvec, ctime*(float)path->len);
|
|
|
|
|
VECADD(vec, vec, dvec);
|
|
|
|
|
}
|
|
|
|
|
else if(ctime > 1.0) {
|
|
|
|
|
VecSubf(dvec, path->data+4*path->len-4, path->data+4*path->len-8);
|
|
|
|
|
VecMulf(dvec, (ctime-1.0)*(float)path->len);
|
|
|
|
|
VECADD(vec, vec, dvec);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* for each point, rotate & translate to curve */
|
|
|
|
|
/* use path, since it has constant distances */
|
|
|
|
|
/* co: local coord, result local too */
|
|
|
|
|
static void calc_curve_deform(Object *par, float *co, short axis, CurveDeform *cd)
|
|
|
|
|
{
|
|
|
|
|
Curve *cu= par->data;
|
2005-05-13 13:06:20 +00:00
|
|
|
float fac, loc[4], dir[3], *quat, q[4], mat[3][3], cent[3];
|
2004-09-14 19:03:11 +00:00
|
|
|
short upflag, index;
|
|
|
|
|
|
|
|
|
|
if(axis==OB_POSX || axis==OB_NEGX) {
|
|
|
|
|
upflag= OB_POSZ;
|
|
|
|
|
cent[0]= 0.0;
|
|
|
|
|
cent[1]= co[1];
|
|
|
|
|
cent[2]= co[2];
|
|
|
|
|
index= 0;
|
|
|
|
|
}
|
|
|
|
|
else if(axis==OB_POSY || axis==OB_NEGY) {
|
|
|
|
|
upflag= OB_POSZ;
|
|
|
|
|
cent[0]= co[0];
|
|
|
|
|
cent[1]= 0.0;
|
|
|
|
|
cent[2]= co[2];
|
|
|
|
|
index= 1;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
upflag= OB_POSY;
|
|
|
|
|
cent[0]= co[0];
|
|
|
|
|
cent[1]= co[1];
|
|
|
|
|
cent[2]= 0.0;
|
|
|
|
|
index= 2;
|
|
|
|
|
}
|
|
|
|
|
/* to be sure */
|
2004-09-27 10:39:18 +00:00
|
|
|
if(cu->path==NULL) {
|
|
|
|
|
calc_curvepath(par);
|
|
|
|
|
if(cu->path==NULL) return; // happens on append...
|
|
|
|
|
}
|
2004-09-14 19:03:11 +00:00
|
|
|
/* options */
|
|
|
|
|
if(cu->flag & CU_STRETCH)
|
|
|
|
|
fac= (co[index]-cd->dmin[index])/(cd->dmax[index] - cd->dmin[index]);
|
|
|
|
|
else
|
|
|
|
|
fac= (cd->dloc[index])/(cu->path->totdist) + (co[index]-cd->dmin[index])/(cu->path->totdist);
|
|
|
|
|
|
|
|
|
|
if( where_on_path_deform(par, fac, loc, dir)) { /* returns OK */
|
|
|
|
|
|
|
|
|
|
quat= vectoquat(dir, axis, upflag);
|
2005-05-13 13:06:20 +00:00
|
|
|
|
|
|
|
|
/* the tilt */
|
|
|
|
|
if(loc[3]!=0.0) {
|
|
|
|
|
Normalise(dir);
|
|
|
|
|
q[0]= (float)cos(0.5*loc[3]);
|
|
|
|
|
fac= (float)sin(0.5*loc[3]);
|
|
|
|
|
q[1]= -fac*dir[0];
|
|
|
|
|
q[2]= -fac*dir[1];
|
|
|
|
|
q[3]= -fac*dir[2];
|
|
|
|
|
QuatMul(quat, q, quat);
|
|
|
|
|
}
|
2004-09-14 19:03:11 +00:00
|
|
|
QuatToMat3(quat, mat);
|
|
|
|
|
|
|
|
|
|
/* local rotation */
|
|
|
|
|
Mat3MulVecfl(mat, cent);
|
|
|
|
|
|
|
|
|
|
/* translation */
|
|
|
|
|
VECADD(co, cent, loc);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
void curve_deform_verts(Object *cuOb, Object *target, float (*vertexCos)[3], int numVerts)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2005-07-19 20:14:17 +00:00
|
|
|
Curve *cu = cuOb->data;
|
|
|
|
|
int a, flag = cu->flag;
|
|
|
|
|
CurveDeform cd;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
cu->flag |= (CU_PATH|CU_FOLLOW); // needed for path & bevlist
|
|
|
|
|
|
|
|
|
|
init_curve_deform(cuOb, target, &cd);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
INIT_MINMAX(cd.dmin, cd.dmax);
|
|
|
|
|
|
|
|
|
|
for(a=0; a<numVerts; a++) {
|
|
|
|
|
Mat4MulVecfl(cd.curvespace, vertexCos[a]);
|
|
|
|
|
DO_MINMAX(vertexCos[a], cd.dmin, cd.dmax);
|
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
for(a=0; a<numVerts; a++) {
|
|
|
|
|
calc_curve_deform(cuOb, vertexCos[a], target->trackflag, &cd);
|
|
|
|
|
Mat4MulVecfl(cd.objectspace, vertexCos[a]);
|
|
|
|
|
}
|
2005-07-19 02:36:21 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
cu->flag = flag;
|
|
|
|
|
}
|
2005-07-19 02:36:21 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
void lattice_deform_verts(Object *laOb, Object *target, float (*vertexCos)[3], int numVerts)
|
|
|
|
|
{
|
|
|
|
|
int a;
|
2005-07-19 02:36:21 +00:00
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
init_latt_deform(laOb, target);
|
|
|
|
|
|
|
|
|
|
for(a=0; a<numVerts; a++) {
|
|
|
|
|
calc_latt_deform(vertexCos[a]);
|
2005-07-19 02:36:21 +00:00
|
|
|
}
|
|
|
|
|
|
2005-07-19 20:14:17 +00:00
|
|
|
end_latt_deform();
|
|
|
|
|
}
|
|
|
|
|
|
2005-08-14 06:08:41 +00:00
|
|
|
int object_deform_mball(Object *ob)
|
2005-07-19 02:36:21 +00:00
|
|
|
{
|
2005-08-14 06:08:41 +00:00
|
|
|
if(ob->parent && ob->parent->type==OB_LATTICE && ob->partype==PARSKEL) {
|
|
|
|
|
DispList *dl;
|
2005-07-19 02:36:21 +00:00
|
|
|
|
2005-08-14 06:08:41 +00:00
|
|
|
for (dl=ob->disp.first; dl; dl=dl->next) {
|
|
|
|
|
lattice_deform_verts(ob->parent, ob, (float(*)[3]) dl->verts, dl->nr);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2005-08-14 06:08:41 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
return 1;
|
2005-08-14 06:08:41 +00:00
|
|
|
} else {
|
|
|
|
|
return 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2005-08-14 06:08:41 +00:00
|
|
|
}
|
|
|
|
|
|
2005-08-15 10:30:53 +00:00
|
|
|
static BPoint *latt_bp(Lattice *lt, int u, int v, int w)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
return lt->def+ u + v*lt->pntsu + w*lt->pntsu*lt->pntsv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void outside_lattice(Lattice *lt)
|
|
|
|
|
{
|
|
|
|
|
BPoint *bp, *bp1, *bp2;
|
|
|
|
|
int u, v, w;
|
|
|
|
|
float fac1, du=0.0, dv=0.0, dw=0.0;
|
|
|
|
|
|
|
|
|
|
bp= lt->def;
|
|
|
|
|
|
|
|
|
|
if(lt->pntsu>1) du= 1.0f/((float)lt->pntsu-1);
|
|
|
|
|
if(lt->pntsv>1) dv= 1.0f/((float)lt->pntsv-1);
|
|
|
|
|
if(lt->pntsw>1) dw= 1.0f/((float)lt->pntsw-1);
|
|
|
|
|
|
|
|
|
|
for(w=0; w<lt->pntsw; w++) {
|
|
|
|
|
|
|
|
|
|
for(v=0; v<lt->pntsv; v++) {
|
|
|
|
|
|
|
|
|
|
for(u=0; u<lt->pntsu; u++, bp++) {
|
|
|
|
|
if(u==0 || v==0 || w==0 || u==lt->pntsu-1 || v==lt->pntsv-1 || w==lt->pntsw-1);
|
|
|
|
|
else {
|
|
|
|
|
|
|
|
|
|
bp->hide= 1;
|
|
|
|
|
bp->f1 &= ~SELECT;
|
|
|
|
|
|
|
|
|
|
/* u extrema */
|
|
|
|
|
bp1= latt_bp(lt, 0, v, w);
|
|
|
|
|
bp2= latt_bp(lt, lt->pntsu-1, v, w);
|
|
|
|
|
|
|
|
|
|
fac1= du*u;
|
|
|
|
|
bp->vec[0]= (1.0f-fac1)*bp1->vec[0] + fac1*bp2->vec[0];
|
|
|
|
|
bp->vec[1]= (1.0f-fac1)*bp1->vec[1] + fac1*bp2->vec[1];
|
|
|
|
|
bp->vec[2]= (1.0f-fac1)*bp1->vec[2] + fac1*bp2->vec[2];
|
|
|
|
|
|
|
|
|
|
/* v extrema */
|
|
|
|
|
bp1= latt_bp(lt, u, 0, w);
|
|
|
|
|
bp2= latt_bp(lt, u, lt->pntsv-1, w);
|
|
|
|
|
|
|
|
|
|
fac1= dv*v;
|
|
|
|
|
bp->vec[0]+= (1.0f-fac1)*bp1->vec[0] + fac1*bp2->vec[0];
|
|
|
|
|
bp->vec[1]+= (1.0f-fac1)*bp1->vec[1] + fac1*bp2->vec[1];
|
|
|
|
|
bp->vec[2]+= (1.0f-fac1)*bp1->vec[2] + fac1*bp2->vec[2];
|
|
|
|
|
|
|
|
|
|
/* w extrema */
|
|
|
|
|
bp1= latt_bp(lt, u, v, 0);
|
|
|
|
|
bp2= latt_bp(lt, u, v, lt->pntsw-1);
|
|
|
|
|
|
|
|
|
|
fac1= dw*w;
|
|
|
|
|
bp->vec[0]+= (1.0f-fac1)*bp1->vec[0] + fac1*bp2->vec[0];
|
|
|
|
|
bp->vec[1]+= (1.0f-fac1)*bp1->vec[1] + fac1*bp2->vec[1];
|
|
|
|
|
bp->vec[2]+= (1.0f-fac1)*bp1->vec[2] + fac1*bp2->vec[2];
|
|
|
|
|
|
|
|
|
|
VecMulf(bp->vec, 0.3333333f);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
2005-08-15 10:30:53 +00:00
|
|
|
|
|
|
|
|
float (*lattice_getVertexCos(struct Object *ob, int *numVerts_r))[3]
|
|
|
|
|
{
|
|
|
|
|
Lattice *lt = (G.obedit==ob)?editLatt:ob->data;
|
|
|
|
|
int i, numVerts = *numVerts_r = lt->pntsu*lt->pntsv*lt->pntsw;
|
|
|
|
|
float (*vertexCos)[3] = MEM_mallocN(sizeof(*vertexCos)*numVerts,"lt_vcos");
|
|
|
|
|
|
|
|
|
|
for (i=0; i<numVerts; i++) {
|
|
|
|
|
VECCOPY(vertexCos[i], lt->def[i].vec);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return vertexCos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void lattice_applyVertexCos(struct Object *ob, float (*vertexCos)[3])
|
|
|
|
|
{
|
|
|
|
|
Lattice *lt = ob->data;
|
|
|
|
|
int i, numVerts = lt->pntsu*lt->pntsv*lt->pntsw;
|
|
|
|
|
|
|
|
|
|
for (i=0; i<numVerts; i++) {
|
|
|
|
|
VECCOPY(lt->def[i].vec, vertexCos[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void lattice_calc_modifiers(Object *ob)
|
|
|
|
|
{
|
|
|
|
|
float (*vertexCos)[3] = NULL;
|
|
|
|
|
ModifierData *md = modifiers_getVirtualModifierList(ob);
|
|
|
|
|
int numVerts, editmode = G.obedit==ob;
|
|
|
|
|
|
|
|
|
|
freedisplist(&ob->disp);
|
|
|
|
|
|
|
|
|
|
if (!editmode) {
|
2005-09-26 15:34:21 +00:00
|
|
|
do_ob_key(ob);
|
2005-08-15 10:30:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (; md; md=md->next) {
|
|
|
|
|
ModifierTypeInfo *mti = modifierType_getInfo(md->type);
|
|
|
|
|
|
|
|
|
|
if (!(md->mode&(1<<0))) continue;
|
|
|
|
|
if (editmode && !(md->mode&eModifierMode_Editmode)) continue;
|
|
|
|
|
if (mti->isDisabled && mti->isDisabled(md)) continue;
|
|
|
|
|
if (mti->type!=eModifierTypeType_OnlyDeform) continue;
|
|
|
|
|
|
|
|
|
|
if (!vertexCos) vertexCos = lattice_getVertexCos(ob, &numVerts);
|
|
|
|
|
mti->deformVerts(md, ob, NULL, vertexCos, numVerts);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (vertexCos) {
|
|
|
|
|
DispList *dl = MEM_callocN(sizeof(*dl), "lt_dl");
|
|
|
|
|
dl->type = DL_VERTS;
|
|
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = numVerts;
|
|
|
|
|
dl->verts = (float*) vertexCos;
|
|
|
|
|
|
|
|
|
|
BLI_addtail(&ob->disp, dl);
|
|
|
|
|
}
|
|
|
|
|
}
|