带有静态服务器端呈现(静态 SSR)的 ASP.NET Core Blazor JavaScript
注意
此版本不是本文的最新版本。 有关当前版本,请参阅本文的 .NET 9 版本。
本文介绍如何在带有静态服务器端呈现(静态 SSR)和JS的 Blazor Web App 中加载 JavaScript ()。
某些应用依赖于 JS 来执行特定于每个页面的初始化任务。 使用 Blazor 的增强导航功能时(用户可避免重新加载整个页面),每次发生增强的页面导航时,特定于页面的 JS 可能不会按预期再次执行。
为了避免此问题,我们不建议依赖应用于组件的布局文件之外的特定于页面的 <script>
元素。 相反,脚本应注册 afterWebStarted
JS 初始化器 来执行初始化逻辑,并使用事件监听器来监听由增强的导航引起的页面更新。
事件
在以下事件侦听器示例中,{CALLBACK}
占位符是回调函数。
增强导航开始 (
enhancednavigationstart
) 会在增强导航发生之前触发回调。blazor.addEventListener("enhancednavigationstart", {CALLBACK});
增强导航结束 (
enhancednavigationend
) 会在增强导航发生后触发回调:blazor.addEventListener("enhancednavigationend", {CALLBACK});
增强导航页面加载 (
enhancedload
) 会在每次因增强导航而更新页面触发回调函数,包括流式处理更新:blazor.addEventListener("enhancedload", {CALLBACK});
为了避免此问题,我们不建议依赖应用于组件的布局文件之外的特定于页面的 <script>
元素。 相反,脚本应注册 afterWebStarted
JS 初始化器 来执行初始化逻辑,并使用事件监听器监听因增强导航而引发的页面更新。
blazor.addEventListener("enhancedload", {CALLBACK});
在前面的示例中,{CALLBACK}
占位符是回调函数。
增强的页面加载脚本示例
以下示例演示了一种配置 JS 代码的方法,以便代码在最初加载或更新具有增强导航功能的静态呈现页面时运行。
以下 PageWithScript
组件示例是应用中的一个组件,它要求脚本使用静态 SSR 和增强型导航运行。 以下组件示例包括 PageScript
类库 (RCL) 中的 Razor 组件,该组件将在本文后面部分添加到解决方案中。
Components/Pages/PageWithScript.razor
:
@page "/page-with-script"
@using BlazorPageScript
<PageTitle>Enhanced Load Script Example</PageTitle>
<PageScript Src="./Components/Pages/PageWithScript.razor.js" />
Welcome to my page.
在 Blazor Web App 中,添加以下并置 JS 文件:
onLoad
将在脚本添加到页面时调用。onUpdate
将在增强型更新后脚本仍存在于页面时调用。onDispose
将在增强型更新后从页面删除脚本时调用。
Components/Pages/PageWithScript.razor.js
:
export function onLoad() {
console.log('Loaded');
}
export function onUpdate() {
console.log('Updated');
}
export function onDispose() {
console.log('Disposed');
}
在Razor类库 (RCL)(示例 RCL 命名为 BlazorPageScript
)中,添加以下模块。
wwwroot/BlazorPageScript.lib.module.js
:
const pageScriptInfoBySrc = new Map();
function registerPageScriptElement(src) {
if (!src) {
throw new Error('Must provide a non-empty value for the "src" attribute.');
}
let pageScriptInfo = pageScriptInfoBySrc.get(src);
if (pageScriptInfo) {
pageScriptInfo.referenceCount++;
} else {
pageScriptInfo = { referenceCount: 1, module: null };
pageScriptInfoBySrc.set(src, pageScriptInfo);
initializePageScriptModule(src, pageScriptInfo);
}
}
function unregisterPageScriptElement(src) {
if (!src) {
return;
}
const pageScriptInfo = pageScriptInfoBySrc.get(src);
if (!pageScriptInfo) {
return;
}
pageScriptInfo.referenceCount--;
}
async function initializePageScriptModule(src, pageScriptInfo) {
if (src.startsWith("./")) {
src = new URL(src.substr(2), document.baseURI).toString();
}
const module = await import(src);
if (pageScriptInfo.referenceCount <= 0) {
return;
}
pageScriptInfo.module = module;
module.onLoad?.();
module.onUpdate?.();
}
function onEnhancedLoad() {
for (const [src, { module, referenceCount }] of pageScriptInfoBySrc) {
if (referenceCount <= 0) {
module?.onDispose?.();
pageScriptInfoBySrc.delete(src);
}
}
for (const { module } of pageScriptInfoBySrc.values()) {
module?.onUpdate?.();
}
}
export function afterWebStarted(blazor) {
customElements.define('page-script', class extends HTMLElement {
static observedAttributes = ['src'];
attributeChangedCallback(name, oldValue, newValue) {
if (name !== 'src') {
return;
}
this.src = newValue;
unregisterPageScriptElement(oldValue);
registerPageScriptElement(newValue);
}
disconnectedCallback() {
unregisterPageScriptElement(this.src);
}
});
blazor.addEventListener('enhancedload', onEnhancedLoad);
}
在 RCL 中添加以下 PageScript
组件。
PageScript.razor
:
<page-script src="@Src"></page-script>
@code {
[Parameter]
[EditorRequired]
public string Src { get; set; } = default!;
}
组件 PageScript
通常在页面的顶层运行。
如果将 PageScript
组件放置在应用布局中(例如 MainLayout.razor
)(这会导致使用该布局的页面之间共享 PageScript
),则该组件仅在整个页面重新加载后运行 onLoad
,并在发生任何增强型页面更新(包括增强型导航)时运行 onUpdate
。
若要在页面之间重复使用同一模块,但在每个页面更改中调用 onLoad
和 onDispose
回调,请将查询字符串追加到脚本末尾,以便将其识别为其他模块。 应用可以采用将组件名称用作查询字符串值的约定。 在以下示例中,查询字符串为“counter
”,因为此 PageScript
组件引用放置在 Counter
组件中。 这只是一个建议,你可以使用你喜欢的任何查询字符串方案。
<PageScript Src="./Components/Pages/PageWithScript.razor.js?counter" />
要监视特定 DOM 元素中的更改,请使用客户端上 MutationObserver
中的 JS 模式。 有关详细信息,请参阅 ASP.NET Core BlazorJavaScript 互操作性(JS 互操作)。
不使用 RCL 的示例实现
本文中所述的方法可以直接在 Blazor Web App 中实现,而无需使用 Razor 类库 (RCL)。 有关示例,请参阅在 ASP.NET Core Blazor Web App 中为 TOTP 验证器应用启用 QR 码生成。