Difference between revisions of "LlSetKeyframedMotion"

From Second Life Wiki
Jump to navigation Jump to search
m (Updated "Continuous Spin" example script to a working example)
 
(18 intermediate revisions by 11 users not shown)
Line 1: Line 1:
{{LSL_Function
{{LSL_Function
|inject-2={{Issues/SCR-345}}{{Issues/SCR-346}}{{Issues/SVC-7796}}{{Issues/SCR-373}}
|inject-2={{Issues/SCR-345}}{{Issues/SCR-346}}{{Issues/SVC-7796}}{{Issues/BUG-7743}}{{Issues/BUG-7767}}{{Issues/BUG-7882}}
|func=llSetKeyframedMotion
|func=llSetKeyframedMotion
|func_id=?
|func_id=?
|func_desc=Specify a list of times, positions, and orientations to be followed by an object. The object will be smoothly moved between keyframes by the simulator. Collisions with other nonphysical or keyframed objects will be ignored (no script events will fire and collision processing will not occur). Collisions with physical objects will be computed and reported, but the keyframed object will be unaffected by those collisions. (The physical object will be affected, however.)  
|func_desc=Specify a list of positions, orientations, and timings to be followed by an object. The object will be smoothly moved between those keyframes by the simulator.
 
Collisions with other nonphysical or keyframed objects will be ignored (no script events will fire and collision processing will not occur). Collisions with physical objects will be computed and reported, but the keyframed object will be unaffected by those collisions. (The physical object will be affected, however.)  
|func_sleep
|func_sleep
|func_energy
|func_energy
Line 13: Line 15:
* ''float'' time
* ''float'' time


Each keyframe is interpreted relative to the previous transform of the object. For example, consider the following list of keyframes: [<0, 0, 10>, [[ZERO_ROTATION]], 5, <0, 0, 0>, [[ZERO_ROTATION]], 5, <0, 0, -10>, [[ZERO_ROTATION]], 5]. This would cause the object to move up 10m over the course of 5s. It would then remain at the location for 5s before moving down 10m over the course of another 5s.
Each keyframe is interpreted '''relative''' to the previous transform of the object. Time values must be 1/9s. or greater. For example, consider the following list of keyframes:
Time values must be 1/9s. or greater.
[<0, 0, 10>, [[ZERO_ROTATION]], 5, <0, 0, 0>, [[ZERO_ROTATION]], 5, <0, 0, -10>, [[ZERO_ROTATION]], 5]
Linear and angular velocities will be clamped to limits set by the simulator (values TBD).
This would cause the object to move up 10m over the course of 5s. It would then remain at the location for 5s before moving down 10m over the course of another 5s.
 
An empty list will terminate any keyframed animation currently playing.  
An empty list will terminate any keyframed animation currently playing.  
|p2_type=list|p2_name=options|p2_desc=modifiers and future options  
|p2_type=list|p2_subtype=instructions|p2_name=options|p2_desc=modifiers and future options  
|spec=
|spec=
The following flags will be supported within the ''options'' list:
The following flags will be supported within the ''options'' list:
Line 36: Line 39:
*This function can only be called on the root prim of a linkset.
*This function can only be called on the root prim of a linkset.
*This function requires the linkset to use the Prim Equivalency system. However, it keyframed objects will not receive the dynamics penalty and can have a physics weight of up to 64.
*This function requires the linkset to use the Prim Equivalency system. However, it keyframed objects will not receive the dynamics penalty and can have a physics weight of up to 64.
*llSetKeyframedMotion is implemented in terms of frames and not real time. To avoid a drift from the expected positions and rotations, use times which are integer multiples of 1/45, e.g. 20.0/45.0, 40.0/45.0, 90.0/45.0, etc. [http://community.secondlife.com/t5/LSL-Scripting/llSetKeyframedMotion-turning-a-corner/td-p/1225219 Forum Thread]
*Linear velocity greater than Second Life's maximum of 250 meters per second will produce an error on DEBUG_CHANNEL.
**Natural drift can still occur; for best results use [[moving_end]] to determine when the animation has ended and confirm the target position with [[llSetPos]].
*llSetKeyframedMotion is implemented in terms of frames and not real time. To avoid a drift from the expected positions and rotations, use times which are integer multiples of 1/45, e.g. 20.0/45.0, 40.0/45.0, 90.0/45.0, etc. [http://community.secondlife.com/t5/LSL-Scripting/llSetKeyframedMotion-turning-a-corner/td-p/1225219 Forum Thread] KFM claims delta times must be larger than 0.1 seconds; in practice, however, delta times slightly over 0.1 will produce an error on DEBUG_CHANNEL. Testing shows a minimum delta time is about 6/45 (.13333) seconds.
**Natural drift can still occur; for best results use [[moving_end]] to determine when the animation has ended and confirm the target position with [[llSetPos]].  Or you can use [[at_target]] or [[at_rot_target]].
*There are a few bugs in the avatar animation system that may cause strange looking animations to play when standing on a moving platform (e.g., walking in place, feet-at-pelvis). We hope to fix these in the future, but doing so is out of scope for this feature.
*There are a few bugs in the avatar animation system that may cause strange looking animations to play when standing on a moving platform (e.g., walking in place, feet-at-pelvis). We hope to fix these in the future, but doing so is out of scope for this feature.
*As with dynamic objects, objects moving using this function are paused when they are selected by an avatar with adequate permissions (object owner, passenger, etc). When such an avatar deselects the object, motion resumes, even if the object had been paused using [[KFM_CMD_PAUSE]].
*As with dynamic objects, objects moving using this function are paused when they are selected by an avatar with adequate permissions (object owner, passenger, etc). When such an avatar deselects the object, motion resumes, even if the object had been paused using [[KFM_CMD_PAUSE]].
*A Key Framed Motion is a '''prim property''' in some respect. When a KFM_LOOP or [[KFM_PING_PONG]] is initiated not only '''the motion''' but also the '''prim position''' is '''preserved after the script is removed'''. I.E. The prim will continue what motion it had and it will snap back when moving is attempted. It will not survive take and copy. It will survive server restart.
*A Key Framed Motion is a '''prim property''' in some respect. When a [[KFM_LOOP]] or [[KFM_PING_PONG]] is initiated the '''the motion''' is '''preserved after the script is removed'''. I.E. The prim will continue what motion it had. It will survive take and copy. It will survive server restart.
*Inter region movement is far from perfect. Crossing from sim 1 to sim 2 using a Key Frame list with numerous frames on a curved path will pick up conspicuous errors on the crossing and when reversed by [ [[KFM_MODE]], [[KFM_REVERSE]] ] the object will stay in sim 2 and never return to sim 1
*Inter region movement is far from perfect. Crossing from sim 1 to sim 2 using a Key Frame list with numerous frames on a curved path will pick up conspicuous errors on the crossing and when reversed by [ [[KFM_MODE]], [[KFM_REVERSE]] ] the object will stay in sim 2 and never return to sim 1
|examples=
|examples=
<lsl>
A KFM object must use the new Prim Equivalency system in order to work. One of many ways to achieve this is to make sure the object uses Convex Hull for its physics shape. (Mainly new prims need a workaround like this. Mesh objects always use the new Prim Equivalency system.)
// If your client is not mesh-aware use the following line:
<source lang="lsl2">
llSetLinkPrimitiveParamsFast(LINK_THIS, [PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);
</source>
 
This is a simple example of back-and-forth or "ping pong" motion, using only positions and time.
<source lang="lsl2">
llSetKeyframedMotion([<0.0, 0.0, 10.0>, 5, <0.0, 0.0, -10.0>, 5], [KFM_DATA, KFM_TRANSLATION, KFM_MODE, KFM_PING_PONG]);
</source>


llSetLinkPrimitiveParamsFast(LINK_THIS,
This example includes position, rotation, and time.
    [PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);
<source lang="lsl2">
</lsl>
<lsl>
llSetKeyframedMotion(
    [<0.0, 0.0, 10.0>,  5,
        <0.0, 0.0, -10.0>,  5],
    [KFM_DATA, KFM_TRANSLATION, KFM_MODE, KFM_PING_PONG]);
</lsl>
<lsl>
llSetKeyframedMotion(
llSetKeyframedMotion(
     [<0.0, 0.0, 10.0>, llEuler2Rot(<90, 45, 180> * DEG_TO_RAD), 5,
     [<0.0, 0.0, 10.0>, llEuler2Rot(<90, 45, 180> * DEG_TO_RAD), 5,
        <0.0, 0.0, -10.0>, llEuler2Rot(<270, 225, 360> * DEG_TO_RAD), 5],
    <0.0, 0.0, -10.0>, llEuler2Rot(<270, 225, 360> * DEG_TO_RAD), 5],
     [KFM_MODE, KFM_REVERSE]);
     [KFM_MODE, KFM_REVERSE]);
</lsl>
</source>
 
=== Pause-Play ===
=== Pause-Play ===
<lsl>llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PAUSE]);
If you need to update a KFM object's position, you will need to pause KFM and continue it afterwards.
//modify prim positions here
<source lang="lsl2">
llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PLAY]);</lsl>
llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PAUSE]);
// Modify prim positions here
llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PLAY]);
</source>
 
