mixedreality/com.microsoft.mixedreality..../Runtime/WebView.cs

348 lines
12 KiB
C#

// <copyright file="WebView.cs" company="Microsoft Corporation">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
namespace Microsoft.MixedReality.WebView
{
using System;
using System.Threading.Tasks;
using UnityEngine;
using UnityEngine.Scripting;
#if UNITY_EDITOR
using UnityEditor;
#endif
/// <summary>
/// A high-level script that facilitates adding WebView's to your scene.
/// Use either the built-in WebView prefab or your own quad to position and
/// render your WebView's contents with the help of WebView.cs.
/// </summary>
[Preserve]
[AddComponentMenu("WebView")]
// Necessary because WebViewSystem attaches the WebView texture to
// the GameObject's renderer's material.
[RequireComponent(typeof(Renderer))]
public class WebView : MonoBehaviour
{
[Tooltip("Scale the brightness. A value of 1 matches what you see in your browser.")]
[SerializeField]
[OnRangeChangedCall(0f, 2f, "OnBrightnessScaleChanged")]
private float brightnessScale = 1.0f;
public enum ImageQuality
{
Low,
Good,
Great,
Excellent
}
[Tooltip("Configure image quality. Higher quality looks better but can decrease performance.")]
[SerializeField]
[OnChangedCall("OnImageQualityChanged")]
private ImageQuality imageQuality = ImageQuality.Low;
private float TextureScale
{
get
{
if (webView is IWithContentScale)
{
switch (this.imageQuality)
{
case ImageQuality.Low:
return 1;
case ImageQuality.Good:
return 3;
case ImageQuality.Great:
return 6;
case ImageQuality.Excellent:
return 9;
}
}
return 1;
}
}
[Tooltip("The URL that is first loaded when the WebView initializes in the scene.")]
[SerializeField]
[OnChangedCall("OnAbsoluteUrlChanged")]
private string currentURL = "https://www.microsoft.com";
/// <summary>
/// This flag signifies if the current url field is unsynced with the IWebView instance.
/// </summary>
private bool currentURLDirty = false;
private IWebView webView = null;
private Exception creationException = null;
/// <summary>
/// Invoked once the WebView plugin instance is initialized and has begun loading the currentUrl.
/// </summary>
public event EventHandler<IWebView> WebViewReady;
private readonly TaskCompletionSource<IWebView> webViewTCS = new TaskCompletionSource<IWebView>();
/// <summary>
/// An awaitable Task that returns the WebView's IWebView instance after it has been initialized.
/// </summary>
public Task<IWebView> WebViewTask
{
get { return webViewTCS.Task; }
}
/// <summary>
/// Invoked once the WebView has finished navigating to a URL.
/// The event arguments are a tuple of the WebView instance and the URL to which it navigated.
/// </summary>
public event EventHandler<(IWebView, string)> WebViewNavigated;
public event EventHandler<Exception> WebViewCreationFailed;
public Exception WebViewCreationException { get { return creationException; } }
/// <summary>
/// Gets the URL currently loaded by the WebView.
/// </summary>
public Uri CurrentURL
{
get { return this.webView?.Page; }
}
public void Awake()
{
this.CreateAndConfigureWebView();
this.UpdateBrightness();
}
public void PostMessage(string message)
{
if (this.webView is IWithPostMessage withPostMessage)
{
withPostMessage.PostMessage(message);
}
}
public void NavigateToString(string htmlContent)
{
if (this.webView is IWithHTMLInjection withHTMLInjection)
{
withHTMLInjection.LoadHTMLContent(htmlContent);
}
}
/// <summary>
/// Loads an absolute URL, such as about:blank or https://www.microsoft.com
/// The URL string will be validated to make sure it's well-formed and absolute.
/// If the internal WebView instance is not yet initialized, the load will be enqueued for when the instance is ready.
/// Calls to this method will not overwrite currentUrl, they will load in sequence.
/// </summary>
/// <param name="url">The URL to load.</param>
public void Load(string url)
{
if (string.IsNullOrWhiteSpace(url))
{
Debug.Log($"Current url is empty. Ignoring load request.");
}
else
{
this.Load(new Uri(url));
}
}
/// <summary>
/// Uri overload for Load(string url).
/// </summary>
/// <param name="uri">The URI to load.</param>
/// <exception cref="ArgumentException">If the URI is invalid.</exception>
public void Load(Uri uri)
{
if (uri is null)
{
throw new ArgumentNullException(nameof(uri));
}
var absolutePath = uri.AbsoluteUri;
if (!uri.IsWellFormedOriginalString() || !uri.IsAbsoluteUri)
{
throw new ArgumentException($"\"{absolutePath}\" is invalid: it must be a well-formed, absolute Uri.");
}
if (this.webView is null)
{
// Enqueue a load on the webview.
this.WebViewReady += new EventHandler<IWebView>((object s, IWebView wv) =>
{
wv.Load(uri);
});
}
else
{
this.webView.Load(uri);
}
}
public IWebView GetWebView()
{
return this.webView;
}
/// <summary>
/// Takes a callback that is invoked either immediately if the IWebView instance has already been created, or once the IWebView instance is created.
/// </summary>
/// <param name="callback">The callback, which can be a lambda or any function taking an IWebView instance as the only argument.</param>
public void GetWebViewWhenReady(Action<IWebView> callback)
{
if (this.webView is null)
{
this.WebViewReady += (object s, IWebView wv) => callback(wv);
}
else
{
callback(this.webView);
}
}
public void GetWebViewCreationFailed(Action<Exception> callback)
{
if (this.webView is null && this.creationException is null)
{
this.WebViewCreationFailed += (object s, Exception e) => callback(e);
}
else if (this.creationException is not null)
{
callback(this.creationException);
}
}
private void OnAbsoluteUrlChanged()
{
this.currentURLDirty = !this.CurrentURL?.AbsoluteUri.Equals(this.currentURL) ?? false;
}
private void OnImageQualityChanged()
{
if (webView is IWithContentScale)
{
MatchTextureSizeToQuad();
}
}
private void OnBrightnessScaleChanged()
{
this.UpdateBrightness();
}
private void UpdateBrightness()
{
GetComponent<Renderer>().material.SetFloat("_Brightness", brightnessScale);
}
private void CreateAndConfigureWebView()
{
string parentHWNDHint = null;
#if UNITY_EDITOR_WIN
parentHWNDHint = "UnityEditor.GameView:UnityGUIViewWndClass";
#endif
var newWebView = WebViewSystem.CreateWebView(this.gameObject, 1280, 720, parentHWNDHint);
if (newWebView is null)
{
creationException = new Exception("Failed to create webview");
this.WebViewCreationFailed?.Invoke(this, creationException);
return;
}
newWebView.Navigated += path =>
{
this.currentURL = path;
this.currentURLDirty = false;
this.WebViewNavigated?.Invoke(this, (newWebView, path));
};
newWebView.OnceCreated.ContinueWith((task) =>
{
// Disposed before the new webview could be created.
if (this == null)
{
newWebView.Dispose();
}
else if (task.IsFaulted)
{
creationException = task.Exception;
this.WebViewCreationFailed?.Invoke(this, creationException);
}
else
{
this.webView = newWebView;
MatchTextureSizeToQuad();
this.WebViewReady?.Invoke(this, this.webView);
this.webViewTCS.SetResult(this.webView);
}
}, TaskScheduler.FromCurrentSynchronizationContext());
this.Load(this.currentURL);
}
private void MatchTextureSizeToQuad()
{
// Adjust width and height
Vector3 lossyScale = transform.lossyScale;
float scaleHorizontal = lossyScale.x;
// Sometimes quads are aligned with their vertical component on y, and some on z.
float scaleVertical = Math.Max(lossyScale.y, lossyScale.z);
int smallDimension = (int)(720 * TextureScale);
// For D3D11, texture dimensions must be between 1 and 16384, inclusively.
// This is a reasonable cap for all platforms.
int maxSize = 16384;
int width, height;
bool landscape = scaleHorizontal > scaleVertical;
var aspectRatio = landscape ? scaleHorizontal / scaleVertical : scaleVertical / scaleHorizontal;
int bigDimension = (int)(smallDimension * aspectRatio);
// Adjust if the larger dimension is too big.
if (bigDimension > maxSize)
{
bigDimension = maxSize;
smallDimension = (int)(bigDimension / aspectRatio);
}
width = landscape ? bigDimension : smallDimension;
height = landscape ? smallDimension : bigDimension;
(webView as IWithContentScale)?.SetContentScale(TextureScale);
webView.Resize(width, height);
}
private void OnDestroy()
{
this.webView?.Dispose();
}
#if UNITY_EDITOR
[CustomEditor(typeof(WebView))]
private class WebViewInspector : Editor
{
public override void OnInspectorGUI()
{
var webViewComponent = (WebView)this.target;
DrawDefaultInspector();
if (webViewComponent.currentURLDirty)
{
if (GUILayout.Button("Navigate"))
{
webViewComponent.Load(webViewComponent.currentURL);
}
if (GUILayout.Button("Cancel"))
{
webViewComponent.currentURL = webViewComponent.CurrentURL.AbsoluteUri;
webViewComponent.currentURLDirty = false;
}
}
}
}
#endif
}
}