tauri项目添加多文件下载功能,并支持下载进度回调显示在前端页面上

支持多文件同时下载,以及选中下载目录,并且支持下载进度回显在界面上,下载是在rust端实现的,通过emit事件来将下载进度发送给前端,前端通过listen监听下载进度的事件,然后通过传递下载进度的消息,区分每一个文件的下载进度,再显示到界面上

rust端多文件下载

通过定义一个download_file下载函数,然后传递进来下载url,保存的路径,以及文件id,保存路径可以为空字符串,这样的话,就会默认保存到电脑的下载文件夹中,file_id是文件标识,可以是id或者文件名等,要保证唯一,因为是通过这个id来回调下载进度事件通知的

rust 复制代码
#[derive(Clone, Serialize)]
#[serde(rename_all = "camelCase")]
struct DownloadProgress {
    file_id: String,
    downloaded: u64,
    total: u64,
}

#[tauri::command]
pub async fn download_file(
    app: AppHandle,
    url: String,
    save_path: String,
    file_id: String,
) -> Result<(), String> {
    let client = Client::new();
    let resp = client.get(&url).send().await.map_err(|e| e.to_string())?;
    // if save path is empty
    let mut save_path = save_path;
    let file_name = url.split('/').last().unwrap();
    if save_path.is_empty() {
        let file_path = app
            .path()
            .resolve(file_name, BaseDirectory::Download)
            .expect("failed to resolve resource");
        save_path = file_path.to_str().unwrap().to_string();
    }
    let total_size = resp.content_length();
    let mut stream = resp.bytes_stream();
    let mut file = File::create(&save_path).map_err(|e| e.to_string())?;
    let mut downloaded: u64 = 0;

    while let Some(chunk) = stream.next().await {
        let chunk = chunk.map_err(|e| e.to_string())?;
        file.write_all(&chunk).map_err(|e| e.to_string())?;
        downloaded += chunk.len() as u64;
        app.emit(
            "download_progress",
            DownloadProgress {
                file_id: file_id.clone(),
                downloaded,
                total: total_size.unwrap_or(0),
            },
        )
        .unwrap();
    }
    Ok(())
}

前端调用下载

通过invoke来触发一个下载事件

rust 复制代码
    await invoke('download_file', {
        url,
        savePath,
        fileId,
    })

然后通过监听download_progress事件,拿到下载进度的消息:

javascript 复制代码
const downloadProgress = ref(0)

listen('download_progress', (event: any) => {
    console.log(`downloading fileId--- ${event.payload.fileId}`)
    console.log(`downloading downloaded--- ${event.payload.downloaded}`)
    console.log(`downloading total--- ${event.payload.total}`)
    downloadProgress.value = Number(
        ((event.payload.downloaded / event.payload.total) * 100).toFixed(2)
    )
})

完整项目代码可以在我的开源项目中查看:

github.com/Sjj1024/Pak...

相关推荐
五月君_17 分钟前
炸裂!Claude Opus 4.6 与 GPT-5.3 同日发布:前端人的“自动驾驶“时刻到了?
前端·gpt
Mr Xu_21 分钟前
前端开发中CSS代码的优化与复用:从公共样式提取到CSS变量的最佳实践
前端·css
低代码布道师43 分钟前
Next.js 16 全栈实战(一):从零打造“教培管家”系统——环境与脚手架搭建
开发语言·javascript·ecmascript
鹏北海-RemHusband1 小时前
从零到一:基于 micro-app 的企业级微前端模板完整实现指南
前端·微服务·架构
LYFlied1 小时前
AI大时代下前端跨端解决方案的现状与演进路径
前端·人工智能
光影少年1 小时前
AI 前端 / 高级前端
前端·人工智能·状态模式
一位搞嵌入式的 genius1 小时前
深入 JavaScript 函数式编程:从基础到实战(含面试题解析)
前端·javascript·函数式
anOnion1 小时前
构建无障碍组件之Alert Dialog Pattern
前端·html·交互设计
choke2331 小时前
[特殊字符] Python 文件与路径操作
java·前端·javascript
云飞云共享云桌面1 小时前
高性能图形工作站的资源如何共享给10个SolidWorks研发设计用
linux·运维·服务器·前端·网络·数据库·人工智能