创建 Blazor 项目

创建新的 Blazor WebAssembly App

blazor如何使用JavaScript blazor javascript_互操作

将其命名为BlazerWithTSInterop在您选择的目录中。

blazor如何使用JavaScript blazor javascript_Interop_02

仅使用 .NET 6.0 客户端,无安全性且无 PWA。

blazor如何使用JavaScript blazor javascript_缓存_03

CTRL+F5 在热重载模式下生成和运行。

blazor如何使用JavaScript blazor javascript_Interop_04

总结已经创建了一个准备演示 JavaScript 互操作演练的项目。忽略模板附带的“计数器”和“获取数据”页。此演示将仅使用主页。

实现 JavaScript 互操作

在我们开始Typescript之前,让我们看看JavaScript是如何互操作的。

1. 调用 JavaScript 浏览器 API

在 Pages/Index.razor 的所有内容替换为以下代码片段。

@page "/"

@inject IJSRuntime JS

<h1>Hello, Interop!</h1>
<hr />@Message<hr />
<h4>JS Interop</h4>
<button class="btn btn-primary" @onclick="@Prompt">Prompt</button>
<hr>

@code{
    string Message { get; set; } = string.Empty;

    async void Prompt(){

        string answer = await JS.InvokeAsync<string>("propmpt","say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing":answer)}";
    }
}

解释一下:

JS.InvokeAsync:

第一个参数是:执行JavaScript 函数的名称

第二个参数是:执行JavaScript 函数的参数

返回一个字符串

保存以在热重载模式下运行并进行测试。

blazor如何使用JavaScript blazor javascript_Interop_05

2. 调用嵌入式 JavaScript

创建新的 JavaScript 文件。为JavaScript和Typescript文件创建新的'js'文件夹。
创建新的“wwwroot/js/script.js”文件。

blazor如何使用JavaScript blazor javascript_互操作_06

添加下面的代码到script.js

function ScriptPrompt(message) {
    return prompt(message);
}

function ScriptAlert(message) {
    alert(message);
}

ScriptPrompt 和 ScriptAlert 将静态加载和全局。
其他 JavaScript 模块(包括隔离模块)可访问。

请注意,脚本方法调用浏览器 API 提示并恭敬地发出警报。

在“wwwroot/index.html”中的“webassemly.js”之后添加“script.js”作为静态资源

<body>
...
    <script src="_framework/blazor.webassembly.js"></script>
    <script src="src/script.js"></script>
...
</body>

请将所有“Pags/Index.razor”内容替换为以下代码片段。
使用操作方法添加脚本提示和脚本警报按钮。

@page "/"

@inject IJSRuntime JS

<h1>Hello, Interop!</h1><br />
<h4 style="background-color:aliceblue; padding:20px">JavaScript Interop</h4>
@Message<hr />

<button class="btn btn-primary" @onclick="@Prompt">Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptPrompt">Script Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptAlert">Script Alert</button>
<hr>

@code{
    string Message { get; set; } = string.Empty;

    async void Prompt()
    {

        string answer = await JS.InvokeAsync<string>("prompt","say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing":answer)}";

        StateHasChanged();
    }


    async void SrciptPrompt()
    {
        // 
        string answer = await JS.InvokeAsync<string>("ScriptPrompt", "say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing" : answer)}";

        StateHasChanged();
    }

    async void ScriptAlert()
    {
        await JS.InvokeAsync("ScriptAlert", "Script Alert");
    }

}

运行并测试。

blazor如何使用JavaScript blazor javascript_缓存_07

3. 调用隔离的JavaScript

创建新的 'wwwroot/js/script.module.js' JavaScript 文件。

blazor如何使用JavaScript blazor javascript_互操作_08

将下面的代码添加到'wwwroot/js/script.module.js'

export function ModulePrompt(message) {
    return ScriptPrompt(message);
}

export function ModulAlert(message) {
    return ScriptAlert(message);
}

模块方法演示如何调用全局脚本方法。

请注意“export”方法前缀。这是将代码标记为可导入的 ES 模块语法。全局嵌入式 script.js 不使用“export”。

将所有“Index.razor”内容替换为以下代码片段,以添加模块按钮和方法。

@page "/"

@inject IJSRuntime JS
@implements IAsyncDisposable

<h1>Hello, Interop!</h1><br />
<h4 style="background-color:aliceblue; padding:20px">JavaScript Interop</h4>
@Message<hr />

<button class="btn btn-primary" @onclick="@Prompt">Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptPrompt">Script Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptAlert">Script Alert</button>
<hr>

