Compare commits
34 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0e5d38907f | ||
|
|
7ba7abd817 | ||
|
|
cedcf8619f | ||
|
|
8ff2d5c89c | ||
|
|
c4629e0c28 | ||
|
|
ab5e2ea6ae | ||
|
|
63117e2605 | ||
|
|
d22609f403 | ||
|
|
cb6a7f44fe | ||
|
|
85cc2d78ee | ||
|
|
d3a0fd04a4 | ||
|
|
52e63667c9 | ||
|
|
f399f589fb | ||
|
|
2058f124fd | ||
|
|
ebdbe71ac6 | ||
|
|
3fe6d71234 | ||
|
|
0e5b1cf659 | ||
|
|
6494557979 | ||
|
|
f68678f687 | ||
|
|
e31df6e721 | ||
|
|
9509a65625 | ||
|
|
bc84a05378 | ||
|
|
a249c37612 | ||
|
|
c7fa6e836a | ||
|
|
87a62e7bb6 | ||
|
|
11241cef0b | ||
|
|
87d2a74db0 | ||
|
|
41f7d58edf | ||
|
|
583ddfde0b | ||
|
|
383869906a | ||
|
|
54553ffae1 | ||
|
|
d8e7e0f6c2 | ||
|
|
7230d82be3 | ||
|
|
02b3930ef9 |
1
.github/FUNDING.yml
vendored
Normal file
1
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
github: nwg-piotr
|
||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -15,5 +15,7 @@
|
|||||||
bin
|
bin
|
||||||
nwg-drawer
|
nwg-drawer
|
||||||
|
|
||||||
|
/.idea
|
||||||
|
|
||||||
# Dependency directories (remove the comment below to include it)
|
# Dependency directories (remove the comment below to include it)
|
||||||
# vendor/
|
# vendor/
|
||||||
|
|||||||
14
go.mod
14
go.mod
@@ -1,12 +1,20 @@
|
|||||||
module github.com/nwg-piotr/nwg-drawer
|
module github.com/nwg-piotr/nwg-drawer
|
||||||
|
|
||||||
go 1.16
|
go 1.19
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37
|
github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37
|
||||||
github.com/dlasky/gotk3-layershell v0.0.0-20210827021656-e6ecab2731f7
|
github.com/dlasky/gotk3-layershell v0.0.0-20210827021656-e6ecab2731f7
|
||||||
github.com/fsnotify/fsnotify v1.5.1
|
github.com/fsnotify/fsnotify v1.5.1
|
||||||
github.com/gotk3/gotk3 v0.6.1
|
github.com/gotk3/gotk3 v0.6.1
|
||||||
github.com/joshuarubin/go-sway v0.0.4
|
github.com/joshuarubin/go-sway v1.2.0
|
||||||
github.com/sirupsen/logrus v1.8.1
|
github.com/sirupsen/logrus v1.9.0
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/joshuarubin/lifecycle v1.0.0 // indirect
|
||||||
|
go.uber.org/atomic v1.3.2 // indirect
|
||||||
|
go.uber.org/multierr v1.1.0 // indirect
|
||||||
|
golang.org/x/sync v0.0.0-20190412183630-56d357773e84 // indirect
|
||||||
|
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
19
go.sum
19
go.sum
@@ -9,24 +9,27 @@ github.com/fsnotify/fsnotify v1.5.1 h1:mZcQUHVQUQWoPXXtuf9yuEXKudkV2sx1E06UadKWp
|
|||||||
github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU=
|
github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU=
|
||||||
github.com/gotk3/gotk3 v0.6.1 h1:GJ400a0ecEEWrzjBvzBzH+pB/esEMIGdB9zPSmBdoeo=
|
github.com/gotk3/gotk3 v0.6.1 h1:GJ400a0ecEEWrzjBvzBzH+pB/esEMIGdB9zPSmBdoeo=
|
||||||
github.com/gotk3/gotk3 v0.6.1/go.mod h1:/hqFpkNa9T3JgNAE2fLvCdov7c5bw//FHNZrZ3Uv9/Q=
|
github.com/gotk3/gotk3 v0.6.1/go.mod h1:/hqFpkNa9T3JgNAE2fLvCdov7c5bw//FHNZrZ3Uv9/Q=
|
||||||
github.com/joshuarubin/go-sway v0.0.4 h1:dpmIwQ/LytG+oMrjmaVKdk1aPdW2feXK/+wAcLKIx4A=
|
github.com/joshuarubin/go-sway v1.2.0 h1:t3eqW504//uj9PDwFf0+IVfkD+WoOGaDX5gYIe0BHyM=
|
||||||
github.com/joshuarubin/go-sway v0.0.4/go.mod h1:qcDd6f25vJ0++wICwA1BainIcRC67p2Mb4lsrZ0k3/k=
|
github.com/joshuarubin/go-sway v1.2.0/go.mod h1:qcDd6f25vJ0++wICwA1BainIcRC67p2Mb4lsrZ0k3/k=
|
||||||
github.com/joshuarubin/lifecycle v1.0.0 h1:N/lPEC8f+dBZ1Tn99vShqp36LwB+LI7XNAiNadZeLUQ=
|
github.com/joshuarubin/lifecycle v1.0.0 h1:N/lPEC8f+dBZ1Tn99vShqp36LwB+LI7XNAiNadZeLUQ=
|
||||||
github.com/joshuarubin/lifecycle v1.0.0/go.mod h1:sRy++ATvR9Ee21tkRdFkQeywAWvDsue66V70K0Dnl54=
|
github.com/joshuarubin/lifecycle v1.0.0/go.mod h1:sRy++ATvR9Ee21tkRdFkQeywAWvDsue66V70K0Dnl54=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/sirupsen/logrus v1.8.1 h1:dJKuHgqk1NNQlqoA6BTlM1Wf9DOH3NBjQyu0h9+AZZE=
|
github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0=
|
||||||
github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
|
github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
|
||||||
github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
|
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
|
||||||
|
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4=
|
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4=
|
||||||
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
||||||
go.uber.org/multierr v1.1.0 h1:HoEmRHQPVSqub6w2z2d2EOVs2fjyFRGyofhKuyDq0QI=
|
go.uber.org/multierr v1.1.0 h1:HoEmRHQPVSqub6w2z2d2EOVs2fjyFRGyofhKuyDq0QI=
|
||||||
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
||||||
golang.org/x/sync v0.0.0-20190412183630-56d357773e84 h1:IqXQ59gzdXv58Jmm2xn0tSOR9i6HqroaOFRQ3wR/dJQ=
|
golang.org/x/sync v0.0.0-20190412183630-56d357773e84 h1:IqXQ59gzdXv58Jmm2xn0tSOR9i6HqroaOFRQ3wR/dJQ=
|
||||||
golang.org/x/sync v0.0.0-20190412183630-56d357773e84/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20190412183630-56d357773e84/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c h1:F1jZWGFhYfh0Ci55sIpILtKKK8p3i2/krTr0H1rg74I=
|
|
||||||
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8 h1:0A+M6Uqn+Eje4kHMK80dtF3JCXC4ykBgQG4Fe06QRhQ=
|
||||||
|
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
|||||||
43
main.go
43
main.go
@@ -21,7 +21,7 @@ import (
|
|||||||
"github.com/gotk3/gotk3/gtk"
|
"github.com/gotk3/gotk3/gtk"
|
||||||
)
|
)
|
||||||
|
|
||||||
const version = "0.3.0"
|
const version = "0.3.7"
|
||||||
|
|
||||||
var (
|
var (
|
||||||
appDirs []string
|
appDirs []string
|
||||||
@@ -189,7 +189,7 @@ func main() {
|
|||||||
// Otherwise the command may behave in two ways:
|
// Otherwise the command may behave in two ways:
|
||||||
// 1. kill the running non-residennt instance and exit;
|
// 1. kill the running non-residennt instance and exit;
|
||||||
// 2. die if a resident instance found.
|
// 2. die if a resident instance found.
|
||||||
lockFilePath := path.Join(tempDir(), "nwg-drawer.lock")
|
lockFilePath := path.Join(dataDir(), "nwg-drawer.lock")
|
||||||
lockFile, err := singleinstance.CreateLockFile(lockFilePath)
|
lockFile, err := singleinstance.CreateLockFile(lockFilePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
pid, err := readTextFile(lockFilePath)
|
pid, err := readTextFile(lockFilePath)
|
||||||
@@ -200,7 +200,10 @@ func main() {
|
|||||||
log.Warnf("Resident instance already running (PID %v)", i)
|
log.Warnf("Resident instance already running (PID %v)", i)
|
||||||
} else {
|
} else {
|
||||||
log.Infof("Showing resident instance (PID %v)", i)
|
log.Infof("Showing resident instance (PID %v)", i)
|
||||||
syscall.Kill(i, syscall.SIGUSR1)
|
err := syscall.Kill(i, syscall.SIGUSR1)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -243,7 +246,10 @@ func main() {
|
|||||||
|
|
||||||
// Copy default style sheet if not found
|
// Copy default style sheet if not found
|
||||||
if !pathExists(filepath.Join(configDirectory, "drawer.css")) {
|
if !pathExists(filepath.Join(configDirectory, "drawer.css")) {
|
||||||
copyFile(filepath.Join(getDataHome(), "nwg-drawer/drawer.css"), filepath.Join(configDirectory, "drawer.css"))
|
err := copyFile("/usr/share/nwg-drawer/drawer.css", filepath.Join(configDirectory, "drawer.css"))
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("Failed copying 'drawer.css' file: %s", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cacheDirectory := cacheDir()
|
cacheDirectory := cacheDir()
|
||||||
@@ -312,7 +318,11 @@ func main() {
|
|||||||
log.Infof("User demanded theme: %s", *gtkTheme)
|
log.Infof("User demanded theme: %s", *gtkTheme)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
settings.SetProperty("gtk-application-prefer-dark-theme", true)
|
err := settings.SetProperty("gtk-application-prefer-dark-theme", true)
|
||||||
|
if err != nil {
|
||||||
|
log.Error("Error setting 'gtk-application-prefer-dark-theme' property")
|
||||||
|
return
|
||||||
|
}
|
||||||
log.Info("Preferring dark theme variants")
|
log.Info("Preferring dark theme variants")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -480,6 +490,7 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
userDirsMap = mapXdgUserDirs()
|
userDirsMap = mapXdgUserDirs()
|
||||||
|
log.Debugf("User dirs map: %s", userDirsMap)
|
||||||
|
|
||||||
placeholder, _ := gtk.BoxNew(gtk.ORIENTATION_VERTICAL, 0)
|
placeholder, _ := gtk.BoxNew(gtk.ORIENTATION_VERTICAL, 0)
|
||||||
resultsWrapper.PackStart(placeholder, true, true, 0)
|
resultsWrapper.PackStart(placeholder, true, true, 0)
|
||||||
@@ -539,12 +550,14 @@ func main() {
|
|||||||
fileSearchResultWrapper.Hide()
|
fileSearchResultWrapper.Hide()
|
||||||
}
|
}
|
||||||
// focus 1st element
|
// focus 1st element
|
||||||
b := appFlowBox.GetChildAtIndex(0)
|
var button gtk.IWidget
|
||||||
if b != nil {
|
if pinnedFlowBox.GetChildren().Length() > 0 {
|
||||||
button, err := b.GetChild()
|
button, err = pinnedFlowBox.GetChildAtIndex(0).GetChild()
|
||||||
if err == nil {
|
} else {
|
||||||
button.ToWidget().GrabFocus()
|
button, err = appFlowBox.GetChildAtIndex(0).GetChild()
|
||||||
}
|
}
|
||||||
|
if err == nil {
|
||||||
|
button.ToWidget().GrabFocus()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -570,7 +583,9 @@ func main() {
|
|||||||
|
|
||||||
func restoreStateAndHide() {
|
func restoreStateAndHide() {
|
||||||
timeStart1 := time.Now()
|
timeStart1 := time.Now()
|
||||||
win.Hide()
|
if win != nil {
|
||||||
|
win.Hide()
|
||||||
|
}
|
||||||
|
|
||||||
// clear search
|
// clear search
|
||||||
searchEntry.SetText("")
|
searchEntry.SetText("")
|
||||||
@@ -583,7 +598,9 @@ func restoreStateAndHide() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// scroll to the top
|
// scroll to the top
|
||||||
resultWindow.GetVAdjustment().SetValue(0)
|
if resultWindow != nil {
|
||||||
|
resultWindow.GetVAdjustment().SetValue(0)
|
||||||
|
}
|
||||||
|
|
||||||
t := time.Now()
|
t := time.Now()
|
||||||
log.Debugf(fmt.Sprintf("UI hidden and restored in the backgroud in %v ms", t.Sub(timeStart1).Milliseconds()))
|
log.Debugf(fmt.Sprintf("UI hidden and restored in the backgroud in %v ms", t.Sub(timeStart1).Milliseconds()))
|
||||||
|
|||||||
108
tools.go
108
tools.go
@@ -7,7 +7,6 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/fs"
|
"io/fs"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path"
|
"path"
|
||||||
@@ -79,8 +78,9 @@ func mapXdgUserDirs() map[string]string {
|
|||||||
result["pictures"] = filepath.Join(home, "Pictures")
|
result["pictures"] = filepath.Join(home, "Pictures")
|
||||||
result["videos"] = filepath.Join(home, "Videos")
|
result["videos"] = filepath.Join(home, "Videos")
|
||||||
|
|
||||||
userDirsFile := filepath.Join(home, ".config/user-dirs.dirs")
|
userDirsFile := filepath.Join(filepath.Join(os.Getenv("XDG_CONFIG_HOME"), "user-dirs.dirs"))
|
||||||
if pathExists(userDirsFile) {
|
if pathExists(userDirsFile) {
|
||||||
|
log.Debugf("userDirsFile found: %s", userDirsFile)
|
||||||
log.Info(fmt.Sprintf("Using XDG user dirs from %s", userDirsFile))
|
log.Info(fmt.Sprintf("Using XDG user dirs from %s", userDirsFile))
|
||||||
lines, _ := loadTextFile(userDirsFile)
|
lines, _ := loadTextFile(userDirsFile)
|
||||||
for _, l := range lines {
|
for _, l := range lines {
|
||||||
@@ -105,7 +105,7 @@ func mapXdgUserDirs() map[string]string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
log.Warnf("%s file not found, using defaults", userDirsFile)
|
log.Warnf("userDirsFile %s not found, using defaults", userDirsFile)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
@@ -131,19 +131,8 @@ func cacheDir() string {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func tempDir() string {
|
|
||||||
if os.Getenv("TMPDIR") != "" {
|
|
||||||
return os.Getenv("TMPDIR")
|
|
||||||
} else if os.Getenv("TEMP") != "" {
|
|
||||||
return os.Getenv("TEMP")
|
|
||||||
} else if os.Getenv("TMP") != "" {
|
|
||||||
return os.Getenv("TMP")
|
|
||||||
}
|
|
||||||
return "/tmp"
|
|
||||||
}
|
|
||||||
|
|
||||||
func readTextFile(path string) (string, error) {
|
func readTextFile(path string) (string, error) {
|
||||||
bytes, err := ioutil.ReadFile(path)
|
bytes, err := os.ReadFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
@@ -177,6 +166,20 @@ func configDir() string {
|
|||||||
return dir
|
return dir
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func dataDir() string {
|
||||||
|
var dir string
|
||||||
|
if os.Getenv("XDG_DATA_HOME") != "" {
|
||||||
|
dir = path.Join(os.Getenv("XDG_DATA_HOME"), "nwg-drawer")
|
||||||
|
} else if os.Getenv("HOME") != "" {
|
||||||
|
dir = path.Join(os.Getenv("HOME"), ".local/share/nwg-drawer")
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Infof("Data dir: %s", dir)
|
||||||
|
createDir(dir)
|
||||||
|
|
||||||
|
return dir
|
||||||
|
}
|
||||||
|
|
||||||
func createDir(dir string) {
|
func createDir(dir string) {
|
||||||
if _, err := os.Stat(dir); os.IsNotExist(err) {
|
if _, err := os.Stat(dir); os.IsNotExist(err) {
|
||||||
err := os.MkdirAll(dir, os.ModePerm)
|
err := os.MkdirAll(dir, os.ModePerm)
|
||||||
@@ -197,12 +200,22 @@ func copyFile(src, dst string) error {
|
|||||||
if srcfd, err = os.Open(src); err != nil {
|
if srcfd, err = os.Open(src); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer srcfd.Close()
|
defer func(srcfd *os.File) {
|
||||||
|
err := srcfd.Close()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("Error closing file: %v", srcfd)
|
||||||
|
}
|
||||||
|
}(srcfd)
|
||||||
|
|
||||||
if dstfd, err = os.Create(dst); err != nil {
|
if dstfd, err = os.Create(dst); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer dstfd.Close()
|
defer func(dstfd *os.File) {
|
||||||
|
err := dstfd.Close()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("Error closing file: %v", dstfd)
|
||||||
|
}
|
||||||
|
}(dstfd)
|
||||||
|
|
||||||
if _, err = io.Copy(dstfd, srcfd); err != nil {
|
if _, err = io.Copy(dstfd, srcfd); err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -213,13 +226,6 @@ func copyFile(src, dst string) error {
|
|||||||
return os.Chmod(dst, srcinfo.Mode())
|
return os.Chmod(dst, srcinfo.Mode())
|
||||||
}
|
}
|
||||||
|
|
||||||
func getDataHome() string {
|
|
||||||
if os.Getenv("XDG_DATA_HOME") != "" {
|
|
||||||
return os.Getenv("XDG_DATA_HOME")
|
|
||||||
}
|
|
||||||
return "/usr/share/"
|
|
||||||
}
|
|
||||||
|
|
||||||
func getAppDirs() []string {
|
func getAppDirs() []string {
|
||||||
var dirs []string
|
var dirs []string
|
||||||
xdgDataDirs := ""
|
xdgDataDirs := ""
|
||||||
@@ -261,12 +267,20 @@ func loadPreferredApps(path string) (map[string]interface{}, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
defer jsonFile.Close()
|
defer func(jsonFile *os.File) {
|
||||||
|
err := jsonFile.Close()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("Error closing file: %v", jsonFile)
|
||||||
|
}
|
||||||
|
}(jsonFile)
|
||||||
|
|
||||||
byteValue, _ := ioutil.ReadAll(jsonFile)
|
byteValue, _ := io.ReadAll(jsonFile)
|
||||||
|
|
||||||
var result map[string]interface{}
|
var result map[string]interface{}
|
||||||
json.Unmarshal([]byte(byteValue), &result)
|
err = json.Unmarshal(byteValue, &result)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
if len(result) == 0 {
|
if len(result) == 0 {
|
||||||
return nil, errors.New("json invalid or empty")
|
return nil, errors.New("json invalid or empty")
|
||||||
@@ -275,8 +289,8 @@ func loadPreferredApps(path string) (map[string]interface{}, error) {
|
|||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func listFiles(dir string) ([]fs.FileInfo, error) {
|
func listFiles(dir string) ([]fs.DirEntry, error) {
|
||||||
files, err := ioutil.ReadDir(dir)
|
files, err := os.ReadDir(dir)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
return files, nil
|
return files, nil
|
||||||
}
|
}
|
||||||
@@ -300,12 +314,12 @@ func listDesktopFiles() []string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func setUpCategories() {
|
func setUpCategories() {
|
||||||
path := filepath.Join(getDataHome(), "nwg-drawer/desktop-directories")
|
|
||||||
var other category
|
var other category
|
||||||
|
|
||||||
for _, cName := range categoryNames {
|
for _, cName := range categoryNames {
|
||||||
fileName := fmt.Sprintf("%s.directory", cName)
|
fileName := fmt.Sprintf("%s.directory", cName)
|
||||||
lines, err := loadTextFile(filepath.Join(path, fileName))
|
fp := filepath.Join("/usr/share/nwg-drawer/desktop-directories", fileName)
|
||||||
|
lines, err := loadTextFile(fp)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
var cat category
|
var cat category
|
||||||
cat.Name = cName
|
cat.Name = cName
|
||||||
@@ -350,6 +364,8 @@ func setUpCategories() {
|
|||||||
} else {
|
} else {
|
||||||
other = cat
|
other = cat
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
log.Errorf("Couldn't open %s", fp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sort.Slice(categories, func(i, j int) bool {
|
sort.Slice(categories, func(i, j int) bool {
|
||||||
@@ -463,7 +479,7 @@ func pathExists(name string) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func loadTextFile(path string) ([]string, error) {
|
func loadTextFile(path string) ([]string, error) {
|
||||||
bytes, err := ioutil.ReadFile(path)
|
bytes, err := os.ReadFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -514,14 +530,20 @@ func savePinned() {
|
|||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
defer f.Close()
|
defer func(f *os.File) {
|
||||||
|
err := f.Close()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("Error closing file: %v", f)
|
||||||
|
}
|
||||||
|
}(f)
|
||||||
|
|
||||||
for _, line := range pinned {
|
for _, line := range pinned {
|
||||||
if line != "" {
|
//skip invalid lines
|
||||||
|
if line != "" && id2entry[line].DesktopID != "" {
|
||||||
_, err := f.WriteString(line + "\n")
|
_, err := f.WriteString(line + "\n")
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Errorf("Error saving pinned", err)
|
log.Error("Error saving pinned", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -669,3 +691,19 @@ func mapOutputs() (map[string]*gdk.Monitor, error) {
|
|||||||
}
|
}
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// KAdot / https://stackoverflow.com/a/38537764/4040598 - thanks!
|
||||||
|
func substring(s string, start int, end int) string {
|
||||||
|
startStrIdx := 0
|
||||||
|
i := 0
|
||||||
|
for j := range s {
|
||||||
|
if i == start {
|
||||||
|
startStrIdx = j
|
||||||
|
}
|
||||||
|
if i == end {
|
||||||
|
return s[startStrIdx:j]
|
||||||
|
}
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
return s[startStrIdx:]
|
||||||
|
}
|
||||||
|
|||||||
@@ -32,6 +32,10 @@ func setUpPinnedFlowBox() *gtk.FlowBox {
|
|||||||
if len(pinned) > 0 {
|
if len(pinned) > 0 {
|
||||||
for _, desktopID := range pinned {
|
for _, desktopID := range pinned {
|
||||||
entry := id2entry[desktopID]
|
entry := id2entry[desktopID]
|
||||||
|
if entry.DesktopID == "" {
|
||||||
|
log.Debugf("Pinned item doesn't seem to exist: %s", desktopID)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
btn, _ := gtk.ButtonNew()
|
btn, _ := gtk.ButtonNew()
|
||||||
|
|
||||||
@@ -54,9 +58,8 @@ func setUpPinnedFlowBox() *gtk.FlowBox {
|
|||||||
name = entry.Name
|
name = entry.Name
|
||||||
}
|
}
|
||||||
if len(name) > 20 {
|
if len(name) > 20 {
|
||||||
r := []rune(name)
|
r := substring(name, 0, 17)
|
||||||
name = string(r[:17])
|
name = fmt.Sprintf("%s…", string(r))
|
||||||
name = fmt.Sprintf("%s…", name)
|
|
||||||
}
|
}
|
||||||
btn.SetLabel(name)
|
btn.SetLabel(name)
|
||||||
|
|
||||||
@@ -248,9 +251,8 @@ func flowBoxButton(entry desktopEntry) *gtk.Button {
|
|||||||
button.SetImagePosition(gtk.POS_TOP)
|
button.SetImagePosition(gtk.POS_TOP)
|
||||||
name := entry.NameLoc
|
name := entry.NameLoc
|
||||||
if len(name) > 20 {
|
if len(name) > 20 {
|
||||||
r := []rune(name)
|
r := substring(name, 0, 17)
|
||||||
name = string(r[:17])
|
name = fmt.Sprintf("%s…", string(r))
|
||||||
name = fmt.Sprintf("%s…", name)
|
|
||||||
}
|
}
|
||||||
button.SetLabel(name)
|
button.SetLabel(name)
|
||||||
|
|
||||||
@@ -259,9 +261,8 @@ func flowBoxButton(entry desktopEntry) *gtk.Button {
|
|||||||
terminal := entry.Terminal
|
terminal := entry.Terminal
|
||||||
desc := entry.CommentLoc
|
desc := entry.CommentLoc
|
||||||
if len(desc) > 120 {
|
if len(desc) > 120 {
|
||||||
r := []rune(desc)
|
r := substring(desc, 0, 117)
|
||||||
desc = string(r[:117])
|
desc = fmt.Sprintf("%s…", string(r))
|
||||||
desc = fmt.Sprintf("%s…", desc)
|
|
||||||
}
|
}
|
||||||
button.Connect("button-release-event", func(btn *gtk.Button, e *gdk.Event) bool {
|
button.Connect("button-release-event", func(btn *gtk.Button, e *gdk.Event) bool {
|
||||||
btnEvent := gdk.EventButtonNewFromEvent(e)
|
btnEvent := gdk.EventButtonNewFromEvent(e)
|
||||||
@@ -430,6 +431,7 @@ func searchUserDir(dir string) {
|
|||||||
fileSearchResultFlowBox.Add(btn)
|
fileSearchResultFlowBox.Add(btn)
|
||||||
|
|
||||||
for _, path := range fileSearchResults {
|
for _, path := range fileSearchResults {
|
||||||
|
log.Debugf("Path: %s", path)
|
||||||
partOfPathToShow := strings.Split(path, userDirsMap[dir])[1]
|
partOfPathToShow := strings.Split(path, userDirsMap[dir])[1]
|
||||||
if partOfPathToShow != "" {
|
if partOfPathToShow != "" {
|
||||||
if !(strings.HasPrefix(path, "#is_dir#") && isExcluded(path)) {
|
if !(strings.HasPrefix(path, "#is_dir#") && isExcluded(path)) {
|
||||||
|
|||||||
@@ -54,7 +54,34 @@ func parseDesktopEntry(id string, in io.Reader) (entry desktopEntry, err error)
|
|||||||
case "Terminal":
|
case "Terminal":
|
||||||
entry.Terminal, _ = strconv.ParseBool(value)
|
entry.Terminal, _ = strconv.ParseBool(value)
|
||||||
case "NoDisplay":
|
case "NoDisplay":
|
||||||
entry.NoDisplay, _ = strconv.ParseBool(value)
|
if !entry.NoDisplay {
|
||||||
|
entry.NoDisplay, _ = strconv.ParseBool(value)
|
||||||
|
}
|
||||||
|
case "Hidden":
|
||||||
|
if !entry.NoDisplay {
|
||||||
|
entry.NoDisplay, _ = strconv.ParseBool(value)
|
||||||
|
}
|
||||||
|
case "OnlyShowIn":
|
||||||
|
if !entry.NoDisplay {
|
||||||
|
entry.NoDisplay = true
|
||||||
|
currentDesktop := os.Getenv("XDG_CURRENT_DESKTOP")
|
||||||
|
if currentDesktop != "" {
|
||||||
|
for _, ele := range strings.Split(value, ";") {
|
||||||
|
if ele == currentDesktop && ele != "" {
|
||||||
|
entry.NoDisplay = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "NotShowIn":
|
||||||
|
currentDesktop := os.Getenv("XDG_CURRENT_DESKTOP")
|
||||||
|
if !entry.NoDisplay && currentDesktop != "" {
|
||||||
|
for _, ele := range strings.Split(value, ";") {
|
||||||
|
if ele == currentDesktop && ele != "" {
|
||||||
|
entry.NoDisplay = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
case "Exec":
|
case "Exec":
|
||||||
entry.Exec = cleanexec.Replace(value)
|
entry.Exec = cleanexec.Replace(value)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user