mirror of https://github.com/OpenMW/openmw.git
Move some of player controls logic from C++ to Lua
parent
80e2cd79ec
commit
1869aeae5c
@ -0,0 +1,6 @@
|
||||
Interface Controls
|
||||
==================
|
||||
|
||||
.. raw:: html
|
||||
:file: generated_html/scripts_omw_playercontrols.html
|
||||
|
@ -0,0 +1,16 @@
|
||||
ControlsPage: "OpenMW Controls"
|
||||
ControlsPageDescription: "Additional settings related to player controls"
|
||||
|
||||
MovementSettings: "Movement"
|
||||
|
||||
alwaysRun: "Always run"
|
||||
alwaysRunDescription: |
|
||||
If this setting is true, the character is running by default, otherwise the character is walking by default.
|
||||
The shift key will temporarily invert this setting, and the caps lock key will invert this setting while it's "locked".
|
||||
|
||||
toggleSneak: "Toggle sneak"
|
||||
toggleSneakDescription: |
|
||||
This setting causes the behavior of the sneak key (bound to Ctrl by default)
|
||||
to toggle sneaking on and off rather than requiring the key to be held down while sneaking.
|
||||
Players that spend significant time sneaking may find the character easier to control with this option enabled.
|
||||
|
@ -0,0 +1,189 @@
|
||||
local core = require('openmw.core')
|
||||
local input = require('openmw.input')
|
||||
local self = require('openmw.self')
|
||||
local util = require('openmw.util')
|
||||
local ui = require('openmw.ui')
|
||||
local Actor = require('openmw.types').Actor
|
||||
local Player = require('openmw.types').Player
|
||||
|
||||
local storage = require('openmw.storage')
|
||||
local I = require('openmw.interfaces')
|
||||
|
||||
local settingsGroup = 'SettingsOMWControls'
|
||||
|
||||
local function boolSetting(key, default)
|
||||
return {
|
||||
key = key,
|
||||
renderer = 'checkbox',
|
||||
name = key,
|
||||
description = key..'Description',
|
||||
default = default,
|
||||
}
|
||||
end
|
||||
|
||||
I.Settings.registerPage({
|
||||
key = 'OMWControls',
|
||||
l10n = 'OMWControls',
|
||||
name = 'ControlsPage',
|
||||
description = 'ControlsPageDescription',
|
||||
})
|
||||
|
||||
I.Settings.registerGroup({
|
||||
key = settingsGroup,
|
||||
page = 'OMWControls',
|
||||
l10n = 'OMWControls',
|
||||
name = 'MovementSettings',
|
||||
permanentStorage = true,
|
||||
settings = {
|
||||
boolSetting('alwaysRun', false),
|
||||
boolSetting('toggleSneak', false),
|
||||
},
|
||||
})
|
||||
|
||||
local settings = storage.playerSection(settingsGroup)
|
||||
|
||||
local attemptJump = false
|
||||
local startAttack = false
|
||||
local autoMove = false
|
||||
local movementControlsOverridden = false
|
||||
local combatControlsOverridden = false
|
||||
|
||||
local function processMovement()
|
||||
local controllerMovement = -input.getAxisValue(input.CONTROLLER_AXIS.MoveForwardBackward)
|
||||
local controllerSideMovement = input.getAxisValue(input.CONTROLLER_AXIS.MoveLeftRight)
|
||||
if controllerMovement ~= 0 or controllerSideMovement ~= 0 then
|
||||
-- controller movement
|
||||
if util.vector2(controllerMovement, controllerSideMovement):length2() < 0.25
|
||||
and not self.controls.sneak and Actor.isOnGround(self) and not Actor.isSwimming(self) then
|
||||
self.controls.run = false
|
||||
self.controls.movement = controllerMovement * 2
|
||||
self.controls.sideMovement = controllerSideMovement * 2
|
||||
else
|
||||
self.controls.run = true
|
||||
self.controls.movement = controllerMovement
|
||||
self.controls.sideMovement = controllerSideMovement
|
||||
end
|
||||
else
|
||||
-- keyboard movement
|
||||
self.controls.movement = 0
|
||||
self.controls.sideMovement = 0
|
||||
if input.isActionPressed(input.ACTION.MoveLeft) then
|
||||
self.controls.sideMovement = self.controls.sideMovement - 1
|
||||
end
|
||||
if input.isActionPressed(input.ACTION.MoveRight) then
|
||||
self.controls.sideMovement = self.controls.sideMovement + 1
|
||||
end
|
||||
if input.isActionPressed(input.ACTION.MoveBackward) then
|
||||
self.controls.movement = self.controls.movement - 1
|
||||
end
|
||||
if input.isActionPressed(input.ACTION.MoveForward) then
|
||||
self.controls.movement = self.controls.movement + 1
|
||||
end
|
||||
self.controls.run = input.isActionPressed(input.ACTION.Run) ~= settings:get('alwaysRun')
|
||||
end
|
||||
if self.controls.movement ~= 0 or not Actor.canMove(self) then
|
||||
autoMove = false
|
||||
elseif autoMove then
|
||||
self.controls.movement = 1
|
||||
end
|
||||
self.controls.jump = attemptJump and input.getControlSwitch(input.CONTROL_SWITCH.Jumping)
|
||||
if not settings:get('toggleSneak') then
|
||||
self.controls.sneak = input.isActionPressed(input.ACTION.Sneak)
|
||||
end
|
||||
end
|
||||
|
||||
local function processAttacking()
|
||||
if startAttack then
|
||||
self.controls.use = 1
|
||||
elseif Actor.stance(self) == Actor.STANCE.Spell then
|
||||
self.controls.use = 0
|
||||
elseif input.getAxisValue(input.CONTROLLER_AXIS.TriggerRight) < 0.6
|
||||
and not input.isActionPressed(input.ACTION.Use) then
|
||||
-- The value "0.6" shouldn't exceed the triggering threshold in BindingsManager::actionValueChanged.
|
||||
-- TODO: Move more logic from BindingsManager to Lua and consider to make this threshold configurable.
|
||||
self.controls.use = 0
|
||||
end
|
||||
end
|
||||
|
||||
local function onFrame(dt)
|
||||
controlsAllowed = input.getControlSwitch(input.CONTROL_SWITCH.Controls) and not core.isWorldPaused()
|
||||
if not movementControlsOverridden then
|
||||
if controlsAllowed then
|
||||
processMovement()
|
||||
else
|
||||
self.controls.movement = 0
|
||||
self.controls.sideMovement = 0
|
||||
self.controls.jump = false
|
||||
end
|
||||
end
|
||||
if controlsAllowed and not combatControlsOverridden then
|
||||
processAttacking()
|
||||
end
|
||||
attemptJump = false
|
||||
startAttack = false
|
||||
end
|
||||
|
||||
local function onInputAction(action)
|
||||
if core.isWorldPaused() or not input.getControlSwitch(input.CONTROL_SWITCH.Controls) then
|
||||
return
|
||||
end
|
||||
|
||||
if action == input.ACTION.Jump then
|
||||
attemptJump = true
|
||||
elseif action == input.ACTION.Use then
|
||||
startAttack = true
|
||||
elseif action == input.ACTION.AutoMove and not movementControlsOverridden then
|
||||
autoMove = true
|
||||
elseif action == input.ACTION.AlwaysRun and not movementControlsOverridden then
|
||||
settings:set('alwaysRun', not settings:get('alwaysRun'))
|
||||
elseif action == input.ACTION.Sneak and not movementControlsOverridden then
|
||||
if settings:get('toggleSneak') then
|
||||
self.controls.sneak = not self.controls.sneak
|
||||
end
|
||||
elseif action == input.ACTION.ToggleSpell and not combatControlsOverridden then
|
||||
if Actor.stance(self) == Actor.STANCE.Spell then
|
||||
Actor.setStance(self, Actor.STANCE.Nothing)
|
||||
elseif input.getControlSwitch(input.CONTROL_SWITCH.Magic) then
|
||||
if Player.isWerewolf(self) then
|
||||
ui.showMessage(core.getGMST('sWerewolfRefusal'))
|
||||
else
|
||||
Actor.setStance(self, Actor.STANCE.Spell)
|
||||
end
|
||||
end
|
||||
elseif action == input.ACTION.ToggleWeapon and not combatControlsOverridden then
|
||||
if Actor.stance(self) == Actor.STANCE.Weapon then
|
||||
Actor.setStance(self, Actor.STANCE.Nothing)
|
||||
elseif input.getControlSwitch(input.CONTROL_SWITCH.Fighting) then
|
||||
Actor.setStance(self, Actor.STANCE.Weapon)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
return {
|
||||
engineHandlers = {
|
||||
onFrame = onFrame,
|
||||
onInputAction = onInputAction,
|
||||
},
|
||||
interfaceName = 'Controls',
|
||||
---
|
||||
-- @module Controls
|
||||
-- @usage require('openmw.interfaces').Controls
|
||||
interface = {
|
||||
--- Interface version
|
||||
-- @field [parent=#Controls] #number version
|
||||
version = 0,
|
||||
|
||||
--- When set to true then the movement controls including jump and sneak are not processed and can be handled by another script.
|
||||
-- If movement should be dissallowed completely, consider to use `input.setControlSwitch` instead.
|
||||
-- @function [parent=#Controls] overrideMovementControls
|
||||
-- @param #boolean value
|
||||
overrideMovementControls = function(v) movementControlsOverridden = v end,
|
||||
|
||||
--- When set to true then the controls "attack", "toggle spell", "toggle weapon" are not processed and can be handled by another script.
|
||||
-- If combat should be dissallowed completely, consider to use `input.setControlSwitch` instead.
|
||||
-- @function [parent=#Controls] overrideCombatControls
|
||||
-- @param #boolean value
|
||||
overrideCombatControls = function(v) combatControlsOverridden = v end,
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1 @@
|
||||
# It is an empty file that overrides builtin.omwscripts and disables builtin scripts
|
Loading…
Reference in New Issue