client/systray: support several different color themes
Currently we only have a dark theme icon with white and grey dots over a black background. For some desktops, a logo with black and grey dots over a white background might be preferable. And for desktops where the bar is *almost* black or white, but not quite, an option to render the logo with dots only and no background can look really nice. Add a new -theme flag to the systray command with the default staying the same as it is today. Updates #18303 Change-Id: Ia101a4a3005adb9118051b3416f5a64a4a45987d Signed-off-by: Will Norris <will@tailscale.com>
This commit is contained in:
+42
-4
@@ -11,6 +11,7 @@ import (
|
|||||||
"image"
|
"image"
|
||||||
"image/color"
|
"image/color"
|
||||||
"image/png"
|
"image/png"
|
||||||
|
"log"
|
||||||
"runtime"
|
"runtime"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
@@ -204,12 +205,49 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
bg = color.NRGBA{0, 0, 0, 255}
|
black = color.NRGBA{0, 0, 0, 255}
|
||||||
fg = color.NRGBA{255, 255, 255, 255}
|
white = color.NRGBA{255, 255, 255, 255}
|
||||||
gray = color.NRGBA{255, 255, 255, 102}
|
darkGray = color.NRGBA{102, 102, 102, 255}
|
||||||
red = color.NRGBA{229, 111, 74, 255}
|
lightGray = color.NRGBA{153, 153, 153, 255}
|
||||||
|
red = color.NRGBA{229, 111, 74, 255}
|
||||||
|
transparent = color.NRGBA{}
|
||||||
|
|
||||||
|
// default values to dark theme
|
||||||
|
bg = black
|
||||||
|
fg = white
|
||||||
|
gray = darkGray
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// SetTheme sets the color theme of the systray icon.
|
||||||
|
//
|
||||||
|
// Supported themes are:
|
||||||
|
// - dark - white and gray dots over black background
|
||||||
|
// - dark:nobg - white and grey dots over transparent background
|
||||||
|
// - light - black and gray dots over white background
|
||||||
|
// - light:nobg - black and grey dots over transparent background
|
||||||
|
func SetTheme(theme string) {
|
||||||
|
switch theme {
|
||||||
|
case "dark":
|
||||||
|
bg = black
|
||||||
|
fg = white
|
||||||
|
gray = darkGray
|
||||||
|
case "dark:nobg":
|
||||||
|
bg = transparent
|
||||||
|
fg = white
|
||||||
|
gray = darkGray
|
||||||
|
case "light":
|
||||||
|
bg = white
|
||||||
|
fg = black
|
||||||
|
gray = lightGray
|
||||||
|
case "light:nobg":
|
||||||
|
bg = transparent
|
||||||
|
fg = black
|
||||||
|
gray = lightGray
|
||||||
|
default:
|
||||||
|
log.Printf("unknown theme: %q", theme)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// render returns a PNG image of the logo.
|
// render returns a PNG image of the logo.
|
||||||
func (logo tsLogo) render() *bytes.Buffer {
|
func (logo tsLogo) render() *bytes.Buffer {
|
||||||
const borderUnits = 1
|
const borderUnits = 1
|
||||||
|
|||||||
@@ -15,9 +15,11 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var socket = flag.String("socket", paths.DefaultTailscaledSocket(), "path to tailscaled socket")
|
var socket = flag.String("socket", paths.DefaultTailscaledSocket(), "path to tailscaled socket")
|
||||||
|
var theme = flag.String("theme", "dark", "color theme for Tailscale icon: dark, dark:nobg, light, light:nobg")
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
lc := &local.Client{Socket: *socket}
|
lc := &local.Client{Socket: *socket}
|
||||||
|
systray.SetTheme(*theme)
|
||||||
new(systray.Menu).Run(lc)
|
new(systray.Menu).Run(lc)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ func init() {
|
|||||||
maybeSystrayCmd = systrayConfigCmd
|
maybeSystrayCmd = systrayConfigCmd
|
||||||
}
|
}
|
||||||
|
|
||||||
var systrayArgs struct {
|
var configSystrayArgs struct {
|
||||||
initSystem string
|
initSystem string
|
||||||
installStartup bool
|
installStartup bool
|
||||||
}
|
}
|
||||||
@@ -32,7 +32,7 @@ func systrayConfigCmd() *ffcli.Command {
|
|||||||
Exec: configureSystray,
|
Exec: configureSystray,
|
||||||
FlagSet: (func() *flag.FlagSet {
|
FlagSet: (func() *flag.FlagSet {
|
||||||
fs := newFlagSet("systray")
|
fs := newFlagSet("systray")
|
||||||
fs.StringVar(&systrayArgs.initSystem, "enable-startup", "",
|
fs.StringVar(&configSystrayArgs.initSystem, "enable-startup", "",
|
||||||
"Install startup script for init system. Currently supported systems are [systemd, freedesktop].")
|
"Install startup script for init system. Currently supported systems are [systemd, freedesktop].")
|
||||||
return fs
|
return fs
|
||||||
})(),
|
})(),
|
||||||
@@ -40,8 +40,8 @@ func systrayConfigCmd() *ffcli.Command {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func configureSystray(_ context.Context, _ []string) error {
|
func configureSystray(_ context.Context, _ []string) error {
|
||||||
if systrayArgs.initSystem != "" {
|
if configSystrayArgs.initSystem != "" {
|
||||||
if err := systray.InstallStartupScript(systrayArgs.initSystem); err != nil {
|
if err := systray.InstallStartupScript(configSystrayArgs.initSystem); err != nil {
|
||||||
fmt.Printf("%s\n\n", err.Error())
|
fmt.Printf("%s\n\n", err.Error())
|
||||||
return flag.ErrHelp
|
return flag.ErrHelp
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ package cli
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"flag"
|
||||||
|
|
||||||
"github.com/peterbourgon/ff/v3/ffcli"
|
"github.com/peterbourgon/ff/v3/ffcli"
|
||||||
"tailscale.com/client/systray"
|
"tailscale.com/client/systray"
|
||||||
@@ -17,10 +18,20 @@ var systrayCmd = &ffcli.Command{
|
|||||||
ShortUsage: "tailscale systray",
|
ShortUsage: "tailscale systray",
|
||||||
ShortHelp: "Run a systray application to manage Tailscale",
|
ShortHelp: "Run a systray application to manage Tailscale",
|
||||||
LongHelp: "Run a systray application to manage Tailscale.",
|
LongHelp: "Run a systray application to manage Tailscale.",
|
||||||
Exec: runSystray,
|
FlagSet: (func() *flag.FlagSet {
|
||||||
|
fs := newFlagSet("systray")
|
||||||
|
fs.StringVar(&systrayArgs.theme, "theme", "dark", "color theme for Tailscale icon: dark, dark:nobg, light, light:nobg")
|
||||||
|
return fs
|
||||||
|
})(),
|
||||||
|
Exec: runSystray,
|
||||||
|
}
|
||||||
|
|
||||||
|
var systrayArgs struct {
|
||||||
|
theme string
|
||||||
}
|
}
|
||||||
|
|
||||||
func runSystray(ctx context.Context, _ []string) error {
|
func runSystray(ctx context.Context, _ []string) error {
|
||||||
|
systray.SetTheme(systrayArgs.theme)
|
||||||
new(systray.Menu).Run(&localClient)
|
new(systray.Menu).Run(&localClient)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user