Scripting Custom Chunks - The Basics

The MAXScript Online Help provides an advanced example of using the PArray to generate debris then animated by PFlow.
The 3dsmax Online Tutorials also provide an example of a chunky explosion.

The simple tutorial provides the basics of setting up an explosion to chunks using a Scripted Birth Event.
It does not provide any information about the origin of the chunks (we will use simple boxes building a larger box),
but the technique applies to any situation where the chunks are already existing and require only animation through PFlow...

On a side note, a similar but much more complex setup was used by Frantic Films to animate the hundreds of frozen
Tar Monster chunks seen in the new Scooby Doo 2 movie in theatres since March 26th 2004.

Click here to download the MAX Scene (55K, 3dsmax 5.1 with PFlow, 3dsmax 6 or higher)

Natural Language - Setup
Particle View Setup

Script

on ChannelsUsed pCont do
(
    pCont.useAge = true
    pCont.useTM = true
    pCont.useShape = true
)

on Init pCont do
(
    global ChunksArray = $Box* as array
)

on Proceed pCont do
(
    t = pCont.getTimeStart() as float

    if t < 0 do
    (
        NumChunks = ChunksArray.count

        for i = 1 to NumChunks do
        (
            pCont.AddParticle()
            pCont.particleIndex = pCont.NumParticles()
            pCont.particleAge = 0
            pCont.particleTM = ChunksArray[i].transform
            pCont.particleShape = ChunksArray[i].mesh
        )
    )
)

on Release pCont do
(

)

 

Results


 

This is the array of boxes building a larger box

 

This is the array of boxes in "shade selected" mode AND the particles as wireframe on frame 10

 

These are the particles in shaded mode on frame 10.


They are moving away from the center of the explosion defined by the position of the PF icon.

 

Step-By-Step Comments

on ChannelsUsed pCont do
(

The ChannelsUsed handler defines the channels to be used by the Script Operator - you cannot get or set particle related values from the particle container without specifying which properties you need access to. This way, Particle Flow does not have to provide the Script Operator with all possible channels (and there can be an arbitrary number of channels in Particle Flow) but only with those that are actually needed. This conserves memory!
The parameter pCont contains  the Particle Container.

pCont.useAge = true

Because we are going to create new particles in the Birth Script, we will have to access their Age property to set it to 0.

pCont.useTM = true

We will need to copy the complete transformation matrix from the chunks in the scene into the particles in the flow.

pCont.useShape = true

We will also copy the mesh of the particles into the shape property of the particles.

)

on Init pCont do
(

The Init  handler is used to initialize the Birth Script Operator. 
The parameter pCont contains  the Particle Container.

global ChunksArray = $Box* as array

In this case, we define a global variable containing an array listing the scene objects to be used as sources for the chunks.

NOTE that it is a good idea to give both the variable and the scene objects rather unique names to avoid conflicts. In this case though, we are using some simple names...

)

on Proceed pCont do
(

The Proceed  handler is called every time the Script Operator is evaluated by Particle Flow. 
It contains the actual body of the script.
The parameter pCont contains  the Particle Container which contains all particles the Operator is applied to.


t  = pCont.getTimeStart() as float
 

This variable will contain the start time of the evaluation.


if t < 0 do
(

The  start interval will be negative on frame 0 and below. You should not alow negative frames in your scene anyway,
so the following code will be evaluated only of frame 0

NumChunks = ChunksArray.count

This variable will contain the number of scene objects in the array to be turned into particles.

for i = 1 to NumChunks do
(

Now we will create as many particles as objects in the array. The variable i will contain the index of the particle

pCont.AddParticle()

We add a new particle to the container, basically giving it birth!

pCont.particleIndex = pCont.NumParticles()

We read the count of particles in the container and assign to the particleIndex property of the container,
thus making the last created particle the current one!

pCont.particleAge = 0

We set the age of the particle to 0 in case future events would ask for it...

pCont.particleTM = ChunksArray[i].transform

But the important part is to "steal" the whole transformation matrix of the scene object the particle references -
this basically places the particle at the same position as the scene object and also aligns orientation and inherits scaling...


pCont.particleShape = ChunksArray[i].mesh

Finally, we snapshot the mesh from the top of the stack of the scene object and assign to the Shape of the particle.
At this point, the particle looks like the scene object and also occupies the same position in space!

)
)
)

on Release pCont do
(

)

Copyright © 2004 by Borislav 'Bobo' Petrov.