Add initial prototype.
This commit is contained in:
292
4-high/gel/source/world/gel-world-server.adb
Normal file
292
4-high/gel/source/world/gel-world-server.adb
Normal file
@@ -0,0 +1,292 @@
|
||||
with
|
||||
gel.Events,
|
||||
physics.Forge,
|
||||
openGL.Renderer.lean,
|
||||
lace.Event.utility,
|
||||
|
||||
ada.Text_IO,
|
||||
ada.unchecked_Deallocation;
|
||||
|
||||
|
||||
package body gel.World.server
|
||||
is
|
||||
use gel.Sprite,
|
||||
linear_Algebra_3D,
|
||||
|
||||
lace.Event.utility,
|
||||
lace.Event;
|
||||
|
||||
|
||||
procedure log (Message : in String)
|
||||
renames ada.text_IO.put_Line;
|
||||
pragma Unreferenced (log);
|
||||
|
||||
|
||||
---------
|
||||
--- Forge
|
||||
--
|
||||
|
||||
procedure free (Self : in out View)
|
||||
is
|
||||
procedure deallocate is new ada.unchecked_Deallocation (Item'Class, View);
|
||||
begin
|
||||
deallocate (Self);
|
||||
end free;
|
||||
|
||||
|
||||
|
||||
procedure define (Self : in out Item'Class; Name : in String;
|
||||
Id : in world_Id;
|
||||
space_Kind : in physics.space_Kind;
|
||||
Renderer : access openGL.Renderer.lean.item'Class);
|
||||
|
||||
overriding
|
||||
procedure destroy (Self : in out Item)
|
||||
is
|
||||
begin
|
||||
physics.Space.free (Self.physics_Space);
|
||||
|
||||
lace.Subject_and_deferred_Observer.item (Self).destroy; -- Destroy base class.
|
||||
lace.Subject_and_deferred_Observer.free (Self.local_Subject_and_deferred_Observer);
|
||||
end destroy;
|
||||
|
||||
|
||||
|
||||
package body Forge
|
||||
is
|
||||
|
||||
function to_World (Name : in String;
|
||||
Id : in world_Id;
|
||||
space_Kind : in physics.space_Kind;
|
||||
Renderer : access openGL.Renderer.lean.item'Class) return gel.World.server.item
|
||||
is
|
||||
use lace.Subject_and_deferred_Observer.Forge;
|
||||
begin
|
||||
return Self : gel.World.server.item := (to_Subject_and_Observer (Name => Name & " world" & Id'Image)
|
||||
with others => <>)
|
||||
do
|
||||
Self.define (Name, Id, space_Kind, Renderer);
|
||||
end return;
|
||||
end to_World;
|
||||
|
||||
|
||||
|
||||
function new_World (Name : in String;
|
||||
Id : in world_Id;
|
||||
space_Kind : in physics.space_Kind;
|
||||
Renderer : access openGL.Renderer.lean.item'Class) return gel.World.server.view
|
||||
is
|
||||
use lace.Subject_and_deferred_Observer.Forge;
|
||||
|
||||
Self : constant gel.World.server.view
|
||||
:= new gel.World.server.item' (to_Subject_and_Observer (name => Name & " world" & Id'Image)
|
||||
with others => <>);
|
||||
begin
|
||||
Self.define (Name, Id, space_Kind, Renderer);
|
||||
return Self;
|
||||
end new_World;
|
||||
|
||||
end Forge;
|
||||
|
||||
|
||||
|
||||
function to_Sprite (the_Pair : in remote.World.sprite_model_Pair;
|
||||
the_Models : in Id_Maps_of_Model .Map;
|
||||
the_physics_Models : in Id_Maps_of_physics_Model.Map;
|
||||
the_World : in gel.World.view) return gel.Sprite.view
|
||||
is
|
||||
the_graphics_Model : access openGL .Model.item'Class;
|
||||
the_physics_Model : access physics.Model.item'Class;
|
||||
the_Sprite : gel.Sprite.view;
|
||||
|
||||
use openGL;
|
||||
begin
|
||||
the_graphics_Model := openGL .Model.view (the_Models .Element (the_Pair.graphics_Model_Id));
|
||||
the_physics_Model := physics.Model.view (the_physics_Models.Element (the_Pair. physics_Model_Id));
|
||||
|
||||
the_Sprite := gel.Sprite.forge.new_Sprite ("Sprite" & the_Pair.sprite_Id'Image,
|
||||
sprite.World_view (the_World),
|
||||
get_Translation (the_Pair.Transform),
|
||||
the_graphics_Model,
|
||||
the_physics_Model,
|
||||
owns_Graphics => False,
|
||||
owns_Physics => False,
|
||||
is_Kinematic => the_Pair.Mass /= 0.0);
|
||||
|
||||
the_Sprite.Id_is (Now => the_Pair.sprite_Id);
|
||||
the_Sprite.is_Visible (Now => the_Pair.is_Visible);
|
||||
|
||||
the_Sprite.Site_is (get_Translation (the_Pair.Transform));
|
||||
the_Sprite.Spin_is (get_Rotation (the_Pair.Transform));
|
||||
|
||||
the_Sprite.desired_Dynamics_are (Site => the_Sprite.Site,
|
||||
Spin => to_Quaternion (get_Rotation (the_Sprite.Transform)));
|
||||
|
||||
-- the_Sprite.desired_Site_is (the_Sprite.Site);
|
||||
-- the_Sprite.desired_Spin_is (to_Quaternion (get_Rotation (the_Sprite.Transform)));
|
||||
|
||||
return the_Sprite;
|
||||
end to_Sprite;
|
||||
pragma Unreferenced (to_Sprite);
|
||||
|
||||
|
||||
----------
|
||||
--- Define
|
||||
--
|
||||
|
||||
procedure define (Self : in out Item'Class; Name : in String;
|
||||
Id : in world_Id;
|
||||
space_Kind : in physics.space_Kind;
|
||||
Renderer : access openGL.Renderer.lean.Item'Class)
|
||||
is
|
||||
use lace.Subject_and_deferred_Observer.Forge;
|
||||
begin
|
||||
Self.local_Subject_and_deferred_Observer := new_Subject_and_Observer (name => Name & " world" & Id'Image);
|
||||
|
||||
Self.Id := Id;
|
||||
Self.space_Kind := space_Kind;
|
||||
Self.Renderer := Renderer;
|
||||
Self.physics_Space := physics.Forge.new_Space (space_Kind);
|
||||
end define;
|
||||
|
||||
|
||||
|
||||
--------------
|
||||
--- Operations
|
||||
--
|
||||
|
||||
overriding
|
||||
procedure evolve (Self : in out Item)
|
||||
is
|
||||
begin
|
||||
gel.World.item (Self).evolve; -- Evolve the base class.
|
||||
|
||||
-- Update dynamics in client worlds.
|
||||
--
|
||||
declare
|
||||
use id_Maps_of_sprite,
|
||||
remote.World;
|
||||
|
||||
all_Sprites : constant id_Maps_of_sprite.Map := Self.all_Sprites.fetch;
|
||||
Cursor : id_Maps_of_sprite.Cursor := all_Sprites.First;
|
||||
|
||||
|
||||
the_Sprite : gel.Sprite.view;
|
||||
|
||||
is_a_mirrored_World : constant Boolean := not Self.Clients.Is_Empty;
|
||||
mirror_Updates_are_due : constant Boolean := Self.Age >= Self.Age_at_last_Clients_update + client_update_Period;
|
||||
updates_Count : Natural := 0;
|
||||
|
||||
the_motion_Updates : remote.World.motion_Updates (1 .. Integer (all_Sprites.Length));
|
||||
|
||||
begin
|
||||
if is_a_mirrored_World
|
||||
and mirror_Updates_are_due
|
||||
then
|
||||
while has_Element (Cursor)
|
||||
loop
|
||||
the_Sprite := Sprite.view (Element (Cursor));
|
||||
|
||||
updates_Count := updates_Count + 1;
|
||||
the_motion_Updates (updates_Count) := (Id => the_Sprite.Id,
|
||||
Site => coarsen (the_Sprite.Site),
|
||||
Spin => coarsen (to_Quaternion (the_Sprite.Spin)));
|
||||
-- Spin => the_Sprite.Spin);
|
||||
|
||||
-- log (Image (Quaternion' (refined (the_motion_Updates (updates_Count).Spin))));
|
||||
|
||||
next (Cursor);
|
||||
end loop;
|
||||
|
||||
-- Send updated sprite motions to all registered client worlds.
|
||||
--
|
||||
Self.Age_at_last_clients_update := Self.Age;
|
||||
|
||||
if updates_Count > 0
|
||||
then
|
||||
declare
|
||||
use World.server.world_Vectors;
|
||||
|
||||
Cursor : world_Vectors.Cursor := Self.Clients.First;
|
||||
the_Mirror : remote.World.view;
|
||||
begin
|
||||
while has_Element (Cursor)
|
||||
loop
|
||||
the_Mirror := Element (Cursor);
|
||||
the_Mirror.motion_Updates_are (the_motion_Updates (1 .. updates_Count));
|
||||
|
||||
next (Cursor);
|
||||
end loop;
|
||||
end;
|
||||
end if;
|
||||
|
||||
end if;
|
||||
end;
|
||||
|
||||
end evolve;
|
||||
|
||||
|
||||
|
||||
overriding
|
||||
function fetch (From : in sprite_Map) return id_Maps_of_sprite.Map
|
||||
is
|
||||
begin
|
||||
return From.Map;
|
||||
end fetch;
|
||||
|
||||
|
||||
|
||||
overriding
|
||||
procedure add (To : in out sprite_Map; the_Sprite : in Sprite.view)
|
||||
is
|
||||
begin
|
||||
To.Map.insert (the_Sprite.Id, the_Sprite);
|
||||
end add;
|
||||
|
||||
|
||||
|
||||
overriding
|
||||
procedure rid (From : in out sprite_Map; the_Sprite : in Sprite.view)
|
||||
is
|
||||
begin
|
||||
From.Map.delete (the_Sprite.Id);
|
||||
end rid;
|
||||
|
||||
|
||||
|
||||
overriding
|
||||
function all_Sprites (Self : access Item) return access World.sprite_Map'Class
|
||||
is
|
||||
begin
|
||||
return Self.all_Sprites'Access;
|
||||
end all_Sprites;
|
||||
|
||||
|
||||
|
||||
-----------------------
|
||||
--- Client Registration
|
||||
--
|
||||
|
||||
overriding
|
||||
procedure register (Self : access Item; the_Mirror : in remote.World.view;
|
||||
Mirror_as_observer : in lace.Observer.view)
|
||||
is
|
||||
begin
|
||||
Self.Clients.append (the_Mirror);
|
||||
|
||||
Self.register (Mirror_as_observer, to_Kind (remote.World. new_model_Event'Tag));
|
||||
Self.register (Mirror_as_observer, to_Kind (gel.events. new_sprite_Event'Tag));
|
||||
Self.register (Mirror_as_observer, to_Kind (gel.events.my_new_sprite_added_to_world_Event'Tag));
|
||||
end register;
|
||||
|
||||
|
||||
|
||||
overriding
|
||||
procedure deregister (Self : access Item; the_Mirror : in remote.World.view)
|
||||
is
|
||||
begin
|
||||
Self.Clients.delete (Self.Clients.find_Index (the_Mirror));
|
||||
end deregister;
|
||||
|
||||
|
||||
end gel.World.server;
|
||||
Reference in New Issue
Block a user