2021-07-10 13:43:53 +02:00
|
|
|
local testing = require('testing_util')
|
|
|
|
local self = require('openmw.self')
|
|
|
|
local util = require('openmw.util')
|
|
|
|
local core = require('openmw.core')
|
|
|
|
local input = require('openmw.input')
|
|
|
|
local types = require('openmw.types')
|
2022-07-10 17:27:00 +02:00
|
|
|
local nearby = require('openmw.nearby')
|
2021-07-10 13:43:53 +02:00
|
|
|
|
2023-10-10 00:23:30 +02:00
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.Fighting, false)
|
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.Jumping, false)
|
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.Looking, false)
|
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.Magic, false)
|
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.VanityMode, false)
|
|
|
|
types.Player.setControlSwitch(self, types.Player.CONTROL_SWITCH.ViewMode, false)
|
2021-07-10 13:43:53 +02:00
|
|
|
|
2024-06-22 02:58:30 +02:00
|
|
|
local function rotate(object, targetPitch, targetYaw)
|
|
|
|
local endTime = core.getSimulationTime() + 1
|
|
|
|
while core.getSimulationTime() < endTime do
|
|
|
|
object.controls.jump = false
|
|
|
|
object.controls.run = true
|
|
|
|
object.controls.movement = 0
|
|
|
|
object.controls.sideMovement = 0
|
|
|
|
if targetPitch ~= nil then
|
|
|
|
object.controls.pitchChange = util.normalizeAngle(targetPitch - object.rotation:getPitch()) * 0.5
|
|
|
|
end
|
|
|
|
if targetYaw ~= nil then
|
|
|
|
object.controls.yawChange = util.normalizeAngle(targetYaw - object.rotation:getYaw()) * 0.5
|
2021-07-10 13:43:53 +02:00
|
|
|
end
|
2024-06-22 02:58:30 +02:00
|
|
|
coroutine.yield()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function rotateByYaw(object, target)
|
|
|
|
rotate(object, nil, target)
|
|
|
|
end
|
|
|
|
|
|
|
|
local function rotateByPitch(object, target)
|
|
|
|
rotate(object, target, nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
testing.registerLocalTest('playerYawRotation',
|
|
|
|
function()
|
|
|
|
local initialAlphaXZ, initialGammaXZ = self.rotation:getAnglesXZ()
|
|
|
|
local initialAlphaZYX, initialBetaZYX, initialGammaZYX = self.rotation:getAnglesZYX()
|
|
|
|
|
|
|
|
local targetYaw = math.rad(90)
|
|
|
|
rotateByYaw(self, targetYaw)
|
|
|
|
|
|
|
|
testing.expectEqualWithDelta(self.rotation:getYaw(), targetYaw, 0.05, 'Incorrect yaw rotation')
|
|
|
|
|
|
|
|
local alpha1, gamma1 = self.rotation:getAnglesXZ()
|
|
|
|
testing.expectEqualWithDelta(alpha1, initialAlphaXZ, 0.05, 'Alpha rotation in XZ convention should not change')
|
|
|
|
testing.expectEqualWithDelta(gamma1, targetYaw, 0.05, 'Incorrect gamma rotation in XZ convention')
|
|
|
|
|
|
|
|
local alpha2, beta2, gamma2 = self.rotation:getAnglesZYX()
|
|
|
|
testing.expectEqualWithDelta(alpha2, targetYaw, 0.05, 'Incorrect alpha rotation in ZYX convention')
|
|
|
|
testing.expectEqualWithDelta(beta2, initialBetaZYX, 0.05, 'Beta rotation in ZYX convention should not change')
|
|
|
|
testing.expectEqualWithDelta(gamma2, initialGammaZYX, 0.05, 'Gamma rotation in ZYX convention should not change')
|
|
|
|
end)
|
|
|
|
|
|
|
|
testing.registerLocalTest('playerPitchRotation',
|
|
|
|
function()
|
|
|
|
local initialAlphaXZ, initialGammaXZ = self.rotation:getAnglesXZ()
|
|
|
|
local initialAlphaZYX, initialBetaZYX, initialGammaZYX = self.rotation:getAnglesZYX()
|
|
|
|
|
|
|
|
local targetPitch = math.rad(90)
|
|
|
|
rotateByPitch(self, targetPitch)
|
|
|
|
|
|
|
|
testing.expectEqualWithDelta(self.rotation:getPitch(), targetPitch, 0.05, 'Incorrect pitch rotation')
|
|
|
|
|
|
|
|
local alpha1, gamma1 = self.rotation:getAnglesXZ()
|
|
|
|
testing.expectEqualWithDelta(alpha1, targetPitch, 0.05, 'Incorrect alpha rotation in XZ convention')
|
|
|
|
testing.expectEqualWithDelta(gamma1, initialGammaXZ, 0.05, 'Gamma rotation in XZ convention should not change')
|
|
|
|
|
|
|
|
local alpha2, beta2, gamma2 = self.rotation:getAnglesZYX()
|
|
|
|
testing.expectEqualWithDelta(alpha2, initialAlphaZYX, 0.05, 'Alpha rotation in ZYX convention should not change')
|
|
|
|
testing.expectEqualWithDelta(beta2, initialBetaZYX, 0.05, 'Beta rotation in ZYX convention should not change')
|
|
|
|
testing.expectEqualWithDelta(gamma2, targetPitch, 0.05, 'Incorrect gamma rotation in ZYX convention')
|
|
|
|
end)
|
|
|
|
|
|
|
|
testing.registerLocalTest('playerPitchAndYawRotation',
|
|
|
|
function()
|
|
|
|
local targetPitch = math.rad(-30)
|
|
|
|
local targetYaw = math.rad(-60)
|
|
|
|
rotate(self, targetPitch, targetYaw)
|
|
|
|
|
|
|
|
testing.expectEqualWithDelta(self.rotation:getPitch(), targetPitch, 0.05, 'Incorrect pitch rotation')
|
|
|
|
testing.expectEqualWithDelta(self.rotation:getYaw(), targetYaw, 0.05, 'Incorrect yaw rotation')
|
|
|
|
|
|
|
|
local alpha1, gamma1 = self.rotation:getAnglesXZ()
|
|
|
|
testing.expectEqualWithDelta(alpha1, targetPitch, 0.05, 'Incorrect alpha rotation in XZ convention')
|
|
|
|
testing.expectEqualWithDelta(gamma1, targetYaw, 0.05, 'Incorrect gamma rotation in XZ convention')
|
|
|
|
|
|
|
|
local alpha2, beta2, gamma2 = self.rotation:getAnglesZYX()
|
|
|
|
testing.expectEqualWithDelta(alpha2, math.rad(-56), 0.05, 'Incorrect alpha rotation in ZYX convention')
|
|
|
|
testing.expectEqualWithDelta(beta2, math.rad(-25), 0.05, 'Incorrect beta rotation in ZYX convention')
|
|
|
|
testing.expectEqualWithDelta(gamma2, math.rad(-16), 0.05, 'Incorrect gamma rotation in ZYX convention')
|
2022-06-06 01:10:33 +02:00
|
|
|
end)
|
2021-07-10 13:43:53 +02:00
|
|
|
|
2024-06-23 14:59:45 +02:00
|
|
|
testing.registerLocalTest('playerRotation',
|
|
|
|
function()
|
|
|
|
local rotation = math.sqrt(2)
|
|
|
|
local endTime = core.getSimulationTime() + 3
|
|
|
|
while core.getSimulationTime() < endTime do
|
|
|
|
self.controls.jump = false
|
|
|
|
self.controls.run = true
|
|
|
|
self.controls.movement = 0
|
|
|
|
self.controls.sideMovement = 0
|
|
|
|
self.controls.pitchChange = rotation
|
|
|
|
self.controls.yawChange = rotation
|
|
|
|
coroutine.yield()
|
|
|
|
|
|
|
|
local alpha1, gamma1 = self.rotation:getAnglesXZ()
|
|
|
|
testing.expectThat(alpha1, testing.isNotNan(), 'Alpha rotation in XZ convention is nan')
|
|
|
|
testing.expectThat(gamma1, testing.isNotNan(), 'Gamma rotation in XZ convention is nan')
|
|
|
|
|
|
|
|
local alpha2, beta2, gamma2 = self.rotation:getAnglesZYX()
|
|
|
|
testing.expectThat(alpha2, testing.isNotNan(), 'Alpha rotation in ZYX convention is nan')
|
|
|
|
testing.expectThat(beta2, testing.isNotNan(), 'Beta rotation in ZYX convention is nan')
|
|
|
|
testing.expectThat(gamma2, testing.isNotNan(), 'Gamma rotation in ZYX convention is nan')
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
|
2022-06-06 01:10:33 +02:00
|
|
|
testing.registerLocalTest('playerForwardRunning',
|
|
|
|
function()
|
|
|
|
local startPos = self.position
|
|
|
|
local endTime = core.getSimulationTime() + 1
|
|
|
|
while core.getSimulationTime() < endTime do
|
2021-07-10 13:43:53 +02:00
|
|
|
self.controls.jump = false
|
|
|
|
self.controls.run = true
|
|
|
|
self.controls.movement = 1
|
|
|
|
self.controls.sideMovement = 0
|
|
|
|
self.controls.yawChange = 0
|
|
|
|
coroutine.yield()
|
|
|
|
end
|
2022-06-06 01:10:33 +02:00
|
|
|
local direction, distance = (self.position - startPos):normalize()
|
2023-12-24 21:26:18 +01:00
|
|
|
testing.expectGreaterThan(distance, 0, 'Run forward, distance')
|
2022-06-06 01:10:33 +02:00
|
|
|
testing.expectEqualWithDelta(direction.x, 0, 0.1, 'Run forward, X coord')
|
|
|
|
testing.expectEqualWithDelta(direction.y, 1, 0.1, 'Run forward, Y coord')
|
|
|
|
end)
|
2021-07-10 13:43:53 +02:00
|
|
|
|
2022-06-06 01:10:33 +02:00
|
|
|
testing.registerLocalTest('playerDiagonalWalking',
|
|
|
|
function()
|
|
|
|
local startPos = self.position
|
|
|
|
local endTime = core.getSimulationTime() + 1
|
|
|
|
while core.getSimulationTime() < endTime do
|
2021-07-10 13:43:53 +02:00
|
|
|
self.controls.jump = false
|
|
|
|
self.controls.run = false
|
|
|
|
self.controls.movement = -1
|
|
|
|
self.controls.sideMovement = -1
|
|
|
|
self.controls.yawChange = 0
|
|
|
|
coroutine.yield()
|
|
|
|
end
|
2022-06-06 01:10:33 +02:00
|
|
|
local direction, distance = (self.position - startPos):normalize()
|
2023-12-24 21:26:18 +01:00
|
|
|
testing.expectGreaterThan(distance, 0, 'Walk diagonally, distance')
|
2021-07-10 13:43:53 +02:00
|
|
|
testing.expectEqualWithDelta(direction.x, -0.707, 0.1, 'Walk diagonally, X coord')
|
2022-06-06 01:10:33 +02:00
|
|
|
testing.expectEqualWithDelta(direction.y, -0.707, 0.1, 'Walk diagonally, Y coord')
|
2021-07-10 13:43:53 +02:00
|
|
|
end)
|
|
|
|
|
2022-07-10 17:27:00 +02:00
|
|
|
testing.registerLocalTest('findPath',
|
|
|
|
function()
|
|
|
|
local src = util.vector3(4096, 4096, 867.237)
|
|
|
|
local dst = util.vector3(4500, 4500, 700.216)
|
|
|
|
local options = {
|
|
|
|
agentBounds = types.Actor.getPathfindingAgentBounds(self),
|
|
|
|
includeFlags = nearby.NAVIGATOR_FLAGS.Walk + nearby.NAVIGATOR_FLAGS.Swim,
|
|
|
|
areaCosts = {
|
|
|
|
water = 1,
|
|
|
|
door = 2,
|
|
|
|
ground = 1,
|
|
|
|
pathgrid = 1,
|
|
|
|
},
|
|
|
|
destinationTolerance = 1,
|
|
|
|
}
|
2023-07-16 17:29:34 +02:00
|
|
|
local status, path = nearby.findPath(src, dst, options)
|
2022-07-10 17:27:00 +02:00
|
|
|
testing.expectEqual(status, nearby.FIND_PATH_STATUS.Success, 'Status')
|
2023-02-04 13:49:31 +01:00
|
|
|
testing.expectLessOrEqual((path[path:size()] - dst):length(), 1,
|
|
|
|
'Last path point ' .. testing.formatActualExpected(path[path:size()], dst))
|
2022-07-10 17:27:00 +02:00
|
|
|
end)
|
|
|
|
|
|
|
|
testing.registerLocalTest('findRandomPointAroundCircle',
|
|
|
|
function()
|
|
|
|
local position = util.vector3(4096, 4096, 867.237)
|
|
|
|
local maxRadius = 100
|
|
|
|
local options = {
|
|
|
|
agentBounds = types.Actor.getPathfindingAgentBounds(self),
|
|
|
|
includeFlags = nearby.NAVIGATOR_FLAGS.Walk,
|
|
|
|
}
|
|
|
|
local result = nearby.findRandomPointAroundCircle(position, maxRadius, options)
|
2023-02-04 13:49:31 +01:00
|
|
|
testing.expectGreaterThan((result - position):length(), 1,
|
|
|
|
'Random point ' .. testing.formatActualExpected(result, position))
|
2022-07-10 17:27:00 +02:00
|
|
|
end)
|
|
|
|
|
|
|
|
testing.registerLocalTest('castNavigationRay',
|
|
|
|
function()
|
|
|
|
local src = util.vector3(4096, 4096, 867.237)
|
|
|
|
local dst = util.vector3(4500, 4500, 700.216)
|
|
|
|
local options = {
|
|
|
|
agentBounds = types.Actor.getPathfindingAgentBounds(self),
|
|
|
|
includeFlags = nearby.NAVIGATOR_FLAGS.Walk + nearby.NAVIGATOR_FLAGS.Swim,
|
|
|
|
}
|
|
|
|
local result = nearby.castNavigationRay(src, dst, options)
|
2023-02-04 13:49:31 +01:00
|
|
|
testing.expectLessOrEqual((result - dst):length(), 1,
|
|
|
|
'Navigation hit point ' .. testing.formatActualExpected(result, dst))
|
|
|
|
end)
|
|
|
|
|
|
|
|
testing.registerLocalTest('findNearestNavMeshPosition',
|
|
|
|
function()
|
|
|
|
local position = util.vector3(4096, 4096, 1000)
|
|
|
|
local options = {
|
|
|
|
agentBounds = types.Actor.getPathfindingAgentBounds(self),
|
|
|
|
includeFlags = nearby.NAVIGATOR_FLAGS.Walk + nearby.NAVIGATOR_FLAGS.Swim,
|
|
|
|
searchAreaHalfExtents = util.vector3(1000, 1000, 1000),
|
|
|
|
}
|
|
|
|
local result = nearby.findNearestNavMeshPosition(position, options)
|
|
|
|
local expected = util.vector3(4096, 4096, 872.674)
|
|
|
|
testing.expectLessOrEqual((result - expected):length(), 1,
|
|
|
|
'Navigation mesh position ' .. testing.formatActualExpected(result, expected))
|
2022-07-10 17:27:00 +02:00
|
|
|
end)
|
|
|
|
|
2021-07-10 13:43:53 +02:00
|
|
|
return {
|
|
|
|
engineHandlers = {
|
|
|
|
onUpdate = testing.updateLocal,
|
|
|
|
},
|
|
|
|
eventHandlers = testing.eventHandlers
|
|
|
|
}
|