mixedreality/com.microsoft.mixedreality..../Runtime/Platform/BaseWebView.cs

310 lines
12 KiB
C#

// <copyright file="BaseWebView.cs" company="Microsoft Corporation">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
namespace Microsoft.MixedReality.WebView
{
using System;
using System.Threading;
using System.Threading.Tasks;
using AOT;
using UnityEngine;
using System.Runtime.InteropServices;
internal abstract class BaseWebView : IWebView, IWithPostMessage, IWithMouseEvents, IWithBrowserHistory, IWithHTMLInjection, IWithContentScale, IWithNavigationFiltering
{
#region Events
public event WebView_OnNavigated Navigated;
public event WebView_OnPostMessage MessageReceived;
public event WebView_OnCanGoForwardUpdated CanGoForwardUpdated;
public event WebView_OnCanGoBackUpdated CanGoBackUpdated;
public event WebView_OnNewWindowRequested NewWindowRequested;
public event WebView_OnCloseRequested WindowCloseRequested;
public event WebView_OnNavigationBlocked NavigationBlocked;
#endregion
private readonly TaskCompletionSource<bool> whenReadyTCS = new TaskCompletionSource<bool>();
private readonly WeakReference gameObject;
private int activeResizeRequestCount = 0;
private Texture2D targetTexture;
public IntPtr InstanceId { get; private set; }
private int width;
private int height;
public int Width
{
get => width;
set { Resize(value, Height); }
}
public int Height
{
get => height;
set { Resize(Width, value); }
}
// Unity API's (e.g. UnityEngine.Behaviour::set_enabled) must be called from Unity's main thread (i.e. game thread or app thread).
// However, WebView2 callbacks are executed on WinRT's main thread.
// Use SynchronizationContext to ensure any Unity API calls are made from Unity's main thread and at the plugin layer so that
// consumers of the plugin do not run into wrong thread type violations
protected SynchronizationContext UnityGameThreadContext;
private TaskScheduler UnityGameThreadScheduler;
public GameObject GameObject => (GameObject)gameObject.Target;
public Texture2D Texture => targetTexture;
public BaseWebView(GameObject gameObject, int width, int height, string parentHWNDHint = null)
{
#if UNITY_WSA && !UNITY_EDITOR
// WebViews created in the editor don't need this check as the editor is a Win32 app.
// The UNITY_WSA refers to the currently set target platform, so we also need to check if !UNITY_EDITOR.
Debug.Assert(WebViewSystem.AppThread == System.Threading.Thread.CurrentThread, "WebView was not created on the game thread");
#endif
UnityGameThreadContext = SynchronizationContext.Current;
UnityGameThreadScheduler = TaskScheduler.FromCurrentSynchronizationContext();
this.gameObject = new WeakReference(gameObject);
this.width = width;
this.height = height;
#if UNITY_WSA
// We need to create the WebView on the main thread.
// All other WebViewNative functions are thread-safe.
UnityEngine.WSA.Application.InvokeOnUIThread(() => {
#endif
int errorCode = 0;
InstanceId = WebViewNative.InitializeWebView(width, height, parentHWNDHint, ref errorCode);
if (errorCode != 0)
{
Debug.LogError($"Failed to create native WebView control. Exception code: {errorCode}");
var creationException = Marshal.GetExceptionForHR(errorCode);
UnityGameThreadContext.Post(_ =>
{
this.whenReadyTCS.SetException(creationException);
}, null);
return;
}
WebViewNative.SetUrlChangedCallback(InstanceId, OnUrlChangedCallback);
WebViewNative.SetCanGoBackUpdatedCallback(InstanceId, OnGoBackStatusUpdated);
WebViewNative.SetCanGoForwardUpdatedCallback(InstanceId, OnGoForwardUpdated);
WebViewNative.SetPostMessageCallback(InstanceId, OnPostMessageCallback);
WebViewNative.SetReadyCallback(InstanceId, OnReadyCallback);
WebViewNative.SetTextureAvailableCallback(InstanceId, OnTextureAvailable);
WebViewNative.SetNewWindowRequestedCallback(InstanceId, OnNewWindowRequested);
WebViewNative.SetWindowCloseRequestedCallback(InstanceId, OnWindowCloseRequested);
WebViewNative.SetNavigationBlockedCallback(InstanceId, OnNavigationBlocked);
#if UNITY_WSA
}, true);
#endif
}
public Uri Page { get; private set; }
public void Load(Uri url)
{
Page = url;
WebViewNative.SetWebViewUrl(InstanceId, url.AbsoluteUri);
}
public void MouseEvent(WebViewMouseEventData mouseEvent)
{
WebViewNative.HandlePointerInput(InstanceId, mouseEvent.X, mouseEvent.Y, (int)mouseEvent.Device, (int)mouseEvent.Type, (int)mouseEvent.Button, (int)mouseEvent.WheelY);
}
public void Resize(int width, int height)
{
// Run on the same thread as SetTargetTexture so we can properly synchronize.
OnceCreated.ContinueWith(_ =>
{
targetTexture = null;
this.width = width;
this.height = height;
activeResizeRequestCount++;
WebViewNative.SetWebViewSize(InstanceId, width, height);
}, UnityGameThreadScheduler);
}
private void SetTargetTexture(IntPtr externalTexturePtr)
{
// We need this to be the case so that:
// (1) We can access Unity GameObject methods
// (2) We properly synchronize against resize requests.
Debug.Assert(SynchronizationContext.Current == UnityGameThreadContext);
// Consume resize request.
if (activeResizeRequestCount >= 1)
{
activeResizeRequestCount--;
}
// Check if we're already asking for more resizes that will invalidate the target texture.
if (activeResizeRequestCount > 0)
{
return;
}
targetTexture = Texture2D.CreateExternalTexture(width, height, TextureFormat.RGBA32, false, true, externalTexturePtr);
// Set point filtering just so we can see the pixels clearly
targetTexture.filterMode = FilterMode.Point;
targetTexture.wrapMode = TextureWrapMode.Clamp;
if (this.gameObject.Target != null)
{
var webViewRenderer = this.GameObject.GetComponent<Renderer>();
webViewRenderer.material.mainTexture = targetTexture;
}
}
public void PostMessage(string message, bool isJSON = false)
{
WebViewNative.PostWebMessage(InstanceId, message, isJSON);
}
public void LoadHTMLContent(string htmlContent)
{
WebViewNative.LoadHTMLContent(InstanceId, htmlContent);
}
public void GoBack()
{
WebViewNative.GoBackOnWebView(InstanceId);
}
public void GoForward()
{
WebViewNative.GoForwardOnWebView(InstanceId);
}
public void Dispose()
{
WebViewNative.DestroyWebView(InstanceId);
WebViewSystem.ViewDestroyed(InstanceId);
}
#region Callbacks
[MonoPInvokeCallback(typeof(WebViewNative.NewWindowRequestedDelegate))]
private static void OnNewWindowRequested(IntPtr instanceId, string uri)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ => webView.NewWindowRequested?.Invoke(uri), null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.WindowCloseRequestedDelegate))]
private static void OnWindowCloseRequested(IntPtr instanceId)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ => webView.WindowCloseRequested?.Invoke(), null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.OnReadyDelegate))]
private static void OnReadyCallback(IntPtr instanceId)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ =>
{
webView.whenReadyTCS.SetResult(true);
}, null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.TextureAvailableDelegate))]
private static void OnTextureAvailable(IntPtr instanceId, IntPtr texturePtr)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ =>
{
webView.SetTargetTexture(texturePtr);
}, null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.PostMessageToUnityDelegate))]
private static void OnPostMessageCallback(IntPtr instanceId, string message)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ => webView.MessageReceived?.Invoke(message), null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.UrlChangedDelegate))]
private static void OnUrlChangedCallback(IntPtr instanceId, string url)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ =>
{
webView.Page = new Uri(url);
webView.Navigated?.Invoke(url);
}, null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.NavigationBlockedDelegate))]
private static void OnNavigationBlocked(IntPtr instanceId, string blockedUri)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
webView?.UnityGameThreadContext.Post(_ => webView.NavigationBlocked?.Invoke(blockedUri), null);
}
public bool CanGoBack { get; private set; } = false;
public bool CanGoForward { get; private set; } = false;
[MonoPInvokeCallback(typeof(WebViewNative.NavigationButtonStatusUpdatedDelegate))]
private static void OnGoBackStatusUpdated(IntPtr instanceId, bool value)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ =>
{
webView.CanGoBack = value;
webView.CanGoBackUpdated?.Invoke(webView.CanGoBack);
}, null);
}
}
[MonoPInvokeCallback(typeof(WebViewNative.NavigationButtonStatusUpdatedDelegate))]
private static void OnGoForwardUpdated(IntPtr instanceId, bool value)
{
BaseWebView webView = WebViewSystem.FindWebView(instanceId) as BaseWebView;
if (webView != null)
{
webView.UnityGameThreadContext.Post(_ =>
{
webView.CanGoForward = value;
webView.CanGoForwardUpdated?.Invoke(webView.CanGoForward);
}, null);
}
}
#endregion
public Task OnceCreated => whenReadyTCS.Task;
public void SetContentScale(double scale)
{
WebViewNative.SetWebViewContentScale(InstanceId, scale);
}
}
}