32 Commits

Author SHA1 Message Date
Piotr Miller
7b76c271d9 Merge pull request #25 from nwg-piotr/logging
Logging + 1 bug fixed
2021-09-17 02:52:44 +02:00
piotr
3768e21b61 bump to 0.1.12 2021-09-17 02:38:09 +02:00
piotr
c1a807615f avoid closing window on item pinned 2021-09-17 02:35:43 +02:00
piotr
b6dd4099fd logging with logrus 2021-09-17 02:33:39 +02:00
Piotr Miller
6ca8e6a175 Merge pull request #24 from nwg-piotr/fix_nodisplay
Fix nodisplay, fix running in terminal
2021-09-07 02:51:26 +02:00
piotr
f12be46583 fix #23 2021-09-07 02:47:17 +02:00
piotr
9d50258042 don't skip hidden entries: fixes #22 added in #19 2021-09-07 02:07:29 +02:00
piotr
1be8e432fa print errors in human-readable format 2021-09-05 01:57:24 +02:00
piotr
aa227b51b8 update README 2021-09-05 01:35:44 +02:00
piotr
56d6cd264d update README 2021-09-05 01:34:57 +02:00
piotr
bea2fdf90c verrsion bump 2021-09-05 01:26:36 +02:00
Piotr Miller
6a0f9d4a93 Merge pull request #20 from james-lawrence/use-terminal-env
use terminal environment variable if present
2021-09-05 01:12:30 +02:00
James Lawrence
206a602e77 use terminal environment variable if present 2021-09-04 15:29:58 -04:00
Piotr Miller
c655f99cd9 Merge pull request #19 from james-lawrence/bugfix/whitespace-handling
Bugfix/whitespace handling
2021-09-01 02:18:26 +02:00
James Lawrence
3a59228eaa remove old parse entry version 2021-08-31 06:33:04 -04:00
James Lawrence
57826c064d improve parsing of desktop entries.
- speed up parsing by not creating nearly as many strings.
- fix basic whitespace handling. fixes #4

goos: linux
goarch: amd64
pkg: github.com/nwg-piotr/nwg-drawer
cpu: AMD Ryzen 7 1800X Eight-Core Processor
BenchmarkDesktopEntryParserOld-16    	   10000	    146094 ns/op
BenchmarkDesktopEntryParser-16       	   26097	     43303 ns/op
PASS
ok  	github.com/nwg-piotr/nwg-drawer	3.090s
2021-08-31 06:32:40 -04:00
Piotr Miller
c5e39f179f Merge pull request #18 from nwg-piotr/searchres
focus first search result #17
2021-08-30 03:17:49 +02:00
piotr
ce48848589 focus first search result #17 2021-08-29 03:37:00 +02:00
Piotr Miller
9d0c4d2e5f Merge pull request #16 from nwg-piotr/fixnofs
fix crash on a category button click while file search turned off (`-nofs`)
2021-08-27 01:33:01 +02:00
piotr
a7968d8510 version bump 2021-08-27 01:23:16 +02:00
piotr
e601316480 add missing nil check #15 2021-08-27 01:21:41 +02:00
piotr
4818dc3358 update README 2021-08-21 01:45:22 +02:00
piotr
9afd7dca20 update README 2021-08-21 01:41:45 +02:00
piotr
eb1e9994e7 update README 2021-08-21 01:40:19 +02:00
Piotr Miller
3e2b9395d5 Merge pull request #14 from nwg-piotr/files
xdg-open support + customization + 1 bug fixed
2021-08-21 00:49:24 +02:00
piotr
061d5e1ca9 avoid unwanted closes 2021-08-21 00:34:50 +02:00
piotr
dc50203ee3 update README 2021-08-20 05:43:16 +02:00
piotr
656347855b update README 2021-08-20 05:40:51 +02:00
piotr
81aeb10311 update README 2021-08-20 05:36:27 +02:00
piotr
04a292c2ec check err just in case 2021-08-20 05:02:37 +02:00
piotr
6912e7f320 xdg-open or open in FM 2021-08-20 04:46:26 +02:00
piotr
41d48a0824 version bump 2021-08-17 22:49:09 +02:00
10 changed files with 345 additions and 148 deletions

View File

@@ -5,6 +5,7 @@ get:
go get github.com/dlasky/gotk3-layershell/layershell go get github.com/dlasky/gotk3-layershell/layershell
go get github.com/joshuarubin/go-sway go get github.com/joshuarubin/go-sway
go get github.com/allan-simon/go-singleinstance go get github.com/allan-simon/go-singleinstance
go get "github.com/sirupsen/logrus"
build: build:
go build -o bin/nwg-drawer *.go go build -o bin/nwg-drawer *.go
@@ -20,4 +21,5 @@ uninstall:
rm /usr/bin/nwg-drawer rm /usr/bin/nwg-drawer
run: run:
go run *.go go build -o bin/nwg-drawer *.go
bin/nwg-drawer

