107 Commits

Author SHA1 Message Date
piotr
67b16ea6a6 bump to 0.4.0 2023-11-08 02:30:01 +01:00
piotr
3a02b16761 update versions 2023-11-08 02:25:43 +01:00
piotr
c67fef6671 fix line formatting 2023-11-08 02:24:41 +01:00
Piotr Miller
444617865e Merge pull request #102 from gouvinb/main
add support to hyprlandctl
2023-11-08 01:01:10 +01:00
gouvinb
e3de9cc67b make better log with -wm flag 2023-11-07 10:32:23 +01:00
gouvinb
1c11329581 Reworked command building.
Typical shell-supported command sets:

- `cmd`
- `cmd --arg`
- `cmd --arg=value`
- `ENV_VAR="value" cmd`
- `ENV_VAR="value" cmd --arg`
- `ENV_VAR="value" cmd --arg=value`

The listed commands above are now better managed.

Initially, the command types `cmd --arg=value` or `ENV_VAR="value" cmd --arg=value` were not correctly parsed. After finding a solution to this problem, I conducted further tests and discovered cases that were not currently handled. These cases are as follows:

- `cmd "arg with space"`
- `ENV_VAR="value with space" cmd`
2023-11-06 15:35:42 +01:00
gouvinb
87ee23afe2 print the correct executed command (prefix supported) 2023-11-06 12:40:10 +01:00
gouvinb
808fd032a0 fix prefix command with hypr or hyprland and update -wm description 2023-11-06 00:05:16 +01:00
gouvinb
1d3f023dc8 add support to hyprlandctl
BREAKING CHANGE: '-swaymsg' removed, use '-wm sway' instead
2023-11-05 23:36:18 +01:00
Piotr Miller
ccfe7776c6 Merge pull request #101 from gouvinb/main
Add the `swaymsg` flag to launch a program with `swaymsg exec` and prevent potential involuntary closing with Jetbrains IDEs
2023-11-05 14:03:26 +01:00
gouvinb
5a7dbefeb0 adds the "swaymsg" flag to launch a program with "swaymsg exec" and prevent potential involuntary closing with Jetbrains IDEs 2023-11-01 00:44:50 +01:00
piotr
fd26bb39e8 adjust help descriptions 2023-07-28 01:18:08 +02:00
piotr
bc74ebc927 update usage 2023-07-28 01:15:31 +02:00
piotr
d65d075bdb update usage 2023-07-28 01:13:57 +02:00
Piotr Miller
58ea8ea8db Merge pull request #95 from trinitronx/add-keyboard-mode-flag
Add keyboard mode flag to support both: "on-demand" & "exclusive" GTK layer-shell modes
2023-07-28 00:51:52 +02:00
Piotr Miller
94b8b8d7f0 Merge pull request #96 from nwg-piotr/theme
Force GTK_THEME for libadwaita apps
2023-07-28 00:49:47 +02:00
piotr
ebfaba1881 undo version bump 2023-07-28 00:47:27 +02:00
piotr
6656568c6a update -ft description 2023-07-27 03:44:29 +02:00
piotr
b3fa492b44 bump to 0.3.10 2023-07-27 03:27:56 +02:00
piotr
c0faecdb8b add -ft argument 2023-07-27 03:27:27 +02:00
James Cuzella
830edefa22 Refactor -k flag as Bool to toggle keyboard mode on-demand / exclusive 2023-07-24 20:48:35 -06:00
James Cuzella
aeeb4e4890 Fix warn messages for short -k flag 2023-07-23 22:49:52 -06:00
James Cuzella
6bbfbea1a8 Handle empty string passed to --keyboard/-k flag 2023-07-23 22:45:16 -06:00
James Cuzella
d88d9795d1 Document --keyboard/-k flag in README 2023-07-23 22:22:41 -06:00
James Cuzella
f85356bdc7 Drop keyboard mode: 'none' b/c it doesn't make sense with this app to not have keyboard input at all 2023-07-23 20:30:14 -06:00
James Cuzella
37c9c2d520 Add --keyboard/-k flag to set GTK layer shell keyboard mode (default: exclusive) 2023-07-23 20:26:09 -06:00
Piotr Miller
2903abd831 Merge pull request #94 from 6543-forks/smal_code_refactor
smal code refactors
2023-07-07 22:48:39 +02:00
Piotr Miller
9cc81fa38a Merge pull request #93 from 6543-forks/fix_lint_errors
log actual error message too
2023-07-07 22:45:03 +02:00
Piotr Miller
edf4c81f85 Merge pull request #92 from 6543-forks/reduce_syscalls
Save result of os.Getenv()
2023-07-07 22:43:50 +02:00
Piotr Miller
9582726cb9 Merge pull request #90 from 6543-forks/update_go_deps
update golang lib dependencies
2023-07-07 22:26:53 +02:00
6543
7a6ae82a75 log the unknown signal 2023-07-06 18:03:28 +02:00
6543
22ddb71603 declare unused var with _ and always name event the same 2023-07-06 17:56:40 +02:00
6543
ea7813761a gitignore vendor directory 2023-07-06 17:16:26 +02:00
6543
0cafb3c3ad save result of os.Getenv and use that if possible to reduce some syscalls 2023-07-06 17:15:49 +02:00
6543
89cffad81d log actual error message too 2023-07-06 17:14:55 +02:00
6543
1e854558cf update golang lib dependencies 2023-07-06 16:10:31 +02:00
Piotr Miller
3df7a30533 Update README.md 2023-03-11 02:10:14 +01:00
piotr
e383f7a470 bump to 0.3.9 (go 1.20) 2023-02-06 01:57:11 +01:00
Piotr Miller
820848d984 Merge pull request #85 from nwg-piotr/add84
Add `-i` flag to force an icon theme
2023-01-22 22:41:42 +01:00
piotr
fa01d5bb32 bump to 0.3.8 2023-01-22 22:35:00 +01:00
piotr
8be1fc1ea2 allow forcing icon theme #84 2023-01-22 22:30:09 +01:00
Piotr Miller
0e5d38907f Merge pull request #83 from nwg-piotr/fix82
fixed trimming strings on non-ASCII characters
2023-01-12 22:50:59 +01:00
piotr
7ba7abd817 bump to 0.3.7 2023-01-12 22:48:52 +01:00
piotr
cedcf8619f fix cutting utf-8 strings #82 2023-01-12 02:27:27 +01:00
Piotr Miller
8ff2d5c89c Create FUNDING.yml 2022-12-15 12:05:39 +01:00
Piotr Miller
c4629e0c28 Merge pull request #80 from nwg-piotr/lockfile
Move lock file to XDG_DATA_HOME/nwg-drawer/
2022-12-11 23:21:32 +01:00
piotr
ab5e2ea6ae move lock file to dataDir #79 2022-12-11 23:05:57 +01:00
piotr
63117e2605 bump to 0.3.6 2022-12-11 21:44:01 +01:00
Piotr Miller
d22609f403 Merge pull request #77 from nwg-piotr/debu68
fix #68
2022-12-01 00:12:46 +01:00
piotr
cb6a7f44fe go-sway -> 1.2.0 2022-12-01 00:03:17 +01:00
piotr
85cc2d78ee fix userDirsFile path #68 2022-11-28 23:39:36 +01:00
piotr
d3a0fd04a4 log if userDirsFile found #68 2022-11-28 23:26:19 +01:00
piotr
52e63667c9 print userDirsMap -> log.Debugf #68 2022-11-28 23:03:12 +01:00
piotr
f399f589fb add a line to debug #68 2022-11-18 01:40:50 +01:00
Piotr Miller
2058f124fd Merge pull request #75 from nwg-piotr/fix74
sanitize no longer valid pinned items #74
2022-11-12 15:40:14 +01:00
piotr
ebdbe71ac6 hide and delete invalid pinned items #74 2022-11-12 03:41:24 +01:00
Piotr Miller
3fe6d71234 Merge pull request #73 from nwg-piotr/fix-72
Fix inconsistent behaviour of normal resident instance
2022-10-30 23:54:41 +01:00
piotr
0e5b1cf659 bump to 0.3.3 2022-10-30 22:58:19 +01:00
piotr
6494557979 focus 1st pinned if any, or focus 1st appBox entry; fixes #72 2022-10-30 22:53:06 +01:00
Piotr Miller
f68678f687 Merge pull request #71 from nwg-piotr/fix-45-revisited
Fix 45 revisited
2022-10-26 02:13:02 +02:00
piotr
e31df6e721 bump to 0.3.2 2022-10-25 04:06:30 +02:00
piotr
9509a65625 attempt to fix #45 (revisited) 2022-10-25 04:02:05 +02:00
Piotr Miller
bc84a05378 Merge pull request #70 from nwg-piotr/check69
Fix #69
2022-09-25 20:24:41 +02:00
piotr
a249c37612 del redundant type conversion 2022-09-21 00:56:09 +02:00
piotr
c7fa6e836a fix log line formatting 2022-09-21 00:54:42 +02:00
piotr
87a62e7bb6 handle error closing file 2022-09-21 00:53:48 +02:00
piotr
11241cef0b replace deprecated types 2022-09-21 00:48:28 +02:00
piotr
87d2a74db0 fix setting data dir #69 2022-09-21 00:32:29 +02:00
piotr
41f7d58edf debug #69 2022-09-20 23:01:35 +02:00
piotr
583ddfde0b update .gitignore 2022-06-23 00:16:56 +02:00
Piotr Miller
383869906a Merge pull request #65 from SPFabGerman/OnlyShowIn
Added support for Hidden, OnlyShowIn and NotShowIn
2022-06-11 00:19:45 +02:00
SPFabGerman
54553ffae1 Added support for Hidden, OnlyShowIn and NotShowIn 2022-06-09 21:32:38 +02:00
Piotr Miller
d8e7e0f6c2 Merge pull request #63 from nwg-piotr/fix62
Fixes accidental crash on trimming long descriptions
2022-05-16 22:27:54 +02:00
piotr
7230d82be3 fix trimming names as well #62 2022-05-16 01:16:02 +02:00
piotr
02b3930ef9 trim string, not runes #62 2022-05-16 01:01:25 +02:00
piotr
f3b3635bd8 trim comments > 120 chars #61 2022-05-08 01:43:16 +02:00
nwg-piotr
bb84a2fc46 remove accidental binary 2022-04-13 11:36:19 +02:00
Piotr Miller
93587a47db Merge pull request #60 from nwg-piotr/dark
prefer dark theme
2022-04-13 11:22:15 +02:00
nwg-piotr
afe99b42f0 prefer dark theme 2022-04-13 11:14:50 +02:00
Piotr Miller
e052a86b12 Merge pull request #58 from jovanlanik/main
Support for changing GTK theme
2022-03-18 02:06:06 +01:00
Jovan Lanik
45cb286296 Add support for changing GTK theme 2022-03-13 21:58:42 +01:00
piotr
d050598086 bump to 0.2.8 2022-02-15 01:19:30 +01:00
Piotr Miller
42ddd815f4 Merge pull request #52 from snehrbass/main
Add flags to set window margins
2022-02-15 01:02:48 +01:00
Ste_ph_en
8b1dfb4421 Add flags to set margins 2022-02-14 18:57:57 -05:00
Piotr Miller
d572c1125d Merge pull request #54 from Raffy23/51-fix-zombie-processes
Collect signals of child processes
2022-02-15 00:42:56 +01:00
Piotr Miller
6b055bb48a Merge pull request #53 from Raffy23/50-fix-high-cpu-load
Fix high CPU load
2022-02-15 00:37:54 +01:00
Raphael Ludwig
165b925da2 Fix regression that prevented pinned items to update while window is open 2022-02-14 20:59:48 +01:00
Raphael Ludwig
058f8684aa Use a channel and a goroutine instead of busy polling a boolean flag to reduce cpu consumption 2022-02-13 23:21:21 +01:00
Raphael Ludwig
f4b51b0706 Collect signals of child processes
Add logging if starting of a process was not successful
2022-02-13 23:06:21 +01:00
Piotr Miller
b781f58097 Update README.md 2022-02-04 11:45:17 +01:00
Piotr Miller
68137935a9 Update README.md 2022-02-02 14:47:28 +01:00
piotr
98efb36614 bump version 2022-01-15 23:58:23 +01:00
Piotr Miller
59533536b1 Merge pull request #48 from nightly-brew/detach-from-process-group
Use Setsid when spawning programs
2022-01-15 03:58:52 +01:00
nightly-brew
07821f39b7 Use Setsid when starting the selected program to make sure it's not killed by signals sent to the drawer's process group. 2022-01-15 02:36:07 +01:00
piotr
4ea160e524 update README 2022-01-13 23:58:10 +01:00
Piotr Miller
8fb4209a97 Merge pull request #47 from nwg-piotr/not-wayland
remove bin
2022-01-13 23:46:57 +01:00
piotr
c85c364ba5 ignore bin 2022-01-13 23:42:42 +01:00
piotr
8239254485 remove bin #40 2022-01-13 23:42:31 +01:00
Piotr Miller
7681055a23 Merge pull request #46 from nwg-piotr/fix45
check if fileSearchResultWrapper exists #45
2022-01-12 11:17:56 +01:00
piotr
15c9029935 check if fileSearchResultWrapper exists #45 2022-01-12 11:06:05 +01:00
Piotr Miller
f67f1a9950 Merge pull request #43 from nwg-piotr/fix-pinned
create empty pinned file if not found on startup (and some other minor issues)
2022-01-11 14:41:39 +01:00
piotr
2c63e256ba update readme 2022-01-11 14:36:56 +01:00
piotr
92095d5b97 hide fileSearchResultWrapper initially 2022-01-11 12:31:47 +01:00
piotr
a6977d9444 create pinned file if not found 2022-01-11 11:45:05 +01:00
Piotr Miller
378aa33cf3 Merge pull request #42 from nwg-piotr/issue36
Close on RMB only #36
2022-01-10 23:55:16 +01:00
piotr
9d6d572f72 close on LMB only #36 2022-01-10 22:04:58 +01:00
piotr
8fdb174643 hot fix to #41 2022-01-09 23:20:48 +01:00
11 changed files with 436 additions and 175 deletions

