From 9d3ede7575e8e6c2fb890c32e12be031bccaa00d Mon Sep 17 00:00:00 2001 From: Mads Buvik Sandvei Date: Wed, 27 Dec 2023 19:11:49 +0000 Subject: [PATCH] Revert "Merge branch 'skating-olympics' into 'master'" This reverts merge request !3631 --- CHANGELOG.md | 1 - apps/openmw/mwbase/world.hpp | 6 +- apps/openmw/mwmechanics/character.cpp | 6 +- apps/openmw/mwphysics/movementsolver.cpp | 11 +-- apps/openmw/mwphysics/mtphysics.cpp | 88 +++-------------------- apps/openmw/mwphysics/mtphysics.hpp | 8 +-- apps/openmw/mwphysics/physicssystem.cpp | 17 ++--- apps/openmw/mwphysics/physicssystem.hpp | 3 +- apps/openmw/mwphysics/ptrholder.hpp | 19 +---- apps/openmw/mwworld/projectilemanager.cpp | 7 +- apps/openmw/mwworld/worldimp.cpp | 4 +- apps/openmw/mwworld/worldimp.hpp | 4 +- 12 files changed, 32 insertions(+), 142 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8d8d6f8ccf..f0181da99a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,7 +2,6 @@ ------ Bug #2623: Snowy Granius doesn't prioritize conjuration spells - Bug #3330: Backward displacement of the character when attacking in 3rd person Bug #3438: NPCs can't hit bull netch with melee weapons Bug #3842: Body part skeletons override the main skeleton Bug #4127: Weapon animation looks choppy diff --git a/apps/openmw/mwbase/world.hpp b/apps/openmw/mwbase/world.hpp index 23961554ca..14e3b2b3b7 100644 --- a/apps/openmw/mwbase/world.hpp +++ b/apps/openmw/mwbase/world.hpp @@ -295,13 +295,9 @@ namespace MWBase /// relative to \a referenceObject (but the object may be placed somewhere else if the wanted location is /// obstructed). - virtual void queueMovement( - const MWWorld::Ptr& ptr, const osg::Vec3f& velocity, float duration, bool jump = false) - = 0; + virtual void queueMovement(const MWWorld::Ptr& ptr, const osg::Vec3f& velocity) = 0; ///< Queues movement for \a ptr (in local space), to be applied in the next call to /// doPhysics. - /// \param duration The duration this speed shall be held, starting at current simulation time - /// \param jump Whether the movement shall be run over time, or immediately added as inertia instead virtual void updateAnimatedCollisionShape(const MWWorld::Ptr& ptr) = 0; diff --git a/apps/openmw/mwmechanics/character.cpp b/apps/openmw/mwmechanics/character.cpp index 58a5dfbba4..20c7fd0a92 100644 --- a/apps/openmw/mwmechanics/character.cpp +++ b/apps/openmw/mwmechanics/character.cpp @@ -2416,9 +2416,7 @@ namespace MWMechanics } if (!isMovementAnimationControlled() && !isScriptedAnimPlaying()) - { - world->queueMovement(mPtr, vec, duration, mInJump && mJumpState == JumpState_None); - } + world->queueMovement(mPtr, vec); } movement = vec; @@ -2491,7 +2489,7 @@ namespace MWMechanics } // Update movement - world->queueMovement(mPtr, movement, duration, mInJump && mJumpState == JumpState_None); + world->queueMovement(mPtr, movement); } mSkipAnim = false; diff --git a/apps/openmw/mwphysics/movementsolver.cpp b/apps/openmw/mwphysics/movementsolver.cpp index 63ffb055dd..c0b5014b31 100644 --- a/apps/openmw/mwphysics/movementsolver.cpp +++ b/apps/openmw/mwphysics/movementsolver.cpp @@ -170,8 +170,6 @@ namespace MWPhysics } // Now that we have the effective movement vector, apply wind forces to it - // TODO: This will cause instability in idle animations and other in-place animations. Should include a flag for - // this when queueing up movement if (worldData.mIsInStorm && velocity.length() > 0) { osg::Vec3f stormDirection = worldData.mStormDirection; @@ -202,8 +200,7 @@ namespace MWPhysics for (int iterations = 0; iterations < sMaxIterations && remainingTime > 0.0001f; ++iterations) { - osg::Vec3f diff = velocity * remainingTime; - osg::Vec3f nextpos = newPosition + diff; + osg::Vec3f nextpos = newPosition + velocity * remainingTime; bool underwater = newPosition.z() < swimlevel; // If not able to fly, don't allow to swim up into the air @@ -215,11 +212,7 @@ namespace MWPhysics continue; // velocity updated, calculate nextpos again } - // Note, we use an epsilon of 1e-6 instead of std::numeric_limits::epsilon() to avoid doing extremely - // small steps. But if we make it any larger we'll start rejecting subtle movements from e.g. idle - // animations. Note that, although both these comparisons to 1e-6 are logically the same, they test separate - // floating point accuracy cases. - if (diff.length2() > 1e-6 && (newPosition - nextpos).length2() > 1e-6) + if ((newPosition - nextpos).length2() > 0.0001) { // trace to where character would go if there were no obstructions tracer.doTrace(actor.mCollisionObject, newPosition, nextpos, collisionWorld, actor.mIsOnGround); diff --git a/apps/openmw/mwphysics/mtphysics.cpp b/apps/openmw/mwphysics/mtphysics.cpp index 238d00deac..52b96d9d13 100644 --- a/apps/openmw/mwphysics/mtphysics.cpp +++ b/apps/openmw/mwphysics/mtphysics.cpp @@ -25,7 +25,6 @@ #include "../mwrender/bulletdebugdraw.hpp" #include "../mwworld/class.hpp" -#include "../mwworld/datetimemanager.hpp" #include "../mwbase/environment.hpp" #include "../mwbase/world.hpp" @@ -36,7 +35,6 @@ #include "object.hpp" #include "physicssystem.hpp" #include "projectile.hpp" -#include "ptrholder.hpp" namespace MWPhysics { @@ -197,67 +195,6 @@ namespace void operator()(MWPhysics::ProjectileSimulation& /*sim*/) const {} }; - struct InitMovement - { - int mSteps = 0; - float mDelta = 0.f; - float mSimulationTime = 0.f; - - // Returns how the actor or projectile wants to move between startTime and endTime - osg::Vec3f takeMovement(MWPhysics::PtrHolder& actor, float startTime, float endTime) const - { - osg::Vec3f movement = osg::Vec3f(); - actor.eraseMovementIf([&](MWPhysics::Movement& v) { - if (v.mJump) - return false; - float start = std::max(v.mSimulationTimeStart, startTime); - float stop = std::min(v.mSimulationTimeStop, endTime); - movement += v.mVelocity * (stop - start); - if (std::abs(stop - v.mSimulationTimeStop) < 0.0001f) - return true; - return false; - }); - - return movement; - } - - std::optional takeInertia(MWPhysics::PtrHolder& actor, float startTime) const - { - std::optional inertia = std::nullopt; - actor.eraseMovementIf([&](MWPhysics::Movement& v) { - if (v.mJump && v.mSimulationTimeStart >= startTime) - { - inertia = v.mVelocity; - return true; - } - return false; - }); - return inertia; - } - - void operator()(auto& sim) const - { - if (mSteps <= 0 || mDelta < 0.00001f) - return; - - auto locked = sim.lock(); - if (!locked.has_value()) - return; - - auto& [ptrHolder, frameDataRef] = *locked; - - // Because takeMovement() returns movement instead of velocity, convert it back to velocity for the - // movement solver - osg::Vec3f velocity - = takeMovement(*ptrHolder, mSimulationTime, mSimulationTime + mDelta * mSteps) / (mSteps * mDelta); - // takeInertia() returns a velocity and should be taken over the velocity calculated above to initiate a - // jump - auto inertia = takeInertia(*ptrHolder, mSimulationTime); - - frameDataRef.get().mMovement += inertia.value_or(velocity); - } - }; - struct PreStep { btCollisionWorld* mCollisionWorld; @@ -564,18 +501,18 @@ namespace MWPhysics return std::make_tuple(numSteps, actualDelta); } - void PhysicsTaskScheduler::applyQueuedMovements(float& timeAccum, float simulationTime, - std::vector& simulations, osg::Timer_t frameStart, unsigned int frameNumber, osg::Stats& stats) + void PhysicsTaskScheduler::applyQueuedMovements(float& timeAccum, std::vector& simulations, + osg::Timer_t frameStart, unsigned int frameNumber, osg::Stats& stats) { assert(mSimulations != &simulations); waitForWorkers(); - prepareWork(timeAccum, simulationTime, simulations, frameStart, frameNumber, stats); + prepareWork(timeAccum, simulations, frameStart, frameNumber, stats); if (mWorkersSync != nullptr) mWorkersSync->wakeUpWorkers(); } - void PhysicsTaskScheduler::prepareWork(float& timeAccum, float simulationTime, std::vector& simulations, + void PhysicsTaskScheduler::prepareWork(float& timeAccum, std::vector& simulations, osg::Timer_t frameStart, unsigned int frameNumber, osg::Stats& stats) { // This function run in the main thread. @@ -585,9 +522,6 @@ namespace MWPhysics double timeStart = mTimer->tick(); - // The simulation time when the movement solving begins. - float simulationTimeStart = simulationTime - timeAccum; - // start by finishing previous background computation if (mNumThreads != 0) { @@ -602,15 +536,10 @@ namespace MWPhysics timeAccum -= numSteps * newDelta; // init - const Visitors::InitPosition initPositionVisitor{ mCollisionWorld }; + const Visitors::InitPosition vis{ mCollisionWorld }; for (auto& sim : simulations) { - std::visit(initPositionVisitor, sim); - } - const Visitors::InitMovement initMovementVisitor{ numSteps, newDelta, simulationTimeStart }; - for (auto& sim : simulations) - { - std::visit(initMovementVisitor, sim); + std::visit(vis, sim); } mPrevStepCount = numSteps; mRemainingSteps = numSteps; @@ -623,10 +552,10 @@ namespace MWPhysics mNextJob.store(0, std::memory_order_release); if (mAdvanceSimulation) - { mWorldFrameData = std::make_unique(); + + if (mAdvanceSimulation) mBudgetCursor += 1; - } if (mNumThreads == 0) { @@ -935,7 +864,6 @@ namespace MWPhysics std::remove_if(mLOSCache.begin(), mLOSCache.end(), [](const LOSRequest& req) { return req.mStale; }), mLOSCache.end()); } - mTimeEnd = mTimer->tick(); if (mWorkersSync != nullptr) mWorkersSync->workIsDone(); diff --git a/apps/openmw/mwphysics/mtphysics.hpp b/apps/openmw/mwphysics/mtphysics.hpp index 986f2be973..57f3711096 100644 --- a/apps/openmw/mwphysics/mtphysics.hpp +++ b/apps/openmw/mwphysics/mtphysics.hpp @@ -46,8 +46,8 @@ namespace MWPhysics /// @param timeAccum accumulated time from previous run to interpolate movements /// @param actorsData per actor data needed to compute new positions /// @return new position of each actor - void applyQueuedMovements(float& timeAccum, float simulationTime, std::vector& simulations, - osg::Timer_t frameStart, unsigned int frameNumber, osg::Stats& stats); + void applyQueuedMovements(float& timeAccum, std::vector& simulations, osg::Timer_t frameStart, + unsigned int frameNumber, osg::Stats& stats); void resetSimulation(const ActorMap& actors); @@ -87,8 +87,8 @@ namespace MWPhysics void afterPostSim(); void syncWithMainThread(); void waitForWorkers(); - void prepareWork(float& timeAccum, float simulationTime, std::vector& simulations, - osg::Timer_t frameStart, unsigned int frameNumber, osg::Stats& stats); + void prepareWork(float& timeAccum, std::vector& simulations, osg::Timer_t frameStart, + unsigned int frameNumber, osg::Stats& stats); std::unique_ptr mWorldFrameData; std::vector* mSimulations = nullptr; diff --git a/apps/openmw/mwphysics/physicssystem.cpp b/apps/openmw/mwphysics/physicssystem.cpp index 9a85ee009f..2196834a50 100644 --- a/apps/openmw/mwphysics/physicssystem.cpp +++ b/apps/openmw/mwphysics/physicssystem.cpp @@ -43,7 +43,6 @@ #include "../mwrender/bulletdebugdraw.hpp" #include "../mwworld/class.hpp" -#include "../mwworld/datetimemanager.hpp" #include "actor.hpp" #include "collisiontype.hpp" @@ -624,20 +623,18 @@ namespace MWPhysics return false; } - void PhysicsSystem::queueObjectMovement( - const MWWorld::Ptr& ptr, const osg::Vec3f& velocity, float duration, bool jump) + void PhysicsSystem::queueObjectMovement(const MWWorld::Ptr& ptr, const osg::Vec3f& velocity) { - float start = MWBase::Environment::get().getWorld()->getTimeManager()->getSimulationTime(); ActorMap::iterator found = mActors.find(ptr.mRef); if (found != mActors.end()) - found->second->queueMovement(velocity, start, start + duration, jump); + found->second->setVelocity(velocity); } void PhysicsSystem::clearQueuedMovement() { for (const auto& [_, actor] : mActors) { - actor->clearMovement(); + actor->setVelocity(osg::Vec3f()); actor->setInertialForce(osg::Vec3f()); } } @@ -725,10 +722,8 @@ namespace MWPhysics { std::vector& simulations = mSimulations[mSimulationsCounter++ % mSimulations.size()]; prepareSimulation(mTimeAccum >= mPhysicsDt, simulations); - float simulationTime = MWBase::Environment::get().getWorld()->getTimeManager()->getSimulationTime() + dt; // modifies mTimeAccum - mTaskScheduler->applyQueuedMovements( - mTimeAccum, simulationTime, simulations, frameStart, frameNumber, stats); + mTaskScheduler->applyQueuedMovements(mTimeAccum, simulations, frameStart, frameNumber, stats); } } @@ -912,7 +907,7 @@ namespace MWPhysics ->mValue.getFloat())) , mSlowFall(slowFall) , mRotation() - , mMovement() + , mMovement(actor.velocity()) , mWaterlevel(waterlevel) , mHalfExtentsZ(actor.getHalfExtents().z()) , mOldHeight(0) @@ -927,7 +922,7 @@ namespace MWPhysics ProjectileFrameData::ProjectileFrameData(Projectile& projectile) : mPosition(projectile.getPosition()) - , mMovement() + , mMovement(projectile.velocity()) , mCaster(projectile.getCasterCollisionObject()) , mCollisionObject(projectile.getCollisionObject()) , mProjectile(&projectile) diff --git a/apps/openmw/mwphysics/physicssystem.hpp b/apps/openmw/mwphysics/physicssystem.hpp index 7758c6dfd7..ad56581eb3 100644 --- a/apps/openmw/mwphysics/physicssystem.hpp +++ b/apps/openmw/mwphysics/physicssystem.hpp @@ -245,8 +245,7 @@ namespace MWPhysics /// Queues velocity movement for a Ptr. If a Ptr is already queued, its velocity will /// be overwritten. Valid until the next call to stepSimulation - void queueObjectMovement( - const MWWorld::Ptr& ptr, const osg::Vec3f& velocity, float duration, bool jump = false); + void queueObjectMovement(const MWWorld::Ptr& ptr, const osg::Vec3f& velocity); /// Clear the queued movements list without applying. void clearQueuedMovement(); diff --git a/apps/openmw/mwphysics/ptrholder.hpp b/apps/openmw/mwphysics/ptrholder.hpp index 16c3db0691..fc8fd94c30 100644 --- a/apps/openmw/mwphysics/ptrholder.hpp +++ b/apps/openmw/mwphysics/ptrholder.hpp @@ -1,7 +1,6 @@ #ifndef OPENMW_MWPHYSICS_PTRHOLDER_H #define OPENMW_MWPHYSICS_PTRHOLDER_H -#include #include #include #include @@ -14,14 +13,6 @@ namespace MWPhysics { - struct Movement - { - osg::Vec3f mVelocity = osg::Vec3f(); - float mSimulationTimeStart = 0.f; // The time at which this movement begun - float mSimulationTimeStop = 0.f; // The time at which this movement finished - bool mJump = false; - }; - class PtrHolder { public: @@ -41,13 +32,9 @@ namespace MWPhysics btCollisionObject* getCollisionObject() const { return mCollisionObject.get(); } - void clearMovement() { mMovement = {}; } - void queueMovement(osg::Vec3f velocity, float simulationTimeStart, float simulationTimeStop, bool jump = false) - { - mMovement.push_back(Movement{ velocity, simulationTimeStart, simulationTimeStop, jump }); - } + void setVelocity(osg::Vec3f velocity) { mVelocity = velocity; } - void eraseMovementIf(const auto& predicate) { std::erase_if(mMovement, predicate); } + osg::Vec3f velocity() { return std::exchange(mVelocity, osg::Vec3f()); } void setSimulationPosition(const osg::Vec3f& position) { mSimulationPosition = position; } @@ -66,7 +53,7 @@ namespace MWPhysics protected: MWWorld::Ptr mPtr; std::unique_ptr mCollisionObject; - std::list mMovement; + osg::Vec3f mVelocity; osg::Vec3f mSimulationPosition; osg::Vec3d mPosition; osg::Vec3d mPreviousPosition; diff --git a/apps/openmw/mwworld/projectilemanager.cpp b/apps/openmw/mwworld/projectilemanager.cpp index 1cb593f208..3cb08be5fa 100644 --- a/apps/openmw/mwworld/projectilemanager.cpp +++ b/apps/openmw/mwworld/projectilemanager.cpp @@ -30,7 +30,6 @@ #include #include "../mwworld/class.hpp" -#include "../mwworld/datetimemanager.hpp" #include "../mwworld/esmstore.hpp" #include "../mwworld/inventorystore.hpp" #include "../mwworld/manualref.hpp" @@ -455,8 +454,7 @@ namespace MWWorld } osg::Vec3f direction = orient * osg::Vec3f(0, 1, 0); direction.normalize(); - float start = MWBase::Environment::get().getWorld()->getTimeManager()->getSimulationTime(); - projectile->queueMovement(direction * speed, start, start + duration); + projectile->setVelocity(direction * speed); update(magicBoltState, duration); @@ -484,8 +482,7 @@ namespace MWWorld projectileState.mVelocity -= osg::Vec3f(0, 0, Constants::GravityConst * Constants::UnitsPerMeter * 0.1f) * duration; - float start = MWBase::Environment::get().getWorld()->getTimeManager()->getSimulationTime(); - projectile->queueMovement(projectileState.mVelocity, start, start + duration); + projectile->setVelocity(projectileState.mVelocity); // rotation does not work well for throwing projectiles - their roll angle will depend on shooting // direction. diff --git a/apps/openmw/mwworld/worldimp.cpp b/apps/openmw/mwworld/worldimp.cpp index e42f75eb97..07334396b7 100644 --- a/apps/openmw/mwworld/worldimp.cpp +++ b/apps/openmw/mwworld/worldimp.cpp @@ -1448,9 +1448,9 @@ namespace MWWorld return placed; } - void World::queueMovement(const Ptr& ptr, const osg::Vec3f& velocity, float duration, bool jump) + void World::queueMovement(const Ptr& ptr, const osg::Vec3f& velocity) { - mPhysics->queueObjectMovement(ptr, velocity, duration, jump); + mPhysics->queueObjectMovement(ptr, velocity); } void World::updateAnimatedCollisionShape(const Ptr& ptr) diff --git a/apps/openmw/mwworld/worldimp.hpp b/apps/openmw/mwworld/worldimp.hpp index aa5f9d56f0..4b9a0ccb98 100644 --- a/apps/openmw/mwworld/worldimp.hpp +++ b/apps/openmw/mwworld/worldimp.hpp @@ -383,11 +383,9 @@ namespace MWWorld float getMaxActivationDistance() const override; - void queueMovement(const Ptr& ptr, const osg::Vec3f& velocity, float duration, bool jump = false) override; + void queueMovement(const Ptr& ptr, const osg::Vec3f& velocity) override; ///< Queues movement for \a ptr (in local space), to be applied in the next call to /// doPhysics. - /// \param duration The duration this speed shall be held, starting at current simulation time - /// \param jump Whether the movement shall be run over time, or immediately added as inertia instead void updateAnimatedCollisionShape(const Ptr& ptr) override;