1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-24 12:56:36 +00:00
openmw/files/data/scripts/omw/mechanics/animationcontroller.lua
2025-08-01 04:18:03 +03:00

156 lines
5.7 KiB
Lua

local anim = require('openmw.animation')
local self = require('openmw.self')
local playBlendedHandlers = {}
local function onPlayBlendedAnimation(groupname, options)
for i = #playBlendedHandlers, 1, -1 do
if playBlendedHandlers[i](groupname, options) == false then
return
end
end
end
local function playBlendedAnimation(groupname, options)
onPlayBlendedAnimation(groupname, options)
if options.skip then
return
end
anim.playBlended(self, groupname, options)
end
local textKeyHandlers = {}
local function onAnimationTextKey(groupname, key)
local handlers = textKeyHandlers[groupname]
if handlers then
for i = #handlers, 1, -1 do
if handlers[i](groupname, key) == false then
return
end
end
end
handlers = textKeyHandlers['']
if handlers then
for i = #handlers, 1, -1 do
if handlers[i](groupname, key) == false then
return
end
end
end
end
local initialized = false
local function onUpdate(dt)
if dt <= 0 then
return
end
-- The script is loaded before the actor's CharacterController object is initialized, therefore
-- we have to delay this initialization step or the call won't have any effect.
if not initialized then
self:_enableLuaAnimations(true)
initialized = true
end
end
return {
engineHandlers = {
_onPlayAnimation = playBlendedAnimation,
_onAnimationTextKey = onAnimationTextKey,
onUpdate = onUpdate,
},
interfaceName = 'AnimationController',
---
-- Animation controller interface
-- @module AnimationController
-- @context local
-- @usage local anim = require('openmw.animation')
-- local I = require('openmw.interfaces')
--
-- -- play spellcast animation
-- I.AnimationController.playBlendedAnimation('spellcast', { startkey = 'self start', stopkey = 'self stop', priority = {
-- [anim.BONE_GROUP.RightArm] = anim.PRIORITY.Weapon,
-- [anim.BONE_GROUP.LeftArm] = anim.PRIORITY.Weapon,
-- [anim.BONE_GROUP.Torso] = anim.PRIORITY.Weapon,
-- [anim.BONE_GROUP.LowerBody] = anim.PRIORITY.WeaponLowerBody
-- } })
--
-- @usage -- react to the spellcast release textkey
-- I.AnimationController.addTextKeyHandler('spellcast', function(groupname, key)
-- -- Note, Lua is 1-indexed so have to subtract 1 less than the length of 'release'
-- if key.sub(key, #key - 6) == 'release' then
-- print('Abra kadabra!')
-- end
-- end)
--
-- @usage -- Add a text key handler that will react to all keys
-- I.AnimationController.addTextKeyHandler('', function(groupname, key)
-- if key.sub(key, #key - 2) == 'hit' and not key.sub(key, #key - 7) == ' min hit' then
-- print('Hit!')
-- end
-- end)
--
-- @usage -- Make a handler that changes player attack speed based on current fatigue
-- I.AnimationController.addPlayBlendedAnimationHandler(function (groupname, options)
-- local stop = options.stopkey
-- if #stop > 10 and stop.sub(stop, #stop - 10) == ' max attack' then
-- -- This is an attack wind up animation, scale its speed by attack
-- local fatigue = Actor.stats.dynamic.fatigue(self)
-- local factor = 1 - fatigue.current / fatigue.base
-- speed = 1 - factor * 0.8
-- options.speed = speed
-- end
-- end)
--
interface = {
--- Interface version
-- @field [parent=#AnimationController] #number version
version = 0,
--- AnimationController Package
-- @type Package
--- Make this actor play an animation. Makes a call to @{openmw.animation#playBlended}, after invoking handlers added through addPlayBlendedAnimationHandler
-- @function [parent=#AnimationController] playBlendedAnimation
-- @param #string groupname The animation group to be played
-- @param #table options The table of play options that will be passed to @{openmw.animation#playBlended}
playBlendedAnimation = playBlendedAnimation,
--- Add new playBlendedAnimation handler for this actor
-- If `handler(groupname, options)` returns false, other handlers for
-- the call will be skipped.
-- @function [parent=#AnimationController] addPlayBlendedAnimationHandler
-- @param #function handler The handler.
addPlayBlendedAnimationHandler = function(handler)
playBlendedHandlers[#playBlendedHandlers + 1] = handler
end,
--- Add new text key handler for this actor
-- While playing, some animations emit text key events. Register a handle to listen for all
-- text key events associated with this actor's animations.
-- If `handler(groupname, key)` returns false, other handlers for
-- the call will be skipped.
-- @function [parent=#AnimationController] addTextKeyHandler
-- @param #string groupname Name of the animation group to listen to keys for. If the empty string or nil, all keys will be received
-- @param #function handler The handler.
addTextKeyHandler = function(groupname, handler)
if not groupname then
groupname = ""
end
local handlers = textKeyHandlers[groupname]
if handlers == nil then
handlers = {}
textKeyHandlers[groupname] = handlers
end
handlers[#handlers + 1] = handler
end,
},
eventHandlers = {
AddVfx = function(data)
anim.addVfx(self, data.model, data.options)
end,
}
}