1
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1 @@
github: nwg-piotr

8
.gitignore vendored
View File

@@ -11,5 +11,11 @@
# Output of the go coverage tool, specifically when used with LiteIDE # Output of the go coverage tool, specifically when used with LiteIDE
*.out *.out
# Binaries built with make
bin
nwg-drawer
/.idea
# Dependency directories (remove the comment below to include it) # Dependency directories (remove the comment below to include it)
# vendor/ vendor/

View File

@@ -1,6 +1,8 @@
# nwg-drawer # nwg-drawer
This application is a part of the [nwg-shell](https://github.com/nwg-piotr/nwg-shell) project. This application is a part of the [nwg-shell](https://nwg-piotr.github.io/nwg-shell) project.
**Contributing:** please read the [general contributing rules for the nwg-shell project](https://nwg-piotr.github.io/nwg-shell/contribution).
Nwg-drawer is a golang replacement to the `nwggrid` command Nwg-drawer is a golang replacement to the `nwggrid` command
(a part of [nwg-launchers](https://github.com/nwg-piotr/nwg-launchers)). It's being developed with (a part of [nwg-launchers](https://github.com/nwg-piotr/nwg-launchers)). It's being developed with
@@ -15,22 +17,24 @@ You may pin applications by right-clicking them. Pinned items will appear above
a pinned item to unpin it. The pinned items cache is shared with [nwg-menu](https://github.com/nwg-piotr/nwg-menu) a pinned item to unpin it. The pinned items cache is shared with [nwg-menu](https://github.com/nwg-piotr/nwg-menu)
and `nwggrid`. and `nwggrid`.
![screenshot-01.png](https://scrot.cloud/images/2021/05/30/screenshot-01.png) ![screenshot.png](https://raw.githubusercontent.com/nwg-piotr/nwg-shell-resources/master/images/nwg-shell/nwg-drawer.png)
[more screenshots](https://scrot.cloud/album/nwg-drawer.Bogd) | [see on YouTube](https://youtu.be/iIgxJQhCQf0) [see on YouTube](https://youtu.be/iIgxJQhCQf0)
[![Packaging status](https://repology.org/badge/vertical-allrepos/nwg-drawer.svg)](https://repology.org/project/nwg-drawer/versions) [![Packaging status](https://repology.org/badge/vertical-allrepos/nwg-drawer.svg)](https://repology.org/project/nwg-drawer/versions)
To close the window w/o running a program, you may use `Esc` key, or right-click the window next to the icons.
## v0.2.x note ## v0.2.x note
1. Placing config files in the nwg-panel config directory was a mistake, sorry. The 0.2.0 version migrates them to `~/.config/nwg-drawer`. 1. Placing config files in the nwg-panel config directory was a mistake, sorry. The 0.2.0 version migrates them to `~/.config/nwg-drawer`.
2. From now on you may run the program residently, which should speed it up. See "Running" below. 2. From now on you may run the program residently, which should speed it up (but also occupy some resources!). See "Running" below.
## Installation ## Installation
### Dependencies ### Dependencies
- go >=1.16 (just to build) - go >=1.20 (just to build)
- gtk3 - gtk3
- gtk-layer-shell - gtk-layer-shell
- xdg-utils - xdg-utils
@@ -38,7 +42,7 @@ and `nwggrid`.
Optional (recommended): Optional (recommended):
- thunar - thunar
- alacritty - foot
You may use another file manager and terminal emulator (see command line arguments), but mentioned above have been You may use another file manager and terminal emulator (see command line arguments), but mentioned above have been
confirmed to work well with the program. Also see **Files** below. confirmed to work well with the program. Also see **Files** below.
@@ -50,9 +54,6 @@ confirmed to work well with the program. Also see **Files** below.
3. `make build` 3. `make build`
4. `sudo make install` 4. `sudo make install`
Building the gotk3 library takes quite a lot of time. If your machine is x86_64, you may skip steps 2-3, and
install the provided binary by executing step 4.
## Command line arguments ## Command line arguments
```text ```text
@@ -67,10 +68,25 @@ Usage of nwg-drawer:
File Search result COLumns (default 2) File Search result COLumns (default 2)
-fslen int -fslen int
File Search name LENgth Limit (default 80) File Search name LENgth Limit (default 80)
-ft
Force Theme for libadwaita apps, by adding 'GTK_THEME=<default-gtk-theme>' env var
-g string
GTK theme name
-i string
GTK icon theme name
-is int -is int
Icon Size (default 64) Icon Size (default 64)
-k set GTK layer shell Keyboard interactivity to 'on-demand' mode
-lang string -lang string
force lang, e.g. "en", "pl" force lang, e.g. "en", "pl"
-mb int
Margin Bottom
-ml int
Margin Left
-mr int
Margin Right
-mt int
Margin Top
-nocats -nocats
Disable filtering by category Disable filtering by category
-nofs -nofs
@@ -85,7 +101,7 @@ Usage of nwg-drawer:
-spacing uint -spacing uint
icon spacing (default 20) icon spacing (default 20)
-term string -term string
Terminal emulator (default "alacritty") Terminal emulator (default "foot")
-v display Version information -v display Version information
``` ```
@@ -133,7 +149,7 @@ Edit `~/.config/nwg-drawer/drawer.css` to your taste.
When the search phrase is at least 3 characters long, your XDG user directories are being searched. 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) ![screenshot-02.png](https://raw.githubusercontent.com/nwg-piotr/nwg-shell/main/images/nwg-drawer/search.png)
Use the **left mouse button** to open a file with the `xdg-open` command. As configuring file associations for it is 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. PITA, you may override them, by creating the `~/.config/nwg-panel/preferred-apps.json` file with your own definitions.
@@ -180,4 +196,4 @@ Copyright (c) 2015-2018 gotk3 contributors
- [go-sway](https://github.com/joshuarubin/go-sway) Copyright (c) 2019 Joshua Rubin - [go-sway](https://github.com/joshuarubin/go-sway) Copyright (c) 2019 Joshua Rubin
- [go-singleinstance](github.com/allan-simon/go-singleinstance) Copyright (c) 2015 Allan Simon - [go-singleinstance](github.com/allan-simon/go-singleinstance) Copyright (c) 2015 Allan Simon
- [logrus](https://github.com/sirupsen/logrus) Copyright (c) 2014 Simon Eskildsen - [logrus](https://github.com/sirupsen/logrus) Copyright (c) 2014 Simon Eskildsen
- [fsnotify](https://github.com/fsnotify/fsnotify) Copyright (c) 2012-2019 fsnotify Authors - [fsnotify](https://github.com/fsnotify/fsnotify) Copyright (c) 2012-2019 fsnotify Authors

Binary file not shown.

19
go.mod
View File

@@ -1,12 +1,19 @@
module github.com/nwg-piotr/nwg-drawer module github.com/nwg-piotr/nwg-drawer
go 1.16 go 1.21
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-20230802002603-b0c42cd8474f
github.com/fsnotify/fsnotify v1.5.1 github.com/fsnotify/fsnotify v1.7.0
github.com/gotk3/gotk3 v0.6.1 github.com/gotk3/gotk3 v0.6.2
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.3
)
require (
github.com/joshuarubin/lifecycle v1.1.4 // indirect
go.uber.org/multierr v1.11.0 // indirect
golang.org/x/sync v0.5.0 // indirect
golang.org/x/sys v0.14.0 // indirect
) )

51
go.sum
View File

@@ -3,30 +3,45 @@ github.com/allan-simon/go-singleinstance v0.0.0-20210120080615-d0997106ab37/go.m
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 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/dlasky/gotk3-layershell v0.0.0-20210827021656-e6ecab2731f7 h1:LDo0kwt+oW9a4lWlj2OakIgMW1ySXXVRGFt8GHUdYYA= github.com/dlasky/gotk3-layershell v0.0.0-20221218201547-1f6674a3f872 h1:16qcNl+UgbvudN7wPv+zq4mmDSYJWdLv5jbVhS7+OVI=
github.com/dlasky/gotk3-layershell v0.0.0-20210827021656-e6ecab2731f7/go.mod h1:JHLx2Wz4mAPVwn4PFhC69ydwyHP4A3wQvlg7HKVVc1U= github.com/dlasky/gotk3-layershell v0.0.0-20221218201547-1f6674a3f872/go.mod h1:JHLx2Wz4mAPVwn4PFhC69ydwyHP4A3wQvlg7HKVVc1U=
github.com/fsnotify/fsnotify v1.5.1 h1:mZcQUHVQUQWoPXXtuf9yuEXKudkV2sx1E06UadKWpgI= github.com/dlasky/gotk3-layershell v0.0.0-20230802002603-b0c42cd8474f h1:qDnUQAD7tVX/gnL6uSgouzfGNA4xXH+B/fd6Ko19GgM=
github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= github.com/dlasky/gotk3-layershell v0.0.0-20230802002603-b0c42cd8474f/go.mod h1:JHLx2Wz4mAPVwn4PFhC69ydwyHP4A3wQvlg7HKVVc1U=
github.com/gotk3/gotk3 v0.6.1 h1:GJ400a0ecEEWrzjBvzBzH+pB/esEMIGdB9zPSmBdoeo= github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY=
github.com/fsnotify/fsnotify v1.6.0/go.mod h1:sl3t1tCWJFWoRz9R8WJCbQihKKwmorjAbSClcnxKAGw=
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
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/gotk3/gotk3 v0.6.2 h1:sx/PjaKfKULJPTPq8p2kn2ZbcNFxpOJqi4VLzMbEOO8=
github.com/joshuarubin/go-sway v0.0.4/go.mod h1:qcDd6f25vJ0++wICwA1BainIcRC67p2Mb4lsrZ0k3/k= github.com/gotk3/gotk3 v0.6.2/go.mod h1:/hqFpkNa9T3JgNAE2fLvCdov7c5bw//FHNZrZ3Uv9/Q=
github.com/joshuarubin/lifecycle v1.0.0 h1:N/lPEC8f+dBZ1Tn99vShqp36LwB+LI7XNAiNadZeLUQ= github.com/joshuarubin/go-sway v1.2.0 h1:t3eqW504//uj9PDwFf0+IVfkD+WoOGaDX5gYIe0BHyM=
github.com/joshuarubin/go-sway v1.2.0/go.mod h1:qcDd6f25vJ0++wICwA1BainIcRC67p2Mb4lsrZ0k3/k=
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/joshuarubin/lifecycle v1.1.4 h1:9ZjvYSsWax9DC3Jpz6vGf/0KnU8FNMjh0/vJ3SpSBRQ=
github.com/joshuarubin/lifecycle v1.1.4/go.mod h1:QqHrqwMPMA9dbJY3XgIyVLhzHMSGOFrcCAQ59bke1mo=
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.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.9.3/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=
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4= 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/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/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= go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
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/sync v0.3.0 h1:ftCYgMx6zT/asHUrPw8BLLscYtGznsLAnjq5RH9P66E=
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c h1:F1jZWGFhYfh0Ci55sIpILtKKK8p3i2/krTr0H1rg74I= golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y=
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sync v0.5.0 h1:60k92dhOjHxJkrqnwsfl8KuaHbn/5dl0lUPUklKo3qE=
golang.org/x/sync v0.5.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220908164124-27713097b956/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.10.0 h1:SqMFp9UcQJZa+pmYuAKjd9xq1f0j5rLcDIk0mj4qAsA=
golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.14.0 h1:Vz7Qs629MkJkGyHxUlRHizWJRG2j8fbQKjELVSNhy7Q=
golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=

201
main.go
View File

@@ -21,7 +21,7 @@ import (
"github.com/gotk3/gotk3/gtk" "github.com/gotk3/gotk3/gtk"
) )
const version = "0.2.3" const version = "0.4.0"
var ( var (
appDirs []string appDirs []string
@@ -100,12 +100,11 @@ var (
statusLabel *gtk.Label statusLabel *gtk.Label
status string status string
ignore string ignore string
showWindowTrigger bool
desktopTrigger bool desktopTrigger bool
pinnedTrigger bool pinnedItemsChanged chan interface{} = make(chan interface{}, 1)
) )
func defaultStringIfBlank(s, fallback string) string { func defaultTermIfBlank(s, fallback string) string {
s = strings.TrimSpace(s) s = strings.TrimSpace(s)
// os.Getenv("TERM") returns "linux" instead of empty string, if program has been started // 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. // from a key binding defined in the config file. See #23.
@@ -115,18 +114,42 @@ func defaultStringIfBlank(s, fallback string) string {
return s return s
} }
func defaultStringIfBlank(s, fallback string) string {
s = strings.TrimSpace(s)
if s == "" {
return fallback
}
return s
}
func validateWm() {
if !(*wm == "sway" || *wm == "hyprland") && *wm != "" {
*wm = ""
log.Warn("-wm argument supports only sway or hyprland string.")
}
}
// 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)")
var displayVersion = flag.Bool("v", false, "display Version information") var displayVersion = flag.Bool("v", false, "display Version information")
var keyboard = flag.Bool("k", false, "set GTK layer shell Keyboard interactivity to 'on-demand' mode")
var overlay = flag.Bool("ovl", false, "use OVerLay layer") var overlay = flag.Bool("ovl", false, "use OVerLay layer")
var gtkTheme = flag.String("g", "", "GTK theme name")
var gtkIconTheme = flag.String("i", "", "GTK icon theme name")
var iconSize = flag.Int("is", 64, "Icon Size") var iconSize = flag.Int("is", 64, "Icon Size")
var marginTop = flag.Int("mt", 0, "Margin Top")
var marginLeft = flag.Int("ml", 0, "Margin Left")
var marginRight = flag.Int("mr", 0, "Margin Right")
var marginBottom = flag.Int("mb", 0, "Margin Bottom")
var fsColumns = flag.Uint("fscol", 2, "File Search result COLumns") var fsColumns = flag.Uint("fscol", 2, "File Search result COLumns")
var forceTheme = flag.Bool("ft", false, "Force Theme for libadwaita apps, by adding 'GTK_THEME=<default-gtk-theme>' env var")
var columnsNumber = flag.Uint("c", 6, "number of Columns") 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", defaultStringIfBlank(os.Getenv("TERM"), "foot"), "Terminal emulator") var term = flag.String("term", defaultTermIfBlank(os.Getenv("TERM"), "foot"), "Terminal emulator")
var wm = flag.String("wm", defaultStringIfBlank(os.Getenv("XDG_CURRENT_DESKTOP"), ""), "Use swaymsg (with 'sway' argument) or hyprctl (with 'hyprland')")
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")
@@ -146,8 +169,11 @@ func main() {
os.Exit(0) os.Exit(0)
} }
validateWm()
// Gentle SIGTERM handler thanks to reiki4040 https://gist.github.com/reiki4040/be3705f307d3cd136e85 // Gentle SIGTERM handler thanks to reiki4040 https://gist.github.com/reiki4040/be3705f307d3cd136e85
// v0.2: we also need to support SIGUSR from now on // v0.2: we also need to support SIGUSR from now on
showWindowChannel := make(chan interface{}, 1)
signalChan := make(chan os.Signal, 1) signalChan := make(chan os.Signal, 1)
signal.Notify(signalChan, syscall.SIGTERM, syscall.SIGUSR1) signal.Notify(signalChan, syscall.SIGTERM, syscall.SIGUSR1)
@@ -164,7 +190,7 @@ func main() {
// let's just set e helper variable here. We'll be checking it with glib.TimeoutAdd. // let's just set e helper variable here. We'll be checking it with glib.TimeoutAdd.
if !win.IsVisible() { if !win.IsVisible() {
log.Debug("SIGUSR1 received, showing the window") log.Debug("SIGUSR1 received, showing the window")
showWindowTrigger = true showWindowChannel <- struct{}{}
} else { } else {
log.Debug("SIGUSR1 received, hiding the window") log.Debug("SIGUSR1 received, hiding the window")
restoreStateAndHide() restoreStateAndHide()
@@ -174,7 +200,7 @@ func main() {
gtk.MainQuit() gtk.MainQuit()
} }
default: default:
log.Info("Unknown signal") log.Infof("Unknown signal: %s", s.String())
} }
} }
}() }()
@@ -184,7 +210,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)
@@ -195,7 +221,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
}
} }
} }
} }
@@ -238,7 +267,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()
@@ -251,6 +283,7 @@ func main() {
pinned, err = loadTextFile(pinnedFile) pinned, err = loadTextFile(pinnedFile)
if err != nil { if err != nil {
pinned = nil pinned = nil
savePinned()
} }
log.Info(fmt.Sprintf("Found %v pinned items", len(pinned))) log.Info(fmt.Sprintf("Found %v pinned items", len(pinned)))
@@ -297,6 +330,32 @@ func main() {
// USER INTERFACE // USER INTERFACE
gtk.Init(nil) gtk.Init(nil)
settings, _ := gtk.SettingsGetDefault()
if *gtkTheme != "" {
err = settings.SetProperty("gtk-theme-name", *gtkTheme)
if err != nil {
log.Error("Unable to set theme:", err)
} else {
log.Infof("User demanded theme: %s", *gtkTheme)
}
} else {
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")
}
if *gtkIconTheme != "" {
err = settings.SetProperty("gtk-icon-theme-name", *gtkIconTheme)
if err != nil {
log.Error("Unable to set icon theme:", err)
} else {
log.Infof("User demanded icon theme: %s", *gtkIconTheme)
}
}
cssProvider, _ := gtk.CssProviderNew() cssProvider, _ := gtk.CssProviderNew()
err = cssProvider.LoadFromPath(*cssFileName) err = cssProvider.LoadFromPath(*cssFileName)
@@ -342,17 +401,28 @@ func main() {
layershell.SetLayer(win, layershell.LAYER_SHELL_LAYER_TOP) layershell.SetLayer(win, layershell.LAYER_SHELL_LAYER_TOP)
} }
layershell.SetKeyboardMode(win, layershell.LAYER_SHELL_KEYBOARD_MODE_EXCLUSIVE) layershell.SetMargin(win, layershell.LAYER_SHELL_EDGE_TOP, *marginTop)
layershell.SetMargin(win, layershell.LAYER_SHELL_EDGE_LEFT, *marginLeft)
layershell.SetMargin(win, layershell.LAYER_SHELL_EDGE_RIGHT, *marginRight)
layershell.SetMargin(win, layershell.LAYER_SHELL_EDGE_BOTTOM, *marginBottom)
if *keyboard {
log.Info("Setting GTK layer shell keyboard mode to: on-demand")
layershell.SetKeyboardMode(win, layershell.LAYER_SHELL_KEYBOARD_MODE_ON_DEMAND)
} else {
log.Info("Setting GTK layer shell keyboard mode to default: exclusive")
layershell.SetKeyboardMode(win, layershell.LAYER_SHELL_KEYBOARD_MODE_EXCLUSIVE)
}
} }
win.Connect("destroy", func() { win.Connect("destroy", func() {
gtk.MainQuit() gtk.MainQuit()
}) })
win.Connect("key-release-event", func(window *gtk.Window, event *gdk.Event) bool { win.Connect("key-release-event", func(_ *gtk.Window, event *gdk.Event) bool {
key := &gdk.EventKey{Event: event} key := &gdk.EventKey{Event: event}
switch key.KeyVal() { if key.KeyVal() == gdk.KEY_Escape {
case gdk.KEY_Escape:
s, _ := searchEntry.GetText() s, _ := searchEntry.GetText()
if s != "" { if s != "" {
searchEntry.GrabFocus() searchEntry.GrabFocus()
@@ -365,15 +435,22 @@ func main() {
} }
} }
return true return true
}
return false
})
win.Connect("key-press-event", func(_ *gtk.Window, event *gdk.Event) bool {
key := &gdk.EventKey{Event: event}
switch key.KeyVal() {
case gdk.KEY_downarrow, gdk.KEY_Up, gdk.KEY_Down, gdk.KEY_Left, gdk.KEY_Right, gdk.KEY_Tab, case gdk.KEY_downarrow, gdk.KEY_Up, gdk.KEY_Down, gdk.KEY_Left, gdk.KEY_Right, gdk.KEY_Tab,
gdk.KEY_Return, gdk.KEY_Page_Up, gdk.KEY_Page_Down, gdk.KEY_Home, gdk.KEY_End: gdk.KEY_Return, gdk.KEY_Page_Up, gdk.KEY_Page_Down, gdk.KEY_Home, gdk.KEY_End:
return true return false
default: default:
if !searchEntry.IsFocus() { if !searchEntry.IsFocus() {
searchEntry.GrabFocusWithoutSelecting() searchEntry.GrabFocusWithoutSelecting()
} }
return true return false
} }
}) })
@@ -417,9 +494,9 @@ func main() {
resultWindow.SetEvents(int(gdk.ALL_EVENTS_MASK)) resultWindow.SetEvents(int(gdk.ALL_EVENTS_MASK))
resultWindow.SetPolicy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC) resultWindow.SetPolicy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
resultWindow.Connect("button-release-event", func(sw *gtk.ScrolledWindow, e *gdk.Event) bool { resultWindow.Connect("button-release-event", func(_ *gtk.ScrolledWindow, event *gdk.Event) bool {
btnEvent := gdk.EventButtonNewFromEvent(e) btnEvent := gdk.EventButtonNewFromEvent(event)
if btnEvent.Button() == 1 || btnEvent.Button() == 3 { if btnEvent.Button() == 3 {
if !*resident { if !*resident {
gtk.MainQuit() gtk.MainQuit()
} else { } else {
@@ -450,6 +527,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)
@@ -485,38 +563,55 @@ func main() {
log.Info(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()))
// Check if showing the window has been requested (SIGUSR1) // Check if showing the window has been requested (SIGUSR1)
glib.TimeoutAdd(uint(1), func() bool { go func() {
if showWindowTrigger && win != nil && !win.IsVisible() { for {
win.ShowAll() select {
// focus 1st element case <-showWindowChannel:
b := appFlowBox.GetChildAtIndex(0) log.Debug("Showing window")
if b != nil { glib.TimeoutAdd(0, func() bool {
button, err := b.GetChild() if win != nil && !win.IsVisible() {
if err == nil {
button.ToWidget().GrabFocus() // Refresh files before displaying the root window
} // some .desktop file changed
if desktopTrigger {
log.Debug(".desktop file changed")
desktopFiles = listDesktopFiles()
status = parseDesktopFiles(desktopFiles)
appFlowBox = setUpAppsFlowBox(nil, "")
desktopTrigger = false
}
// Show window and focus the search box
win.ShowAll()
if fileSearchResultWrapper != nil {
fileSearchResultWrapper.Hide()
}
// focus 1st element
var button gtk.IWidget
if pinnedFlowBox.GetChildren().Length() > 0 {
button, err = pinnedFlowBox.GetChildAtIndex(0).GetChild()
} else {
button, err = appFlowBox.GetChildAtIndex(0).GetChild()
}
if err == nil {
button.ToWidget().GrabFocus()
}
}
return false
})
case <-pinnedItemsChanged:
glib.TimeoutAdd(0, func() bool {
log.Debug("pinned file changed")
pinned, _ = loadTextFile(pinnedFile)
pinnedFlowBox = setUpPinnedFlowBox()
return false
})
} }
} }
showWindowTrigger = false }()
// some .desktop file changed
if desktopTrigger {
log.Debug(".desktop file changed")
desktopFiles = listDesktopFiles()
status = parseDesktopFiles(desktopFiles)
appFlowBox = setUpAppsFlowBox(nil, "")
desktopTrigger = false
}
// pinned file changed
if pinnedTrigger {
log.Debug("pinned file changed")
pinnedTrigger = false
pinned, _ = loadTextFile(pinnedFile)
pinnedFlowBox = setUpPinnedFlowBox()
}
return true
})
go watchFiles() go watchFiles()
@@ -525,7 +620,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("")
@@ -538,7 +635,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()))

233
tools.go
View File

@@ -7,7 +7,6 @@ import (
"fmt" "fmt"
"io" "io"
"io/fs" "io/fs"
"io/ioutil"
"os" "os"
"os/exec" "os/exec"
"path" "path"
@@ -15,6 +14,7 @@ import (
"regexp" "regexp"
"sort" "sort"
"strings" "strings"
"syscall"
"time" "time"
log "github.com/sirupsen/logrus" log "github.com/sirupsen/logrus"
@@ -78,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 {
@@ -104,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
@@ -120,29 +121,18 @@ func getUserDir(home, line string) string {
} }
func cacheDir() string { func cacheDir() string {
if os.Getenv("XDG_CACHE_HOME") != "" { if xdgCache := os.Getenv("XDG_CACHE_HOME"); xdgCache != "" {
return os.Getenv("XDG_CONFIG_HOME") return xdgCache
} }
if os.Getenv("HOME") != "" && pathExists(filepath.Join(os.Getenv("HOME"), ".cache")) { if home := os.Getenv("HOME"); home != "" && pathExists(filepath.Join(home, ".cache")) {
p := filepath.Join(os.Getenv("HOME"), ".cache") p := filepath.Join(home, ".cache")
return p return p
} }
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
} }
@@ -151,11 +141,11 @@ func readTextFile(path string) (string, error) {
} }
func oldConfigDir() (string, error) { func oldConfigDir() (string, error) {
if os.Getenv("XDG_CONFIG_HOME") != "" { if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); xdgConfig != "" {
dir := path.Join(os.Getenv("XDG_CONFIG_HOME"), "nwg-panel") dir := path.Join(xdgConfig, "nwg-panel")
return dir, nil return dir, nil
} else if os.Getenv("HOME") != "" { } else if home := os.Getenv("HOME"); home != "" {
dir := path.Join(os.Getenv("HOME"), ".config/nwg-panel") dir := path.Join(home, ".config/nwg-panel")
return dir, nil return dir, nil
} }
@@ -164,10 +154,10 @@ func oldConfigDir() (string, error) {
func configDir() string { func configDir() string {
var dir string var dir string
if os.Getenv("XDG_CONFIG_HOME") != "" { if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); xdgConfig != "" {
dir = path.Join(os.Getenv("XDG_CONFIG_HOME"), "nwg-drawer") dir = path.Join(xdgConfig, "nwg-drawer")
} else if os.Getenv("HOME") != "" { } else if home := os.Getenv("HOME"); home != "" {
dir = path.Join(os.Getenv("HOME"), ".config/nwg-drawer") dir = path.Join(home, ".config/nwg-drawer")
} }
log.Infof("Config dir: %s", dir) log.Infof("Config dir: %s", dir)
@@ -176,6 +166,20 @@ func configDir() string {
return dir return dir
} }
func dataDir() string {
var dir string
if xdgData := os.Getenv("XDG_DATA_HOME"); xdgData != "" {
dir = path.Join(xdgData, "nwg-drawer")
} else if home := os.Getenv("HOME"); home != "" {
dir = path.Join(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)
@@ -196,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
@@ -212,22 +226,13 @@ 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 := ""
home := os.Getenv("HOME") home := os.Getenv("HOME")
xdgDataHome := os.Getenv("XDG_DATA_HOME") xdgDataHome := os.Getenv("XDG_DATA_HOME")
if os.Getenv("XDG_DATA_DIRS") != "" { xdgDataDirs := os.Getenv("XDG_DATA_DIRS")
xdgDataDirs = os.Getenv("XDG_DATA_DIRS") if xdgDataDirs == "" {
} else {
xdgDataDirs = "/usr/local/share/:/usr/share/" xdgDataDirs = "/usr/local/share/:/usr/share/"
} }
if xdgDataHome != "" { if xdgDataHome != "" {
@@ -260,12 +265,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")
@@ -274,8 +287,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
} }
@@ -299,12 +312,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
@@ -349,6 +362,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 {
@@ -462,7 +477,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
} }
@@ -513,20 +528,35 @@ 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)
} }
} }
} }
} }
func launch(command string, terminal bool) { func launch(command string, terminal bool) {
themeToPrepend := ""
// add "GTK_THEME=<default_gtk_theme>" environment variable
if *forceTheme {
settings, _ := gtk.SettingsGetDefault()
th, err := settings.GetProperty("gtk-theme-name")
if err == nil {
themeToPrepend = th.(string)
}
}
// trim % and everything afterwards // trim % and everything afterwards
if strings.Contains(command, "%") { if strings.Contains(command, "%") {
cutAt := strings.Index(command, "%") cutAt := strings.Index(command, "%")
@@ -537,36 +567,51 @@ func launch(command string, terminal bool) {
elements := strings.Split(command, " ") elements := strings.Split(command, " ")
// find prepended env variables, if any envVarsNum := 0
envVarsNum := strings.Count(command, "=") if strings.Contains(elements[0], "=") {
var envVars []string for _, element := range elements {
if strings.Contains(element, "=") {
cmdIdx := -1 envVarsNum++
} else {
if envVarsNum > 0 { break
for idx, item := range elements {
if strings.Contains(item, "=") {
envVars = append(envVars, item)
} else if !strings.HasPrefix(item, "-") && cmdIdx == -1 {
cmdIdx = idx
} }
} }
} }
if cmdIdx == -1 {
cmdIdx = 0 // find prepended env variables, if any
var envVars []string
if themeToPrepend != "" {
envVars = append(envVars, fmt.Sprintf("GTK_THEME=%s", themeToPrepend))
} }
cmd := exec.Command(elements[cmdIdx], elements[1+cmdIdx:]...) cmdIdx := envVarsNum
firstArgIdx := envVarsNum + 1
if terminal { if envVarsNum > 0 {
var args []string for idx, item := range elements {
if *term != "foot" { if envVarsNum > idx && strings.Contains(item, "=") {
args = []string{"-e", elements[cmdIdx]} envVars = append(envVars, item)
} else { }
args = []string{elements[cmdIdx]}
} }
}
cmd = exec.Command(*term, args...) cmd := exec.Command(elements[cmdIdx], elements[firstArgIdx:]...)
var prefixCommand string
var args []string
if terminal {
prefixCommand = *term
if *term != "foot" {
args = []string{"-e", strings.Join(elements, " ")}
} else {
args = elements[cmdIdx:]
}
cmd = exec.Command(prefixCommand, args...)
} else if *wm == "sway" {
cmd = exec.Command("swaymsg", "exec", strings.Join(elements, " "))
} else if *wm == "hyprland" {
cmd = exec.Command("hyprctl", "dispatch", "exec", strings.Join(elements, " "))
} }
// set env variables // set env variables
@@ -575,10 +620,22 @@ func launch(command string, terminal bool) {
cmd.Env = append(cmd.Env, envVars...) cmd.Env = append(cmd.Env, envVars...)
} }
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, cmd.Args[0], cmd.Args[1:])
log.Info(msg) log.Info(msg)
cmd.Start() cmd.SysProcAttr = &syscall.SysProcAttr{
Setsid: true,
}
if cmd.Start() != nil {
log.Warn("Unable to launch terminal emulator!")
} else {
// Collect the exit code of the child process to prevent zombies
// if the drawer runs in resident mode
go func() {
_ = cmd.Wait()
}()
}
if *resident { if *resident {
restoreStateAndHide() restoreStateAndHide()
@@ -604,7 +661,15 @@ func open(filePath string, xdgOpen bool) {
} }
log.Infof("Executing: %s", cmd) log.Infof("Executing: %s", cmd)
cmd.Start() if cmd.Start() != nil {
log.Warn("Unable to execute command!")
} else {
// Collect the exit code of the child process to prevent zombies
// if the drawer runs in resident mode
go func() {
_ = cmd.Wait()
}()
}
if *resident { if *resident {
restoreStateAndHide() restoreStateAndHide()
@@ -648,3 +713,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:]
}

View File

@@ -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)
@@ -258,6 +260,10 @@ func flowBoxButton(entry desktopEntry) *gtk.Button {
exec := entry.Exec exec := entry.Exec
terminal := entry.Terminal terminal := entry.Terminal
desc := entry.CommentLoc desc := entry.CommentLoc
if len(desc) > 120 {
r := substring(desc, 0, 117)
desc = fmt.Sprintf("%s…", string(r))
}
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)
if btnEvent.Button() == 1 { if btnEvent.Button() == 1 {
@@ -425,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)) {

View File

@@ -20,12 +20,12 @@ func watchFiles() {
defer watcher.Close() defer watcher.Close()
if err := watcher.Add(pinnedFile); err != nil { if err := watcher.Add(pinnedFile); err != nil {
log.Errorf("ERROR", err) log.Errorf("ERROR: %s", err)
} }
for _, fp := range appDirs { for _, fp := range appDirs {
if err := filepath.Walk(fp, watchDir); err != nil { if err := filepath.Walk(fp, watchDir); err != nil {
log.Errorf("ERROR", err) log.Errorf("ERROR: %s", err)
} }
} }
@@ -41,11 +41,13 @@ func watchFiles() {
event.Op.String() == "RENAME") { event.Op.String() == "RENAME") {
desktopTrigger = true desktopTrigger = true
} else if event.Name == pinnedFile { } else if event.Name == pinnedFile {
pinnedTrigger = true // TODO: This can be used to propagate information about the changed file to the
// GUI to avoid recreating everything
pinnedItemsChanged <- struct{}{}
} }
case err := <-watcher.Errors: case err := <-watcher.Errors:
log.Errorf("ERROR", err) log.Errorf("ERROR: %s", err)
} }
} }
}() }()

View File

@@ -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)
} }