===Sample Script - Key Framed Follower===
===Sample Script - Key Framed Follower===
Under some circumstances, rotations will generate a run-time error unless they are normalized. This script illustrates a way to use llKeyframedMotion to create a follower -- think, for example, of a cart behind a vehicle -- using normalized target rotation.
Under some circumstances, rotations will generate a run-time error unless they are normalized. This script illustrates a way to use llKeyframedMotion to create a follower -- think, for example, of a cart behind a vehicle -- using normalized target rotation.
<lsl>
<source lang="lsl2">
key scriptedObjectToFollow = "daf4847b-967d-bff6-69c0-fd7022241be7";
integer isSensorRepeatOn;
integer isSensorRepeatOn;


rotation NormRot(rotation Q)
rotation NormalizeRotation(rotation Q)
{
{
     float MagQ = llSqrt(Q.x*Q.x + Q.y*Q.y +Q.z*Q.z + Q.s*Q.s);
     float MagQ = llSqrt(Q.x*Q.x + Q.y*Q.y +Q.z*Q.z + Q.s*Q.s);
 
     return <Q.x/MagQ, Q.y/MagQ, Q.z/MagQ, Q.s/MagQ>;
     return
        <Q.x/MagQ, Q.y/MagQ, Q.z/MagQ, Q.s/MagQ>;
}
}


