微软在asp.net6中给blazor新增了一个IJSStreamReference的接口。

我们今天的所有内容,都要依赖这个接口,因为它可以把流直接传到c#中,这样我们就可以做很多的骚操作了。

今天我们来做一个简单的文件上传,这里以bootstrapblazor中新的CherryMarkdown组件为例。

首先,CherryMarkdown本身就支持文件上传处理,所以我们可以直接拿到js中的file,这里就不用考虑获取文件的方式了。

这里我们直接用window来保存这个file对象,这样操作应该是最简单的。

                    fileUpload(file, callback) {
                        window.files = {};
                        window.files[0] = file;
                        obj.invokeMethodAsync('Upload', {
                            fileName: file.name,
                            fileSize: file.size,
                            contentType: file.type,
                            lastModified: new Date(file.lastModified).toISOString(),
                        }).then(data => {
                            if (data !== "") {
                                callback(data);
                            }
                        })
                    },

这里我们定义了window.files[0]为我们要上传的文件内容。

然后再写一个方法来返回这个window.files[0]

export function bb_cherry_markdown_file(){
    return window.files[0];
}

这样,我们的js部分就搞定了,无需webapi,也无需其他的支持。

下面我们来看c#部分,也是相当简单。

首先我们写一个Upload方法来接收文件上传的请求。

    /// <summary>
    /// 文件上传回调
    /// </summary>
    /// <param name="uploadFile"></param>
    [JSInvokable]
    public async Task<string> Upload(CherryMarkdownUploadFile uploadFile)
    {
        var stream = await Module.InvokeAsync<IJSStreamReference>("bb_cherry_markdown_file");
        var data = await stream.OpenReadStreamAsync();
        uploadFile.UploadStream = data;
        if (OnFileUpload == null)
        {
            return "";
        }
        return await OnFileUpload.Invoke(uploadFile);
    }

这里的CherryMarkdownUploadFile如下:

/// <summary>
/// 文件信息
/// </summary>
public class CherryMarkdownUploadFile
{
    /// <summary>
    /// 文件名
    /// </summary>
    public string? FileName { get; set; }

    /// <summary>
    /// 文件大小
    /// </summary>
    public long FileSize { get; set; }

    /// <summary>
    /// 最后修改日期
    /// </summary>
    public string? LastModified { get; set; }

    /// <summary>
    /// 文件类型
    /// </summary>
    public string? ContentType { get; set; }

    /// <summary>
    /// 上传的文件流
    /// </summary>
    public Stream? UploadStream { get; set; }

    /// <summary>
    /// 返回码,0为成功,非0失败
    /// </summary>
    public int Code { get; set; }

    /// <summary>
    /// 错误信息
    /// </summary>
    public string? Error { get; set; }

    /// <summary>
    /// 保存到文件
    /// </summary>
    /// <param name="fileName"></param>
    /// <param name="token"></param>
    /// <returns></returns>
    public async Task<bool> SaveToFile(string fileName, CancellationToken token = default)
    {
        var ret = false;
        if (UploadStream != null)
        {
            // 文件保护,如果文件存在则先删除
            if (System.IO.File.Exists(fileName))
            {
                try
                {
                    System.IO.File.Delete(fileName);
                }
                catch (Exception ex)
                {
                    Code = 1002;
                    Error = ex.Message;
                }
            }

            var folder = Path.GetDirectoryName(fileName);
            if (!string.IsNullOrEmpty(folder) && !Directory.Exists(folder))
            {
                Directory.CreateDirectory(folder);
            }

            if (Code == 0)
            {
                using var uploadFile = File.OpenWrite(fileName);

                try
                {
                    // 打开文件流
                    var stream = UploadStream;

                    var buffer = new byte[4 * 1096];
                    int bytesRead = 0;

                    // 开始读取文件
                    while ((bytesRead = await stream.ReadAsync(buffer, token)) > 0)
                    {
                        await uploadFile.WriteAsync(buffer.AsMemory(0, bytesRead), token);

                    }
                    ret = true;
                }
                catch (Exception ex)
                {
                    Code = 1003;
                    Error = ex.Message;
                }
            }
        }
        return ret;
    }

}

可以用来接收js中的

obj.invokeMethodAsync('Upload', {
                            fileName: file.name,
                            fileSize: file.size,
                            contentType: file.type,
                            lastModified: new Date(file.lastModified).toISOString(),
                        }).then(data => {
                            if (data !== "") {
                                callback(data);
                            }
                        })

并且有一个SaveToFile方法可以将流保存为文件。

然后就是最关键的这行代码:

var stream = await Module.InvokeAsync<IJSStreamReference>("bb_cherry_markdown_file");

我们调用刚刚js中的bb_cherry_markdown_file方法来获取浏览器中的window.files[0],就可以返回一个stream,然后我们就可以结合CherryMarkdownUploadFile来将文件保存。

作者:|jvx|,原文链接: https://www.cnblogs.com/j4587698/p/16301706.html

文章推荐

Python趣味入门9:函数是你走过的套路,详解函数、调用、参...

PostgreSQL 的窗口函数 OVER, WINDOW, PARTITION BY, RANGE

【Azure 存储服务】Java Azure Storage SDK V12使用Endpoint...

一次线上MySQL分页事故,搞了半夜...

浅析kubernetes中client-go Informer

基于Python的渗透测试信息收集系统的设计和实现

基于SqlSugar的开发框架循序渐进介绍(6)-- 在基类接口中注...

为什么我写了路由懒加载但代码却没有分割?

普通人如何理解递归算法

一篇文章带你搞定JavaScript 性能调优

【组件封装】改造 Element-UI 多选框组件 (el-check

群晖下 gitea+drone+harbor实现CI/CD 发布到云服务器