2021-06-14 17:58:04 +00:00
|
|
|
local camera = require('openmw.camera')
|
|
|
|
local self = require('openmw.self')
|
|
|
|
local util = require('openmw.util')
|
2022-05-14 11:37:32 +00:00
|
|
|
local async = require('openmw.async')
|
2021-06-14 17:58:04 +00:00
|
|
|
|
2022-02-15 18:38:47 +00:00
|
|
|
local Actor = require('openmw.types').Actor
|
|
|
|
|
2022-05-14 11:37:32 +00:00
|
|
|
local M = {}
|
|
|
|
|
|
|
|
local settings = require('scripts.omw.camera.settings').headBobbing
|
|
|
|
|
|
|
|
local doubleStepLength, stepHeight, maxRoll
|
|
|
|
|
|
|
|
local function updateSettings()
|
|
|
|
M.enabled = settings:get('enabled')
|
|
|
|
doubleStepLength = settings:get('step') * 2
|
|
|
|
stepHeight = settings:get('height')
|
|
|
|
maxRoll = math.rad(settings:get('roll'))
|
|
|
|
end
|
|
|
|
|
|
|
|
updateSettings()
|
|
|
|
settings:subscribe(async:callback(updateSettings))
|
2021-06-14 17:58:04 +00:00
|
|
|
|
|
|
|
local effectWeight = 0
|
|
|
|
local totalMovement = 0
|
|
|
|
|
|
|
|
-- Trajectory of each step is a scaled arc of 60 degrees.
|
|
|
|
local halfArc = math.rad(30)
|
|
|
|
local sampleArc = function(x) return 1 - math.cos(x * halfArc) end
|
|
|
|
local arcHeight = sampleArc(1)
|
|
|
|
|
|
|
|
function M.update(dt, smoothedSpeed)
|
2022-02-15 18:38:47 +00:00
|
|
|
local speed = Actor.currentSpeed(self)
|
2021-06-14 17:58:04 +00:00
|
|
|
speed = speed / (1 + speed / 500) -- limit bobbing frequency if the speed is very high
|
|
|
|
totalMovement = totalMovement + speed * dt
|
|
|
|
if not M.enabled or camera.getMode() ~= camera.MODE.FirstPerson then
|
|
|
|
effectWeight = 0
|
|
|
|
return
|
|
|
|
end
|
2022-02-15 18:38:47 +00:00
|
|
|
if Actor.isOnGround(self) then
|
2021-06-14 17:58:04 +00:00
|
|
|
effectWeight = math.min(1, effectWeight + dt * 5)
|
|
|
|
else
|
|
|
|
effectWeight = math.max(0, effectWeight - dt * 5)
|
|
|
|
end
|
|
|
|
|
|
|
|
local doubleStepState = totalMovement / doubleStepLength
|
|
|
|
doubleStepState = doubleStepState - math.floor(doubleStepState) -- from 0 to 1 during 2 steps
|
|
|
|
local stepState = math.abs(doubleStepState * 4 - 2) - 1 -- from -1 to 1 on even steps and from 1 to -1 on odd steps
|
|
|
|
local effect = sampleArc(stepState) / arcHeight -- range from 0 to 1
|
|
|
|
|
|
|
|
-- Smoothly reduce the effect to zero when the player stops
|
|
|
|
local coef = math.min(smoothedSpeed / 300, 1) * effectWeight
|
|
|
|
|
|
|
|
local zOffset = (0.5 - effect) * coef * stepHeight -- range from -stepHeight/2 to stepHeight/2
|
|
|
|
local roll = ((stepState > 0 and 1) or -1) * effect * coef * maxRoll -- range from -maxRoll to maxRoll
|
|
|
|
camera.setFirstPersonOffset(camera.getFirstPersonOffset() + util.vector3(0, 0, zOffset))
|
2022-05-12 20:57:00 +00:00
|
|
|
camera.setExtraRoll(camera.getExtraRoll() + roll)
|
2021-06-14 17:58:04 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return M
|
|
|
|
|