3

提前致谢。

在我的 WPF 应用程序中,我能够使用 WebBrowser 的 InvokeScript(String, Object[]) 方法成功地将消息从 WPF 传递到 JavaScript。我在 Object[] 参数中传递了我的消息,JS 代码很好地接受了它。(见下面的代码)

我如何在 WebView2 中实现相同的功能,有什么方法可以像在 WebBrowser 控件中一样将参数传递给 JS 代码?

窗口.xaml

 <Grid>
            <DockPanel> 
                <StackPanel>        
                <TextBox x:Name="txtMessageFromWPF" Width="150" FontSize="20"></TextBox>
                <Button x:Name="btnCallDocument" Click="btnCallDocument_Click" Content="CallDocument" />
                </StackPanel>
                <WebBrowser x:Name="webBrowser" DockPanel.Dock="Top" Margin="30"/>           
            </DockPanel>
        </Grid>

窗口.xaml.cs

        private void btnCallDocument_Click(object sender, RoutedEventArgs e)
        {
            webBrowser.InvokeScript("WriteMessageFromWPF", new object[] { this.txtMessageFromWPF.Text });
        }

JS代码:

 <script type="text/javascript">
            function getAlert()
            {
                alert("Hi the page is loaded!!!");
            }
            window.onload = getAlert;
            
            function WriteMessageFromWPF(message){
                document.write("Message from WPF: " + message);
            }
        </script>
4

2 回答 2

6

更新:扩展的第 2 版更简单、更通用。它使用 JSON 作为“中间站”——使用 NewtonSoft 或内置的 JSON 转换器。

这是我所做的扩展(只需创建一个类文件并粘贴它)。

ExecuteScriptFunctionAsync构建必要的字符串,ExecuteScriptAsync然后执行它:

//using Microsoft.Web.WebView2.WinForms; //Uncomment the one you use
//using Microsoft.Web.WebView2.Wpf; //Uncomment the one you use
using Newtonsoft.Json;
using System.Threading.Tasks;

public static class Extensions
{
    public static async Task<string> ExecuteScriptFunctionAsync(this WebView2 webView2, string functionName, params object[] parameters)
    {
        string script = functionName + "(";
        for (int i = 0; i < parameters.Length; i++)
        {
            script += JsonConvert.SerializeObject(parameters[i]);
            if (i < parameters.Length - 1)
            {
                script += ", ";
            }
        }
        script += ");";
        return await webView2.ExecuteScriptAsync(script);
    }
}

将 javascript 函数名称作为第一个参数传递,然后是函数参数。

该代码使得可以有任意数量的所有类型的参数,这些参数可以序列化为 JSON:objectarraystringall numbersboolean

使用示例(来自问题):

private async void btnCallDocument_Click(object sender, RoutedEventArgs e)
{
    await webBrowser.ExecuteScriptFunctionAsync("WriteMessageFromWPF", this.txtMessageFromWPF.Text);
}

另一个例子(这会将窗口滚动到底部):

await webView21.ExecuteScriptFunctionAsync("window.scrollTo", 0, 10000);
于 2020-07-10T18:51:36.893 回答
2

目前在 WebView2 中没有直接等效的 InvokeScript。最接近的是CoreWebView2.ExecuteScriptAsync。此方法需要一个脚本字符串来运行,并以 JSON 字符串的形式异步返回脚本执行的结果。

如果要调用函数,则需要创建一个脚本字符串来执行此操作并将其提供给 ExecuteScriptAsync。在您的示例中,这意味着生成字符串WriteMessageFromWPF("...")并将其提供给 ExecuteScriptAsync。

棘手的部分是正确编码您的函数参数。如果字符串参数中有引号,那么您需要斜杠转义引号,否则会产生不正确的脚本。您应该能够使用 String.Replace 斜线转义引号:

string script = "WriteMessageFromWPF(\"" + 
    this.txtMessageFromWPF.Text.Replace("\"", "\\\"")
    + "\")";
await webview2.CoreWebView2.ExecuteScriptAsync(script);
于 2020-07-10T16:55:47.380 回答