<button class="btn btn-primary" @onclick="@ModulelPrompt">Module Prompt</button>
<button class="btn btn-primary" @onclick="@ModulelAlert">Module Alert</button>
<hr>

@code{
    private IJSObjectReference module;

    string Message { get; set; } = string.Empty;

    string Version{ get => $"?v={DateTime.Now.Ticks.ToString()}"; }


    async ValueTask IAsyncDisposable.DisposeAsync()
    {
        if (module is not null)
            await module.DisposeAsync();
    }

    protected override async Task OnAfterRenderAsync(bool firstRender)
    {
        if(firstRender){
            module = await JS.InvokeAsync<IJSObjectReference>("import", $"./src/script.module.js{Version}");
        }
    }

    async void ModulelAlert()
    {
        await module.InvokeVoidAsync("ModulAlert", "Modulel Alert");
    }

    async void ModulelPrompt()
    {
        string answer = await module.InvokeAsync<string>("ModulePrompt", "Module Prompt say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing" : answer)}";
        StateHasChanged();
    }

    async void Prompt()
    {

        string answer = await JS.InvokeAsync<string>("prompt","say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing":answer)}";

        StateHasChanged();
    }

    async void ScriptPrompt()
    {
        string answer = await JS.InvokeAsync<string>("ScriptPrompt", "say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing" : answer)}";

        StateHasChanged();
    }

    async void ScriptAlert()
    {
        await JS.InvokeVoidAsync("ScriptAlert", "Script Alert");
    }

}

隔离的模型支持 IAsyncDisposable 与 DisposeAsync 在不再需要时清理模块资源。
模块在首次呈现后由 OnAfterRenderAsync 方法加载。
ModulePrompt 演示如何调用静态方法 ScriptPrompt。
ModuleAlert 演示如何从同一模块调用另一个导出的方法。

通知模块在加载时附加一个唯一参数 Version 标记:

...
string Version { get { return "?v=" + DateTime.Now.Ticks.ToString(); } }
...
module = await JS.InvokeAsync<IJSObjectReference>
                    ("import", "./js/script.module.js" + Version);
...

script.module.js 避免了由唯一参数标记缓存。

blazor如何使用JavaScript blazor javascript_互操作_09

这是一个绕过浏览器缓存的黑客,该缓存可能会在开发过程中粘附。
要重新获得缓存性能,版本值可以替换为应用程序发行版号。
然后,这将在新版本的第一次客户端运行时强制缓存刷新一次。

构建并运行。

blazor如何使用JavaScript blazor javascript_缓存_10

**总结 ** 第 2 部分介绍了如何调用独立和嵌入式 JavaScript。从 Blazor 调用 TypeScript 互操作的前身。

调试 JavaScript

现在是回顾从Visual Studio调试JavaScript的好时机

Visual Studio可能会犹豫是否要附加到Chrome调试器。这个问题不是Blazor独有的。

随着JavaScript代码和符号的增长,这一点更加明显。以下是一些可能有帮助的情况和解决方法。

在中设置断点script.js如图所示

blazor如何使用JavaScript blazor javascript_Interop_11

在调试模式 F5 下运行应用程序。
如果断点红色圆圈为空心,则不会附加调试器。

blazor如何使用JavaScript blazor javascript_Interop_12

您可以在“脚本文档”文件夹中看到缓存的文件。
单击文件以查看缓存的内容是否来自以前的版本。

blazor如何使用JavaScript blazor javascript_互操作_13

请尝试删除断点并重新应用。
调试器可能会重新附加。

应用运行时,在浏览器中按 CTR+Shift+I 可查看开发人员工具。
在“source”面板中选择 js/script.js,并在显示处设置断点。
这将触发Visual Studio调试器重新附加到Chrome。如果这不起作用,则在Chrome中调试就足够了。

blazor如何使用JavaScript blazor javascript_缓存_14

**总结 ** 第 3 部分介绍了调试和调试器附件解决方法。建议执行调试代码演练以查看操作中的互操作

实现 TypeScript 互操作

1. 调用隔离 TypeScript

创建新的 'scripts/hello.ts' TypeScript 文件。

blazor如何使用JavaScript blazor javascript_缓存_15

将代码添加到“hello.ts”。
注意 类方法从嵌入式“script.js”访问脚本警报

declare function ScriptAlert(message: string);

export class Hello{

    hello(): void {
        ScriptAlert("hello");
    }

    static goodbye(): void {
        ScriptAlert("goodbye");
    }

}


export var HelloInstance = new Hello();

安装 Microsoft.TypeScript.MSBuild

右键单击项目节点,然后选择“添加”>“新项”。 选择“TypeScript JSON 配置文件”,然后单击“添加”。

Visual Studio 会将 tsconfig.json 文件添加到项目根目录中。 可以使用此文件为 TypeScript 编译器配置选项

打开 tsconfig.json 并更新,以设置所需的编译器选项。

设置版本:ECMAScript, TSX:无, 模块:ES2015 在项目/属性/类型脚本构建

{
  "compilerOptions": {
    "noImplicitAny": false,
    "noEmitOnError": true,
    "removeComments": false,
    "sourceMap": true,
    "target": "es2015",
    "outDir": "wwwroot/js"
  },
  "include": [
    "scripts/**/*"
  ]
}

将所有“Index.razor”内容替换为以下代码片段,以添加模块按钮和方法。

@page "/"

@inject IJSRuntime JS
@implements IAsyncDisposable

<h1>Hello, Interop!</h1><br />
<h4 style="background-color:aliceblue; padding:20px">JavaScript Interop</h4>
@Message<hr />

<button class="btn btn-primary" @onclick="@Prompt">Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptPrompt">Script Prompt</button>
<button class="btn btn-primary" @onclick="@ScriptAlert">Script Alert</button>
<hr>

<button class="btn btn-primary" @onclick="@ModulelPrompt">Module Prompt</button>
<button class="btn btn-primary" @onclick="@ModulelAlert">Module Alert</button>
<hr>

<h4 style="background-color:aliceblue; padding:20px">TypeScript Interop</h4><hr>
<button class="btn btn-primary" @onclick="@HelloAlert">Hello Alert</button>

@code{
    private IJSObjectReference module;
    private IJSObjectReference hello;

    string Message { get; set; } = string.Empty;

    string Version{ get => $"?v={DateTime.Now.Ticks.ToString()}"; }


    async ValueTask IAsyncDisposable.DisposeAsync()
    {
        if (module is not null)
            await module.DisposeAsync();
    }

    protected override async Task OnAfterRenderAsync(bool firstRender)
    {
        if(firstRender){
            module = await JS.InvokeAsync<IJSObjectReference>("import", $"./js/script.module.js{Version}");
            hello = await JS.InvokeAsync<IJSObjectReference>("import", $"./js/hello.js{Version}");
        }
    }

    async void ModulelAlert()
    {
        await module.InvokeVoidAsync("ModulAlert", "Modulel Alert");
    }

    async void ModulelPrompt()
    {
        string answer = await module.InvokeAsync<string>("ModulePrompt", "Module Prompt say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing" : answer)}";
        StateHasChanged();
    }

    async void Prompt()
    {

        string answer = await JS.InvokeAsync<string>("prompt","say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing":answer)}";

        StateHasChanged();
    }


    async void ScriptPrompt()
    {
        string answer = await JS.InvokeAsync<string>("ScriptPrompt", "say what?");
        Message = $"Prompt:{(string.IsNullOrEmpty(answer) ? "nothing" : answer)}";

        StateHasChanged();
    }

    async void ScriptAlert()
    {
        await JS.InvokeVoidAsync("ScriptAlert", "Script Alert");
    }


    async void HelloAlert()
    {
        await hello.InvokeVoidAsync("HelloInstance.hello");
        await hello.InvokeVoidAsync("Hello.goodbye");
    }

}

添加了另一个模块“hello”来加载由“hello.ts”生成的JavaScript文件“hello.js”。HelloAlert 方法演示调用 TypeScript 类方法“goodbye”和对象实例方法“hello”。这些方法使用嵌入式脚本“script.js”中的脚本警报函数

构建、运行和测试 Hello 警报。

blazor如何使用JavaScript blazor javascript_缓存_16

2. 设置 Webpack 构建管道

右键单击“BlazerWithTSInterop”文件夹,然后选择弹出菜单项“在终端中打开”。

这将在编辑器中打开 PowerShell 终端窗口。

blazor如何使用JavaScript blazor javascript_互操作_17

执行以下命令以创建 package.json

npm init -y

blazor如何使用JavaScript blazor javascript_缓存_18

执行以下命令以安装 webpack 和 typescript 工具

npm i ts-loader typescript webpack webpack-cli

在 'package.json' 中添加脚本条目 "build": "webpack" 或者将 'package.json' 替换为下面的 json 内容。

