2021-06-14 17:58:04 +00:00
|
|
|
local camera = require('openmw.camera')
|
|
|
|
local util = require('openmw.util')
|
|
|
|
local self = require('openmw.self')
|
|
|
|
local nearby = require('openmw.nearby')
|
2022-05-14 11:37:32 +00:00
|
|
|
local async = require('openmw.async')
|
2024-01-10 19:32:34 +00:00
|
|
|
local storage = require('openmw.storage')
|
2021-06-14 17:58:04 +00:00
|
|
|
|
2022-02-15 18:38:47 +00:00
|
|
|
local Actor = require('openmw.types').Actor
|
|
|
|
|
2024-01-10 19:32:34 +00:00
|
|
|
local settings = storage.playerSection('SettingsOMWCameraThirdPerson')
|
2022-05-14 11:37:32 +00:00
|
|
|
|
2021-06-14 17:58:04 +00:00
|
|
|
local MODE = camera.MODE
|
|
|
|
local STATE = { RightShoulder = 0, LeftShoulder = 1, Combat = 2, Swimming = 3 }
|
|
|
|
|
|
|
|
local M = {
|
2022-05-14 11:37:32 +00:00
|
|
|
baseDistance = 192,
|
2021-06-14 17:58:04 +00:00
|
|
|
preferredDistance = 0,
|
|
|
|
standingPreview = false,
|
2023-09-10 14:12:26 +00:00
|
|
|
noOffsetControl = {},
|
2021-06-14 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
2022-05-14 11:37:32 +00:00
|
|
|
local viewOverShoulder, autoSwitchShoulder
|
|
|
|
local shoulderOffset
|
|
|
|
local zoomOutWhenMoveCoef
|
2021-06-14 17:58:04 +00:00
|
|
|
|
2022-05-14 11:37:32 +00:00
|
|
|
local defaultShoulder, rightShoulderOffset, leftShoulderOffset
|
2021-06-14 17:58:04 +00:00
|
|
|
local combatOffset = util.vector2(0, 15)
|
|
|
|
|
2022-05-14 11:37:32 +00:00
|
|
|
local noThirdPersonLastFrame = true
|
|
|
|
|
|
|
|
local function updateSettings()
|
|
|
|
viewOverShoulder = settings:get('viewOverShoulder')
|
|
|
|
autoSwitchShoulder = settings:get('autoSwitchShoulder')
|
|
|
|
shoulderOffset = util.vector2(settings:get('shoulderOffsetX'),
|
2024-01-10 19:32:34 +00:00
|
|
|
settings:get('shoulderOffsetY'))
|
2022-05-14 11:37:32 +00:00
|
|
|
zoomOutWhenMoveCoef = settings:get('zoomOutWhenMoveCoef')
|
|
|
|
|
|
|
|
defaultShoulder = (shoulderOffset.x > 0 and STATE.RightShoulder) or STATE.LeftShoulder
|
|
|
|
rightShoulderOffset = util.vector2(math.abs(shoulderOffset.x), shoulderOffset.y)
|
|
|
|
leftShoulderOffset = util.vector2(-math.abs(shoulderOffset.x), shoulderOffset.y)
|
|
|
|
noThirdPersonLastFrame = true
|
|
|
|
end
|
|
|
|
updateSettings()
|
|
|
|
settings:subscribe(async:callback(updateSettings))
|
|
|
|
|
2021-06-14 17:58:04 +00:00
|
|
|
local state = defaultShoulder
|
|
|
|
|
|
|
|
local function ray(from, angle, limit)
|
|
|
|
local to = from + util.transform.rotateZ(angle) * util.vector3(0, limit, 0)
|
2024-01-10 19:32:34 +00:00
|
|
|
local res = nearby.castRay(from, to, { collisionType = camera.getCollisionType() })
|
2021-06-14 17:58:04 +00:00
|
|
|
if res.hit then
|
|
|
|
return (res.hitPos - from):length()
|
|
|
|
else
|
|
|
|
return limit
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function trySwitchShoulder()
|
2024-01-10 19:32:34 +00:00
|
|
|
local limitToSwitch = 120 -- switch to other shoulder if wall is closer than this limit
|
|
|
|
local limitToSwitchBack = 300 -- switch back to default shoulder if there is no walls at this distance
|
2021-06-14 17:58:04 +00:00
|
|
|
|
|
|
|
local pos = camera.getTrackedPosition()
|
|
|
|
local rayRight = ray(pos, camera.getYaw() + math.rad(90), limitToSwitchBack + 1)
|
|
|
|
local rayLeft = ray(pos, camera.getYaw() - math.rad(90), limitToSwitchBack + 1)
|
|
|
|
local rayRightForward = ray(pos, camera.getYaw() + math.rad(30), limitToSwitchBack + 1)
|
|
|
|
local rayLeftForward = ray(pos, camera.getYaw() - math.rad(30), limitToSwitchBack + 1)
|
|
|
|
|
|
|
|
local distRight = math.min(rayRight, rayRightForward)
|
|
|
|
local distLeft = math.min(rayLeft, rayLeftForward)
|
|
|
|
|
|
|
|
if distLeft < limitToSwitch and distRight > limitToSwitchBack then
|
|
|
|
state = STATE.RightShoulder
|
|
|
|
elseif distRight < limitToSwitch and distLeft > limitToSwitchBack then
|
|
|
|
state = STATE.LeftShoulder
|
|
|
|
elseif distRight > limitToSwitchBack and distLeft > limitToSwitchBack then
|
|
|
|
state = defaultShoulder
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function calculateDistance(smoothedSpeed)
|
|
|
|
local smoothedSpeedSqr = smoothedSpeed * smoothedSpeed
|
|
|
|
return (M.baseDistance + math.max(camera.getPitch(), 0) * 50
|
2024-01-10 19:32:34 +00:00
|
|
|
+ smoothedSpeedSqr / (smoothedSpeedSqr + 300 * 300) * zoomOutWhenMoveCoef)
|
2021-06-14 17:58:04 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
local function updateState()
|
|
|
|
local mode = camera.getMode()
|
|
|
|
local oldState = state
|
2023-05-07 22:30:24 +00:00
|
|
|
if Actor.getStance(self) ~= Actor.STANCE.Nothing and mode == MODE.ThirdPerson then
|
2021-06-14 17:58:04 +00:00
|
|
|
state = STATE.Combat
|
2022-02-15 18:38:47 +00:00
|
|
|
elseif Actor.isSwimming(self) then
|
2021-06-14 17:58:04 +00:00
|
|
|
state = STATE.Swimming
|
|
|
|
elseif oldState == STATE.Combat or oldState == STATE.Swimming then
|
|
|
|
state = defaultShoulder
|
2022-04-03 18:42:19 +00:00
|
|
|
elseif not state then
|
|
|
|
state = defaultShoulder
|
2021-06-14 17:58:04 +00:00
|
|
|
end
|
2023-05-07 22:30:24 +00:00
|
|
|
if (mode == MODE.ThirdPerson or Actor.getCurrentSpeed(self) > 0 or state ~= oldState or noThirdPersonLastFrame)
|
2024-01-10 19:32:34 +00:00
|
|
|
and (state == STATE.LeftShoulder or state == STATE.RightShoulder) then
|
2022-05-14 11:37:32 +00:00
|
|
|
if autoSwitchShoulder then
|
|
|
|
trySwitchShoulder()
|
|
|
|
else
|
|
|
|
state = defaultShoulder
|
|
|
|
end
|
2021-06-14 17:58:04 +00:00
|
|
|
end
|
|
|
|
if oldState ~= state or noThirdPersonLastFrame then
|
|
|
|
-- State was changed, start focal point transition.
|
|
|
|
if mode == MODE.Vanity then
|
|
|
|
-- Player doesn't touch controls for a long time. Transition should be very slow.
|
|
|
|
camera.setFocalTransitionSpeed(0.2)
|
|
|
|
elseif (oldState == STATE.Combat or state == STATE.Combat) and
|
2024-01-10 19:32:34 +00:00
|
|
|
(mode ~= MODE.Preview or M.standingPreview) then
|
2021-06-14 17:58:04 +00:00
|
|
|
-- Transition to/from combat mode and we are not in preview mode. Should be fast.
|
|
|
|
camera.setFocalTransitionSpeed(5.0)
|
|
|
|
else
|
2024-01-10 19:32:34 +00:00
|
|
|
camera.setFocalTransitionSpeed(1.0) -- Default transition speed.
|
2021-06-14 17:58:04 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if state == STATE.RightShoulder then
|
|
|
|
camera.setFocalPreferredOffset(rightShoulderOffset)
|
|
|
|
elseif state == STATE.LeftShoulder then
|
|
|
|
camera.setFocalPreferredOffset(leftShoulderOffset)
|
|
|
|
else
|
|
|
|
camera.setFocalPreferredOffset(combatOffset)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function M.update(dt, smoothedSpeed)
|
|
|
|
local mode = camera.getMode()
|
|
|
|
if mode == MODE.FirstPerson or mode == MODE.Static then
|
|
|
|
noThirdPersonLastFrame = true
|
|
|
|
return
|
|
|
|
end
|
|
|
|
if not viewOverShoulder then
|
|
|
|
M.preferredDistance = M.baseDistance
|
|
|
|
camera.setPreferredThirdPersonDistance(M.baseDistance)
|
2022-05-14 11:37:32 +00:00
|
|
|
if noThirdPersonLastFrame then
|
|
|
|
camera.setFocalPreferredOffset(util.vector2(0, 0))
|
|
|
|
camera.instantTransition()
|
|
|
|
noThirdPersonLastFrame = false
|
|
|
|
end
|
2021-06-14 17:58:04 +00:00
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2023-09-10 14:12:26 +00:00
|
|
|
if not next(M.noOffsetControl) then
|
2021-06-14 17:58:04 +00:00
|
|
|
updateState()
|
|
|
|
else
|
|
|
|
state = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
M.preferredDistance = calculateDistance(smoothedSpeed)
|
2024-01-10 19:32:34 +00:00
|
|
|
if noThirdPersonLastFrame then -- just switched to third person view
|
2021-06-14 17:58:04 +00:00
|
|
|
camera.setPreferredThirdPersonDistance(M.preferredDistance)
|
|
|
|
camera.instantTransition()
|
|
|
|
noThirdPersonLastFrame = false
|
|
|
|
else
|
|
|
|
local maxIncrease = dt * (100 + M.baseDistance)
|
|
|
|
camera.setPreferredThirdPersonDistance(math.min(
|
|
|
|
M.preferredDistance, camera.getThirdPersonDistance() + maxIncrease))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return M
|