View File

@@ -25,17 +25,18 @@ and `nwggrid`.
### Dependencies ### Dependencies
- go 1.16 (just to build) - go >=1.16 (just to build)
- gtk3 - gtk3
- gtk-layer-shell - gtk-layer-shell
- xdg-utils
Optional (recommended): Optional (recommended):
- thunar - thunar
- alacritty - alacritty
You may use another file manager and terminal emulator (see command line arguments), but for now the program has You may use another file manager and terminal emulator (see command line arguments), but mentioned above have been
only been tested with the two mentioned above. confirmed to work well with the program. Also see **Files** below.
### Steps ### Steps
@@ -81,10 +82,43 @@ Usage of nwg-drawer:
-v display Version information -v display Version information
``` ```
*NOTE: the `$TERM` environment variable overrides the `-term` argument if defined.*
## Styling ## Styling
Edit `~/.config/nwg-panel/drawer.css` to your taste. Edit `~/.config/nwg-panel/drawer.css` to your taste.
## Files
When the search phrase is at least 3 characters long, your XDG user directories are being searched.
![screenshot-03.png](https://scrot.cloud/images/2021/05/30/screenshot-03.png)
Use the **left mouse button** to open a file with the `xdg-open` command. As configuring file associations for it is
PITA, you may override them, by creating the `~/.config/nwg-panel/preferred-apps.json` file with your own definitions.
### Sample file content
```json
{
"\\.pdf$": "atril",
"\\.svg$": "inkscape",
"\\.(jpg|png|tiff|gif)$": "feh",
"\\.(mp3|ogg|flac|wav|wma)$": "audacious",
"\\.(avi|mp4|mkv|mov|wav)$": "mpv",
"\\.(doc|docx|xls|xlsx)$": "libreoffice"
}
```
Use the **right mouse button** to open the file with your file manager (see `-fm` argument). The result depends on the
file manager you use.
- thunar will open the file location
- pcmanfm will open the file with its associated program
- caja won't open anything, except for directories
I've noy yet tried other file managers.
## Credits ## Credits
This program uses some great libraries: This program uses some great libraries:

Binary file not shown.

1
go.mod
View File

@@ -7,4 +7,5 @@ require (
github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261 github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261
github.com/gotk3/gotk3 v0.6.0 github.com/gotk3/gotk3 v0.6.0
github.com/joshuarubin/go-sway v0.0.4 github.com/joshuarubin/go-sway v0.0.4
github.com/sirupsen/logrus v1.8.1
) )

8
go.sum
View File

@@ -1,7 +1,8 @@
github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37 h1:28uU3TtuvQ6KRndxg9TrC868jBWmSKgh0GTXkACCXmA= github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37 h1:28uU3TtuvQ6KRndxg9TrC868jBWmSKgh0GTXkACCXmA=
github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37/go.mod h1:6AXRstqK+32jeFmw89QGL2748+dj34Av4xc/I9oo9BY= github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37/go.mod h1:6AXRstqK+32jeFmw89QGL2748+dj34Av4xc/I9oo9BY=
github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261 h1:eoXn91ckLWKMXmQKX34UHEF2XMyQpRnnP80fDiu+kys= github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261 h1:eoXn91ckLWKMXmQKX34UHEF2XMyQpRnnP80fDiu+kys=
github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261/go.mod h1:d56Gslp3IaiT8lqxD/lO1Msz1wYgD8D/HTKHgSdg9tU= github.com/dlasky/gotk3-layershell v0.0.0-20210331230524-5cca0b819261/go.mod h1:d56Gslp3IaiT8lqxD/lO1Msz1wYgD8D/HTKHgSdg9tU=
github.com/gotk3/gotk3 v0.5.3-0.20210223154815-289cfb6dbf32/go.mod h1:/hqFpkNa9T3JgNAE2fLvCdov7c5bw//FHNZrZ3Uv9/Q= github.com/gotk3/gotk3 v0.5.3-0.20210223154815-289cfb6dbf32/go.mod h1:/hqFpkNa9T3JgNAE2fLvCdov7c5bw//FHNZrZ3Uv9/Q=
@@ -13,7 +14,10 @@ github.com/joshuarubin/lifecycle v1.0.0 h1:N/lPEC8f+dBZ1Tn99vShqp36LwB+LI7XNAiNa
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.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
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 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=
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4= go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4=
@@ -22,3 +26,5 @@ 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 h1:YyJpGZS1sBuBCzLAR1VEpK193GlqGZbnPFnPV/5Rsb4=
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=

62
main.go
View File

