百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

golang 托盘菜单应用及打开系统默认浏览器

bigegpt 2025-07-10 13:19 10 浏览

之前看到一个应用,用go语言编写,说是某某程序的windows图形化客户端,体验一下发现只是一个托盘,然后托盘菜单的控制面板功能直接打开本地浏览器访问程序启动的web server网页完成gui相关功能。顿时感觉,嗯,是个曲线绕开类似electron等框架的方法。

这种方式的好处是,可以把擅长写web服务的应用桌面化,当需要gui的时候,直接托盘菜单启动浏览器,完成相关功能后,直接关闭浏览器省内存。

我的gost-ui-3程序用electron编写,内部集成浏览器,安装包60-~85M,启动后内存占用超过100M,所以后面考虑节省资源的方式,可以使用托盘图标+默认浏览器的方式解决。

当然如果对 cgo 敏感的话,就不能用了,三个平台都依赖 cgo

1. golang 托盘图标的使用

因为最近常用的是windows,家里的mac已经吃灰很久了。所以暂时默认适配的是windows环境。理论上mac和linux上也是可以适配的。

最核心的是应用了
github.com/getlantern/systray 这个库。

一个最简单的示例如下:

注意提前准备好相应的图标文件

package main

import (
	_ "embed"
	"fmt"

	"github.com/getlantern/systray"
)

// embed 指令直接读取icon文件并在编译时嵌入程序中
//
//go:embed icon.ico
var iconWin []byte

// 托盘菜单描述,自定义,为了方便定义和事件管理
type Menu struct {
	Title   string
	Tips    string
	Icon    []byte
	OnClick func(m *systray.MenuItem)
}

// 添加菜单
func AddMenu(menu *Menu) *systray.MenuItem {
	m := systray.AddMenuItem(menu.Title, menu.Tips)
	if len(menu.Icon) > 0 {
		m.SetIcon(menu.Icon)
	}
	go func() {
		for range m.ClickedCh {
			menu.OnClick(m)
		}
	}()

	return m
}

// 添加checkbox菜单
func AddCheckboxMenu(menu *Menu, checked bool) *systray.MenuItem {
	m := systray.AddMenuItemCheckbox(menu.Title, menu.Tips, checked)
	if len(menu.Icon) > 0 {
		m.SetIcon(menu.Icon)
	}
	go func() {
		for range m.ClickedCh {
			menu.OnClick(m)
		}
	}()

	return m
}

func main() {
	systray.Run(onReady, onExit)
}

func onReady() {
	systray.SetIcon(iconWin)
	systray.SetTitle("托盘图标示例")
	systray.SetTooltip("托盘图标示例提示")

	// 选择框和动态菜单综合示例
    AddCheckboxMenu(&Menu{
		Title: "启动",
		OnClick: func(m *systray.MenuItem) {
			if m.Checked() {
				m.SetTitle("启动")
				m.Uncheck()
			} else {
				m.SetTitle("停止")
				m.Check()
			}
		},
	}, false)

	// 添加退出菜单
	AddMenu(&Menu{
		Title: "退出",
		Tips:  "退出程序",
		OnClick: func(m *systray.MenuItem) {
			systray.Quit()
		},
	})
}

func onExit() {
	fmt.Printf("退出喽")
}

同时看到一些可能在应用中需要用到的api方法如下

// 设置主托盘图标, 比如一个服务分别在启动状态和关闭状态使用不同的图标
systray.SetTemplateIcon(_icon, _icon)

// 菜单可以显示和隐藏
systray.MenuItem.Show(); systray.MenuItem.Hide()
// 菜单可禁止和启用
systray.MenuItem.Disable(); systray.MenuItem.Enable()
// 添加一组菜单的方式
func AddMenuGroup(title string, sub []*Menu) {
	boot := systray.AddMenuItem(title, "")
	for _, v := range sub {
		mi := boot.AddSubMenuItem(v.Title, v.Title)
		_v := v
		go func() {
			for {
				select {
				case <-mi.ClickedCh:
					_v.OnClick(mi)
				}
			}
		}()
	}
}

2. 用 golang 打开默认浏览器

打开默认浏览器其实就是执行对应平台的系统命令。

结合我的托盘示例,一个完整的例子程序如下。主要看Open(uri string)方法。

//go:generate goversioninfo
package main

import (
    _ "embed"
    "fmt"
    "os/exec"
    "runtime"

    "github.com/getlantern/systray"
)

//go:embed icon/icon.png
var icon []byte

//go:embed icon/icon_off.png
var iconOff []byte

//go:embed icon/icon.ico
var iconWin []byte

//go:embed icon/icon_off.ico
var iconOffWin []byte

//go:embed icon/logo.png
var logo []byte

// 不同平台打开浏览器对应的命令
var commands = map[string]string{
    "windows": "cmd",
    "darwin":  "open",
    "linux":   "xdg-open",
}

// 托盘菜单自定义数据结构
type Menu struct {
    Title   string
    Tips    string
    Icon    []byte
    OnClick func(m *systray.MenuItem)
}

func main() {
    systray.Run(onReady, onExit)
}

