2004-06-26 18:18:11 +00:00
|
|
|
/* effect.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 <math.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_listBase.h"
|
|
|
|
#include "DNA_effect_types.h"
|
|
|
|
#include "DNA_object_types.h"
|
2005-05-02 13:28:13 +00:00
|
|
|
#include "DNA_object_force.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "DNA_mesh_types.h"
|
2004-03-20 22:55:42 +00:00
|
|
|
#include "DNA_meshdata_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "DNA_material_types.h"
|
|
|
|
#include "DNA_curve_types.h"
|
|
|
|
#include "DNA_key_types.h"
|
|
|
|
#include "DNA_texture_types.h"
|
|
|
|
#include "DNA_scene_types.h"
|
|
|
|
#include "DNA_lattice_types.h"
|
2004-06-26 18:18:11 +00:00
|
|
|
#include "DNA_ipo_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
|
|
|
#include "BLI_arithb.h"
|
|
|
|
#include "BLI_rand.h"
|
|
|
|
|
2004-12-03 20:22:37 +00:00
|
|
|
#include "BKE_action.h"
|
2005-09-15 20:34:41 +00:00
|
|
|
#include "BKE_armature.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_bad_level_calls.h"
|
|
|
|
#include "BKE_blender.h"
|
2004-12-03 20:22:37 +00:00
|
|
|
#include "BKE_constraint.h"
|
|
|
|
#include "BKE_deform.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_displist.h"
|
2005-03-27 22:42:57 +00:00
|
|
|
#include "BKE_DerivedMesh.h"
|
2004-12-03 20:22:37 +00:00
|
|
|
#include "BKE_effect.h"
|
|
|
|
#include "BKE_global.h"
|
|
|
|
#include "BKE_ipo.h"
|
|
|
|
#include "BKE_key.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_lattice.h"
|
2004-06-26 18:18:11 +00:00
|
|
|
#include "BKE_mesh.h"
|
2004-12-03 20:22:37 +00:00
|
|
|
#include "BKE_material.h"
|
|
|
|
#include "BKE_main.h"
|
|
|
|
#include "BKE_object.h"
|
|
|
|
#include "BKE_screen.h"
|
|
|
|
#include "BKE_utildefines.h"
|
Phew, a lot of work, and no new features...
Main target was to make the inner rendering loop using no globals anymore.
This is essential for proper usage while raytracing, it caused a lot of
hacks in the raycode as well, which even didn't work correctly for all
situations (textures especially).
Done this by creating a new local struct RenderInput, which replaces usage
of the global struct Render R. The latter now only is used to denote
image size, viewmatrix, and the like.
Making the inner render loops using no globals caused 1000s of vars to
be changed... but the result definitely is much nicer code, which enables
making 'real' shaders in a next stage.
It also enabled me to remove the hacks from ray.c
Then i went to the task of removing redundant code. Especially the calculus
of texture coords took place (identical) in three locations.
Most obvious is the change in the unified render part, which is much less
code now; it uses the same rendering routines as normal render now.
(Note; not for halos yet!)
I also removed 6 files called 'shadowbuffer' something. This was experimen-
tal stuff from NaN days. And again saved a lot of double used code.
Finally I went over the blenkernel and blender/src calls to render stuff.
Here the same local data is used now, resulting in less dependency.
I also moved render-texture to the render module, this was still in Kernel.
(new file: texture.c)
So! After this commit I will check on the autofiles, to try to fix that.
MSVC people have to do it themselves.
This commit will need quite some testing help, but I'm around!
2003-12-21 21:52:51 +00:00
|
|
|
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
#include "render.h" // externtex, bad level call (ton)
|
|
|
|
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
Effect *add_effect(int type)
|
|
|
|
{
|
|
|
|
Effect *eff=0;
|
|
|
|
PartEff *paf;
|
|
|
|
int a;
|
|
|
|
|
|
|
|
switch(type) {
|
|
|
|
case EFF_PARTICLE:
|
|
|
|
paf= MEM_callocN(sizeof(PartEff), "neweff");
|
|
|
|
eff= (Effect *)paf;
|
|
|
|
|
|
|
|
paf->sta= 1.0;
|
|
|
|
paf->end= 100.0;
|
|
|
|
paf->lifetime= 50.0;
|
|
|
|
for(a=0; a<PAF_MAXMULT; a++) {
|
|
|
|
paf->life[a]= 50.0;
|
|
|
|
paf->child[a]= 4;
|
|
|
|
paf->mat[a]= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
paf->totpart= 1000;
|
|
|
|
paf->totkey= 8;
|
|
|
|
paf->staticstep= 5;
|
|
|
|
paf->defvec[2]= 1.0f;
|
|
|
|
paf->nabla= 0.05f;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
eff->type= eff->buttype= type;
|
|
|
|
eff->flag |= SELECT;
|
|
|
|
|
|
|
|
return eff;
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_effect(Effect *eff)
|
|
|
|
{
|
|
|
|
PartEff *paf;
|
|
|
|
|
|
|
|
if(eff->type==EFF_PARTICLE) {
|
|
|
|
paf= (PartEff *)eff;
|
|
|
|
if(paf->keys) MEM_freeN(paf->keys);
|
|
|
|
}
|
|
|
|
MEM_freeN(eff);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void free_effects(ListBase *lb)
|
|
|
|
{
|
|
|
|
Effect *eff;
|
|
|
|
|
|
|
|
eff= lb->first;
|
|
|
|
while(eff) {
|
|
|
|
BLI_remlink(lb, eff);
|
|
|
|
free_effect(eff);
|
|
|
|
eff= lb->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Effect *copy_effect(Effect *eff)
|
|
|
|
{
|
|
|
|
Effect *effn;
|
2004-10-14 22:20:42 +00:00
|
|
|
|
2004-10-14 22:39:13 +00:00
|
|
|
effn= MEM_dupallocN(eff);
|
2002-10-12 11:37:38 +00:00
|
|
|
if(effn->type==EFF_PARTICLE) ((PartEff *)effn)->keys= 0;
|
|
|
|
|
|
|
|
return effn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void copy_act_effect(Object *ob)
|
|
|
|
{
|
2003-04-25 15:48:11 +00:00
|
|
|
/* return a copy of the active effect */
|
2002-10-12 11:37:38 +00:00
|
|
|
Effect *effn, *eff;
|
|
|
|
|
|
|
|
eff= ob->effect.first;
|
|
|
|
while(eff) {
|
|
|
|
if(eff->flag & SELECT) {
|
|
|
|
|
|
|
|
effn= copy_effect(eff);
|
|
|
|
BLI_addtail(&ob->effect, effn);
|
|
|
|
|
|
|
|
eff->flag &= ~SELECT;
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
eff= eff->next;
|
|
|
|
}
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* when it comes here: add new effect */
|
2005-07-20 04:14:21 +00:00
|
|
|
eff= add_effect(EFF_PARTICLE);
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(&ob->effect, eff);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void copy_effects(ListBase *lbn, ListBase *lb)
|
|
|
|
{
|
|
|
|
Effect *eff, *effn;
|
|
|
|
|
|
|
|
lbn->first= lbn->last= 0;
|
|
|
|
|
|
|
|
eff= lb->first;
|
|
|
|
while(eff) {
|
|
|
|
effn= copy_effect(eff);
|
|
|
|
BLI_addtail(lbn, effn);
|
|
|
|
|
|
|
|
eff= eff->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void deselectall_eff(Object *ob)
|
|
|
|
{
|
|
|
|
Effect *eff= ob->effect.first;
|
|
|
|
|
|
|
|
while(eff) {
|
|
|
|
eff->flag &= ~SELECT;
|
|
|
|
eff= eff->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ***************** PARTICLES ***************** */
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
static Particle *new_particle(PartEff *paf)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
static Particle *pa;
|
|
|
|
static int cur;
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* we agree: when paf->keys==0: alloc */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(paf->keys==0) {
|
|
|
|
pa= paf->keys= MEM_callocN( paf->totkey*paf->totpart*sizeof(Particle), "particlekeys" );
|
|
|
|
cur= 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if(cur && cur<paf->totpart) pa+=paf->totkey;
|
|
|
|
cur++;
|
|
|
|
}
|
|
|
|
return pa;
|
|
|
|
}
|
|
|
|
|
|
|
|
PartEff *give_parteff(Object *ob)
|
|
|
|
{
|
|
|
|
PartEff *paf;
|
|
|
|
|
|
|
|
paf= ob->effect.first;
|
|
|
|
while(paf) {
|
|
|
|
if(paf->type==EFF_PARTICLE) return paf;
|
|
|
|
paf= paf->next;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void where_is_particle(PartEff *paf, Particle *pa, float ctime, float *vec)
|
|
|
|
{
|
|
|
|
Particle *p[4];
|
|
|
|
float dt, t[4];
|
|
|
|
int a;
|
|
|
|
|
|
|
|
if(paf->totkey==1) {
|
|
|
|
VECCOPY(vec, pa->co);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* first find the first particlekey */
|
2002-10-12 11:37:38 +00:00
|
|
|
a= (int)((paf->totkey-1)*(ctime-pa->time)/pa->lifetime);
|
|
|
|
if(a>=paf->totkey) a= paf->totkey-1;
|
2004-12-03 20:22:37 +00:00
|
|
|
else if(a<0) a= 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
pa+= a;
|
|
|
|
|
|
|
|
if(a>0) p[0]= pa-1; else p[0]= pa;
|
|
|
|
p[1]= pa;
|
|
|
|
|
|
|
|
if(a+1<paf->totkey) p[2]= pa+1; else p[2]= pa;
|
|
|
|
if(a+2<paf->totkey) p[3]= pa+2; else p[3]= p[2];
|
|
|
|
|
|
|
|
if(p[1]==p[2]) dt= 0.0;
|
|
|
|
else dt= (ctime-p[1]->time)/(p[2]->time - p[1]->time);
|
|
|
|
|
|
|
|
if(paf->flag & PAF_BSPLINE) set_four_ipo(dt, t, KEY_BSPLINE);
|
|
|
|
else set_four_ipo(dt, t, KEY_CARDINAL);
|
|
|
|
|
|
|
|
vec[0]= t[0]*p[0]->co[0] + t[1]*p[1]->co[0] + t[2]*p[2]->co[0] + t[3]*p[3]->co[0];
|
|
|
|
vec[1]= t[0]*p[0]->co[1] + t[1]*p[1]->co[1] + t[2]*p[2]->co[1] + t[3]*p[3]->co[1];
|
|
|
|
vec[2]= t[0]*p[0]->co[2] + t[1]*p[1]->co[2] + t[2]*p[2]->co[2] + t[3]*p[3]->co[2];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
static void particle_tex(MTex *mtex, PartEff *paf, float *co, float *no)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
float tin, tr, tg, tb, ta;
|
2002-10-12 11:37:38 +00:00
|
|
|
float old;
|
|
|
|
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
externtex(mtex, co, &tin, &tr, &tg, &tb, &ta);
|
2004-07-11 13:17:03 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
if(paf->texmap==PAF_TEXINT) {
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
tin*= paf->texfac;
|
|
|
|
no[0]+= tin*paf->defvec[0];
|
|
|
|
no[1]+= tin*paf->defvec[1];
|
|
|
|
no[2]+= tin*paf->defvec[2];
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
else if(paf->texmap==PAF_TEXRGB) {
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
no[0]+= (tr-0.5f)*paf->texfac;
|
|
|
|
no[1]+= (tg-0.5f)*paf->texfac;
|
|
|
|
no[2]+= (tb-0.5f)*paf->texfac;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
else { /* PAF_TEXGRAD */
|
|
|
|
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
old= tin;
|
2002-10-12 11:37:38 +00:00
|
|
|
co[0]+= paf->nabla;
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
externtex(mtex, co, &tin, &tr, &tg, &tb, &ta);
|
|
|
|
no[0]+= (old-tin)*paf->texfac;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
co[0]-= paf->nabla;
|
|
|
|
co[1]+= paf->nabla;
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
externtex(mtex, co, &tin, &tr, &tg, &tb, &ta);
|
|
|
|
no[1]+= (old-tin)*paf->texfac;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
co[1]-= paf->nabla;
|
|
|
|
co[2]+= paf->nabla;
|
Biiig commit! Thanks to 2-3 weeks of cvs freeze...
Render:
- New; support for dual CPU render (SDL thread)
Currently only works with alternating scanlines, but gives excellent
performance. For both normal render as unified implemented.
Note the "mutex" locks on z-transp buffer render and imbuf loads.
- This has been made possible by major cleanups in render code, especially
getting rid of globals (example Tin Tr Tg Tb Ta for textures) or struct
OSA or using Materials or Texture data to write to.
- Made normal render fully 4x32 floats too, and removed all old optimizes
with chars or shorts.
- Made normal render and unified render use same code for sky and halo
render, giving equal (and better) results for halo render. Old render
now also uses PostProcess options (brightness, mul, gamma)
- Added option ("FBuf") in F10 Output Panel, this keeps a 4x32 bits buffer
after render. Using PostProcess menu you will note an immediate re-
display of image too (32 bits RGBA)
- Added "Hue" and "Saturation" sliders to PostProcess options
- Render module is still not having a "nice" API, but amount of dependencies
went down a lot. Next todo: remove abusive "previewrender" code.
The last main global in Render (struct Render) now can be re-used for fully
controlling a render, to allow multiple "instances" of render to open.
- Renderwindow now displays a smal bar on top with the stats, and keeps the
stats after render too. Including "spare" page support.
Not only easier visible that way, but also to remove the awkward code that
was drawing stats in the Info header (extreme slow on some ATIs too)
- Cleaned up blendef.h and BKE_utildefines.h, these two had overlapping
defines.
- I might have forgotten stuff... and will write a nice doc on the architecture!
2004-12-27 19:28:52 +00:00
|
|
|
externtex(mtex, co, &tin, &tr, &tg, &tb, &ta);
|
|
|
|
no[2]+= (old-tin)*paf->texfac;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2005-04-04 18:09:47 +00:00
|
|
|
/* -------- pdDoEffector() --------
|
|
|
|
generic force/speed system, now used for particles and softbodies
|
|
|
|
opco = global coord, as input
|
|
|
|
force = force accumulator
|
|
|
|
speed = speed accumulator
|
|
|
|
cur_time = in frames
|
|
|
|
par_layer = layer the caller is in
|
|
|
|
|
|
|
|
*/
|
2005-04-05 20:28:32 +00:00
|
|
|
void pdDoEffector(float *opco, float *force, float *speed, float cur_time, unsigned int par_layer,unsigned int flags)
|
2004-07-09 14:39:06 +00:00
|
|
|
{
|
2004-07-10 09:17:25 +00:00
|
|
|
/*
|
|
|
|
Modifies the force on a particle according to its
|
|
|
|
relation with the effector object
|
|
|
|
Different kind of effectors include:
|
|
|
|
Forcefields: Gravity-like attractor
|
|
|
|
(force power is related to the inverse of distance to the power of a falloff value)
|
|
|
|
Vortex fields: swirling effectors
|
|
|
|
(particles rotate around Z-axis of the object. otherwise, same relation as)
|
|
|
|
(Forcefields, but this is not done through a force/acceleration)
|
|
|
|
|
|
|
|
*/
|
2004-07-09 14:39:06 +00:00
|
|
|
Object *ob;
|
|
|
|
Base *base;
|
2005-04-04 18:09:47 +00:00
|
|
|
PartDeflect *pd;
|
2004-07-09 14:39:06 +00:00
|
|
|
float vect_to_vert[3];
|
|
|
|
float force_vec[3];
|
|
|
|
float f_force, distance;
|
2005-04-04 18:09:47 +00:00
|
|
|
float *obloc;
|
2004-07-09 14:39:06 +00:00
|
|
|
float force_val, ffall_val;
|
|
|
|
short cur_frame;
|
|
|
|
|
2004-07-10 09:17:25 +00:00
|
|
|
/* Cycle through objects, get total of (1/(gravity_strength * dist^gravity_power)) */
|
2005-04-04 18:09:47 +00:00
|
|
|
/* Check for min distance here? (yes would be cool to add that, ton) */
|
|
|
|
|
|
|
|
for(base = G.scene->base.first; base; base= base->next) {
|
|
|
|
if( (base->lay & par_layer) && base->object->pd) {
|
2004-07-09 14:39:06 +00:00
|
|
|
ob= base->object;
|
2005-04-04 18:09:47 +00:00
|
|
|
pd= ob->pd;
|
|
|
|
|
|
|
|
/* checking if to continue or not */
|
|
|
|
if(pd->forcefield==0) continue;
|
|
|
|
|
|
|
|
/* Get IPO force strength and fall off values here */
|
|
|
|
if (has_ipo_code(ob->ipo, OB_PD_FSTR))
|
|
|
|
force_val = IPO_GetFloatValue(ob->ipo, OB_PD_FSTR, cur_time);
|
|
|
|
else
|
|
|
|
force_val = pd->f_strength;
|
|
|
|
|
|
|
|
if (has_ipo_code(ob->ipo, OB_PD_FFALL))
|
|
|
|
ffall_val = IPO_GetFloatValue(ob->ipo, OB_PD_FFALL, cur_time);
|
|
|
|
else
|
|
|
|
ffall_val = pd->f_power;
|
|
|
|
|
|
|
|
|
|
|
|
/* Need to set r.cfra for paths (investigate, ton) (uses ob->ctime now, ton) */
|
|
|
|
if(ob->ctime!=cur_time) {
|
2004-07-09 14:39:06 +00:00
|
|
|
cur_frame = G.scene->r.cfra;
|
|
|
|
G.scene->r.cfra = (short)cur_time;
|
|
|
|
where_is_object_time(ob, cur_time);
|
|
|
|
G.scene->r.cfra = cur_frame;
|
2005-04-04 18:09:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* use center of object for distance calculus */
|
|
|
|
obloc= ob->obmat[3];
|
|
|
|
VECSUB(vect_to_vert, obloc, opco);
|
|
|
|
distance = VecLength(vect_to_vert);
|
|
|
|
|
|
|
|
if((pd->flag & PFIELD_USEMAX) && distance>pd->maxdist)
|
|
|
|
; /* don't do anything */
|
|
|
|
else if(pd->forcefield == PFIELD_WIND) {
|
|
|
|
VECCOPY(force_vec, ob->obmat[2]);
|
2004-07-09 14:39:06 +00:00
|
|
|
|
2005-04-04 18:09:47 +00:00
|
|
|
/* wind works harder perpendicular to normal, would be nice for softbody later (ton) */
|
2004-07-09 14:39:06 +00:00
|
|
|
|
2005-04-04 18:09:47 +00:00
|
|
|
/* Limit minimum distance to vertex so that */
|
|
|
|
/* the force is not too big */
|
|
|
|
if (distance < 0.001) distance = 0.001f;
|
|
|
|
f_force = (force_val)*(1/(1000 * (float)pow((double)distance, (double)ffall_val)));
|
2005-04-05 20:28:32 +00:00
|
|
|
if(flags &&PE_WIND_AS_SPEED){
|
2005-11-04 12:38:46 +00:00
|
|
|
speed[0] -= (force_vec[0] * f_force );
|
|
|
|
speed[1] -= (force_vec[1] * f_force );
|
|
|
|
speed[2] -= (force_vec[2] * f_force );
|
2005-04-05 20:28:32 +00:00
|
|
|
}
|
|
|
|
else{
|
2005-11-04 12:38:46 +00:00
|
|
|
force[0] += force_vec[0]*f_force;
|
|
|
|
force[1] += force_vec[1]*f_force;
|
|
|
|
force[2] += force_vec[2]*f_force;
|
2005-04-05 20:28:32 +00:00
|
|
|
}
|
2005-04-04 18:09:47 +00:00
|
|
|
}
|
|
|
|
else if(pd->forcefield == PFIELD_FORCE) {
|
|
|
|
|
|
|
|
/* only use center of object */
|
|
|
|
obloc= ob->obmat[3];
|
2004-07-09 14:39:06 +00:00
|
|
|
|
2004-07-10 09:17:25 +00:00
|
|
|
/* Now calculate the gravitational force */
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(vect_to_vert, obloc, opco);
|
2005-04-04 18:09:47 +00:00
|
|
|
distance = VecLength(vect_to_vert);
|
2004-07-09 14:39:06 +00:00
|
|
|
|
|
|
|
/* Limit minimum distance to vertex so that */
|
|
|
|
/* the force is not too big */
|
2004-09-07 01:39:44 +00:00
|
|
|
if (distance < 0.001) distance = 0.001f;
|
|
|
|
f_force = (force_val)*(1/(1000 * (float)pow((double)distance, (double)ffall_val)));
|
2004-07-10 09:17:25 +00:00
|
|
|
force[0] += (vect_to_vert[0] * f_force );
|
|
|
|
force[1] += (vect_to_vert[1] * f_force );
|
|
|
|
force[2] += (vect_to_vert[2] * f_force );
|
2004-07-09 14:39:06 +00:00
|
|
|
|
|
|
|
}
|
2005-04-04 18:09:47 +00:00
|
|
|
else if(pd->forcefield == PFIELD_VORTEX) {
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2005-04-04 18:09:47 +00:00
|
|
|
/* only use center of object */
|
|
|
|
obloc= ob->obmat[3];
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-07-10 09:17:25 +00:00
|
|
|
/* Now calculate the vortex force */
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(vect_to_vert, obloc, opco);
|
2005-04-04 18:09:47 +00:00
|
|
|
distance = VecLength(vect_to_vert);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-07-10 09:17:25 +00:00
|
|
|
Crossf(force_vec, ob->obmat[2], vect_to_vert);
|
|
|
|
Normalise(force_vec);
|
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
/* Limit minimum distance to vertex so that */
|
|
|
|
/* the force is not too big */
|
2004-09-07 01:39:44 +00:00
|
|
|
if (distance < 0.001) distance = 0.001f;
|
|
|
|
f_force = (force_val)*(1/(100 * (float)pow((double)distance, (double)ffall_val)));
|
2004-07-10 09:17:25 +00:00
|
|
|
speed[0] -= (force_vec[0] * f_force );
|
|
|
|
speed[1] -= (force_vec[1] * f_force );
|
|
|
|
speed[2] -= (force_vec[2] * f_force );
|
2004-06-26 18:18:11 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
static void cache_object_vertices(Object *ob)
|
|
|
|
{
|
|
|
|
Mesh *me;
|
|
|
|
MVert *mvert;
|
|
|
|
float *fp;
|
|
|
|
int a;
|
|
|
|
|
|
|
|
me= ob->data;
|
|
|
|
if(me->totvert==0) return;
|
|
|
|
|
|
|
|
fp= ob->sumohandle= MEM_mallocN(3*sizeof(float)*me->totvert, "cache particles");
|
|
|
|
mvert= me->mvert;
|
|
|
|
a= me->totvert;
|
|
|
|
while(a--) {
|
|
|
|
VECCOPY(fp, mvert->co);
|
|
|
|
Mat4MulVecfl(ob->obmat, fp);
|
|
|
|
mvert++;
|
|
|
|
fp+= 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
static int pdDoDeflection(RNG *rng, float opco[3], float npco[3], float opno[3],
|
2004-06-26 18:18:11 +00:00
|
|
|
float npno[3], float life, float force[3], int def_depth,
|
|
|
|
float cur_time, unsigned int par_layer, int *last_object,
|
|
|
|
int *last_face, int *same_face)
|
|
|
|
{
|
|
|
|
/* Particle deflection code */
|
|
|
|
/* The code is in two sections: the first part checks whether a particle has */
|
|
|
|
/* intersected a face of a deflector mesh, given its old and new co-ords, opco and npco */
|
|
|
|
/* and which face it hit first */
|
|
|
|
/* The second part calculates the new co-ordinates given that collision and updates */
|
|
|
|
/* the new co-ordinates accordingly */
|
|
|
|
Base *base;
|
|
|
|
Object *ob, *deflection_object = NULL;
|
|
|
|
Mesh *def_mesh;
|
|
|
|
MFace *mface, *deflection_face = NULL;
|
2004-07-11 13:17:03 +00:00
|
|
|
float *v1, *v2, *v3, *v4, *vcache=NULL;
|
2004-06-26 18:18:11 +00:00
|
|
|
float nv1[3], nv2[3], nv3[3], nv4[3], edge1[3], edge2[3];
|
|
|
|
float dv1[3], dv2[3], dv3[3];
|
|
|
|
float vect_to_int[3], refl_vel[3];
|
|
|
|
float d_intersect_co[3], d_intersect_vect[3], d_nvect[3], d_i_co_above[3];
|
|
|
|
float forcec[3];
|
|
|
|
float k_point3, dist_to_plane;
|
|
|
|
float first_dist, ref_plane_mag;
|
|
|
|
float dk_plane=0, dk_point1=0;
|
|
|
|
float icalctop, icalcbot, n_mag;
|
|
|
|
float mag_iv, x_m,y_m,z_m;
|
|
|
|
float damping, perm_thresh;
|
|
|
|
float perm_val, rdamp_val;
|
|
|
|
int a, deflected=0, deflected_now=0;
|
2005-04-13 20:32:28 +00:00
|
|
|
float t,t2, min_t;
|
2004-06-26 18:18:11 +00:00
|
|
|
float mat[3][3], obloc[3];
|
|
|
|
short cur_frame;
|
|
|
|
float time_before, time_after;
|
|
|
|
float force_mag_norm;
|
|
|
|
int d_object=0, d_face=0, ds_object=0, ds_face=0;
|
|
|
|
|
|
|
|
first_dist = 200000;
|
|
|
|
min_t = 200000;
|
|
|
|
|
|
|
|
/* The first part of the code, finding the first intersected face*/
|
|
|
|
base= G.scene->base.first;
|
|
|
|
while (base) {
|
|
|
|
/*Only proceed for mesh object in same layer */
|
|
|
|
if(base->object->type==OB_MESH && (base->lay & par_layer)) {
|
|
|
|
ob= base->object;
|
|
|
|
/* only with deflecting set */
|
|
|
|
if(ob->pd && ob->pd->deflect) {
|
|
|
|
def_mesh= ob->data;
|
|
|
|
|
|
|
|
d_object = d_object + 1;
|
|
|
|
|
|
|
|
d_face = d_face + 1;
|
|
|
|
mface= def_mesh->mface;
|
|
|
|
a = def_mesh->totface;
|
2004-07-11 13:17:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
if(ob->parent==NULL && ob->ipo==NULL) { // static
|
|
|
|
if(ob->sumohandle==NULL) cache_object_vertices(ob);
|
|
|
|
vcache= ob->sumohandle;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/*Find out where the object is at this time*/
|
|
|
|
cur_frame = G.scene->r.cfra;
|
|
|
|
G.scene->r.cfra = (short)cur_time;
|
|
|
|
where_is_object_time(ob, cur_time);
|
|
|
|
G.scene->r.cfra = cur_frame;
|
|
|
|
|
|
|
|
/*Pass the values from ob->obmat to mat*/
|
|
|
|
/*and the location values to obloc */
|
|
|
|
Mat3CpyMat4(mat,ob->obmat);
|
|
|
|
obloc[0] = ob->obmat[3][0];
|
|
|
|
obloc[1] = ob->obmat[3][1];
|
|
|
|
obloc[2] = ob->obmat[3][2];
|
2005-04-18 21:51:45 +00:00
|
|
|
vcache= NULL;
|
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
}
|
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
while (a--) {
|
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
if(vcache) {
|
|
|
|
v1= vcache+ 3*(mface->v1);
|
|
|
|
VECCOPY(nv1, v1);
|
|
|
|
v1= vcache+ 3*(mface->v2);
|
|
|
|
VECCOPY(nv2, v1);
|
|
|
|
v1= vcache+ 3*(mface->v3);
|
|
|
|
VECCOPY(nv3, v1);
|
|
|
|
v1= vcache+ 3*(mface->v4);
|
|
|
|
VECCOPY(nv4, v1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Calculate the global co-ordinates of the vertices*/
|
|
|
|
v1= (def_mesh->mvert+(mface->v1))->co;
|
|
|
|
v2= (def_mesh->mvert+(mface->v2))->co;
|
|
|
|
v3= (def_mesh->mvert+(mface->v3))->co;
|
|
|
|
v4= (def_mesh->mvert+(mface->v4))->co;
|
|
|
|
|
|
|
|
VECCOPY(nv1, v1);
|
|
|
|
VECCOPY(nv2, v2);
|
|
|
|
VECCOPY(nv3, v3);
|
|
|
|
VECCOPY(nv4, v4);
|
|
|
|
|
|
|
|
/*Apply the objects deformation matrix*/
|
|
|
|
Mat3MulVecfl(mat, nv1);
|
|
|
|
Mat3MulVecfl(mat, nv2);
|
|
|
|
Mat3MulVecfl(mat, nv3);
|
|
|
|
Mat3MulVecfl(mat, nv4);
|
|
|
|
|
|
|
|
VECADD(nv1, nv1, obloc);
|
|
|
|
VECADD(nv2, nv2, obloc);
|
|
|
|
VECADD(nv3, nv3, obloc);
|
|
|
|
VECADD(nv4, nv4, obloc);
|
|
|
|
}
|
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
deflected_now = 0;
|
2005-04-18 21:51:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2005-04-13 20:32:28 +00:00
|
|
|
// t= 0.5; // this is labda of line, can use it optimize quad intersection
|
|
|
|
// sorry but no .. see below (BM)
|
2005-08-19 22:55:05 +00:00
|
|
|
if( LineIntersectsTriangle(opco, npco, nv1, nv2, nv3, &t) ) {
|
2004-07-11 13:17:03 +00:00
|
|
|
if (t < min_t) {
|
|
|
|
deflected = 1;
|
|
|
|
deflected_now = 1;
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
}
|
2005-04-13 20:32:28 +00:00
|
|
|
// else if (mface->v4 && (t>=0.0 && t<=1.0)) {
|
|
|
|
// no, you can't skip testing the other triangle
|
|
|
|
// it might give a smaller t on (close to) the edge .. this is numerics not esoteric maths :)
|
|
|
|
// note: the 2 triangles don't need to share a plane ! (BM)
|
|
|
|
if (mface->v4) {
|
2005-08-19 22:55:05 +00:00
|
|
|
if( LineIntersectsTriangle(opco, npco, nv1, nv3, nv4, &t2) ) {
|
2005-04-13 20:32:28 +00:00
|
|
|
if (t2 < min_t) {
|
2004-07-11 13:17:03 +00:00
|
|
|
deflected = 1;
|
|
|
|
deflected_now = 2;
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
}
|
|
|
|
}
|
2004-07-11 13:17:03 +00:00
|
|
|
|
2005-04-13 20:32:28 +00:00
|
|
|
if ((deflected_now > 0) && ((t < min_t) ||(t2 < min_t))) {
|
2004-06-26 18:18:11 +00:00
|
|
|
min_t = t;
|
|
|
|
ds_object = d_object;
|
|
|
|
ds_face = d_face;
|
|
|
|
deflection_object = ob;
|
|
|
|
deflection_face = mface;
|
|
|
|
if (deflected_now==1) {
|
2005-04-13 20:32:28 +00:00
|
|
|
min_t = t;
|
2004-06-26 18:18:11 +00:00
|
|
|
VECCOPY(dv1, nv1);
|
|
|
|
VECCOPY(dv2, nv2);
|
|
|
|
VECCOPY(dv3, nv3);
|
|
|
|
}
|
|
|
|
else {
|
2005-04-13 20:32:28 +00:00
|
|
|
min_t = t2;
|
2004-06-26 18:18:11 +00:00
|
|
|
VECCOPY(dv1, nv1);
|
|
|
|
VECCOPY(dv2, nv3);
|
|
|
|
VECCOPY(dv3, nv4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mface++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
base = base->next;
|
|
|
|
}
|
|
|
|
|
2005-04-12 21:36:21 +00:00
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
/* Here's the point to do the permeability calculation */
|
|
|
|
/* Set deflected to 0 if a random number is below the value */
|
|
|
|
/* Get the permeability IPO here*/
|
|
|
|
if (deflected) {
|
|
|
|
|
|
|
|
if (has_ipo_code(deflection_object->ipo, OB_PD_PERM))
|
|
|
|
perm_val = IPO_GetFloatValue(deflection_object->ipo, OB_PD_PERM, cur_time);
|
|
|
|
else
|
|
|
|
perm_val = deflection_object->pd->pdef_perm;
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
perm_thresh = rng_getFloat(rng) - perm_val;
|
2004-06-26 18:18:11 +00:00
|
|
|
if (perm_thresh < 0 ) {
|
|
|
|
deflected = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now for the second part of the deflection code - work out the new speed */
|
|
|
|
/* and position of the particle if a collision occurred */
|
|
|
|
if (deflected) {
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(edge1, dv1, dv2);
|
|
|
|
VECSUB(edge2, dv3, dv2);
|
2004-06-26 18:18:11 +00:00
|
|
|
Crossf(d_nvect, edge2, edge1);
|
|
|
|
n_mag = Normalise(d_nvect);
|
2004-07-11 13:17:03 +00:00
|
|
|
dk_plane = INPR(d_nvect, nv1);
|
|
|
|
dk_point1 = INPR(d_nvect,opco);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(d_intersect_vect, npco, opco);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
|
|
|
d_intersect_co[0] = opco[0] + (min_t * (npco[0] - opco[0]));
|
|
|
|
d_intersect_co[1] = opco[1] + (min_t * (npco[1] - opco[1]));
|
|
|
|
d_intersect_co[2] = opco[2] + (min_t * (npco[2] - opco[2]));
|
|
|
|
|
2004-09-07 01:39:44 +00:00
|
|
|
d_i_co_above[0] = (d_intersect_co[0] + (0.001f * d_nvect[0]));
|
|
|
|
d_i_co_above[1] = (d_intersect_co[1] + (0.001f * d_nvect[1]));
|
|
|
|
d_i_co_above[2] = (d_intersect_co[2] + (0.001f * d_nvect[2]));
|
2004-06-26 18:18:11 +00:00
|
|
|
mag_iv = Normalise(d_intersect_vect);
|
|
|
|
VECCOPY(npco, d_intersect_co);
|
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(vect_to_int, opco, d_intersect_co);
|
2004-06-26 18:18:11 +00:00
|
|
|
first_dist = Normalise(vect_to_int);
|
|
|
|
|
|
|
|
/* Work out the lengths of time before and after collision*/
|
|
|
|
time_before = (life*(first_dist / (mag_iv)));
|
|
|
|
time_after = (life*((mag_iv - first_dist) / (mag_iv)));
|
|
|
|
|
|
|
|
/* We have to recalculate what the speed would have been at the */
|
|
|
|
/* point of collision, not the key frame time */
|
|
|
|
npno[0]= opno[0] + time_before*force[0];
|
|
|
|
npno[1]= opno[1] + time_before*force[1];
|
|
|
|
npno[2]= opno[2] + time_before*force[2];
|
|
|
|
|
|
|
|
|
|
|
|
/* Reflect the speed vector in the face */
|
|
|
|
x_m = (2 * npno[0] * d_nvect[0]);
|
|
|
|
y_m = (2 * npno[1] * d_nvect[1]);
|
|
|
|
z_m = (2 * npno[2] * d_nvect[2]);
|
|
|
|
refl_vel[0] = npno[0] - (d_nvect[0] * (x_m + y_m + z_m));
|
|
|
|
refl_vel[1] = npno[1] - (d_nvect[1] * (x_m + y_m + z_m));
|
|
|
|
refl_vel[2] = npno[2] - (d_nvect[2] * (x_m + y_m + z_m));
|
|
|
|
|
|
|
|
/*A random variation in the damping factor........ */
|
|
|
|
/*Get the IPO values for damping here*/
|
|
|
|
|
|
|
|
if (has_ipo_code(deflection_object->ipo, OB_PD_SDAMP))
|
|
|
|
damping = IPO_GetFloatValue(deflection_object->ipo, OB_PD_SDAMP, cur_time);
|
|
|
|
else
|
|
|
|
damping = deflection_object->pd->pdef_damp;
|
|
|
|
|
|
|
|
if (has_ipo_code(deflection_object->ipo, OB_PD_RDAMP))
|
|
|
|
rdamp_val = IPO_GetFloatValue(deflection_object->ipo, OB_PD_RDAMP, cur_time);
|
|
|
|
else
|
|
|
|
rdamp_val = deflection_object->pd->pdef_rdamp;
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
damping = damping + ((1.0f - damping) * rng_getFloat(rng) *rdamp_val);
|
2004-06-26 18:18:11 +00:00
|
|
|
damping = damping * damping;
|
2004-07-11 13:17:03 +00:00
|
|
|
ref_plane_mag = INPR(refl_vel,d_nvect);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-09-07 01:39:44 +00:00
|
|
|
if (damping > 0.999) damping = 0.999f;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
|
|
|
/* Now add in the damping force - only damp in the direction of */
|
|
|
|
/* the faces normal vector */
|
|
|
|
npno[0] = (refl_vel[0] - (d_nvect[0] * ref_plane_mag * damping));
|
|
|
|
npno[1] = (refl_vel[1] - (d_nvect[1] * ref_plane_mag * damping));
|
|
|
|
npno[2] = (refl_vel[2] - (d_nvect[2] * ref_plane_mag * damping));
|
|
|
|
|
|
|
|
/* Now reset opno */
|
|
|
|
VECCOPY(opno,npno);
|
|
|
|
VECCOPY(forcec, force);
|
|
|
|
|
|
|
|
/* If the particle has bounced more than four times on the same */
|
|
|
|
/* face within this cycle (depth > 4, same face > 4 ) */
|
|
|
|
/* Then set the force to be only that component of the force */
|
|
|
|
/* in the same direction as the face normal */
|
|
|
|
/* i.e. subtract the component of the force in the direction */
|
|
|
|
/* of the face normal from the actual force */
|
|
|
|
if ((ds_object == *last_object) && (ds_face == *last_face)) {
|
|
|
|
/* Increment same_face */
|
|
|
|
*same_face = *same_face + 1;
|
|
|
|
if ((*same_face > 3) && (def_depth > 3)) {
|
2004-07-11 13:17:03 +00:00
|
|
|
force_mag_norm = INPR(forcec, d_nvect);
|
2004-06-26 18:18:11 +00:00
|
|
|
forcec[0] = forcec[0] - (d_nvect[0] * force_mag_norm);
|
|
|
|
forcec[1] = forcec[1] - (d_nvect[1] * force_mag_norm);
|
|
|
|
forcec[2] = forcec[2] - (d_nvect[2] * force_mag_norm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else *same_face = 1;
|
|
|
|
|
|
|
|
*last_object = ds_object;
|
|
|
|
*last_face = ds_face;
|
|
|
|
|
|
|
|
/* We have the particles speed at the point of collision */
|
|
|
|
/* Now we want the particles speed at the current key frame */
|
|
|
|
|
|
|
|
npno[0]= npno[0] + time_after*forcec[0];
|
|
|
|
npno[1]= npno[1] + time_after*forcec[1];
|
|
|
|
npno[2]= npno[2] + time_after*forcec[2];
|
|
|
|
|
|
|
|
/* Now we have to recalculate pa->co for the remainder*/
|
|
|
|
/* of the time since the intersect*/
|
|
|
|
npco[0]= npco[0] + time_after*npno[0];
|
|
|
|
npco[1]= npco[1] + time_after*npno[1];
|
|
|
|
npco[2]= npco[2] + time_after*npno[2];
|
|
|
|
|
|
|
|
/* And set the old co-ordinates back to the point just above the intersection */
|
|
|
|
VECCOPY(opco, d_i_co_above);
|
|
|
|
|
|
|
|
/* Finally update the time */
|
|
|
|
life = time_after;
|
|
|
|
cur_time += time_before;
|
|
|
|
|
|
|
|
/* The particle may have fallen through the face again by now!!*/
|
|
|
|
/* So check if the particle has changed sides of the plane compared*/
|
|
|
|
/* the co-ordinates at the last keyframe*/
|
|
|
|
/* But only do this as a last resort, if we've got to the end of the */
|
|
|
|
/* number of collisions allowed */
|
|
|
|
if (def_depth==9) {
|
2004-07-11 13:17:03 +00:00
|
|
|
k_point3 = INPR(d_nvect,npco);
|
2004-06-26 18:18:11 +00:00
|
|
|
if (((dk_plane > k_point3) && (dk_plane < dk_point1))||((dk_plane < k_point3) && (dk_plane > dk_point1))) {
|
|
|
|
|
|
|
|
/* Yup, the pesky particle may have fallen through a hole!!! */
|
|
|
|
/* So we'll cheat a bit and move the particle along the normal vector */
|
|
|
|
/* until it's just the other side of the plane */
|
|
|
|
icalctop = (dk_plane - d_nvect[0]*npco[0] - d_nvect[1]*npco[1] - d_nvect[2]*npco[2]);
|
|
|
|
icalcbot = (d_nvect[0]*d_nvect[0] + d_nvect[1]*d_nvect[1] + d_nvect[2]*d_nvect[2]);
|
|
|
|
dist_to_plane = icalctop / icalcbot;
|
|
|
|
|
|
|
|
/* Now just increase the distance a little to place */
|
|
|
|
/* the point the other side of the plane */
|
2004-09-07 01:39:44 +00:00
|
|
|
dist_to_plane *= 1.1f;
|
2004-06-26 18:18:11 +00:00
|
|
|
npco[0]= npco[0] + (dist_to_plane * d_nvect[0]);
|
|
|
|
npco[1]= npco[1] + (dist_to_plane * d_nvect[1]);
|
|
|
|
npco[2]= npco[2] + (dist_to_plane * d_nvect[2]);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return deflected;
|
|
|
|
}
|
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
/*
|
|
|
|
rng= random number generator
|
|
|
|
ob = object that spawns the particles
|
|
|
|
depth = for fireworks
|
|
|
|
nr = index nr of current particle
|
|
|
|
paf = the particle system
|
|
|
|
part = current particle
|
|
|
|
force = force vector
|
|
|
|
deform = flag to indicate lattice deform
|
|
|
|
cfraont = current frame
|
|
|
|
*/
|
|
|
|
static void make_particle_keys(RNG *rng, Object *ob, int depth, int nr, PartEff *paf, Particle *part, float *force, int deform, MTex *mtex, unsigned int par_layer, int cfraont)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
Particle *pa, *opa = NULL;
|
2004-06-26 18:18:11 +00:00
|
|
|
float damp, deltalife, life;
|
|
|
|
float cur_time;
|
2004-07-09 14:39:06 +00:00
|
|
|
float opco[3], opno[3], npco[3], npno[3], new_force[3], new_speed[3];
|
2004-06-26 18:18:11 +00:00
|
|
|
int b, rt1, rt2, deflected, deflection, finish_defs, def_count;
|
|
|
|
int last_ob, last_fc, same_fc;
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
damp= 1.0f-paf->damp;
|
|
|
|
pa= part;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* start speed: random */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(paf->randfac!=0.0) {
|
2005-07-25 21:53:58 +00:00
|
|
|
pa->no[0]+= paf->randfac*(rng_getFloat(rng) - 0.5f);
|
|
|
|
pa->no[1]+= paf->randfac*(rng_getFloat(rng) - 0.5f);
|
|
|
|
pa->no[2]+= paf->randfac*(rng_getFloat(rng) - 0.5f);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* start speed: texture */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(mtex && paf->texfac!=0.0) {
|
2004-07-03 14:18:21 +00:00
|
|
|
particle_tex(mtex, paf, pa->co, pa->no);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-03-23 16:01:32 +00:00
|
|
|
if(paf->totkey>1) deltalife= pa->lifetime/(paf->totkey-1);
|
|
|
|
else deltalife= pa->lifetime;
|
|
|
|
|
|
|
|
opa= pa;
|
|
|
|
pa++;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-03-23 16:01:32 +00:00
|
|
|
b= paf->totkey-1;
|
|
|
|
while(b--) {
|
2003-04-25 15:48:11 +00:00
|
|
|
/* new time */
|
2003-03-23 16:01:32 +00:00
|
|
|
pa->time= opa->time+deltalife;
|
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
/* set initial variables */
|
|
|
|
opco[0] = opa->co[0];
|
|
|
|
opco[1] = opa->co[1];
|
|
|
|
opco[2] = opa->co[2];
|
|
|
|
|
|
|
|
new_force[0] = force[0];
|
|
|
|
new_force[1] = force[1];
|
|
|
|
new_force[2] = force[2];
|
2004-07-09 14:39:06 +00:00
|
|
|
new_speed[0] = 0.0;
|
|
|
|
new_speed[1] = 0.0;
|
|
|
|
new_speed[2] = 0.0;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
/* handle differences between static (local coords, fixed frae) and dynamic */
|
|
|
|
if(paf->flag & PAF_STATIC) {
|
|
|
|
float opco1[3], new_force1[3], new_speed1[3];
|
|
|
|
|
|
|
|
/* move to global coords */
|
|
|
|
VECCOPY(opco1, opco);
|
|
|
|
Mat4MulVecfl(ob->obmat, opco1);
|
|
|
|
|
|
|
|
VECCOPY(new_force1, new_force);
|
|
|
|
Mat4Mul3Vecfl(ob->obmat, new_force1);
|
|
|
|
VECCOPY(new_speed1, new_speed);
|
|
|
|
Mat4Mul3Vecfl(ob->obmat, new_speed1);
|
|
|
|
|
|
|
|
cur_time = cfraont;
|
|
|
|
|
|
|
|
/* force fields */
|
|
|
|
pdDoEffector(opco1, new_force1, new_speed1, cur_time, par_layer, 0);
|
|
|
|
|
|
|
|
/* move back to local */
|
|
|
|
VECCOPY(opco, opco1);
|
|
|
|
Mat4MulVecfl(ob->imat, opco);
|
|
|
|
|
|
|
|
VECCOPY(new_force, new_force1);
|
|
|
|
Mat4Mul3Vecfl(ob->imat, new_force);
|
|
|
|
VECCOPY(new_speed, new_speed1);
|
|
|
|
Mat4Mul3Vecfl(ob->imat, new_speed);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cur_time = pa->time;
|
|
|
|
|
|
|
|
/* force fields */
|
|
|
|
pdDoEffector(opco, new_force, new_speed, cur_time, par_layer,0);
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* new location */
|
2004-09-07 01:39:44 +00:00
|
|
|
pa->co[0]= opa->co[0] + deltalife * (opa->no[0] + new_speed[0] + 0.5f*new_force[0]);
|
|
|
|
pa->co[1]= opa->co[1] + deltalife * (opa->no[1] + new_speed[1] + 0.5f*new_force[1]);
|
|
|
|
pa->co[2]= opa->co[2] + deltalife * (opa->no[2] + new_speed[2] + 0.5f*new_force[2]);
|
2003-03-23 16:01:32 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* new speed */
|
2004-06-26 18:18:11 +00:00
|
|
|
pa->no[0]= opa->no[0] + deltalife*new_force[0];
|
|
|
|
pa->no[1]= opa->no[1] + deltalife*new_force[1];
|
|
|
|
pa->no[2]= opa->no[2] + deltalife*new_force[2];
|
|
|
|
|
|
|
|
/* Particle deflection code */
|
|
|
|
deflection = 0;
|
|
|
|
finish_defs = 1;
|
|
|
|
def_count = 0;
|
|
|
|
|
|
|
|
VECCOPY(opno, opa->no);
|
|
|
|
VECCOPY(npco, pa->co);
|
|
|
|
VECCOPY(npno, pa->no);
|
|
|
|
|
|
|
|
life = deltalife;
|
|
|
|
cur_time -= deltalife;
|
|
|
|
|
|
|
|
last_ob = -1;
|
|
|
|
last_fc = -1;
|
|
|
|
same_fc = 0;
|
|
|
|
|
|
|
|
/* First call the particle deflection check for the particle moving */
|
|
|
|
/* between the old co-ordinates and the new co-ordinates */
|
|
|
|
/* If a deflection occurs, call the code again, this time between the */
|
|
|
|
/* intersection point and the updated new co-ordinates */
|
|
|
|
/* Bail out if we've done the calculation 10 times - this seems ok */
|
|
|
|
/* for most scenes I've tested */
|
|
|
|
while (finish_defs) {
|
2005-07-25 21:53:58 +00:00
|
|
|
deflected = pdDoDeflection(rng, opco, npco, opno, npno, life, new_force,
|
2004-06-26 18:18:11 +00:00
|
|
|
def_count, cur_time, par_layer,
|
|
|
|
&last_ob, &last_fc, &same_fc);
|
|
|
|
if (deflected) {
|
|
|
|
def_count = def_count + 1;
|
|
|
|
deflection = 1;
|
|
|
|
if (def_count==10) finish_defs = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
finish_defs = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only update the particle positions and speed if we had a deflection */
|
|
|
|
if (deflection) {
|
|
|
|
pa->co[0] = npco[0];
|
|
|
|
pa->co[1] = npco[1];
|
|
|
|
pa->co[2] = npco[2];
|
|
|
|
pa->no[0] = npno[0];
|
|
|
|
pa->no[1] = npno[1];
|
|
|
|
pa->no[2] = npno[2];
|
|
|
|
}
|
|
|
|
|
2003-03-23 16:01:32 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* speed: texture */
|
2003-03-23 16:01:32 +00:00
|
|
|
if(mtex && paf->texfac!=0.0) {
|
2004-07-11 13:17:03 +00:00
|
|
|
particle_tex(mtex, paf, pa->co, pa->no);
|
2003-03-23 16:01:32 +00:00
|
|
|
}
|
|
|
|
if(damp!=1.0) {
|
|
|
|
pa->no[0]*= damp;
|
|
|
|
pa->no[1]*= damp;
|
|
|
|
pa->no[2]*= damp;
|
|
|
|
}
|
|
|
|
|
2004-06-26 18:18:11 +00:00
|
|
|
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
opa= pa;
|
|
|
|
pa++;
|
2003-04-25 15:48:11 +00:00
|
|
|
/* opa is used later on too! */
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(deform) {
|
2003-04-25 15:48:11 +00:00
|
|
|
/* deform all keys */
|
2002-10-12 11:37:38 +00:00
|
|
|
pa= part;
|
|
|
|
b= paf->totkey;
|
|
|
|
while(b--) {
|
2005-10-20 18:52:29 +00:00
|
|
|
calc_latt_deform(pa->co, 1.0f);
|
2002-10-12 11:37:38 +00:00
|
|
|
pa++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-23 16:01:32 +00:00
|
|
|
/* the big multiplication */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(depth<PAF_MAXMULT && paf->mult[depth]!=0.0) {
|
|
|
|
|
2003-03-23 16:01:32 +00:00
|
|
|
/* new 'child' emerges from an average 'mult' part from
|
|
|
|
the particles */
|
2002-10-12 11:37:38 +00:00
|
|
|
damp = (float)nr;
|
|
|
|
rt1= (int)(damp*paf->mult[depth]);
|
|
|
|
rt2= (int)((damp+1.0)*paf->mult[depth]);
|
|
|
|
if(rt1!=rt2) {
|
|
|
|
|
|
|
|
for(b=0; b<paf->child[depth]; b++) {
|
|
|
|
pa= new_particle(paf);
|
|
|
|
*pa= *opa;
|
|
|
|
pa->lifetime= paf->life[depth];
|
|
|
|
if(paf->randlife!=0.0) {
|
2005-07-25 21:53:58 +00:00
|
|
|
pa->lifetime*= 1.0f + paf->randlife*(rng_getFloat(rng) - 0.5f);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
pa->mat_nr= paf->mat[depth];
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
make_particle_keys(rng, ob, depth+1, b, paf, pa, force, deform, mtex, par_layer, cfraont);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
static void init_mv_jit(float *jit, int num,int seed2)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2005-07-25 21:53:58 +00:00
|
|
|
RNG *rng;
|
2002-10-12 11:37:38 +00:00
|
|
|
float *jit2, x, rad1, rad2, rad3;
|
2002-10-30 00:37:19 +00:00
|
|
|
int i, num2;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
if(num==0) return;
|
|
|
|
|
|
|
|
rad1= (float)(1.0/sqrt((float)num));
|
|
|
|
rad2= (float)(1.0/((float)num));
|
|
|
|
rad3= (float)sqrt((float)num)/((float)num);
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
rng = rng_new(31415926 + num + seed2);
|
2002-10-12 11:37:38 +00:00
|
|
|
x= 0;
|
2002-10-30 00:37:19 +00:00
|
|
|
num2 = 2 * num;
|
|
|
|
for(i=0; i<num2; i+=2) {
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
jit[i]= x + rad1*(0.5f - rng_getFloat(rng));
|
|
|
|
jit[i+1]= i/(2.0f*num) + rad1*(0.5f - rng_getFloat(rng));
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
jit[i]-= (float)floor(jit[i]);
|
|
|
|
jit[i+1]-= (float)floor(jit[i+1]);
|
|
|
|
|
|
|
|
x+= rad3;
|
|
|
|
x -= (float)floor(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
jit2= MEM_mallocN(12 + 2*sizeof(float)*num, "initjit");
|
|
|
|
|
|
|
|
for (i=0 ; i<4 ; i++) {
|
|
|
|
RE_jitterate1(jit, jit2, num, rad1);
|
|
|
|
RE_jitterate1(jit, jit2, num, rad1);
|
|
|
|
RE_jitterate2(jit, jit2, num, rad2);
|
|
|
|
}
|
|
|
|
MEM_freeN(jit2);
|
2005-07-25 21:53:58 +00:00
|
|
|
rng_free(rng);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-27 22:42:57 +00:00
|
|
|
static void give_mesh_mvert(Mesh *me, DispListMesh *dlm, int nr, float *co, short *no, int seed2)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
static float *jit=0;
|
|
|
|
static int jitlevel=1;
|
2004-07-03 18:39:19 +00:00
|
|
|
MVert *mvert, *mvertbase=NULL;
|
|
|
|
MFace *mface, *mfacebase=NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
float u, v, *v1, *v2, *v3, *v4;
|
2004-07-03 18:39:19 +00:00
|
|
|
int totface=0, totvert=0, curface, curjit;
|
2002-10-12 11:37:38 +00:00
|
|
|
short *n1, *n2, *n3, *n4;
|
|
|
|
|
|
|
|
/* signal */
|
|
|
|
if(me==0) {
|
|
|
|
if(jit) MEM_freeN(jit);
|
|
|
|
jit= 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-03-27 22:42:57 +00:00
|
|
|
if(dlm) {
|
|
|
|
mvertbase= dlm->mvert;
|
|
|
|
mfacebase= dlm->mface;
|
|
|
|
totface= dlm->totface;
|
|
|
|
totvert= dlm->totvert;
|
2004-07-03 18:39:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(totvert==0) {
|
|
|
|
mvertbase= me->mvert;
|
|
|
|
mfacebase= me->mface;
|
|
|
|
totface= me->totface;
|
|
|
|
totvert= me->totvert;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(totface==0 || nr<totvert) {
|
|
|
|
mvert= mvertbase + (nr % totvert);
|
2002-10-12 11:37:38 +00:00
|
|
|
VECCOPY(co, mvert->co);
|
|
|
|
VECCOPY(no, mvert->no);
|
|
|
|
}
|
|
|
|
else {
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
nr-= totvert;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
if(jit==0) {
|
2004-07-03 18:39:19 +00:00
|
|
|
jitlevel= nr/totface;
|
2002-10-12 11:37:38 +00:00
|
|
|
if(jitlevel==0) jitlevel= 1;
|
|
|
|
if(jitlevel>100) jitlevel= 100;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
jit= MEM_callocN(2+ jitlevel*2*sizeof(float), "jit");
|
2004-01-22 15:45:42 +00:00
|
|
|
init_mv_jit(jit, jitlevel,seed2);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
curjit= nr/totface;
|
2002-10-12 11:37:38 +00:00
|
|
|
curjit= curjit % jitlevel;
|
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
curface= nr % totface;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
mface= mfacebase;
|
2002-10-12 11:37:38 +00:00
|
|
|
mface+= curface;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
v1= (mvertbase+(mface->v1))->co;
|
|
|
|
v2= (mvertbase+(mface->v2))->co;
|
|
|
|
n1= (mvertbase+(mface->v1))->no;
|
|
|
|
n2= (mvertbase+(mface->v2))->no;
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
v3= (mvertbase+(mface->v3))->co;
|
|
|
|
n3= (mvertbase+(mface->v3))->no;
|
|
|
|
if(mface->v4==0) {
|
2004-07-03 18:39:19 +00:00
|
|
|
v4= (mvertbase+(mface->v1))->co;
|
|
|
|
n4= (mvertbase+(mface->v1))->no;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
else {
|
2004-07-03 18:39:19 +00:00
|
|
|
v4= (mvertbase+(mface->v4))->co;
|
|
|
|
n4= (mvertbase+(mface->v4))->no;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u= jit[2*curjit];
|
|
|
|
v= jit[2*curjit+1];
|
|
|
|
|
|
|
|
co[0]= (float)((1.0-u)*(1.0-v)*v1[0] + (1.0-u)*(v)*v2[0] + (u)*(v)*v3[0] + (u)*(1.0-v)*v4[0]);
|
|
|
|
co[1]= (float)((1.0-u)*(1.0-v)*v1[1] + (1.0-u)*(v)*v2[1] + (u)*(v)*v3[1] + (u)*(1.0-v)*v4[1]);
|
|
|
|
co[2]= (float)((1.0-u)*(1.0-v)*v1[2] + (1.0-u)*(v)*v2[2] + (u)*(v)*v3[2] + (u)*(1.0-v)*v4[2]);
|
|
|
|
|
|
|
|
no[0]= (short)((1.0-u)*(1.0-v)*n1[0] + (1.0-u)*(v)*n2[0] + (u)*(v)*n3[0] + (u)*(1.0-v)*n4[0]);
|
|
|
|
no[1]= (short)((1.0-u)*(1.0-v)*n1[1] + (1.0-u)*(v)*n2[1] + (u)*(v)*n3[1] + (u)*(1.0-v)*n4[1]);
|
|
|
|
no[2]= (short)((1.0-u)*(1.0-v)*n1[2] + (1.0-u)*(v)*n2[2] + (u)*(v)*n3[2] + (u)*(1.0-v)*n4[2]);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void build_particle_system(Object *ob)
|
|
|
|
{
|
2005-07-25 21:53:58 +00:00
|
|
|
RNG *rng;
|
2004-07-11 13:17:03 +00:00
|
|
|
Base *base;
|
2002-10-12 11:37:38 +00:00
|
|
|
Object *par;
|
|
|
|
PartEff *paf;
|
|
|
|
Particle *pa;
|
|
|
|
Mesh *me;
|
|
|
|
MTex *mtexmove=0;
|
|
|
|
Material *ma;
|
2005-03-27 22:42:57 +00:00
|
|
|
DispListMesh *dlm;
|
2005-07-17 05:20:57 +00:00
|
|
|
int dmNeedsFree;
|
|
|
|
DerivedMesh *dm;
|
2002-10-12 11:37:38 +00:00
|
|
|
float framelenont, ftime, dtime, force[3], imat[3][3], vec[3];
|
|
|
|
float fac, prevobmat[4][4], sfraont, co[3];
|
2005-07-17 17:33:09 +00:00
|
|
|
int deform=0, a, cur, cfraont, cfralast, totpart, totvert;
|
2002-10-12 11:37:38 +00:00
|
|
|
short no[3];
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
if(ob->type!=OB_MESH) return;
|
|
|
|
me= ob->data;
|
|
|
|
if(me->totvert==0) return;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
ma= give_current_material(ob, 1);
|
|
|
|
if(ma) {
|
|
|
|
mtexmove= ma->mtex[7];
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
paf= give_parteff(ob);
|
2004-07-03 18:39:19 +00:00
|
|
|
if(paf==NULL) return;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
waitcursor(1);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
disable_speed_curve(1);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* generate all particles */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(paf->keys) MEM_freeN(paf->keys);
|
2004-07-03 18:39:19 +00:00
|
|
|
paf->keys= NULL;
|
2004-06-26 18:18:11 +00:00
|
|
|
new_particle(paf);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
/* reset deflector cache, sumohandle is free, but its still sorta abuse... (ton) */
|
|
|
|
for(base= G.scene->base.first; base; base= base->next) {
|
|
|
|
base->object->sumohandle= NULL;
|
|
|
|
}
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
cfraont= G.scene->r.cfra;
|
|
|
|
cfralast= -1000;
|
|
|
|
framelenont= G.scene->r.framelen;
|
|
|
|
G.scene->r.framelen= 1.0;
|
|
|
|
sfraont= ob->sf;
|
|
|
|
ob->sf= 0.0;
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* mult generations? */
|
2002-10-12 11:37:38 +00:00
|
|
|
totpart= paf->totpart;
|
|
|
|
for(a=0; a<PAF_MAXMULT; a++) {
|
|
|
|
if(paf->mult[a]!=0.0) {
|
2003-04-25 15:48:11 +00:00
|
|
|
/* interessant formula! this way after 'x' generations the total is paf->totpart */
|
2002-10-12 11:37:38 +00:00
|
|
|
totpart= (int)(totpart / (1.0+paf->mult[a]*paf->child[a]));
|
|
|
|
}
|
|
|
|
else break;
|
|
|
|
}
|
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
if (paf->flag & PAF_STATIC) {
|
|
|
|
ftime = cfraont;
|
|
|
|
dtime = 0;
|
|
|
|
} else {
|
|
|
|
ftime= paf->sta;
|
|
|
|
dtime= (paf->end - paf->sta)/totpart;
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2005-07-23 04:37:02 +00:00
|
|
|
/* this call returns NULL during editmode, just ignore it and
|
|
|
|
* particles should be recalc'd on exit.
|
|
|
|
*/
|
|
|
|
dm = mesh_get_derived_final(ob, &dmNeedsFree);
|
|
|
|
if (!dm) {
|
|
|
|
waitcursor(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* WARNING!!!! pushdata and popdata actually copy object memory!!!!
|
|
|
|
* Changes between these calls will be lost!!!
|
|
|
|
*/
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* remember full hierarchy */
|
2002-10-12 11:37:38 +00:00
|
|
|
par= ob;
|
|
|
|
while(par) {
|
|
|
|
pushdata(par, sizeof(Object));
|
|
|
|
par= par->parent;
|
|
|
|
}
|
2004-11-22 18:13:56 +00:00
|
|
|
|
|
|
|
/* for static particles, calculate system on current frame */
|
2004-12-02 14:52:40 +00:00
|
|
|
if(ma) do_mat_ipo(ma);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* set it all at first frame */
|
2002-10-12 11:37:38 +00:00
|
|
|
G.scene->r.cfra= cfralast= (int)floor(ftime);
|
|
|
|
par= ob;
|
|
|
|
while(par) {
|
|
|
|
/* do_ob_ipo(par); */
|
|
|
|
do_ob_key(par);
|
|
|
|
par= par->parent;
|
|
|
|
}
|
2004-11-22 18:13:56 +00:00
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
/* matrix stuff for static too */
|
|
|
|
Mat4Invert(ob->imat, ob->obmat);
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
if((paf->flag & PAF_STATIC)==0) {
|
2004-12-02 14:52:40 +00:00
|
|
|
if(ma) do_mat_ipo(ma); // nor for static
|
2004-11-22 18:13:56 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
where_is_object(ob);
|
|
|
|
Mat4CpyMat4(prevobmat, ob->obmat);
|
|
|
|
Mat3CpyMat4(imat, ob->imat);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Mat4One(prevobmat);
|
|
|
|
Mat3One(imat);
|
|
|
|
}
|
|
|
|
|
2005-07-25 21:53:58 +00:00
|
|
|
rng = rng_new(paf->seed);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* otherwise it goes way too fast */
|
2002-10-12 11:37:38 +00:00
|
|
|
force[0]= paf->force[0]*0.05f;
|
|
|
|
force[1]= paf->force[1]*0.05f;
|
|
|
|
force[2]= paf->force[2]*0.05f;
|
|
|
|
|
2004-07-03 18:39:19 +00:00
|
|
|
if( paf->flag & PAF_STATIC ) deform= 0;
|
|
|
|
else {
|
2005-08-14 10:35:58 +00:00
|
|
|
deform= (ob->parent && ob->parent->type==OB_LATTICE && ob->partype==PARSKEL);
|
2004-07-03 18:39:19 +00:00
|
|
|
if(deform) init_latt_deform(ob->parent, 0);
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
/* init */
|
2005-07-20 02:23:09 +00:00
|
|
|
|
2005-08-07 02:30:29 +00:00
|
|
|
dlm = dm->convertToDispListMesh(dm, 1);
|
2005-07-17 17:33:09 +00:00
|
|
|
totvert = dlm->totvert;
|
2005-03-27 22:42:57 +00:00
|
|
|
|
|
|
|
give_mesh_mvert(me, dlm, totpart, co, no, paf->seed);
|
2004-12-03 20:22:37 +00:00
|
|
|
|
|
|
|
if(G.f & G_DEBUG) {
|
|
|
|
printf("\n");
|
|
|
|
printf("Calculating particles......... \n");
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
for(a=0; a<totpart; a++, ftime+=dtime) {
|
|
|
|
|
|
|
|
pa= new_particle(paf);
|
|
|
|
pa->time= ftime;
|
|
|
|
|
2004-12-03 20:22:37 +00:00
|
|
|
if(G.f & G_DEBUG) {
|
|
|
|
int b, c;
|
|
|
|
|
|
|
|
c = totpart/100;
|
|
|
|
if (c==0){
|
|
|
|
c = 1;
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2004-12-03 20:22:37 +00:00
|
|
|
b=(a%c);
|
|
|
|
if (b==0) {
|
|
|
|
printf("\r Particle: %d / %d ", a, totpart);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
2004-06-26 18:18:11 +00:00
|
|
|
}
|
2003-04-25 15:48:11 +00:00
|
|
|
/* set ob at correct time */
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
if((paf->flag & PAF_STATIC)==0) {
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
cur= (int)floor(ftime) + 1 ; /* + 1 has a reason: (obmat/prevobmat) otherwise comet-tails start too late */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(cfralast != cur) {
|
|
|
|
G.scene->r.cfra= cfralast= cur;
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* added later: blur? */
|
2002-10-12 11:37:38 +00:00
|
|
|
bsystem_time(ob, ob->parent, (float)G.scene->r.cfra, 0.0);
|
2004-06-26 18:18:11 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
par= ob;
|
|
|
|
while(par) {
|
|
|
|
/* do_ob_ipo(par); */
|
|
|
|
par->ctime= -1234567.0;
|
|
|
|
do_ob_key(par);
|
2005-04-11 17:55:20 +00:00
|
|
|
if(par->type==OB_ARMATURE) {
|
2005-10-10 18:05:30 +00:00
|
|
|
do_all_pose_actions(par); // only does this object actions
|
2005-09-15 20:34:41 +00:00
|
|
|
where_is_pose(par);
|
2005-04-11 17:55:20 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
par= par->parent;
|
|
|
|
}
|
2004-12-02 14:52:40 +00:00
|
|
|
if(ma) do_mat_ipo(ma);
|
2002-10-12 11:37:38 +00:00
|
|
|
Mat4CpyMat4(prevobmat, ob->obmat);
|
|
|
|
where_is_object(ob);
|
|
|
|
Mat4Invert(ob->imat, ob->obmat);
|
|
|
|
Mat3CpyMat4(imat, ob->imat);
|
|
|
|
}
|
|
|
|
}
|
2003-04-25 15:48:11 +00:00
|
|
|
/* get coordinates */
|
2005-04-11 17:55:20 +00:00
|
|
|
if(paf->flag & PAF_FACE) give_mesh_mvert(me, dlm, a, co, no, paf->seed);
|
2002-10-12 11:37:38 +00:00
|
|
|
else {
|
2005-07-17 17:33:09 +00:00
|
|
|
if (totvert) {
|
|
|
|
VECCOPY(co, dlm->mvert[a%totvert].co);
|
|
|
|
VECCOPY(no, dlm->mvert[a%totvert].no);
|
|
|
|
} else {
|
|
|
|
co[0] = co[1] = co[2] = 0.0f;
|
|
|
|
no[0] = no[1] = no[2] = 0;
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VECCOPY(pa->co, co);
|
|
|
|
|
|
|
|
if(paf->flag & PAF_STATIC);
|
|
|
|
else {
|
|
|
|
Mat4MulVecfl(ob->obmat, pa->co);
|
|
|
|
|
|
|
|
VECCOPY(vec, co);
|
|
|
|
Mat4MulVecfl(prevobmat, vec);
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* first start speed: object */
|
2004-07-11 13:17:03 +00:00
|
|
|
VECSUB(pa->no, pa->co, vec);
|
2002-10-12 11:37:38 +00:00
|
|
|
VecMulf(pa->no, paf->obfac);
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* calculate the correct inter-frame */
|
2002-10-12 11:37:38 +00:00
|
|
|
fac= (ftime- (float)floor(ftime));
|
|
|
|
pa->co[0]= fac*pa->co[0] + (1.0f-fac)*vec[0];
|
|
|
|
pa->co[1]= fac*pa->co[1] + (1.0f-fac)*vec[1];
|
|
|
|
pa->co[2]= fac*pa->co[2] + (1.0f-fac)*vec[2];
|
|
|
|
}
|
2003-04-25 15:48:11 +00:00
|
|
|
|
|
|
|
/* start speed: normal */
|
2002-10-12 11:37:38 +00:00
|
|
|
if(paf->normfac!=0.0) {
|
|
|
|
/* transpose ! */
|
|
|
|
vec[0]= imat[0][0]*no[0] + imat[0][1]*no[1] + imat[0][2]*no[2];
|
|
|
|
vec[1]= imat[1][0]*no[0] + imat[1][1]*no[1] + imat[1][2]*no[2];
|
|
|
|
vec[2]= imat[2][0]*no[0] + imat[2][1]*no[1] + imat[2][2]*no[2];
|
|
|
|
|
|
|
|
Normalise(vec);
|
|
|
|
VecMulf(vec, paf->normfac);
|
2004-07-11 13:17:03 +00:00
|
|
|
VECADD(pa->no, pa->no, vec);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
pa->lifetime= paf->lifetime;
|
|
|
|
if(paf->randlife!=0.0) {
|
2005-07-25 21:53:58 +00:00
|
|
|
pa->lifetime*= 1.0f + paf->randlife*(rng_getFloat(rng) - 0.5f);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
pa->mat_nr= 1;
|
|
|
|
|
2005-11-04 12:38:46 +00:00
|
|
|
make_particle_keys(rng, ob, 0, a, paf, pa, force, deform, mtexmove, ob->lay, cfraont);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2004-12-03 20:22:37 +00:00
|
|
|
if(G.f & G_DEBUG) {
|
|
|
|
printf("\r Particle: %d / %d \n", totpart, totpart);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
if(deform) end_latt_deform();
|
|
|
|
|
|
|
|
/* restore */
|
|
|
|
G.scene->r.cfra= cfraont;
|
|
|
|
G.scene->r.framelen= framelenont;
|
2005-03-27 22:42:57 +00:00
|
|
|
give_mesh_mvert(0, 0, 0, 0, 0,paf->seed);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* put hierarchy back */
|
2002-10-12 11:37:38 +00:00
|
|
|
par= ob;
|
|
|
|
while(par) {
|
|
|
|
popfirst(par);
|
2003-04-25 15:48:11 +00:00
|
|
|
/* do not do ob->ipo: keep insertkey */
|
2002-10-12 11:37:38 +00:00
|
|
|
do_ob_key(par);
|
2005-04-11 17:55:20 +00:00
|
|
|
|
|
|
|
if(par->type==OB_ARMATURE) {
|
2005-10-10 18:05:30 +00:00
|
|
|
do_all_pose_actions(par); // only does this object actions
|
2005-09-15 20:34:41 +00:00
|
|
|
where_is_pose(par);
|
2005-04-11 17:55:20 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
par= par->parent;
|
|
|
|
}
|
|
|
|
|
2004-07-11 13:17:03 +00:00
|
|
|
/* reset deflector cache */
|
|
|
|
for(base= G.scene->base.first; base; base= base->next) {
|
|
|
|
if(base->object->sumohandle) {
|
|
|
|
MEM_freeN(base->object->sumohandle);
|
|
|
|
base->object->sumohandle= NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* restore: AFTER popfirst */
|
2002-10-12 11:37:38 +00:00
|
|
|
ob->sf= sfraont;
|
|
|
|
|
2004-12-02 14:52:40 +00:00
|
|
|
if(ma) do_mat_ipo(ma); // set back on current time
|
2002-10-12 11:37:38 +00:00
|
|
|
disable_speed_curve(0);
|
2004-12-03 20:22:37 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
waitcursor(0);
|
|
|
|
|
2005-07-17 05:20:57 +00:00
|
|
|
displistmesh_free(dlm);
|
|
|
|
if (dmNeedsFree) dm->release(dm);
|
2005-07-25 21:53:58 +00:00
|
|
|
|
|
|
|
rng_free(rng);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|