Line 82: Line 89:
     state_entry()
     state_entry()
     {
     {
         llSetLinkPrimitiveParamsFast(LINK_ROOT,
         llSetLinkPrimitiveParamsFast(LINK_ROOT, [
                [PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX,
            PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX,
             PRIM_LINK_TARGET, LINK_ALL_CHILDREN,
             PRIM_LINK_TARGET, LINK_ALL_CHILDREN,
                PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_NONE]);
            PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_NONE
        ]);
     }
     }


Line 94: Line 102:
         if (isSensorRepeatOn)
         if (isSensorRepeatOn)
         {
         {
        //  PUBLIC_CHANNEL has the integer value 0
             llSay(PUBLIC_CHANNEL, "Sensor repeat switched on!");
             llSay(PUBLIC_CHANNEL, "Sensor repeat switched on!");
 
             llSensorRepeat("", scriptedObjectToFollow, SCRIPTED, 10.0, PI, 0.3);
             llSensorRepeat("", "91b39b5b-13b1-2517-273a-67360b842c02", SCRIPTED, 10.0, PI, 0.1);
         }
         }
         else
         else
//      {
        {
             llSensorRemove();
             llSensorRemove();
//      }
        }
     }
     }


Line 112: Line 118:
         rotation detectedRotation = llDetectedRot(0);
         rotation detectedRotation = llDetectedRot(0);


         llSetKeyframedMotion(
         vector kfmPosition = (detectedPosition - ownPosition) + <-1.0, 0.0, 0.2> * detectedRotation;
            [(detectedPosition - ownPosition) + <-1.0, 0.0, 0.2>*detectedRotation,
        rotation kfmRotation = NormalizeRotation(detectedRotation / ownRotation);
            NormRot(detectedRotation/ownRotation), 0.12],
        llSetKeyframedMotion([kfmPosition, kfmRotation, 0.15], []);
            []);
     }
     }
}
}
</lsl>
</source>
===Universal Hinged Motion in 8 Key Frames===
===Universal Hinged Motion in 8 Key Frames===
The script will turn a box prim around one edge parallel to the prim's Y-axis<br>
The script will turn a box prim around one edge parallel to the prim's Y-axis<br>
The script will work for any prim orientation<br>
The script will work for any prim orientation<br>
Note that the smallest accepted time per frame is 1/9S=0.11111111S and NOT 0.1S<br>
Note that the smallest accepted time per frame is 1/9S=0.11111111S and NOT 0.1S<br>
<lsl>
<source lang="lsl2">
float angleEnd = PI_BY_TWO;
float angleEnd=PI_BY_TWO;
 
