Interagir avec les composants et objets de jeu Unity
Azure Remote Rendering (ARR) est optimisé pour un grand nombre d’objets (cf. Limitations). Même s’il est possible de gérer des hiérarchies volumineuses et complexes sur l’hôte, il est impossible de toutes les répliquer dans Unity sur des appareils de faible puissance.
Ainsi, quand un modèle est chargé sur l’hôte, Azure Remote Rendering met en miroir les informations sur la structure du modèle sur l’appareil client (ce qui entraîne un trafic réseau), mais ne réplique pas les objets et les composants dans Unity. Au lieu de cela, il s’attend à ce que vous demandiez les composants et objets de jeu Unity manuellement pour limiter la surcharge à ce qui est réellement nécessaire. Vous avez ainsi davantage de contrôle sur les performances côté client.
L’intégration à Unity d’Azure Remote Rendering offre donc une fonctionnalité supplémentaire permettant de répliquer la structure Remote Rendering à la demande.
Quand vous chargez un modèle, vous obtenez une référence à l’objet racine du modèle chargé. Cette référence n’est pas un objet de jeu Unity. Mais vous pouvez en faire un tel objet en utilisant la méthode d’extension Entity.GetOrCreateGameObject()
. Cette fonction attend un argument de type UnityCreationMode
. Si vous transmettez CreateUnityComponents
, le nouvel objet de jeu Unity sera également renseigné avec les composants de proxy pour tous les composants Remote Rendering qui existent sur l’hôte. Toutefois, il est recommandé de privilégier DoNotCreateUnityComponents
pour minimiser la surcharge.
IEnumerator LoadModelWithCoroutine(RenderingSession session)
{
float currentProgress = 0.0f;
var task = session.Connection.LoadModelFromSasAsync(new LoadModelFromSasOptions("builtin://Engine"),
(float progress) =>
{
currentProgress = progress;
});
while (!task.IsCompleted && !task.IsFaulted)
{
int percentage = (int)(currentProgress * 100.0f);
yield return null;
}
if (!task.IsFaulted)
{
var gameObject = task.Result.Root?.GetOrCreateGameObject(UnityCreationMode.DoNotCreateUnityComponents);
}
}
async void LoadModelWithAwait(RenderingSession session)
{
var result = await session.Connection.LoadModelFromSasAsync(new LoadModelFromSasOptions("builtin://Engine"), null);
var gameObject = result.Root?.GetOrCreateGameObject(UnityCreationMode.DoNotCreateUnityComponents);
}
Les exemples de code ci-dessus utilisent le chemin de chargement du modèle par le biais d’une SAS, car le modèle intégré est chargé. L’envoi du modèle par le biais de conteneurs d’objets Blob (avec LoadModelAsync
et LoadModelOptions
) fonctionne exactement de la même manière.
La création d’un objet de jeu Unity ajoute implicitement un composant RemoteEntitySyncObject
à l’objet de jeu. Ce composant est utilisé pour synchroniser la transformation d’entité avec le serveur. Par défaut, RemoteEntitySyncObject
implique que l’utilisateur appelle explicitement SyncToRemote()
pour synchroniser l’état local dans Unity avec le serveur. L’activation de SyncEveryFrame
synchronise automatiquement l’objet.
Les enfants distants des objets avec un RemoteEntitySyncObject
peuvent être instanciés et vous pouvez les afficher dans l’éditeur Unity à l’aide du bouton Show children.
Les composants attachés aux entités Remote Rendering sont exposés sur Unity par le biais de proxys MonoBehavior
. Ces proxys représentent le composant distant dans Unity et transfèrent toutes les modifications à l’hôte.
Pour créer des composants Remote Rendering proxys, utilisez la méthode d’extension GetOrCreateArrComponent
:
var cutplane = gameObject.GetOrCreateArrComponent<ARRCutPlaneComponent>(RemoteManagerUnity.CurrentSession);
Les durées de vie d’une entité distante et d’un objet de jeu Unity sont couplées quand ces derniers sont liés avec RemoteEntitySyncObject
. Si vous appelez UnityEngine.Object.Destroy(...)
avec un objet de jeu de ce type, l’entité distante est également supprimée.
Pour détruire l’objet de jeu Unity sans affecter l’entité distante, vous devez d’abord appeler Unbind()
sur le RemoteEntitySyncObject
.
Il en va de même pour tous les composants de proxy. Pour détruire uniquement la représentation côté client, vous devez d’abord appeler Unbind()
sur le composant de proxy :
var cutplane = gameObject.GetComponent<ARRCutPlaneComponent>();
if (cutplane != null)
{
cutplane.Unbind();
UnityEngine.Object.Destroy(cutplane);
}