Technical Review
Gravity Ruckus
Philipp Welsch, Philipp Wimmer,
Lukas Rosenberger
Pooling
Most weapons are projectile based -> Lots of GameObjects in play at once.
Continously deleting and allocating projectiles would be terrible for performance.
GC considered harmful.
Alternative: Allocate all memory upfront and only reuse.
Pooling - ObjectPool
Encapsuates a List of PooledGameObject.
Ctor is passed size of pool and allocates accordingly.
Also supports Allocate(int), however this is never used.
Internally uses CircularListIterator to keep moving through pool and find unused object.
If all instances are in use, null is returned.
Pooling - PooledGameObject
Two approaches:
Be intrusive and add PoolingComponent to GameObject
or
Encapsulate GameObject and move all pool operations into owner class
Pooling - PooledGameObject
Two approaches:
or
Keep internals of GameObject and pooling seperate.
Different GOs have different state, needs to be reset accordingly.
ResetComponent is used to reset state.
Encapsulate GameObject and move all pool operations into owner class
Be intrusive and add PoolingComponent to GameObject
Pooling - PooledGameObject
Request() changes the object to owned.
Instance is used to access the internal GameObject.
Release() resets the object and clears it for reacquisition.
Also calls each ResetComponent.
{
"Prefabs": [
[ "Prefab/SphereProjectile", 30 ],
[ "Prefab/BounceProjectile", 180 ],
[ "Prefab/GravityProjectile", 60],
[ "Prefab/BlobProjectile", 60 ],
[ "Prefab/UltimateProjectile", 4 ]
],
"Usage": [
"This file defines the path to all GameObject prefabs which should",
"have GoPools created for them on startup.",
"[Path, Instances]",
"Path: A string that defines the prefabs location relative to Assets/.",
"Instances: An integer that defines the amount of Prefab ",
"instances that should be allocated.",
"This may be omitted. In this case, 8 instances will be allocated."
]
}
Pooling - JSON
Pooling - GameObjectPoolManager
Reads the config file and creates the pools from it.
Stores a collection of GameObjectPools.
Requests are issued using the prefab:
Request(GameObject prefab)
Exists(GameObject prefab)
Allocate(GameObject prefab)
GetNumInstances(GameObject prefab)
Reset() needs to be called when changing scenes, as Unity kills all GameObjects in a scene.
Gravity
Dynamic gravity switching based on proximity didn't work out
- Unexpected behaviour for the player
- Calculating the proper rotations in 3D very hard
- Pretty resource-intensive (OverlapSphere every frame)
Switching gravity based on player input (gravity switch button) proved to be less confusing and easier to execute.
Gravity
Basic execution order for a gravity switch:
- Query player input
- Calculate a new Gravity Vector
- Set Switch Vector (local Rotation)
- Normalize local Rotation
- Calculate New Rotation
- Set new Gravity
- Notify Gravity Observers (Gravity Launcher)
FixedUpdate:
- Add Current Gravity via RigidBody.AddForce with ForceMode Acceleration
- Finish gravity rotations
void FixedUpdate()
{
m_rigidBody.AddForce(m_gravityVec*100, ForceMode.Acceleration);
if (m_fRotationTime <= 1.1f)
{
transform.localRotation = Quaternion.Slerp(m_quatOldRotation, m_quatNewRotation, m_fRotationTime);
m_fRotationTime += 0.1f;
}
}
Using Quaternion.Slerp with the old Rotation effectively locks the player into the rotation for its duration.
Since its only 10 fixedUpdate frames (=0.2 seconds) it's not really noticeable.
It also prevents the rotation from going wrong and the player 'tilting'.
Input - First Attempt
Using Unitys built in input manager and set the bindings in the editor.
This turned out to cause massive problems when more than one controler was plugged in.
- Different controlers would have different layouts
- Sometimes the controls would not work at all or cause random behaviour.
Input - Second Attempt
Write our own input handler and set the bindings via a JSON file to remain flexible.
Use XboxCtrlrInput(XCI) as a wrapper, since it handles multiple controlers well and is easy to use.
The bindings are stored in three dictionaries (Axis, Buttons, DPad) within the players PlayerInput class.
A reference to the PlayerInput class is then used to get the abstracted Input.
Code Review
By Philipp Welsch
Code Review
- 1,340