float speed=0.2; // m/S
float speedInMetersPerSecond = 0.2;
float steps=8.0; // number of Key Frames
 
float step=0.0;
float numberOfKeyframes = 8.0;
list KFMlist=[];
float step = 0.0;
 
list listOfKeyframes;
 
vector V;
vector V;
integer open = TRUE;
integer open=TRUE;
vector basePos;
rotation baseRot;


vector startPosition;
float motion_time( float mt)
rotation startRotation;
float motion_time(float mt)
{
{
     mt = llRound(45.0*mt)/45.0;
     mt = llRound(45.0*mt)/45.0;
    if ( mt > 0.11111111 ) return mt;
    else return 0.11111111;
}


    if (0.11111111 < mt)
        return mt;
//  else
        return 0.11111111;
}
default
default
{
{
    on_rez(integer start_param)
    {
        llResetScript();
    }
     state_entry()
     state_entry()
     {
     {
         llSetMemoryLimit(0x2000);
         llSetMemoryLimit(0x2000);
 
         llSetPrimitiveParams([PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);
         llSetLinkPrimitiveParamsFast(LINK_THIS,
         basePos = llGetPos();
            [PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);
         baseRot = llGetRot();
 
         vector v1 = 0.5*llGetScale()*llGetRot();
         vector scale = llGetScale();
         rotation deltaRot = llEuler2Rot(< 0.0, angleEnd/steps, 0.0>);
        startPosition = llGetPos();
         while ( step < steps )
         startRotation = llGetRot();
 
         vector v1 = 0.5*scale*startRotation;
 
         rotation deltaRot = llEuler2Rot(< 0.0, angleEnd/numberOfKeyframes, 0.0>);
 
         while (step < numberOfKeyframes)
         {
         {
             vector ownRot2Left = llRot2Left(llGetRot());
             V = v1*llAxisAngle2Rot(llRot2Left(llGetRot()), angleEnd*step/steps);
 
             V = v1*llAxisAngle2Rot(llRot2Left(llGetRot()), angleEnd*(step+1.0)/steps) - V;
            V = v1*llAxisAngle2Rot(ownRot2Left, angleEnd*step/numberOfKeyframes);
             KFMlist += [V, deltaRot, motion_time(llVecMag(V)/speed)];
             V = v1*llAxisAngle2Rot(ownRot2Left, angleEnd*(step + 1.0)/numberOfKeyframes) - V;
 
             listOfKeyframes += [V, deltaRot, motion_time(llVecMag(V)/speedInMetersPerSecond)];
 
             step += 1.0;
             step += 1.0;
         }
         }
     }
     }
 
     touch_end( integer n)
     touch_start(integer num_detected)
     {
     {
         llSetKeyframedMotion([], []);
         llSetKeyframedMotion( [], []);
 
         if ( open )
         if (open)
         {
         {
             llSetLinkPrimitiveParamsFast(LINK_THIS,
             llSetPrimitiveParams([PRIM_POSITION, basePos, PRIM_ROTATION, baseRot]);
                [PRIM_POSITION, startPosition,
             llSetKeyframedMotion( KFMlist, []);
                PRIM_ROTATION, startRotation]);
 
             llSetKeyframedMotion(listOfKeyframes, []);
         }
         }
         else
         else
//      {
        {
             llSetKeyframedMotion(listOfKeyframes, [KFM_MODE, KFM_REVERSE]);
             llSetKeyframedMotion( KFMlist, [KFM_MODE, KFM_REVERSE]);
//      }
        }
 
         open = !open;
         open = !open;
     }
     }
    on_rez( integer n) { llResetScript(); }
}
}
</lsl>
</source>
After editing prim position, rotation and/or size the script should be reset in order to update the motion
After editing prim position, rotation and/or size the script should be reset in order to update the motion
===Continuous Spin===
The following example does the same thing as using llTargetOmega(<0.0,0.0,1.0>,PI,1.0) to make a prim rotate continuously around its Z-axis, assuming that the prim is set to convex hull and is non-physical.
<source lang="lsl2">
integer gON;
default
{
    touch_end(integer total_number)
    {
        if (gON = !gON)
        {
            // Make repeated rotations of PI radians, each taking 1 seconds
            rotation r = llEuler2Rot(<0.0, 0.0, PI>);
            llSetKeyframedMotion(
                [
                    r, 1,
                    r, 1
                ],
                [KFM_DATA, KFM_ROTATION, KFM_MODE, KFM_LOOP]
            );
            return;
        }
        llSetKeyframedMotion([], []);
    }
}
</source>
===More Examples===
===More Examples===
* [[User:Tapple_Gao/3D_Spinning_Pendulum_Motion|3D Spinning Pendulum Motion]], Suitable for spinning tops, swings, tire swings. Highly configurable
* [[User:Dora_Gustafson/Harmonic_Oscillator_motion|Oscillator Motion in 12 Key Frames]]
* [[User:Dora_Gustafson/Pendulum_motion|Simple Pendulum Motion in 24 Key Frames]], a good motion for a swing
* [[User:Dora_Gustafson/Pendulum_motion|Simple Pendulum Motion in 24 Key Frames]], a good motion for a swing
* [[User:Dora_Gustafson/Harmonic_Oscillator_motion|Oscillator Motion in 12 Key Frames]]
|notes=
|notes=
Potential Use Cases:
Potential Use Cases:
Line 225: Line 228:
|cat3
|cat3
|cat4
|cat4
|haiku={{Haiku|author={{User|Progenitor Resident}}|This is how they move,|Non-phys life for my critters|Smooth, no collisions.}}
|history = Date of Release  [[ Release_Notes/Second_Life_Server/11#11.11.09.244706 | 09/11/2011 ]]
|history = Date of Release  [[ Release_Notes/Second_Life_Server/11#11.11.09.244706 | 09/11/2011 ]]
}}
}}

Latest revision as of 08:25, 7 September 2024

Summary

Function: llSetKeyframedMotion( list keyframes, list options );

Specify a list of positions, orientations, and timings to be followed by an object. The object will be smoothly moved between those keyframes by the simulator.

Collisions with other nonphysical or keyframed objects will be ignored (no script events will fire and collision processing will not occur). Collisions with physical objects will be computed and reported, but the keyframed object will be unaffected by those collisions. (The physical object will be affected, however.)

• list keyframes Strided keyframe list of the form:

Each keyframe is interpreted relative to the previous transform of the object. Time values must be 1/9s. or greater. For example, consider the following list of keyframes:

[<0, 0, 10>, ZERO_ROTATION, 5, <0, 0, 0>, ZERO_ROTATION, 5, <0, 0, -10>, ZERO_ROTATION, 5]

This would cause the object to move up 10m over the course of 5s. It would then remain at the location for 5s before moving down 10m over the course of another 5s.

An empty list will terminate any keyframed animation currently playing.

• list options modifiers and future options

Specification

The following flags will be supported within the options list:

  • KFM_MODE followed by one of: KFM_LOOP, KFM_REVERSE, KFM_FORWARD, or KFM_PING_PONG will specify the playback mode. Defaults to KFM_FORWARD. Must be specified when the keyframe list is provided.
  • KFM_DATA followed by a bitwise combination of: KFM_TRANSLATION and KFM_ROTATION. By default both rotations and translations must be provided. If you specify one or the other, you should only include translations or rotations in your keyframe list. Must be specified at the time the keyframe list is provided.
  • KFM_COMMAND followed by one of: KFM_CMD_STOP, KFM_CMD_PLAY, KFM_CMD_PAUSE. STOP will pause the animation AND reset it to the beginning. PAUSE will pause the animation without resetting. PLAY will resume a paused or stopped animation.

Note that if KFM_COMMAND is provided in the options list, it must be the only option in the list, and cannot be specified in the same function call that sets the keyframes list.

Inter-region movement is supported; simply specify a target keyframe that would place the object outside the sim boundaries and it will cross over. This seems to provide the same effect as pausing the animation when it is in limbo between the two regions, but the viewer continues to interpolate its movement. It will jump back to the sim border and continue on the same speed, although the time between waypoints will be increased due to the sim crossing. Make sure you factor this into your system.

Since keyframed motion is a prim property rather than restricted to the script that called it, you can use multiple scripts to control, pause, or restart the same keyframed motion without passing the original keyframe list provided. Just pass an empty list for keyframes list, and the KFM_COMMAND followed by the command in the options list.

options flags Additional Parameters Description
KFM_COMMAND ] 0 [ integer command ] Sets the command
KFM_CMD_PLAY 0 ?
KFM_CMD_STOP 1 ?
KFM_CMD_PAUSE 2 ?
KFM_MODE ] 1 [ integer mode ] Sets the playback modeDefaults to KFM_FORWARD
KFM_FORWARD 0 ?
KFM_LOOP 1 ?
KFM_PING_PONG 2 ?
KFM_REVERSE 3 ?
KFM_DATA ] 2 [ integer fields ] Sets what data the keyframes containDefaults to (KFM_ROTATION | KFM_TRANSLATION)
KFM_ROTATION 0x1 ?
KFM_TRANSLATION 0x2 ?