@@ -3,7 +3,6 @@ package main
import ( import (
"flag" "flag"
"fmt" "fmt"
"log"
"os" "os"
"os/signal" "os/signal"
"path/filepath" "path/filepath"
@@ -12,6 +11,8 @@ import (
"syscall" "syscall"
"time" "time"
log "github.com/sirupsen/logrus"
"github.com/allan-simon/go-singleinstance" "github.com/allan-simon/go-singleinstance"
"github.com/dlasky/gotk3-layershell/layershell" "github.com/dlasky/gotk3-layershell/layershell"
"github.com/gotk3/gotk3/gdk" "github.com/gotk3/gotk3/gdk"
@@ -19,7 +20,7 @@ import (
"github.com/gotk3/gotk3/gtk" "github.com/gotk3/gotk3/gtk"
) )
const version = "0.1.5" const version = "0.1.12"
var ( var (
appDirs []string appDirs []string
@@ -28,6 +29,7 @@ var (
pinned []string pinned []string
src glib.SourceHandle src glib.SourceHandle
id2entry map[string]desktopEntry id2entry map[string]desktopEntry
preferredApps map[string]interface{}
) )
var categoryNames = [...]string{ var categoryNames = [...]string{
@@ -58,6 +60,7 @@ type desktopEntry struct {
CommentLoc string CommentLoc string
Icon string Icon string
Exec string Exec string
Category string
Terminal bool Terminal bool
NoDisplay bool NoDisplay bool
} }
@@ -97,6 +100,16 @@ var (
ignore string ignore string
) )
func defaultStringIfBlank(s, fallback string) string {
s = strings.TrimSpace(s)
// os.Getenv("TERM") returns "linux" instead of empty string, if program has been started
// from a key binding defined in the config file. See #23.
if s == "" || s == "linux" {
return fallback
}
return s
}
// Flags // Flags
var cssFileName = flag.String("s", "drawer.css", "Styling: css file name") var cssFileName = flag.String("s", "drawer.css", "Styling: css file name")
var targetOutput = flag.String("o", "", "name of the Output to display the drawer on (sway only)") var targetOutput = flag.String("o", "", "name of the Output to display the drawer on (sway only)")
@@ -108,7 +121,7 @@ var columnsNumber = flag.Uint("c", 6, "number of Columns")
var itemSpacing = flag.Uint("spacing", 20, "icon spacing") var itemSpacing = flag.Uint("spacing", 20, "icon spacing")
var lang = flag.String("lang", "", "force lang, e.g. \"en\", \"pl\"") var lang = flag.String("lang", "", "force lang, e.g. \"en\", \"pl\"")
var fileManager = flag.String("fm", "thunar", "File Manager") var fileManager = flag.String("fm", "thunar", "File Manager")
var term = flag.String("term", "alacritty", "Terminal emulator") var term = flag.String("term", defaultStringIfBlank(os.Getenv("TERM"), "alacritty"), "Terminal emulator")
var nameLimit = flag.Int("fslen", 80, "File Search name length Limit") var nameLimit = flag.Int("fslen", 80, "File Search name length Limit")
var noCats = flag.Bool("nocats", false, "Disable filtering by category") var noCats = flag.Bool("nocats", false, "Disable filtering by category")
var noFS = flag.Bool("nofs", false, "Disable file search") var noFS = flag.Bool("nofs", false, "Disable file search")
@@ -129,7 +142,7 @@ func main() {
for { for {
s := <-signalChan s := <-signalChan
if s == syscall.SIGTERM { if s == syscall.SIGTERM {
println("SIGTERM received, bye bye!") log.Info("SIGTERM received, bye bye!")
gtk.MainQuit() gtk.MainQuit()
} }
} }
@@ -143,7 +156,7 @@ func main() {
if err == nil { if err == nil {
i, err := strconv.Atoi(pid) i, err := strconv.Atoi(pid)
if err == nil { if err == nil {
println("Running instance found, sending SIGTERM and exiting...") log.Info("Running instance found, sending SIGTERM and exiting...")
syscall.Kill(i, syscall.SIGTERM) syscall.Kill(i, syscall.SIGTERM)
} }
} }
@@ -155,7 +168,7 @@ func main() {
if *lang == "" && os.Getenv("LANG") != "" { if *lang == "" && os.Getenv("LANG") != "" {
*lang = strings.Split(os.Getenv("LANG"), ".")[0] *lang = strings.Split(os.Getenv("LANG"), ".")[0]
} }
println(fmt.Sprintf("lang: %s", *lang)) log.Info(fmt.Sprintf("lang: %s", *lang))
// ENVIRONMENT // ENVIRONMENT
configDirectory = configDir() configDirectory = configDir()
@@ -175,7 +188,7 @@ func main() {
if err != nil { if err != nil {
pinned = nil pinned = nil
} }
println(fmt.Sprintf("Found %v pinned items", len(pinned))) log.Info(fmt.Sprintf("Found %v pinned items", len(pinned)))
cssFile := filepath.Join(configDirectory, *cssFileName) cssFile := filepath.Join(configDirectory, *cssFileName)
@@ -184,10 +197,20 @@ func main() {
setUpCategories() setUpCategories()
desktopFiles := listDesktopFiles() desktopFiles := listDesktopFiles()
println(fmt.Sprintf("Found %v desktop files", len(desktopFiles))) log.Info(fmt.Sprintf("Found %v desktop files", len(desktopFiles)))
status = parseDesktopFiles(desktopFiles) status = parseDesktopFiles(desktopFiles)
// For opening files we use xdg-open. As its configuration is PITA, we may override some associations
// in the ~/.config/nwg-panel/preferred-apps.json file.
paFile := filepath.Join(configDirectory, "preferred-apps.json")
preferredApps, err = loadPreferredApps(paFile)
if err != nil {
log.Error(fmt.Sprintf("Custom associations file %s not found or invalid", paFile))
} else {
log.Info(fmt.Sprintf("Found %v associations in %s", len(preferredApps), paFile))
}
// USER INTERFACE // USER INTERFACE
gtk.Init(nil) gtk.Init(nil)
@@ -195,10 +218,10 @@ func main() {
err = cssProvider.LoadFromPath(cssFile) err = cssProvider.LoadFromPath(cssFile)
if err != nil { if err != nil {
println(fmt.Sprintf("ERROR: %s css file not found or erroneous. Using GTK styling.", cssFile)) log.Errorf("ERROR: %s css file not found or erroneous. Using GTK styling.", cssFile)
println(fmt.Sprintf(">>> %s", err)) log.Errorf("%s", err)
} else { } else {
println(fmt.Sprintf("Using style from %s", cssFile)) log.Info(fmt.Sprintf("Using style from %s", cssFile))
screen, _ := gdk.ScreenGetDefault() screen, _ := gdk.ScreenGetDefault()
gtk.AddProviderForScreen(screen, cssProvider, gtk.STYLE_PROVIDER_PRIORITY_APPLICATION) gtk.AddProviderForScreen(screen, cssProvider, gtk.STYLE_PROVIDER_PRIORITY_APPLICATION)
} }
@@ -220,7 +243,7 @@ func main() {
layershell.SetMonitor(win, monitor) layershell.SetMonitor(win, monitor)
} else { } else {
println(err) log.Errorf("%s", err)
} }
} }
@@ -243,15 +266,6 @@ func main() {
gtk.MainQuit() gtk.MainQuit()
}) })
win.Connect("button-release-event", func(sw *gtk.Window, e *gdk.Event) bool {
btnEvent := gdk.EventButtonNewFromEvent(e)
if btnEvent.Button() == 1 || btnEvent.Button() == 3 {
gtk.MainQuit()
return true
}
return false
})
win.Connect("key-press-event", func(window *gtk.Window, event *gdk.Event) bool { win.Connect("key-press-event", func(window *gtk.Window, event *gdk.Event) bool {
key := &gdk.EventKey{Event: event} key := &gdk.EventKey{Event: event}
switch key.KeyVal() { switch key.KeyVal() {
@@ -270,7 +284,7 @@ func main() {
default: default:
if !searchEntry.IsFocus() { if !searchEntry.IsFocus() {
searchEntry.GrabFocus() searchEntry.GrabFocusWithoutSelecting()
} }
return false return false
} }
@@ -294,7 +308,7 @@ func main() {
This feature is not really supported and will stay undocumented. This feature is not really supported and will stay undocumented.
*/ */
if !wayland() { if !wayland() {
println("Not Wayland, oh really?") log.Info("Not Wayland, oh really?")
win.SetDecorated(false) win.SetDecorated(false)
win.Maximize() win.Maximize()
} }
@@ -387,6 +401,6 @@ func main() {
} }
t := time.Now() t := time.Now()
println(fmt.Sprintf("UI created in %v ms. Thank you for your patience.", t.Sub(timeStart).Milliseconds())) log.Info(fmt.Sprintf("UI created in %v ms. Thank you for your patience.", t.Sub(timeStart).Milliseconds()))
gtk.Main() gtk.Main()
} }

170
tools.go
View File

@@ -2,18 +2,22 @@ package main
import ( import (
"context" "context"
"encoding/json"
"errors"
"fmt" "fmt"
"io" "io"
"io/fs" "io/fs"
"io/ioutil" "io/ioutil"
"log"
"os" "os"
"os/exec" "os/exec"
"path/filepath" "path/filepath"
"regexp"
"sort" "sort"
"strings" "strings"
"time" "time"
log "github.com/sirupsen/logrus"
"github.com/gotk3/gotk3/gdk" "github.com/gotk3/gotk3/gdk"
"github.com/gotk3/gotk3/glib" "github.com/gotk3/gotk3/glib"
"github.com/gotk3/gotk3/gtk" "github.com/gotk3/gotk3/gtk"
@@ -46,7 +50,7 @@ func createPixbuf(icon string, size int) (*gdk.Pixbuf, error) {
if strings.Contains(icon, "/") { if strings.Contains(icon, "/") {
pixbuf, err := gdk.PixbufNewFromFileAtSize(icon, size, size) pixbuf, err := gdk.PixbufNewFromFileAtSize(icon, size, size)
if err != nil { if err != nil {
println(err) log.Errorf("%s", err)
return nil, err return nil, err
} }
return pixbuf, nil return pixbuf, nil
@@ -89,7 +93,7 @@ func mapXdgUserDirs() map[string]string {
userDirsFile := filepath.Join(home, ".config/user-dirs.dirs") userDirsFile := filepath.Join(home, ".config/user-dirs.dirs")
if pathExists(userDirsFile) { if pathExists(userDirsFile) {
println(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 {
if strings.HasPrefix(l, "XDG_DOCUMENTS_DIR") { if strings.HasPrefix(l, "XDG_DOCUMENTS_DIR") {
@@ -113,7 +117,7 @@ func mapXdgUserDirs() map[string]string {
} }
} }
} else { } else {
println(fmt.Sprintf("%s file not found, using defaults", userDirsFile)) log.Warnf("%s file not found, using defaults", userDirsFile)
} }
return result return result
@@ -174,13 +178,13 @@ 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)
if err == nil { if err == nil {
fmt.Println("Creating dir:", dir) log.Infof("Creating dir: %s", dir)
} }
} }
} }
func copyFile(src, dst string) error { func copyFile(src, dst string) error {
fmt.Println("Copying file:", dst) log.Infof("Copying file: %s", dst)
var err error var err error
var srcfd *os.File var srcfd *os.File
@@ -243,6 +247,25 @@ func getAppDirs() []string {
return dirs return dirs
} }
func loadPreferredApps(path string) (map[string]interface{}, error) {
jsonFile, err := os.Open(path)
if err != nil {
return nil, err
}
defer jsonFile.Close()
byteValue, _ := ioutil.ReadAll(jsonFile)
var result map[string]interface{}
json.Unmarshal([]byte(byteValue), &result)
if len(result) == 0 {
return nil, errors.New("json invalid or empty")
}
return result, nil
}
func listFiles(dir string) ([]fs.FileInfo, error) { func listFiles(dir string) ([]fs.FileInfo, error) {
files, err := ioutil.ReadDir(dir) files, err := ioutil.ReadDir(dir)
if err == nil { if err == nil {
@@ -328,109 +351,35 @@ func setUpCategories() {
func parseDesktopFiles(desktopFiles []string) string { func parseDesktopFiles(desktopFiles []string) string {
id2entry = make(map[string]desktopEntry) id2entry = make(map[string]desktopEntry)
var added []string
skipped := 0 skipped := 0
hidden := 0 hidden := 0
for _, file := range desktopFiles { for _, file := range desktopFiles {
lines, err := loadTextFile(file) id := filepath.Base(file)
if err == nil { if _, ok := id2entry[id]; ok {
parts := strings.Split(file, "/") skipped++
desktopID := parts[len(parts)-1] continue
name := "" }
nameLoc := ""
comment := ""
commentLoc := ""
icon := ""
exec := ""
terminal := false
noDisplay := false
categories := "" entry, err := parseDesktopEntryFile(id, file)
if err != nil {
continue
}
for _, l := range lines { if entry.NoDisplay {
if strings.HasPrefix(l, "[") && l != "[Desktop Entry]" {
break
}
if strings.HasPrefix(l, "Name=") {
name = strings.Split(l, "=")[1]
continue
}
if strings.HasPrefix(l, fmt.Sprintf("Name[%s]=", strings.Split(*lang, "_")[0])) {
nameLoc = strings.Split(l, "=")[1]
continue
}
if strings.HasPrefix(l, "Comment=") {
comment = strings.Split(l, "=")[1]
continue
}
if strings.HasPrefix(l, fmt.Sprintf("Comment[%s]=", strings.Split(*lang, "_")[0])) {
commentLoc = strings.Split(l, "=")[1]
continue
}
if strings.HasPrefix(l, "Icon=") {
icon = strings.Split(l, "=")[1]
continue
}
if strings.HasPrefix(l, "Exec=") {
exec = strings.Split(l, "Exec=")[1]
disallowed := [2]string{"\"", "'"}
for _, char := range disallowed {
exec = strings.Replace(exec, char, "", -1)
}
continue
}
if strings.HasPrefix(l, "Categories=") {
categories = strings.Split(l, "Categories=")[1]
continue
}
if l == "Terminal=true" {
terminal = true
continue
}
if l == "NoDisplay=true" {
noDisplay = true
hidden++ hidden++
continue // We still need hidden entries, so `continue` is disallowed here
// Fixes introduced in #19
} }
}
// if name[ln] not found, let's try to find name[ln_LN]
if nameLoc == "" {
nameLoc = name
}
if commentLoc == "" {
commentLoc = comment
}
if !isIn(added, desktopID) {
added = append(added, desktopID)
var entry desktopEntry
entry.DesktopID = desktopID
entry.Name = name
entry.NameLoc = nameLoc
entry.Comment = comment
entry.CommentLoc = commentLoc
entry.Icon = icon
entry.Exec = exec
entry.Terminal = terminal
entry.NoDisplay = noDisplay
desktopEntries = append(desktopEntries, entry)
id2entry[entry.DesktopID] = entry id2entry[entry.DesktopID] = entry
desktopEntries = append(desktopEntries, entry)
assignToLists(entry.DesktopID, categories) assignToLists(entry.DesktopID, entry.Category)
} else {
skipped++
}
}
} }
sort.Slice(desktopEntries, func(i, j int) bool { sort.Slice(desktopEntries, func(i, j int) bool {
return desktopEntries[i].NameLoc < desktopEntries[j].NameLoc return desktopEntries[i].NameLoc < desktopEntries[j].NameLoc
}) })
summary := fmt.Sprintf("%v entries (+%v hidden)", len(desktopEntries)-hidden, hidden) summary := fmt.Sprintf("%v entries (+%v hidden)", len(desktopEntries)-hidden, hidden)
println(fmt.Sprintf("Skipped %v duplicates; %v .desktop entries hidden by \"NoDisplay=true\"", skipped, hidden)) log.Infof("Skipped %v duplicates; %v .desktop entries hidden by \"NoDisplay=true\"", skipped, hidden)
return summary return summary
} }
@@ -523,20 +472,20 @@ func loadTextFile(path string) ([]string, error) {
func pinItem(itemID string) { func pinItem(itemID string) {
for _, item := range pinned { for _, item := range pinned {
if item == itemID { if item == itemID {
println(item, "already pinned") log.Warn(item, "already pinned")
return return
} }
} }
pinned = append(pinned, itemID) pinned = append(pinned, itemID)
savePinned() savePinned()
println(itemID, "pinned") log.Infof("%s pinned", itemID)
} }
func unpinItem(itemID string) { func unpinItem(itemID string) {
if isIn(pinned, itemID) { if isIn(pinned, itemID) {
pinned = remove(pinned, itemID) pinned = remove(pinned, itemID)
savePinned() savePinned()
println(itemID, "unpinned") log.Infof("%s unpinned", itemID)
} }
} }
@@ -562,7 +511,7 @@ func savePinned() {
_, err := f.WriteString(line + "\n") _, err := f.WriteString(line + "\n")
if err != nil { if err != nil {
println("Error saving pinned", err) log.Errorf("Error saving pinned", err)
} }
} }
} }
@@ -612,7 +561,7 @@ func launch(command string, terminal bool) {
} }
msg := fmt.Sprintf("env vars: %s; command: '%s'; args: %s\n", envVars, elements[cmdIdx], elements[1+cmdIdx:]) msg := fmt.Sprintf("env vars: %s; command: '%s'; args: %s\n", envVars, elements[cmdIdx], elements[1+cmdIdx:])
println(msg) log.Info(msg)
go cmd.Run() go cmd.Run()
@@ -622,14 +571,25 @@ func launch(command string, terminal bool) {
}) })
} }
func open(filePath string) { func open(filePath string, xdgOpen bool) {
cmd := exec.Command(*fileManager, filePath) var cmd *exec.Cmd
if xdgOpen {
cmd = exec.Command("xdg-open", filePath)
// Look for possible custom file association
for key, element := range preferredApps {
r, err := regexp.Compile(key)
if err == nil && r.MatchString(filePath) {
cmd = exec.Command(fmt.Sprintf("%v", element), filePath)
break
}
}
} else {
cmd = exec.Command(*fileManager, filePath)
}
fmt.Printf("Executing: %s", cmd)
cmd.Start() cmd.Start()
glib.TimeoutAdd(uint(150), func() bool {
gtk.MainQuit() gtk.MainQuit()
return false
})
} }
// Returns map output name -> gdk.Monitor // Returns map output name -> gdk.Monitor

View File

@@ -146,7 +146,9 @@ func setUpCategoriesButtonBox() *gtk.EventBox {
w := b.GetAllocatedWidth() w := b.GetAllocatedWidth()
b.SetImagePosition(gtk.POS_TOP) b.SetImagePosition(gtk.POS_TOP)
b.SetSizeRequest(w, 0) b.SetSizeRequest(w, 0)
if fileSearchResultWrapper != nil {
fileSearchResultWrapper.Hide() fileSearchResultWrapper.Hide()
}
}) })
} }
} }
@@ -261,6 +263,7 @@ func flowBoxButton(entry desktopEntry) *gtk.Button {
} else if btnEvent.Button() == 3 { } else if btnEvent.Button() == 3 {
pinItem(ID) pinItem(ID)
pinnedFlowBox = setUpPinnedFlowBox() pinnedFlowBox = setUpPinnedFlowBox()
return true
} }
return false return false
}) })
@@ -349,6 +352,29 @@ func setUpSearchEntry() *gtk.SearchEntry {
fileSearchResultWrapper.Hide() fileSearchResultWrapper.Hide()
} }
} }
// focus 1st search result #17
var w *gtk.Widget
if appFlowBox != nil {
b := appFlowBox.GetChildAtIndex(0)
if b != nil {
button, err := b.GetChild()
if err == nil {
button.ToWidget().GrabFocus()
w = button.ToWidget()
}
}
}
if w == nil && fileSearchResultFlowBox != nil {
f := fileSearchResultFlowBox.GetChildAtIndex(0)
if f != nil {
//f.SetCanFocus(false)
button, err := f.GetChild()
if err == nil {
button.ToWidget().SetCanFocus(true)
button.ToWidget().GrabFocus()
}
}
}
} else { } else {
// clear search results // clear search results
appFlowBox = setUpAppsFlowBox(nil, "") appFlowBox = setUpAppsFlowBox(nil, "")
@@ -362,9 +388,9 @@ func setUpSearchEntry() *gtk.SearchEntry {
} }
} }
}) })
searchEntry.Connect("focus-in-event", func() { /*searchEntry.Connect("focus-in-event", func() {
searchEntry.SetText("") searchEntry.SetText("")
}) })*/
return searchEntry return searchEntry
} }
@@ -387,7 +413,8 @@ func searchUserDir(dir string) {
} }
fileSearchResultFlowBox.Hide() fileSearchResultFlowBox.Hide()
statusLabel.SetText(fmt.Sprintf("%v results", fileSearchResultFlowBox.GetChildren().Length())) statusLabel.SetText(fmt.Sprintf("%v results | LMB: xdg-open | RMB: file manager",
fileSearchResultFlowBox.GetChildren().Length()))
num := uint(fileSearchResultFlowBox.GetChildren().Length() / *fsColumns) num := uint(fileSearchResultFlowBox.GetChildren().Length() / *fsColumns)
fileSearchResultFlowBox.SetMinChildrenPerLine(num + 1) fileSearchResultFlowBox.SetMinChildrenPerLine(num + 1)
fileSearchResultFlowBox.SetMaxChildrenPerLine(num + 1) fileSearchResultFlowBox.SetMaxChildrenPerLine(num + 1)
@@ -406,6 +433,7 @@ func setUpUserDirButton(iconName, displayName, entryName string, userDirsMap map
} }
box, _ := gtk.BoxNew(gtk.ORIENTATION_HORIZONTAL, 0) box, _ := gtk.BoxNew(gtk.ORIENTATION_HORIZONTAL, 0)
button, _ := gtk.ButtonNew() button, _ := gtk.ButtonNew()
button.SetAlwaysShowImage(true)
img, _ := gtk.ImageNewFromIconName(iconName, gtk.ICON_SIZE_MENU) img, _ := gtk.ImageNewFromIconName(iconName, gtk.ICON_SIZE_MENU)
button.SetImage(img) button.SetImage(img)
@@ -414,8 +442,16 @@ func setUpUserDirButton(iconName, displayName, entryName string, userDirsMap map
} }
button.SetLabel(displayName) button.SetLabel(displayName)
button.Connect("clicked", func() { button.Connect("button-release-event", func(btn *gtk.Button, e *gdk.Event) bool {
launch(fmt.Sprintf("%s %s", *fileManager, userDirsMap[entryName]), false) btnEvent := gdk.EventButtonNewFromEvent(e)
if btnEvent.Button() == 1 {
open(userDirsMap[entryName], true)
return true
} else if btnEvent.Button() == 3 {
open(userDirsMap[entryName], false)
return true
}
return false
}) })
box.PackStart(button, false, true, 0) box.PackStart(button, false, true, 0)
@@ -430,6 +466,7 @@ func setUpUserFileSearchResultButton(fileName, filePath string) *gtk.Box {
if strings.HasPrefix(filePath, "#is_dir#") { if strings.HasPrefix(filePath, "#is_dir#") {
filePath = filePath[8:] filePath = filePath[8:]
img, _ := gtk.ImageNewFromIconName("folder", gtk.ICON_SIZE_MENU) img, _ := gtk.ImageNewFromIconName("folder", gtk.ICON_SIZE_MENU)
button.SetAlwaysShowImage(true)
button.SetImage(img) button.SetImage(img)
} }
@@ -443,8 +480,20 @@ func setUpUserFileSearchResultButton(fileName, filePath string) *gtk.Box {
button.SetTooltipText(tooltipText) button.SetTooltipText(tooltipText)
} }
button.Connect("clicked", func() { button.Connect("button-release-event", func(btn *gtk.Button, e *gdk.Event) bool {
open(filePath) btnEvent := gdk.EventButtonNewFromEvent(e)
if btnEvent.Button() == 1 {
open(filePath, true)
return true
} else if btnEvent.Button() == 3 {
open(filePath, false)
return true
}
return false
})
button.Connect("activate", func() {
open(filePath, true)
}) })
box.PackStart(button, false, true, 0) box.PackStart(button, false, true, 0)

