diff --git a/Uchu.Master/ServerInstance.cs b/Uchu.Master/ServerInstance.cs
index 545e69b7..6adcd36f 100644
--- a/Uchu.Master/ServerInstance.cs
+++ b/Uchu.Master/ServerInstance.cs
@@ -40,7 +40,7 @@ public void Start(string location, string dotnet)
var useDotNet = !string.IsNullOrWhiteSpace(dotnet);
if (useDotNet && dotnet?.ToLower() == "dotnet" && !File.Exists(dotnet))
{
- var pathDirectories = (Environment.GetEnvironmentVariable("PATH") ?? "").Split(";");
+ var pathDirectories = (Environment.GetEnvironmentVariable("PATH") ?? "").Split(new[] { ';', ':' });
var dotNetInPath = pathDirectories.Any(pathDirectory => File.Exists(Path.Combine(pathDirectory, dotnet)));
if (!dotNetInPath)
{
diff --git a/Uchu.Physics.Test/CollisionTest.cs b/Uchu.Physics.Test/CollisionTest.cs
index 299c1bdf..4091087b 100644
--- a/Uchu.Physics.Test/CollisionTest.cs
+++ b/Uchu.Physics.Test/CollisionTest.cs
@@ -129,5 +129,39 @@ public void BoxBoxCollision()
Assert.IsFalse(PhysicsSimulation.BoxBoxCollision(smallBox5, box7));
Assert.IsTrue(PhysicsSimulation.BoxBoxCollision(box9, box8Rotated));
}
+
+ [Test]
+ public void CapsuleSphereCollision()
+ {
+ // r=1.5 at (0,8,0)
+ var sphere = SphereBody.Create(_simulation,
+ new Vector3(0, 8, 0),
+ 1.5f);
+
+ // pointing straight up, should hit sphere
+ var capsule1 = CapsuleBody.Create(this._simulation,
+ new Vector3(0, 0, 0),
+ Quaternion.Identity,
+ 1f,
+ 6f);
+
+ // rotated 45deg, should miss sphere
+ var capsule2 = CapsuleBody.Create(this._simulation,
+ new Vector3(0, 0, 0),
+ Quaternion.CreateFromAxisAngle(Vector3.UnitX, (float) (0.125 * Math.Tau)),
+ 1f,
+ 6f);
+
+ // floating, rotated 90deg, should hit sphere
+ var capsule3 = CapsuleBody.Create(this._simulation,
+ new Vector3(8, 8, 0),
+ Quaternion.CreateFromAxisAngle(Vector3.UnitZ, (float) (0.25 * Math.Tau)),
+ 1f,
+ 6f);
+
+ Assert.IsTrue(PhysicsSimulation.CapsuleSphereCollision(capsule1, sphere));
+ Assert.IsFalse(PhysicsSimulation.CapsuleSphereCollision(capsule2, sphere));
+ Assert.IsTrue(PhysicsSimulation.CapsuleSphereCollision(capsule3, sphere));
+ }
}
}
diff --git a/Uchu.Physics/Objects/CapsuleBody.cs b/Uchu.Physics/Objects/CapsuleBody.cs
new file mode 100644
index 00000000..f1f462a8
--- /dev/null
+++ b/Uchu.Physics/Objects/CapsuleBody.cs
@@ -0,0 +1,51 @@
+using System.Numerics;
+
+namespace Uchu.Physics
+{
+ public class CapsuleBody : PhysicsObject
+ {
+ ///
+ /// Value used to determine the order in which objects are passed to the collision detection functions.
+ ///
+ public override int CollisionPrecedence { get; } = 2;
+
+ ///
+ /// Creates a capsule body.
+ ///
+ /// The simulation to use.
+ public CapsuleBody(PhysicsSimulation simulation) : base(simulation)
+ {
+ }
+
+ ///
+ /// Creates a capsule body.
+ ///
+ /// The simulation to use.
+ /// The position of the body.
+ /// The rotation of the body.
+ /// The radius of the capsule.
+ /// The height of the cylinder part of the capsule.
+ /// The capsule body that was created.
+ public static CapsuleBody Create(PhysicsSimulation simulation, Vector3 position, Quaternion rotation,
+ float radius, float height)
+ {
+ var obj = new CapsuleBody(simulation);
+ obj.Position = position;
+ obj.Rotation = rotation;
+ obj.Radius = radius;
+ obj.Height = height;
+ simulation.Register(obj);
+ return obj;
+ }
+
+ ///
+ /// Radius of the capsule.
+ ///
+ public float Radius;
+
+ ///
+ /// Height of the cylinder part of the capsule.
+ ///
+ public float Height;
+ }
+}
diff --git a/Uchu.Physics/PhysicsSimulation.cs b/Uchu.Physics/PhysicsSimulation.cs
index c73fe582..8aae587f 100644
--- a/Uchu.Physics/PhysicsSimulation.cs
+++ b/Uchu.Physics/PhysicsSimulation.cs
@@ -72,14 +72,20 @@ public static bool Collides(PhysicsObject firstObject, PhysicsObject secondObjec
}
// First precedence value >= second precedence value
- // Cube = 1, sphere = 0
-
- if (first is BoxBody box && second is SphereBody sphere)
- return BoxSphereCollision(box, sphere);
- if (first is SphereBody sphere1 && second is SphereBody sphere2)
- return SphereSphereCollision(sphere1, sphere2);
- if (first is BoxBody box1 && second is BoxBody box2)
- return BoxBoxCollision(box1, box2);
+ // Capsule = 2, cube = 1, sphere = 0
+
+ if (first is BoxBody boxSphereFirst && second is SphereBody boxSphereSecond)
+ return BoxSphereCollision(boxSphereFirst, boxSphereSecond);
+ if (first is SphereBody sphereSphereFirst && second is SphereBody sphereSphereSecond)
+ return SphereSphereCollision(sphereSphereFirst, sphereSphereSecond);
+ if (first is BoxBody boxBoxFirst && second is BoxBody boxBoxSecond)
+ return BoxBoxCollision(boxBoxFirst, boxBoxSecond);
+ if (first is CapsuleBody capsuleSphereFirst && second is SphereBody capsuleSphereSecond)
+ return CapsuleSphereCollision(capsuleSphereFirst, capsuleSphereSecond);
+ if (first is CapsuleBody capsuleBoxFirst && second is BoxBody capsuleBoxSecond)
+ return CapsuleBoxCollision(capsuleBoxFirst, capsuleBoxSecond);
+ if (first is CapsuleBody capsuleCapsuleFirst && second is CapsuleBody capsuleCapsuleSecond)
+ return CapsuleCapsuleCollision(capsuleCapsuleFirst, capsuleCapsuleSecond);
throw new NotSupportedException();
}
@@ -144,6 +150,54 @@ public static bool SphereSphereCollision(SphereBody firstSphere, SphereBody seco
return distanceSquared < maxDistanceSquared;
}
+ ///
+ /// Determine whether a capsule and a sphere intersect.
+ ///
+ /// Capsule physics object
+ /// Sphere physics object
+ public static bool CapsuleSphereCollision(CapsuleBody capsule, SphereBody sphere)
+ {
+ // reference frame: capsule at (0, 0, 0), pointing in positive y direction
+ var sphereCoordsRelative = sphere.Position - capsule.Position;
+
+ // to transform coordinate system to have capsule be axis-aligned, we apply
+ // the reverse of the rotation to the sphere coordinates
+ var inverse = new Quaternion(capsule.Rotation.X, capsule.Rotation.Y, capsule.Rotation.Z, -capsule.Rotation.W);
+ var sphereCoordsTransformed = Vector3.Transform(sphereCoordsRelative, inverse);
+
+ // coordinates of the line in the centre of the cylinder part
+ const int capsuleMinY = 0;
+ var capsuleMaxY = capsule.Height;
+
+ const int closestPointToSphereX = 0;
+ var closestPointToSphereY = Math.Clamp(sphereCoordsTransformed.Y, capsuleMinY, capsuleMaxY);
+ const int closestPointToSphereZ = 0;
+
+ return Vector3.DistanceSquared(sphereCoordsTransformed,
+ new Vector3(closestPointToSphereX, closestPointToSphereY, closestPointToSphereZ))
+ < Math.Pow(capsule.Radius + sphere.Radius, 2);
+ }
+
+ ///
+ /// Determine whether two capsules intersect.
+ ///
+ /// The first capsule
+ /// The second capsule
+ public static bool CapsuleCapsuleCollision(CapsuleBody firstCapsule, CapsuleBody secondCapsule)
+ {
+ throw new NotImplementedException("Capsule-capsule collision checks are not implemented.");
+ }
+
+ ///
+ /// Determine whether a capsule and a box intersect.
+ ///
+ /// The capsule
+ /// The box
+ public static bool CapsuleBoxCollision(CapsuleBody capsule, BoxBody box)
+ {
+ throw new NotImplementedException("Capsule-box collision checks are not implemented.");
+ }
+
///
/// Registers a physics object.
///
diff --git a/Uchu.StandardScripts/AvantGardens/MaelstromSample.cs b/Uchu.StandardScripts/AvantGardens/MaelstromSample.cs
index 56261625..ed71c826 100644
--- a/Uchu.StandardScripts/AvantGardens/MaelstromSample.cs
+++ b/Uchu.StandardScripts/AvantGardens/MaelstromSample.cs
@@ -5,9 +5,9 @@
namespace Uchu.StandardScripts.AvantGardens
{
///
- /// Native implementation of scripts/02_client/map/ag/l_ag_maelstrom_sample.lua
+ /// Script to show/hide maelstrom samples based on whether the player has the relevant mission
///
- [ScriptName("l_ag_maelstrom_sample.lua")]
+ [LotSpecific(14718)]
public class MaelstromSample : ObjectScript
{
///
@@ -21,4 +21,4 @@ public MaelstromSample(GameObject gameObject) : base(gameObject)
missionFilter.AddMissionIdToFilter(MissionId.SampleforScience);
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.StandardScripts/AvantGardens/MaelstromVacuum.cs b/Uchu.StandardScripts/AvantGardens/MaelstromVacuum.cs
index 3e1998fb..59d3cfd4 100644
--- a/Uchu.StandardScripts/AvantGardens/MaelstromVacuum.cs
+++ b/Uchu.StandardScripts/AvantGardens/MaelstromVacuum.cs
@@ -7,9 +7,9 @@
namespace Uchu.StandardScripts.AvantGardens
{
///
- /// Native implementation of scripts/02_client/Equipment/l_maelstrom_extracticator_client.lua
+ /// Script for spawned maelstrom vacuum (https://lu.lcdruniverse.org/explorer/objects/14596)
///
- [ScriptName("l_maelstrom_extracticator_client.lua")]
+ [ScriptName("ScriptComponent_1582_script_name__removed")]
public class MaelstromVacuum : ObjectScript
{
///
@@ -57,4 +57,4 @@ public MaelstromVacuum(GameObject gameObject) : base(gameObject)
});
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.StandardScripts/AvantGardens/TargetFilter.cs b/Uchu.StandardScripts/AvantGardens/TargetFilter.cs
index e31d9432..90d3eb67 100644
--- a/Uchu.StandardScripts/AvantGardens/TargetFilter.cs
+++ b/Uchu.StandardScripts/AvantGardens/TargetFilter.cs
@@ -5,9 +5,9 @@
namespace Uchu.StandardScripts.AvantGardens
{
///
- /// Native implementation of scripts/02_client/map/ag/l_ag_plunger_target.lua
+ /// Script to show/hide targets based on whether the player has the relevant mission
///
- [ScriptName("l_ag_plunger_target.lua")]
+ [LotSpecific(14380)]
public class TargetFilter : ObjectScript
{
///
@@ -20,4 +20,4 @@ public TargetFilter(GameObject gameObject) : base(gameObject)
missionFilter.AddMissionIdToFilter(MissionId.SixShooter);
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.StandardScripts/General/Binoculars.cs b/Uchu.StandardScripts/General/Binoculars.cs
index c357a3fc..b9b69ec1 100644
--- a/Uchu.StandardScripts/General/Binoculars.cs
+++ b/Uchu.StandardScripts/General/Binoculars.cs
@@ -4,9 +4,14 @@
namespace Uchu.StandardScripts.General
{
///
- /// Native implementation of scripts/02_client/map/general/l_binoculars_client.lua
+ /// Script to set flags for binoculars (e.g. https://lu.lcdruniverse.org/explorer/objects/6700) the player uses
///
- [ScriptName("l_binoculars_client.lua")]
+ [ScriptName("ScriptComponent_1002_script_name__removed")] // Binoculars, id 6700
+ [ScriptName("ScriptComponent_952_script_name__removed")] // AG - Spaceship Binoculars, id 6842
+ [ScriptName("ScriptComponent_975_script_name__removed")] // GF - Binoculars, id 6958
+ [ScriptName("ScriptComponent_1020_script_name__removed")] // PR - Binoculars, id 7607
+ [ScriptName("ScriptComponent_1021_script_name__removed")] // NS - Binoculars, id 7608
+ [ScriptName("ScriptComponent_1335_script_name__removed")] // FB - Binoculars, id 12306
public class Binoculars : ObjectScript
{
///
@@ -40,4 +45,4 @@ public Binoculars(GameObject gameObject) : base(gameObject)
});
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.StandardScripts/General/DeathPlane/DeathPlane.cs b/Uchu.StandardScripts/General/DeathPlane/DeathPlane.cs
new file mode 100644
index 00000000..7e6e3be0
--- /dev/null
+++ b/Uchu.StandardScripts/General/DeathPlane/DeathPlane.cs
@@ -0,0 +1,31 @@
+using System.Threading.Tasks;
+using Uchu.World;
+using Uchu.World.Scripting.Native;
+
+namespace Uchu.StandardScripts.General.DeathPlane
+{
+ ///
+ /// Native implementation of scripts/ai/act/l_act_player_death_trigger.lua
+ ///
+ [ScriptName("l_act_player_death_trigger.lua")]
+ public class DeathPlane : ObjectScript
+ {
+ ///
+ /// Creates the object script.
+ ///
+ /// Game object to control with the script.
+ public DeathPlane(GameObject gameObject) : base(gameObject)
+ {
+ var physics = gameObject.GetComponent();
+ if (physics == default) return;
+ Listen(physics.OnEnter, other =>
+ {
+ if (!(other.GameObject is Player player)) return;
+ Task.Run(async () =>
+ {
+ await player.GetComponent().SmashAsync(gameObject);
+ });
+ });
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.StandardScripts/General/DeathPlane/PlayerFallDeath.cs b/Uchu.StandardScripts/General/DeathPlane/PlayerFallDeath.cs
new file mode 100644
index 00000000..6b720afe
--- /dev/null
+++ b/Uchu.StandardScripts/General/DeathPlane/PlayerFallDeath.cs
@@ -0,0 +1,32 @@
+using System.Threading.Tasks;
+using Uchu.World;
+using Uchu.World.Scripting.Native;
+
+namespace Uchu.StandardScripts.General.DeathPlane
+{
+ ///
+ /// Native implementation of scripts/ai/gf/l_player_fall_death.lua
+ ///
+ [ScriptName("l_player_fall_death.lua")]
+ public class PlayerFallDeath : ObjectScript
+ {
+ ///
+ /// Creates the object script.
+ ///
+ /// Game object to control with the script.
+ public PlayerFallDeath(GameObject gameObject) : base(gameObject)
+ {
+ var physics = gameObject.GetComponent();
+ if (physics == default) return;
+ Listen(physics.OnEnter, other =>
+ {
+ if (!(other.GameObject is Player player)) return;
+ Task.Run(async () =>
+ {
+ await Task.Delay(2000);
+ await player.GetComponent().SmashAsync(gameObject);
+ });
+ });
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.StandardScripts/General/DeathPlane/PropertyDeathPlane.cs b/Uchu.StandardScripts/General/DeathPlane/PropertyDeathPlane.cs
new file mode 100644
index 00000000..9fee430f
--- /dev/null
+++ b/Uchu.StandardScripts/General/DeathPlane/PropertyDeathPlane.cs
@@ -0,0 +1,28 @@
+using Uchu.World;
+using Uchu.World.Scripting.Native;
+
+namespace Uchu.StandardScripts.General.DeathPlane
+{
+ ///
+ /// Native implementation of scripts/ai/ns/ns_pp_01/l_ns_pp_01_teleport.lua
+ ///
+ [ScriptName("l_ns_pp_01_teleport.lua")]
+ public class PropertyDeathPlane : ObjectScript
+ {
+ ///
+ /// Creates the object script.
+ ///
+ /// Game object to control with the script.
+ public PropertyDeathPlane(GameObject gameObject) : base(gameObject)
+ {
+ var physics = gameObject.GetComponent();
+ if (physics == default) return;
+ Listen(physics.OnEnter, other =>
+ {
+ if (!(other.GameObject is Player player)) return;
+ var teleportObject = this.GetGroup("Teleport")[0];
+ player.Teleport(teleportObject.Transform.Position, ignore: false);
+ });
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.StandardScripts/General/DeathPlane/SharkDeathPlane.cs b/Uchu.StandardScripts/General/DeathPlane/SharkDeathPlane.cs
new file mode 100644
index 00000000..b9badff9
--- /dev/null
+++ b/Uchu.StandardScripts/General/DeathPlane/SharkDeathPlane.cs
@@ -0,0 +1,39 @@
+using System.Numerics;
+using System.Threading.Tasks;
+using Uchu.Physics;
+using Uchu.World;
+using Uchu.World.Scripting.Native;
+
+namespace Uchu.StandardScripts.General.DeathPlane
+{
+ ///
+ /// Native implementation of scripts/ai/act/l_act_shark_player_death_trigger.lua
+ ///
+ [ScriptName("l_act_shark_player_death_trigger.lua")]
+ public class SharkDeathPlane : ObjectScript
+ {
+ ///
+ /// Creates the object script.
+ ///
+ /// Game object to control with the script.
+ public SharkDeathPlane(GameObject gameObject) : base(gameObject)
+ {
+ var physics = gameObject.GetComponent();
+ if (physics == default) return;
+ Listen(physics.OnEnter, other =>
+ {
+ if (!(other.GameObject is Player player)) return;
+ var character = player.GetComponent();
+ var missionInventoryComponent = player.GetComponent();
+ Task.Run(async () =>
+ {
+ //achievements require this order
+ await missionInventoryComponent.ScriptAsync(665, gameObject.Lot);
+ await missionInventoryComponent.ScriptAsync(664, gameObject.Lot);
+ await missionInventoryComponent.ScriptAsync(663, gameObject.Lot);
+ });
+ player.GetComponent().SmashAsync(gameObject, animation: "big-shark-death");
+ });
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.StandardScripts/General/DeathPlane/WorldDeathPlane.cs b/Uchu.StandardScripts/General/DeathPlane/WorldDeathPlane.cs
new file mode 100644
index 00000000..6d0c1337
--- /dev/null
+++ b/Uchu.StandardScripts/General/DeathPlane/WorldDeathPlane.cs
@@ -0,0 +1,50 @@
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+using Uchu.Core;
+using Uchu.World;
+using Uchu.World.Scripting.Native;
+
+namespace Uchu.StandardScripts.General.DeathPlane
+{
+ public class WorldDeathPlane: NativeScript
+ {
+ public override Task LoadAsync()
+ {
+ List dead = new List();
+ Dictionary heights = new Dictionary(){
+ {1300, 80},
+ {1400, -200},
+ {1800, -100},
+ //if we decide that -100 is fine for every world, switch this to a
+ //list containing just the zoneids, replace the if check below with
+ //heights.contains, and replace height in the listen statements
+ //with -100
+ };
+ if (!heights.ContainsKey(Zone.ZoneId))
+ {
+ return Task.CompletedTask;
+ }
+ var height = heights[Zone.ZoneId];
+ Listen(Zone.OnPlayerLoad, player =>
+ {
+ var destructibleComponent = player.GetComponent();
+ Listen(player.OnPositionUpdate, (position, rotation) =>
+ {
+ if (position.Y < height && destructibleComponent.Alive && !dead.Contains(player))
+ {
+ dead.Add(player);
+ destructibleComponent.SmashAsync(player);
+ }
+ });
+ Listen(destructibleComponent.OnResurrect, async () =>
+ {
+ await Task.Delay(1000);
+ dead.Remove(player);
+ });
+ });
+
+ return Task.CompletedTask;
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.StandardScripts/General/NexusForcePlaque.cs b/Uchu.StandardScripts/General/NexusForcePlaque.cs
index 3d8a034f..72a30837 100644
--- a/Uchu.StandardScripts/General/NexusForcePlaque.cs
+++ b/Uchu.StandardScripts/General/NexusForcePlaque.cs
@@ -4,9 +4,9 @@
namespace Uchu.StandardScripts.General
{
///
- /// Native implementation of scripts/02_client/map/general/l_story_box_interact_client.lua
+ /// Script to set flags for story plaques (https://lu.lcdruniverse.org/explorer/objects/8139) the player uses
///
- [ScriptName("l_story_box_interact_client.lua")]
+ [ScriptName("ScriptComponent_1054_script_name__removed")]
public class NexusForcePlaque : ObjectScript
{
///
@@ -41,4 +41,4 @@ public NexusForcePlaque(GameObject gameObject) : base(gameObject)
});
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.StandardScripts/NexusTower/Vault.cs b/Uchu.StandardScripts/NexusTower/Vault.cs
index 77fbf4ce..b3539a0f 100644
--- a/Uchu.StandardScripts/NexusTower/Vault.cs
+++ b/Uchu.StandardScripts/NexusTower/Vault.cs
@@ -5,9 +5,10 @@
namespace Uchu.StandardScripts.NexusTower
{
///
- /// Native implementation of scripts/02_client/map/general/l_bank_interact_client.lua
+ /// Script to handle interactions with the Vault
///
- [ScriptName("l_bank_interact_client.lua")]
+ [ScriptName("ScriptComponent_1480_script_name__removed")]
+ [ScriptName("ScriptComponent_1481_script_name__removed")]
public class Vault : ObjectScript
{
///
diff --git a/Uchu.StandardScripts/VentureExplorer/MissionConsole.cs b/Uchu.StandardScripts/VentureExplorer/MissionConsole.cs
index 68b1fe92..3d19c827 100644
--- a/Uchu.StandardScripts/VentureExplorer/MissionConsole.cs
+++ b/Uchu.StandardScripts/VentureExplorer/MissionConsole.cs
@@ -7,9 +7,9 @@
namespace Uchu.StandardScripts.VentureExplorer
{
///
- /// Native implementation of scripts/02_client/map/ve/l_mission_console_client.lua
+ /// Script for VE console (https://lu.lcdruniverse.org/explorer/objects/12551)
///
- [ScriptName("l_mission_console_client.lua")]
+ [ScriptName("ScriptComponent_1372_script_name__removed")]
public class MissionConsole : ObjectScript
{
///
@@ -85,4 +85,4 @@ public override void CompleteOnce()
});
}
}
-}
\ No newline at end of file
+}
diff --git a/Uchu.World/Objects/Components/ReplicaComponents/LuaScriptComponent.cs b/Uchu.World/Objects/Components/ReplicaComponents/LuaScriptComponent.cs
index c08342f8..270ee5c6 100644
--- a/Uchu.World/Objects/Components/ReplicaComponents/LuaScriptComponent.cs
+++ b/Uchu.World/Objects/Components/ReplicaComponents/LuaScriptComponent.cs
@@ -31,26 +31,22 @@ protected LuaScriptComponent()
return;
}
- if (GameObject.Settings.TryGetValue("custom_script_server", out var scriptOverride) && (string) scriptOverride != "")
- {
- ScriptName = (string) scriptOverride;
- }
+ // Set the server script name.
+ if (GameObject.Settings.TryGetValue("custom_script_server", out var serverScriptOverride) && (string) serverScriptOverride != "")
+ this.ScriptName = (string) serverScriptOverride;
else
- {
- ScriptName = script.Scriptname;
- }
-
- // Set the script name.
- this.ClientScriptName = script.Clientscriptname;
+ this.ScriptName = script.Scriptname;
+
+ // Set the client script name.
+ if (GameObject.Settings.TryGetValue("custom_script_client", out var clientScriptOverride) && (string) clientScriptOverride != "")
+ this.ClientScriptName = (string) clientScriptOverride;
+ else
+ this.ClientScriptName = script.Scriptname;
+
Logger.Debug($"{GameObject} -> {this.ScriptName}, {this.ClientScriptName}");
// Start the object script.
- var newObjectScriptName = Zone.ScriptManager.ObjectScriptTypes.Keys.FirstOrDefault(
- objectScriptName => (this.ScriptName ?? "").ToLower().EndsWith(objectScriptName))
- ?? Zone.ScriptManager.ObjectScriptTypes.Keys.FirstOrDefault(
- objectScriptName => (this.ClientScriptName ?? "").ToLower().EndsWith(objectScriptName));
- if (newObjectScriptName == null) return;
- this.Zone.LoadObjectScript(this.GameObject, Zone.ScriptManager.ObjectScriptTypes[newObjectScriptName]);
+ this.Zone.LoadScriptForObject(this.GameObject);
});
}
diff --git a/Uchu.World/Objects/Components/Server/PhysicsComponent.cs b/Uchu.World/Objects/Components/Server/PhysicsComponent.cs
index 555b3f77..bd237108 100644
--- a/Uchu.World/Objects/Components/Server/PhysicsComponent.cs
+++ b/Uchu.World/Objects/Components/Server/PhysicsComponent.cs
@@ -124,7 +124,7 @@ public void SetPhysicsByPath(string path) // We can't read HKX so this is basica
// 640 x 1 x 640. used for https://lu.lcdruniverse.org/explorer/objects/5633
else if (path.Contains("misc_phys_640x640"))
{
- size = new Vector3(640, 640, 12.5f)* Math.Abs(GameObject.Transform.Scale);
+ size = new Vector3(640, 12.5f, 640)* Math.Abs(GameObject.Transform.Scale);
finalObject = BoxBody.Create(Zone.Simulation, Transform.Position, Transform.Rotation, size );
}
// 20 x 50 x 1. used for https://lu.lcdruniverse.org/explorer/objects/8575
diff --git a/Uchu.World/Objects/Components/Server/PrimitiveModelComponent.cs b/Uchu.World/Objects/Components/Server/PrimitiveModelComponent.cs
index 0076f74e..4fa53fa6 100644
--- a/Uchu.World/Objects/Components/Server/PrimitiveModelComponent.cs
+++ b/Uchu.World/Objects/Components/Server/PrimitiveModelComponent.cs
@@ -42,7 +42,7 @@ private PhysicsObject LoadPrimitiveModel()
Z = (float) GameObject.Settings["primitiveModelValueZ"]
};
- return BoxBody.Create(Zone.Simulation, Transform.Position, Transform.Rotation, cuboidSize);
+ return BoxBody.Create(Zone.Simulation, Transform.Position + new Vector3(0, cuboidSize.Y / 2, 0), Transform.Rotation, cuboidSize);
case PrimitiveModelType.Cone:
break;
case PrimitiveModelType.Cylinder:
@@ -52,7 +52,7 @@ private PhysicsObject LoadPrimitiveModel()
Y = (float) GameObject.Settings["primitiveModelValueY"]
};
- return SphereBody.Create(Zone.Simulation, Transform.Position, cylinderSize.X);
+ return CapsuleBody.Create(Zone.Simulation, Transform.Position, Transform.Rotation, cylinderSize.X, cylinderSize.Y);
case PrimitiveModelType.Sphere:
break;
case PrimitiveModelType.Invalid:
diff --git a/Uchu.World/Objects/GameObjects/GameObject.cs b/Uchu.World/Objects/GameObjects/GameObject.cs
index 69df22de..21932518 100644
--- a/Uchu.World/Objects/GameObjects/GameObject.cs
+++ b/Uchu.World/Objects/GameObjects/GameObject.cs
@@ -152,22 +152,10 @@ protected GameObject()
// Load the script for the object.
// Special case for when custom_script_server but there is no script component.
- if (!this.Settings.TryGetValue("custom_script_server", out var scriptNameValue) ||
- this.GetComponent() != default || this.GetComponent() != default) return;
- var scriptName = ((string) scriptNameValue).ToLower();
- Logger.Debug($"{this} -> {scriptNameValue}");
- var scriptLoaded = false;
- foreach (var (objectScriptName, objectScriptType) in Zone.ScriptManager.ObjectScriptTypes)
- {
- if (!scriptName.EndsWith(objectScriptName)) continue;
- this.Zone.LoadObjectScript(this, objectScriptType);
- scriptLoaded = true;
- break;
- }
- //if it attempted to load a script that isn't here yet, log it
- if (!scriptLoaded && scriptName != ""){
- Logger.Debug($"Did not load script: {scriptName} Object LOT: {Lot.Id}");
- }
+ var hasCustomServerScript = this.Settings.TryGetValue("custom_script_server", out _);
+ var hasLotScript = this.Zone.ScriptManager.LotObjectScriptTypes.ContainsKey(this.Lot);
+ if ((hasCustomServerScript || hasLotScript) && this.GetComponent() == default && this.GetComponent() == default)
+ this.Zone.LoadScriptForObject(this);
});
Listen(OnDestroyed, () =>
diff --git a/Uchu.World/Objects/GameObjects/Player.cs b/Uchu.World/Objects/GameObjects/Player.cs
index 0d89a75f..634d5b38 100644
--- a/Uchu.World/Objects/GameObjects/Player.cs
+++ b/Uchu.World/Objects/GameObjects/Player.cs
@@ -363,7 +363,9 @@ private void UpdatePhysics(Vector3 position, Quaternion rotation)
physicsComponent.Physics is { } physics))
return;
- physics.Position = Transform.Position;
+ // The Transform.Position of the player refers to the bottom of the player, not the center. Since the player
+ // is treated as a sphere of radius 1, moving the sphere up by 1 corrects this.
+ physics.Position = Transform.Position + new Vector3(0, 1, 0);
physics.Rotation = Transform.Rotation;
}
diff --git a/Uchu.World/Objects/Zone.cs b/Uchu.World/Objects/Zone.cs
index f49dc1cc..39a81780 100644
--- a/Uchu.World/Objects/Zone.cs
+++ b/Uchu.World/Objects/Zone.cs
@@ -147,6 +147,66 @@ public async Task InitializeAsync()
Loaded = true;
}
+ ///
+ /// Sets up the object script for an object.
+ ///
+ /// Game object to load.
+ public void LoadScriptForObject(GameObject gameObject)
+ {
+ // Get the script names to try to load.
+ var scriptNames = new List();
+ var scriptComponent = gameObject.GetComponent();
+
+ if (gameObject.Settings.TryGetValue("custom_script_server", out var serverScriptOverride) &&
+ (string) serverScriptOverride != "")
+ {
+ scriptNames.Add(((string) serverScriptOverride).ToLower());
+ }
+
+ if (!string.IsNullOrWhiteSpace(scriptComponent?.ScriptName) &&
+ !scriptNames.Contains(scriptComponent.ScriptName.ToLower()))
+ {
+ scriptNames.Add(scriptComponent.ScriptName.ToLower());
+ }
+
+ // Log the script names.
+ var hasLotScript = this.ScriptManager.LotObjectScriptTypes.ContainsKey(gameObject.Lot);
+ if (scriptNames.Count == 0 && !hasLotScript) return;
+ string scriptNamesOutput = default;
+ if (scriptNames.Count > 0)
+ {
+ scriptNamesOutput = scriptNames.Aggregate((i, j) => i + ", " + j);
+ Logger.Debug($"Loading script for {gameObject} (LOT {gameObject.Lot}): {scriptNamesOutput}");
+ }
+ else if (hasLotScript)
+ {
+ Logger.Debug($"Loading LOT script for {gameObject} (LOT {gameObject.Lot})");
+ }
+
+ // Attempt to load a script.
+ var scriptLoaded = false;
+ foreach (var scriptName in scriptNames)
+ {
+ if (scriptLoaded) break;
+ foreach (var (objectScriptName, objectScriptType) in this.ScriptManager.NameObjectScriptTypes)
+ {
+ if (!scriptName.EndsWith(objectScriptName)) continue;
+ this.LoadObjectScript(gameObject, objectScriptType);
+ scriptLoaded = true;
+ break;
+ }
+ }
+ if (!scriptLoaded && hasLotScript)
+ {
+ this.LoadObjectScript(gameObject, this.ScriptManager.LotObjectScriptTypes[gameObject.Lot]);
+ scriptLoaded = true;
+ }
+
+ // Output if no script was loaded.
+ if (scriptLoaded || scriptNamesOutput == default) return;
+ Logger.Debug($"No script found for {gameObject} (LOT {gameObject.Lot}): {scriptNamesOutput}");
+ }
+
///
/// Loads an object script. Either happens immediately or is queued
/// for when all the objects load depending on if the zone has loaded.
diff --git a/Uchu.World/Scripting/Native/Attributes/LotSpecificAttribute.cs b/Uchu.World/Scripting/Native/Attributes/LotSpecificAttribute.cs
new file mode 100644
index 00000000..b1be7d22
--- /dev/null
+++ b/Uchu.World/Scripting/Native/Attributes/LotSpecificAttribute.cs
@@ -0,0 +1,22 @@
+using System;
+
+namespace Uchu.World.Scripting.Native
+{
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
+ public class LotSpecific : Attribute
+ {
+ ///
+ /// LOT of the script that the class applies to.
+ ///
+ public Lot Lot { get; }
+
+ ///
+ /// Creates the attribute.
+ ///
+ /// LOT of the script that the class applies to.
+ public LotSpecific(int lot)
+ {
+ this.Lot = lot;
+ }
+ }
+}
\ No newline at end of file
diff --git a/Uchu.World/Scripting/Native/NativeScriptPack.cs b/Uchu.World/Scripting/Native/NativeScriptPack.cs
index 44ba42bc..9b1629a5 100644
--- a/Uchu.World/Scripting/Native/NativeScriptPack.cs
+++ b/Uchu.World/Scripting/Native/NativeScriptPack.cs
@@ -11,10 +11,15 @@ namespace Uchu.World.Scripting.Native
internal class NativeScriptPack : ScriptPack
{
///
- /// Object script types in the script pack.
+ /// Object script types in the script pack referenced by name.
///
- public Dictionary ObjectScriptTypes { get; } = new Dictionary();
+ public Dictionary NameObjectScriptTypes { get; } = new Dictionary();
+ ///
+ /// Object script types in the script pack referenced by LOT.
+ ///
+ public Dictionary LotObjectScriptTypes { get; } = new Dictionary();
+
private List _scripts;
private Assembly _assembly;
@@ -59,11 +64,17 @@ internal void ReadAssembly()
{
// Ignore non-object scripts.
if (!type.IsAssignableTo(typeof(ObjectScript))) continue;
+
+ // Add the object scripts.
var scriptNames = type.GetCustomAttributes();
foreach (var scriptName in scriptNames)
{
- // Add the object script.
- this.ObjectScriptTypes.Add(scriptName.Name.ToLower(), type);
+ this.NameObjectScriptTypes.Add(scriptName.Name.ToLower(), type);
+ }
+ var scriptLots = type.GetCustomAttributes();
+ foreach (var scriptLot in scriptLots)
+ {
+ this.LotObjectScriptTypes.Add(scriptLot.Lot, type);
}
}
}
diff --git a/Uchu.World/Scripting/ScriptManager.cs b/Uchu.World/Scripting/ScriptManager.cs
index 97596f27..33e7cdac 100644
--- a/Uchu.World/Scripting/ScriptManager.cs
+++ b/Uchu.World/Scripting/ScriptManager.cs
@@ -13,9 +13,14 @@ namespace Uchu.World.Scripting
public class ScriptManager
{
///
- /// Object scripts types in the script manager.
+ /// Object scripts types in the script manager referenced by name.
///
- public Dictionary ObjectScriptTypes { get; } = new Dictionary();
+ public Dictionary NameObjectScriptTypes { get; } = new Dictionary();
+
+ ///
+ /// Object scripts types in the script manager referenced by LOT.
+ ///
+ public Dictionary LotObjectScriptTypes { get; } = new Dictionary();
private Zone Zone { get; }
@@ -44,15 +49,24 @@ internal async Task LoadDefaultScriptsAsync()
foreach (var pack in ScriptPacks)
{
if (!(pack is NativeScriptPack nativePack)) continue;
- foreach (var (scriptName, objectScriptType) in nativePack.ObjectScriptTypes)
+ foreach (var (scriptName, objectScriptType) in nativePack.NameObjectScriptTypes)
{
- if (ObjectScriptTypes.ContainsKey(scriptName))
+ if (NameObjectScriptTypes.ContainsKey(scriptName))
{
- Logger.Warning($"Object script time registered multiple times: {scriptName}");
+ Logger.Warning($"Name-based object script time registered multiple times: {scriptName}");
}
- ObjectScriptTypes[scriptName] = objectScriptType;
+ NameObjectScriptTypes[scriptName] = objectScriptType;
Logger.Information($"Registered object script {scriptName}");
}
+ foreach (var (scriptLot, objectScriptType) in nativePack.LotObjectScriptTypes)
+ {
+ if (LotObjectScriptTypes.ContainsKey(scriptLot))
+ {
+ Logger.Warning($"LOT-based object script time registered multiple times: {scriptLot}");
+ }
+ LotObjectScriptTypes[scriptLot] = objectScriptType;
+ Logger.Information($"Registered object script {objectScriptType.Name} for LOT {scriptLot}");
+ }
}
}