Saves a shape key for this Geometry and creates a new ShapeClip
object in the mixer to represent this shape. Shape keys have two parts: sources (a shape that you
have stored) and clips (an instance of a shape key (source) at a particular position along a track
in the animation mixer).
There are two possible workflows for using this method: the first is to change the actual shape of
the primitive and then call this method (which is similar to the SaveShapeKey
command); the second is to specify the shape using the PositionArray argument. The example below
illustrates these two approaches.
The actual data for the shape is stored as a ClusterProperty. It is possible to
store a shape key that defines positions for only a subset of the points, by using the IndexArray
argument. This can save memory on a large object. Each subset of points with a shape key is
represented by a separate Cluster.
ShapeClip Geometry.SaveShapeKey( Double in_time, Double in_clipDuration, siShapeReferenceMode in_refMode, siShapeInstanceMode in_instMode, String in_name, Object in_indexArray, Object in_positionArray, Object in_normalArray ); |
oReturn = Geometry.SaveShapeKey( Time, [ClipDuration], [RefMode], [InstMode], [Name], [IndexArray], [PositionArray], [NormalArray] ); |
Parameter | Type | Description |
---|---|---|
Time | Double | Time in seconds. The shape of the object is evaluated at this time, unless the shape is explicitly defined by the PositionArray argument. The time argument is also used when adding the ShapeClip to the Mixer. For example, if siShapeMixedWeightMode is specified as the InstMode argument, then the new shape will account for 100% of the shape at the specified time. Only a single shape key can be specified at any particular time. |
ClipDuration | Double |
Clip duration in seconds. Specify -1.0 to have the default behavior, which depends
on the InstMode argument. For example if siShapeMixedWeightMode is specified then
the ClipDuration is the entire scene duration, but if siShapeTransitionMode is
specified then clip will last 1 second.
Default Value: -1.0 |
RefMode | siShapeReferenceMode |
Shape reference mode. This specifies how the shape is stored and how the shape will
change if the underlying reference shape changes. The reference mode is ignored when
more than one shape is saved on the same cluster, the reference mode of the first shape
key is used instead.
Default Value: siShapeLocalReferenceMode |
InstMode | siShapeInstanceMode |
Shape instantiation mode. This controls the way the new ShapeClip is
added to the Mixer. With siShapeMixedWeightMode a new track is created, the new clip
added to this track, and the fcurves controlling the weighting are adjusted. With
siShapeInstanceOnlyMode the clips are added to separate tracks, but no fcurves are used
to weight between the shapes. With siShapeTransitionMode all clips are added to the same
track, with transition effects applied between them.
Default Value: siShapeMixedWeightMode |
Name | String |
Name to use for the new ShapeClip. The ClusterProperty
associated with this shape is not affected by this argument.
Default Value: "Point_AUTO_ShapeAction_Clip" |
IndexArray | Array |
A 1-dimensional array of geometry point indices. Only these points will be included in the
shape definition. If no other shapes use this subset of points then a new
Cluster will be created.
Default Value: If not specified the shape key is saved for every point on the geometry |
PositionArray | Array |
A 1- or 2- dimensional array of key position points used for computing the offsets of the geometry
points referred by the array of indices. The points must be specified in absolute coordinates.
A 1-dimensional array must be formatted as a sequence of x,y,z values: {Xo,Yo,Zo,...X(n-1),Y(n-1),Z(n-1)}.
A 2 dimensional array must be formatted as a Nx3 array of x,y,z values. This argument is only considered
if the IndexArray argument is specified, and must specify the same number of points as the IndexArray.
Default Value: Shape key is based on the current positions of the geometry points |
NormalArray | Array | A 1 or 2 dimensional array of normal points. Normals are not yet supported. |
' ' This example shows how it is possible to create shape animations from the object model. ' Two techniques are used to create the same animation on two polygons ' dim oGrid1, oGrid2 ' Technique one is similar to the process used when interactively ' saving shape keys - we change the shape of the object and then save a ' shape key ' Create a polygon set oGrid1 = CreatePrim( "Grid", "MeshSurface" ) SetValue oGrid1 & ".polymsh.geom.subdivu", 1 SetValue oGrid1 & ".polymsh.geom.subdivv", 1 Scale oGrid1, 0.25, 0.25, 0.25, siAbsolute, siParent, siObj, siXYZ Translate oGrid1, -2, 0, 0, siRelative, siParent, siObj, siXYZ ' Our animation will start with the initial shape at frame 1 and ' morph to the second shape until frame 30 dim frameRate, frame1Seconds, frame30Seconds frameRate = GetValue( "PlayControl.Rate" ) frame1Seconds = 1 / frameRate frame30Seconds = 30 / frameRate ' Save the current shape as the reference ' This remembers the locations of each vertex. This information will ' be all zeros because we haven't moved any vertices and use siShapeLocalReferenceMode call oGrid1.ActivePrimitive.Geometry.SaveShapeKey( frame1Seconds, -1.0, _ siShapeLocalReferenceMode, siShapeMixedWeightMode ) ' Move some of the points Translate oGrid1 & ".pnt[0]" , 0, 0, -1, siRelative, siGlobal, siObj, siXYZ Translate oGrid1 & ".pnt[2]" , 0, 0, 1, siRelative, siGlobal, siObj, siXYZ Translate oGrid1 & ".pnt[3]" , 0, 0, -1, siRelative, siGlobal, siObj, siXYZ ' Take a second snap shot of the object call oGrid1.ActivePrimitive.Geometry.SaveShapeKey( frame30Seconds, -1.0, _ siShapeLocalReferenceMode, siShapeMixedWeightMode ) ' Technique two is direct. We describe the actual positions ' of the vertices for each shape. set oGrid2 = CreatePrim( "Grid", "MeshSurface" ) SetValue oGrid2 & ".polymsh.geom.subdivu", 1 SetValue oGrid2 & ".polymsh.geom.subdivv", 1 Scale oGrid2, 0.25, 0.25, 0.25, siAbsolute, siParent, siObj, siXYZ Translate oGrid2, +2, 0, 0, siRelative, siParent, siObj, siXYZ ' We only specify the vertices that actually move. In the case of ' a large object this could save a lot of memory call oGrid2.ActivePrimitive.Geometry.SaveShapeKey( frame1Seconds, -1.0, _ siShapeLocalReferenceMode, siShapeMixedWeightMode, ,_ Array( 0,2,3 ), Array( -4,0,-4, 4,0,-4, 4,0,4) ) call oGrid2.ActivePrimitive.Geometry.SaveShapeKey( frame30Seconds, -1.0, _ siShapeLocalReferenceMode, siShapeMixedWeightMode, ,_ Array( 0,2,3 ), Array( -4,0,-8, 4,0,0, 4,0,0) ) |
' ' Three cones are created with identical animation ' although they use different Reference Modes to record the shape ' set root = Application.ActiveProject.ActiveScene.Root set obj = root.AddGeometry( "Cone", "MeshSurface" ) ApplyOp "bend", obj obj.posz.Value = -3 set geometry = obj.ActivePrimitive.Geometry ' Move the point to 1,1,0 in local mode set clip = geometry.SaveShapeKey( 1, 1, siShapeLocalReferenceMode, _ siShapeInstanceOnlyMode, "MyShapeKey", Array(1), Array(1.,1.,0) ) set obj = root.AddGeometry( "Cone", "MeshSurface" ) ApplyOp "bend", obj set geometry = obj.ActivePrimitive.Geometry ' Move the point to 1,1,0 in absolute mode set clip = geometry.SaveShapeKey( 1, 1, siShapeAbsoluteReferenceMode, _ siShapeInstanceOnlyMode, "MyShapeKey", Array(1), Array(1.,1.,0) ) set obj = root.AddGeometry( "Cone", "MeshSurface" ) ApplyOp "bend", obj obj.posz.value = 3 set geometry = obj.ActivePrimitive.Geometry ' Move the point to 1,1,0 in object mode set clip = geometry.SaveShapeKey( 1, 1, siShapeObjectReferenceMode, _ siShapeInstanceOnlyMode, "MyShapeKey", Array(1), Array(1.,1.,0) ) |
' ' This example demonstrates how you access the shape information for a particular shape key by looking ' at the cluster properties which store this data ' option explicit ' Set up a little scene dim oCircle set oCircle = CreatePrim( "Sphere", "MeshSurface" ) ' Move the top and bottom point of the sphere outwards oCircle.ActivePrimitive.Geometry.SaveShapeKey 0.5, 3.0, _ siShapeAbsoluteReferenceMode, siShapeInstanceOnlyMode, ,_ Array( 1, 0 ), Array( 0,6,0, 0,-6,0 ) ' Move two different points - this creates a new, independent cluster ' We can use a different reference mode for this, but we still specify the ' point positions in Absolute terms. oCircle.ActivePrimitive.Geometry.SaveShapeKey 0.8, 1.0, _ siShapeLocalReferenceMode, siShapeInstanceOnlyMode,, _ Array( 5, 33 ), Array( -2,0,0, 2,0,0 ) ShowShapeInformation( oCircle ) sub ShowShapeInformation( in_obj ) dim oClusters, oCluster, oProperties, oProperty, cntFoundShapes on error resume next set oClusters = in_obj.ShapeAnimatedClusters if ( err <> 0 ) then Application.LogMessage "Please select a 3D Object" exit sub end if on error goto 0 Application.LogMessage "Dump of shape key data on object " & in_obj.Fullname cntFoundShapes = 0 ' Go through the various clusters on the object ' (In practice only the Point clusters will have shape keys) for each oCluster in oClusters ' Cluster indices are not the same as the indices on the ' geometry, but we can easily determine the relationship with ' this array: dim oClusterElementsCollection, aElements set oClusterElementsCollection = oCluster.Elements aElements = oClusterElementsCollection.Array ' Only search for the shape keys, which have type "clskey" set oProperties = oCluster.LocalProperties.filter( "clskey" ) for each oProperty in oProperties ' Found a shape key dim i, XYZArray cntFoundShapes = cntFoundShapes + 1 if InstrRev( oProperty, ".ResultClusterKey" ) <> 0 then ' There may also be an internal cluster property ' called "ResultClusterKey" which stores the ' result of blending the various shapes at the ' current time Application.LogMessage "Blended shape at this time: " & oProperty else Application.LogMessage "Shape key: " & oProperty end if ' The reference mode is available from the KeyType parameter dim ReferenceType ReferenceType = oProperty.Parameters("KeyType").Value if ( ReferenceType = 0 ) then Logmessage " Uses: Absolute Reference Mode" elseif ( ReferenceType = 1 ) then Logmessage " Uses: Local Reference Mode" elseif ( ReferenceType = 2 ) then Logmessage " Uses: Object Reference Mode" end if ' The contents of the cluster can be found in this safearray XYZArray = oProperty.Elements.Array For i=0 to UBound(oProperty.Elements.Array,2) ' Print out the x,y,z values LogMessage " pnt["& aElements( i ) & "] has position (" _ & Round(XYZArray(0,i),3) & "," & Round(XYZArray(1,i),3) & ","_ & Round(XYZArray(2,i),3) & ")" Next next next if ( cntFoundShapes = 0 ) then LogMessage "There are no shapes on this object" end if end sub |