78
xdgdesktop_parser.go Normal file
View File

@@ -0,0 +1,78 @@
package main
import (
"bufio"
"fmt"
"io"
"os"
"strconv"
"strings"
)
func parseDesktopEntryFile(id string, path string) (e desktopEntry, err error) {
o, err := os.Open(path)
if err != nil {
return e, err
}
defer o.Close()
return parseDesktopEntry(id, o)
}
func parseDesktopEntry(id string, in io.Reader) (entry desktopEntry, err error) {
cleanexec := strings.NewReplacer("\"", "", "'", "")
entry.DesktopID = id
localizedName := fmt.Sprintf("Name[%s]", strings.Split(*lang, "_")[0])
localizedComment := fmt.Sprintf("Comment[%s]", strings.Split(*lang, "_")[0])
scanner := bufio.NewScanner(in)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
l := scanner.Text()
if strings.HasPrefix(l, "[") && l != "[Desktop Entry]" {
break
}
name, value := parseKeypair(l)
if value == "" {
continue
}
switch name {
case "Name":
entry.Name = value
case localizedName:
entry.NameLoc = value
case "Comment":
entry.Comment = value
case localizedComment:
entry.CommentLoc = value
case "Icon":
entry.Icon = value
case "Categories":
entry.Category = value
case "Terminal":
entry.Terminal, _ = strconv.ParseBool(value)
case "NoDisplay":
entry.NoDisplay, _ = strconv.ParseBool(value)
case "Exec":
entry.Exec = cleanexec.Replace(value)
}
}
// if name[ln] not found, let's try to find name[ln_LN]
if entry.NameLoc == "" {
entry.NameLoc = entry.Name
}
if entry.CommentLoc == "" {
entry.CommentLoc = entry.Comment
}
return entry, err
}
func parseKeypair(s string) (string, string) {
if idx := strings.IndexRune(s, '='); idx > 0 {
return strings.TrimSpace(s[:idx]), strings.TrimSpace(s[idx+1:])
}
return s, ""
}

