mirror of
https://codeberg.org/scip/swaycycle.git
synced 2025-12-17 04:21:01 +01:00
Compare commits
3 Commits
dependabot
...
i3ipc
| Author | SHA1 | Date | |
|---|---|---|---|
| 83ab028c66 | |||
| ab8b3a7816 | |||
| 14cc48feb4 |
4
.github/workflows/release.yaml
vendored
4
.github/workflows/release.yaml
vendored
@@ -10,10 +10,10 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v5
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v6
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.23.5
|
go-version: 1.23.5
|
||||||
|
|
||||||
|
|||||||
@@ -24,13 +24,6 @@ Add such a line to your sway config file (e.g. in `$HOME/.config/sway/config`):
|
|||||||
bindsym $mod+Tab exec ~/bin/swaycycle
|
bindsym $mod+Tab exec ~/bin/swaycycle
|
||||||
```
|
```
|
||||||
|
|
||||||
You may also add a second key binding to do the reverse, which is
|
|
||||||
sometimes very useful:
|
|
||||||
|
|
||||||
```default
|
|
||||||
bindsym $mod+Shift+Tab exec ~/bin/swaycycle --prev
|
|
||||||
```
|
|
||||||
|
|
||||||
## Debugging
|
## Debugging
|
||||||
|
|
||||||
You may call `swaycycle` in a terminal window on a workspace with at
|
You may call `swaycycle` in a terminal window on a workspace with at
|
||||||
|
|||||||
14
go.mod
14
go.mod
@@ -3,14 +3,16 @@ module swaycycle
|
|||||||
go 1.23
|
go 1.23
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/alecthomas/repr v0.5.1 // indirect
|
github.com/lmittmann/tint v1.1.2
|
||||||
|
github.com/mattn/go-isatty v0.0.20
|
||||||
|
github.com/spf13/pflag v1.0.7
|
||||||
|
github.com/tlinden/swayipc v0.3.0
|
||||||
|
github.com/tlinden/yadu v0.1.3
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
github.com/fatih/color v1.16.0 // indirect
|
github.com/fatih/color v1.16.0 // indirect
|
||||||
github.com/lmittmann/tint v1.1.2 // indirect
|
|
||||||
github.com/mattn/go-colorable v0.1.13 // indirect
|
github.com/mattn/go-colorable v0.1.13 // indirect
|
||||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
|
||||||
github.com/spf13/pflag v1.0.7 // indirect
|
|
||||||
github.com/tlinden/i3ipc v0.0.0-20250815101608-4f7e27528be3 // indirect
|
|
||||||
github.com/tlinden/yadu v0.1.3 // indirect
|
|
||||||
golang.org/x/sys v0.14.0 // indirect
|
golang.org/x/sys v0.14.0 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
10
go.sum
10
go.sum
@@ -1,5 +1,3 @@
|
|||||||
github.com/alecthomas/repr v0.5.1 h1:E3G4t2QbHTSNpPKBgMTln5KLkZHLOcU7r37J4pXBuIg=
|
|
||||||
github.com/alecthomas/repr v0.5.1/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4=
|
|
||||||
github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM=
|
github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM=
|
||||||
github.com/fatih/color v1.16.0/go.mod h1:fL2Sau1YI5c0pdGEVCbKQbLXB6edEj1ZgiY4NijnWvE=
|
github.com/fatih/color v1.16.0/go.mod h1:fL2Sau1YI5c0pdGEVCbKQbLXB6edEj1ZgiY4NijnWvE=
|
||||||
github.com/lmittmann/tint v1.1.2 h1:2CQzrL6rslrsyjqLDwD11bZ5OpLBPU+g3G/r5LSfS8w=
|
github.com/lmittmann/tint v1.1.2 h1:2CQzrL6rslrsyjqLDwD11bZ5OpLBPU+g3G/r5LSfS8w=
|
||||||
@@ -11,15 +9,17 @@ github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWE
|
|||||||
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
||||||
github.com/spf13/pflag v1.0.7 h1:vN6T9TfwStFPFM5XzjsvmzZkLuaLX+HS+0SeFLRgU6M=
|
github.com/spf13/pflag v1.0.7 h1:vN6T9TfwStFPFM5XzjsvmzZkLuaLX+HS+0SeFLRgU6M=
|
||||||
github.com/spf13/pflag v1.0.7/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
github.com/spf13/pflag v1.0.7/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||||
github.com/tlinden/i3ipc v0.0.0-20250815101608-4f7e27528be3 h1:/kIZO4852sAVemXtqnsBid0r4Q1h87jDwHa8f7v1h5I=
|
github.com/tlinden/swayipc v0.0.0-20250816175030-177eecd4757f h1:SP/fEurr6crxQI+j85L61rMppsHmOlJwxVzXnCvYJ40=
|
||||||
github.com/tlinden/i3ipc v0.0.0-20250815101608-4f7e27528be3/go.mod h1:mc0toDHmgqgX6FpE69U5yMPnHuLTdekHRslSLDp8xSE=
|
github.com/tlinden/swayipc v0.0.0-20250816175030-177eecd4757f/go.mod h1:JwlMIC7eBwV8soCt2UDqlAyBudobLo07ZvepIA0irY8=
|
||||||
|
github.com/tlinden/swayipc v0.3.0 h1:hGNWeEZUZIHfeP+MxpAKsUzPf3YSJ0FYX2XEu/yqNXA=
|
||||||
|
github.com/tlinden/swayipc v0.3.0/go.mod h1:JwlMIC7eBwV8soCt2UDqlAyBudobLo07ZvepIA0irY8=
|
||||||
github.com/tlinden/yadu v0.1.3 h1:5cRCUmj+l5yvlM2irtpFBIJwVV2DPEgYSaWvF19FtcY=
|
github.com/tlinden/yadu v0.1.3 h1:5cRCUmj+l5yvlM2irtpFBIJwVV2DPEgYSaWvF19FtcY=
|
||||||
github.com/tlinden/yadu v0.1.3/go.mod h1:l3bRmHKL9zGAR6pnBHY2HRPxBecf7L74BoBgOOpTcUA=
|
github.com/tlinden/yadu v0.1.3/go.mod h1:l3bRmHKL9zGAR6pnBHY2HRPxBecf7L74BoBgOOpTcUA=
|
||||||
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.6.0 h1:MVltZSvRTcU2ljQOhs94SXPftV6DCNnZViHeQps87pQ=
|
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.14.0 h1:Vz7Qs629MkJkGyHxUlRHizWJRG2j8fbQKjELVSNhy7Q=
|
golang.org/x/sys v0.14.0 h1:Vz7Qs629MkJkGyHxUlRHizWJRG2j8fbQKjELVSNhy7Q=
|
||||||
golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
|||||||
80
main.go
80
main.go
@@ -24,11 +24,10 @@ import (
|
|||||||
"log/slog"
|
"log/slog"
|
||||||
"os"
|
"os"
|
||||||
"runtime/debug"
|
"runtime/debug"
|
||||||
"sort"
|
|
||||||
|
|
||||||
"github.com/lmittmann/tint"
|
"github.com/lmittmann/tint"
|
||||||
"github.com/mattn/go-isatty"
|
"github.com/mattn/go-isatty"
|
||||||
"github.com/tlinden/i3ipc"
|
"github.com/tlinden/swayipc"
|
||||||
"github.com/tlinden/yadu"
|
"github.com/tlinden/yadu"
|
||||||
|
|
||||||
flag "github.com/spf13/pflag"
|
flag "github.com/spf13/pflag"
|
||||||
@@ -43,23 +42,14 @@ const (
|
|||||||
|
|
||||||
LevelNotice = slog.Level(2)
|
LevelNotice = slog.Level(2)
|
||||||
|
|
||||||
VERSION = "v0.3.1"
|
VERSION = "v0.4.0"
|
||||||
|
|
||||||
IPC_HEADER_SIZE = 14
|
|
||||||
IPC_MAGIC = "i3-ipc"
|
|
||||||
|
|
||||||
// message types
|
|
||||||
IPC_GET_TREE = 4
|
|
||||||
IPC_RUN_COMMAND = 0
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
Visibles = []*i3ipc.Node{}
|
Visibles = []*swayipc.Node{}
|
||||||
CurrentWorkspace = ""
|
CurrentWorkspace = ""
|
||||||
Previous = false
|
|
||||||
Debug = false
|
Debug = false
|
||||||
Dumptree = false
|
Dumptree = false
|
||||||
Dumpvisibles = false
|
|
||||||
Version = false
|
Version = false
|
||||||
Verbose = false
|
Verbose = false
|
||||||
Notswitch = false
|
Notswitch = false
|
||||||
@@ -72,11 +62,9 @@ const Usage string = `This is swaycycle - cycle focus through all visible window
|
|||||||
Usage: swaycycle [-vdDn] [-l <log>]
|
Usage: swaycycle [-vdDn] [-l <log>]
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
-p, --prev cycle backward
|
|
||||||
-n, --no-switch do not switch windows
|
-n, --no-switch do not switch windows
|
||||||
-d, --debug enable debugging
|
-d, --debug enable debugging
|
||||||
-D, --dump dump the sway tree (needs -d as well)
|
-D, --dump dump the sway tree (needs -d as well)
|
||||||
--dump-visibles dump a list of visible windows on current workspace (needs -d)
|
|
||||||
-l, --logfile string write output to logfile
|
-l, --logfile string write output to logfile
|
||||||
-v, --version show program version
|
-v, --version show program version
|
||||||
|
|
||||||
@@ -85,10 +73,8 @@ Licensed under the terms of the GNU GPL version 3.
|
|||||||
`
|
`
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.BoolVarP(&Previous, "prev", "p", false, "cycle backward")
|
|
||||||
flag.BoolVarP(&Debug, "debug", "d", false, "enable debugging")
|
flag.BoolVarP(&Debug, "debug", "d", false, "enable debugging")
|
||||||
flag.BoolVarP(&Dumptree, "dump", "D", false, "dump the sway tree (needs -d as well)")
|
flag.BoolVarP(&Dumptree, "dump", "D", false, "dump the sway tree (needs -d as well)")
|
||||||
flag.BoolVarP(&Dumpvisibles, "dump-visibles", "", false, "dump a list of visible windows on current workspace (needs -d)")
|
|
||||||
flag.BoolVarP(&Notswitch, "no-switch", "n", false, "do not switch windows")
|
flag.BoolVarP(&Notswitch, "no-switch", "n", false, "do not switch windows")
|
||||||
flag.BoolVarP(&Version, "version", "v", false, "show program version")
|
flag.BoolVarP(&Version, "version", "v", false, "show program version")
|
||||||
flag.BoolVarP(&Showhelp, "help", "h", Showhelp, "show help")
|
flag.BoolVarP(&Showhelp, "help", "h", Showhelp, "show help")
|
||||||
@@ -119,7 +105,7 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// connect to sway unix socket
|
// connect to sway unix socket
|
||||||
ipc := i3ipc.NewI3ipc()
|
ipc := swayipc.NewSwayIPC()
|
||||||
|
|
||||||
err := ipc.Connect()
|
err := ipc.Connect()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -141,14 +127,8 @@ func main() {
|
|||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
}
|
}
|
||||||
|
|
||||||
id := 0
|
id := findNextWindow()
|
||||||
if Previous {
|
|
||||||
id = findPrevWindow()
|
|
||||||
slog.Debug("findPrevWindow", "nextid", id)
|
|
||||||
} else {
|
|
||||||
id = findNextWindow()
|
|
||||||
slog.Debug("findNextWindow", "nextid", id)
|
slog.Debug("findNextWindow", "nextid", id)
|
||||||
}
|
|
||||||
|
|
||||||
if id > 0 && !Notswitch {
|
if id > 0 && !Notswitch {
|
||||||
switchFocus(id, ipc)
|
switchFocus(id, ipc)
|
||||||
@@ -157,7 +137,7 @@ func main() {
|
|||||||
|
|
||||||
// get into the sway tree, determine current workspace and extract all
|
// get into the sway tree, determine current workspace and extract all
|
||||||
// its visible windows, store them in the global var Visibles
|
// its visible windows, store them in the global var Visibles
|
||||||
func processJSON(sway *i3ipc.Node) error {
|
func processJSON(sway *swayipc.Node) error {
|
||||||
if !istype(sway, root) && len(sway.Nodes) == 0 {
|
if !istype(sway, root) && len(sway.Nodes) == 0 {
|
||||||
return fmt.Errorf("Invalid or empty JSON structure")
|
return fmt.Errorf("Invalid or empty JSON structure")
|
||||||
}
|
}
|
||||||
@@ -175,9 +155,7 @@ func processJSON(sway *i3ipc.Node) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if Dumpvisibles {
|
slog.Debug("processed visible windows", "visibles", Visibles)
|
||||||
dumpVisibles()
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@@ -209,26 +187,8 @@ func findNextWindow() int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func findPrevWindow() int {
|
|
||||||
vislen := len(Visibles)
|
|
||||||
if vislen == 0 {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
prevnode := Visibles[vislen-1].Id
|
|
||||||
|
|
||||||
for _, node := range Visibles {
|
|
||||||
if node.Focused {
|
|
||||||
return prevnode
|
|
||||||
}
|
|
||||||
prevnode = node.Id
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// actually switch focus using a swaymsg command
|
// actually switch focus using a swaymsg command
|
||||||
func switchFocus(id int, ipc *i3ipc.I3ipc) error {
|
func switchFocus(id int, ipc *swayipc.SwayIPC) error {
|
||||||
responses, err := ipc.RunContainerCommand(id, "focus")
|
responses, err := ipc.RunContainerCommand(id, "focus")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("failed to send focus command to container %d: %w (%s)",
|
log.Fatalf("failed to send focus command to container %d: %w (%s)",
|
||||||
@@ -241,19 +201,13 @@ func switchFocus(id int, ipc *i3ipc.I3ipc) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// iterate recursively over given node list extracting visible windows
|
// iterate recursively over given node list extracting visible windows
|
||||||
func recurseNodes(nodes []*i3ipc.Node) {
|
func recurseNodes(nodes []*swayipc.Node) {
|
||||||
for _, node := range nodes {
|
for _, node := range nodes {
|
||||||
|
// we handle nodes and floating_nodes identical
|
||||||
|
node.Nodes = append(node.Nodes, node.FloatingNodes...)
|
||||||
|
|
||||||
if istype(node, workspace) {
|
if istype(node, workspace) {
|
||||||
if node.Name == CurrentWorkspace {
|
if node.Name == CurrentWorkspace {
|
||||||
//floating_nodes need to be sorted because
|
|
||||||
//order changes each time they are focused.
|
|
||||||
FloatVis := node.FloatingNodes
|
|
||||||
sort.Slice(FloatVis, func(i, j int) bool {
|
|
||||||
return FloatVis[i].Id < FloatVis[j].Id
|
|
||||||
})
|
|
||||||
//now we can handle nodes and floating_nodes identical
|
|
||||||
node.Nodes = append(node.Nodes, FloatVis...)
|
|
||||||
recurseNodes(node.Nodes)
|
recurseNodes(node.Nodes)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -273,16 +227,6 @@ func recurseNodes(nodes []*i3ipc.Node) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func dumpVisibles() {
|
|
||||||
windows := make([]string, len(Visibles))
|
|
||||||
|
|
||||||
for idx, node := range Visibles {
|
|
||||||
windows[idx] = fmt.Sprintf("id: %02d, focus: %5t, name: %s", node.Id, node.Focused, node.Name)
|
|
||||||
}
|
|
||||||
|
|
||||||
slog.Debug("visible windows on current workspace", "visibles", windows)
|
|
||||||
}
|
|
||||||
|
|
||||||
// we use line wise logging, unless debugging is enabled
|
// we use line wise logging, unless debugging is enabled
|
||||||
func setupLogging(output io.Writer) {
|
func setupLogging(output io.Writer) {
|
||||||
logLevel := &slog.LevelVar{}
|
logLevel := &slog.LevelVar{}
|
||||||
@@ -324,7 +268,7 @@ func setupLogging(output io.Writer) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// little helper to distinguish sway tree node types
|
// little helper to distinguish sway tree node types
|
||||||
func istype(nd *i3ipc.Node, which int) bool {
|
func istype(nd *swayipc.Node, which int) bool {
|
||||||
switch nd.Type {
|
switch nd.Type {
|
||||||
case "root":
|
case "root":
|
||||||
return which == root
|
return which == root
|
||||||
|
|||||||
Reference in New Issue
Block a user