Caveats

  • This function does not work in attachments.
  • This function can only be called on NON-physical objects. In the future it could be extended to support physical objects, but this is more complicated as collisions could prevent the object from reaching its goal positions on time.
  • You cannot use scripts to move any prim in the linkset while the keyframed motion is active. If you must, first pause and restart it with KFM_CMD_PLAY when done. #Example
  • Collisions with avatars affect the angular movement of the object and it may not reach the final rotation.
  • This function can only be called on the root prim of a linkset.
  • This function requires the linkset to use the Prim Equivalency system. However, it keyframed objects will not receive the dynamics penalty and can have a physics weight of up to 64.
  • Linear velocity greater than Second Life's maximum of 250 meters per second will produce an error on DEBUG_CHANNEL.
  • llSetKeyframedMotion is implemented in terms of frames and not real time. To avoid a drift from the expected positions and rotations, use times which are integer multiples of 1/45, e.g. 20.0/45.0, 40.0/45.0, 90.0/45.0, etc. Forum Thread KFM claims delta times must be larger than 0.1 seconds; in practice, however, delta times slightly over 0.1 will produce an error on DEBUG_CHANNEL. Testing shows a minimum delta time is about 6/45 (.13333) seconds.
  • There are a few bugs in the avatar animation system that may cause strange looking animations to play when standing on a moving platform (e.g., walking in place, feet-at-pelvis). We hope to fix these in the future, but doing so is out of scope for this feature.
  • As with dynamic objects, objects moving using this function are paused when they are selected by an avatar with adequate permissions (object owner, passenger, etc). When such an avatar deselects the object, motion resumes, even if the object had been paused using KFM_CMD_PAUSE.
  • A Key Framed Motion is a prim property in some respect. When a KFM_LOOP or KFM_PING_PONG is initiated the the motion is preserved after the script is removed. I.E. The prim will continue what motion it had. It will survive take and copy. It will survive server restart.
  • Inter region movement is far from perfect. Crossing from sim 1 to sim 2 using a Key Frame list with numerous frames on a curved path will pick up conspicuous errors on the crossing and when reversed by [ KFM_MODE, KFM_REVERSE ] the object will stay in sim 2 and never return to sim 1

