content/hifi-content/dave/walk-beta/walk.js
2022-02-13 22:49:05 +01:00

503 lines
No EOL
24 KiB
JavaScript

//
// walk.js
// version 2.01
//
// Created by David Wooldridge, June 2015
// Copyright © 2014 - 2016 High Fidelity, Inc.
//
// Animates an avatar using procedural animation techniques combined with harmonics derived from motion capture data.
//
// Editing tools available here:
// https://github.com/DaveDubUK/walkTools
// and here:
// https://hifi-content.s3.amazonaws.com/dave/walk-tools/walk.js
//
// Distributed under the Apache License, Version 2.0.
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
//
// animations, reach poses, reach pose parameters, transitions, transition parameters, sounds, image/s and reference files
var pathToAssets = "https://hifi-content.s3.amazonaws.com/dave/walk-beta/assets/";
print('walk.js: Started. Loading assets from ' + pathToAssets);
Script.include([
"./libraries/walkConstants.js",
"./libraries/walkFilters.js",
"./libraries/walkApi.js",
pathToAssets + "walkAssets.js"
]);
// construct Avatar, Motion and (null) Transition
var avatar = new Avatar();
var motion = new Motion();
var nullTransition = new Transition();
motion.currentTransition = nullTransition;
// create settings (gets initial values from avatar)
Script.include("./libraries/walkSettings.js");
print('walk.js: Ready');
// Main loop
Script.update.connect(function(deltaTime) {
if (motion.isLive) {
// assess current locomotion state
motion.assess(deltaTime);
// decide which animation should be playing
composeAnimation();
// advance the animation cycle/s by the correct amount/s
advanceAnimations();
// update the progress of any live transitions
updateTransitions();
// apply translation and rotations
renderMotion();
// save this frame's parameters
motion.saveHistory();
}
});
// helper function for composeAnimation()
function setTransition(nextAnimation, playTransitionReachPoses, inAnticipation) {
var lastTransition = motion.currentTransition;
var lastAnimation = avatar.currentAnimation;
// if already transitioning from a blended walk need to maintain the previous walk's direction
if (lastAnimation.lastDirection) {
switch(lastAnimation.lastDirection) {
case FORWARDS:
lastAnimation = avatar.selectedWalk;
break;
case BACKWARDS:
lastAnimation = avatar.selectedWalkBackwards;
break;
case LEFT:
lastAnimation = avatar.selectedSideStepLeft;
break;
case RIGHT:
lastAnimation = avatar.selectedSideStepRight;
break;
}
}
motion.currentTransition = new Transition(nextAnimation, lastAnimation, lastTransition, playTransitionReachPoses);
avatar.currentAnimation = nextAnimation;
// reset default first footstep
if (nextAnimation === avatar.selectedWalkBlend && lastTransition === nullTransition) {
avatar.nextStep = RIGHT;
}
}
// fly animation blending: smoothing / damping filters
const FLY_BLEND_DAMPING = 50;
var verticalFilter = filter.createAveragingFilter(FLY_BLEND_DAMPING);
var lateralFilter = filter.createAveragingFilter(FLY_BLEND_DAMPING);
var forwardFilter = filter.createAveragingFilter(FLY_BLEND_DAMPING);
var turningFilter = filter.createAveragingFilter(FLY_BLEND_DAMPING);
var slowFlyFilter = filter.createAveragingFilter(FLY_BLEND_DAMPING);
// select / blend the appropriate animation for the current state of motion
function composeAnimation() {
// select appropriate animation. create transitions where appropriate
var playTransitionReachPoses = true;
switch (motion.nextState) {
case STATIC: {
if (avatar.distanceFromSurface <= ON_SURFACE_THRESHOLD) {
if (motion.yawDelta < -YAW_THRESHOLD &&
avatar.currentAnimation !== avatar.selectedTurnLeft) {
setTransition(avatar.selectedTurnLeft, playTransitionReachPoses);
} else if (motion.yawDelta > YAW_THRESHOLD &&
avatar.currentAnimation !== avatar.selectedTurnRight) {
setTransition(avatar.selectedTurnRight, playTransitionReachPoses);
} else if (motion.yawDelta > -YAW_THRESHOLD && motion.yawDelta < YAW_THRESHOLD &&
avatar.currentAnimation !== avatar.selectedIdle) {
setTransition(avatar.selectedIdle, playTransitionReachPoses);
}
} else if (avatar.distanceFromSurface > ON_SURFACE_THRESHOLD &&
avatar.currentAnimation !== avatar.selectedHover) {
setTransition(avatar.selectedHover, playTransitionReachPoses);
}
motion.state = STATIC;
avatar.selectedWalkBlend.lastDirection = NONE;
break;
}
case SURFACE_MOTION: {
// walk transition reach poses are currently only specified for starting to walk forwards
playTransitionReachPoses = (motion.direction === FORWARDS);
var isAlreadyWalking = (avatar.currentAnimation === avatar.selectedWalkBlend);
switch (motion.direction) {
case FORWARDS:
if (avatar.selectedWalkBlend.lastDirection !== FORWARDS) {
animationOperations.deepCopy(avatar.selectedWalk, avatar.selectedWalkBlend);
}
avatar.selectedWalkBlend.lastDirection = FORWARDS;
break;
case BACKWARDS:
if (avatar.selectedWalkBlend.lastDirection !== BACKWARDS) {
animationOperations.deepCopy(avatar.selectedWalkBackwards, avatar.selectedWalkBlend);
}
avatar.selectedWalkBlend.lastDirection = BACKWARDS;
break;
case LEFT:
animationOperations.deepCopy(avatar.selectedSideStepLeft, avatar.selectedWalkBlend);
avatar.selectedWalkBlend.lastDirection = LEFT;
break
case RIGHT:
animationOperations.deepCopy(avatar.selectedSideStepRight, avatar.selectedWalkBlend);
avatar.selectedWalkBlend.lastDirection = RIGHT;
break;
default:
// condition occurs when the avi goes through the floor due to collision hull errors
animationOperations.deepCopy(avatar.selectedWalk, avatar.selectedWalkBlend);
avatar.selectedWalkBlend.lastDirection = FORWARDS;
break;
}
if (!isAlreadyWalking && !motion.isComingToHalt) {
setTransition(avatar.selectedWalkBlend, playTransitionReachPoses);
}
motion.state = SURFACE_MOTION;
break;
}
case AIR_MOTION: {
// blend the up, down, forward and backward flying animations relative to motion speed and direction
animationOperations.zeroAnimation(avatar.selectedFlyBlend);
// calculate influences based on velocity and direction
var speed = Vec3.length(motion.velocity);
var sumOfSpeeds = Math.abs(motion.velocity.x) +
Math.abs(motion.velocity.y) +
Math.abs(motion.velocity.z);
var verticalProportion = motion.velocity.y / sumOfSpeeds;
var lateralProportion = motion.velocity.x / sumOfSpeeds;
var forwardProportion = -motion.velocity.z / sumOfSpeeds;
// factor in slow flying and turning influences (overwrite velocity / direction influences)
const FLY_SPEED_MULTIPLIER = 2;
var flyingSlowlySpeed = MAX_WALK_SPEED * FLY_SPEED_MULTIPLIER *
(GRAVITY_THRESHOLD - avatar.distanceFromSurface) > 0 ?
MAX_WALK_SPEED * FLY_SPEED_MULTIPLIER *
(GRAVITY_THRESHOLD - avatar.distanceFromSurface) : 0;
var slowFlyComponent = speed > flyingSlowlySpeed ? 0 :
(flyingSlowlySpeed - speed) / flyingSlowlySpeed;
// only use slow flying animation when moving forwards
slowFlyComponent *= motion.direction === FORWARDS ? 1 : 0;
var turningComponent = Math.abs(motion.yawDelta) / DELTA_YAW_MAX > 1 ? 1 :
Math.abs(motion.yawDelta) / DELTA_YAW_MAX;
turningComponent *= (1 - slowFlyComponent); // reduce turning influence at low speeds
// final proportions
var overallDirectionInfluence = 1 - slowFlyComponent - turningComponent > 0 ?
1 - slowFlyComponent - turningComponent : 0;
verticalProportion *= overallDirectionInfluence;
lateralProportion *= overallDirectionInfluence;
forwardProportion *= overallDirectionInfluence;
// smooth / damp to add visual 'weight'
verticalProportion = verticalFilter.process(verticalProportion);
lateralProportion = lateralFilter.process(lateralProportion);
forwardProportion = forwardFilter.process(forwardProportion);
slowFlyComponent = turningFilter.process(slowFlyComponent);
turningComponent = slowFlyFilter.process(turningComponent);
// blend animations proportionally
if (verticalProportion > 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFlyUp.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFlyUp,
avatar.selectedFlyBlend,
verticalProportion);
}
if (verticalProportion < 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFlyDown.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFlyDown,
avatar.selectedFlyBlend,
-verticalProportion);
}
if (forwardProportion > 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFly.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFly,
avatar.selectedFlyBlend,
forwardProportion);
}
if (forwardProportion < 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFlyBackwards.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFlyBackwards,
avatar.selectedFlyBlend,
-forwardProportion);
}
if (slowFlyComponent > 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFlySlow.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFlySlow,
avatar.selectedFlyBlend,
slowFlyComponent);
}
if (turningComponent > 0) {
avatar.currentAnimation.calibration.frequency = avatar.selectedFlyTurning.calibration.frequency
animationOperations.blendAnimation(avatar.selectedFlyTurning,
avatar.selectedFlyBlend,
turningComponent);
}
// set transition?
if (avatar.currentAnimation !== avatar.selectedFlyBlend) {
setTransition(avatar.selectedFlyBlend, playTransitionReachPoses);
}
motion.state = AIR_MOTION;
avatar.selectedWalkBlend.lastDirection = NONE;
break;
}
} // end switch next state of motion
}
// determine the length of stride. advance the frequency time wheels. advance frequency time wheels for any live transitions
function advanceAnimations() {
var wheelAdvance = 0;
if (avatar.currentAnimation === avatar.selectedWalkBlend) {
// Using technique described here: http://www.gdcvault.com/play/1020583/Animation-Bootcamp-An-Indie-Approach
// wrap the stride length around a 'surveyor's wheel' twice and calculate the angular speed at the given (linear) speed
// omega = v / r , where r = circumference / 2 PI and circumference = 2 * stride length
var speed = Vec3.length(motion.velocity);
motion.frequencyTimeWheelRadius = avatar.currentAnimation.calibration.strideLength / Math.PI;
var ftWheelAngularVelocity = speed / motion.frequencyTimeWheelRadius;
// calculate the degrees turned (at this angular speed) since last frame
wheelAdvance = filter.radToDeg(motion.deltaTime * ftWheelAngularVelocity);
} else {
// turn the frequency time wheel by the amount specified for this animation
wheelAdvance = filter.radToDeg(avatar.currentAnimation.calibration.frequency * motion.deltaTime);
}
if (motion.currentTransition !== nullTransition) {
// the last animation is still playing so we turn it's frequency time wheel to maintain the animation
if (motion.currentTransition.lastAnimation === motion.selectedWalkBlend) {
// if at a stop angle (i.e. feet now under the avi) hold the wheel position for remainder of transition
var tolerance = motion.currentTransition.lastFrequencyTimeIncrement + 0.1;
if ((motion.currentTransition.lastFrequencyTimeWheelPos >
(motion.currentTransition.stopAngle - tolerance)) &&
(motion.currentTransition.lastFrequencyTimeWheelPos <
(motion.currentTransition.stopAngle + tolerance))) {
motion.currentTransition.lastFrequencyTimeIncrement = 0;
}
}
motion.currentTransition.advancePreviousFrequencyTimeWheel(motion.deltaTime);
}
// advance the walk wheel the appropriate amount
motion.advanceFrequencyTimeWheel(wheelAdvance);
}
// initialise a new transition. update progress of a live transition
function updateTransitions() {
if (motion.currentTransition !== nullTransition) {
// is this a new transition?
if (motion.currentTransition.progress === 0) {
// do we have overlapping transitions?
if (motion.currentTransition.lastTransition !== nullTransition) {
// is the last animation for the nested transition the same as the new animation?
if (motion.currentTransition.lastTransition.lastAnimation === avatar.currentAnimation) {
// then sync the nested transition's frequency time wheel for a smooth animation blend
motion.frequencyTimeWheelPos = motion.currentTransition.lastTransition.lastFrequencyTimeWheelPos;
}
}
}
if (motion.currentTransition.updateProgress() === TRANSITION_COMPLETE) {
motion.currentTransition = nullTransition;
}
}
}
// helper function for renderMotion(). calculate the amount to lean forwards (or backwards) based on the avi's velocity
var leanPitchSmoothingFilter = filter.createButterworthFilter();
function getLeanPitch() {
var leanProgress = 0;
if (motion.direction === DOWN ||
motion.direction === FORWARDS ||
motion.direction === BACKWARDS) {
leanProgress = -motion.velocity.z / TOP_SPEED;
}
// use filters to shape the walking acceleration response
leanProgress = leanPitchSmoothingFilter.process(leanProgress);
return PITCH_MAX * leanProgress;
}
// helper function for renderMotion(). calculate the angle at which to bank into corners whilst turning
var leanRollSmoothingFilter = filter.createAveragingFilter(20);
function getLeanRoll() {
var leanRollProgress = 0;
var linearContribution = 0;
const LOG_SCALER = 8;
if (Vec3.length(motion.velocity) > 0) {
linearContribution = (Math.log(Vec3.length(motion.velocity) / TOP_SPEED) + LOG_SCALER) / LOG_SCALER;
}
var angularContribution = Math.min(Math.abs(motion.yawDelta) / DELTA_YAW_MAX, 1);
leanRollProgress = linearContribution;
leanRollProgress *= angularContribution;
// shape the response curve
leanRollProgress = filter.bezier(leanRollProgress, {x: 1, y: 0}, {x: 1, y: 0});
// which way to lean?
var turnSign = (motion.yawDelta >= 0) ? 1 : -1;
if (motion.direction === BACKWARDS ||
motion.direction === LEFT) {
turnSign *= -1;
}
// filter progress
leanRollProgress = leanRollSmoothingFilter.process(turnSign * leanRollProgress);
return ROLL_MAX * leanRollProgress;
}
// animate the avatar using sine waves, geometric waveforms and harmonic generators
function renderMotion() {
// leaning in response to speed and acceleration
var leanPitch = motion.state === SURFACE_MOTION ? getLeanPitch() : 0;
var leanRoll = motion.state === STATIC ? 0 : getLeanRoll();
var lastDirection = motion.lastDirection;
// hips translations from currently playing animations
var hipsTranslations = {x:0, y:0, z:0};
if (motion.currentTransition !== nullTransition) {
// maintain previous direction when transitioning from a walk
if (motion.currentTransition.lastAnimation === avatar.selectedWalkBlend) {
motion.lastDirection = motion.currentTransition.lastDirection;
}
hipsTranslations = motion.currentTransition.blendTranslations(motion.frequencyTimeWheelPos,
motion.lastDirection);
} else {
hipsTranslations = animationOperations.calculateTranslations(avatar.currentAnimation,
motion.frequencyTimeWheelPos,
motion.direction);
}
// animation translations are calibrated for a 1m hips to feet height, so we adjust for this particular avatar
hipsTranslations = Vec3.multiply(hipsTranslations, avatar.calibration.hipsToFeet);
// factor any leaning into the hips offset
hipsTranslations.z += avatar.calibration.hipsToFeet * Math.sin(filter.degToRad(leanPitch));
hipsTranslations.x += avatar.calibration.hipsToFeet * Math.sin(filter.degToRad(leanRoll));
// ensure skeleton offsets are within the 1m limit
hipsTranslations.x = hipsTranslations.x > 1 ? 1 : hipsTranslations.x;
hipsTranslations.x = hipsTranslations.x < -1 ? -1 : hipsTranslations.x;
hipsTranslations.y = hipsTranslations.y > 1 ? 1 : hipsTranslations.y;
hipsTranslations.y = hipsTranslations.y < -1 ? -1 : hipsTranslations.y;
hipsTranslations.z = hipsTranslations.z > 1 ? 1 : hipsTranslations.z;
hipsTranslations.z = hipsTranslations.z < -1 ? -1 : hipsTranslations.z;
// apply translations
MyAvatar.setSkeletonOffset(hipsTranslations);
// play footsteps sound?
var producingFootstepSounds = (avatar.currentAnimation === avatar.selectedWalkBlend) && avatar.makesFootStepSounds;
// is there an animation in the transition that would make footstep sounds?
if ((motion.currentTransition !== nullTransition && avatar.makesFootStepSounds) ||
(motion.currentTransition.nextAnimation === avatar.selectedWalkBlend ||
motion.currentTransition.lastAnimation === avatar.selectedWalkBlend)) {
producingFootstepSounds = true;
}
if (producingFootstepSounds) {
var ftWheelPosition = motion.frequencyTimeWheelPos;
if (motion.currentTransition !== nullTransition &&
motion.currentTransition.lastAnimation === avatar.selectedWalkBlend) {
ftWheelPosition = motion.currentTransition.lastFrequencyTimeWheelPos;
}
if (avatar.nextStep === LEFT && ftWheelPosition > THREE_QUARTER_CYCLE) {
avatar.makeFootStepSound();
} else if (avatar.nextStep === RIGHT && (ftWheelPosition < THREE_QUARTER_CYCLE && ftWheelPosition > QUARTER_CYCLE)) {
avatar.makeFootStepSound();
}
}
// apply joint rotations
for (jointName in avatar.currentAnimation.joints) {
var joint = null;
var jointRotations = {x:0, y:0, z:0};
if (walkAssets.animationReference.joints[jointName]) {
joint = walkAssets.animationReference.joints[jointName];
}
// ignore arms / head / fingers rotations (dependant on options selected in the settings)
if (avatar.armsNotAnimated && (joint.IKChain === "LeftArm" || joint.IKChain === "RightArm" ||
joint.IKParent === "LeftHand" || joint.IKParent === "RightHand") ||
avatar.headNotAnimated && joint.IKChain === "Head") {
continue;
}
// if there's a live transition, blend the rotations with the last animation's rotations
if (motion.currentTransition !== nullTransition) {
jointRotations = motion.currentTransition.blendRotations(jointName,
motion.frequencyTimeWheelPos,
motion.lastDirection);
} else {
jointRotations = animationOperations.calculateRotations(jointName,
avatar.currentAnimation,
motion.frequencyTimeWheelPos,
motion.direction);
}
// apply any pre-state change reach poses
for (reachPose in motion.preReachPoses) {
var reachPoseStrength = motion.preReachPoses[reachPose].currentStrength();
var poseRotations = animationOperations.calculateRotations(jointName,
motion.preReachPoses[reachPose].animation,
motion.frequencyTimeWheelPos,
motion.direction);
// don't use Vec3 operations here, as if x,y or z is zero, the reach pose should not have any influence
if (Math.abs(poseRotations.x) > 0) {
jointRotations.x = reachPoseStrength * poseRotations.x + (1 - reachPoseStrength) * jointRotations.x;
}
if (Math.abs(poseRotations.y) > 0) {
jointRotations.y = reachPoseStrength * poseRotations.y + (1 - reachPoseStrength) * jointRotations.y;
}
if (Math.abs(poseRotations.z) > 0) {
jointRotations.z = reachPoseStrength * poseRotations.z + (1 - reachPoseStrength) * jointRotations.z;
}
}
// apply angular velocity and speed induced leaning
if (jointName === "Hips") {
jointRotations.x += leanPitch;
jointRotations.z += leanRoll;
}
// apply pre-rotations?
var rotationsQ = Quat.fromVec3Degrees(jointRotations);
if (avatar.isUsingHiFiPreRotations) {
//var jointNumber = MyAvatar.getJointIndex(jointName);
var jointNumber = walkAssets.animationReference.joints[jointName].number;
if (jointNumber > 0) {
var preRotationsQ = MyAvatar.getDefaultJointRotation(jointNumber);
rotationsQ = Quat.multiply(preRotationsQ, rotationsQ);
}
}
// apply rotations
MyAvatar.setJointRotation(jointName, rotationsQ);
}
}