You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
o3de/Gems/PhysX/Code/Source/JointComponent.cpp

196 lines
7.1 KiB
C++

/*
* Copyright (c) Contributors to the Open 3D Engine Project. For complete copyright and license terms please see the LICENSE at the root of this distribution.
*
* SPDX-License-Identifier: Apache-2.0 OR MIT
*
*/
#include <PhysX_precompiled.h>
#include <AzCore/Component/ComponentApplicationBus.h>
#include <AzFramework/Physics/RigidBodyBus.h>
#include <AzFramework/Physics/Common/PhysicsSimulatedBody.h>
#include <AzFramework/Physics/SimulatedBodies/RigidBody.h>
#include <PhysX/NativeTypeIdentifiers.h>
#include <PhysX/PhysXLocks.h>
#include <Source/Joint.h>
#include <Source/JointComponent.h>
#include <Source/Utils.h>
namespace PhysX
{
void JointComponent::Reflect(AZ::ReflectContext* context)
{
if (auto* serializeContext = azrtti_cast<AZ::SerializeContext*>(context))
{
serializeContext->Class<JointComponent, AZ::Component>()
->Version(1)
->Field("Joint Configuration", &JointComponent::m_configuration)
->Field("Joint Limits", &JointComponent::m_limits)
;
}
}
JointComponent::JointComponent(const GenericJointConfiguration& config)
: m_configuration(config)
{
}
JointComponent::JointComponent(const GenericJointConfiguration& config
, const GenericJointLimitsConfiguration& limits)
: m_configuration(AZStd::move(config))
, m_limits(limits)
{
}
void JointComponent::Activate()
{
if (m_configuration.m_followerEntity.IsValid())
{
if (m_configuration.m_followerEntity == m_configuration.m_leadEntity)
{
AZ_Error("JointComponent::Activate()",
false,
"Joint's lead entity cannot be the same as the entity in which the joint resides. Joint failed to initialize.");
return;
}
AZ::EntityBus::Handler::BusConnect(m_configuration.m_followerEntity);
}
}
void JointComponent::Deactivate()
{
AZ::EntityBus::Handler::BusDisconnect();
m_joint.reset();
}
physx::PxTransform JointComponent::GetJointLocalPose(const physx::PxRigidActor* actor
, const physx::PxTransform& jointPose)
{
if (!actor)
{
AZ_Error("JointComponent::GetJointLocalPose", false, "Can't get pose for invalid actor pointer.");
return physx::PxTransform();
}
PHYSX_SCENE_READ_LOCK(actor->getScene());
physx::PxTransform actorPose = actor->getGlobalPose();
physx::PxTransform actorTranslateInv(-actorPose.p);
physx::PxTransform actorRotateInv(actorPose.q);
actorRotateInv = actorRotateInv.getInverse();
return actorRotateInv * actorTranslateInv * jointPose;
}
AZ::Transform JointComponent::GetJointTransform(AZ::EntityId entityId
, const GenericJointConfiguration& jointConfig)
{
AZ::Transform jointTransform = PhysX::Utils::GetEntityWorldTransformWithoutScale(entityId);
jointTransform = jointTransform * jointConfig.m_localTransformFromFollower;
return jointTransform;
}
void JointComponent::InitGenericProperties()
{
if (!m_joint)
{
return;
}
physx::PxJoint* jointNative = static_cast<physx::PxJoint*>(m_joint->GetNativePointer());
if (!jointNative)
{
return;
}
PHYSX_SCENE_WRITE_LOCK(jointNative->getScene());
jointNative->setConstraintFlag(
physx::PxConstraintFlag::eCOLLISION_ENABLED,
m_configuration.GetFlag(GenericJointConfiguration::GenericJointFlag::SelfCollide));
if (m_configuration.GetFlag(GenericJointConfiguration::GenericJointFlag::Breakable))
{
jointNative->setBreakForce(m_configuration.m_forceMax
, m_configuration.m_torqueMax);
}
}
void JointComponent::ObtainLeadFollowerInfo(JointComponent::LeadFollowerInfo& info)
{
info = LeadFollowerInfo();
if (!m_configuration.m_followerEntity.IsValid())
{
return;
}
if (m_configuration.m_leadEntity.IsValid())
{
Physics::RigidBodyRequestBus::EventResult(info.m_leadBody
, m_configuration.m_leadEntity
, &Physics::RigidBodyRequests::GetRigidBody);
if (!info.m_leadBody)
{
const AZStd::string entityWithoutBodyWarningMsg("Rigid body not found in lead entity associated with joint. Joint treated as constraint on global position.");
WarnInvalidJointSetup(m_configuration.m_leadEntity, entityWithoutBodyWarningMsg);
}
}
Physics::RigidBodyRequestBus::EventResult(info.m_followerBody
, m_configuration.m_followerEntity
, &Physics::RigidBodyRequests::GetRigidBody);
if (!info.m_followerBody)
{
const AZStd::string entityWithoutBodyWarningMsg("Rigid body not found in follower entity associated with joint. Please add a rigid body component to the entity.");
WarnInvalidJointSetup(m_configuration.m_followerEntity, entityWithoutBodyWarningMsg);
return;
}
if (info.m_leadBody)
{
info.m_leadActor = static_cast<physx::PxRigidActor*>(info.m_leadBody->GetNativePointer());
}
info.m_followerActor = static_cast<physx::PxRigidActor*>(info.m_followerBody->GetNativePointer());
const AZ::Transform jointTransform = GetJointTransform(GetEntityId(), m_configuration);
physx::PxTransform jointPose = PxMathConvert(jointTransform);
if (info.m_leadActor)
{
info.m_leadLocal = GetJointLocalPose(info.m_leadActor, jointPose); // joint position & orientation in lead actor's frame.
}
else
{
info.m_leadLocal = jointPose; // lead is null, attaching follower to global position of joint.
}
info.m_followerLocal = PxMathConvert(m_configuration.m_localTransformFromFollower);// joint position & orientation in follower actor's frame.
}
void JointComponent::WarnInvalidJointSetup(AZ::EntityId entityId, const AZStd::string& message)
{
const AZStd::vector<AZ::EntityId> entityIds = { entityId };
const char* category = "PhysX Joint";
PhysX::Utils::WarnEntityNames(entityIds, category, message.c_str());
}
void JointComponent::OnEntityActivated(const AZ::EntityId& entityId)
{
AZ::EntityBus::Handler::BusDisconnect();
// If joint has no lead entity, it is a constraint on a global frame (position & orientation).
// or, follower has been activated, this is the lead being activated.
if (!m_configuration.m_leadEntity.IsValid() || entityId == m_configuration.m_leadEntity)
{
InitNativeJoint(); // Invoke overriden specific joint type instantiation
InitGenericProperties();
}
// Else, follower entity is activated, subscribe to be notified that lead entity is activated.
else
{
AZ::EntityBus::Handler::BusConnect(m_configuration.m_leadEntity);
}
}
} // namespace PhysX