Examples

A KFM object must use the new Prim Equivalency system in order to work. One of many ways to achieve this is to make sure the object uses Convex Hull for its physics shape. (Mainly new prims need a workaround like this. Mesh objects always use the new Prim Equivalency system.)

llSetLinkPrimitiveParamsFast(LINK_THIS, [PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);

This is a simple example of back-and-forth or "ping pong" motion, using only positions and time.

llSetKeyframedMotion([<0.0, 0.0, 10.0>, 5, <0.0, 0.0, -10.0>, 5], [KFM_DATA, KFM_TRANSLATION, KFM_MODE, KFM_PING_PONG]);

This example includes position, rotation, and time.

llSetKeyframedMotion(
    [<0.0, 0.0, 10.0>, llEuler2Rot(<90, 45, 180> * DEG_TO_RAD), 5,
     <0.0, 0.0, -10.0>, llEuler2Rot(<270, 225, 360> * DEG_TO_RAD), 5],
    [KFM_MODE, KFM_REVERSE]);

Pause-Play

If you need to update a KFM object's position, you will need to pause KFM and continue it afterwards.

llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PAUSE]);
// Modify prim positions here
llSetKeyframedMotion([],[KFM_COMMAND, KFM_CMD_PLAY]);

Sample Script - Key Framed Follower

