• 教程六 在Go中使用Energy创建跨平台GUI - 应用下载事件


     教程-示例-文档 


    介绍

    Energy应用下载文件时触发的下载事件和使用

    我们在页面上下载文件时,可以对文件下载时的处理,例如:保存路径,下载取消,开始、暂停。

    下面将用代码和注释,和简要的说明来演示


    Go代码示例

    想要运行此示例,需要安装好Go和Energy开发环境:教程一 环境安装

    此示例中采用了内置http服务访问内嵌资源: 内置http服务

    • 主窗口初始化回调函数里设置事件
    cef.BrowserWindow.SetBrowserInit(func(event *cef.BrowserEvent, browserWindow *cef.TCefWindowInfo)
    • 下载事件注册

    下载之前事件

    • 在页面中下载时触发该函数OnBeforeDownload,例如点击某下载链接
    • OnBeforeDownload事件,可以设置保存路径,显示保存提示框和不显示保存提示框
    • Linux下由于上游原因,需要自定义保存提示框,否则无法显示。
    • Windows和MacOSX保存提示框都正常
    • 值得注意的是 cef.QueueSyncCall 函数,其作用是将UI相关的操作移至到UI线程中执行,该函数是异步的。详情参考源码注释
    • 这里针对linux系统自定义了保存提示框 
      lcl.NewSaveDialog
    event.SetOnBeforeDownload(func(sender lcl.IObject, browser *cef.ICefBrowser, beforeDownloadItem *cef.DownloadItem, suggestedName string, callback *cef.ICefBeforeDownloadCallback) 

    下载更新事件

    • 下载更新事件OnDownloadUpdated,该函数实时更新下载项的进度
    •  在该函数中可以对每个下载项操作,下载取消,开始、暂停, 恢复下载。
      • 下载项操作需要在函数执行周期内操作,不可在函数执行周期之外。
    • 示例中下载进度使用了事件机制,把下载信息发送给web端展示: Go调用JS事件
    event.SetOnDownloadUpdated(func(sender lcl.IObject, browser *cef.ICefBrowser, downloadItem *cef.DownloadItem, callback *cef.ICefDownloadItemCallback)
    1. package main
    2. import (
    3. "embed"
    4. "fmt"
    5. "github.com/energye/energy/cef"
    6. "github.com/energye/energy/common"
    7. "github.com/energye/energy/common/assetserve"
    8. "github.com/energye/energy/consts"
    9. "github.com/energye/energy/ipc"
    10. "github.com/energye/golcl/lcl"
    11. )
    12. //资源目录,内置到执行程序中
    13. //go:embed resources
    14. var resources embed.FS
    15. //这个示例使用了几个事件来演示下载文件
    16. //在cef.BrowserWindow.SetBrowserInit初始化函数中设置event.SetOnBeforeDownload,用于设置保存目录
    17. //并且设置event.SetOnDownloadUpdated获取下载进度信息
    18. func main() {
    19. //全局初始化 每个应用都必须调用的
    20. cef.GlobalCEFInit(nil, &resources)
    21. //创建应用
    22. cefApp := cef.NewApplication(nil)
    23. //主窗口的配置
    24. //指定一个URL地址,或本地html文件目录
    25. cef.BrowserWindow.Config.DefaultUrl = "http://localhost:22022/download.html"
    26. //在主窗口初始化回调函数里设置浏览器事件
    27. cef.BrowserWindow.SetBrowserInit(func(event *cef.BrowserEvent, browserWindow *cef.TCefWindowInfo) {
    28. //linux 下载文件 系统弹出保存对话框不启作用
    29. //所以 自己调用系统的保存对话框获得保存路径
    30. linuxDlSave := lcl.NewSaveDialog(browserWindow.Window)
    31. linuxDlSave.SetTitle("保存对话框标题")
    32. //下载之前事件
    33. event.SetOnBeforeDownload(func(sender lcl.IObject, browser *cef.ICefBrowser, beforeDownloadItem *cef.DownloadItem, suggestedName string, callback *cef.ICefBeforeDownloadCallback) {
    34. fmt.Println("下载之前事件")
    35. //设置下载目录, 和弹出保存窗口
    36. if common.IsLinux() {
    37. //linux 在大多数据情况操作UI相关的需要使用 QueueSyncCall 函数包起来
    38. cef.QueueSyncCall(func(id int) {
    39. linuxDlSave.SetFileName(suggestedName)
    40. if linuxDlSave.Execute() {
    41. // showDialog = false 不显示保存对话框
    42. callback.Cont(linuxDlSave.FileName(), false)
    43. }
    44. })
    45. } else {
    46. //windows macosx
    47. callback.Cont(consts.ExePath+consts.Separator+suggestedName, true)
    48. }
    49. })
    50. //下载更新事件
    51. //1. 返回下载进度
    52. //2. downloadItem 下载项
    53. //3. callback 下载状态的控制, 下载暂停,开始、取消
    54. //4. 将下载进度通过事件机制发送到html中展示
    55. event.SetOnDownloadUpdated(func(sender lcl.IObject, browser *cef.ICefBrowser, downloadItem *cef.DownloadItem, callback *cef.ICefDownloadItemCallback) {
    56. //传递数据参数到html中
    57. //这些参数按下标顺序对应到js函数参数位置
    58. //演示只传递了几个参数
    59. var argumentList = ipc.NewArgumentList()
    60. //1个参数下载的ID
    61. argumentList.SetInt32(0, downloadItem.Id)
    62. //2个参数文件名
    63. argumentList.SetString(1, downloadItem.FullPath, true)
    64. //3个参数 接收的字节数 - 这里需要注意的是,IPC消息 数字类型只支持32位的,如果大于32位,需要转换成string类型发送, 否则直接使用int64类型会导致消息接收不到
    65. argumentList.SetInt32(2, int32(downloadItem.ReceivedBytes))
    66. //4个参数 文件总大小字节数 - 例如: int64转成string做为参数
    67. argumentList.SetString(3, fmt.Sprintf("%d", downloadItem.TotalBytes), true)
    68. browserWindow.Chromium().Emit("downloadUpdateDemo", argumentList, browser)
    69. })
    70. })
    71. //在主进程启动成功之后执行
    72. //在这里启动内置http服务
    73. //内置http服务需要使用 go:embed resources 内置资源到执行程序中
    74. cef.SetBrowserProcessStartAfterCallback(func(b bool) {
    75. fmt.Println("主进程启动 创建一个内置http服务")
    76. //通过内置http服务加载资源
    77. server := assetserve.NewAssetsHttpServer()
    78. server.PORT = 22022 //服务端口号
    79. server.AssetsFSName = "resources" //必须设置目录名和资源文件夹同名
    80. server.Assets = &resources
    81. go server.StartHttpServer()
    82. })
    83. //运行应用
    84. cef.Run(cefApp)
    85. }

    html代码示例

    html中写了几个下载链接

    html中定义了事件监听,ipc.on,事件名downloadUpdateDemo

    downloadUpdateDemo事件接收Go触发的事件,Go发送的消息由此事件函数接收

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <title>下载文件示例title>
    6. <script>
    7. console.log("下载文件示例")
    8. //使用ipc.on监听事件
    9. ipc.on("downloadUpdateDemo", function (id, suggestedFileName, receivedBytes, totalBytes) {
    10. console.log(id, suggestedFileName, receivedBytes, totalBytes)
    11. var downloadMessage = document.getElementById("downloadMessage")
    12. downloadMessage.innerHTML = downloadMessage.innerHTML + ("id: " + id + " suggestedFileName: " + suggestedFileName + " receivedBytes: " + receivedBytes + " / totalBytes: " + totalBytes) + "
      "
    13. });
    14. script>
    15. head>
    16. <body style="overflow: hidden;margin: 10px;padding: 10px;">
    17. <h3>download demo:h3><br>
    18. <a href="https://gitee.com/energye/energy/releases/download/v1.107.1.11/Windows%2032%20bits.zip">Windows 32 bitsa>
    19. <br>
    20. <a href="https://gitee.com/energye/energy/releases/download/v1.107.1.11/Windows%2064%20bits.zip">Windows 64 bitsa>
    21. <br>
    22. <a href="https://gitee.com/energye/energy/releases/download/v1.107.1.11/Linux%20x86%2064%20bits.zip">Linux x86 64
    23. bitsa> <br>
    24. <a href="https://gitee.com/energye/energy/releases/download/v1.107.1.11/MacOSX%20x86%2064%20bits.zip">MacOSX x86 64
    25. bitsa> <br>
    26. <div id="downloadMessage">
    27. div>
    28. body>
    29. html>

    运行效果图 - download

  • 相关阅读:
    数据库顶会 VLDB 2023 论文解读 - Krypton: 字节跳动实时服务分析 SQL 引擎设计
    #gStore-weekly | gStore最新版本0.9.1之BIND函数的使用
    京东商品详情API接口(PC端和APP端),京东详情页,商品属性接口,商品信息查询
    LeetCode220729_59、打家劫舍
    工程水文学试卷
    yarn 包管理器设置淘宝镜像和sass镜像
    C++------继承
    术语与定义
    【node和node-sass版本不一致导致报错】解决sass-loader和node-sass版本冲突问题,本人尝试过,真的解决错误了
    Coding:小写一个debugfs
  • 原文地址:https://blog.csdn.net/snxamdf/article/details/128068098