53
xdgdesktop_parser_test.go Normal file
View File

@@ -0,0 +1,53 @@
package main
import (
"strings"
"testing"
)
var result desktopEntry
func BenchmarkDesktopEntryParser(b *testing.B) {
var entry desktopEntry
for n := 0; n < b.N; n++ {
entry, _ = parseDesktopEntryFile("id", "./desktop-directories/game.directory")
}
result = entry
}
func TestWhitespaceHandling(t *testing.T) {
const whitespace = `[Desktop Entry]
Categories = Debugger; Development; Git; IDE; Programming; TextEditor;
Comment = Editor for building and debugging modern web and cloud applications
Exec = bash -c "code-insiders ~/Workspaces/Linux/Flutter.code-workspace"
GenericName = Text Editor
Icon = vscode-flutter
Keywords = editor; IDE; plaintext; text; write;
MimeType = application/x-shellscript; inode/directory; text/english; text/plain; text/x-c; text/x-c++; text/x-c++hdr; text/x-c++src; text/x-chdr; text/x-csrc; text/x-java; text/x-makefile; text/x-moc; text/x-pascal; text/x-tcl; text/x-tex;
Name = VSCode Insiders with Flutter
Name[pt] = VSCode Insiders com Flutter
StartupNotify = true
StartupWMClass = code - insiders
Terminal = false
NoDisplay = false
Type = Application
Version = 1.0`
*lang = "pt"
entry, err := parseDesktopEntry("id", strings.NewReader(whitespace))
if err != nil {
t.Fatal(err)
}
if entry.Name != "VSCode Insiders with Flutter" {
t.Error("failed to parse desktop entry name")
}
if entry.NameLoc != "VSCode Insiders com Flutter" {
t.Error("failed to parse localized name")
}
if entry.NoDisplay {
t.Error("failed to parse desktop entry no display")
}
}