mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 21:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			164 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| local camera = require('openmw.camera')
 | |
| local util = require('openmw.util')
 | |
| local self = require('openmw.self')
 | |
| local nearby = require('openmw.nearby')
 | |
| local async = require('openmw.async')
 | |
| local storage = require('openmw.storage')
 | |
| 
 | |
| local Actor = require('openmw.types').Actor
 | |
| 
 | |
| local settings = storage.playerSection('SettingsOMWCameraThirdPerson')
 | |
| 
 | |
| local MODE = camera.MODE
 | |
| local STATE = { RightShoulder = 0, LeftShoulder = 1, Combat = 2, Swimming = 3 }
 | |
| 
 | |
| local M = {
 | |
|     baseDistance = 192,
 | |
|     preferredDistance = 0,
 | |
|     standingPreview = false,
 | |
|     noOffsetControl = {},
 | |
| }
 | |
| 
 | |
| local viewOverShoulder, autoSwitchShoulder
 | |
| local shoulderOffset
 | |
| local zoomOutWhenMoveCoef
 | |
| 
 | |
| local defaultShoulder, rightShoulderOffset, leftShoulderOffset
 | |
| local combatOffset = util.vector2(0, 15)
 | |
| 
 | |
| local noThirdPersonLastFrame = true
 | |
| 
 | |
| local function updateSettings()
 | |
|     viewOverShoulder = settings:get('viewOverShoulder')
 | |
|     autoSwitchShoulder = settings:get('autoSwitchShoulder')
 | |
|     shoulderOffset = util.vector2(settings:get('shoulderOffsetX'),
 | |
|         settings:get('shoulderOffsetY'))
 | |
|     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))
 | |
| 
 | |
| local state = defaultShoulder
 | |
| 
 | |
| local function ray(from, angle, limit)
 | |
|     local to = from + util.transform.rotateZ(angle) * util.vector3(0, limit, 0)
 | |
|     local res = nearby.castRay(from, to, { collisionType = camera.getCollisionType() })
 | |
|     if res.hit then
 | |
|         return (res.hitPos - from):length()
 | |
|     else
 | |
|         return limit
 | |
|     end
 | |
| end
 | |
| 
 | |
| local function trySwitchShoulder()
 | |
|     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
 | |
| 
 | |
|     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
 | |
|         + smoothedSpeedSqr / (smoothedSpeedSqr + 300 * 300) * zoomOutWhenMoveCoef)
 | |
| end
 | |
| 
 | |
| local function updateState()
 | |
|     local mode = camera.getMode()
 | |
|     local oldState = state
 | |
|     if Actor.getStance(self) ~= Actor.STANCE.Nothing and mode == MODE.ThirdPerson then
 | |
|         state = STATE.Combat
 | |
|     elseif Actor.isSwimming(self) then
 | |
|         state = STATE.Swimming
 | |
|     elseif oldState == STATE.Combat or oldState == STATE.Swimming then
 | |
|         state = defaultShoulder
 | |
|     elseif not state then
 | |
|         state = defaultShoulder
 | |
|     end
 | |
|     if (mode == MODE.ThirdPerson or Actor.getCurrentSpeed(self) > 0 or state ~= oldState or noThirdPersonLastFrame)
 | |
|         and (state == STATE.LeftShoulder or state == STATE.RightShoulder) then
 | |
|         if autoSwitchShoulder then
 | |
|             trySwitchShoulder()
 | |
|         else
 | |
|             state = defaultShoulder
 | |
|         end
 | |
|     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
 | |
|             (mode ~= MODE.Preview or M.standingPreview) then
 | |
|             -- Transition to/from combat mode and we are not in preview mode. Should be fast.
 | |
|             camera.setFocalTransitionSpeed(5.0)
 | |
|         else
 | |
|             camera.setFocalTransitionSpeed(1.0) -- Default transition speed.
 | |
|         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)
 | |
|         if noThirdPersonLastFrame then
 | |
|             camera.setFocalPreferredOffset(util.vector2(0, 0))
 | |
|             camera.instantTransition()
 | |
|             noThirdPersonLastFrame = false
 | |
|         end
 | |
|         return
 | |
|     end
 | |
| 
 | |
|     if not next(M.noOffsetControl) then
 | |
|         updateState()
 | |
|     else
 | |
|         state = nil
 | |
|     end
 | |
| 
 | |
|     M.preferredDistance = calculateDistance(smoothedSpeed)
 | |
|     if noThirdPersonLastFrame then -- just switched to third person view
 | |
|         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
 |