Under some circumstances, rotations will generate a run-time error unless they are normalized. This script illustrates a way to use llKeyframedMotion to create a follower -- think, for example, of a cart behind a vehicle -- using normalized target rotation.

key scriptedObjectToFollow = "daf4847b-967d-bff6-69c0-fd7022241be7";
integer isSensorRepeatOn;

rotation NormalizeRotation(rotation Q)
{
    float MagQ = llSqrt(Q.x*Q.x + Q.y*Q.y +Q.z*Q.z + Q.s*Q.s);
    return <Q.x/MagQ, Q.y/MagQ, Q.z/MagQ, Q.s/MagQ>;
}

default
{
    state_entry()
    {
        llSetLinkPrimitiveParamsFast(LINK_ROOT, [
            PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX,
            PRIM_LINK_TARGET, LINK_ALL_CHILDREN,
            PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_NONE
        ]);
    }

    touch_start(integer total_number)
    {
        isSensorRepeatOn = !isSensorRepeatOn;

        if (isSensorRepeatOn)
        {
            llSay(PUBLIC_CHANNEL, "Sensor repeat switched on!");
            llSensorRepeat("", scriptedObjectToFollow, SCRIPTED, 10.0, PI, 0.3);
        }
        else
        {
            llSensorRemove();
        }
    }

    sensor(integer num_detected)
    {
        vector ownPosition = llGetPos();
        rotation ownRotation = llGetRot();
        vector detectedPosition = llDetectedPos(0);
        rotation detectedRotation = llDetectedRot(0);

        vector kfmPosition = (detectedPosition - ownPosition) + <-1.0, 0.0, 0.2> * detectedRotation;
        rotation kfmRotation = NormalizeRotation(detectedRotation / ownRotation);
        llSetKeyframedMotion([kfmPosition, kfmRotation, 0.15], []);
    }
}

