Add namespace
This commit is contained in:
parent
34e83ed312
commit
d8f5282892
|
@ -2,35 +2,38 @@ using System.Collections.Generic;
|
||||||
using TMPro;
|
using TMPro;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ConfigureNavBar : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class ConfigureNavBar : MonoBehaviour
|
||||||
private EndpointLoader endpointLoader;
|
|
||||||
|
|
||||||
private bool isVisible = false;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Toggles the visibility of the address field in the nav bar.
|
|
||||||
/// </summary>
|
|
||||||
public void ToggleVisibilityMethod()
|
|
||||||
{
|
{
|
||||||
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
[SerializeField]
|
||||||
isVisible = !isVisible;
|
private EndpointLoader endpointLoader;
|
||||||
foreach (GameObject canvas in canvases)
|
|
||||||
{
|
|
||||||
TMP_InputField inputField = canvas.GetComponentInChildren<TMP_InputField>(true);
|
|
||||||
if (inputField != null)
|
|
||||||
{
|
|
||||||
Debug.Log("Setting address field visibility to " + isVisible);
|
|
||||||
inputField.gameObject.SetActive(isVisible);
|
|
||||||
}
|
|
||||||
|
|
||||||
BoxCollider boxCollider = canvas.GetComponent<BoxCollider>();
|
private bool isVisible = false;
|
||||||
if (boxCollider != null)
|
|
||||||
|
/// <summary>
|
||||||
|
/// Toggles the visibility of the address field in the nav bar.
|
||||||
|
/// </summary>
|
||||||
|
public void ToggleVisibilityMethod()
|
||||||
|
{
|
||||||
|
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
||||||
|
isVisible = !isVisible;
|
||||||
|
foreach (GameObject canvas in canvases)
|
||||||
{
|
{
|
||||||
boxCollider.size = new Vector3(boxCollider.size.x, isVisible ? 400 : 370, boxCollider.size.z);
|
TMP_InputField inputField = canvas.GetComponentInChildren<TMP_InputField>(true);
|
||||||
boxCollider.center = new Vector3(0, isVisible ? 0 : -16, 0);
|
if (inputField != null)
|
||||||
|
{
|
||||||
|
Debug.Log("Setting address field visibility to " + isVisible);
|
||||||
|
inputField.gameObject.SetActive(isVisible);
|
||||||
|
}
|
||||||
|
|
||||||
|
BoxCollider boxCollider = canvas.GetComponent<BoxCollider>();
|
||||||
|
if (boxCollider != null)
|
||||||
|
{
|
||||||
|
boxCollider.size = new Vector3(boxCollider.size.x, isVisible ? 400 : 370, boxCollider.size.z);
|
||||||
|
boxCollider.center = new Vector3(0, isVisible ? 0 : -16, 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -2,23 +2,26 @@ using Microsoft.MixedReality.Toolkit;
|
||||||
using Microsoft.MixedReality.Toolkit.SpatialAwareness;
|
using Microsoft.MixedReality.Toolkit.SpatialAwareness;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ConfigureObservers : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
private IMixedRealitySpatialAwarenessSystem spatialAwarenessSystem;
|
public class ConfigureObservers : MonoBehaviour
|
||||||
|
|
||||||
private void Start()
|
|
||||||
{
|
{
|
||||||
spatialAwarenessSystem =
|
private IMixedRealitySpatialAwarenessSystem spatialAwarenessSystem;
|
||||||
MixedRealityToolkit.Instance.GetService<IMixedRealitySpatialAwarenessSystem>();
|
|
||||||
|
|
||||||
if (spatialAwarenessSystem != null)
|
private void Start()
|
||||||
{
|
{
|
||||||
spatialAwarenessSystem.SuspendObservers();
|
spatialAwarenessSystem =
|
||||||
Debug.Log("Spatial observers suspended");
|
MixedRealityToolkit.Instance.GetService<IMixedRealitySpatialAwarenessSystem>();
|
||||||
}
|
|
||||||
else
|
if (spatialAwarenessSystem != null)
|
||||||
{
|
{
|
||||||
Debug.LogWarning("SAS is not available");
|
spatialAwarenessSystem.SuspendObservers();
|
||||||
|
Debug.Log("Spatial observers suspended");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug.LogWarning("SAS is not available");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -2,85 +2,88 @@ using System.Collections.Generic;
|
||||||
using Microsoft.MixedReality.Toolkit.Utilities.Solvers;
|
using Microsoft.MixedReality.Toolkit.Utilities.Solvers;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ConfigureOrbital : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class ConfigureOrbital : MonoBehaviour
|
||||||
private EndpointLoader endpointLoader;
|
|
||||||
|
|
||||||
private bool orbitalEnabled = false;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Toggles the orbital behavior (solver) of the canvases.
|
|
||||||
/// </summary>
|
|
||||||
public void ToggleOrbital()
|
|
||||||
{
|
{
|
||||||
orbitalEnabled = !orbitalEnabled;
|
[SerializeField]
|
||||||
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
private EndpointLoader endpointLoader;
|
||||||
|
|
||||||
foreach (GameObject canvas in canvases)
|
private bool orbitalEnabled = false;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Toggles the orbital behavior (solver) of the canvases.
|
||||||
|
/// </summary>
|
||||||
|
public void ToggleOrbital()
|
||||||
{
|
{
|
||||||
Orbital orbital = canvas.GetComponent<Orbital>();
|
orbitalEnabled = !orbitalEnabled;
|
||||||
SolverHandler solverHandler = canvas.GetComponent<SolverHandler>();
|
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
||||||
|
|
||||||
if (orbital != null && solverHandler != null)
|
foreach (GameObject canvas in canvases)
|
||||||
{
|
{
|
||||||
orbital.enabled = orbitalEnabled;
|
Orbital orbital = canvas.GetComponent<Orbital>();
|
||||||
|
SolverHandler solverHandler = canvas.GetComponent<SolverHandler>();
|
||||||
|
|
||||||
if (orbitalEnabled)
|
if (orbital != null && solverHandler != null)
|
||||||
{
|
{
|
||||||
Vector3 headPosition = Camera.main.transform.position;
|
orbital.enabled = orbitalEnabled;
|
||||||
Quaternion headRotation = Camera.main.transform.rotation;
|
|
||||||
Vector3 relativePosition =
|
|
||||||
Quaternion.Inverse(headRotation) * (orbital.transform.position - headPosition);
|
|
||||||
|
|
||||||
orbital.LocalOffset = relativePosition;
|
if (orbitalEnabled)
|
||||||
|
{
|
||||||
|
Vector3 headPosition = Camera.main.transform.position;
|
||||||
|
Quaternion headRotation = Camera.main.transform.rotation;
|
||||||
|
Vector3 relativePosition =
|
||||||
|
Quaternion.Inverse(headRotation) * (orbital.transform.position - headPosition);
|
||||||
|
|
||||||
solverHandler.UpdateSolvers = true;
|
orbital.LocalOffset = relativePosition;
|
||||||
}
|
|
||||||
else
|
solverHandler.UpdateSolvers = true;
|
||||||
{
|
}
|
||||||
solverHandler.UpdateSolvers = false;
|
else
|
||||||
|
{
|
||||||
|
solverHandler.UpdateSolvers = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Rotates the canvases to face the user.
|
/// Rotates the canvases to face the user.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public void RotateCanvasToFaceUser()
|
public void RotateCanvasToFaceUser()
|
||||||
{
|
|
||||||
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
|
||||||
|
|
||||||
foreach (GameObject canvas in canvases)
|
|
||||||
{
|
{
|
||||||
Vector3 directionToCamera = canvas.transform.position - Camera.main.transform.position;
|
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
||||||
canvas.transform.rotation = Quaternion.LookRotation(directionToCamera);
|
|
||||||
|
foreach (GameObject canvas in canvases)
|
||||||
|
{
|
||||||
|
Vector3 directionToCamera = canvas.transform.position - Camera.main.transform.position;
|
||||||
|
canvas.transform.rotation = Quaternion.LookRotation(directionToCamera);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Centers the canvases in front of the user.
|
||||||
|
/// </summary>
|
||||||
|
public void CenterCanvasesToUser()
|
||||||
|
{
|
||||||
|
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
||||||
|
|
||||||
|
Vector3 localOffset = new Vector3(-0.4f, 0.1f, 1f);
|
||||||
|
|
||||||
|
foreach (GameObject canvas in canvases)
|
||||||
|
{
|
||||||
|
Transform cameraTransform = Camera.main.transform;
|
||||||
|
|
||||||
|
canvas.transform.position =
|
||||||
|
cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
||||||
|
canvas.transform.rotation = Quaternion.LookRotation(cameraTransform.forward, cameraTransform.up);
|
||||||
|
|
||||||
|
localOffset = new Vector3(
|
||||||
|
localOffset.x + endpointLoader.GetItemWidth(canvas),
|
||||||
|
localOffset.y,
|
||||||
|
localOffset.z
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
/// <summary>
|
|
||||||
/// Centers the canvases in front of the user.
|
|
||||||
/// </summary>
|
|
||||||
public void CenterCanvasesToUser()
|
|
||||||
{
|
|
||||||
List<GameObject> canvases = endpointLoader.GetInstantiatedItems();
|
|
||||||
|
|
||||||
Vector3 localOffset = new Vector3(-0.4f, 0.1f, 1f);
|
|
||||||
|
|
||||||
foreach (GameObject canvas in canvases)
|
|
||||||
{
|
|
||||||
Transform cameraTransform = Camera.main.transform;
|
|
||||||
|
|
||||||
canvas.transform.position =
|
|
||||||
cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
|
||||||
canvas.transform.rotation = Quaternion.LookRotation(cameraTransform.forward, cameraTransform.up);
|
|
||||||
|
|
||||||
localOffset = new Vector3(
|
|
||||||
localOffset.x + endpointLoader.GetItemWidth(canvas),
|
|
||||||
localOffset.y,
|
|
||||||
localOffset.z
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,34 +1,37 @@
|
||||||
using Microsoft.MixedReality.Toolkit.Input;
|
using Microsoft.MixedReality.Toolkit.Input;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ConfigurePointer : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
private bool handRayPointerEnabled = true;
|
public class ConfigurePointer : MonoBehaviour
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Toggles the hand ray pointer on and off.
|
|
||||||
/// </summary>
|
|
||||||
public void ToggleHandRayPointer()
|
|
||||||
{
|
{
|
||||||
if (handRayPointerEnabled)
|
private bool handRayPointerEnabled = true;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Toggles the hand ray pointer on and off.
|
||||||
|
/// </summary>
|
||||||
|
public void ToggleHandRayPointer()
|
||||||
{
|
{
|
||||||
DisableHandRayPointer();
|
if (handRayPointerEnabled)
|
||||||
handRayPointerEnabled = false;
|
{
|
||||||
|
DisableHandRayPointer();
|
||||||
|
handRayPointerEnabled = false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
EnableHandRayPointer();
|
||||||
|
handRayPointerEnabled = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
private void EnableHandRayPointer()
|
||||||
{
|
{
|
||||||
EnableHandRayPointer();
|
PointerUtils.SetHandRayPointerBehavior(PointerBehavior.AlwaysOn);
|
||||||
handRayPointerEnabled = true;
|
}
|
||||||
|
|
||||||
|
private void DisableHandRayPointer()
|
||||||
|
{
|
||||||
|
PointerUtils.SetHandRayPointerBehavior(PointerBehavior.AlwaysOff);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
private void EnableHandRayPointer()
|
|
||||||
{
|
|
||||||
PointerUtils.SetHandRayPointerBehavior(PointerBehavior.AlwaysOn);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void DisableHandRayPointer()
|
|
||||||
{
|
|
||||||
PointerUtils.SetHandRayPointerBehavior(PointerBehavior.AlwaysOff);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -2,35 +2,38 @@ using System;
|
||||||
using Microsoft.MixedReality.Toolkit.UI;
|
using Microsoft.MixedReality.Toolkit.UI;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class DialogHandler : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class DialogHandler : MonoBehaviour
|
||||||
private GameObject dialogPrefab;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Opens a dialog with a title, question, and action.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="title"></param>
|
|
||||||
/// <param name="question"></param>
|
|
||||||
/// <param name="action"></param>
|
|
||||||
public void OpenDialog(string title, string question, Action action)
|
|
||||||
{
|
{
|
||||||
Dialog dialog = Dialog.Open(
|
[SerializeField]
|
||||||
dialogPrefab,
|
private GameObject dialogPrefab;
|
||||||
DialogButtonType.Yes | DialogButtonType.No,
|
|
||||||
title,
|
/// <summary>
|
||||||
question,
|
/// Opens a dialog with a title, question, and action.
|
||||||
true
|
/// </summary>
|
||||||
);
|
/// <param name="title"></param>
|
||||||
if (dialog != null)
|
/// <param name="question"></param>
|
||||||
|
/// <param name="action"></param>
|
||||||
|
public void OpenDialog(string title, string question, Action action)
|
||||||
{
|
{
|
||||||
dialog.OnClosed += (x) =>
|
Dialog dialog = Dialog.Open(
|
||||||
|
dialogPrefab,
|
||||||
|
DialogButtonType.Yes | DialogButtonType.No,
|
||||||
|
title,
|
||||||
|
question,
|
||||||
|
true
|
||||||
|
);
|
||||||
|
if (dialog != null)
|
||||||
{
|
{
|
||||||
if (x.Result == DialogButtonType.Yes)
|
dialog.OnClosed += (x) =>
|
||||||
{
|
{
|
||||||
action?.Invoke();
|
if (x.Result == DialogButtonType.Yes)
|
||||||
}
|
{
|
||||||
};
|
action?.Invoke();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -5,341 +5,344 @@ using Microsoft.MixedReality.WebView;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
using UnityEngine.Networking;
|
using UnityEngine.Networking;
|
||||||
|
|
||||||
public class EndpointLoader : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class EndpointLoader : MonoBehaviour
|
||||||
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 float loadTimeout = 10f;
|
|
||||||
private bool areItemsVisible = true;
|
|
||||||
|
|
||||||
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;
|
[SerializeField]
|
||||||
StartCoroutine(TimeoutFallback());
|
private GameObject dynamicItem;
|
||||||
StartCoroutine(LoadEndpoints());
|
|
||||||
}
|
|
||||||
|
|
||||||
private IEnumerator TimeoutFallback()
|
[SerializeField]
|
||||||
{
|
private ServiceDiscovery serviceDiscovery;
|
||||||
float timer = 0f;
|
|
||||||
|
|
||||||
while (timer < loadTimeout && availableServices.Count == 0)
|
[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()
|
||||||
{
|
{
|
||||||
yield return new WaitForSeconds(1f);
|
apiUrl = defaultApiUrl;
|
||||||
timer += 1f;
|
StartCoroutine(TimeoutFallback());
|
||||||
|
StartCoroutine(LoadEndpoints());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (availableServices.Count == 0)
|
private IEnumerator TimeoutFallback()
|
||||||
{
|
{
|
||||||
Debug.LogWarning("Timeout reached. Loading default endpoints...");
|
float timer = 0f;
|
||||||
dialogHandler.OpenDialog(
|
|
||||||
"Timeout reached",
|
while (timer < loadTimeout && availableServices.Count == 0)
|
||||||
"No services were found within the time limit.\r\n"
|
{
|
||||||
+ "Would you like to load the default endpoints now?\r\n"
|
yield return new WaitForSeconds(1f);
|
||||||
+ "If you click \"No\", we will continue waiting for mDNS services to appear.",
|
timer += 1f;
|
||||||
() =>
|
}
|
||||||
{
|
|
||||||
StartCoroutine(TryLoadingFromDefaultEndpoints());
|
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()
|
private Vector3 CalculateNextPosition()
|
||||||
{
|
|
||||||
Transform cameraTransform = Camera.main.transform;
|
|
||||||
Vector3 localOffset = new Vector3(-0.4f, 0.1f, 1f);
|
|
||||||
|
|
||||||
if (instantiatedItems.Count == 0)
|
|
||||||
{
|
{
|
||||||
|
Transform cameraTransform = Camera.main.transform;
|
||||||
|
Vector3 localOffset = new Vector3(-0.4f, 0.1f, 1f);
|
||||||
|
|
||||||
|
if (instantiatedItems.Count == 0)
|
||||||
|
{
|
||||||
|
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
||||||
|
}
|
||||||
|
|
||||||
|
GameObject lastItem = instantiatedItems[instantiatedItems.Count - 1];
|
||||||
|
localOffset = new Vector3(localOffset.x + GetItemWidth(lastItem), localOffset.y, localOffset.z);
|
||||||
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
||||||
}
|
}
|
||||||
|
|
||||||
GameObject lastItem = instantiatedItems[instantiatedItems.Count - 1];
|
/// <summary>
|
||||||
localOffset = new Vector3(localOffset.x + GetItemWidth(lastItem), localOffset.y, localOffset.z);
|
/// Toggles the visibility of the items spawned by this script.
|
||||||
return cameraTransform.position + cameraTransform.TransformDirection(localOffset);
|
/// </summary>
|
||||||
}
|
public void ToggleItemsVisibility()
|
||||||
|
|
||||||
/// <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);
|
areItemsVisible = !areItemsVisible;
|
||||||
}
|
foreach (var item in instantiatedItems)
|
||||||
}
|
|
||||||
|
|
||||||
/// <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 + 0.2f;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0.8f;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <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);
|
item.SetActive(areItemsVisible);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
/// <summary>
|
||||||
|
/// Returns the width of the item in world space units.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="item"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
public float GetItemWidth(GameObject item)
|
||||||
{
|
{
|
||||||
Debug.LogError("Dynamic item is not assigned.");
|
RectTransform rectTransform = item.GetComponent<RectTransform>();
|
||||||
}
|
if (rectTransform != null)
|
||||||
}
|
|
||||||
|
|
||||||
/// <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");
|
return rectTransform.rect.width * rectTransform.lossyScale.x + 0.2f;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0.8f;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <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;
|
yield break;
|
||||||
}
|
}
|
||||||
|
|
||||||
Debug.LogWarning("Trying to load from default endpoints");
|
if (defaultEndpointLoaded)
|
||||||
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)
|
Debug.Log("Default endpoint already loaded");
|
||||||
{
|
yield break;
|
||||||
Destroy(item);
|
|
||||||
}
|
|
||||||
instantiatedItems.Clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var endpoint in endpoints)
|
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
|
||||||
|
)
|
||||||
{
|
{
|
||||||
if (endpoint.url == null || endpoint.url.Length == 0)
|
Debug.LogWarning($"Error loading endpoints: {request.error}");
|
||||||
|
|
||||||
|
if (triedMulticast)
|
||||||
{
|
{
|
||||||
Debug.LogWarning($"Endpoint URL is null for endpoint");
|
Debug.LogError("Multicast also failed");
|
||||||
continue;
|
yield break;
|
||||||
}
|
}
|
||||||
SpawnItem(endpoint.url);
|
|
||||||
|
Debug.LogWarning("Trying to load from default endpoints");
|
||||||
|
yield return StartCoroutine(TryLoadingFromDefaultEndpoints());
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void StartListeningForMulticast()
|
if (defaultEndpointLoaded)
|
||||||
{
|
|
||||||
Debug.Log("Starting multicast discovery for endpoints");
|
|
||||||
|
|
||||||
triedMulticast = true;
|
|
||||||
serviceDiscovery.StartListening(
|
|
||||||
(service) =>
|
|
||||||
{
|
{
|
||||||
bool wasAdded = availableServices.Add(service);
|
Debug.Log("At least one default endpoint loaded successfully");
|
||||||
if (wasAdded)
|
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)
|
||||||
{
|
{
|
||||||
AddServiceToTable(service);
|
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 AddServiceToTable(MdnsService service)
|
private void StartListeningForMulticast()
|
||||||
{
|
|
||||||
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);
|
Debug.Log("Starting multicast discovery for endpoints");
|
||||||
return wrapper.Items;
|
|
||||||
|
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]
|
[Serializable]
|
||||||
private class Wrapper<T>
|
public class Endpoint
|
||||||
{
|
{
|
||||||
public T[] Items;
|
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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,46 +1,49 @@
|
||||||
public class MdnsService
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
public string IpAddress { get; }
|
public class MdnsService
|
||||||
public int Port { get; }
|
|
||||||
public string Path { get; }
|
|
||||||
public string Host { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Represents a service discovered via mDNS.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="ipAddress"></param>
|
|
||||||
/// <param name="port"></param>
|
|
||||||
/// <param name="path"></param>
|
|
||||||
/// <param name="host"></param>
|
|
||||||
public MdnsService(string ipAddress, int port, string path, string host)
|
|
||||||
{
|
{
|
||||||
IpAddress = ipAddress;
|
public string IpAddress { get; }
|
||||||
Port = port;
|
public int Port { get; }
|
||||||
Path = path;
|
public string Path { get; }
|
||||||
Host = host;
|
public string Host { get; }
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
/// <summary>
|
||||||
{
|
/// Represents a service discovered via mDNS.
|
||||||
return $"IpAddress: {IpAddress}, Port: {Port}, Path: {Path}, Host: {Host}";
|
/// </summary>
|
||||||
}
|
/// <param name="ipAddress"></param>
|
||||||
|
/// <param name="port"></param>
|
||||||
|
/// <param name="path"></param>
|
||||||
|
/// <param name="host"></param>
|
||||||
|
public MdnsService(string ipAddress, int port, string path, string host)
|
||||||
|
{
|
||||||
|
IpAddress = ipAddress;
|
||||||
|
Port = port;
|
||||||
|
Path = path;
|
||||||
|
Host = host;
|
||||||
|
}
|
||||||
|
|
||||||
public override bool Equals(object obj)
|
public override string ToString()
|
||||||
{
|
{
|
||||||
return obj is MdnsService service
|
return $"IpAddress: {IpAddress}, Port: {Port}, Path: {Path}, Host: {Host}";
|
||||||
&& IpAddress == service.IpAddress
|
}
|
||||||
&& Host == service.Host
|
|
||||||
&& Port == service.Port
|
|
||||||
&& Path == service.Path;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override int GetHashCode()
|
public override bool Equals(object obj)
|
||||||
{
|
{
|
||||||
int hash = 17;
|
return obj is MdnsService service
|
||||||
hash = hash * 31 + (IpAddress?.GetHashCode() ?? 0);
|
&& IpAddress == service.IpAddress
|
||||||
hash = hash * 31 + Port.GetHashCode();
|
&& Host == service.Host
|
||||||
hash = hash * 31 + (Path?.GetHashCode() ?? 0);
|
&& Port == service.Port
|
||||||
hash = hash * 31 + (Host?.GetHashCode() ?? 0);
|
&& Path == service.Path;
|
||||||
return hash;
|
}
|
||||||
|
|
||||||
|
public override int GetHashCode()
|
||||||
|
{
|
||||||
|
int hash = 17;
|
||||||
|
hash = hash * 31 + (IpAddress?.GetHashCode() ?? 0);
|
||||||
|
hash = hash * 31 + Port.GetHashCode();
|
||||||
|
hash = hash * 31 + (Path?.GetHashCode() ?? 0);
|
||||||
|
hash = hash * 31 + (Host?.GetHashCode() ?? 0);
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,23 +1,26 @@
|
||||||
using Microsoft.MixedReality.Toolkit.UI;
|
using Microsoft.MixedReality.Toolkit.UI;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ScrollablePagination : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class ScrollablePagination : MonoBehaviour
|
||||||
private ScrollingObjectCollection scrollView;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Scrolls the collection by a specified amount.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="amount"></param>
|
|
||||||
public void ScrollByTier(int amount)
|
|
||||||
{
|
{
|
||||||
if (scrollView == null)
|
[SerializeField]
|
||||||
{
|
private ScrollingObjectCollection scrollView;
|
||||||
Debug.LogError("ScrollingObjectCollection is not set.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
scrollView.MoveByTiers(amount);
|
/// <summary>
|
||||||
|
/// Scrolls the collection by a specified amount.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="amount"></param>
|
||||||
|
public void ScrollByTier(int amount)
|
||||||
|
{
|
||||||
|
if (scrollView == null)
|
||||||
|
{
|
||||||
|
Debug.LogError("ScrollingObjectCollection is not set.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollView.MoveByTiers(amount);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -6,367 +6,370 @@ using System.Net.Sockets;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ServiceDiscovery : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
private UdpClient udpClient;
|
public class ServiceDiscovery : MonoBehaviour
|
||||||
private Action<MdnsService> action;
|
|
||||||
|
|
||||||
private string receivedIp;
|
|
||||||
private string receivedPort;
|
|
||||||
private string receivedPath;
|
|
||||||
private string receivedHost;
|
|
||||||
|
|
||||||
private IPAddress defaultIP;
|
|
||||||
|
|
||||||
private const string multicastAddress = "224.0.0.251";
|
|
||||||
private const int multicastPort = 5353;
|
|
||||||
|
|
||||||
private Queue<MdnsService> serviceQueue = new Queue<MdnsService>();
|
|
||||||
|
|
||||||
private IPAddress GetDefaultInterfaceIP()
|
|
||||||
{
|
{
|
||||||
foreach (NetworkInterface ni in NetworkInterface.GetAllNetworkInterfaces())
|
private UdpClient udpClient;
|
||||||
|
private Action<MdnsService> action;
|
||||||
|
|
||||||
|
private string receivedIp;
|
||||||
|
private string receivedPort;
|
||||||
|
private string receivedPath;
|
||||||
|
private string receivedHost;
|
||||||
|
|
||||||
|
private IPAddress defaultIP;
|
||||||
|
|
||||||
|
private const string multicastAddress = "224.0.0.251";
|
||||||
|
private const int multicastPort = 5353;
|
||||||
|
|
||||||
|
private Queue<MdnsService> serviceQueue = new Queue<MdnsService>();
|
||||||
|
|
||||||
|
private IPAddress GetDefaultInterfaceIP()
|
||||||
{
|
{
|
||||||
if (ni.OperationalStatus == OperationalStatus.Up)
|
foreach (NetworkInterface ni in NetworkInterface.GetAllNetworkInterfaces())
|
||||||
{
|
{
|
||||||
var ipProps = ni.GetIPProperties();
|
if (ni.OperationalStatus == OperationalStatus.Up)
|
||||||
if (ipProps.GatewayAddresses.Count > 0)
|
|
||||||
{
|
{
|
||||||
foreach (UnicastIPAddressInformation ip in ipProps.UnicastAddresses)
|
var ipProps = ni.GetIPProperties();
|
||||||
|
if (ipProps.GatewayAddresses.Count > 0)
|
||||||
{
|
{
|
||||||
if (ip.Address.AddressFamily == AddressFamily.InterNetwork)
|
foreach (UnicastIPAddressInformation ip in ipProps.UnicastAddresses)
|
||||||
{
|
{
|
||||||
return ip.Address;
|
if (ip.Address.AddressFamily == AddressFamily.InterNetwork)
|
||||||
|
{
|
||||||
|
return ip.Address;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
private List<IPAddress> GetRoutableLocalIPs()
|
||||||
}
|
|
||||||
|
|
||||||
private List<IPAddress> GetRoutableLocalIPs()
|
|
||||||
{
|
|
||||||
List<IPAddress> localIPs = new List<IPAddress>();
|
|
||||||
|
|
||||||
foreach (IPAddress local in Dns.GetHostEntry(Dns.GetHostName()).AddressList)
|
|
||||||
{
|
{
|
||||||
if (local.AddressFamily == AddressFamily.InterNetwork)
|
List<IPAddress> localIPs = new List<IPAddress>();
|
||||||
|
|
||||||
|
foreach (IPAddress local in Dns.GetHostEntry(Dns.GetHostName()).AddressList)
|
||||||
{
|
{
|
||||||
byte[] bytes = local.GetAddressBytes();
|
if (local.AddressFamily == AddressFamily.InterNetwork)
|
||||||
if (bytes[0] == 169 && bytes[1] == 254)
|
|
||||||
{
|
{
|
||||||
Debug.Log($"Skipping non-routable address: {local}");
|
byte[] bytes = local.GetAddressBytes();
|
||||||
continue;
|
if (bytes[0] == 169 && bytes[1] == 254)
|
||||||
|
{
|
||||||
|
Debug.Log($"Skipping non-routable address: {local}");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
localIPs.Add(local);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return localIPs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Starts listening for mDNS service announcements.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="action"></param>
|
||||||
|
public void StartListening(Action<MdnsService> action)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
defaultIP = GetDefaultInterfaceIP();
|
||||||
|
if (defaultIP == null)
|
||||||
|
{
|
||||||
|
Debug.LogError("No default interface found. Cannot start multicast listener.");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
localIPs.Add(local);
|
udpClient = new UdpClient();
|
||||||
|
udpClient.Client.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
|
||||||
|
udpClient.Client.Bind(new IPEndPoint(defaultIP, multicastPort));
|
||||||
|
udpClient.JoinMulticastGroup(IPAddress.Parse(multicastAddress), defaultIP);
|
||||||
|
|
||||||
|
this.action = action;
|
||||||
|
|
||||||
|
Debug.Log("Listening for service announcements...");
|
||||||
|
|
||||||
|
SendMdnsQuery("_http._tcp.local");
|
||||||
|
|
||||||
|
udpClient.BeginReceive(OnReceive, null);
|
||||||
}
|
}
|
||||||
}
|
catch (Exception ex)
|
||||||
|
|
||||||
return localIPs;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Starts listening for mDNS service announcements.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="action"></param>
|
|
||||||
public void StartListening(Action<MdnsService> action)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
defaultIP = GetDefaultInterfaceIP();
|
|
||||||
if (defaultIP == null)
|
|
||||||
{
|
{
|
||||||
Debug.LogError("No default interface found. Cannot start multicast listener.");
|
Debug.LogError($"Error starting UDP listener: {ex.Message}");
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
udpClient = new UdpClient();
|
|
||||||
udpClient.Client.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
|
|
||||||
udpClient.Client.Bind(new IPEndPoint(defaultIP, multicastPort));
|
|
||||||
udpClient.JoinMulticastGroup(IPAddress.Parse(multicastAddress), defaultIP);
|
|
||||||
|
|
||||||
this.action = action;
|
|
||||||
|
|
||||||
Debug.Log("Listening for service announcements...");
|
|
||||||
|
|
||||||
SendMdnsQuery("_http._tcp.local");
|
|
||||||
|
|
||||||
udpClient.BeginReceive(OnReceive, null);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Debug.LogError($"Error starting UDP listener: {ex.Message}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void SendMdnsQuery(string serviceName)
|
|
||||||
{
|
|
||||||
byte[] query = CreateMdnsQuery(serviceName);
|
|
||||||
Debug.Log($"Sending mDNS query for {serviceName}");
|
|
||||||
|
|
||||||
udpClient.Send(query, query.Length, new IPEndPoint(IPAddress.Parse(multicastAddress), multicastPort));
|
|
||||||
}
|
|
||||||
|
|
||||||
private byte[] CreateMdnsQuery(string serviceName)
|
|
||||||
{
|
|
||||||
ushort transactionId = 0;
|
|
||||||
ushort flags = 0x0100;
|
|
||||||
ushort questions = 1;
|
|
||||||
byte[] header = new byte[12];
|
|
||||||
Array.Copy(
|
|
||||||
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)transactionId)),
|
|
||||||
0,
|
|
||||||
header,
|
|
||||||
0,
|
|
||||||
2
|
|
||||||
);
|
|
||||||
Array.Copy(
|
|
||||||
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)flags)),
|
|
||||||
0,
|
|
||||||
header,
|
|
||||||
2,
|
|
||||||
2
|
|
||||||
);
|
|
||||||
Array.Copy(
|
|
||||||
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)questions)),
|
|
||||||
0,
|
|
||||||
header,
|
|
||||||
4,
|
|
||||||
2
|
|
||||||
);
|
|
||||||
|
|
||||||
byte[] name = EncodeName(serviceName);
|
|
||||||
byte[] query = new byte[header.Length + name.Length + 4];
|
|
||||||
Array.Copy(header, query, header.Length);
|
|
||||||
Array.Copy(name, 0, query, header.Length, name.Length);
|
|
||||||
|
|
||||||
query[query.Length - 4] = 0x00;
|
|
||||||
query[query.Length - 3] = 0x0C;
|
|
||||||
query[query.Length - 2] = 0x00;
|
|
||||||
query[query.Length - 1] = 0x01;
|
|
||||||
|
|
||||||
return query;
|
|
||||||
}
|
|
||||||
|
|
||||||
private byte[] EncodeName(string name)
|
|
||||||
{
|
|
||||||
string[] parts = name.Split('.');
|
|
||||||
byte[] result = new byte[name.Length + 2];
|
|
||||||
int offset = 0;
|
|
||||||
|
|
||||||
foreach (string part in parts)
|
|
||||||
{
|
|
||||||
result[offset++] = (byte)part.Length;
|
|
||||||
Array.Copy(Encoding.UTF8.GetBytes(part), 0, result, offset, part.Length);
|
|
||||||
offset += part.Length;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
result[offset] = 0;
|
private void SendMdnsQuery(string serviceName)
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnReceive(IAsyncResult result)
|
|
||||||
{
|
|
||||||
if (udpClient == null)
|
|
||||||
{
|
{
|
||||||
return;
|
byte[] query = CreateMdnsQuery(serviceName);
|
||||||
|
Debug.Log($"Sending mDNS query for {serviceName}");
|
||||||
|
|
||||||
|
udpClient.Send(query, query.Length, new IPEndPoint(IPAddress.Parse(multicastAddress), multicastPort));
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
private byte[] CreateMdnsQuery(string serviceName)
|
||||||
{
|
{
|
||||||
IPEndPoint remoteEndPoint = new IPEndPoint(IPAddress.Any, multicastPort);
|
ushort transactionId = 0;
|
||||||
byte[] receivedBytes = udpClient.EndReceive(result, ref remoteEndPoint);
|
ushort flags = 0x0100;
|
||||||
|
ushort questions = 1;
|
||||||
ushort flags = BitConverter.ToUInt16(new byte[] { receivedBytes[3], receivedBytes[2] }, 0);
|
byte[] header = new byte[12];
|
||||||
if (flags == 0x0100)
|
Array.Copy(
|
||||||
{
|
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)transactionId)),
|
||||||
udpClient?.BeginReceive(OnReceive, null);
|
0,
|
||||||
return;
|
header,
|
||||||
}
|
0,
|
||||||
|
2
|
||||||
ParseMdnsResponse(receivedBytes);
|
);
|
||||||
|
Array.Copy(
|
||||||
udpClient?.BeginReceive(OnReceive, null);
|
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)flags)),
|
||||||
}
|
0,
|
||||||
catch (Exception ex)
|
header,
|
||||||
{
|
2,
|
||||||
Debug.LogError($"Error receiving UDP message: {ex.Message}");
|
2
|
||||||
}
|
);
|
||||||
}
|
Array.Copy(
|
||||||
|
BitConverter.GetBytes((ushort)IPAddress.HostToNetworkOrder((short)questions)),
|
||||||
private void AddMdnsService()
|
0,
|
||||||
{
|
header,
|
||||||
if (receivedIp != null && receivedPort != null && receivedHost != null && receivedPath != null)
|
4,
|
||||||
{
|
2
|
||||||
MdnsService currentService = new MdnsService(
|
|
||||||
receivedIp,
|
|
||||||
int.Parse(receivedPort),
|
|
||||||
receivedPath,
|
|
||||||
receivedHost
|
|
||||||
);
|
);
|
||||||
serviceQueue.Enqueue(currentService);
|
|
||||||
Debug.Log($"Added service: {currentService}");
|
|
||||||
receivedIp = null;
|
|
||||||
receivedPort = null;
|
|
||||||
receivedPath = null;
|
|
||||||
receivedHost = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void ParseMdnsResponse(byte[] data)
|
byte[] name = EncodeName(serviceName);
|
||||||
{
|
byte[] query = new byte[header.Length + name.Length + 4];
|
||||||
int offset = 12;
|
Array.Copy(header, query, header.Length);
|
||||||
ushort questions = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 4));
|
Array.Copy(name, 0, query, header.Length, name.Length);
|
||||||
ushort answerRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 6));
|
|
||||||
ushort additionalRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 10));
|
|
||||||
|
|
||||||
for (int i = 0; i < questions; i++)
|
query[query.Length - 4] = 0x00;
|
||||||
{
|
query[query.Length - 3] = 0x0C;
|
||||||
offset = SkipName(data, offset);
|
query[query.Length - 2] = 0x00;
|
||||||
offset += 4;
|
query[query.Length - 1] = 0x01;
|
||||||
|
|
||||||
|
return query;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < answerRRs; i++)
|
private byte[] EncodeName(string name)
|
||||||
{
|
{
|
||||||
offset = ParseRecord(data, offset);
|
string[] parts = name.Split('.');
|
||||||
}
|
byte[] result = new byte[name.Length + 2];
|
||||||
|
int offset = 0;
|
||||||
|
|
||||||
for (int i = 0; i < additionalRRs; i++)
|
foreach (string part in parts)
|
||||||
{
|
|
||||||
offset = ParseRecord(data, offset);
|
|
||||||
AddMdnsService();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int ParseRecord(byte[] data, int offset)
|
|
||||||
{
|
|
||||||
string name;
|
|
||||||
(name, offset) = ReadName(data, offset);
|
|
||||||
|
|
||||||
ushort recordType = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset));
|
|
||||||
ushort recordClass = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 2));
|
|
||||||
uint ttl = (uint)IPAddress.NetworkToHostOrder(BitConverter.ToInt32(data, offset + 4));
|
|
||||||
ushort dataLength = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 8));
|
|
||||||
offset += 10;
|
|
||||||
|
|
||||||
if (ttl == 0)
|
|
||||||
{
|
|
||||||
Debug.LogWarning($"Zero TTL for {name}");
|
|
||||||
return offset + dataLength;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (recordType == 1) // A Record
|
|
||||||
{
|
|
||||||
IPAddress ipAddress = new IPAddress(new ArraySegment<byte>(data, offset, dataLength).ToArray());
|
|
||||||
receivedIp = ipAddress.ToString();
|
|
||||||
receivedHost = name;
|
|
||||||
}
|
|
||||||
else if (recordType == 12) // PTR Record
|
|
||||||
{
|
|
||||||
string target;
|
|
||||||
(target, _) = ReadName(data, offset);
|
|
||||||
}
|
|
||||||
else if (recordType == 33) // SRV Record
|
|
||||||
{
|
|
||||||
ushort priority = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset));
|
|
||||||
ushort weight = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 2));
|
|
||||||
ushort port = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 4));
|
|
||||||
string target;
|
|
||||||
(target, _) = ReadName(data, offset + 6);
|
|
||||||
receivedPort = port.ToString();
|
|
||||||
}
|
|
||||||
else if (recordType == 16) // TXT Record
|
|
||||||
{
|
|
||||||
string txtData = Encoding.UTF8.GetString(data, offset, dataLength);
|
|
||||||
if (txtData.Contains("path"))
|
|
||||||
{
|
{
|
||||||
receivedPath = txtData.Split('=')[1];
|
result[offset++] = (byte)part.Length;
|
||||||
|
Array.Copy(Encoding.UTF8.GetBytes(part), 0, result, offset, part.Length);
|
||||||
|
offset += part.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
result[offset] = 0;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnReceive(IAsyncResult result)
|
||||||
|
{
|
||||||
|
if (udpClient == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
IPEndPoint remoteEndPoint = new IPEndPoint(IPAddress.Any, multicastPort);
|
||||||
|
byte[] receivedBytes = udpClient.EndReceive(result, ref remoteEndPoint);
|
||||||
|
|
||||||
|
ushort flags = BitConverter.ToUInt16(new byte[] { receivedBytes[3], receivedBytes[2] }, 0);
|
||||||
|
if (flags == 0x0100)
|
||||||
|
{
|
||||||
|
udpClient?.BeginReceive(OnReceive, null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ParseMdnsResponse(receivedBytes);
|
||||||
|
|
||||||
|
udpClient?.BeginReceive(OnReceive, null);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Debug.LogError($"Error receiving UDP message: {ex.Message}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (recordType == 47) // NSEC Record
|
|
||||||
{
|
|
||||||
// Debug.Log($"NSEC Record: {name}");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Debug.Log($"Unknown Record Type {recordType} for {name}");
|
|
||||||
}
|
|
||||||
|
|
||||||
return offset + dataLength;
|
private void AddMdnsService()
|
||||||
}
|
|
||||||
|
|
||||||
private (string, int) ReadName(byte[] data, int offset)
|
|
||||||
{
|
|
||||||
StringBuilder name = new StringBuilder();
|
|
||||||
int originalOffset = offset;
|
|
||||||
bool jumped = false;
|
|
||||||
|
|
||||||
while (data[offset] != 0)
|
|
||||||
{
|
{
|
||||||
if ((data[offset] & 0xC0) == 0xC0)
|
if (receivedIp != null && receivedPort != null && receivedHost != null && receivedPath != null)
|
||||||
{
|
{
|
||||||
if (!jumped)
|
MdnsService currentService = new MdnsService(
|
||||||
|
receivedIp,
|
||||||
|
int.Parse(receivedPort),
|
||||||
|
receivedPath,
|
||||||
|
receivedHost
|
||||||
|
);
|
||||||
|
serviceQueue.Enqueue(currentService);
|
||||||
|
Debug.Log($"Added service: {currentService}");
|
||||||
|
receivedIp = null;
|
||||||
|
receivedPort = null;
|
||||||
|
receivedPath = null;
|
||||||
|
receivedHost = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ParseMdnsResponse(byte[] data)
|
||||||
|
{
|
||||||
|
int offset = 12;
|
||||||
|
ushort questions = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 4));
|
||||||
|
ushort answerRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 6));
|
||||||
|
ushort additionalRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 10));
|
||||||
|
|
||||||
|
for (int i = 0; i < questions; i++)
|
||||||
|
{
|
||||||
|
offset = SkipName(data, offset);
|
||||||
|
offset += 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < answerRRs; i++)
|
||||||
|
{
|
||||||
|
offset = ParseRecord(data, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < additionalRRs; i++)
|
||||||
|
{
|
||||||
|
offset = ParseRecord(data, offset);
|
||||||
|
AddMdnsService();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private int ParseRecord(byte[] data, int offset)
|
||||||
|
{
|
||||||
|
string name;
|
||||||
|
(name, offset) = ReadName(data, offset);
|
||||||
|
|
||||||
|
ushort recordType = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset));
|
||||||
|
ushort recordClass = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 2));
|
||||||
|
uint ttl = (uint)IPAddress.NetworkToHostOrder(BitConverter.ToInt32(data, offset + 4));
|
||||||
|
ushort dataLength = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 8));
|
||||||
|
offset += 10;
|
||||||
|
|
||||||
|
if (ttl == 0)
|
||||||
|
{
|
||||||
|
Debug.LogWarning($"Zero TTL for {name}");
|
||||||
|
return offset + dataLength;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recordType == 1) // A Record
|
||||||
|
{
|
||||||
|
IPAddress ipAddress = new IPAddress(new ArraySegment<byte>(data, offset, dataLength).ToArray());
|
||||||
|
receivedIp = ipAddress.ToString();
|
||||||
|
receivedHost = name;
|
||||||
|
}
|
||||||
|
else if (recordType == 12) // PTR Record
|
||||||
|
{
|
||||||
|
string target;
|
||||||
|
(target, _) = ReadName(data, offset);
|
||||||
|
}
|
||||||
|
else if (recordType == 33) // SRV Record
|
||||||
|
{
|
||||||
|
ushort priority = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset));
|
||||||
|
ushort weight = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 2));
|
||||||
|
ushort port = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, offset + 4));
|
||||||
|
string target;
|
||||||
|
(target, _) = ReadName(data, offset + 6);
|
||||||
|
receivedPort = port.ToString();
|
||||||
|
}
|
||||||
|
else if (recordType == 16) // TXT Record
|
||||||
|
{
|
||||||
|
string txtData = Encoding.UTF8.GetString(data, offset, dataLength);
|
||||||
|
if (txtData.Contains("path"))
|
||||||
{
|
{
|
||||||
originalOffset = offset + 2;
|
receivedPath = txtData.Split('=')[1];
|
||||||
}
|
}
|
||||||
offset = ((data[offset] & 0x3F) << 8) | data[offset + 1];
|
}
|
||||||
jumped = true;
|
else if (recordType == 47) // NSEC Record
|
||||||
|
{
|
||||||
|
// Debug.Log($"NSEC Record: {name}");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
int length = data[offset++];
|
Debug.Log($"Unknown Record Type {recordType} for {name}");
|
||||||
name.Append(Encoding.UTF8.GetString(data, offset, length) + ".");
|
|
||||||
offset += length;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return offset + dataLength;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (name.ToString().TrimEnd('.'), jumped ? originalOffset : offset + 1);
|
private (string, int) ReadName(byte[] data, int offset)
|
||||||
}
|
|
||||||
|
|
||||||
private int SkipName(byte[] data, int offset)
|
|
||||||
{
|
|
||||||
while (data[offset] != 0)
|
|
||||||
{
|
{
|
||||||
if ((data[offset] & 0xC0) == 0xC0)
|
StringBuilder name = new StringBuilder();
|
||||||
{
|
int originalOffset = offset;
|
||||||
return offset + 2;
|
bool jumped = false;
|
||||||
}
|
|
||||||
offset += data[offset] + 1;
|
|
||||||
}
|
|
||||||
return offset + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void Update()
|
while (data[offset] != 0)
|
||||||
{
|
|
||||||
if (serviceQueue.Count > 0)
|
|
||||||
{
|
|
||||||
Debug.Log($"Queued services: {serviceQueue.Count}");
|
|
||||||
|
|
||||||
while (serviceQueue.Count > 0)
|
|
||||||
{
|
{
|
||||||
MdnsService service = serviceQueue.Dequeue();
|
if ((data[offset] & 0xC0) == 0xC0)
|
||||||
if (service == null)
|
|
||||||
{
|
{
|
||||||
continue;
|
if (!jumped)
|
||||||
|
{
|
||||||
|
originalOffset = offset + 2;
|
||||||
|
}
|
||||||
|
offset = (data[offset] & 0x3F) << 8 | data[offset + 1];
|
||||||
|
jumped = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
int length = data[offset++];
|
||||||
|
name.Append(Encoding.UTF8.GetString(data, offset, length) + ".");
|
||||||
|
offset += length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (name.ToString().TrimEnd('.'), jumped ? originalOffset : offset + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private int SkipName(byte[] data, int offset)
|
||||||
|
{
|
||||||
|
while (data[offset] != 0)
|
||||||
|
{
|
||||||
|
if ((data[offset] & 0xC0) == 0xC0)
|
||||||
|
{
|
||||||
|
return offset + 2;
|
||||||
|
}
|
||||||
|
offset += data[offset] + 1;
|
||||||
|
}
|
||||||
|
return offset + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Update()
|
||||||
|
{
|
||||||
|
if (serviceQueue.Count > 0)
|
||||||
|
{
|
||||||
|
Debug.Log($"Queued services: {serviceQueue.Count}");
|
||||||
|
|
||||||
|
while (serviceQueue.Count > 0)
|
||||||
|
{
|
||||||
|
MdnsService service = serviceQueue.Dequeue();
|
||||||
|
if (service == null)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
Debug.Log($"Invoking action with: {service}");
|
||||||
|
action?.Invoke(service);
|
||||||
}
|
}
|
||||||
Debug.Log($"Invoking action with: {service}");
|
|
||||||
action?.Invoke(service);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private void OnDestroy()
|
private void OnDestroy()
|
||||||
{
|
{
|
||||||
StopListening();
|
StopListening();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void StopListening()
|
private void StopListening()
|
||||||
{
|
{
|
||||||
udpClient?.DropMulticastGroup(IPAddress.Parse(multicastAddress));
|
udpClient?.DropMulticastGroup(IPAddress.Parse(multicastAddress));
|
||||||
udpClient?.Close();
|
udpClient?.Close();
|
||||||
udpClient = null;
|
udpClient = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -4,100 +4,103 @@ using Microsoft.MixedReality.Toolkit.Utilities;
|
||||||
using TMPro;
|
using TMPro;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class ServicesListPopulator : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class ServicesListPopulator : MonoBehaviour
|
||||||
private ScrollingObjectCollection scrollView;
|
|
||||||
|
|
||||||
[SerializeField]
|
|
||||||
private GameObject dynamicItem;
|
|
||||||
|
|
||||||
[SerializeField]
|
|
||||||
private GridObjectCollection gridObjectCollection;
|
|
||||||
|
|
||||||
private bool isVisible = true;
|
|
||||||
|
|
||||||
private const string apiUrlPrefix = "API URL: ";
|
|
||||||
private const string hostPrefix = "Host: ";
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Adds an item to the table from a service.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="service"></param>
|
|
||||||
/// <param name="action"></param>
|
|
||||||
public void AddItemFromService(MdnsService service, Action action)
|
|
||||||
{
|
{
|
||||||
GameObject itemInstance = Instantiate(dynamicItem, gridObjectCollection.transform);
|
[SerializeField]
|
||||||
itemInstance.SetActive(true);
|
private ScrollingObjectCollection scrollView;
|
||||||
|
|
||||||
Debug.Log($"Adding service to table: {service}");
|
[SerializeField]
|
||||||
TextMeshPro[] textMeshes = itemInstance.GetComponentsInChildren<TextMeshPro>();
|
private GameObject dynamicItem;
|
||||||
if (textMeshes.Length < 2)
|
|
||||||
|
[SerializeField]
|
||||||
|
private GridObjectCollection gridObjectCollection;
|
||||||
|
|
||||||
|
private bool isVisible = true;
|
||||||
|
|
||||||
|
private const string apiUrlPrefix = "API URL: ";
|
||||||
|
private const string hostPrefix = "Host: ";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adds an item to the table from a service.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="service"></param>
|
||||||
|
/// <param name="action"></param>
|
||||||
|
public void AddItemFromService(MdnsService service, Action action)
|
||||||
{
|
{
|
||||||
Debug.LogError("Not enough text meshes found in dynamic item");
|
GameObject itemInstance = Instantiate(dynamicItem, gridObjectCollection.transform);
|
||||||
return;
|
itemInstance.SetActive(true);
|
||||||
}
|
|
||||||
textMeshes[0].text = $"{apiUrlPrefix}http://{service.IpAddress}:{service.Port}{service.Path}";
|
Debug.Log($"Adding service to table: {service}");
|
||||||
textMeshes[1].text = $"{hostPrefix}{service.Host}";
|
TextMeshPro[] textMeshes = itemInstance.GetComponentsInChildren<TextMeshPro>();
|
||||||
itemInstance
|
if (textMeshes.Length < 2)
|
||||||
.GetComponentInChildren<Interactable>()
|
|
||||||
.OnClick.AddListener(() =>
|
|
||||||
{
|
{
|
||||||
Debug.Log($"Clicked on service: {service.Host}");
|
Debug.LogError("Not enough text meshes found in dynamic item");
|
||||||
action.Invoke();
|
return;
|
||||||
ToggleVisibility();
|
|
||||||
});
|
|
||||||
|
|
||||||
gridObjectCollection.UpdateCollection();
|
|
||||||
scrollView.UpdateContent();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes all items from the table.
|
|
||||||
/// </summary>
|
|
||||||
public void RemoveAllItems()
|
|
||||||
{
|
|
||||||
foreach (Transform child in gridObjectCollection.transform)
|
|
||||||
{
|
|
||||||
Destroy(child.gameObject);
|
|
||||||
}
|
|
||||||
|
|
||||||
Debug.Log("Removed all services from table");
|
|
||||||
gridObjectCollection.UpdateCollection();
|
|
||||||
scrollView.UpdateContent();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes an item from the table by service.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="service"></param>
|
|
||||||
public void RemoveItemByService(MdnsService service)
|
|
||||||
{
|
|
||||||
string apiUrl = $"{apiUrlPrefix}http://{service.IpAddress}:{service.Port}{service.Path}";
|
|
||||||
string hostname = $"{hostPrefix}{service.Host}";
|
|
||||||
|
|
||||||
foreach (Transform child in gridObjectCollection.transform)
|
|
||||||
{
|
|
||||||
TextMeshPro[] textMeshes = child.GetComponentsInChildren<TextMeshPro>();
|
|
||||||
if (textMeshes.Length >= 2 && textMeshes[0].text == apiUrl && textMeshes[1].text == hostname)
|
|
||||||
{
|
|
||||||
Debug.Log($"Removing service from table: {service}");
|
|
||||||
Destroy(child.gameObject);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
textMeshes[0].text = $"{apiUrlPrefix}http://{service.IpAddress}:{service.Port}{service.Path}";
|
||||||
|
textMeshes[1].text = $"{hostPrefix}{service.Host}";
|
||||||
|
itemInstance
|
||||||
|
.GetComponentInChildren<Interactable>()
|
||||||
|
.OnClick.AddListener(() =>
|
||||||
|
{
|
||||||
|
Debug.Log($"Clicked on service: {service.Host}");
|
||||||
|
action.Invoke();
|
||||||
|
ToggleVisibility();
|
||||||
|
});
|
||||||
|
|
||||||
|
gridObjectCollection.UpdateCollection();
|
||||||
|
scrollView.UpdateContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
gridObjectCollection.UpdateCollection();
|
/// <summary>
|
||||||
scrollView.UpdateContent();
|
/// Removes all items from the table.
|
||||||
}
|
/// </summary>
|
||||||
|
public void RemoveAllItems()
|
||||||
|
{
|
||||||
|
foreach (Transform child in gridObjectCollection.transform)
|
||||||
|
{
|
||||||
|
Destroy(child.gameObject);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
Debug.Log("Removed all services from table");
|
||||||
/// Toggles the visibility of the table.
|
gridObjectCollection.UpdateCollection();
|
||||||
/// </summary>
|
scrollView.UpdateContent();
|
||||||
public void ToggleVisibility()
|
}
|
||||||
{
|
|
||||||
isVisible = !isVisible;
|
|
||||||
|
|
||||||
gameObject.SetActive(isVisible);
|
/// <summary>
|
||||||
|
/// Removes an item from the table by service.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="service"></param>
|
||||||
|
public void RemoveItemByService(MdnsService service)
|
||||||
|
{
|
||||||
|
string apiUrl = $"{apiUrlPrefix}http://{service.IpAddress}:{service.Port}{service.Path}";
|
||||||
|
string hostname = $"{hostPrefix}{service.Host}";
|
||||||
|
|
||||||
|
foreach (Transform child in gridObjectCollection.transform)
|
||||||
|
{
|
||||||
|
TextMeshPro[] textMeshes = child.GetComponentsInChildren<TextMeshPro>();
|
||||||
|
if (textMeshes.Length >= 2 && textMeshes[0].text == apiUrl && textMeshes[1].text == hostname)
|
||||||
|
{
|
||||||
|
Debug.Log($"Removing service from table: {service}");
|
||||||
|
Destroy(child.gameObject);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gridObjectCollection.UpdateCollection();
|
||||||
|
scrollView.UpdateContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Toggles the visibility of the table.
|
||||||
|
/// </summary>
|
||||||
|
public void ToggleVisibility()
|
||||||
|
{
|
||||||
|
isVisible = !isVisible;
|
||||||
|
|
||||||
|
gameObject.SetActive(isVisible);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -7,13 +7,15 @@ using UnityEngine.Windows.WebCam;
|
||||||
using Windows.Storage;
|
using Windows.Storage;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
public class VideoCaptureHandler : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class VideoCaptureHandler : MonoBehaviour
|
||||||
private GameObject videoCaptureButton = null;
|
{
|
||||||
|
[SerializeField]
|
||||||
|
private GameObject videoCaptureButton = null;
|
||||||
|
|
||||||
private Interactable videoCaptureButtonInteractable = null;
|
private Interactable videoCaptureButtonInteractable = null;
|
||||||
private VideoCapture videoCapture = null;
|
private VideoCapture videoCapture = null;
|
||||||
|
|
||||||
#if WINDOWS_UWP && !UNITY_EDITOR
|
#if WINDOWS_UWP && !UNITY_EDITOR
|
||||||
private const string freeSpace = "System.FreeSpace";
|
private const string freeSpace = "System.FreeSpace";
|
||||||
|
@ -49,122 +51,123 @@ public class VideoCaptureHandler : MonoBehaviour
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Starts recording a video.
|
/// Starts recording a video.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public void StartRecordingVideo()
|
public void StartRecordingVideo()
|
||||||
{
|
{
|
||||||
#if WINDOWS_UWP && !UNITY_EDITOR
|
#if WINDOWS_UWP && !UNITY_EDITOR
|
||||||
VideoCapture.CreateAsync(true, OnVideoCaptureCreated);
|
VideoCapture.CreateAsync(true, OnVideoCaptureCreated);
|
||||||
#else
|
#else
|
||||||
VideoCapture.CreateAsync(false, OnVideoCaptureCreated);
|
VideoCapture.CreateAsync(false, OnVideoCaptureCreated);
|
||||||
#endif
|
#endif
|
||||||
if (videoCaptureButtonInteractable == null)
|
if (videoCaptureButtonInteractable == null)
|
||||||
{
|
|
||||||
videoCaptureButtonInteractable = videoCaptureButton.GetComponent<Interactable>();
|
|
||||||
}
|
|
||||||
videoCaptureButtonInteractable.IsToggled = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Stops recording a video.
|
|
||||||
/// </summary>
|
|
||||||
public void StopRecordingVideo()
|
|
||||||
{
|
|
||||||
videoCapture.StopRecordingAsync(OnStoppedRecordingVideo);
|
|
||||||
videoCaptureButtonInteractable.IsToggled = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Toggles the recording of a video.
|
|
||||||
/// </summary>
|
|
||||||
public void ToggleRecordingVideo()
|
|
||||||
{
|
|
||||||
if (videoCapture == null)
|
|
||||||
{
|
|
||||||
StartRecordingVideo();
|
|
||||||
}
|
|
||||||
else if (videoCapture.IsRecording)
|
|
||||||
{
|
|
||||||
StopRecordingVideo();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnVideoCaptureCreated(VideoCapture videoCapture)
|
|
||||||
{
|
|
||||||
if (videoCapture != null)
|
|
||||||
{
|
|
||||||
this.videoCapture = videoCapture;
|
|
||||||
|
|
||||||
Resolution cameraResolution = new Resolution();
|
|
||||||
foreach (Resolution resolution in VideoCapture.SupportedResolutions)
|
|
||||||
{
|
{
|
||||||
if (resolution.width * resolution.height > cameraResolution.width * cameraResolution.height)
|
videoCaptureButtonInteractable = videoCaptureButton.GetComponent<Interactable>();
|
||||||
{
|
|
||||||
cameraResolution = resolution;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
videoCaptureButtonInteractable.IsToggled = true;
|
||||||
|
}
|
||||||
|
|
||||||
float cameraFramerate = 0.0f;
|
/// <summary>
|
||||||
foreach (float framerate in VideoCapture.GetSupportedFrameRatesForResolution(cameraResolution))
|
/// Stops recording a video.
|
||||||
|
/// </summary>
|
||||||
|
public void StopRecordingVideo()
|
||||||
|
{
|
||||||
|
videoCapture.StopRecordingAsync(OnStoppedRecordingVideo);
|
||||||
|
videoCaptureButtonInteractable.IsToggled = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Toggles the recording of a video.
|
||||||
|
/// </summary>
|
||||||
|
public void ToggleRecordingVideo()
|
||||||
|
{
|
||||||
|
if (videoCapture == null)
|
||||||
{
|
{
|
||||||
if (framerate > cameraFramerate)
|
StartRecordingVideo();
|
||||||
{
|
}
|
||||||
cameraFramerate = framerate;
|
else if (videoCapture.IsRecording)
|
||||||
}
|
{
|
||||||
|
StopRecordingVideo();
|
||||||
}
|
}
|
||||||
|
|
||||||
CameraParameters cameraParameters = new CameraParameters();
|
|
||||||
cameraParameters.hologramOpacity = 0.75f;
|
|
||||||
cameraParameters.frameRate = cameraFramerate;
|
|
||||||
cameraParameters.cameraResolutionWidth = cameraResolution.width;
|
|
||||||
cameraParameters.cameraResolutionHeight = cameraResolution.height;
|
|
||||||
cameraParameters.pixelFormat = CapturePixelFormat.BGRA32;
|
|
||||||
|
|
||||||
this.videoCapture.StartVideoModeAsync(
|
|
||||||
cameraParameters,
|
|
||||||
VideoCapture.AudioState.ApplicationAndMicAudio,
|
|
||||||
OnStartedVideoCaptureMode
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
private void OnVideoCaptureCreated(VideoCapture videoCapture)
|
||||||
{
|
{
|
||||||
Debug.LogError("Failed to create VideoCapture instance");
|
if (videoCapture != null)
|
||||||
}
|
{
|
||||||
}
|
this.videoCapture = videoCapture;
|
||||||
|
|
||||||
private void OnStartedVideoCaptureMode(VideoCapture.VideoCaptureResult result)
|
Resolution cameraResolution = new Resolution();
|
||||||
{
|
foreach (Resolution resolution in VideoCapture.SupportedResolutions)
|
||||||
if (result.success)
|
{
|
||||||
|
if (resolution.width * resolution.height > cameraResolution.width * cameraResolution.height)
|
||||||
|
{
|
||||||
|
cameraResolution = resolution;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
float cameraFramerate = 0.0f;
|
||||||
|
foreach (float framerate in VideoCapture.GetSupportedFrameRatesForResolution(cameraResolution))
|
||||||
|
{
|
||||||
|
if (framerate > cameraFramerate)
|
||||||
|
{
|
||||||
|
cameraFramerate = framerate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CameraParameters cameraParameters = new CameraParameters();
|
||||||
|
cameraParameters.hologramOpacity = 0.75f;
|
||||||
|
cameraParameters.frameRate = cameraFramerate;
|
||||||
|
cameraParameters.cameraResolutionWidth = cameraResolution.width;
|
||||||
|
cameraParameters.cameraResolutionHeight = cameraResolution.height;
|
||||||
|
cameraParameters.pixelFormat = CapturePixelFormat.BGRA32;
|
||||||
|
|
||||||
|
this.videoCapture.StartVideoModeAsync(
|
||||||
|
cameraParameters,
|
||||||
|
VideoCapture.AudioState.ApplicationAndMicAudio,
|
||||||
|
OnStartedVideoCaptureMode
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug.LogError("Failed to create VideoCapture instance");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnStartedVideoCaptureMode(VideoCapture.VideoCaptureResult result)
|
||||||
{
|
{
|
||||||
string filename = string.Format(
|
if (result.success)
|
||||||
"WebView_{0}.mp4",
|
{
|
||||||
DateTime.UtcNow.ToString("yyyy-MM-ddTHHmmssZ")
|
string filename = string.Format(
|
||||||
);
|
"WebView_{0}.mp4",
|
||||||
string filepath = Path.Combine(Application.persistentDataPath, filename);
|
DateTime.UtcNow.ToString("yyyy-MM-ddTHHmmssZ")
|
||||||
Debug.Log("Saving video to: " + filepath);
|
);
|
||||||
|
string filepath = Path.Combine(Application.persistentDataPath, filename);
|
||||||
|
Debug.Log("Saving video to: " + filepath);
|
||||||
|
|
||||||
videoCapture.StartRecordingAsync(filepath, OnStartedRecordingVideo);
|
videoCapture.StartRecordingAsync(filepath, OnStartedRecordingVideo);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private void OnStartedRecordingVideo(VideoCapture.VideoCaptureResult result)
|
private void OnStartedRecordingVideo(VideoCapture.VideoCaptureResult result)
|
||||||
{
|
{
|
||||||
Debug.Log("Started recording video");
|
Debug.Log("Started recording video");
|
||||||
#if WINDOWS_UWP && !UNITY_EDITOR
|
#if WINDOWS_UWP && !UNITY_EDITOR
|
||||||
StartCoroutine(CheckAvailableStorageSpace());
|
StartCoroutine(CheckAvailableStorageSpace());
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnStoppedRecordingVideo(VideoCapture.VideoCaptureResult result)
|
private void OnStoppedRecordingVideo(VideoCapture.VideoCaptureResult result)
|
||||||
{
|
{
|
||||||
Debug.Log("Stopped recording video");
|
Debug.Log("Stopped recording video");
|
||||||
videoCapture.StopVideoModeAsync(OnStoppedVideoCaptureMode);
|
videoCapture.StopVideoModeAsync(OnStoppedVideoCaptureMode);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnStoppedVideoCaptureMode(VideoCapture.VideoCaptureResult result)
|
private void OnStoppedVideoCaptureMode(VideoCapture.VideoCaptureResult result)
|
||||||
{
|
{
|
||||||
videoCapture.Dispose();
|
videoCapture.Dispose();
|
||||||
videoCapture = null;
|
videoCapture = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -3,43 +3,46 @@ using Microsoft.MixedReality.WebView;
|
||||||
using TMPro;
|
using TMPro;
|
||||||
using UnityEngine;
|
using UnityEngine;
|
||||||
|
|
||||||
public class WebViewBrowser : MonoBehaviour
|
namespace WebViewStream
|
||||||
{
|
{
|
||||||
[SerializeField]
|
public class WebViewBrowser : MonoBehaviour
|
||||||
private TMP_InputField URLField;
|
|
||||||
|
|
||||||
private void Start()
|
|
||||||
{
|
{
|
||||||
var webViewComponent = gameObject.GetComponent<WebView>();
|
[SerializeField]
|
||||||
webViewComponent.GetWebViewWhenReady(
|
private TMP_InputField URLField;
|
||||||
(IWebView webView) =>
|
|
||||||
{
|
|
||||||
URLField.onSubmit.AddListener((text) => LoadUrl(webView));
|
|
||||||
|
|
||||||
webView.Navigated += OnNavigated;
|
private void Start()
|
||||||
|
{
|
||||||
if (webView.Page != null)
|
var webViewComponent = gameObject.GetComponent<WebView>();
|
||||||
|
webViewComponent.GetWebViewWhenReady(
|
||||||
|
(IWebView webView) =>
|
||||||
{
|
{
|
||||||
URLField.text = webView.Page.AbsoluteUri;
|
URLField.onSubmit.AddListener((text) => LoadUrl(webView));
|
||||||
|
|
||||||
|
webView.Navigated += OnNavigated;
|
||||||
|
|
||||||
|
if (webView.Page != null)
|
||||||
|
{
|
||||||
|
URLField.text = webView.Page.AbsoluteUri;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnNavigated(string path)
|
||||||
|
{
|
||||||
|
URLField.text = path;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void LoadUrl(IWebView webView)
|
||||||
|
{
|
||||||
|
if (Uri.TryCreate(URLField.text, UriKind.Absolute, out Uri uriResult))
|
||||||
|
{
|
||||||
|
webView.Load(uriResult);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug.LogWarning("Invalid URL entered.");
|
||||||
}
|
}
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnNavigated(string path)
|
|
||||||
{
|
|
||||||
URLField.text = path;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void LoadUrl(IWebView webView)
|
|
||||||
{
|
|
||||||
if (Uri.TryCreate(URLField.text, UriKind.Absolute, out Uri uriResult))
|
|
||||||
{
|
|
||||||
webView.Load(uriResult);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Debug.LogWarning("Invalid URL entered.");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue