c# - 如何在 .Net Blazor 中获取 Javascript 回调?

标签 c# interop blazor

有没有办法向 Javascript 添加回调并在 Blazor 中获取结果?除了 JS Promises。

例如,假设我要加载一个文件

Javascript 代码

window.readFile = function(filePath, callBack) {
    var reader = new FileReader();
    reader.onload = function (evt) {
        callBack(evt.target.result);
    };
    reader.readAsText(filePath);
}

我可以在 Blazor C# 中有这样的东西吗

    // read file content and output result to console
    void GetFileContent() {
        JsRuntime.InvokeAsync<object>("readFile", "file.txt", (string text) => {
            Console.Write(text);
        });
    }

或者可能是这样的

    // read with javascript
    void ReadFileContent() {
        JsRuntime.InvokeAsync<object>("readFile", "file.txt", "resultCallbackMethod");
    }

    // output result callback to console
    void resultCallbackMethod(string text) {
        Console.Write(text);
    }

谢谢

最佳答案

更新 1:

在重新阅读您的问题后,我认为这将涵盖您的第二个示例

我认为您可以选择实现处理调用的 JS 代理函数。像这样:

更新 2:

代码已更新为功能性(但未经过深入测试)版本,您还可以在 blazorfiddle.com 中找到工作示例

JavaScript 代码

// Target Javascript function
window.readFile = function (filePath, callBack) {

    var fileInput = document.getElementById('fileInput');
    var file = fileInput.files[0];

    var reader = new FileReader();

    reader.onload = function (evt) {
        callBack(evt.target.result);
    };

    reader.readAsText(file);

}

// Proxy function
// blazorInstance: A reference to the actual C# class instance, required to invoke C# methods inside it
// blazorCallbackName: parameter that will get the name of the C# method used as callback
window.readFileProxy = (instance, callbackMethod, fileName) => {

    // Execute function that will do the actual job
    window.readFile(fileName, result => {
        // Invoke the C# callback method passing the result as parameter
        instance.invokeMethodAsync(callbackMethod, result);
    });

}

C# 代码

@page "/"

@inject IJSRuntime jsRuntime

<div>
    Select a text file:
    <input type="file" id="fileInput" @onchange="@ReadFileContent" />
</div>
<pre>
    @fileContent
</pre>

Welcome to your new app.

@code{

    private string fileContent { get; set; }

    public static object CreateDotNetObjectRefSyncObj = new object();

    public async Task ReadFileContent(UIChangeEventArgs ea)
    {
        // Fire & Forget: ConfigureAwait(false) is telling "I'm not expecting this call to return a thing"
        await jsRuntime.InvokeAsync<object>("readFileProxy", CreateDotNetObjectRef(this), "ReadFileCallback", ea.Value.ToString()).ConfigureAwait(false);
    }


    [JSInvokable] // This is required in order to JS be able to execute it
    public void ReadFileCallback(string response)
    {
        fileContent = response?.ToString();
        StateHasChanged();
    }

    // Hack to fix https://github.com/aspnet/AspNetCore/issues/11159    
    protected DotNetObjectRef<T> CreateDotNetObjectRef<T>(T value) where T : class
    {
        lock (CreateDotNetObjectRefSyncObj)
        {
            JSRuntime.SetCurrentJSRuntime(jsRuntime);
            return DotNetObjectRef.Create(value);
        }
    }

}

关于c# - 如何在 .Net Blazor 中获取 Javascript 回调?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56627649/

相关文章:

c# - 算法问题

c# - 处理从 C# 服务器返回的 SAFEARRAY

c# - 从静态方法访问隐藏属性

c# - 使用编译绑定(bind) (x :bind), 为什么我必须调用 Bindings.Update()?

c# - 如何在C#中处理excel工作表的另存为对话框

c# - 如何在 C# 中记录低级操作系统文件事务?

c# - 为什么我得到 ImageSharp 不支持同步读取?

asp.net-core - 找不到类型或命名空间名称 'IWebHostEnvironment'(是否缺少 using 指令或程序集引用?)

Blazor wasm - 调试 - 无法查看任何变量的值

c# - 格式化 ESRI 地理编码器的 RestSharp 请求