{
  "name": "blazerwithtsinterop",
  "version": "1.0.0",
  "description": "",
  "main": "index.js",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1",
    "build": "webpack"
  },
  "keywords": [],
  "author": "",
  "license": "ISC",
  "dependencies": {
    "ts-loader": "^9.3.1",
    "typescript": "^4.8.2",
    "webpack": "^5.74.0",
    "webpack-cli": "^4.10.0"
  }
}

修改tsconfig.json

{
  "display": "Node 14",
  "compilerOptions": {
    "allowJs": true,
    "noImplicitAny": false,
    "noEmitOnError": true,
    "removeComments": false,
    "sourceMap": true,
    "lib": [ "es2020", "DOM" ],
    "target": "es6",
    "strict": true,
    "esModuleInterop": true,
    "skipLibCheck": true,
    "forceConsistentCasingInFileNames": true,
    "experimentalDecorators": true,
    "emitDecoratorMetadata": true,
    "moduleResolution": "node"
  },
  "include": [
    "scripts/**/*"
  ],
  "exclude": [
    "node_modules",
    "wwwroot"
  ]
}

这将禁用不再适用的项目 TypeScript 生成属性。Visual Studio现在将使用tsconfig.json进行TypeScript配置。

在“BlazerWithTSInterop”文件夹中创建webpack.config.json,其中包含以下内容。

const path = require("path");

module.exports = {
    mode: 'development',
    devtool: 'eval-source-map',
    module: {
        rules: [
            {
                test: /\.(ts)$/,
                exclude: /node_modules/,
                include: [path.resolve(__dirname, 'scripts')],
                use: 'ts-loader',
            }
        ]
    },
    resolve: {
        extensions: ['.ts', '.js'],
    },
    entry: {
        index: ['./scripts/index']
    },
    output: {
        path: path.resolve(__dirname, './wwwroot/public'),
        filename: '[name]-bundle.js',
        library: "[name]"
    }
};

此脚本告诉 webpack 使用 ts-loader 将 .ts 文件转译为.js。
对于每个条目 [name] 创建一个 JavaScript 库 [name]。
文件 [name]-bundle 在“wwwroot/public”文件夹中进行类型化。
此脚本有一个名为“index”的条目。
将输入文件 './scripts/index.ts' 转译为输出文件 './scripts/index.js'。
第二个传递将 “./scripts/index.js” 与依赖项代码捆绑在一起,并输出到文件 '../wwwroot/public/index-bundle.js'

<Project Sdk="Microsoft.NET.Sdk.BlazorWebAssembly">
...
  <Target Name="PreBuild" BeforeTargets="PreBuildEvent">
    <Exec Command="npm install" WorkingDirectory="wwwroot" />
    <Exec Command="npm run build" WorkingDirectory="wwwroot" />
  </Target>
...
</Project>

不再需要 Microsoft.TypeScript.MSBuild 过程,因为它绕过了 webpack typescript pre-build。

没有做任何伤害,让它留在这个演示中。或者,您可以选择并删除以将其删除。

blazor如何使用JavaScript blazor javascript_Interop_19

将代码添加到“index.ts”。索引类是 Hello 类包装器。索引模块还导出 Hello 类和 HelloInstance 对象。

import { Hello, HelloInstance } from "./hello";
export { Hello, HelloInstance } from "./hello";

export class Index {

    hello(): void {
        HelloInstance.hello();
    }

    static goodbye(): void{
        Hello.goodbye();
    }
}

export var IndexInstance = new Index()

生成 Ctrl+Shift+B 会在“公共”目录中创建“index.js”和“index_bundle.js”。

blazor如何使用JavaScript blazor javascript_缓存_20

在新的“index-bundle.js”中搜索“ScriptAlert”。 验证 'index.js' 包包含 'hello.js' 依赖代码。 捆绑包包括相关代码。 然而; 一个包将只有一个模块条目。 Interop 只能访问从模块中导出的项目。

在 'script.js' 之后的 'Index.html' 中添加 'index-bundle.js' 作为静态资源。

<body>
...
    <script src="_framework/blazor.webassembly.js"></script>
    <script src="js/script.js"></script>
    <script src="public/index-bundle.js"></script>
...
</body>

在 'Index.razor' html 部分中,将此行添加为最后一个按钮。

<button class="btn btn-primary" @onclick="@BundleIndexHello">Bundle Index Hello</button>

在“Index.razor”代码部分添加这个作为最后一个方法。

async void BundleIndexHello()
{
    await JS.InvokeVoidAsync("index.IndexInstance.hello");
    await JS.InvokeVoidAsync("index.Index.goodbye");
}

构建并运行。

blazor如何使用JavaScript blazor javascript_Interop_21