Universal Hinged Motion in 8 Key Frames

The script will turn a box prim around one edge parallel to the prim's Y-axis
The script will work for any prim orientation
Note that the smallest accepted time per frame is 1/9S=0.11111111S and NOT 0.1S

float angleEnd=PI_BY_TWO;
float speed=0.2; // m/S
float steps=8.0; // number of Key Frames
float step=0.0;
list KFMlist=[];
vector V;
integer open=TRUE;
vector basePos;
rotation baseRot;

float motion_time( float mt)
{
    mt = llRound(45.0*mt)/45.0;
    if ( mt > 0.11111111 ) return mt;
    else return 0.11111111;
}

default
{
    state_entry()
    {
        llSetMemoryLimit(0x2000);
        llSetPrimitiveParams([PRIM_PHYSICS_SHAPE_TYPE, PRIM_PHYSICS_SHAPE_CONVEX]);
        basePos = llGetPos();
        baseRot = llGetRot();
        vector v1 = 0.5*llGetScale()*llGetRot();
        rotation deltaRot = llEuler2Rot(< 0.0, angleEnd/steps, 0.0>);
        while ( step < steps )
        {
            V = v1*llAxisAngle2Rot(llRot2Left(llGetRot()), angleEnd*step/steps);
            V = v1*llAxisAngle2Rot(llRot2Left(llGetRot()), angleEnd*(step+1.0)/steps) - V;
            KFMlist += [V, deltaRot, motion_time(llVecMag(V)/speed)];
            step += 1.0;
        }
    }
    touch_end( integer n)
    {
        llSetKeyframedMotion( [], []);
        if ( open )
        {
            llSetPrimitiveParams([PRIM_POSITION, basePos, PRIM_ROTATION, baseRot]);
            llSetKeyframedMotion( KFMlist, []);
        }
        else
        {
            llSetKeyframedMotion( KFMlist, [KFM_MODE, KFM_REVERSE]);
        }
        open = !open;
    }
    on_rez( integer n) { llResetScript(); }
}

After editing prim position, rotation and/or size the script should be reset in order to update the motion

Continuous Spin

The following example does the same thing as using llTargetOmega(<0.0,0.0,1.0>,PI,1.0) to make a prim rotate continuously around its Z-axis, assuming that the prim is set to convex hull and is non-physical.

integer gON;

default
{
    touch_end(integer total_number)
    {
        if (gON = !gON)
        {
            // Make repeated rotations of PI radians, each taking 1 seconds
            rotation r = llEuler2Rot(<0.0, 0.0, PI>);
            llSetKeyframedMotion(
                [
                    r, 1,
                    r, 1
                ], 
                [KFM_DATA, KFM_ROTATION, KFM_MODE, KFM_LOOP]
            );
            return;
        }

        llSetKeyframedMotion([], []);
    }
}

More Examples

Notes

Potential Use Cases:

  • Elevators
  • Moving platforms
  • Trains/Fixed-Track Vehicles
  • Moving doors/walls/gates
  • Windmills and other machines

Targeted coordinate systems: The Translation is in Global coordinates, the Rotation in Local coordinates

  • Say: a move on the X-axis will move the object along the global, region X-axis no matter how the object is rotated
  • Say: a rotate around the X-axis will rotate the object around it's local, prim X-axis no matter the object's rotation
  • When the object is not rotated in the global system you won't notice the difference

Deep Notes

History

Date of Release 09/11/2011

Signature

function void llSetKeyframedMotion( list keyframes, list options );

Haiku

This is how they move,
Non-phys life for my critters
Smooth, no collisions.
Progenitor Resident