从 ASP.NET Core Blazor 中的.NET 方法调用 JavaScript 函数
- Blazor 应用程序可以从.NET 方法调用 JavaScript 函数,并可以从 JavaScript 函数调用.NET 方法。这些场景称为 JavaScript 互操作性(JS interop)。
- 本文介绍了从.NET 调用 JavaScript 函数的方法。有关如何从 JavaScript 调用.NET 方法的信息,请参见从 ASP.NET Core Blazor 中的 JavaScript 函数调用.NET 方法。
使用
- 要从.NET 调用 JavaScript,请使用 IJSRuntime 抽象。要发出 JS 互操作调用,请将 IJSRuntime 抽象注入到组件中。该 InvokeAsync方法获取您希望调用的 JavaScript 函数的标识符以及任意数量的 JSON 可序列化的参数。功能标识符是相对于全局范围(window)的。如果您要呼叫 window.someScope.someFunction,标识符为 someScope.someFunction。无需在调用函数之前先进行注册。返回类型 T 还必须是 JSON 可序列化的。T 应该与最能映射到返回的 JSON 类型的.NET 类型匹配。
- 对于启用了预渲染的 Blazor Server 应用,在初始预渲染期间无法调用 JavaScript。必须将 JavaScript 互操作调用推迟到与浏览器建立连接之后。有关更多信息,请参见检测 Blazor Server 应用何时渲染部分。
- 以下示例基于基于 JavaScript 的解码器 TextDecoder。该示例演示了如何从 C#方法调用 JavaScript 函数。JavaScript 函数从 C#方法接受一个字节数组,对该数组进行解码,然后将文本返回给组件以进行显示。
- 在 wwwroot / index.html(Blazor WebAssembly)或 Pages / _Host.cshtml(Blazor Server)的元素内,提供一个 JavaScript 函数,该函数用于解码传递的数组并返回解码后的值:TextDecoder
<script>
window.convertArray = win1251Array => {
var win1251decoder = new TextDecoder('windows-1251')
var bytes = new Uint8Array(win1251Array)
var decodedArray = win1251decoder.decode(bytes)
console.log(decodedArray)
return decodedArray
}
</script>
JavaScript 代码(例如前面示例中显示的代码)也可以从 JavaScript 文件(.js)加载,并引用该脚本文件:
<script src="exampleJsInterop.js"></script>
-
以下组件:
- 选择组件按钮(“ 转换数组”)时 convertArray 使用调用 JavaScript 函数。JSRuntime
- 调用 JavaScript 函数后,将传递的数组转换为字符串。字符串将返回到组件以显示。
@page "/call-js-example"
@inject IJSRuntime JSRuntime;
<h1>Call JavaScript Function Example</h1>
<button type="button" class="btn btn-primary" @onclick="ConvertArray">
Convert Array
</button>
<p class="mt-2" >
<span class="badge badge-success">
@_convertedText
</span>
</p>
@code {
// Quote (c)2005 Universal Pictures: Serenity
// https://www.uphe.com/movies/serenity
// David Krumholtz on IMDB: https://www.imdb.com/name/nm0472710/
private MarkupString _convertedText =
new MarkupString("Select the <b>Convert Array</b> button.");
private uint[] _quoteArray = new uint[]
{
60, 101, 109, 62, 67, 97, 110, 39, 116, 32, 115, 116, 111, 112, 32,
116, 104, 101, 32, 115, 105, 103, 110, 97, 108, 44, 32, 77, 97,
108, 46, 60, 47, 101, 109, 62, 32, 45, 32, 77, 114, 46, 32, 85, 110,
105, 118, 101, 114, 115, 101, 10, 10,
};
private async Task ConvertArray()
{
var text =
await JSRuntime.InvokeAsync<string>("convertArray", _quoteArray);
_convertedText = new MarkupString(text);
StateHasChanged();
}
}
IJSRuntime
-
要使用 IJSRuntime 抽象,请采用以下任何一种方法:
- 将 IJSRuntime 抽象注入 Razor 组件(.razor):
@inject IJSRuntime JSRuntime
@code {
protected override void OnInitialized()
{
StocksService.OnStockTickerUpdated += stockUpdate =>
{
JSRuntime.InvokeVoidAsync("handleTickerChanged",
stockUpdate.symbol, stockUpdate.price);
};
}
}
在 wwwroot / index.html(Blazor WebAssembly)或 Pages / _Host.cshtml(Blazor Server)的元素内,提供 JavaScript 函数。该函数使用调用,但不返回值:handleTickerChangedIJSRuntime.InvokeVoidAsync
<script>
window.handleTickerChanged = (symbol, price) => {
// ... client-side processing/display code ...
}
</script>
- 将 IJSRuntime 抽象注入到类(.cs)中:
public class JsInteropClasses
{
private readonly IJSRuntime \_jsRuntime;
public JsInteropClasses(IJSRuntime jsRuntime)
{
_jsRuntime = jsRuntime;
}
public ValueTask<string> TickerChanged(string data)
{
return _jsRuntime.InvokeAsync<string>(
"handleTickerChanged",
stockUpdate.symbol,
stockUpdate.price);
}
}
在 wwwroot / index.html(Blazor WebAssembly)或 Pages / _Host.cshtml(Blazor Server)的元素内,提供 JavaScript 函数。该函数使用调用并返回一个值:handleTickerChangedJSRuntime.InvokeAsync
<script>
window.handleTickerChanged = (symbol, price) => {
// ... client-side processing/display code ...
return 'Done!'
}
</script>
- 为了使用 BuildRenderTree 动态生成内容,请使用以下[Inject]属性:
[Inject]
IJSRuntime JSRuntime { get; set; }
-
在此主题附带的客户端示例应用程序中,该应用程序可以使用两个 JavaScript 函数,这些函数与 DOM 交互以接收用户输入并显示欢迎消息:
- showPrompt –产生提示以接受用户输入(用户名),并将名称返回给呼叫者。
- displayWelcome-分配从呼叫者到 DOM 对象与一个欢迎消息 id 的 welcome。
-
wwwroot / exampleJsInterop.js:
window.exampleJsFunctions = {
showPrompt: function (text) {
return prompt(text, 'Type your name here');
},
displayWelcome: function (welcomeMessage) {
document.getElementById('welcome').innerText = welcomeMessage;
},
returnArrayAsyncJs: function () {
DotNet.invokeMethodAsync('BlazorSample', 'ReturnArrayAsync')
.then(data => {
data.push(4);
console.log(data);
});
},
sayHello: function (dotnetHelper) {
return dotnetHelper.invokeMethodAsync('SayHello')
.then(r => console.log(r));
}
};
- 将
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<title>Blazor WebAssembly Sample</title>
<base href="/" />
<link href="css/bootstrap/bootstrap.min.css" rel="stylesheet" />
<link href="css/site.css" rel="stylesheet" />
</head>
<body>
<app>Loading...</app>
<div id="blazor-error-ui">
An unhandled error has occurred.
<a href="" class="reload">Reload</a>
<a class="dismiss">🗙</a>
</div>
<script src="_framework/blazor.webassembly.js"></script>
<script src="exampleJsInterop.js"></script>
</body>
</html>
- Pages / _Host.cshtml(Blazor 服务器):
@page "/" @namespace BlazorSample.Pages @addTagHelper \*,
Microsoft.AspNetCore.Mvc.TagHelpers @{ Layout = null; }
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Blazor Server Sample</title>
<base href="~/" />
<link rel="stylesheet" href="css/bootstrap/bootstrap.min.css" />
<link href="css/site.css" rel="stylesheet" />
</head>
<body>
<app>
<component type="typeof(App)" render-mode="ServerPrerendered" />
</app>
<div id="blazor-error-ui">
<environment include="Staging,Production">
An error has occurred. This application may no longer respond until
reloaded.
</environment>
<environment include="Development">
An unhandled exception has occurred. See browser dev tools for details.
</environment>
<a href="" class="reload">Reload</a>
<a class="dismiss">🗙</a>
</div>
<script src="_framework/blazor.server.js"></script>
<script src="exampleJsInterop.js"></script>
</body>
</html>
- 不要
@page "/JSInterop"
@using BlazorSample.JsInteropClasses
@inject IJSRuntime JSRuntime
<h1>JavaScript Interop</h1>
<h2>Invoke JavaScript functions from .NET methods</h2>
<button type="button" class="btn btn-primary" @onclick="TriggerJsPrompt">
Trigger JavaScript Prompt
</button>
<h3 id="welcome" ></h3>
@code {
public async Task TriggerJsPrompt()
{
var name = await JSRuntime.InvokeAsync<string>(
"exampleJsFunctions.showPrompt",
"What's your name?");
await JSRuntime.InvokeVoidAsync(
"exampleJsFunctions.displayWelcome",
$"Hello {name}! Welcome to Blazor!");
}
}
- 当 TriggerJsPrompt 通过选择组件的执行触发的 JavaScript 提示按钮时,JavaScript showPrompt 在所提供的功能的 wwwroot / exampleJsInterop.js 文件被调用。
- 该 showPrompt 函数接受经过 HTML 编码并返回到组件的用户输入(用户名)。该组件将用户名存储在本地变量中 name。
- 存储在其中的字符串 name 被合并到欢迎消息中,该消息将传递给 JavaScript 函数 displayWelcome,该函数将欢迎消息呈现为标题标记。
调用无效的 JavaScript 函数
- 返回 void(0)/ void 0 或 undefined 的 JavaScript 函数用调用 IJSRuntime.InvokeVoidAsync。
检测 Blazor Server 应用何时渲染
- 在渲染 Blazor Server 应用程序时,由于尚未建立与浏览器的连接,因此无法执行某些操作(如调用 JavaScript)。预渲染时,组件可能需要进行不同的渲染。
- 若要将 JavaScript 互操作调用延迟到与浏览器建立连接之后,可以使用 OnAfterRenderAsync 组件生命周期事件。仅在完全渲染应用程序并建立客户端连接后才调用此事件。
@using Microsoft.JSInterop @inject IJSRuntime JSRuntime
<div @ref="divElement">Text during render</div>
@code { private ElementReference divElement; protected override async Task
OnAfterRenderAsync(bool firstRender) { if (firstRender) { await
JSRuntime.InvokeVoidAsync( "setElementText", divElement, "Text after render"); }
} }
- 对于前面的示例代码,请 setElementText 在 wwwroot / index.html(Blazor WebAssembly)或 Pages / _Host.cshtml(Blazor Server)的元素内提供一个 JavaScript 函数。该函数使用调用,但不返回值:IJSRuntime.InvokeVoidAsync
<script>
window.setElementText = (element, text) => (element.innerText = text)
</script>
-
警告:前面的示例仅出于演示目的直接修改了文档对象模型(DOM)。在大多数情况下,建议不要使用 JavaScript 直接修改 DOM,因为 JavaScript 可能会干扰 Blazor 的更改跟踪。
-
以下组件演示了如何以与预渲染兼容的方式将 JavaScript 互操作用作组件初始化逻辑的一部分。该组件表明可以从内部触发渲染更新 OnAfterRenderAsync。在这种情况下,开发人员必须避免创建无限循环。
仅在,而不是在任何较早的生命周期方法中使用 where JSRuntime.InvokeAsync,因为在呈现组件之前没有 JavaScript 元素。ElementRefOnAfterRenderAsync
调用 StateHasChanged 以使用从 JavaScript 互操作调用获得的新状态来重新呈现组件。该代码不会创建无限循环,因为 StateHasChanged 仅当 infoFromJsis 时才调用 null。
网页 HTML
复制
@page “/prerendered-interop”
@using Microsoft.AspNetCore.Components
@using Microsoft.JSInterop
@inject IJSRuntime JSRuntime
Get value via JS interop call: @(infoFromJs ?? "No value yet")
Set value via JS interop call:
@code {
private string infoFromJs;
private ElementReference divElement;
protected override async Task OnAfterRenderAsync(bool firstRender)
{
if (firstRender && infoFromJs == null)
{
infoFromJs = await JSRuntime.InvokeAsync<string>(
"setElementText", divElement, "Hello from interop call!");
StateHasChanged();
}
}
}
对于前面的示例代码,请 setElementText 在 wwwroot / index.html(Blazor WebAssembly)或 Pages / _Host.cshtml(Blazor Server)的元素内提供一个 JavaScript 函数。该函数使用调用并返回一个值:IJSRuntime.InvokeAsync
的 HTML
复制
-
警告:前面的示例仅出于演示目的直接修改了文档对象模型(DOM)。在大多数情况下,建议不要使用 JavaScript 直接修改 DOM,因为 JavaScript 可能会干扰 Blazor 的更改跟踪。
捕获对元素的引用
某些 JS 互操作方案需要引用 HTML 元素。例如,UI 库可能需要元素引用进行初始化,或者您可能需要在诸如 focus 或的元素上调用类似命令的 API play。
使用以下方法捕获对组件中 HTML 元素的引用:
将@ref 属性添加到 HTML 元素。
定义 ElementReference 名称与@ref 属性值匹配的类型的字段。
以下示例显示了捕获对 username <input>元素的引用:
剃刀
复制
<input @ref=“username” … />
@code {
ElementReference username;
}
警告
仅使用元素引用来改变不与 Blazor 交互的空元素的内容。当第三方 API 向元素提供内容时,此方案很有用。因为 Blazor 不与元素交互,所以 Blazor 的元素表示与 DOM 之间不会发生冲突。
在以下示例中,突变无序列表()的内容很危险,ul 因为 Blazor 与 DOM 交互以填充此元素的列表项(
):剃刀
复制
- @foreach (var item in Todos) {
- @item.Text }
就.NET 代码而言,an ElementReference 是一个不透明的句柄。在只有你可以做的事情 ElementReference 就是通过 JS 互操作,让它通过 JavaScript 代码。当您这样做时,JavaScript 端代码会收到一个 HTMLElement 实例,该实例可与普通 DOM API 一起使用。
例如,以下代码定义了.NET 扩展方法,该方法允许将焦点设置在元素上:
exampleJsInterop.js:
的 JavaScript
复制
window.exampleJsFunctions = {
focusElement : function (element) {
element.focus();
}
}
要调用不返回值的 JavaScript 函数,请使用 IJSRuntime.InvokeVoidAsync。以下代码通过调用捕获的前面的 JavaScript 函数,将焦点放在用户名输入上 ElementReference:
剃刀
复制
@inject IJSRuntime JSRuntime
<input @ref="_username" />
<button @onclick=“SetFocus”>Set focus on username
@code {
private ElementReference _username;
public async Task SetFocus()
{
await JSRuntime.InvokeVoidAsync(
"exampleJsFunctions.focusElement", _username);
}
}
要使用扩展方法,请创建一个接收 IJSRuntime 实例的静态扩展方法:
C#
复制
public static async Task Focus(this ElementReference elementRef, IJSRuntime jsRuntime)
{
await jsRuntime.InvokeVoidAsync(
“exampleJsFunctions.focusElement”, elementRef);
}
该 Focus 方法直接在对象上调用。下面的示例假定该 Focus 方法可从 JsInteropClasses 名称空间使用:
剃刀
复制
@inject IJSRuntime JSRuntime
@using JsInteropClasses
<input @ref="_username" />
<button @onclick=“SetFocus”>Set focus on username
@code {
private ElementReference _username;
public async Task SetFocus()
{
await _username.Focus(JSRuntime);
}
}
重要
该 username 组件被渲染后变量只填充。如果将未填充的 ElementReference 内容传递给 JavaScript 代码,则 JavaScript 代码会收到的值 null。要在组件完成渲染(设置对元素的初始焦点)之后操纵元素引用,请使用 OnAfterRenderAsync 或 OnAfterRender 组件生命周期方法。
使用泛型类型并返回值时,请使用 ValueTask :
C#
复制
public static ValueTask GenericMethod(this ElementReference elementRef,
IJSRuntime jsRuntime)
{
return jsRuntime.InvokeAsync(
“exampleJsFunctions.doSomethingGeneric”, elementRef);
}
GenericMethod 直接在具有类型的对象上调用。以下示例假定 GenericMethod 可以从 JsInteropClasses 名称空间获得:
剃刀
复制
@inject IJSRuntime JSRuntime
@using JsInteropClasses
<input @ref="_username" />
<button @onclick=“OnClickMethod”>Do something generic
_returnValue: @_returnValue
@code {
private ElementReference _username;
private string _returnValue;
private async Task OnClickMethod()
{
_returnValue = await _username.GenericMethod<string>(JSRuntime);
}
}
跨组件的参考元素
一个 ElementReference 时,才能保证在组件的有效 OnAfterRender 方法(和元件参考是 struct),所以一个元件引用不能部件之间传递。
为了使父组件使元素引用可用于其他组件,父组件可以:
允许子组件注册回调。
在 OnAfterRender 事件期间使用传递的元素引用来调用已注册的回调。间接地,这种方法允许子组件与父组件的元素引用进行交互。
以下 Blazor WebAssembly 示例说明了该方法。
在的 wwwroot 文件/ index.html 的:
的 HTML
复制
在的 wwwroot 文件/ index.html 的:
的 HTML
复制
Pages / Index.razor(父组件):
剃刀
复制
@page “/”
Welcome to your new app.
C#
复制
using System;
using System.Collections.Generic;
using Microsoft.AspNetCore.Components;
namespace BlazorSample.Pages
{
public partial class Index :
ComponentBase, IObservable, IDisposable
{
private bool _disposing;
private IList<IObserver> _subscriptions =
new List<IObserver>();
private ElementReference _title;
protected override void OnAfterRender(bool firstRender)
{
base.OnAfterRender(firstRender);
foreach (var subscription in _subscriptions)
{
try
{
subscription.OnNext(_title);
}
catch (Exception)
{
throw;
}
}
}
public void Dispose()
{
_disposing = true;
foreach (var subscription in _subscriptions)
{
try
{
subscription.OnCompleted();
}
catch (Exception)
{
}
}
_subscriptions.Clear();
}
public IDisposable Subscribe(IObserver<ElementReference> observer)
{
if (_disposing)
{
throw new InvalidOperationException("Parent being disposed");
}
_subscriptions.Add(observer);
return new Subscription(observer, this);
}
private class Subscription : IDisposable
{
public Subscription(IObserver<ElementReference> observer, Index self)
{
Observer = observer;
Self = self;
}
public IObserver<ElementReference> Observer { get; }
public Index Self { get; }
public void Dispose()
{
Self._subscriptions.Remove(Observer);
}
}
}
}
共享/SurveyPrompt.razor(子组件):
剃刀
复制
@inject IJSRuntime JS
<span class="text-nowrap">
Please take our
<a target="_blank" class="font-weight-bold"
href="https://go.microsoft.com/fwlink/?linkid=2109206">brief survey</a>
</span>
and tell us what you think.
@code {
[Parameter]
public string Title { get; set; }
}
共享/SurveyPrompt.razor.cs:
C#
复制
using System;
using Microsoft.AspNetCore.Components;
namespace BlazorSample.Shared
{
public partial class SurveyPrompt :
ComponentBase, IObserver, IDisposable
{
private IDisposable _subscription = null;
[Parameter]
public IObservable<ElementReference> Parent { get; set; }
protected override void OnParametersSet()
{
base.OnParametersSet();
if (_subscription != null)
{
_subscription.Dispose();
}
_subscription = Parent.Subscribe(this);
}
public void OnCompleted()
{
_subscription = null;
}
public void OnError(Exception error)
{
_subscription = null;
}
public void OnNext(ElementReference value)
{
JS.InvokeAsync<object>(
"setElementClass", new object[] { value, "red" });
}
public void Dispose()
{
_subscription?.Dispose();
}
}
}
强化 JS 互操作调用
JS 互操作可能由于网络错误而失败,应视为不可靠。默认情况下,Blazor Server 应用程序在一分钟后使服务器上的 JS 互操作调用超时。如果应用程序可以忍受更激进的超时(例如 10 秒),请使用以下方法之一设置超时:
在全局中 Startup.ConfigureServices,指定超时时间:
C#
复制
services.AddServerSideBlazor(
options => options.JSInteropDefaultCallTimeout = TimeSpan.FromSeconds({SECONDS}));
对于组件代码中的每次调用,单个调用可以指定超时:
C#
复制
var result = await JSRuntime.InvokeAsync(“MyJSOperation”,
TimeSpan.FromSeconds({SECONDS}), new[] { “Arg1” });
有关资源耗尽的更多信息,请参见 Secure ASP.NET Core Blazor Server 应用程序。
在类库中共享互操作代码
JS 互操作代码可以包含在类库中,该库使您可以在 NuGet 包中共享代码。
类库处理将 JavaScript 资源嵌入到已生成的程序集中。JavaScript 文件位于 wwwroot 文件夹中。构建库时,该工具负责嵌入资源。
在应用程序的项目文件中引用生成的 NuGet 包的方式与引用任何 NuGet 包的方式相同。还原软件包后,应用程序代码可以像 C#一样调用 JavaScript。
有关更多信息,请参见 ASP.NET Core Razor 组件类库。
额外资源
从 ASP.NET Core Blazor 中的 JavaScript 函数调用.NET 方法
InteropComponent.razor 示例(dotnet / AspNetCore GitHub 存储库,3.1 版本分支)
在 Blazor Server 应用程序中执行大数据传输