350 lines
11 KiB
C#
350 lines
11 KiB
C#
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using Microsoft.MixedReality.WebView;
|
|
using UnityEngine;
|
|
using UnityEngine.Networking;
|
|
|
|
namespace WebViewStream
|
|
{
|
|
public class EndpointLoader : MonoBehaviour
|
|
{
|
|
[SerializeField]
|
|
private GameObject dynamicItem;
|
|
|
|
[SerializeField]
|
|
private ServiceDiscovery serviceDiscovery;
|
|
|
|
[SerializeField]
|
|
private ServicesListPopulator servicesListPopulator;
|
|
|
|
[SerializeField]
|
|
private DialogHandler dialogHandler;
|
|
|
|
private string apiUrl;
|
|
private bool triedMulticast = false;
|
|
private bool defaultEndpointLoaded = false;
|
|
private List<GameObject> instantiatedItems = new List<GameObject>();
|
|
private HashSet<MdnsService> availableServices = new HashSet<MdnsService>();
|
|
private bool areItemsVisible = true;
|
|
|
|
private const float loadTimeout = 20f;
|
|
private const string defaultApiUrl = "http://windows.local:5000/api/endpoints";
|
|
private const string defaultEndpoint1 = "http://windows.local:8100/mystream/";
|
|
private const string defaultEndpoint2 = "http://windows.local:8200/mystream/";
|
|
|
|
private void Start()
|
|
{
|
|
apiUrl = defaultApiUrl;
|
|
StartCoroutine(TimeoutFallback());
|
|
StartCoroutine(LoadEndpoints());
|
|
}
|
|
|
|
private IEnumerator TimeoutFallback()
|
|
{
|
|
float timer = 0f;
|
|
|
|
while (timer < loadTimeout && availableServices.Count == 0)
|
|
{
|
|
yield return new WaitForSeconds(1f);
|
|
timer += 1f;
|
|
}
|
|
|
|
if (availableServices.Count == 0)
|
|
{
|
|
Debug.LogWarning("Timeout reached. Loading default endpoints...");
|
|
dialogHandler.OpenDialog(
|
|
"Timeout reached",
|
|
"No services were found within the time limit.\r\n"
|
|
+ "Would you like to load the default endpoints now?\r\n"
|
|
+ "If you click \"No\", we will continue waiting for mDNS services to appear.",
|
|
() =>
|
|
{
|
|
StartCoroutine(TryLoadingFromDefaultEndpoints());
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
private Vector3 CalculateNextPosition()
|
|
{
|
|
Transform cameraTransform = Camera.main.transform;
|
|
Vector3 localOffset = new Vector3(-0.3f, 0.1f, 1.5f);
|
|
int count = instantiatedItems.Count;
|
|
|
|
if (count == 0)
|
|
{
|
|
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
|
}
|
|
|
|
GameObject lastItem = instantiatedItems[count - 1];
|
|
localOffset = new Vector3(localOffset.x + GetItemWidth(lastItem) * count, localOffset.y, localOffset.z);
|
|
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Toggles the visibility of the items spawned by this script.
|
|
/// </summary>
|
|
public void ToggleItemsVisibility()
|
|
{
|
|
areItemsVisible = !areItemsVisible;
|
|
foreach (var item in instantiatedItems)
|
|
{
|
|
item.SetActive(areItemsVisible);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the width of the item in world space units.
|
|
/// </summary>
|
|
/// <param name="item"></param>
|
|
/// <returns></returns>
|
|
public float GetItemWidth(GameObject item)
|
|
{
|
|
RectTransform rectTransform = item.GetComponent<RectTransform>();
|
|
if (rectTransform != null)
|
|
{
|
|
return rectTransform.rect.width * rectTransform.lossyScale.x;
|
|
}
|
|
|
|
return 0.6f;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Spawns a new item with a WebView component and loads the specified URL.
|
|
/// </summary>
|
|
/// <param name="url"></param>
|
|
public void SpawnItem(string url)
|
|
{
|
|
if (dynamicItem != null)
|
|
{
|
|
Vector3 nextPosition = CalculateNextPosition();
|
|
Transform cameraTransform = Camera.main.transform;
|
|
Quaternion rotation = Quaternion.LookRotation(cameraTransform.forward, cameraTransform.up);
|
|
|
|
GameObject newItem = Instantiate(
|
|
dynamicItem,
|
|
nextPosition,
|
|
rotation,
|
|
dynamicItem.transform.parent
|
|
);
|
|
newItem.SetActive(true);
|
|
|
|
instantiatedItems.Add(newItem);
|
|
|
|
var webView = newItem.GetComponentInChildren<WebView>();
|
|
|
|
if (webView != null)
|
|
{
|
|
webView.Load(url);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Debug.LogError("Dynamic item is not assigned.");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a list of all items instantiated by this script.
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public List<GameObject> GetInstantiatedItems()
|
|
{
|
|
return instantiatedItems;
|
|
}
|
|
|
|
private IEnumerator TryLoadingFromDefaultEndpoints()
|
|
{
|
|
using (UnityWebRequest request = UnityWebRequest.Get(defaultEndpoint1))
|
|
{
|
|
yield return request.SendWebRequest();
|
|
ProcessEndpointResponse(request, defaultEndpoint1, ref defaultEndpointLoaded);
|
|
}
|
|
|
|
using (UnityWebRequest request = UnityWebRequest.Get(defaultEndpoint2))
|
|
{
|
|
yield return request.SendWebRequest();
|
|
ProcessEndpointResponse(request, defaultEndpoint2, ref defaultEndpointLoaded);
|
|
}
|
|
|
|
if (!defaultEndpointLoaded)
|
|
{
|
|
Debug.LogError("Failed to load default endpoints");
|
|
dialogHandler.OpenDialog(
|
|
"Failed to load the default endpoints",
|
|
"Do you want to try one more time?\r\n"
|
|
+ "If you click \"No\", we will continue waiting for mDNS services to appear.",
|
|
() =>
|
|
{
|
|
StartCoroutine(TryLoadingFromDefaultEndpoints());
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
private void ProcessEndpointResponse(UnityWebRequest request, string endpoint, ref bool loadedFlag)
|
|
{
|
|
if (
|
|
request.result == UnityWebRequest.Result.ConnectionError
|
|
|| request.result == UnityWebRequest.Result.ProtocolError
|
|
)
|
|
{
|
|
Debug.LogError($"Error loading from {endpoint}: {request.error}");
|
|
}
|
|
else
|
|
{
|
|
Debug.Log($"Loaded from {endpoint} successfully.");
|
|
SpawnItem(endpoint);
|
|
loadedFlag = true;
|
|
}
|
|
}
|
|
|
|
private IEnumerator LoadEndpoints()
|
|
{
|
|
if (!triedMulticast)
|
|
{
|
|
StartListeningForMulticast();
|
|
yield break;
|
|
}
|
|
|
|
if (defaultEndpointLoaded)
|
|
{
|
|
Debug.Log("Default endpoint already loaded");
|
|
yield break;
|
|
}
|
|
|
|
Debug.Log($"Loading endpoints from {apiUrl}");
|
|
var request = new UnityWebRequest(apiUrl, UnityWebRequest.kHttpVerbGET);
|
|
request.downloadHandler = new DownloadHandlerBuffer();
|
|
request.SetRequestHeader("Content-Type", "application/json");
|
|
|
|
yield return request.SendWebRequest();
|
|
|
|
if (
|
|
request.result == UnityWebRequest.Result.ConnectionError
|
|
|| request.result == UnityWebRequest.Result.ProtocolError
|
|
)
|
|
{
|
|
Debug.LogWarning($"Error loading endpoints: {request.error}");
|
|
|
|
if (triedMulticast)
|
|
{
|
|
Debug.LogError("Multicast also failed");
|
|
yield break;
|
|
}
|
|
|
|
Debug.LogWarning("Trying to load from default endpoints");
|
|
yield return StartCoroutine(TryLoadingFromDefaultEndpoints());
|
|
}
|
|
|
|
if (defaultEndpointLoaded)
|
|
{
|
|
Debug.Log("At least one default endpoint loaded successfully");
|
|
yield break;
|
|
}
|
|
|
|
var json = request.downloadHandler.text;
|
|
json = "{\"Items\":" + json + "}";
|
|
Debug.Log($"Received JSON: {json}");
|
|
|
|
Endpoint[] endpoints = JsonHelper.FromJson<Endpoint>(json);
|
|
|
|
if (endpoints.Length == 0)
|
|
{
|
|
Debug.LogError("Parsed endpoints are empty");
|
|
}
|
|
else
|
|
{
|
|
if (instantiatedItems.Count > 0)
|
|
{
|
|
foreach (var item in instantiatedItems)
|
|
{
|
|
Destroy(item);
|
|
}
|
|
instantiatedItems.Clear();
|
|
}
|
|
|
|
foreach (var endpoint in endpoints)
|
|
{
|
|
if (endpoint.url == null || endpoint.url.Length == 0)
|
|
{
|
|
Debug.LogWarning($"Endpoint URL is null for endpoint");
|
|
continue;
|
|
}
|
|
SpawnItem(endpoint.url);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void StartListeningForMulticast()
|
|
{
|
|
Debug.Log("Starting multicast discovery for endpoints");
|
|
|
|
triedMulticast = true;
|
|
serviceDiscovery.StartListening(
|
|
(service) =>
|
|
{
|
|
bool wasAdded = availableServices.Add(service);
|
|
if (wasAdded)
|
|
{
|
|
AddServiceToTable(service);
|
|
}
|
|
}
|
|
);
|
|
}
|
|
|
|
private void AddServiceToTable(MdnsService service)
|
|
{
|
|
servicesListPopulator.AddItemFromService(
|
|
service,
|
|
() =>
|
|
{
|
|
apiUrl = $"http://{service.IpAddress}:{service.Port}{service.Path}";
|
|
StartCoroutine(LoadEndpoints());
|
|
}
|
|
);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Clears the list of available services.
|
|
/// </summary>
|
|
public void ClearServices()
|
|
{
|
|
availableServices.Clear();
|
|
servicesListPopulator.RemoveAllItems();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reloads the list of available services.
|
|
/// </summary>
|
|
public void ReloadEndpoints()
|
|
{
|
|
triedMulticast = false;
|
|
StartCoroutine(LoadEndpoints());
|
|
}
|
|
|
|
[Serializable]
|
|
public class Endpoint
|
|
{
|
|
public int id;
|
|
public string url;
|
|
}
|
|
|
|
public static class JsonHelper
|
|
{
|
|
public static T[] FromJson<T>(string json)
|
|
{
|
|
Wrapper<T> wrapper = JsonUtility.FromJson<Wrapper<T>>(json);
|
|
return wrapper.Items;
|
|
}
|
|
|
|
[Serializable]
|
|
private class Wrapper<T>
|
|
{
|
|
public T[] Items;
|
|
}
|
|
}
|
|
}
|
|
}
|