Bundle Index Hello 按钮演示调用从“index”库导出的索引类方法

在 'Index.razor' html 部分中,将此行添加为最后一个按钮。

<button class="btn btn-primary" @notallow="@ReExportHello">ReExport Hello</button>

在“Index.razor”代码部分添加这个作为最后一个方法。

async void ReExportHello()
{
    await JS.InvokeVoidAsync("index.HelloInstance.hello");
    await JS.InvokeVoidAsync("index.Hello.goodbye");
}

构建并运行。

blazor如何使用JavaScript blazor javascript_互操作_22

ReExport Hello 按钮演示调用从“index”库导出的 Hello 类方法

总结 本节介绍了从 Webpack 包调用 TypeScript。 包可以通过从包入口模块导出来公开依赖代码。 捆绑包是 Blazor 互操作可通过捆绑包库前缀访问的嵌入式资源。

调用 NPM TypeScript

右键单击项目控制台执行下面的命令,将threejs添加到 package.json

npm i three

创建新的“scripts/cube.ts”文件,将代码复制到“cube.js”。

import * as THREE from 'three';

export class Cube {

    camera: THREE.PerspectiveCamera;
    scene: THREE.Scene;
    renderer: THREE.WebGLRenderer;
    cube: any;

    constructor() {
        this.camera = new THREE.PerspectiveCamera(75, 2, .1, 5);
        this.camera.position.z = 2;
        let canvas = document.querySelector('#cube') as HTMLCanvasElement;
        this.renderer = new THREE.WebGLRenderer({ canvas: canvas, alpha: true, antialias: true });
        this.scene = new THREE.Scene();
        this.scene.background = null;
        const light = new THREE.DirectionalLight(0xFFFFFF, 1);
        light.position.set(-1, 2, 4);
        this.scene.add(light);

        const geometry = new THREE.BoxGeometry(1, 1, 1);
        const loadManager = new THREE.LoadingManager();
        const loader = new THREE.TextureLoader(loadManager);
        const texBlazor = loader.load('images/blazor.png');
        const texInterop = loader.load('images/interop.png');
        const texCircle = loader.load('images/tscircle.png');

        const matBlazor = new THREE.MeshPhongMaterial({ color: 0xffffff, map: texBlazor, transparent: false, opacity: 1 });
        const matInterop = new THREE.MeshPhongMaterial({ color: 0xffffff, map: texInterop, transparent: false, opacity: 1 });
        const matCircle = new THREE.MeshPhongMaterial({ color: 0xffffff, map: texCircle, transparent: false, opacity: 1 });
        const materials = [matBlazor, matInterop, matCircle, matBlazor, matInterop, matCircle];

        loadManager.onLoad = () => {
            this.cube = new THREE.Mesh(geometry, materials);
            this.scene.add(this.cube);
            this.animate();
        };
    }

    animate(time = 0) {
        time = performance.now() * 0.0005;
        this.cube.rotation.x = time;
        this.cube.rotation.y = time;
        this.renderer.render(this.scene, this.camera);
        requestAnimationFrame(this.animate.bind(this));
    }

    static Create(): void {
        new Cube();
    }

}

将多维数据集条目添加到 webpack.config.js用下面的代码段替换 entry: 部分。

entry: {
    index: ['./scripts/index'],
    cube:['./scripts/cube']
}

在“Index.html”中添加“cube-bundle.js”作为静态资产。

<script src="public/cube-bundle.js"></script>

在“Index.razor”html 部分的末尾添加立方体画布。

<canvas id="cube"/>

将“Index.razor”中的 OnAfterRenderAsync 替换为具有多维数据集互操作调用的以下方法。

protected override async Task OnAfterRenderAsync(bool firstRender)
{
    if(firstRender){
        module = await JS.InvokeAsync<IJSObjectReference>("import", $"./js/script.module.js{Version}");
        hello = await JS.InvokeAsync<IJSObjectReference>("import", $"./js/hello.js{Version}");
        await JS.InvokeVoidAsync("cube.Cube.Create");
    }
}

构建并运行。

blazor如何使用JavaScript blazor javascript_缓存_23

互操作软件设计

利用 TypeScript 有利于互操作软件设计。 典型的结构设计模式。
TypeScript 转换为浏览器 JavaScript 准备互操作。
Blazor C# 编译为准备执行的 Wasm 浏览器。

Blazor C# 界面设计与 TypeScript 对应物保持一致。

blazor如何使用JavaScript blazor javascript_缓存_24


Blazor Wasm 通过可互操作的 JavaScript 与浏览器对话。