// 执行打开默认浏览器并访问指定uri的命令
func Open(uri string) error {
    run, ok := commands[runtime.GOOS]
    if !ok {
        return fmt.Errorf("don't know how to open things on %s platform", runtime.GOOS)
    }

    var cmd *exec.Cmd
    if runtime.GOOS == "windows" {
        cmd = exec.Command(run, `/c`, `start`, uri)
		// 无console调用
        cmd.SysProcAttr = &syscall.SysProcAttr{HideWindow: true}
    } else {
        // linux和mac下暂未测试
        cmd = exec.Command(run, uri)
    }

    return cmd.Start()
}

// 添加一个常规菜单
func AddMenu(menu *Menu) *systray.MenuItem {
    m := systray.AddMenuItem(menu.Title, menu.Tips)
    if len(menu.Icon) > 0 {
        m.SetIcon(menu.Icon)
    }
    go func() {
        for range m.ClickedCh {
            menu.OnClick(m)
        }
    }()

    return m
}

// 托盘启动时
func onReady() {
    systray.SetIcon(iconWin)
    systray.SetTitle("托盘图标示例")
    systray.SetTooltip("托盘图标示例提示")

    // 打开一个浏览器网址
    AddMenu(&Menu{
        Title: "我的博客",
        Tips: "blog.wavesxa.com",
        OnClick: func(m *systray.MenuItem) {
            Open("https://blog.wavesxa.com")
        },
    })

    // 退出菜单
    AddMenu(&Menu{
        Title: "退出",
        Tips: "退出程序",
        OnClick: func(m *systray.MenuItem) {
            systray.Quit()
        },
    })
}

// 托盘退出时
func onExit() {
    fmt.Printf("退出喽")
}

相关推荐

Go语言泛型-泛型约束与实践(go1.7泛型)

来源:械说在Go语言中,Go泛型-泛型约束与实践部分主要探讨如何定义和使用泛型约束(Constraints),以及如何在实际开发中利用泛型进行更灵活的编程。以下是详细内容:一、什么是泛型约束?**泛型...

golang总结(golang实战教程)

基础部分Go语言有哪些优势?1简单易学:语法简洁,减少了代码的冗余。高效并发:内置强大的goroutine和channel,使并发编程更加高效且易于管理。内存管理:拥有自动垃圾回收机制,减少内...

Go 官宣:新版 Protobuf API(go pro版本)

原文作者:JoeTsai,DamienNeil和HerbieOng原文链接:https://blog.golang.org/a-new-go-api-for-protocol-buffer...

Golang开发的一些注意事项(一)(golang入门项目)

1.channel关闭后读的问题当channel关闭之后再去读取它,虽然不会引发panic,但会直接得到零值,而且ok的值为false。packagemainimport"...

golang 托盘菜单应用及打开系统默认浏览器

之前看到一个应用,用go语言编写,说是某某程序的windows图形化客户端,体验一下发现只是一个托盘,然后托盘菜单的控制面板功能直接打开本地浏览器访问程序启动的webserver网页完成gui相关功...

golang标准库每日一库之 io/ioutil

一、核心函数概览函数作用描述替代方案(Go1.16+)ioutil.ReadFile(filename)一次性读取整个文件内容(返回[]byte)os.ReadFileioutil.WriteFi...

文件类型更改器——GoLang 中的 CLI 工具

我是如何为一项琐碎的工作任务创建一个简单的工具的,你也可以上周我开始玩GoLang,它是一种由Google制作的类C编译语言,非常轻量和快速,事实上它经常在Techempower的基准测...

Go (Golang) 中的 Channels 简介(golang channel长度和容量)

这篇文章重点介绍Channels(通道)在Go中的工作方式,以及如何在代码中使用它们。在Go中,Channels是一种编程结构,它允许我们在代码的不同部分之间移动数据,通常来自不同的goro...

Golang引入泛型:Go将Interface「」替换为“Any”

现在Go将拥有泛型:Go将Interface{}替换为“Any”,这是一个类型别名:typeany=interface{}这会引入了泛型作好准备,实际上,带有泛型的Go1.18Beta...

一文带你看懂Golang最新特性(golang2.0特性)

作者:腾讯PCG代码委员会经过十余年的迭代,Go语言逐渐成为云计算时代主流的编程语言。下到云计算基础设施,上到微服务,越来越多的流行产品使用Go语言编写。可见其影响力已经非常强大。一、Go语言发展历史...

Go 每日一库之 java 转 go 遇到 Apollo?让 agollo 来平滑迁移

以下文章来源于GoOfficialBlog,作者GoOfficialBlogIntroductionagollo是Apollo的Golang客户端Apollo(阿波罗)是携程框架部门研...

Golang使用grpc详解(golang gcc)

gRPC是Google开源的一种高性能、跨语言的远程过程调用(RPC)框架,它使用ProtocolBuffers作为序列化工具,支持多种编程语言,如C++,Java,Python,Go等。gR...

Etcd服务注册与发现封装实现--golang

服务注册register.gopackageregisterimport("fmt""time"etcd3"github.com/cor...

Golang:将日志以Json格式输出到Kafka

在上一篇文章中我实现了一个支持Debug、Info、Error等多个级别的日志库,并将日志写到了磁盘文件中,代码比较简单,适合练手。有兴趣的可以通过这个链接前往:https://github.com/...

如何从 PHP 过渡到 Golang?(php转golang)

我是PHP开发者,转Go两个月了吧,记录一下使用Golang怎么一步步开发新项目。本着有坑填坑,有错改错的宗旨,从零开始,开始学习。因为我司没有专门的Golang大牛,所以我也只能一步步自己去...