SRMove is a rigidbody character movement solution that uses efficient slope approximation to move smoothly on stairs and obstacles.
This project aims to recreate the low-level character movement logic used in Genshin Impact and Honkai: Star Rail.
In Genshin Impact, even though character IK is disabled while the character is moving, the character is still able to travel smoothly on stairs that have unmodified mesh colliders. This may partially be the result of some slope approximation system, which allows character movement logic to consider stairs as angled ground for the purpose of aligning movement velocity with ground surface.
✔️ Ground detection - Ground contact information is provided each physics frame. Performs ground detection with configurable parameters, using spherecast to support ledge perching
✔️ Slope traversal with ground snapping - Snap to ground surface while moving. Correctly handles velocity on angled surfaces
✔️ Smooth stair traversal - Smoothly move up and down on steps and stairs. Produces reliable smoothing behaviour when combined with slope approximation
✔️ Velocity physics - Optional built-in accleration and deceleration.
✔️ Supports moving surfaces - Correctly handles velocity on moving platforms
✔️ Intuitive collider adjustment - Configure collider height or step height while keeping collider bottom or top fixed
Developed and tested in Unity 2022.3.4f1
CharacterMover
is an equivalent of Unity's CharacterContoller
.
-
Add a
CharacterMover
component to the gameobject you want to move. -
In your own movement controller, add a using directive for
NekoLab.SRMove
. -
Implement your control logic. Reference a
CharacterMover
instance to handle the movement.
A simple example:
using UnityEngine;
using NekoLab.SRMove;
// Example third-person movement controller.
public class MyMovementController : MonoBehaviour
{
[SerializeField] private MyInputSource _inputSource; // Example input source.
[SerializeField] private CharacterMover _characterMover;
[SerializeField] private Transform _cameraTr; // Reference transform used to determine movement direction.
[SerializeField] private float _speed = 3f;
private void FixedUpdate()
{
if (!_inputSource.HasMoveInput())
{
_characterMover.Move(Vector3.zero);
return;
}
// Calculate the direction you want to move along.
Vector3 moveDirection = GetMoveDirection(_input.GetMoveInput(), _cameraTr);
// Move.
_characterMover.Move(_speed * moveDirection);
}
// Convert 2D input vector to 3D worldspace direction relative to the reference transform.
private Vector3 GetMoveDirection(Vector2 input, Transform directionReference)
{
Vector3 direction = (input.x * Vector3.ProjectOnPlane(directionReference.right, Vector3.up)).normalized;
direction += (input.y * Vector3.ProjectOnPlane(directionReference.forward, Vector3.up)).normalized;
return direction.normalized;
}
}
CharacterMover
provides several methods for movement.
This is the easiest way to get moving. Call Move
every fixed update to set the intended movement velocity.
Move(Vector3 velocity);
Move(float speed, Vector3 direction);
activeVelocity
is an internal velocity field that CharacterMover uses to set Rigidbody velocity at the end of each fixed update. When the Velocity Mode inspector field is set to a mode that uses velocity physics (i.e. acceleration, deceleration, friction), activeVelocity
will persist across fixed updates and gradually change based on the configured velocity physics logic.
To ignore any applicable velocity physics and directly set CharacerMover velocity, you can set activeVelocity
by:
SetActiveVelocity(Vector3 velocity);
You can clear the activeVelocity
by:
ClearActiveVelocity();
Directly sets the intended position change for the current fixed update.
This can be used to drive CharacterMover by animation root motion.
MoveDeltaPosition(Vector3 deltaPosition, bool alignToGround, bool restrictToGround)
Use the Velocity Mode inspector field on a CharacerMover to configure velocity physics behaviour.
- Raw: Instant acceleration and deceleration
- Simple: Fixed acceleration and deceleration based on the Speed Change Rate inspector field