Compare commits

...

295 Commits

Author SHA1 Message Date
o9000
01f823cf79 memory tracing 2017-12-19 12:35:35 +01:00
o9000
978e1c90fc Update translations 2017-11-26 21:45:26 +01:00
o9000
3fba8aa1cf tint2conf: Fix tooltip (issue #588) 2017-11-26 21:40:07 +01:00
o9000
26251849c6 Include also timestamp in version number for unstable builds, so that version numbers increase 2017-11-26 18:38:17 +01:00
o9000
abeb7ce2e6 Include commit hash if possible in versioning script, if a non-stable tree is compiled 2017-11-26 18:28:25 +01:00
o9000
abdb1aeff9 Include commit hash if possible in versioning script, if a non-stable tree is compiled 2017-11-26 18:20:26 +01:00
o9000
7e383c395e Include commit hash if possible in versioning script, if a non-stable tree is compiled 2017-11-26 18:14:46 +01:00
o9000
c3ed2dadf3 Fix issue #684 2017-11-26 17:22:58 +01:00
o9000
bd2ca94ffe Cleanup versioning script 2017-11-26 16:22:12 +01:00
o9000
1753641fc9 Fix issue #684 2017-11-26 16:14:31 +01:00
o9000
467ea1332c Taskbar: thumbnails (cleanup) 2017-11-26 10:24:40 +01:00
o9000
68c2ad7062 Taskbar: thumbnails (disable cairo method as it leaks heavily when screensaver is on) 2017-11-25 22:27:46 +01:00
o9000
cc5842463d Simplify get_text_size 2017-11-23 22:38:55 +01:00
o9000
5e124c7a97 Separator: fix memory leak 2017-11-23 22:37:42 +01:00
o9000
07865142b2 Refactoring 2017-11-23 22:31:19 +01:00
o9000
0c71fda5e1 Button: fix memory leak 2017-11-23 22:28:26 +01:00
o9000
8eaf187984 Implement tinting by icon content (refactoring) 2017-11-23 22:05:41 +01:00
o9000
abe8a0eeb1 Implement tinting by icon content (do not apply task state asb effects) 2017-11-22 09:06:10 +01:00
o9000
d82d782541 Update authors 2017-11-21 13:30:46 +01:00
o9000
715eb556da Taskbar: thumbnails (initialize settings) 2017-11-21 12:51:54 +01:00
o9000
b9e64da9da Update translations 2017-11-21 12:40:27 +01:00
o9000
224b7fba82 Regenerate man page 2017-11-21 12:06:01 +01:00
o9000
d72fff9653 Taskbar: thumbnails (update doc) 2017-11-21 12:05:37 +01:00
o9000
67c3e47414 Implement tinting by icon content (config and documentation) 2017-11-21 12:04:36 +01:00
o9000
e96e7fbee7 Taskbar: thumbnails (fix bad copy paste) 2017-11-20 14:12:26 +01:00
o9000
65c91667f9 Implement tinting by icon content (bugfixes) 2017-11-20 10:15:12 +01:00
o9000
9f4087b471 Implement tinting by icon content (issue #638; thanks @heisenbug) 2017-11-19 22:15:24 +01:00
o9000
247687307b Taskbar: thumbnails (update periodically only the active window) 2017-11-17 17:32:49 +01:00
o9000
2c9d1fdf7d Taskbar: thumbnails (twaked blur a bit) 2017-11-17 17:25:53 +01:00
o9000
500b8f5bea Taskbar: thumbnails (improved fixed point precision) 2017-11-17 17:09:46 +01:00
o9000
7e6f7df55e Taskbar: thumbnails (interleave slow periodic throttling with event processing) 2017-11-17 15:53:23 +01:00
o9000
47201cab84 Taskbar: thumbnails (interleave slow periodic throttling with event processing) 2017-11-17 15:40:59 +01:00
o9000
f7d083904f Timer: do not clear timers restarted from their own callback function - fix check 2017-11-17 15:40:48 +01:00
o9000
f11d30f076 Taskbar: thumbnails (only print debug info if DEBUG_THUMBNAILS environment variable is set) 2017-11-17 15:15:15 +01:00
o9000
c0eaa8274f Taskbar: thumbnails (update every 1s for the active tooltip) 2017-11-17 15:08:34 +01:00
o9000
38488b8d75 Timer: do not clear timers restarted from their own callback function 2017-11-17 15:08:27 +01:00
o9000
07339c09a0 Taskbar: thumbnails (update every 1s for the active tooltip) 2017-11-17 11:45:42 +01:00
o9000
0521223899 Taskbar: thumbnails (interleave slow periodic throttling with event processing) 2017-11-17 08:19:23 +01:00
o9000
cbf3cebbb0 Taskbar: thumbnails (XShmGetImage - avoid leaking in X) 2017-11-16 12:14:09 +01:00
o9000
4b50446a7b Taskbar: thumbnails (use slower but safed XGetImage) 2017-11-16 11:39:44 +01:00
o9000
e8a6c93b28 Fix tooltip use after free 2017-11-16 08:32:19 +01:00
o9000
5e6e1184fe Taskbar: thumbnails (small fixes and disable shm for now) 2017-11-15 21:33:52 +01:00
o9000
89ab1fa6c4 Taskbar: thumbnails (make sure we are using rgb24) 2017-11-15 14:11:39 +01:00
o9000
e597973cd7 Taskbar: thumbnails (optimizations - get rid of XGetPixel calls) 2017-11-15 13:00:27 +01:00
o9000
ebe30774ac Taskbar: thumbnails (make sure window is not minimized) 2017-11-15 12:14:37 +01:00
o9000
d463dcb5b4 Taskbar: thumbnails (optimizations) 2017-11-15 12:10:49 +01:00
o9000
23782a4414 Timer: fix use after free 2017-11-15 12:08:29 +01:00
o9000
1be85e66fe Taskbar: thumbnails (optimizations) 2017-11-15 12:08:12 +01:00
o9000
2fe7efd4fe Taskbar: thumbnails (optimizations) 2017-11-15 11:44:35 +01:00
o9000
1b6fd91611 Taskbar: thumbnails (optimizations) 2017-11-15 11:28:03 +01:00
o9000
5730725762 Taskbar: thumbnails (tint2conf) 2017-11-14 17:51:35 +01:00
o9000
87da8c76cc Taskbar: thumbnails (config, tint2conf and doc) 2017-11-14 17:09:04 +01:00
o9000
e5ecc0c15d Taskbar: thumbnails (use image size without cosidering frame) 2017-11-14 15:41:21 +01:00
o9000
812e306376 Taskbar: thumbnails (use gaussian filtering, seems faster) 2017-11-14 13:53:37 +01:00
o9000
3155a5fc89 Taskbar: thumbnails (profiling) 2017-11-14 13:08:53 +01:00
o9000
bc4af51e82 Taskbar: thumbnails (alignment) 2017-11-14 12:58:42 +01:00
o9000
01de174919 Taskbar: thumbnails (timers) 2017-11-14 12:21:08 +01:00
o9000
7ddb373cb4 Taskbar: thumbnails 2017-11-14 11:04:55 +01:00
o9000
8ba1f26309 Taskbar: thumbnails 2017-11-14 10:48:47 +01:00
o9000
5a867a83c6 Taskbar: thumbnails 2017-11-14 10:34:57 +01:00
o9000
8a7e7e4281 Update changelog 2017-11-10 13:52:21 +01:00
o9000
0e8a6dd961 Systray: warn on duplicate config option systray_name_filter (issue #652) 2017-11-10 13:44:04 +01:00
o9000
c4a0ec4140 Executor: update tooltip documentation (fixes issue #676) 2017-11-10 13:37:41 +01:00
o9000
07d907fc43 Add Spanish translation (contributed by Vicmz) 2017-11-10 13:23:03 +01:00
o9000
b1f83baf04 Update periodic testing scripts 2017-11-05 19:54:17 +01:00
o9000
1ff3404e56 Update periodic testing scripts 2017-11-05 19:51:09 +01:00
o9000
41190204b3 Update periodic testing script 2017-11-05 19:33:09 +01:00
o9000
5bc978ee44 Update periodic testing script 2017-11-05 19:16:02 +01:00
o9000
3b4028f443 Update periodic testing script 2017-11-05 19:14:34 +01:00
o9000
acc3ee9205 Update periodic testing script 2017-11-05 15:38:33 +01:00
o9000
d8770ed590 Release 15.3 2017-11-05 12:28:35 +01:00
o9000
be7873a688 Update release script 2017-11-05 12:28:02 +01:00
o9000
cf81f1c9f9 Update release script 2017-11-05 12:27:05 +01:00
o9000
cde05df1bc Update release script 2017-11-05 12:24:46 +01:00
o9000
5fee459945 Update changelog 2017-11-05 12:19:56 +01:00
o9000
6d67291928 Update release script 2017-11-05 12:18:24 +01:00
o9000
4171e23153 Do not run regression tests more than once for a commit 2017-10-16 11:09:56 +02:00
o9000
725f625aba Use correct whitespace in debian/rules (makefile) 2017-10-15 20:48:28 +02:00
o9000
a6ea1eb5a9 Packaging: Workaround cmake issues 2017-10-15 20:41:46 +02:00
o9000
14b983cd0c Packaging: Workaround cmake issues 2017-10-15 20:30:39 +02:00
o9000
23ddb47e0c Reset signal mask at startup 2017-10-15 13:07:12 +02:00
o9000
407aef3786 Reset signal mask before executing commands (issue #674) 2017-10-15 13:00:18 +02:00
o9000
50c7bf77de Do not hardcode path to /etc 2017-10-04 19:15:44 +02:00
o9000
9a7d5a1a51 Release 15.2 2017-10-01 18:45:36 +02:00
o9000
031bd23849 Print tint2: before geometry dumps 2017-10-01 18:33:53 +02:00
o9000
7e2dc91ce7 Silence warning 2017-10-01 14:04:34 +02:00
o9000
c7a81655c4 Merge branch 'warning' into 'master'
Fix warning ISO C does not support __FUNCTION__

See merge request o9000/tint2!28
2017-10-01 11:54:43 +00:00
fafryd
9aa50104db Fix warning ISO C does not support __FUNCTION__
"warning: ISO C does not support ‘__FUNCTION__’ predefined identifier"

Among others -Wpedantic now warns about non-standard predefined identifiers.
The fix is either to use the standard predefined identifier __func__ (since C99),
or to use the __extension__ keyword.
2017-10-01 09:37:05 +02:00
o9000
3320ee8e05 Regression testing 2017-09-29 10:32:24 +02:00
o9000
ec380e25ec Regression testing 2017-09-28 20:06:39 +02:00
o9000
e1b29d0204 Regression testing 2017-09-28 19:39:30 +02:00
o9000
13313f64c5 Regression testing 2017-09-28 19:18:34 +02:00
o9000
9cb64e9cf5 Regression testing 2017-09-28 19:10:10 +02:00
o9000
fd0a1bef65 Regression testing 2017-09-28 18:51:07 +02:00
o9000
ae9ae098cc Regression testing 2017-09-28 18:47:42 +02:00
o9000
18f49d0d52 Regression testing 2017-09-28 18:39:15 +02:00
o9000
a57b6a4706 Regression testing 2017-09-28 18:38:54 +02:00
o9000
f8aa84a91b Regression testing 2017-09-28 18:36:34 +02:00
o9000
0154fe6a5a Regression testing 2017-09-28 15:59:29 +02:00
o9000
6f96818438 Regression testing 2017-09-28 15:43:12 +02:00
o9000
6433767a98 Regression testing 2017-09-28 15:39:10 +02:00
o9000
f8037b7ff5 Regression testing 2017-09-28 15:19:51 +02:00
o9000
7dddd4a5a2 Regression testing 2017-09-28 15:15:00 +02:00
o9000
85ac37d019 Regression testing 2017-09-28 15:11:31 +02:00
o9000
499b2bd938 Regression testing 2017-09-28 15:09:57 +02:00
o9000
5302fb4ba7 Regression testing 2017-09-28 15:08:26 +02:00
o9000
b58a6512b3 Update changelog 2017-09-28 12:27:26 +02:00
o9000
e38ccf5376 Remember window order on panel restart (issue #615) 2017-09-28 12:26:46 +02:00
o9000
3c9a0ff2f7 Updated changelog 2017-09-28 11:28:17 +02:00
o9000
b8675fa208 Compute text size correctly (issue #671) 2017-09-28 11:23:31 +02:00
o9000
ddac8f7802 Battery: add data gathering script 2017-09-17 15:20:38 +02:00
o9000
c21930de2b Battery: check that data files are readable and non-empty 2017-09-17 14:26:29 +02:00
o9000
f2cc1b68c1 More battery data 2017-09-17 14:12:41 +02:00
o9000
fc61676732 Battery: Do not show time remaining if rate is zero 2017-09-17 14:09:48 +02:00
o9000
1772d0a894 More battery data 2017-09-17 14:04:07 +02:00
o9000
3de424b129 Allow batteries that do not provide the charging rate 2017-09-17 14:04:03 +02:00
o9000
d8c289e0cc More battery data 2017-09-17 13:50:58 +02:00
o9000
5a17bb2fda Update packaging info script 2017-09-12 23:41:06 +02:00
o9000
c45b06657a Update packaging info script 2017-09-12 23:38:21 +02:00
o9000
7666077c63 Update packaging info script 2017-09-12 23:34:53 +02:00
o9000
542aa56840 Update packaging info script 2017-09-12 23:28:49 +02:00
o9000
d2f990366b Update packaging info script 2017-09-12 23:26:27 +02:00
o9000
1480faf32f Update packaging info script 2017-09-12 23:15:12 +02:00
o9000
38ff3318eb Update packaging info script 2017-09-12 23:14:08 +02:00
o9000
5bd253c0da Update packaging info script 2017-09-12 23:07:57 +02:00
o9000
9e94ee15e4 Update packaging info script 2017-09-12 22:56:25 +02:00
o9000
a026cd91fe Update memory reporting script 2017-09-12 22:36:52 +02:00
o9000
a52c45bd08 Remove duplicate entry 2017-09-12 22:36:30 +02:00
o9000
88c91aae25 Update version checker 2017-09-12 22:36:13 +02:00
o9000
b793544cf8 Add memory analysis script 2017-09-11 19:02:02 +02:00
o9000
63c3690fa8 Release 15.1 2017-09-08 18:08:28 +02:00
o9000
e2ece2c35b Updated changelog 2017-09-08 18:08:18 +02:00
o9000
86d6e96f7a Fix build on various architectures 2017-09-08 17:37:40 +02:00
o9000
87e1ccc6bf Fix build on various architectures 2017-09-08 17:37:01 +02:00
o9000
aa355e22f9 Fix build on various architectures 2017-09-08 17:04:10 +02:00
o9000
d21f758158 Fix build on various architectures 2017-09-08 16:35:24 +02:00
o9000
f2741116a7 Fix build on non-x86 architectures 2017-09-05 19:11:13 +02:00
o9000
ea92bf7718 Release 15.0 2017-09-02 15:07:36 +02:00
o9000
e2bbbd7835 Update versioning script 2017-09-02 15:07:29 +02:00
o9000
b38ae3aad3 Update versioning script 2017-09-02 15:06:56 +02:00
o9000
68d8b35ab8 Update versioning script 2017-09-02 15:05:42 +02:00
o9000
5d25b0ae9b Update readme 2017-09-02 14:55:04 +02:00
o9000
8fa4df0076 Update screenshots 2017-09-02 14:33:46 +02:00
o9000
909d5ec139 Update screenshots 2017-09-02 14:31:16 +02:00
o9000
64a0bf67db Update themes 2017-09-02 14:21:59 +02:00
o9000
a9047947b0 Update changelog 2017-09-02 13:08:14 +02:00
o9000
d49adfdef3 Launcher: Support %f and %F 2017-09-02 13:04:01 +02:00
o9000
498b665c8a Launcher: Fix drag and drop 2017-09-02 12:39:40 +02:00
o9000
14c3824632 Launcher: Add support for Terminal=true 2017-09-02 10:53:20 +02:00
o9000
eb93af3622 preprent "tint2:" to all logging messages 2017-09-01 18:23:46 +02:00
o9000
b7691afb8d Print logging output consistently to stderr 2017-09-01 18:11:35 +02:00
o9000
61c61c8844 Warn on primary_monitor_first 2017-09-01 18:06:56 +02:00
o9000
120207f1f9 Fix regression in cli handling 2017-09-01 18:02:35 +02:00
o9000
ceb6a44238 Launcher: fix layout on vertical panels 2017-09-01 11:34:57 +02:00
o9000
6a1b2f0610 Icon lookup: allow debugging enabled by env variable 2017-09-01 11:34:28 +02:00
o9000
95d4d90efd Button: fix parsing of empty config options 2017-09-01 11:32:44 +02:00
o9000
6605a1c3c2 Tracing support 2017-08-31 21:46:04 +02:00
o9000
0c754affd9 Refactoring 2017-08-31 18:55:22 +02:00
o9000
1e45abe988 Refactoring 2017-08-31 18:38:31 +02:00
o9000
6852e25372 Refactoring 2017-08-31 16:42:05 +02:00
o9000
ac8256a96b Regression tests 2017-08-31 11:42:19 +02:00
o9000
0786016436 Regression tests 2017-08-30 22:01:46 +02:00
o9000
f67b5db2f9 Regression tests 2017-08-30 17:42:02 +02:00
o9000
949fbdba14 Regression tests 2017-08-30 17:38:13 +02:00
o9000
558ffee93b Regression tests 2017-08-30 17:27:21 +02:00
o9000
69274ed7f9 Regression tests 2017-08-30 16:55:40 +02:00
o9000
7f7d0a0aa2 Regression tests 2017-08-30 16:40:59 +02:00
o9000
683d49bc71 Regression tests 2017-08-30 16:31:01 +02:00
o9000
da77f89910 Regression tests 2017-08-30 16:25:47 +02:00
o9000
9b9310dc84 Regression tests 2017-08-30 16:10:07 +02:00
o9000
1f94752e12 Regression tests 2017-08-30 16:00:01 +02:00
o9000
016958cb41 Regression tests 2017-08-30 15:50:31 +02:00
o9000
b003a2b1e3 Regression tests 2017-08-30 15:39:34 +02:00
o9000
eef665c896 Regression tests 2017-08-30 15:39:10 +02:00
o9000
e7492047c0 Regression tests 2017-08-30 15:30:06 +02:00
o9000
71db6de14f Regression tests 2017-08-30 15:29:52 +02:00
o9000
246048d8be Regression tests 2017-08-30 15:13:25 +02:00
o9000
3a7fb0971b packaging version check script 2017-08-28 18:50:40 +02:00
o9000
b672357f78 Add packaging version check script 2017-08-28 16:29:49 +02:00
o9000
302d8ec3b2 Add packaging version check script 2017-08-28 16:10:25 +02:00
o9000
cdbf8d6fc5 Add packaging version check script 2017-08-28 16:02:56 +02:00
o9000
f5f9d4983f Update changelog 2017-08-25 17:23:57 +02:00
o9000
2d67984536 Revert to the old autohide style (issues #640, #657) 2017-08-25 17:16:41 +02:00
o9000
e042bc3a17 Fix memory leak in icon themes (possibly related to issue #650) 2017-08-25 16:51:56 +02:00
o9000
297e2a13a2 Refresh clock text immediately if empty (issue #659, regression) 2017-08-25 16:31:49 +02:00
o9000
4df8f475ce Fix resizing of text elements (issue #661) 2017-08-24 23:57:48 +02:00
o9000
638264d874 Update changelog 2017-08-21 14:35:37 +02:00
o9000
21785a1a08 Update authors file 2017-08-21 14:30:45 +02:00
o9000
8e50c20c9d Reinitialize timers correctly when created from their own callbacks 2017-08-21 14:26:20 +02:00
o9000
c5d2ddc156 Merge branch 'master' of https://gitlab.com/o9000/tint2 2017-08-21 13:54:44 +02:00
o9000
e9adb18d58 Merge branch 'aaaz/tint2-master' 2017-08-21 13:49:56 +02:00
o9000
146408655b Merge branch 'master' into 'master'
update clock at the beginning of seconds

See merge request !27
2017-08-21 11:48:25 +00:00
o9000
3b7d583d8f packaging: remove yakkety (reached EOL) 2017-07-22 10:45:05 +02:00
o9000
4a704dde0c packaging: remove precise (reached EOL) 2017-07-22 10:35:39 +02:00
o9000
f5f8423364 panel: Set _NET_WM_PID (fixes issue #651) 2017-07-22 09:22:27 +02:00
o9000
00a5f72857 Create dirs with parents 2017-07-21 12:03:49 +02:00
o9000
f4ec61340f Fix fd leak to children 2017-07-16 10:14:40 +02:00
o9000
a46d22b31f Fix minor memory leak 2017-07-16 09:57:59 +02:00
o9000
7ad8bbe6e7 Fix minor memory leak 2017-07-16 09:50:53 +02:00
aaaz
fcbdd00bce update clock at the beginning of seconds 2017-07-08 16:24:45 -05:00
o9000
491a56db10 Clock: Update correctly after suspend 2017-07-02 23:19:24 +02:00
o9000
cbe31981b3 Executor: fallback to old style tooltips if stderr is empty 2017-06-27 12:42:51 +02:00
o9000
dafe9824e3 Update changelog 2017-06-27 12:32:23 +02:00
o9000
b931066573 Executor: truncate very long tooltips 2017-06-27 12:27:50 +02:00
o9000
f45e107207 Executor: if no user tooltip is set, display stderr output 2017-06-27 12:24:52 +02:00
o9000
ece3bc4d85 Add some new themes, remove redundant ones 2017-06-27 11:04:12 +02:00
o9000
4b9ee685ac Update timers correctly after suspend 2017-06-27 00:53:17 +02:00
o9000
cfc43685a4 ChangeLog is in the same directory of get_version.sh, get it from there 2017-06-26 22:46:17 +02:00
o9000
d05d5f594b Revert change to ubuntu script 2017-06-26 22:36:42 +02:00
o9000
425036adc9 Get version script: allow running without path parameter 2017-06-26 22:34:48 +02:00
o9000
cb174592be Merge branch 'version' into 'master'
Fix finding Version

See merge request !26
2017-06-26 20:29:57 +00:00
Chris Mayo
7a350a5e83 Fix finding ChangeLog when building
Keep the first optional argument of get_version.sh as "--strict" and
make the second the PROJECT_SOURCE_DIR plus trailing slash from
CMakeLists.txt
2017-06-26 19:23:58 +01:00
Chris Mayo
afadf3ea3f Revert "Use double quoting in get_version.sh"
Fixed symptom not the underlying problem.

This reverts commit e39a841f6e.
2017-06-26 19:23:58 +01:00
o9000
c8ccf053ff tint2conf: fix segfault in corner case 2017-06-26 11:08:31 +02:00
o9000
2f74250634 Add some new themes, remove redundant ones 2017-06-26 11:01:07 +02:00
o9000
ad7faaab81 Make get_version.sh work in the current dir instead of build 2017-06-26 10:23:50 +02:00
o9000
e39a841f6e Use double quoting in get_version.sh 2017-06-26 10:19:08 +02:00
o9000
fc7c6afa81 Updated man page 2017-06-25 13:02:07 +02:00
o9000
51bc9d1569 Updated translation files 2017-06-22 22:11:06 +02:00
o9000
747bbd7c7b Update changelog 2017-06-22 20:43:13 +02:00
o9000
521ffbfaaf Remove primary_monitor_first and add primary as a possible monitor value (issue #614) 2017-06-22 19:57:23 +02:00
o9000
cd33e5b274 Battery: better Unknown state handling 2017-06-22 19:03:05 +02:00
o9000
201ea843a0 Battery: new config option battery_full_cmd (fixes issue #585) 2017-06-20 20:38:26 +02:00
o9000
dee210ceec Updated French translation 2017-06-20 20:25:07 +02:00
o9000
ea0e52c5a2 Battery format fixes 2017-06-20 18:58:22 +02:00
o9000
b5578ff5bc battery: Do not redefine strncat with different semantics 2017-06-20 16:18:28 +02:00
o9000
f9ad3cb029 Merge battery format from https://gitlab.com/berkley4/tint2 (issue #229) 2017-06-20 13:15:41 +02:00
o9000
30c24c59ed Hide executor if output is empty (issue #648) 2017-06-20 13:01:09 +02:00
o9000
6f05c9d327 Hide executor if output is empty (issue #648) 2017-06-20 12:57:35 +02:00
o9000
30429b88b3 Merge battery format from https://gitlab.com/berkley4/tint2 (issue #229) 2017-06-20 12:36:04 +02:00
o9000
394be61a65 Collapse executor if output is empty (fixes issue #648) 2017-06-20 11:38:37 +02:00
o9000
d55f6d7baa taskbar_hide_different_desktop 2017-06-20 05:06:37 +02:00
o9000
2dfcab170d Update packaging 2017-06-11 13:30:27 +02:00
o9000
59b2094ad1 Update packaging 2017-06-11 11:28:04 +02:00
o9000
8046600a55 Release 0.14.6 2017-06-11 11:15:36 +02:00
o9000
17a655b52a Update changelog 2017-06-11 11:14:57 +02:00
o9000
75b8587216 Update changelog 2017-06-05 17:07:04 +02:00
o9000
7c87ab88bc Reexecute on SIGUSR2 2017-06-05 16:56:40 +02:00
o9000
b87470a284 Update packaging script 2017-06-05 16:21:14 +02:00
o9000
948bf10bad Refactor text rendering 2017-06-05 16:12:51 +02:00
o9000
59c3761455 Refactor text rendering 2017-06-05 16:12:51 +02:00
o9000
b50f5ecf3e executor: do not output last line if it is not terminated by newline 2017-06-02 20:10:41 +02:00
o9000
be9c492406 Update changelog 2017-05-31 23:13:55 +02:00
o9000
85e1a356fe Fix incorrect config warning 2017-05-31 23:06:47 +02:00
o9000
5da4c1653a Clock: take into account borders and padding when computing size 2017-05-29 23:57:38 +02:00
o9000
64edd55add Task: take into account border width when computing text height 2017-05-28 14:46:50 +02:00
o9000
bd28ee77d9 Fix task icon size limits (https://forums.bunsenlabs.org/viewtopic.php?pid=51884) 2017-05-24 21:07:47 +02:00
o9000
d10a505aa9 Release 0.14.5 2017-05-21 13:33:11 +02:00
o9000
77f744eba4 Update changelog 2017-05-21 13:32:49 +02:00
o9000
f8dde00a33 Do not pollute command line of executed processes 2017-05-20 14:48:25 +02:00
o9000
380f260027 Fix some memory leaks (issue #641) 2017-04-30 20:49:08 +02:00
o9000
6df4eb4bee Release 0.14.4 2017-04-29 13:47:21 +02:00
o9000
18fcc4952d Update release script 2017-04-29 13:47:04 +02:00
o9000
3878bd6a49 Update changelog 2017-04-29 13:44:22 +02:00
o9000
067234e9fb Unify code used to execute external programs 2017-04-29 13:42:37 +02:00
o9000
e32d2342a6 Fix regression in executor (issue #639) 2017-04-27 21:57:45 +02:00
o9000
82776df9d6 Fix crash when _NET_WM_ICON is set but empty 2017-04-27 09:04:16 +02:00
o9000
f351b6fd8e Release 0.14.3 2017-04-23 17:33:49 +02:00
o9000
a7e4cf42e2 Release script 2017-04-23 17:33:38 +02:00
o9000
a4542e907e Changelog 2017-04-23 17:30:12 +02:00
o9000
12e3fd64bb Release script 2017-04-23 17:29:03 +02:00
o9000
e5b3a2d7f2 Release 0.14.2 2017-04-23 17:07:45 +02:00
o9000
b973cafce5 Changelog 2017-04-23 17:07:42 +02:00
o9000
162de7fc2b Release script 2017-04-23 17:06:16 +02:00
o9000
b2cc11dd5e Build script 2017-04-23 17:01:07 +02:00
o9000
c53274a576 Doc script 2017-04-23 16:40:31 +02:00
o9000
3120cb9cae Update package 2017-04-23 14:45:25 +02:00
o9000
91f7384fd2 Fix compilation under ubunt 12.04 2017-04-22 22:21:13 +02:00
o9000
2e832a988f Update package 2017-04-22 22:14:45 +02:00
o9000
ae60e8f1e7 Use c99 to fix compilation on old distributions 2017-04-22 22:14:00 +02:00
o9000
e7dc61565d Use correct /etc path 2017-04-22 22:00:11 +02:00
o9000
4b53bc2a22 Update package 2017-04-22 21:46:40 +02:00
o9000
354e23da8f Update package 2017-04-22 21:24:08 +02:00
o9000
f6d121570b Update package 2017-04-22 21:12:58 +02:00
o9000
eddc29a696 Update package 2017-04-22 21:11:18 +02:00
o9000
eca3820e48 Update package 2017-04-22 21:08:59 +02:00
o9000
ef73ff9024 Update package 2017-04-22 21:08:15 +02:00
o9000
b7316ed24a Readme 2017-04-22 19:38:12 +02:00
o9000
49b41667ad tint2conf: Add mime type and full support for editing by path 2017-04-22 19:31:59 +02:00
o9000
5646dca8b7 Update changelog 2017-04-22 19:08:53 +02:00
o9000
701b57e113 Open current tint2 config automatically when started from tint2 2017-04-22 19:07:49 +02:00
o9000
affac4cfa6 Changed indentation everywhere 2017-04-13 14:07:23 +02:00
o9000
c1e5a7b172 Set indentation to use spaces instead of tabs, add auto indentation script 2017-04-13 14:06:37 +02:00
o9000
8f67884119 Turn off debugging output 2017-04-08 14:12:02 +02:00
o9000
0932cc1084 Change _NET_WM_ICON_GEOMETRY on panel shrink (issue #634) 2017-04-08 14:10:39 +02:00
o9000
e7375a186f Fix startup notification issue 2017-04-08 14:10:09 +02:00
o9000
8917a4d15b Remove _NET_WM_ICON_GEOMETRY for hidden task buttons (issue #634) 2017-04-08 13:50:49 +02:00
o9000
87ca207107 Button: prefix environment variables 2017-04-05 20:26:57 +02:00
o9000
982767ce2e Buttons: export alignment variables for menu 2017-04-02 11:18:17 +02:00
o9000
fc5385aa38 Buttons: export alignment variables for menu 2017-04-02 10:09:24 +02:00
o9000
39480f48ac Buttons: export alignment variables for menu 2017-04-01 14:04:08 +02:00
o9000
1c1ba904a4 Update doc 2017-03-26 22:04:35 +02:00
o9000
6cde7bef1e Release 0.14.1 2017-03-26 15:50:55 +02:00
o9000
b9ab0ae58e tint2conf: Fix bad code when compiling with clang 2017-03-26 15:47:16 +02:00
o9000
2c90c5640e tint2conf: Fix bad code when compiling with clang 2017-03-26 15:46:10 +02:00
279 changed files with 37096 additions and 29562 deletions

View File

@@ -63,4 +63,4 @@ SpacesInParentheses: false
SpacesInSquareBrackets: false SpacesInSquareBrackets: false
Standard: Cpp11 Standard: Cpp11
TabWidth: 4 TabWidth: 4
UseTab: ForIndentation UseTab: Never

2
.gitignore vendored
View File

@@ -1,3 +1,5 @@
build build
*.user *.user
version.h version.h
*.todo
*.pyc

View File

@@ -29,6 +29,9 @@ Contributors:
Oskari Rauta : separator plugin, gradients Oskari Rauta : separator plugin, gradients
Michael Messmore : Support for Path in .desktop files Michael Messmore : Support for Path in .desktop files
Matthew Otnel : config option systray_name_filter Matthew Otnel : config option systray_name_filter
Ryan Gray, Jeff Blake (https://gitlab.com/berkley4) : battery format
aaaz (https://gitlab.com/aaaz) : clock fixes
heisenbug (https://gitlab.com/heisenbugh) : taskbar button tinting with icon color
Translations: Translations:
Bosnian: Bosnian:
@@ -43,3 +46,5 @@ Translations:
Daniel Napora <napcok@gmail.com> Daniel Napora <napcok@gmail.com>
Serbian: Serbian:
Dino Duratović <dinomol@mail.com> Dino Duratović <dinomol@mail.com>
Spanish:
Vic <vicmz@yandex.com>

View File

@@ -6,6 +6,7 @@ option( ENABLE_TINT2CONF "Enable tint2conf build, a GTK+2 theme configurator for
option( ENABLE_EXTRA_THEMES "Install additional tint2 themes" ON ) option( ENABLE_EXTRA_THEMES "Install additional tint2 themes" ON )
option( ENABLE_RSVG "Rsvg support (launcher only)" ON ) option( ENABLE_RSVG "Rsvg support (launcher only)" ON )
option( ENABLE_SN "Startup notification support" ON ) option( ENABLE_SN "Startup notification support" ON )
option( ENABLE_TRACING "Build tint2 with tracing instrumentation" OFF )
option( ENABLE_ASAN "Build tint2 with AddressSanitizer" OFF ) option( ENABLE_ASAN "Build tint2 with AddressSanitizer" OFF )
option( ENABLE_BACKTRACE "Dump a backtrace in case of fatal errors (e.g. X11 I/O error)" OFF ) option( ENABLE_BACKTRACE "Dump a backtrace in case of fatal errors (e.g. X11 I/O error)" OFF )
option( ENABLE_BACKTRACE_ON_SIGNAL "Dump a backtrace also when receiving signals such as SIGSEGV" OFF ) option( ENABLE_BACKTRACE_ON_SIGNAL "Dump a backtrace also when receiving signals such as SIGSEGV" OFF )
@@ -24,7 +25,7 @@ endif()
include( FindPkgConfig ) include( FindPkgConfig )
include( CheckLibraryExists ) include( CheckLibraryExists )
include( CheckCSourceCompiles ) include( CheckCSourceCompiles )
pkg_check_modules( X11 REQUIRED x11 xcomposite xdamage xinerama xrender xrandr>=1.3 ) pkg_check_modules( X11 REQUIRED x11 xcomposite xdamage xinerama xext xrender xrandr>=1.3 )
pkg_check_modules( PANGOCAIRO REQUIRED pangocairo ) pkg_check_modules( PANGOCAIRO REQUIRED pangocairo )
pkg_check_modules( PANGO REQUIRED pango ) pkg_check_modules( PANGO REQUIRED pango )
pkg_check_modules( CAIRO REQUIRED cairo ) pkg_check_modules( CAIRO REQUIRED cairo )
@@ -83,10 +84,10 @@ if( NOT IMLIB_BUILD_WITH_X )
endif( NOT IMLIB_BUILD_WITH_X ) endif( NOT IMLIB_BUILD_WITH_X )
add_definitions( -D_BSD_SOURCE -D_DEFAULT_SOURCE -D_WITH_GETLINE )
if(CMAKE_SYSTEM_NAME STREQUAL "Linux") if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
add_definitions( -D_POSIX_C_SOURCE=200809L -D_BSD_SOURCE -D_DEFAULT_SOURCE ) add_definitions( -D_POSIX_C_SOURCE=200809L )
else(CMAKE_SYSTEM_NAME STREQUAL "Linux")
add_definitions( -D_WITH_GETLINE )
endif(CMAKE_SYSTEM_NAME STREQUAL "Linux") endif(CMAKE_SYSTEM_NAME STREQUAL "Linux")
@@ -116,7 +117,12 @@ include_directories( ${PROJECT_BINARY_DIR}
set( SOURCES src/config.c set( SOURCES src/config.c
src/panel.c src/panel.c
src/server.c src/server.c
src/tint.c src/main.c
src/init.c
src/signals.c
src/tracing.c
src/mouse_actions.c
src/drag_and_drop.c
src/clock/clock.c src/clock/clock.c
src/systray/systraybar.c src/systray/systraybar.c
src/launcher/launcher.c src/launcher/launcher.c
@@ -135,11 +141,16 @@ set( SOURCES src/config.c
src/tint2rc.c src/tint2rc.c
src/util/area.c src/util/area.c
src/util/common.c src/util/common.c
src/util/fps_distribution.c
src/util/strnatcmp.c src/util/strnatcmp.c
src/util/timer.c src/util/timer.c
src/util/cache.c src/util/cache.c
src/util/color.c src/util/color.c
src/util/gradient.c src/util/gradient.c
src/util/addr2line.c
src/util/print.c
src/util/mem.c
src/util/uevent.c
src/util/window.c ) src/util/window.c )
if( ENABLE_BATTERY ) if( ENABLE_BATTERY )
@@ -180,7 +191,6 @@ endif( ENABLE_SN)
if( ENABLE_UEVENT ) if( ENABLE_UEVENT )
add_definitions( -DENABLE_UEVENT ) add_definitions( -DENABLE_UEVENT )
set( SOURCES ${SOURCES} src/util/uevent.c)
endif( ENABLE_UEVENT ) endif( ENABLE_UEVENT )
if(ENABLE_BACKTRACE) if(ENABLE_BACKTRACE)
@@ -210,14 +220,23 @@ if( ENABLE_TINT2CONF )
endif( ENABLE_TINT2CONF ) endif( ENABLE_TINT2CONF )
if( ENABLE_ASAN ) if( ENABLE_ASAN )
SET(ASAN_C_FLAGS " -O0 -g3 -gdwarf-2 -fsanitize=address -fno-common -fno-omit-frame-pointer -rdynamic -Wshadow") SET(ASAN_C_FLAGS " -O0 -g3 -fsanitize=address -fno-common -fno-omit-frame-pointer -rdynamic -Wshadow")
SET(ASAN_L_FLAGS " -O0 -g3 -gdwarf-2 -fsanitize=address -fno-common -fno-omit-frame-pointer -rdynamic -fuse-ld=gold ") SET(ASAN_L_FLAGS " -O0 -g3 -fsanitize=address -fno-common -fno-omit-frame-pointer -rdynamic -fuse-ld=gold ")
else() else()
SET(ASAN_C_FLAGS "") SET(ASAN_C_FLAGS "")
SET(ASAN_L_FLAGS "") SET(ASAN_L_FLAGS "")
endif() endif()
add_custom_target( version ALL "${PROJECT_SOURCE_DIR}/get_version.sh" "\"${PROJECT_SOURCE_DIR}\"" ) if( ENABLE_TRACING )
add_definitions( -DHAVE_TRACING )
SET(TRACING_C_FLAGS " -finstrument-functions -finstrument-functions-exclude-file-list=tracing.c -finstrument-functions-exclude-function-list=get_time,gettime -O0 -g3 -fno-common -fno-omit-frame-pointer -rdynamic")
SET(TRACING_L_FLAGS " -O0 -g3 -fno-common -fno-omit-frame-pointer -rdynamic")
else()
SET(TRACING_C_FLAGS "")
SET(TRACING_L_FLAGS "")
endif()
add_custom_target( version ALL "${PROJECT_SOURCE_DIR}/get_version.sh" -- "\"${PROJECT_SOURCE_DIR}/\"" )
link_directories( ${X11_LIBRARY_DIRS} link_directories( ${X11_LIBRARY_DIRS}
${PANGOCAIRO_LIBRARY_DIRS} ${PANGOCAIRO_LIBRARY_DIRS}
@@ -249,10 +268,15 @@ if( RT_LIBRARY )
endif( RT_LIBRARY ) endif( RT_LIBRARY )
target_link_libraries( tint2 m ) target_link_libraries( tint2 m )
if(ENABLE_BACKTRACE)
target_link_libraries( tint2 dl )
target_link_libraries( tint2 z )
target_link_libraries( tint2 bfd )
endif(ENABLE_BACKTRACE)
add_dependencies( tint2 version ) add_dependencies( tint2 version )
set_target_properties( tint2 PROPERTIES COMPILE_FLAGS "-Wall -Wpointer-arith -fno-strict-aliasing -pthread -std=c11 ${ASAN_C_FLAGS}" ) set_target_properties( tint2 PROPERTIES COMPILE_FLAGS "-Wall -Wpointer-arith -fno-strict-aliasing -pthread -std=c11 ${ASAN_C_FLAGS} ${TRACING_C_FLAGS}" )
set_target_properties( tint2 PROPERTIES LINK_FLAGS "-pthread -fno-strict-aliasing ${ASAN_L_FLAGS} ${BACKTRACE_L_FLAGS}" ) set_target_properties( tint2 PROPERTIES LINK_FLAGS "-pthread -fno-strict-aliasing ${ASAN_L_FLAGS} ${BACKTRACE_L_FLAGS} ${TRACING_L_FLAGS}" )
install( TARGETS tint2 DESTINATION bin ) install( TARGETS tint2 DESTINATION bin )
install( FILES tint2.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps ) install( FILES tint2.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps )

View File

@@ -1,3 +1,96 @@
2017-11-10 master
- Enhancements:
- Added Spanish translation (contributed by Vicmz)
- Executor: updated tooltip documentation (issue #676)
- Systray: warn on duplicate config option systray_name_filter (issue #652)
2017-11-05 15.3
- Fixes:
- Launcher: Reset signal mask before executing commands (issue #674)
- cmake: Do not hardcode path to /etc
2017-10-01 15.2
- Fixes:
- Battery info is now again displayed even when current sensor is missing (https://github.com/jmc-88/tint3/issues/34)
- Text elements compute their size correctly (issue #671)
- Window order persists on panel restart (issue #615)
2017-09-08 15.1
- Fixes:
- Fixed build on non-Linux and non-x86 systems
2017-09-02 15.0
- Fixes:
- Clock, executors and other timers fire correctly after waking up from suspend
- One-shot timers are restarted correctly from their own callbacks
- Clock is refreshed with better accuracy (thanks @aaaz)
- Panel: by popular demand, the old struts behavior with autohide has been restored (issue #619);
if you encounter applications that interact poorly with it,
you might have better luck with strut_policy = minimum,
if that does not work, you will have to turn off autohide.
- Enhancements:
- Panel:
- _NET_WM_PID is set correctly, so now tint2 can be interacted with more easily from wmctrl and similar apps
- Taskbar: new config option taskbar_hide_different_desktop
- Battery:
- New config option bat1_format and bat2_format
- New config option battery_full_cmd
- Better "Unknown" state handling
- Executor:
- Hide if output is empty
- If no user tooltip is set, displays the script standard error as tooltip.
Tooltip is multiline, can be cleared with the VT100 clear screen sequence, in shell: (>&2 echo -en "\033[2J").
Long tooltips are truncated to 4096 characters.
- Launcher:
- Drag and drop now handles correctly text/uri-list
- Support for Terminal=true
- Support for %f and %F
- Configuration changes:
- Removed primary_monitor_first as it was conflicting with taskbar behavior; use *_monitor = primary instead.
- Other:
- Major code refactoring
- Dropping "0." from the version number and no longer using semver ("Breaking.Feature.Fix").
- Tint2 will always strive to be backwards compatible with respect to the configuration format.
- Very few configurations changes have been broken between 2010 (0.10) - 2017 (0.14);
in all cases they were minor options that caused incorrect behavior,
and the changes were described better by "Feature" or "Fix".
- Practically all releases starting from 0.10 have been very stable,
so there is no point in staying in "0." anymore.
- But I don't want the project to get stuck in "1." forever.
- The new versioning scheme is the following:
- Version numbers will have the format "Feature.Fix", where:
- "Feature" is increased when significant new features are added.
- "Fix" is increased for bugfixes or minor changes.
- 0.14.6 will be followed by 15.0.
2017-06-11 0.14.6
- Fixes:
- Take into account border width when computing text height
- Taskbar: Fix task icon size limits
- Executor: Do not output last line if it is not terminated by newline
- Enhancements:
- Re-execute tint2 on SIGUSR2.
This is useful for preserving config options and environment when updating tint2.
2017-05-21 0.14.5
- Fixes:
- Fixed a couple of memory leaks
2017-04-29 0.14.4
- Fixes:
- Fix regression in executor (issue #639)
- Fix crash when _NET_WM_ICON is set but empty (https://github.com/jmc-88/tint3/issues/21)
2017-04-23 0.14.3
- Fixes:
- Make versioning more robust when building as package
- Enhancements:
- Tint2conf: Open current tint2 config automatically when started from tint2
2017-03-26 0.14.1
- Fixes:
- Fixed tint2conf problem under FreeBSD (or generally any system using clang).
2017-03-26 0.14 2017-03-26 0.14
- Enhancements: - Enhancements:
- New plugin: button. - New plugin: button.
@@ -874,3 +967,7 @@ released tint-0.2
2008-04-22 2008-04-22
- fork ttm projet from p://code.google.com/p/ttm/ (by Pål Staurland staura@gmail.com) - fork ttm projet from p://code.google.com/p/ttm/ (by Pål Staurland staura@gmail.com)
while the projet is no longer in developpement, have not changed the name of 'tint'. while the projet is no longer in developpement, have not changed the name of 'tint'.
.
.
.
.

View File

@@ -1,5 +1,5 @@
# Latest stable release: 0.14 # Latest stable release: 15.3
Changes: https://gitlab.com/o9000/tint2/blob/0.14/ChangeLog Changes: https://gitlab.com/o9000/tint2/blob/15.3/ChangeLog
Documentation: [doc/tint2.md](doc/tint2.md) Documentation: [doc/tint2.md](doc/tint2.md)
@@ -8,7 +8,7 @@ Compile it with (after you install the [dependencies](https://gitlab.com/o9000/t
``` ```
git clone https://gitlab.com/o9000/tint2.git git clone https://gitlab.com/o9000/tint2.git
cd tint2 cd tint2
git checkout 0.14 git checkout 15.3
mkdir build mkdir build
cd build cd build
cmake .. cmake ..
@@ -19,6 +19,8 @@ To install, run (as root):
``` ```
make install make install
update-icon-caches /usr/local/share/icons/hicolor
update-mime-database /usr/local/share/mime
``` ```
And then you can run the panel `tint2` and the configuration program `tint2conf`. And then you can run the panel `tint2` and the configuration program `tint2conf`.
@@ -83,7 +85,7 @@ tint2 is a simple panel/taskbar made for modern X window managers. It was specif
## Various configs: ## Various configs:
![screenshot](https://gitlab.com/o9000/tint2/wikis/screenshot.png) * [Screenshots](https://gitlab.com/o9000/tint2/wikis/screenshots)
## Demos ## Demos
@@ -91,3 +93,8 @@ tint2 is a simple panel/taskbar made for modern X window managers. It was specif
* [Executor](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.4.gif) * [Executor](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.4.gif)
* [Mouse over effects](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.3.gif) * [Mouse over effects](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.3.gif)
* [Distribute size between taskbars, freespace](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.gif) * [Distribute size between taskbars, freespace](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.gif)
## More
* [Tint2 wiki](https://gitlab.com/o9000/tint2/wikis/Home)
Home)

View File

@@ -2,6 +2,9 @@
# You can install md2man with gem install md2man. You need gem and ruby-dev. # You can install md2man with gem install md2man. You need gem and ruby-dev.
set -e
set -x
md2man-roff tint2.md > tint2.1 md2man-roff tint2.md > tint2.1
cat header.html > manual.html cat header.html > manual.html

View File

@@ -199,7 +199,7 @@ pre {
</style> </style>
</head> </head>
<body> <body>
<h1 id="tint2"><span class="md2man-title">TINT2</span><a name="tint2" href="#tint2" class="md2man-permalink" title="permalink"></a></h1><h2 id="name">NAME<a name="name" href="#name" class="md2man-permalink" title="permalink"></a></h2><p>tint2 - lightweight panel/taskbar</p><h2 id="synopsis">SYNOPSIS<a name="synopsis" href="#synopsis" class="md2man-permalink" title="permalink"></a></h2><p><code>tint2 [-c path_to_config_file]</code></p><h2 id="description">DESCRIPTION<a name="description" href="#description" class="md2man-permalink" title="permalink"></a></h2><p>tint2 is a simple panel/taskbar made for modern X window managers. <h1 id="tint2"><span class="md2man-title">TINT2</span><a name="tint2" href="#tint2" class="md2man-permalink" title="permalink"></a></h1><h2 id="name">NAME<a name="name" href="#name" class="md2man-permalink" title="permalink"></a></h2><p>tint2 - lightweight panel/taskbar</p><h2 id="description">DESCRIPTION<a name="description" href="#description" class="md2man-permalink" title="permalink"></a></h2><p>tint2 is a simple panel/taskbar made for modern X window managers.
It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.).</p><p>Features:</p> It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.).</p><p>Features:</p>
<ul> <ul>
<li>Panel with taskbar, system tray, clock and launcher icons;</li> <li>Panel with taskbar, system tray, clock and launcher icons;</li>
@@ -214,7 +214,7 @@ It was specifically made for Openbox but it should also work with other window m
<li>Follow the freedesktop.org specifications;</li> <li>Follow the freedesktop.org specifications;</li>
<li>Make certain workflows, such as multi-desktop and multi-monitor, easy to use.</li> <li>Make certain workflows, such as multi-desktop and multi-monitor, easy to use.</li>
</ul> </ul>
<h2 id="options">OPTIONS<a name="options" href="#options" class="md2man-permalink" title="permalink"></a></h2><dl><dt><code>-c path_to_config_file</code></dt><dd>Specifies which configuration file to use instead of the default.</dd></dl><h2 id="configuration">CONFIGURATION<a name="configuration" href="#configuration" class="md2man-permalink" title="permalink"></a></h2><h3 id="table-of-contents">Table of contents<a name="table-of-contents" href="#table-of-contents" class="md2man-permalink" title="permalink"></a></h3> <h2 id="synopsis">SYNOPSIS<a name="synopsis" href="#synopsis" class="md2man-permalink" title="permalink"></a></h2><p><code>tint2 [OPTION...]</code></p><h2 id="options">OPTIONS<a name="options" href="#options" class="md2man-permalink" title="permalink"></a></h2><dl><dt><code>-c path_to_config_file</code></dt><dd>Specifies which configuration file to use instead of the default.</dd></dl><dl><dt><code>-v, --version</code></dt><dd>Prints version information and exits.</dd></dl><dl><dt><code>-h, --help</code></dt><dd>Display this help and exits.</dd></dl><h2 id="configuration">CONFIGURATION<a name="configuration" href="#configuration" class="md2man-permalink" title="permalink"></a></h2><h3 id="table-of-contents">Table of contents<a name="table-of-contents" href="#table-of-contents" class="md2man-permalink" title="permalink"></a></h3>
<ul> <ul>
<li><p><a href="#introduction">Introduction</a></p></li> <li><p><a href="#introduction">Introduction</a></p></li>
<li><p><a href="#backgrounds-and-borders">Backgrounds and borders</a></p></li> <li><p><a href="#backgrounds-and-borders">Backgrounds and borders</a></p></li>
@@ -270,6 +270,8 @@ Try to respect as much as possible the order of the options as given below.</p><
<li><code>color</code> is specified in hex RGB, e.g. #ff0000 is red</li> <li><code>color</code> is specified in hex RGB, e.g. #ff0000 is red</li>
<li><code>opacity</code> varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque</li> <li><code>opacity</code> varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque</li>
</ul></li> </ul></li>
<li><p><code>border_content_tint_weight = integer</code> : Mixes the border color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). <em>(since 16.0)</em></p></li>
<li><p><code>background_content_tint_weight = integer</code> : Mixes the background color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). <em>(since 16.0)</em></p></li>
</ul> </ul>
<p>You can define as many backgrounds as you want. For example, the following config defines two backgrounds:</p><pre class="highlight plaintext"><code>rounded = 1 <p>You can define as many backgrounds as you want. For example, the following config defines two backgrounds:</p><pre class="highlight plaintext"><code>rounded = 1
border_width = 0 border_width = 0
@@ -364,12 +366,12 @@ gradient_id_pressed = 2
<li><code>:</code> adds a separator. You can specify more than one. <em>(since 0.13.0)</em></li> <li><code>:</code> adds a separator. You can specify more than one. <em>(since 0.13.0)</em></li>
</ul> </ul>
<p>For example, <code>panel_items = STC</code> will show the systray, the taskbar and the clock (from left to right).</p></li> <p>For example, <code>panel_items = STC</code> will show the systray, the taskbar and the clock (from left to right).</p></li>
<li><p><code>panel_monitor = monitor (all or 1 or 2 or ...)</code> : Which monitor tint2 draws the panel on</p> <li><p><code>panel_monitor = monitor (all or primary or 1 or 2 or ...)</code> : Which monitor tint2 draws the panel on</p>
<ul> <ul>
<li>The first monitor is <code>1</code></li> <li>The first monitor is <code>1</code></li>
<li>Use <code>panel_monitor = all</code> to get a separate panel per monitor</li> <li>Use <code>panel_monitor = all</code> to get a separate panel per monitor</li>
</ul></li> </ul></li>
<li><p><code>primary_monitor_first = boolean (0 or 1)</code> : Place the primary monitor before all the other monitors in the list. <em>(since 0.12.4)</em></p></li> <li><p><code>primary_monitor_first = boolean (0 or 1)</code> : Place the primary monitor before all the other monitors in the list. <em>(since 0.12.4; removed in 1.0, use <code>primary</code> instead)</em></p></li>
</ul> </ul>
<p><img src="images/panel_padding.jpg" alt=""></p> <p><img src="images/panel_padding.jpg" alt=""></p>
<ul> <ul>
@@ -456,6 +458,7 @@ panel_size = 94% 30
<li><p><code>taskbar_active_background_id = integer</code> : Which background to use for the taskbar of the current virtual desktop.</p></li> <li><p><code>taskbar_active_background_id = integer</code> : Which background to use for the taskbar of the current virtual desktop.</p></li>
<li><p><code>taskbar_hide_inactive_tasks = boolean (0 or 1)</code> : If enabled, the taskbar shows only the active task. <em>(since 0.12)</em></p></li> <li><p><code>taskbar_hide_inactive_tasks = boolean (0 or 1)</code> : If enabled, the taskbar shows only the active task. <em>(since 0.12)</em></p></li>
<li><p><code>taskbar_hide_different_monitor = boolean (0 or 1)</code> : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. <em>(since 0.12)</em></p></li> <li><p><code>taskbar_hide_different_monitor = boolean (0 or 1)</code> : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. <em>(since 0.12)</em></p></li>
<li><p><code>taskbar_hide_different_desktop = boolean (0 or 1)</code> : If enabled, the taskbar shows only the tasks from the current desktop. Useful to make multi-desktop taskbars more compact, but still allow desktop switching with mouse click. <em>(since 1.0)</em></p></li>
<li><p><code>taskbar_always_show_all_desktop_tasks = boolean (0 or 1)</code> : Has effect only if <code>taskbar_mode = multi_desktop</code>. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. <em>(since 0.12.4)</em></p></li> <li><p><code>taskbar_always_show_all_desktop_tasks = boolean (0 or 1)</code> : Has effect only if <code>taskbar_mode = multi_desktop</code>. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. <em>(since 0.12.4)</em></p></li>
<li><p><code>taskbar_sort_order = none/title/center</code> : Specifies the sort order of the tasks on the taskbar. <em>(since 0.12)</em></p> <li><p><code>taskbar_sort_order = none/title/center</code> : Specifies the sort order of the tasks on the taskbar. <em>(since 0.12)</em></p>
<ul> <ul>
@@ -480,6 +483,8 @@ panel_size = 94% 30
<li><p><code>task_text = boolean (0 or 1)</code> : Whether to display the task text.</p></li> <li><p><code>task_text = boolean (0 or 1)</code> : Whether to display the task text.</p></li>
<li><p><code>task_centered = boolean (0 or 1)</code> : Whether the task text is centered.</p></li> <li><p><code>task_centered = boolean (0 or 1)</code> : Whether the task text is centered.</p></li>
<li><p><code>task_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for tasks.</p></li> <li><p><code>task_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for tasks.</p></li>
<li><p><code>task_thumbnail = boolean (0 or 1)</code> : Whether to show thumbnail tooltips for tasks. <em>(since 16.0)</em></p></li>
<li><p><code>task_thumbnail_size = width</code> : Thumbnail size. <em>(since 16.0)</em></p></li>
<li><p><code>task_maximum_size = width height</code></p> <li><p><code>task_maximum_size = width height</code></p>
<ul> <ul>
<li><code>width</code> is used with horizontal panels to limit the size of the tasks. Use <code>width = 0</code> to get full taskbar width.</li> <li><code>width</code> is used with horizontal panels to limit the size of the tasks. Use <code>width = 0</code> to get full taskbar width.</li>
@@ -523,7 +528,7 @@ panel_size = 94% 30
<li><p><code>systray_sort = ascending/descending/left2right/right2left</code> : Specifies the sorting order for the icons in the systray: in ascending/descending alphabetical order of the icon title, or always add icons to the right/left (note that with <code>left2right</code> or <code>right2left</code> the order can be different on panel restart).</p></li> <li><p><code>systray_sort = ascending/descending/left2right/right2left</code> : Specifies the sorting order for the icons in the systray: in ascending/descending alphabetical order of the icon title, or always add icons to the right/left (note that with <code>left2right</code> or <code>right2left</code> the order can be different on panel restart).</p></li>
<li><p><code>systray_icon_size = max_icon_size</code> : Set the maximum system tray icon size to <code>number</code>. Set to <code>0</code> for automatic icon sizing.</p></li> <li><p><code>systray_icon_size = max_icon_size</code> : Set the maximum system tray icon size to <code>number</code>. Set to <code>0</code> for automatic icon sizing.</p></li>
<li><p><code>systray_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjust the systray icons color and transparency.</p></li> <li><p><code>systray_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjust the systray icons color and transparency.</p></li>
<li><p><code>systray_monitor = integer (1, 2, ...)</code> : On which monitor to draw the systray. The first monitor is <code>1</code>. <em>(since 0.12)</em></p></li> <li><p><code>systray_monitor = integer (1, 2, ...) or primary</code> : On which monitor to draw the systray. The first monitor is <code>1</code>. <em>(since 0.12)</em></p></li>
<li><p><code>systray_name_filter = string</code> : Regular expression to identify icon names to be hidden. For example, <code>^audacious$</code> will hide icons with the exact name <code>audacious</code>, while <code>aud</code> will hide any icons having <code>aud</code> in the name. <em>(since 0.13.1)</em></p></li> <li><p><code>systray_name_filter = string</code> : Regular expression to identify icon names to be hidden. For example, <code>^audacious$</code> will hide icons with the exact name <code>audacious</code>, while <code>aud</code> will hide any icons having <code>aud</code> in the name. <em>(since 0.13.1)</em></p></li>
</ul> </ul>
<h3 id="clock">Clock<a name="clock" href="#clock" class="md2man-permalink" title="permalink"></a></h3> <h3 id="clock">Clock<a name="clock" href="#clock" class="md2man-permalink" title="permalink"></a></h3>
@@ -566,9 +571,19 @@ panel_size = 94% 30
<li><p><code>battery_hide = never/integer (0 to 100)</code> : At what battery percentage the battery item is hidden.</p></li> <li><p><code>battery_hide = never/integer (0 to 100)</code> : At what battery percentage the battery item is hidden.</p></li>
<li><p><code>battery_low_status = integer</code>: At what battery percentage the low command is executed.</p></li> <li><p><code>battery_low_status = integer</code>: At what battery percentage the low command is executed.</p></li>
<li><p><code>battery_low_cmd = notify-send &quot;battery low&quot;</code> : Command to execute when the battery is low.</p></li> <li><p><code>battery_low_cmd = notify-send &quot;battery low&quot;</code> : Command to execute when the battery is low.</p></li>
<li><p><code>battery_full_cmd = notify-send &quot;battery full&quot;</code> : Command to execute when the battery is full.</p></li>
<li><p><code>bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li> <li><p><code>bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
<li><p><code>bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li> <li><p><code>bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
<li><p><code>battery_font_color = color opacity (0 to 100)</code></p></li> <li><p><code>battery_font_color = color opacity (0 to 100)</code></p></li>
<li><p><code>bat1_format = FORMAT_STRING</code> : Format for battery line 1. Default: %p. <em>(since 1.0)</em> Format specification:</p>
<ul>
<li>%s: State (charging, discharging, full, unknown).</li>
<li>%m: Minutes left until completely charged/discharged (estimated).</li>
<li>%h: Hours left until completely charged/discharged (estimated).</li>
<li>%t: Time left. Shows &quot;hrs:mins&quot; when charging/discharging, or &quot;Ful\&quot; when full.</li>
<li>%p: Percentage. Includes the % sign.</li>
</ul></li>
<li><p><code>bat2_format = FORMAT_STRING</code> : Format for battery line 2. Default: %t. <em>(since 1.0)</em></p></li>
<li><p><code>battery_padding = horizontal_padding vertical_padding</code></p></li> <li><p><code>battery_padding = horizontal_padding vertical_padding</code></p></li>
<li><p><code>battery_background_id = integer</code> : Which background to use for the battery.</p></li> <li><p><code>battery_background_id = integer</code> : Which background to use for the battery.</p></li>
<li><p><code>battery_tooltip_enabled = boolean (0 or 1)</code> : Enable/disable battery tooltips. <em>(since 0.12.3)</em></p></li> <li><p><code>battery_tooltip_enabled = boolean (0 or 1)</code> : Enable/disable battery tooltips. <em>(since 0.12.3)</em></p></li>
@@ -590,7 +605,7 @@ panel_size = 94% 30
<li><p><code>execp_cache_icon = boolean (0 or 1)</code> : If <code>execp_cache_icon = 0</code>, the image is reloaded each time the command is executed (useful if the image file is changed on disk by the program executed by <code>execp_command</code>). <em>(since 0.12.4)</em></p></li> <li><p><code>execp_cache_icon = boolean (0 or 1)</code> : If <code>execp_cache_icon = 0</code>, the image is reloaded each time the command is executed (useful if the image file is changed on disk by the program executed by <code>execp_command</code>). <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_icon_w = integer</code> : You can use <code>execp_icon_w</code> and <code>execp_icon_h</code> to resize the image. If one of them is zero/missing, the image is rescaled proportionally. If both of them are zero/missing, the image is not rescaled. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_icon_w = integer</code> : You can use <code>execp_icon_w</code> and <code>execp_icon_h</code> to resize the image. If one of them is zero/missing, the image is rescaled proportionally. If both of them are zero/missing, the image is not rescaled. <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_icon_h = integer</code> : See <code>execp_icon_w</code>. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_icon_h = integer</code> : See <code>execp_icon_w</code>. <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_tooltip = text</code> : The tooltip. Leave it empty to not display a tooltip. Not specifying this option leads to showing an automatically generated tooltip with information about when the command was last executed. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_tooltip = text</code> : The tooltip. If left empty, no tooltip is displayed. If missing, the standard error of the command is shown as a tooltip (an ANSI clear screen sequence can reset the contents, bash: <code>printf '\e[2J'</code>, C: <code>printf(&quot;\x1b[2J&quot;);</code>). If the standard error is empty, the tooltip will show information about the time when the command was last executed. <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code> : The font used to draw the text. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code> : The font used to draw the text. <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_font_color = color opacity</code> : The font color. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_font_color = color opacity</code> : The font color. <em>(since 0.12.4)</em></p></li>
<li><p><code>execp_markup = boolean (0 or 1)</code> : If non-zero, the output of the command is treated as Pango markup, which allows rich text formatting. The format is <a href="https://developer.gnome.org/pygtk/stable/pango-markup-language.html">documented here</a>. Note that using this with commands that print data downloaded from the Internet is a possible security risk. <em>(since 0.12.4)</em></p></li> <li><p><code>execp_markup = boolean (0 or 1)</code> : If non-zero, the output of the command is treated as Pango markup, which allows rich text formatting. The format is <a href="https://developer.gnome.org/pygtk/stable/pango-markup-language.html">documented here</a>. Note that using this with commands that print data downloaded from the Internet is a possible security risk. <em>(since 0.12.4)</em></p></li>
@@ -683,6 +698,7 @@ execp_interval = 1
It is based on ttm, originally written by Pål Staurland <a href="mailto:staura@gmail.com">staura@gmail.com</a>.</p><p>This manual page was originally written by Daniel Moerner <a href="mailto:dmoerner@gmail.com">dmoerner@gmail.com</a>, for the Debian project (but may be used by others). It is based on ttm, originally written by Pål Staurland <a href="mailto:staura@gmail.com">staura@gmail.com</a>.</p><p>This manual page was originally written by Daniel Moerner <a href="mailto:dmoerner@gmail.com">dmoerner@gmail.com</a>, for the Debian project (but may be used by others).
It was adopted from the tint2 docs.</p><h2 id="see-also">SEE ALSO<a name="see-also" href="#see-also" class="md2man-permalink" title="permalink"></a></h2><p>The main website <a href="https://gitlab.com/o9000/tint2">https://gitlab.com/o9000/tint2</a> It was adopted from the tint2 docs.</p><h2 id="see-also">SEE ALSO<a name="see-also" href="#see-also" class="md2man-permalink" title="permalink"></a></h2><p>The main website <a href="https://gitlab.com/o9000/tint2">https://gitlab.com/o9000/tint2</a>
and the wiki page at <a href="https://gitlab.com/o9000/tint2/wikis/home">https://gitlab.com/o9000/tint2/wikis/home</a>.</p><p>This documentation is also provided in HTML and Markdown format in the system&#39;s default location and the wiki page at <a href="https://gitlab.com/o9000/tint2/wikis/home">https://gitlab.com/o9000/tint2/wikis/home</a>.</p><p>This documentation is also provided in HTML and Markdown format in the system&#39;s default location
for documentation files, usually <code>/usr/share/doc/tint2</code> or <code>/usr/local/share/doc/tint2</code>.</p> for documentation files, usually <code>/usr/share/doc/tint2</code> or <code>/usr/local/share/doc/tint2</code>.
.</p>
</body> </body>
</html> </html>

View File

@@ -199,15 +199,17 @@ pre {
</style> </style>
</head> </head>
<body> <body>
<h1 id="latest-stable-release-0-14"><span class="md2man-title">Latest</span> <span class="md2man-section">stable</span> <span class="md2man-date">release:</span> <span class="md2man-source">0.14</span><a name="latest-stable-release-0-14" href="#latest-stable-release-0-14" class="md2man-permalink" title="permalink"></a></h1><p>Changes: <a href="https://gitlab.com/o9000/tint2/blob/0.14/ChangeLog">https://gitlab.com/o9000/tint2/blob/0.14/ChangeLog</a></p><p>Documentation: <a href="manual.html">manual.html</a></p><p>Compile it with (after you install the <a href="https://gitlab.com/o9000/tint2/wikis/Install#dependencies">dependencies</a>):</p><pre class="highlight plaintext"><code>git clone https://gitlab.com/o9000/tint2.git <h1 id="latest-stable-release-15-3"><span class="md2man-title">Latest</span> <span class="md2man-section">stable</span> <span class="md2man-date">release:</span> <span class="md2man-source">15.3</span><a name="latest-stable-release-15-3" href="#latest-stable-release-15-3" class="md2man-permalink" title="permalink"></a></h1><p>Changes: <a href="https://gitlab.com/o9000/tint2/blob/15.3/ChangeLog">https://gitlab.com/o9000/tint2/blob/15.3/ChangeLog</a></p><p>Documentation: <a href="manual.html">manual.html</a></p><p>Compile it with (after you install the <a href="https://gitlab.com/o9000/tint2/wikis/Install#dependencies">dependencies</a>):</p><pre class="highlight plaintext"><code>git clone https://gitlab.com/o9000/tint2.git
cd tint2 cd tint2
git checkout 0.14 git checkout 15.3
mkdir build mkdir build
cd build cd build
cmake .. cmake ..
make -j4 make -j4
</code></pre> </code></pre>
<p>To install, run (as root):</p><pre class="highlight plaintext"><code>make install <p>To install, run (as root):</p><pre class="highlight plaintext"><code>make install
update-icon-caches /usr/local/share/icons/hicolor
update-mime-database /usr/local/share/mime
</code></pre> </code></pre>
<p>And then you can run the panel <code>tint2</code> and the configuration program <code>tint2conf</code>.</p><p>Please report any problems to <a href="https://gitlab.com/o9000/tint2/issues">https://gitlab.com/o9000/tint2/issues</a>. Your feedback is much appreciated.</p><p>P.S. GitLab is now the official location of the tint2 project, migrated from Google Code, which is shutting down. In case you are wondering why not GitHub, BitBucket etc., we chose GitLab because it is open source, it is mature and works well, looks cool and has a very nice team.</p><h1 id="what-is-tint2">What is tint2?<a name="what-is-tint2" href="#what-is-tint2" class="md2man-permalink" title="permalink"></a></h1><p>tint2 is a simple panel/taskbar made for modern X window managers. It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.). It is based on ttm <a href="https://code.google.com/p/ttm/">https://code.google.com/p/ttm/</a>.</p><h1 id="features">Features<a name="features" href="#features" class="md2man-permalink" title="permalink"></a></h1> <p>And then you can run the panel <code>tint2</code> and the configuration program <code>tint2conf</code>.</p><p>Please report any problems to <a href="https://gitlab.com/o9000/tint2/issues">https://gitlab.com/o9000/tint2/issues</a>. Your feedback is much appreciated.</p><p>P.S. GitLab is now the official location of the tint2 project, migrated from Google Code, which is shutting down. In case you are wondering why not GitHub, BitBucket etc., we chose GitLab because it is open source, it is mature and works well, looks cool and has a very nice team.</p><h1 id="what-is-tint2">What is tint2?<a name="what-is-tint2" href="#what-is-tint2" class="md2man-permalink" title="permalink"></a></h1><p>tint2 is a simple panel/taskbar made for modern X window managers. It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.). It is based on ttm <a href="https://code.google.com/p/ttm/">https://code.google.com/p/ttm/</a>.</p><h1 id="features">Features<a name="features" href="#features" class="md2man-permalink" title="permalink"></a></h1>
<ul> <ul>
@@ -254,12 +256,21 @@ make -j4
<li>Downloads: <a href="https://gitlab.com/o9000/tint2-archive/tree/master">https://gitlab.com/o9000/tint2-archive/tree/master</a> or <a href="https://code.google.com/p/tint2/downloads/list">https://code.google.com/p/tint2/downloads/list</a></li> <li>Downloads: <a href="https://gitlab.com/o9000/tint2-archive/tree/master">https://gitlab.com/o9000/tint2-archive/tree/master</a> or <a href="https://code.google.com/p/tint2/downloads/list">https://code.google.com/p/tint2/downloads/list</a></li>
<li>Old project location (inactive): <a href="https://code.google.com/p/tint2">https://code.google.com/p/tint2</a></li> <li>Old project location (inactive): <a href="https://code.google.com/p/tint2">https://code.google.com/p/tint2</a></li>
</ul> </ul>
<h1 id="screenshots">Screenshots<a name="screenshots" href="#screenshots" class="md2man-permalink" title="permalink"></a></h1><h2 id="default-config">Default config:<a name="default-config" href="#default-config" class="md2man-permalink" title="permalink"></a></h2><p><img src="https://gitlab.com/o9000/tint2/uploads/948fa74eca60864352a033580350b4c3/Screenshot_2016-01-23_14-42-57.png" alt="Screenshot_2016-01-23_14-42-57"></p><h2 id="various-configs">Various configs:<a name="various-configs" href="#various-configs" class="md2man-permalink" title="permalink"></a></h2><p><img src="https://gitlab.com/o9000/tint2/wikis/screenshot.png" alt="screenshot"></p><h2 id="demos">Demos<a name="demos" href="#demos" class="md2man-permalink" title="permalink"></a></h2> <h1 id="screenshots">Screenshots<a name="screenshots" href="#screenshots" class="md2man-permalink" title="permalink"></a></h1><h2 id="default-config">Default config:<a name="default-config" href="#default-config" class="md2man-permalink" title="permalink"></a></h2><p><img src="https://gitlab.com/o9000/tint2/uploads/948fa74eca60864352a033580350b4c3/Screenshot_2016-01-23_14-42-57.png" alt="Screenshot_2016-01-23_14-42-57"></p><h2 id="various-configs">Various configs:<a name="various-configs" href="#various-configs" class="md2man-permalink" title="permalink"></a></h2>
<ul>
<li><a href="https://gitlab.com/o9000/tint2/wikis/screenshots">Screenshots</a></li>
</ul>
<h2 id="demos">Demos<a name="demos" href="#demos" class="md2man-permalink" title="permalink"></a></h2>
<ul> <ul>
<li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.13.0.gif">Compact panel, separator, color gradients</a></li> <li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.13.0.gif">Compact panel, separator, color gradients</a></li>
<li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.4.gif">Executor</a></li> <li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.4.gif">Executor</a></li>
<li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.3.gif">Mouse over effects</a></li> <li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.3.gif">Mouse over effects</a></li>
<li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.gif">Distribute size between taskbars, freespace</a></li> <li><a href="https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.gif">Distribute size between taskbars, freespace</a></li>
</ul> </ul>
<h2 id="more">More<a name="more" href="#more" class="md2man-permalink" title="permalink"></a></h2>
<ul>
<li><a href="https://gitlab.com/o9000/tint2/wikis/Home">Tint2 wiki</a>
Home)</li>
</ul>
</body> </body>
</html> </html>

View File

@@ -1,10 +1,7 @@
.TH TINT2 1 "2017\-03\-26" 0.14 .TH TINT2 1 "2017\-11\-05" 15.3
.SH NAME .SH NAME
.PP .PP
tint2 \- lightweight panel/taskbar tint2 \- lightweight panel/taskbar
.SH SYNOPSIS
.PP
\fB\fCtint2 [\-c path_to_config_file]\fR
.SH DESCRIPTION .SH DESCRIPTION
.PP .PP
tint2 is a simple panel/taskbar made for modern X window managers. tint2 is a simple panel/taskbar made for modern X window managers.
@@ -33,10 +30,19 @@ Follow the freedesktop.org specifications;
.IP \(bu 2 .IP \(bu 2
Make certain workflows, such as multi\-desktop and multi\-monitor, easy to use. Make certain workflows, such as multi\-desktop and multi\-monitor, easy to use.
.RE .RE
.SH SYNOPSIS
.PP
\fB\fCtint2 [OPTION...]\fR
.SH OPTIONS .SH OPTIONS
.TP .TP
\fB\fC\-c path_to_config_file\fR \fB\fC\-c path_to_config_file\fR
Specifies which configuration file to use instead of the default. Specifies which configuration file to use instead of the default.
.TP
\fB\fC\-v, \-\-version\fR
Prints version information and exits.
.TP
\fB\fC\-h, \-\-help\fR
Display this help and exits.
.SH CONFIGURATION .SH CONFIGURATION
.SS Table of contents .SS Table of contents
.RS .RS
@@ -144,6 +150,10 @@ The tint2 config file starts with the options defining background elements with
.IP \(bu 2 .IP \(bu 2
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque \fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque
.RE .RE
.IP \(bu 2
\fB\fCborder_content_tint_weight = integer\fR : Mixes the border color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). \fI(since 16.0)\fP
.IP \(bu 2
\fB\fCbackground_content_tint_weight = integer\fR : Mixes the background color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). \fI(since 16.0)\fP
.RE .RE
.PP .PP
You can define as many backgrounds as you want. For example, the following config defines two backgrounds: You can define as many backgrounds as you want. For example, the following config defines two backgrounds:
@@ -303,7 +313,7 @@ gradient_id_pressed = 2
.PP .PP
For example, \fB\fCpanel_items = STC\fR will show the systray, the taskbar and the clock (from left to right). For example, \fB\fCpanel_items = STC\fR will show the systray, the taskbar and the clock (from left to right).
.IP \(bu 2 .IP \(bu 2
\fB\fCpanel_monitor = monitor (all or 1 or 2 or ...)\fR : Which monitor tint2 draws the panel on \fB\fCpanel_monitor = monitor (all or primary or 1 or 2 or ...)\fR : Which monitor tint2 draws the panel on
.RS .RS
.IP \(bu 2 .IP \(bu 2
The first monitor is \fB\fC1\fR The first monitor is \fB\fC1\fR
@@ -311,7 +321,7 @@ The first monitor is \fB\fC1\fR
Use \fB\fCpanel_monitor = all\fR to get a separate panel per monitor Use \fB\fCpanel_monitor = all\fR to get a separate panel per monitor
.RE .RE
.IP \(bu 2 .IP \(bu 2
\fB\fCprimary_monitor_first = boolean (0 or 1)\fR : Place the primary monitor before all the other monitors in the list. \fI(since 0.12.4)\fP \fB\fCprimary_monitor_first = boolean (0 or 1)\fR : Place the primary monitor before all the other monitors in the list. \fI(since 0.12.4; removed in 1.0, use \fB\fCprimary\fR instead)\fP
.RE .RE
.PP .PP
[](images/panel_padding.jpg) [](images/panel_padding.jpg)
@@ -455,6 +465,8 @@ You can switch between virtual desktops.
.IP \(bu 2 .IP \(bu 2
\fB\fCtaskbar_hide_different_monitor = boolean (0 or 1)\fR : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. \fI(since 0.12)\fP \fB\fCtaskbar_hide_different_monitor = boolean (0 or 1)\fR : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. \fI(since 0.12)\fP
.IP \(bu 2 .IP \(bu 2
\fB\fCtaskbar_hide_different_desktop = boolean (0 or 1)\fR : If enabled, the taskbar shows only the tasks from the current desktop. Useful to make multi\-desktop taskbars more compact, but still allow desktop switching with mouse click. \fI(since 1.0)\fP
.IP \(bu 2
\fB\fCtaskbar_always_show_all_desktop_tasks = boolean (0 or 1)\fR : Has effect only if \fB\fCtaskbar_mode = multi_desktop\fR\&. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. \fI(since 0.12.4)\fP \fB\fCtaskbar_always_show_all_desktop_tasks = boolean (0 or 1)\fR : Has effect only if \fB\fCtaskbar_mode = multi_desktop\fR\&. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. \fI(since 0.12.4)\fP
.IP \(bu 2 .IP \(bu 2
\fB\fCtaskbar_sort_order = none/title/center\fR : Specifies the sort order of the tasks on the taskbar. \fI(since 0.12)\fP \fB\fCtaskbar_sort_order = none/title/center\fR : Specifies the sort order of the tasks on the taskbar. \fI(since 0.12)\fP
@@ -500,6 +512,10 @@ The following options configure the task buttons in the taskbar:
.IP \(bu 2 .IP \(bu 2
\fB\fCtask_tooltip = boolean (0 or 1)\fR : Whether to show tooltips for tasks. \fB\fCtask_tooltip = boolean (0 or 1)\fR : Whether to show tooltips for tasks.
.IP \(bu 2 .IP \(bu 2
\fB\fCtask_thumbnail = boolean (0 or 1)\fR : Whether to show thumbnail tooltips for tasks. \fI(since 16.0)\fP
.IP \(bu 2
\fB\fCtask_thumbnail_size = width\fR : Thumbnail size. \fI(since 16.0)\fP
.IP \(bu 2
\fB\fCtask_maximum_size = width height\fR \fB\fCtask_maximum_size = width height\fR
.RS .RS
.IP \(bu 2 .IP \(bu 2
@@ -572,7 +588,7 @@ The action semantics:
.IP \(bu 2 .IP \(bu 2
\fB\fCsystray_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjust the systray icons color and transparency. \fB\fCsystray_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjust the systray icons color and transparency.
.IP \(bu 2 .IP \(bu 2
\fB\fCsystray_monitor = integer (1, 2, ...)\fR : On which monitor to draw the systray. The first monitor is \fB\fC1\fR\&. \fI(since 0.12)\fP \fB\fCsystray_monitor = integer (1, 2, ...) or primary\fR : On which monitor to draw the systray. The first monitor is \fB\fC1\fR\&. \fI(since 0.12)\fP
.IP \(bu 2 .IP \(bu 2
\fB\fCsystray_name_filter = string\fR : Regular expression to identify icon names to be hidden. For example, \fB\fC^audacious$\fR will hide icons with the exact name \fB\fCaudacious\fR, while \fB\fCaud\fR will hide any icons having \fB\fCaud\fR in the name. \fI(since 0.13.1)\fP \fB\fCsystray_name_filter = string\fR : Regular expression to identify icon names to be hidden. For example, \fB\fC^audacious$\fR will hide icons with the exact name \fB\fCaudacious\fR, while \fB\fCaud\fR will hide any icons having \fB\fCaud\fR in the name. \fI(since 0.13.1)\fP
.RE .RE
@@ -645,12 +661,30 @@ To hide the clock, comment \fB\fCtime1_format\fR and \fB\fCtime2_format\fR\&.
.IP \(bu 2 .IP \(bu 2
\fB\fCbattery_low_cmd = notify\-send "battery low"\fR : Command to execute when the battery is low. \fB\fCbattery_low_cmd = notify\-send "battery low"\fR : Command to execute when the battery is low.
.IP \(bu 2 .IP \(bu 2
\fB\fCbattery_full_cmd = notify\-send "battery full"\fR : Command to execute when the battery is full.
.IP \(bu 2
\fB\fCbat1_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR \fB\fCbat1_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
.IP \(bu 2 .IP \(bu 2
\fB\fCbat2_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR \fB\fCbat2_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
.IP \(bu 2 .IP \(bu 2
\fB\fCbattery_font_color = color opacity (0 to 100)\fR \fB\fCbattery_font_color = color opacity (0 to 100)\fR
.IP \(bu 2 .IP \(bu 2
\fB\fCbat1_format = FORMAT_STRING\fR : Format for battery line 1. Default: %p. \fI(since 1.0)\fP Format specification:
.RS
.IP \(bu 2
%s: State (charging, discharging, full, unknown).
.IP \(bu 2
%m: Minutes left until completely charged/discharged (estimated).
.IP \(bu 2
%h: Hours left until completely charged/discharged (estimated).
.IP \(bu 2
%t: Time left. Shows "hrs:mins" when charging/discharging, or "Ful\[rs]" when full.
.IP \(bu 2
%p: Percentage. Includes the % sign.
.RE
.IP \(bu 2
\fB\fCbat2_format = FORMAT_STRING\fR : Format for battery line 2. Default: %t. \fI(since 1.0)\fP
.IP \(bu 2
\fB\fCbattery_padding = horizontal_padding vertical_padding\fR \fB\fCbattery_padding = horizontal_padding vertical_padding\fR
.IP \(bu 2 .IP \(bu 2
\fB\fCbattery_background_id = integer\fR : Which background to use for the battery. \fB\fCbattery_background_id = integer\fR : Which background to use for the battery.
@@ -690,7 +724,7 @@ To hide the clock, comment \fB\fCtime1_format\fR and \fB\fCtime2_format\fR\&.
.IP \(bu 2 .IP \(bu 2
\fB\fCexecp_icon_h = integer\fR : See \fB\fCexecp_icon_w\fR\&. \fI(since 0.12.4)\fP \fB\fCexecp_icon_h = integer\fR : See \fB\fCexecp_icon_w\fR\&. \fI(since 0.12.4)\fP
.IP \(bu 2 .IP \(bu 2
\fB\fCexecp_tooltip = text\fR : The tooltip. Leave it empty to not display a tooltip. Not specifying this option leads to showing an automatically generated tooltip with information about when the command was last executed. \fI(since 0.12.4)\fP \fB\fCexecp_tooltip = text\fR : The tooltip. If left empty, no tooltip is displayed. If missing, the standard error of the command is shown as a tooltip (an ANSI clear screen sequence can reset the contents, bash: \fB\fCprintf '\\e[2J'\fR, C: \fB\fCprintf("\\x1b[2J");\fR). If the standard error is empty, the tooltip will show information about the time when the command was last executed. \fI(since 0.12.4)\fP
.IP \(bu 2 .IP \(bu 2
\fB\fCexecp_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR : The font used to draw the text. \fI(since 0.12.4)\fP \fB\fCexecp_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR : The font used to draw the text. \fI(since 0.12.4)\fP
.IP \(bu 2 .IP \(bu 2
@@ -869,3 +903,4 @@ and the wiki page at \[la]https://gitlab.com/o9000/tint2/wikis/home\[ra]\&.
.PP .PP
This documentation is also provided in HTML and Markdown format in the system's default location This documentation is also provided in HTML and Markdown format in the system's default location
for documentation files, usually \fB\fC/usr/share/doc/tint2\fR or \fB\fC/usr/local/share/doc/tint2\fR\&. for documentation files, usually \fB\fC/usr/share/doc/tint2\fR or \fB\fC/usr/local/share/doc/tint2\fR\&.
\&.

View File

@@ -1,11 +1,8 @@
# TINT2 1 "2017-03-26" 0.14 # TINT2 1 "2017-11-05" 15.3
## NAME ## NAME
tint2 - lightweight panel/taskbar tint2 - lightweight panel/taskbar
## SYNOPSIS
`tint2 [-c path_to_config_file]`
## DESCRIPTION ## DESCRIPTION
tint2 is a simple panel/taskbar made for modern X window managers. tint2 is a simple panel/taskbar made for modern X window managers.
It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.). It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.).
@@ -24,10 +21,20 @@ Goals:
* Follow the freedesktop.org specifications; * Follow the freedesktop.org specifications;
* Make certain workflows, such as multi-desktop and multi-monitor, easy to use. * Make certain workflows, such as multi-desktop and multi-monitor, easy to use.
## SYNOPSIS
`tint2 [OPTION...]`
## OPTIONS ## OPTIONS
`-c path_to_config_file` `-c path_to_config_file`
Specifies which configuration file to use instead of the default. Specifies which configuration file to use instead of the default.
`-v, --version`
Prints version information and exits.
`-h, --help`
Display this help and exits.
## CONFIGURATION ## CONFIGURATION
### Table of contents ### Table of contents
@@ -112,6 +119,10 @@ The tint2 config file starts with the options defining background elements with
* `color` is specified in hex RGB, e.g. #ff0000 is red * `color` is specified in hex RGB, e.g. #ff0000 is red
* `opacity` varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque * `opacity` varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque
* `border_content_tint_weight = integer` : Mixes the border color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). *(since 16.0)*
* `background_content_tint_weight = integer` : Mixes the background color with the content color (for tasks, this is the average color of the window icon). Values must be between 0 (no mixing) and 100 (fully replaces the color). *(since 16.0)*
You can define as many backgrounds as you want. For example, the following config defines two backgrounds: You can define as many backgrounds as you want. For example, the following config defines two backgrounds:
``` ```
@@ -251,11 +262,11 @@ gradient_id_pressed = 2
For example, `panel_items = STC` will show the systray, the taskbar and the clock (from left to right). For example, `panel_items = STC` will show the systray, the taskbar and the clock (from left to right).
* `panel_monitor = monitor (all or 1 or 2 or ...)` : Which monitor tint2 draws the panel on * `panel_monitor = monitor (all or primary or 1 or 2 or ...)` : Which monitor tint2 draws the panel on
* The first monitor is `1` * The first monitor is `1`
* Use `panel_monitor = all` to get a separate panel per monitor * Use `panel_monitor = all` to get a separate panel per monitor
* `primary_monitor_first = boolean (0 or 1)` : Place the primary monitor before all the other monitors in the list. *(since 0.12.4)* * `primary_monitor_first = boolean (0 or 1)` : Place the primary monitor before all the other monitors in the list. *(since 0.12.4; removed in 1.0, use `primary` instead)*
![](images/panel_padding.jpg) ![](images/panel_padding.jpg)
@@ -363,6 +374,8 @@ panel_size = 94% 30
* `taskbar_hide_different_monitor = boolean (0 or 1)` : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. *(since 0.12)* * `taskbar_hide_different_monitor = boolean (0 or 1)` : If enabled, the taskbar shows only the tasks from the current monitor. Useful when running different tint2 instances on different monitors, each one having its own config. *(since 0.12)*
* `taskbar_hide_different_desktop = boolean (0 or 1)` : If enabled, the taskbar shows only the tasks from the current desktop. Useful to make multi-desktop taskbars more compact, but still allow desktop switching with mouse click. *(since 1.0)*
* `taskbar_always_show_all_desktop_tasks = boolean (0 or 1)` : Has effect only if `taskbar_mode = multi_desktop`. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. *(since 0.12.4)* * `taskbar_always_show_all_desktop_tasks = boolean (0 or 1)` : Has effect only if `taskbar_mode = multi_desktop`. If enabled, tasks that appear on all desktops are shown on all taskbars. Otherwise, they are shown only on the taskbar of the current desktop. *(since 0.12.4)*
* `taskbar_sort_order = none/title/center` : Specifies the sort order of the tasks on the taskbar. *(since 0.12)* * `taskbar_sort_order = none/title/center` : Specifies the sort order of the tasks on the taskbar. *(since 0.12)*
@@ -400,6 +413,10 @@ The following options configure the task buttons in the taskbar:
* `task_tooltip = boolean (0 or 1)` : Whether to show tooltips for tasks. * `task_tooltip = boolean (0 or 1)` : Whether to show tooltips for tasks.
* `task_thumbnail = boolean (0 or 1)` : Whether to show thumbnail tooltips for tasks. *(since 16.0)*
* `task_thumbnail_size = width` : Thumbnail size. *(since 16.0)*
* `task_maximum_size = width height` * `task_maximum_size = width height`
* `width` is used with horizontal panels to limit the size of the tasks. Use `width = 0` to get full taskbar width. * `width` is used with horizontal panels to limit the size of the tasks. Use `width = 0` to get full taskbar width.
* `height` is used with vertical panels. * `height` is used with vertical panels.
@@ -464,7 +481,7 @@ The action semantics:
* `systray_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjust the systray icons color and transparency. * `systray_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjust the systray icons color and transparency.
* `systray_monitor = integer (1, 2, ...)` : On which monitor to draw the systray. The first monitor is `1`. *(since 0.12)* * `systray_monitor = integer (1, 2, ...) or primary` : On which monitor to draw the systray. The first monitor is `1`. *(since 0.12)*
* `systray_name_filter = string` : Regular expression to identify icon names to be hidden. For example, `^audacious$` will hide icons with the exact name `audacious`, while `aud` will hide any icons having `aud` in the name. *(since 0.13.1)* * `systray_name_filter = string` : Regular expression to identify icon names to be hidden. For example, `^audacious$` will hide icons with the exact name `audacious`, while `aud` will hide any icons having `aud` in the name. *(since 0.13.1)*
@@ -527,12 +544,23 @@ The action semantics:
* `battery_low_cmd = notify-send "battery low"` : Command to execute when the battery is low. * `battery_low_cmd = notify-send "battery low"` : Command to execute when the battery is low.
* `battery_full_cmd = notify-send "battery full"` : Command to execute when the battery is full.
* `bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` * `bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
* `bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` * `bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
* `battery_font_color = color opacity (0 to 100)` * `battery_font_color = color opacity (0 to 100)`
* `bat1_format = FORMAT_STRING` : Format for battery line 1. Default: %p. *(since 1.0)* Format specification:
* %s: State (charging, discharging, full, unknown).
* %m: Minutes left until completely charged/discharged (estimated).
* %h: Hours left until completely charged/discharged (estimated).
* %t: Time left. Shows "hrs:mins" when charging/discharging, or "Ful\" when full.
* %p: Percentage. Includes the % sign.
* `bat2_format = FORMAT_STRING` : Format for battery line 2. Default: %t. *(since 1.0)*
* `battery_padding = horizontal_padding vertical_padding` * `battery_padding = horizontal_padding vertical_padding`
* `battery_background_id = integer` : Which background to use for the battery. * `battery_background_id = integer` : Which background to use for the battery.
@@ -571,7 +599,7 @@ The action semantics:
* `execp_icon_h = integer` : See `execp_icon_w`. *(since 0.12.4)* * `execp_icon_h = integer` : See `execp_icon_w`. *(since 0.12.4)*
* `execp_tooltip = text` : The tooltip. Leave it empty to not display a tooltip. Not specifying this option leads to showing an automatically generated tooltip with information about when the command was last executed. *(since 0.12.4)* * `execp_tooltip = text` : The tooltip. If left empty, no tooltip is displayed. If missing, the standard error of the command is shown as a tooltip (an ANSI clear screen sequence can reset the contents, bash: `printf '\e[2J'`, C: `printf("\x1b[2J");`). If the standard error is empty, the tooltip will show information about the time when the command was last executed. *(since 0.12.4)*
* `execp_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` : The font used to draw the text. *(since 0.12.4)* * `execp_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` : The font used to draw the text. *(since 0.12.4)*
@@ -732,3 +760,4 @@ and the wiki page at https://gitlab.com/o9000/tint2/wikis/home.
This documentation is also provided in HTML and Markdown format in the system's default location This documentation is also provided in HTML and Markdown format in the system's default location
for documentation files, usually `/usr/share/doc/tint2` or `/usr/local/share/doc/tint2`. for documentation files, usually `/usr/share/doc/tint2` or `/usr/local/share/doc/tint2`.
.

3
format-code.sh Executable file
View File

@@ -0,0 +1,3 @@
#!/bin/bash
find . '(' -name '*.h' -o -name '*.c' ')' -exec clang-format-3.7 -style=file -i '{}' \;

View File

@@ -1,9 +1,62 @@
#!/bin/sh #!/bin/sh
MAJOR=0.14 SCRIPT_DIR=$(dirname "$0")
DIRTY="" DIRTY=""
VERSION=""
VERSION=0.14 if [ -d ${SCRIPT_DIR}/.git ] && git status 1>/dev/null 2>/dev/null
then
git update-index -q --ignore-submodules --refresh
# Disallow unstaged changes in the working tree
if ! git diff-files --quiet --ignore-submodules --
then
if [ "$1" = "--strict" ]
then
echo >&2 "Error: there are unstaged changes."
git diff-files --name-status -r --ignore-submodules -- >&2
exit 1
else
DIRTY="-dirty"
fi
fi
# Disallow uncommitted changes in the index
if ! git diff-index --cached --quiet HEAD --ignore-submodules --
then
if [ "$1" = "--strict" ]
then
echo >&2 "Error: there are uncommitted changes."
git diff-index --cached --name-status -r --ignore-submodules HEAD -- >&2
exit 1
else
DIRTY="-dirty"
fi
fi
if git describe 1>/dev/null 2>/dev/null
then
VERSION=$(git describe 2>/dev/null)$DIRTY
elif git log -n 1 1>/dev/null 2>/dev/null
then
VERSION=$(head -n 1 "${SCRIPT_DIR}/ChangeLog" | cut -d ' ' -f 2)
if [ "$VERSION" = "master" ]
then
PREVIOUS=$(grep '^2' "${SCRIPT_DIR}/ChangeLog" | head -n 2 | tail -n 1 | cut -d ' ' -f 2)
HASH=$(git log -n 1 --pretty=format:%cI.%ct.%h | tr -d ':' | tr -d '-' | tr '.' '-' | sed 's/T[0-9\+]*//g' 2>/dev/null)
VERSION=$PREVIOUS-next-$HASH
fi
fi
fi
if [ -z "$VERSION" ]
then
VERSION=$(head -n 1 "${SCRIPT_DIR}/ChangeLog" | cut -d ' ' -f 2)
if [ "$VERSION" = "master" ]
then
VERSION=$VERSION-$(head -n 1 "${SCRIPT_DIR}/ChangeLog" | cut -d ' ' -f 1)
fi
fi
VERSION=$(echo "$VERSION" | sed 's/^v//')
echo '#define VERSION_STRING "'$VERSION'"' > version.h echo '#define VERSION_STRING "'$VERSION'"' > version.h
echo $VERSION echo $VERSION

196
new-release.py Executable file
View File

@@ -0,0 +1,196 @@
#!/usr/bin/env python2
import argparse
import datetime
import inspect
import logging
import os
import re
import subprocess
import sys
import time
ansi_brown = "\x1b[0;33;40m"
ansi_yello_bold = "\x1b[1;33;40m"
ansi_lblue = "\x1b[0;36;40m"
ansi_pinky = "\x1b[0;35;40m"
ansi_reset = "\x1b[0m"
log_ts = None
def log_time():
global log_ts
if log_ts == None:
log_ts = time.time()
ts = time.time()
delta_ms = int((ts - log_ts) * 1000)
log_ts = ts
return "{0: >6}".format(delta_ms)
def log_prefix():
line = inspect.stack()[2][2]
function = inspect.stack()[2][3]
return ansi_lblue + "{0} {1}:{2}".format(log_time(), function, line) + ansi_reset
def debug(*args):
parts = [log_prefix()]
for s in args:
parts.append(str(s))
logging.debug(" ".join(parts))
def info(*args):
parts = [log_prefix()]
for s in args:
parts.append(str(s))
logging.info(" ".join(parts))
def cmd(s):
logging.debug(log_prefix() + " Executing: " + ansi_brown + s + ansi_reset)
return s
def run(s):
proc = subprocess.Popen(cmd(s), shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, close_fds=True)
ret = proc.wait()
out = proc.communicate()[0]
for line in out.split("\n"):
debug(ansi_pinky + line + ansi_reset)
debug(ansi_pinky + "Exit code: " + str(ret))
if ret != 0:
raise Exception("Command failed!")
return out
def natsorted(ls):
dre = re.compile(r'(\d+)')
return sorted(ls, key=lambda l: [int(s) if s.isdigit() else s.lower() for s in re.split(dre, l)])
def get_last_version():
tags = natsorted(run("git tag -l 'v*'").split("\n"))
return tags[-1]
def inc_version(v, feature=False):
if v.startswith("v0."):
assert v == "v0.14.6"
return "v15.0"
# v4.11 -> v4.12 or v5.0
parts = v.split(".")
while len(parts) < 2:
parts.append("0")
assert len(parts) == 2
if feature:
parts[-2] = "v" + str(int(parts[-2].replace("v", "")) + 1)
parts[-1] = "0"
else:
parts[-1] = str(int(parts[-1]) + 1)
return ".".join([s for s in parts if s])
def assert_equal(a, b):
if a != b:
info(a, "!=", b)
assert(False)
def test_inc_version():
# auto
assert_equal(inc_version("v0.14.6"), "v15.0")
assert_equal(inc_version("v15"), "v15.1")
assert_equal(inc_version("v15.0"), "v15.1")
assert_equal(inc_version("v16.1"), "v16.2")
assert_equal(inc_version("v16.10"), "v16.11")
# fix
assert_equal(inc_version("v0.14.6", False), "v15.0")
assert_equal(inc_version("v15", False), "v15.1")
assert_equal(inc_version("v15.0", False), "v15.1")
assert_equal(inc_version("v16.1", False), "v16.2")
assert_equal(inc_version("v16.10", False), "v16.11")
# feature
assert_equal(inc_version("v15", True), "v16.0")
assert_equal(inc_version("v15.0", True), "v16.0")
assert_equal(inc_version("v15.1", True), "v16.0")
assert_equal(inc_version("v15.2", True), "v16.0")
assert_equal(inc_version("v15.10", True), "v16.0")
def replace_in_file(path, before, after):
with open(path, "r+") as f:
old = f.read()
new = old.replace(before, after)
f.seek(0)
f.write(new)
def update_man(path, version, date):
with open(path, "r+") as f:
lines = f.read().split("\n")
# # TINT2 1 "2017-03-26" 0.14.1
parts = lines[0].split()
parts[-2] = '"' + date + '"'
parts[-1] = version
lines[0] = " ".join(parts)
f.seek(0)
f.write("\n".join(lines))
run("cd doc ; ./generate-doc.sh")
def update_log(path, version, date):
with open(path, "r+") as f:
lines = f.read().split("\n")
f.seek(0)
assert lines[0].endswith("master")
lines[0] = date + " " + version
f.write("\n".join(lines))
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument("--feature", action="store_true")
parser.add_argument("--undo", action="store_true")
args = parser.parse_args()
logging.basicConfig(format=ansi_lblue + "%(asctime)s %(pathname)s %(levelname)s" + ansi_reset + " %(message)s", level=logging.DEBUG)
test_inc_version()
# Read version from last tag and increment
old_version = get_last_version()
if args.undo:
info("Revering last commit...")
run("git tag -d %s" % old_version)
run("git tag -d %s" % old_version.replace("v", ""))
run("git reset --soft HEAD~")
run("git reset")
run("git stash")
os.system("git log -1")
sys.exit(0)
info("Old version:", old_version)
version = inc_version(old_version, args.feature)
readable_version = version.replace("v", "")
date = datetime.datetime.now().strftime("%Y-%m-%d")
info("New version:", readable_version, version, date)
# Disallow unstaged changes in the working tree
run("git diff-files --quiet --ignore-submodules --")
# Disallow uncommitted changes in the index
run("git diff-index --cached --quiet HEAD --ignore-submodules --")
# Update version string
replace_in_file("README.md", old_version.replace("v", ""), readable_version)
update_man("doc/tint2.md", readable_version, date)
update_log("ChangeLog", readable_version, date)
run("git commit -am 'Release %s'" % readable_version)
run("git tag -a %s -m 'version %s'" % (version, readable_version))
run("git tag -a %s -m 'version %s'" % (readable_version, readable_version))
run("rm -rf tint2-%s* || true" % readable_version)
run("./make_release.sh")
run("tar -xzf tint2-%s.tar.gz" % readable_version)
run("cd tint2-%s ; mkdir build ; cd build ; cmake .. ; make" % readable_version)
assert_equal(run("./tint2-%s/build/tint2 -v" % readable_version).strip(), "tint2 version %s" % readable_version)
os.system("git log -p -1 --word-diff")
print "Does this look correct? [y/n]"
choice = raw_input().lower()
if choice != "y":
run("git reset --hard HEAD~ ; git tag -d %s ; git tag -d %s" % (version, readable_version))

View File

@@ -1,3 +1,35 @@
tint2 (0.12.12-3) unstable; urgency=medium
* Cherry pick upstream fix for use-after-free
-- Sebastian Reichel <sre@debian.org> Sun, 29 Jan 2017 21:18:20 +0100
tint2 (0.12.12-2) unstable; urgency=medium
* Cherry pick a few upstream fixes (Closes: #849228)
-- Sebastian Reichel <sre@debian.org> Fri, 13 Jan 2017 22:01:29 +0100
tint2 (0.12.12-1) unstable; urgency=low
* New upstream release
-- Sebastian Reichel <sre@debian.org> Sun, 21 Aug 2016 22:39:16 +0200
tint2 (0.12.11-1) unstable; urgency=low
* New upstream release (Closes: #824568)
* Update Debian Standards Version to 3.9.8
-- Sebastian Reichel <sre@debian.org> Wed, 01 Jun 2016 00:48:54 +0200
tint2 (0.12.7-2) unstable; urgency=medium
* Add upstream patch to fix build on ppc
* Use https:// vcs urls
-- Sebastian Reichel <sre@debian.org> Mon, 22 Feb 2016 03:23:51 +0100
tint2 (0.12.7-1) unstable; urgency=medium tint2 (0.12.7-1) unstable; urgency=medium
* New upstream release * New upstream release

View File

@@ -1,7 +1,7 @@
Source: tint2 Source: tint2
Section: x11 Section: x11
Priority: optional Priority: optional
Maintainer: o9000 <mrovi9000@gmail.com> Maintainer: Sebastian Reichel <sre@debian.org>
Build-Depends: cmake, Build-Depends: cmake,
debhelper (>= 9), debhelper (>= 9),
libcairo2-dev, libcairo2-dev,
@@ -16,14 +16,15 @@ Build-Depends: cmake,
libxinerama-dev, libxinerama-dev,
libxrandr-dev libxrandr-dev
Standards-Version: 3.9.7 Standards-Version: 3.9.7
Vcs-Git: git://anonscm.debian.org/collab-maint/tint2.git Vcs-Git: https://alioth.debian.org/anonscm/git/collab-maint/tint2.git
Vcs-Browser: http://anonscm.debian.org/gitweb/?p=collab-maint/tint2.git Vcs-Browser: https://anonscm.debian.org/gitweb/?p=collab-maint/tint2.git
Homepage: https://gitlab.com/o9000/tint2/ Homepage: https://gitlab.com/o9000/tint2/
Package: tint2 Package: tint2
Architecture: any Architecture: any
Depends: ${shlibs:Depends}, Depends: ${shlibs:Depends},
${misc:Depends} ${misc:Depends},
procps
Description: lightweight taskbar Description: lightweight taskbar
Tint is a simple panel/taskbar intentionally made for openbox3, but should Tint is a simple panel/taskbar intentionally made for openbox3, but should
also work with other window managers. The taskbar includes transparency and also work with other window managers. The taskbar includes transparency and

View File

@@ -7,6 +7,22 @@ Copyright: 2007-2008 Pål Staurland <staura@gmail.com>
2009 Andreas Fink <andreas.fink85@googlemail.com> 2009 Andreas Fink <andreas.fink85@googlemail.com>
2011-2015 Ovidiu M <mrovi9000@gmail.com> 2011-2015 Ovidiu M <mrovi9000@gmail.com>
License: GPL-2 License: GPL-2
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License version 2
as published by the Free Software Foundation, or (at your option) any
later version.
.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
.
On Debian systems, the full text of the GNU General Public License 2
can be found in `/usr/share/common-licenses/GPL-2'.
Files: src/battery/* Files: src/battery/*
Copyright: 2009 Sebastian Reichel <sre@debian.org> Copyright: 2009 Sebastian Reichel <sre@debian.org>
@@ -43,24 +59,3 @@ License: ISC
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
Files: debian/wiki/*
Copyright: 2009-2012 Andreas Fink <andreas.fink85@googlemail.com>
License: GPL-2
License: GPL-2
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
version 2 as published by the Free Software Foundation.
.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
.
On Debian systems, the full text of the GNU General Public License 2
can be found in `/usr/share/common-licenses/GPL-2'.

View File

@@ -0,0 +1,2 @@
pkill -SIGUSR2 tint2 || true

View File

@@ -4,11 +4,10 @@
dh $@ dh $@
override_dh_auto_configure: override_dh_auto_configure:
dh_auto_configure -- -DCMAKE_BUILD_TYPE=RelWithDebInfo dh_auto_configure -- -DCMAKE_BUILD_TYPE=RelWithDebInfo -DCMAKE_INSTALL_SYSCONFDIR=/etc
override_dh_auto_install: override_dh_auto_install:
dh_auto_install --destdir=$(CURDIR)/debian/tmp dh_auto_install --destdir=$(CURDIR)/debian/tmp
override_dh_installdocs: override_dh_installdocs:
dh_installdocs --link-doc=tint2 dh_installdocs --link-doc=tint2
cp -r debian/wiki debian/tint2/usr/share/doc/tint2/html

View File

View File

@@ -4,9 +4,9 @@
/usr/share/applications/tint2.desktop /usr/share/applications/tint2.desktop
/usr/share/applications/tint2conf.desktop /usr/share/applications/tint2conf.desktop
/usr/share/man/man1/tint2.1 /usr/share/man/man1/tint2.1
/usr/share/doc/tint2/AUTHORS /usr/share/doc/tint2/*
/usr/share/doc/tint2/README.md
/usr/share/icons/hicolor/scalable/apps/tint2.svg /usr/share/icons/hicolor/scalable/apps/tint2.svg
/usr/share/icons/hicolor/scalable/apps/tint2conf.svg /usr/share/icons/hicolor/scalable/apps/tint2conf.svg
/usr/share/locale/* /usr/share/locale/*
/usr/share/tint2/* /usr/share/tint2/*
/usr/share/mime/packages/*

View File

@@ -1,5 +1,7 @@
#!/bin/bash #!/bin/bash
# Requirements: devscripts
set -x set -x
rm -rf tint2* 2>/dev/null || true rm -rf tint2* 2>/dev/null || true
@@ -29,6 +31,8 @@ else
REPO="tint2-git" REPO="tint2-git"
fi fi
set -e
# Export repository contents to source directory # Export repository contents to source directory
DIR=tint2-$VERSION DIR=tint2-$VERSION
echo "Making release $DIR" echo "Making release $DIR"
@@ -43,9 +47,9 @@ rm -f $DIR/make_release.sh
echo "echo \"#define VERSION_STRING \\\"$VERSION\\\"\" > version.h" > $DIR/get_version.sh echo "echo \"#define VERSION_STRING \\\"$VERSION\\\"\" > version.h" > $DIR/get_version.sh
# Copy the debian files into the source directory # Copy the debian files into the source directory
cp -r ubuntu $DIR/debian cp -r debian $DIR/debian
for DISTRO in precise trusty wily xenial for DISTRO in trusty xenial zesty artful
do do
# Cleanup from previous builds # Cleanup from previous builds
rm -rf tint2_$VERSION-* rm -rf tint2_$VERSION-*

View File

@@ -1,4 +0,0 @@
debian/wiki/3265475271_81e8ed56e0.jpg
debian/wiki/3265475337_262799ded3.jpg
debian/wiki/3266303192_476a188196.jpg
debian/wiki/3266303292_14d9b6b623.jpg

Binary file not shown.

Before

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 12 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 11 KiB

View File

@@ -1,405 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>tint2 - Configure</title>
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
</head>
<body class="t6">
<table class="st" width="100%" cellpadding="0" cellspacing="0" align="center" border="0">
<tbody><tr>
<td class="bevel-right" valign="top" align="right"></td>
</tr>
</tbody></table>
<div id="maincol" style="padding: 0pt 3px 3px 0pt; margin: 0pt;">
<!-- IE -->
<div id="wikipage">
<table>
<tbody><tr>
<td style="vertical-align: top; padding-left: 5px;">
<div id="wikicontent">
<div class="vt" id="wikimaincol">
<h1>tint2 - Configure</h1>
<p></p><ul><li><a href="#Background_and_border">Background and border</a></li><li><a href="#Panel">Panel</a></li><li><a href="#Launcher_SVN_Only">Launcher SVN Only</a></li><li><a href="#Taskbar_%28pager_like_capability%29">Taskbar (pager like capability)</a></li><li><a href="#Task">Task</a></li><li><a href="#System_Tray">System Tray</a></li><li><a href="#Clock">Clock</a></li><li><a href="#Tooltip">Tooltip</a></li><li><a href="#Battery">Battery</a></li><li><a href="#Mouse_action_on_task">Mouse action on task</a></li><li><a href="#Panel_autohide">Panel autohide</a></li><li><a href="#An_example_tint2rc">An example tint2rc</a></li></ul> <p></p><p>The first time you run tint2, it will create the config file in <tt>$HOME/.config/tint2/tint2rc</tt> . (this is true if you have done a clean install. running tint2 in the source directory without doing 'make install' will not create the config file.) </p><p>You can specify another file on the command line with -c option. eg. : <tt>tint2 -c $HOME/tint2.conf</tt> </p><p>When you change the config file, the command line 'killall -SIGUSR1 tint2' will force tint2 to reload it. </p><p>Configuration marked as <tt>SVN ONLY</tt> are not yet supported and could change in the next stable release. </p><h1><a name="Background_and_border"></a>Background and border<a href="#Background_and_border" class="section_anchor"></a></h1><blockquote>tint2rc config file starts with background option.
</blockquote><p> </p><ul><li><tt>rounded = number_of_pixels</tt> : How rounded the borders are </li></ul><p></p><ul><li><tt>background_color = color opacity</tt> </li><ul><li><tt>opacity = 0</tt> to get background transparency </li></ul></ul><p> </p><ul><li><tt>border_width = width</tt> </li></ul><p></p><ul><li><tt>border_color = color opacity (0 to 100)</tt> </li><ul><li><tt>opacity = 0</tt> to get border transparency </li></ul></ul><blockquote>you are free to put 1, 2, 3... background
<pre class="prettyprint"> rounded = 1
border_width = 0
background_color = #282828 100
border_color = #000000 0
rounded = 1
border_width = 0
background_color = #f6b655 90
border_color = #cccccc 40</pre>tint2 automatically identifies each background with a number (1, 2, ...).
then you can applied background on objects (panel, taskbar, task, clock, systray) with
<pre class="prettyprint"> panel_background_id = 1
taskbar_background_id = 0
task_background_id = 0
task_active_background_id = 2
systray_background_id = 0
clock_background_id = 0</pre>identifier 0 applied full transparency, identifer 1 applied the first background...
</blockquote><h1><a name="Panel"></a>Panel<a href="#Panel" class="section_anchor"></a></h1><ul><li><tt>panel_monitor = monitor (all, 1, 2, ...)</tt> Which monitor tint2 draws the panel on </li><ul><li>the first monitor is <tt>1</tt> </li><li>use <tt>panel_monitor = all</tt> to get one panel per monitor (unless it's included into another one) </li></ul></ul><p> </p><blockquote><img src="3265475271_81e8ed56e0.jpg">
</blockquote><p></p><p> </p><ul><li><tt>panel_position = vertical_position horizontal_position orientation</tt> </li><ul><li><tt>value for vertical_position : bottom, top, center</tt> </li><li><tt>value for horizontal_position : left, right, center</tt> </li><li><tt>value for orientation : horizontal, vertical</tt> </li></ul></ul> <ul><li><tt>panel_size = width height</tt> </li><ul><li><tt>width = 0</tt> to get full monitor width </li><li>Use <tt>%</tt> to adjust the panel size to monitor size. </li></ul></ul> <blockquote>Example:
<p></p><pre class="prettyprint"> #The panel's width is 94% the size of the monitor
panel_size = 94% 30</pre></blockquote><ul><li><tt>panel_items = LTSBC</tt> define the items tint2 will show and the order of those items. panel_items = STC will show the Systray, the taskbar and the clock (from left to right). SVN ONLY. </li><ul><li>L to show Launcher </li><li>T to show Taskbar </li><li>S to show Systray (also called notification area) </li><li>B to show Battery status </li><li>C to show Clock </li></ul></ul><ul><li><tt>panel_margin = horizontal_margin vertical_margin</tt> </li><ul><li><tt>horizontal_margin = 0</tt> to get full monitor width </li></ul></ul><blockquote><img src="3266303192_476a188196.jpg">
</blockquote><p> </p><ul><li><tt>panel_padding = horizontal_left_right_padding vertical_padding horizontal_spacing</tt> </li></ul><p></p><ul><li><tt>font_shadow = boolean (0 or 1)</tt> </li></ul><ul><li><tt>panel_background_id = integer</tt> : Which background to use </li></ul><ul><li><tt>wm_menu = boolean (0 or 1)</tt> : Allow tint2 to forward mouse event to your Window Manager. </li></ul><ul><li><tt>panel_dock = boolean (0 or 1)</tt> : Put tint2 into the window managers dock. For the openbox window manager I can advise to use also a modifier for the moveButton option, otherwise the mouse click is not forwarded to tint2 (in ~/.config/openbox/rc.xml). </li></ul><p> </p><ul><li><tt>panel_layer = bottom,normal,top</tt> : Puts tint2 into the bottom/normal/top layer. Default is the bottom layer, but for real transparency normal or top layer may be a nice alternative. </li></ul><p></p><ul><li><tt>strut_policy = follow_size/minimum/none</tt> : STRUTs are used by the window manager to decide the size of maximized windows. Should the 'maximized windows' follow tint2 size (follow_size) or use the minimum size (minimum), or use the screen size (none). <tt>minimum</tt> option is interresting when used with <tt>autohide_height</tt> parameter. </li></ul><h1><a name="Launcher_SVN_Only"></a>Launcher SVN Only<a href="#Launcher_SVN_Only" class="section_anchor"></a></h1><ul><li><tt>launcher_icon_theme = name_of_theme</tt> tint2 will follow the theme of your desktop when you have an XSETTINGS manager running. Otherwise, <tt>launcher_icon_theme</tt> parameter is used. </li><li><tt>launcher_padding = horizontal_left_right_padding vertical_padding horizontal_spacing</tt> </li><li><tt>launcher_background_id = integer</tt> : Which background to use </li><li><tt>launcher_icon_size = number</tt> : Set the launcher icon size to <tt>number</tt>. </li><li><tt>launcher_item_app = path_to_application</tt> : Each launcher_item_app must be a full path to a .desktop file. </li></ul><h1><a name="Taskbar_(pager_like_capability)"></a>Taskbar (pager like capability)<a href="#Taskbar_%28pager_like_capability%29" class="section_anchor"></a></h1><ul><li><tt>taskbar_mode = single_desktop or multi_desktop</tt> </li><ul><li><tt>taskbar_mode = single_desktop</tt> : Show one(1) taskbar (current desktop) </li><li><tt>taskbar_mode = multi_desktop</tt> : Pager like capability. Show one(1) taskbar per desktop </li><ul><li>You can drag-and-drop task between desktops` </li><li>And also click on an empty taskbar to switch desktops` </li></ul></ul></ul><p> </p><ul><li><tt>taskbar_padding = horizontal_left_right_padding vertical_padding horizontal_spacing</tt> </li></ul><p></p><ul><li><tt>taskbar_background_id = integer</tt> : Which background to use </li></ul><ul><li><tt>taskbar_active_background_id = integer</tt> : Which background to use for current desktop. </li></ul><blockquote><img src="3266303292_14d9b6b623.jpg">
</blockquote><ul><li><tt>taskbar_name = boolean (0 or 1)</tt> : SVN Only enabled(=1) or disabled(=0) desktop name in taskbar </li></ul><ul><li><tt>taskbar_name_padding = left_right_padding</tt> SVN Only </li></ul><ul><li><tt>taskbar_name_background_id = integer</tt> : SVN Only Which background to use </li></ul><ul><li><tt>taskbar_name_active_background_id = integer</tt> : SVN Only Which background to use for current desktop. </li></ul><ul><li><tt>taskbar_name_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> SVN Only </li></ul><ul><li><tt>taskbar_name_font_color = color opacity (0 to 100)</tt> SVN Only </li></ul><ul><li><tt>taskbar_name_active_font_color = color opacity (0 to 100)</tt> </li></ul><h1><a name="Task"></a>Task<a href="#Task" class="section_anchor"></a></h1><ul><li><tt>urgent_nb_of_blink = integer</tt> : Number of blink on 'get attention' event </li></ul><ul><li><tt>task_icon = boolean (0 or 1)</tt> : Display task icon or not </li></ul><ul><li><tt>task_text = boolean (0 or 1)</tt> : Display task text or not </li></ul><ul><li><tt>task_centered = boolean (0 or 1)</tt> : Task name centered or not </li></ul><ul><li><tt>task_maximum_size = width height</tt> </li><ul><li><tt>width is used with horizontal panel. width = 0 to get full taskbar width</tt> </li><li><tt>height is used with vertical panel</tt> </li></ul></ul><p> </p><ul><li><tt>task_padding = horizontal_padding vertical_padding</tt> </li></ul><p></p><ul><li><tt>task_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><ul><li><tt>task_font_color = color opacity (0 to 100)</tt> </li></ul><p> </p><ul><li><tt>task_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</tt> : adjust icon's color and transparency for task. </li></ul><p></p><ul><li><tt>task_background_id = integer</tt> : Which background to use for non selected tasks </li></ul><blockquote>For the next 3 options STATUS can be <tt>active</tt> / <tt>iconified</tt> / <tt>urgent</tt>.
</blockquote><ul><li><tt>task_STATUS_font_color = color opacity (0 to 100)</tt> </li></ul><ul><li><tt>task_STATUS_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</tt> : adjust icon's color and transparency for active task </li></ul><ul><li><tt>task_STATUS_background_id = integer</tt> : Which background to use when task is selected </li></ul><ul><li><tt>task_tooltip = boolean (0 or 1)</tt> : enabled(=1) or disabled(=0) tooltip on task </li></ul><blockquote><img src="3265475337_262799ded3.jpg">
</blockquote><h1><a name="System_Tray"></a>System Tray<a href="#System_Tray" class="section_anchor"></a></h1><ul><li><tt>systray = boolean (0 or 1)</tt> : enabled(=1) or disabled(=0) systray </li></ul><ul><li><tt>systray_padding = horizontal_left_right_padding vertical_padding horizontal_spacing</tt> </li></ul><ul><li><tt>systray_background_id = integer</tt> : Which background to use for non selected tasks </li></ul><ul><li> <tt>systray_sort = ascending/descending/left2right/right2left</tt> : Sorting order for icon's systray (if left2right or right2left the order can be different on restart) </li></ul><ul><li> <tt>systray_icon_size = max_icon_size</tt> : Set the maximum system tray icon size to <tt>number</tt>. Set to <tt>0</tt> for automatic icon size </li></ul><ul><li><tt>systray_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</tt> : adjust icon's color and transparency for system tray icons </li></ul><h1><a name="Clock"></a>Clock<a href="#Clock" class="section_anchor"></a></h1><ul><li><tt>time1_format = %H:%M</tt> </li><ul><li>time1_format, time2_format and clock_tooltip use 'strftime' syntax. more info on parameters on <a href="http://www.manpagez.com/man/3/strftime/" rel="nofollow">http://www.manpagez.com/man/3/strftime/</a> </li><li>comment time1_format and time2_format to hide clock </li></ul></ul><p> </p><ul><li><tt>time1_timezone = :US/Hawaii</tt> </li><ul><li>time1_timezone, time2_timezone and clock_tooltip_timezone can be used to specify a timezone. If you do not specify a value the system-wide timezone is used. The timezones can usually be found in /usr/share/zoneinfo. If your timezones are in a different directory, you need to specify the absolute path, i.e. time1_timezone = :/different/zoneinfo/dir/US/Hawaii (always prepend with a ':') </li></ul></ul><p></p><ul><li><tt>time1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><ul><li><tt>time2_format = %A %d %B</tt> </li></ul><ul><li><tt>time2_timezone = :Europe/Berlin</tt> </li></ul><ul><li><tt>time2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><ul><li><tt>clock_font_color = color opacity (0 to 100)</tt> </li></ul><ul><li><tt>clock_padding = horizontal_padding vertical_padding</tt> </li></ul><ul><li><tt>clock_background_id = integer</tt> : Which background to use </li></ul><ul><li><tt>clock_tooltip = %a, %d. %b %Y : clock tooltip text</tt> </li></ul><ul><li><tt>clock_tooltip_timezone = :UTC</tt> </li></ul><ul><li><tt>clock_lclick_command = text</tt> : Which command to execute on left click </li></ul><ul><li><tt>clock_rclick_command = text</tt> : Which command to execute on right click </li></ul><h1><a name="Tooltip"></a>Tooltip<a href="#Tooltip" class="section_anchor"></a></h1><ul><li><tt>tooltip_padding = horizontal_padding vertical_padding</tt> </li></ul><ul><li><tt>tooltip_show_timeout = float</tt> : delay to show tooltip in second. Use '.' as decimal separator. </li></ul><ul><li><tt>tooltip_hide_timeout = float</tt> : delay to close tooltip in second. Use '.' as decimal separator. </li></ul><ul><li><tt>tooltip_background_id = integer</tt> : Which background to use (in fake transparency the alpha channel and rounded are not respected) </li></ul><ul><li><tt>tooltip_font_color = color opacity (0 to 100)</tt> </li></ul><ul><li><tt>tooltip_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><h1><a name="Battery"></a>Battery<a href="#Battery" class="section_anchor"></a></h1><ul><li><tt>battery = boolean (0 or 1)</tt> : enabled(=1) or disabled(=0) battery applet </li></ul><ul><li><tt>battery_hide = never or an integer (0 to 100)</tt>: At what battery percent does the battery hide </li></ul><ul><li><tt>battery_low_status = integer</tt>: At what battery percent does the icon show battery low command </li></ul><ul><li><tt>battery_low_cmd = notify-send "battery low"</tt> :What to do when battery is low </li></ul><ul><li><tt>bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><ul><li><tt>bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</tt> </li></ul><ul><li><tt>battery_font_color = color opacity (0 to 100)</tt> </li></ul><ul><li><tt>battery_padding = horizontal_padding vertical_padding</tt> </li></ul><ul><li><tt>battery_background_id = integer</tt> : Which background to use </li></ul><h1><a name="Mouse_action_on_task"></a>Mouse action on task<a href="#Mouse_action_on_task" class="section_anchor"></a></h1><blockquote>Customize mouse action with : <tt>none, close, toggle, iconify, shade, toggle_iconify, maximize_restore, desktop_left, desktop_right, next_task, prev_task</tt>.
</blockquote><blockquote>Sample format:
<pre class="prettyprint"> mouse_middle = none
mouse_right = close
mouse_scroll_up = toggle
mouse_scroll_down = iconify</pre></blockquote><pre class="prettyprint"> close : close the task
toggle : toggle the task
iconify : iconify the task
toggle_iconify : toggle or iconify the task
maximize_restore : maximized or minimized the task
desktop_left : send the task to the desktop on the left
desktop_right : send the task to the desktop on the right
next_task : send the focus to next task
prev_task : send the focus to previous task</pre><p>If a mouse action is set to <tt>none</tt> and <tt>wm_menu = 1</tt> is set, the mouse click is forwarded to the window manager </p><h1><a name="Panel_autohide"></a>Panel autohide<a href="#Panel_autohide" class="section_anchor"></a></h1><ul><li><tt>autohide = boolean (0 or 1)</tt> : enable(=1) or disable(=0) autohiding </li></ul><ul><li><tt>autohide_show_timeout = float</tt> : show timeout in seconds. Use '.' as decimal separator. </li></ul><ul><li><tt>autohide_hide_timeout = float</tt> : hide timeout in seconds. Use '.' as decimal separator. </li></ul><ul><li><tt>autohide_height = integer</tt> : panel height (width for vertical panels) in hidden mode </li></ul><h1><a name="An_example_tint2rc"></a>An example tint2rc<a href="#An_example_tint2rc" class="section_anchor"></a></h1><pre class="prettyprint">#---------------------------------------------
# TINT2 CONFIG FILE
#---------------------------------------------
#---------------------------------------------
# BACKGROUND AND BORDER
#---------------------------------------------
rounded = 7
border_width = 2
background_color = #000000 60
border_color = #ffffff 18
rounded = 5
border_width = 0
background_color = #ffffff 40
border_color = #ffffff 50
rounded = 5
border_width = 0
background_color = #ffffff 18
border_color = #ffffff 70
#---------------------------------------------
# PANEL
#---------------------------------------------
panel_monitor = all
panel_position = bottom center
panel_size = 94% 30
panel_margin = 0 0
panel_padding = 7 0
font_shadow = 0
panel_background_id = 1
wm_menu = 0
panel_dock = 0
panel_layer = bottom
#---------------------------------------------
# TASKBAR
#---------------------------------------------
#taskbar_mode = multi_desktop
taskbar_mode = single_desktop
taskbar_padding = 2 3 2
taskbar_background_id = 0
#taskbar_active_background_id = 0
#---------------------------------------------
# TASKS
#---------------------------------------------
task_icon = 1
task_text = 1
task_maximum_size = 140 35
task_centered = 1
task_padding = 6 3
task_font = sans 7
task_font_color = #ffffff 70
task_background_id = 3
task_icon_asb = 100 0 0
# replace STATUS by 'urgent', 'active' or 'iconfied'
#task_STATUS_background_id = 2
#task_STATUS_font_color = #ffffff 85
#task_STATUS_icon_asb = 100 0 0
# example:
task_active_background_id = 2
task_active_font_color = #ffffff 85
task_active_icon_asb = 100 0 0
urgent_nb_of_blink = 8
#---------------------------------------------
# SYSTRAYBAR
#---------------------------------------------
systray = 1
systray_padding = 0 4 5
systray_background_id = 0
systray_sort = left2right
systray_icon_size = 0
systray_icon_asb = 100 0 0
#---------------------------------------------
# CLOCK
#---------------------------------------------
time1_format = %H:%M
time1_font = sans 8
time2_format = %A %d %B
time2_font = sans 6
clock_font_color = #ffffff 76
clock_padding = 1 0
clock_background_id = 0
#clock_lclick_command = xclock
clock_rclick_command = orage
#clock_tooltip = %A %d %B
#time1_timezone = :US/Hawaii
#time2_timezone = :Europe/Berlin
#clock_tooltip_timezone = :/usr/share/zoneinfo/Europe/Paris
#---------------------------------------------
# BATTERY
#---------------------------------------------
battery = 0
battery_hide = 98
battery_low_status = 10
battery_low_cmd = notify-send "battery low"
bat1_font = sans 8
bat2_font = sans 6
battery_font_color = #ffffff 76
battery_padding = 1 0
battery_background_id = 0
#---------------------------------------------
# TOOLTIP
#---------------------------------------------
tooltip = 0
tooltip_padding = 2 2
tooltip_show_timeout = 0.7
tooltip_hide_timeout = 0.3
tooltip_background_id = 1
tooltip_font_color = #OOOOOO 80
tooltip_font = sans 10
#---------------------------------------------
# MOUSE ACTION ON TASK
#---------------------------------------------
mouse_middle = none
mouse_right = close
mouse_scroll_up = toggle
mouse_scroll_down = iconify
#---------------------------------------------
# AUTOHIDE OPTIONS
#---------------------------------------------
autohide = 0
autohide_show_timeout = 0.3
autohide_hide_timeout = 2
autohide_height = 4
strut_policy = minimum</pre>
</div>
</div>
</td></tr><tr>
</tr></tbody></table>
</div>
<div id="wikicommentcol">
<div class="collapse">
<div id="commentlist">
</div>
</div>
</div>
</div>
<div class="hostedBy" style="margin-top: -20px;">
</div>
</body></html>

View File

@@ -1,217 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>tint2 - FAQ</title>
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
</head>
<body class="t6">
<table class="st" width="100%" cellpadding="0" cellspacing="0" align="center" border="0">
<tbody><tr>
<td class="bevel-right" valign="top" align="right"></td>
</tr>
</tbody></table>
<div id="maincol" style="padding: 0pt 3px 3px 0pt; margin: 0pt;">
<!-- IE -->
<div id="wikipage">
<table>
<tbody><tr>
<td style="vertical-align: top; padding-left: 5px;">
<div id="wikicontent">
<div class="vt" id="wikimaincol">
<h1>tint2 - FAQ</h1>
<p></p><ul><li><a href="#What_is_tint2?">What is tint2?</a></li><li><a href="#Who_is_the_developer_of_tint2_?">Who is the developer of tint2 ?</a></li><li><a href="#Does_tint2_have_a_system_tray/notification_area?">Does tint2 have a system tray/notification area?</a></li><li><a href="#I_have_the_latest_build_of_tint2_but_still_no_system_tray">I have the latest build of tint2 but still no system tray</a></li><li><a href="#tint2_doesn%27t_show_the_systray/notification">tint2 doesn't show the systray/notification</a></li><li><a href="#Why_did_some_systray_icons_disappear_from_the_panel?">Why did some systray icons disappear from the panel?</a></li><li><a href="#Why_don%27t_some_systray_icons_appear_when_you_run_tint2_in_a">Why don't some systray icons appear when you run tint2 in a KDE session?</a></li><li><a href="#I_want_to_disable_the_tint2_systray">I want to disable the tint2 systray</a></li><li><a href="#How_to_configure_real_transparency">How to configure real transparency</a></li><li><a href="#Tint2_doesn%27t_work_on_compiz_correctly?">Tint2 doesn't work on compiz correctly?</a></li><li><a href="#I_want_to_disable_compiz_shadow_on_Tint2">I want to disable compiz shadow on Tint2</a></li><li><a href="#Adding_a_%27show_desktop%27_button">Adding a 'show desktop' button</a></li><li><a href="#Task_list_is_limited">Task list is limited</a></li><li><a href="#How_can_I_get_%27pager_like_capability%27_?">How can I get 'pager like capability' ?</a></li><li><a href="#I_want_to_hide_some_tasks_from_the_panel_?">I want to hide some tasks from the panel ?</a></li><li><a href="#The_tint2_panel_is_blank">The tint2 panel is blank</a></li><li><a href="#How_to_maximize_window_over_tint2">How to maximize window over tint2</a></li><li><a href="#I_get_errors_when_I_try_to_compile_tint2_on_my_own">I get errors when I try to compile tint2 on my own</a></li><li><a href="#I_upgraded_to_tint2_and_I_still_have_the_old_panel_with_its_bugs">I upgraded to tint2 and I still have the old panel with its bugs</a></li><li><a href="#Tint2_doesn%27t_show_in_Pekwm">Tint2 doesn't show in Pekwm</a></li><li><a href="#Is_there_an_IRC_channel?">Is there an IRC channel?</a></li></ul> <p></p><p>Frequently asked questions. Please add any extra tips to the comments. </p><h1><a name="What_is_tint2?"></a>What is tint2?<a href="#What_is_tint2?" class="section_anchor"></a></h1><ul><li>Tint2 is a panel/taskbar for modern x window managers like GNOME, KDE, XFCE, openbox, pekwm,etc.. </li></ul><h1><a name="Who_is_the_developer_of_tint2_?"></a>Who is the developer of tint2 ?<a href="#Who_is_the_developer_of_tint2_?" class="section_anchor"></a></h1><ul><li>Main developers of tint2 are Thierry Lorthiois and Andreas Fink </li></ul><h1><a name="Does_tint2_have_a_system_tray/notification_area?"></a>Does tint2 have a system tray/notification area?<a href="#Does_tint2_have_a_system_tray/notification_area?" class="section_anchor"></a></h1><ul><li>YES! Tint2 finally has system tray/notification area. Please <a href="https://code.google.com/p/tint2/wiki/Install">install</a> the latest build if your current build doesn't have it. </li></ul><h1><a name="I_have_the_latest_build_of_tint2_but_still_no_system_tray"></a>I have the latest build of tint2 but still no system tray<a href="#I_have_the_latest_build_of_tint2_but_still_no_system_tray" class="section_anchor"></a></h1><ul><li>Check your tint2rc to see if you have the systray option configured and the options are not commented out. </li><li>The systray section looks like this </li><pre class="prettyprint">#---------------------------------------------
# SYSTRAYBAR
#---------------------------------------------
systray = 1
systray_padding = 0 3 3
systray_background_id = 0
systray_sort = left2right</pre></ul><h1><a name="tint2_doesn't_show_the_systray/notification"></a>tint2 doesn't show the systray/notification<a href="#tint2_doesn%27t_show_the_systray/notification" class="section_anchor"></a></h1><ul><li>Check your tint2rc to see if you have 'systray = 1' </li><li>run tint2 in a terminal. if you get 'tint2 : another systray is running' it's because you have another systray/notification and you can't run more than one at a time. </li><ul><li>In gnome, remove the systray/notification with a right clic on the separator before your systray/notification -&gt; remove </li></ul></ul><h1><a name="Why_did_some_systray_icons_disappear_from_the_panel?"></a>Why did some systray icons disappear from the panel?<a href="#Why_did_some_systray_icons_disappear_from_the_panel?" class="section_anchor"></a></h1><ul><li>tint2 stops/restarts the systray protocol on some events (like switching theme with tint2conf). But some systray icons do not implement the stop/restart of the manager and so disappear from the panel. You can report the problem to us and also to the systray icon developer. </li></ul><h1><a name="Why_don't_some_systray_icons_appear_when_you_run_tint2_in_a"></a>Why don't some systray icons appear when you run tint2 in a KDE session?<a href="#Why_don%27t_some_systray_icons_appear_when_you_run_tint2_in_a" class="section_anchor"></a></h1><ul><li>Some KDE applications use the KDE status notifier service to show system tray icons. If you disable this service (systemsettings -&gt; Startup and Shutdown -&gt; Service Manager -&gt; uncheck "Status Notifier Manager") all the icons should appear normally in tint2. </li></ul><h1><a name="I_want_to_disable_the_tint2_systray"></a>I want to disable the tint2 systray<a href="#I_want_to_disable_the_tint2_systray" class="section_anchor"></a></h1><ul><li>To disable the systray, just put <tt>systray = 0</tt> in your tint2rc </li><li>Example: </li><pre class="prettyprint">systray = 0
systray_padding = 0 3 3
systray_background_id = 0</pre></ul><h1><a name="How_to_configure_real_transparency"></a>How to configure real transparency<a href="#How_to_configure_real_transparency" class="section_anchor"></a></h1><ul><li>tint2 autodetects your composite manager and uses real transparency when possible. However, xcompmgr have a little bug which prevents tint2 from detecting it. </li><li>If you want autodetect with xcompmgr, download and apply xcompmgr's patch (in bug report). See <a href="http://bugs.freedesktop.org/show_bug.cgi?id=26090" rel="nofollow">xcompmgr patch</a> </li></ul><h1><a name="Tint2_doesn't_work_on_compiz_correctly?"></a>Tint2 doesn't work on compiz correctly?<a href="#Tint2_doesn%27t_work_on_compiz_correctly?" class="section_anchor"></a></h1><ul><li>All my tasks are on one desktop in compiz? </li><ul><li>This is a known issue in compiz. Compiz doesn't support 'Virtual desktop' specification (or you have to disabled visual effect).<br>I look like 'Desktop number' use Viewport instead of 'Virtual desktop'. So pager capabilities are not supported. <br>See <a href="https://bugs.launchpad.net/ubuntu/+source/compiz/+bug/153322" rel="nofollow">https://bugs.launchpad.net/ubuntu/+source/compiz/+bug/153322</a> <br>or <a href="http://forum.compiz.org/viewtopic.php?f=86&amp;t=11943" rel="nofollow">http://forum.compiz.org/viewtopic.php?f=86&amp;t=11943</a> <br>or <a href="https://bugs.launchpad.net/ubuntu/+source/compiz/+bug/150690" rel="nofollow">https://bugs.launchpad.net/ubuntu/+source/compiz/+bug/150690</a> </li></ul><li>Tint is being decorated with a window/ has a shadow /is not fully transparent </li><ul><li>Open CompizConfig Settings manager -&gt; Window Decorations Plugin </li><ul><li>In the box for Decoration windows, put <tt>any &amp; !type=dock</tt> </li><li>In the box for Shadow windows, put <tt>any &amp; !type=dock</tt> </li></ul></ul><blockquote>
</blockquote><li>Tint has ugly white streaks </li><ul><li>Set a background for your desktop </li></ul></ul><h1><a name="I_want_to_disable_compiz_shadow_on_Tint2"></a>I want to disable compiz shadow on Tint2<a href="#I_want_to_disable_compiz_shadow_on_Tint2" class="section_anchor"></a></h1><ul><li>Run ccsm the compiz configuration tool. Go to Window Decorations. For both decoration and shadow windows, replace 'any' with 'any &amp; !name=tint2'. </li></ul><h1><a name="Adding_a_'show_desktop'_button"></a>Adding a 'show desktop' button<a href="#Adding_a_%27show_desktop%27_button" class="section_anchor"></a></h1><ul><li>tint2 doesn't implement 'show desktop' button. But here a way to get it. <br> </li></ul><p>Create a file with following text </p><pre class="prettyprint">#!/bin/sh
if wmctrl -m | grep "mode: ON"; then
exec wmctrl -k off
else
exec wmctrl -k on
fi</pre><p>Save it in /usr/bin/show_desktop.sh and make it executable.<br> Install the package 'wmctrl'. And call the script in tint2rc config file. </p><pre class="prettyprint">time1_format = %H:%M
time1_font = sans 8
time2_format = %A %d %B
time2_font = sans 6
clock_font_color = #ffffff 76
clock_padding = 4 4
clock_background_id = 0
clock_lclick_command = show_desktop.sh</pre><p>Then left click on the clock will do it. </p><h1><a name="Task_list_is_limited"></a>Task list is limited<a href="#Task_list_is_limited" class="section_anchor"></a></h1><ul><li>In your tint2rc file, check the value of <tt>taskbar_mode</tt>. If it's <tt>multi_desktop</tt>, tint2 show one taskbar per desktop. Change the value to <tt>single_desktop</tt> if you want just one taskbar. </li></ul><h1><a name="How_can_I_get_'pager_like_capability'_?"></a>How can I get 'pager like capability' ?<a href="#How_can_I_get_%27pager_like_capability%27_?" class="section_anchor"></a></h1><ul><li>In your tint2rc file, change the value of <tt>taskbar_mode</tt> to <tt>multi_desktop</tt>. This option will show one taskbar per desktop. So you can detect on witch desktop is an application and witch desktop is empty (or full). You can close an application whithout switching to the desktop. You can drag n drop application between desktop. </li></ul><h1><a name="I_want_to_hide_some_tasks_from_the_panel_?"></a>I want to hide some tasks from the panel ?<a href="#I_want_to_hide_some_tasks_from_the_panel_?" class="section_anchor"></a></h1><ul><li>In Openbox config file 'rc.xml' you can assign the property 'skip_taskbar' to your window </li></ul><h1><a name="The_tint2_panel_is_blank"></a>The tint2 panel is blank<a href="#The_tint2_panel_is_blank" class="section_anchor"></a></h1><ul><li>A new option, called <tt>panel_items</tt> has been added to the config file. This option defines the items tint2 will show and the order of those items(from left to right). Please define that in your tint2rc file. The format is shown below. The options are <tt>L</tt> for <tt>Launcher</tt>, <tt>S</tt> for the <tt>Systray</tt>, <tt>T</tt> for the <tt>taskbar</tt>, <tt>C</tt> for the <tt>clock</tt> and <tt>B</tt> for the <tt>battery</tt>. </li><pre class="prettyprint">panel_items = LTSBC </pre></ul><h1><a name="How_to_maximize_window_over_tint2"></a>How to maximize window over tint2<a href="#How_to_maximize_window_over_tint2" class="section_anchor"></a></h1><ul><li>You can manage interaction between tint2 and window with : </li><pre class="prettyprint">panel_layer = bottom,normal,top
strut_policy = follow_size/minimum/none
autohide_height = integer</pre></ul><blockquote>If you want maximized window never cover tint2:
<pre class="prettyprint">panel_layer = top
strut_policy = follow_size</pre>If you want maximized window behind tint2:
<pre class="prettyprint">panel_layer = top
strut_policy = none</pre>If you want maximized window behind tint2, but 2 pixels free so you can use desktop action (right click, ...):
<pre class="prettyprint">panel_layer = top
strut_policy = minimum
autohide_height = 2</pre></blockquote><h1><a name="I_get_errors_when_I_try_to_compile_tint2_on_my_own"></a>I get errors when I try to compile tint2 on my own<a href="#I_get_errors_when_I_try_to_compile_tint2_on_my_own" class="section_anchor"></a></h1><ul><li>Do you have all the dependencies installed? Please check <a href="https://code.google.com/p/tint2/wiki/Install#Dependencies">here</a> </li></ul><h1><a name="I_upgraded_to_tint2_and_I_still_have_the_old_panel_with_its_bugs"></a>I upgraded to tint2 and I still have the old panel with its bugs<a href="#I_upgraded_to_tint2_and_I_still_have_the_old_panel_with_its_bugs" class="section_anchor"></a></h1><ul><li>The tint binary was renamed tint2 to avoid a conflict with another package.So your new tint2 startup command is <strong>tint2</strong> </li><li>Since a new stable release hasn't been released yet, both the old tint and the new tint2 binaries will be on your system when you upgrade just incase you run into some problems and want to fall back on the old binary. </li></ul><h1><a name="Tint2_doesn't_show_in_Pekwm"></a>Tint2 doesn't show in Pekwm<a href="#Tint2_doesn%27t_show_in_Pekwm" class="section_anchor"></a></h1><ul><li>The problem is that pekwm defines by default edge borders, and doesn't allow a panel to get inside these edge borders. The config option is in <tt>~/.pekwm/config</tt> under <tt>Screen/EdgeSize</tt>. You can either disable the edge borders or set tint2 to a size where it doesn't enter these borders (i.e. play with <tt>panel_size</tt> and <tt>panel_margin</tt>) </li><li>Setting the margin to one pixel made it appear on my screen. With horizontal panel: </li><pre class="prettyprint">panel_size = 95% height
panel_margin = 0 1</pre><li>With vertical panel: </li><pre class="prettyprint">panel_size = width 95%
panel_margin = 1 0</pre></ul><h1><a name="Is_there_an_IRC_channel?"></a>Is there an IRC channel?<a href="#Is_there_an_IRC_channel?" class="section_anchor"></a></h1><ul><li>Yes. There is now an irc channel, <strong>#tint2</strong> on <a href="irc://irc.freenode.net" rel="nofollow">irc.freenode.net</a>. </li><li>Just ask your question and stay there, because we are not always watching the channel ;) </li></ul>
</div>
</div>
</td></tr><tr>
</tr></tbody></table>
</div>
<div id="wikicommentcol">
<div class="collapse">
</div>
</div>
<div id="commentform">
</div>
</div>
<div class="hostedBy" style="margin-top: -20px;">
</div>
</body></html>

View File

@@ -0,0 +1,23 @@
#!/bin/bash
export PATH=/usr/local/bin:/usr/bin:/bin:/usr/local/sbin:/usr/sbin:/sbin
set -e
set -x
[ "${FLOCKER}" != "$0" ] && exec env FLOCKER="$0" flock -en "$0" "$0" "$@" || :
exec > ~/tint2.runner-version.log
exec 2>&1
cd ~/tint2.wiki
git reset --hard
git pull
~/tint2/packaging/version_status.py > packaging.tmp.md
cat packaging.tmp.md > packaging.md
rm packaging.tmp.md
git commit -am 'Update packaging info'
git push origin master

422
packaging/version_status.py Executable file
View File

@@ -0,0 +1,422 @@
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
import sys
reload(sys)
sys.setdefaultencoding('utf8')
import datetime
import xml.etree.ElementTree as ET
import ftplib
import gzip
import json
import re
from StringIO import StringIO
import urllib2
# Returns true if `value` is an integer represented as a string.
def is_int(value):
# type: (str) -> bool
try:
value = int(value)
except ValueError:
return False
return True
# Returns a new string with all instances of multiple whitespace
# replaced with a single space.
def collapse_multiple_spaces(line):
# type: (str) -> str
return " ".join(line.split())
# Extracts the file name from a line of an FTP listing.
# The input must be a valid directory entry (starting with "-" or "d").
def ftp_file_name_from_listing(line):
# type: (str) -> str
line = collapse_multiple_spaces(line)
return line.split(" ", 8)[-1]
# Extracts a list of the directories and a list of the files
# from an FTP listing.
def ftp_list_dir_process_listing(lines):
# type: (List[str]) -> List[str], List[str]
dirs = []
files = []
for line in lines:
if line.startswith("d"):
dirs.append(ftp_file_name_from_listing(line))
elif line.startswith("-"):
files.append(ftp_file_name_from_listing(line))
return dirs, files
# Lists the remote FTP directory located at `path`.
# Returns a list of the directories and a list of the files.
def ftp_list_dir(ftp, path):
# type: (ftplib.FTP, str) -> List[str], List[str]
ftp.cwd(path)
lines = []
ftp.retrlines("LIST", lines.append)
return ftp_list_dir_process_listing(lines)
# Downloads a binary file to a string.
# Returns the string.
def ftp_download(ftp, path):
# type: (ftplib.FTP, str) -> str
blocks = []
ftp.retrbinary("RETR {0}".format(path), blocks.append)
return "".join(blocks)
# Extracts the list of links from an HTML string.
def http_links_from_listing(html):
# type: (str) -> List[str]
pattern = re.compile(r"""href=['"]+([^'"]+)['"]+""")
return re.findall(pattern, html)
# Extracts the list of paths (relative links, except to ../*) from an HTML string.
def http_paths_from_listing(html):
# type: (str) -> List[str]
paths = []
for link in http_links_from_listing(html):
if link.startswith(".."):
continue
if link == "./" or link == "/":
continue
if "://" in link:
continue
paths.append(link)
return paths
# Downloads a file as string from an URL. Decodes correctly.
def http_download_txt(url):
# type: (str) -> str
try:
r = urllib2.urlopen(url)
encoding = r.headers.getparam("charset")
if not encoding:
encoding = "utf-8"
return r.read().decode(encoding)
except:
raise
# Extracts the list of paths (relative links, except to ../*) from the HTML code
# located at `url`.
def http_list_dir(url):
# type: (str) -> List[str]
try:
html = http_download_txt(url)
except:
return []
return http_paths_from_listing(html)
# Extracts the version and maintainer info for a package, from a Debian repository Packages file.
def deb_packages_extract_version(packages, name):
# type: (str, str) -> str, str
inside = False
version = None
maintainer = None
for line in packages.split("\n"):
if line == "Package: " + name:
inside = True
elif not line:
if inside:
break
else:
if inside:
if line.startswith("Version:"):
version = line.split(":", 1)[-1].strip()
elif line.startswith("Maintainer:"):
maintainer = line.split(":", 1)[-1].strip()
return version, maintainer
# Extracts the version and maintainer info for a package, from an Arch PKGBUILD file.
def arch_pkgbuild_extract_version(pkgbuild):
# type: (str) -> str, str
version = None
maintainer = None
for line in pkgbuild.split("\n"):
if line.startswith("# Maintainer:"):
maintainer = line.split(":", 1)[-1].strip()
elif line.startswith("pkgver="):
version = line.split("=", 1)[-1].strip()
return version, maintainer
# Debian
def get_debian_release_version(release):
data = http_download_txt("http://metadata.ftp-master.debian.org/changelogs/main/t/tint2/{0}_changelog".format(release))
version = data.split("\n", 1)[0].split("(", 1)[-1].split(")", 1)[0].strip()
maintainer = [line.split("--", 1)[-1] for line in data.split("\n") if line.startswith(" --")][0].split(" ")[0].strip()
return release, version, maintainer
def get_debian_versions():
print >> sys.stderr, "Debian ..."
return "Debian", "debian", [get_debian_release_version(release) for release in ["stable", "testing", "unstable", "experimental"]]
# Ubuntu
def get_ubuntu_versions():
print >> sys.stderr, "Ubuntu ..."
data = http_download_txt("https://api.launchpad.net/1.0/ubuntu/+archive/primary?ws.op=getPublishedSources&source_name=tint2&exact_match=true")
data = json.loads(data)["entries"]
data.reverse()
versions = []
for package in data:
if package["status"] == "Published":
version = package["source_package_version"]
release = package["distro_series_link"].split("/")[-1]
maintainer = package["package_maintainer_link"]
versions.append((release, version, maintainer))
return "Ubuntu", "ubuntu", versions
# BunsenLabs
def get_bunsenlabs_versions():
print >> sys.stderr, "BunsenLabs ..."
dirs = http_list_dir("https://eu.pkg.bunsenlabs.org/debian/dists/")
versions = []
for d in dirs:
if d.endswith("/") and "/" not in d[:-1]:
release = d.replace("/", "")
packages = http_download_txt("https://eu.pkg.bunsenlabs.org/debian/dists/{0}/main/binary-amd64/Packages".format(release))
version, maintainer = deb_packages_extract_version(packages, "tint2")
if version:
versions.append((release, version, maintainer))
return "BunsenLabs", "bunsenlabs", versions
# Arch
def get_arch_versions():
print >> sys.stderr, "Arch ..."
pkgbuild = http_download_txt("https://git.archlinux.org/svntogit/community.git/plain/trunk/PKGBUILD?h=packages/tint2")
version, maintainer = arch_pkgbuild_extract_version(pkgbuild)
return "Arch Linux", "archlinux", [("Community", version, maintainer)]
# Fedora
def get_fedora_versions():
print >> sys.stderr, "Fedora ..."
dirs = http_list_dir("http://mirror.switch.ch/ftp/mirror/fedora/linux/development/")
versions = []
for d in dirs:
if d.endswith("/") and "/" not in d[:-1]:
release = d.replace("/", "")
packages = http_list_dir("http://mirror.switch.ch/ftp/mirror/fedora/linux/development/{0}/Everything/source/tree/Packages/t/".format(release))
for p in packages:
if p.startswith("tint2-"):
version = p.split("-", 1)[-1].split(".fc")[0]
v = (release, version, "")
if v not in versions:
versions.append(v)
return "Fedora", "fedora", versions
# Red Hat (EPEL)
def get_redhat_epel_versions():
print >> sys.stderr, "RedHat ..."
dirs = http_list_dir("http://mirror.switch.ch/ftp/mirror/epel/")
versions = []
for d in dirs:
if d.endswith("/") and "/" not in d[:-1] and is_int(d[:-1]):
release = d.replace("/", "")
packages = http_list_dir("http://mirror.switch.ch/ftp/mirror/epel/{0}/SRPMS/t/".format(release))
for p in packages:
if p.startswith("tint2-"):
version = p.split("-", 1)[-1].split(".el")[0]
v = (release, version, "")
if v not in versions:
versions.append(v)
return "RedHat (EPEL)", "rhel", versions
# SUSE
def get_suse_versions():
print >> sys.stderr, "Suse ..."
ftp = ftplib.FTP("mirror.switch.ch")
ftp.login()
releases, _ = ftp_list_dir(ftp, "/mirror/opensuse/opensuse/ports/update/leap/")
versions = []
for release in releases:
root = "/mirror/opensuse/opensuse/ports/update/leap/{0}/oss/repodata/".format(release)
_, files = ftp_list_dir(ftp, root)
for fname in files:
if fname.endswith("-primary.xml.gz"):
data = ftp_download(ftp, "{0}/{1}".format(root, fname))
xml = gzip.GzipFile(fileobj=StringIO(data)).read()
root = ET.fromstring(xml)
for package in root.findall("{http://linux.duke.edu/metadata/common}package"):
name = package.find("{http://linux.duke.edu/metadata/common}name").text
if name == "tint2":
version = package.find("{http://linux.duke.edu/metadata/common}version").get("ver")
versions.append((release, version, ""))
ftp.quit()
return "OpenSUSE", "opensuse", versions
# Gentoo
def get_gentoo_versions():
print >> sys.stderr, "Gentoo ..."
files = http_list_dir("https://gitweb.gentoo.org/repo/gentoo.git/tree/x11-misc/tint2")
versions = []
for f in files:
if "tint2" in f and f.endswith(".ebuild"):
version = f.split("tint2-")[-1].split(".ebuild")[0]
v = ("", version, "")
if v not in versions:
versions.append(v)
return "Gentoo", "gentoo", versions
# Void
def get_void_versions():
print >> sys.stderr, "Void ..."
template = http_download_txt("https://raw.githubusercontent.com/voidlinux/void-packages/master/srcpkgs/tint2/template")
versions = []
version = None
maintainer = None
for line in template.split("\n"):
if line.startswith("version="):
version = line.split("=", 1)[-1].replace('"', "").strip()
elif line.startswith("maintainer="):
maintainer = line.split("=", 1)[-1].replace('"', "").strip()
if version:
versions.append(("", version, maintainer))
return "Void Linux", "void", versions
# Alpine
def get_alpine_versions():
print >> sys.stderr, "Alpine ..."
apkbuild = http_download_txt("https://git.alpinelinux.org/cgit/aports/plain/community/tint2/APKBUILD")
versions = []
version = None
maintainer = None
for line in apkbuild.split("\n"):
if line.startswith("pkgver="):
version = line.split("=", 1)[-1].replace('"', "").strip()
elif line.startswith("# Maintainer:"):
maintainer = line.split(":", 1)[-1].replace('"', "").strip()
if version:
versions.append(("", version, maintainer))
return "Alpine Linux", "alpine", versions
# Slackware
def get_slack_versions():
print >> sys.stderr, "Slackware ..."
dirs = http_list_dir("https://slackbuilds.org/slackbuilds/")
versions = []
for d in dirs:
if d.endswith("/") and "/" not in d[:-1]:
release = d.replace("/", "")
try:
info = http_download_txt("https://slackbuilds.org/slackbuilds/{0}/desktop/tint2/tint2.info".format(release))
except:
continue
version = None
maintainer = None
for line in info.split("\n"):
if line.startswith("VERSION="):
version = line.split("=", 1)[-1].replace('"', "").strip()
elif line.startswith("MAINTAINER="):
maintainer = line.split("=", 1)[-1].replace('"', "").strip()
if version:
versions.append((release, version, maintainer))
return "Slackware", "slackware", versions
# FreeBSD
def get_freebsd_versions():
print >> sys.stderr, "FreeBSD ..."
makefile = http_download_txt("https://svnweb.freebsd.org/ports/head/x11/tint/Makefile?view=co")
versions = []
version = None
maintainer = None
for line in makefile.split("\n"):
if line.startswith("PORTVERSION="):
version = line.split("=", 1)[-1].strip()
elif line.startswith("MAINTAINER="):
maintainer = line.split("=", 1)[-1].strip()
if version:
versions.append(("", version, maintainer))
return "FreeBSD", "freebsd", versions
# OpenBSD
def get_openbsd_versions():
print >> sys.stderr, "OpenBSD ..."
makefile = http_download_txt("http://cvsweb.openbsd.org/cgi-bin/cvsweb/~checkout~/ports/x11/tint2/Makefile?rev=1.5&content-type=text/plain")
versions = []
version = None
for line in makefile.split("\n"):
if line.startswith("V="):
version = line.split("=", 1)[-1].strip()
if version:
versions.append(("", version, ""))
return "OpenBSD", "openbsd", versions
# Upstream
def get_tint2_version():
print >> sys.stderr, "Upstream ..."
readme = http_download_txt("https://gitlab.com/o9000/tint2/raw/master/README.md")
version = readme.split("\n", 1)[0].split(":", 1)[-1].strip()
return version
def main():
latest = get_tint2_version()
distros = []
distros.append(get_debian_versions())
distros.append(get_bunsenlabs_versions())
distros.append(get_ubuntu_versions())
distros.append(get_fedora_versions())
distros.append(get_redhat_epel_versions())
#distros.append(get_suse_versions())
distros.append(get_alpine_versions())
distros.append(get_slack_versions())
distros.append(get_arch_versions())
distros.append(get_void_versions())
distros.append(get_gentoo_versions())
distros.append(get_freebsd_versions())
distros.append(get_openbsd_versions())
print "| Distribution | Release | Version | Status |"
print "| ------------ | ------- | ------- | ------ |"
for dist, dcode, releases in distros:
icon = "![](numix-icons/distributor-logo-{0}.svg.png)".format(dcode)
for r in releases:
if r[1].split("-", 1)[0] == latest:
status = ":white_check_mark: Latest"
else:
status = ":warning: Out of date"
print "| {0} {1} | {2} | {3} | {4} |".format(icon, dist, r[0], r[1], status)
utc_datetime = datetime.datetime.utcnow()
print ""
print "Last updated:", utc_datetime.strftime("%Y-%m-%d %H:%M UTC")
if __name__ == "__main__":
main()

208
packaging/version_status_test.py Executable file
View File

@@ -0,0 +1,208 @@
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
import unittest
from version_status import *
class TestStringFunctions(unittest.TestCase):
def test_collapse_multiple_spaces(self):
self.assertEqual(collapse_multiple_spaces("asdf"), "asdf")
self.assertEqual(collapse_multiple_spaces("as df"), "as df")
self.assertEqual(collapse_multiple_spaces("as df"), "as df")
self.assertEqual(collapse_multiple_spaces("a s d f"), "a s d f")
class TestFtpFunctions(unittest.TestCase):
def test_ftp_file_name_from_listing(self):
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18 README"), "README")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18:12 README"), "README")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18 READ ME"), "READ ME")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18:12 READ ME"), "READ ME")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18 README"), "README")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18:12 README.txt"), "README.txt")
self.assertEqual(ftp_file_name_from_listing("-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18:12 READ ME.txt"), "READ ME.txt")
def test_ftp_list_dir_process_listing(self):
lines = [ "-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18 README",
"-rw-rw-r-- 1 1176 1176 1063 Jun 15 10:18:11 READ ME.txt",
"drwxr-sr-x 5 1176 1176 4096 Dec 19 2000 pool",
"drwxr-sr-x 4 1176 1176 4096 Nov 17 2008 project",
"drwxr-xr-x 3 1176 1176 4096 Oct 10 2012 tools"]
dirs_check = ["pool", "project", "tools"]
files_check = ["README", "READ ME.txt"]
dirs, files = ftp_list_dir_process_listing(lines)
dirs.sort()
dirs_check.sort()
files.sort()
files_check.sort()
self.assertEqual(dirs, dirs_check)
self.assertEqual(files, files_check)
class TestHttpFunctions(unittest.TestCase):
def test_http_links_from_listing(self):
html = """<html>
<head><title>Index of /debian/dists/</title></head>
<body bgcolor="white">
<a href="http://google.com">google/</a>
<h1>Index of /debian/dists/</h1><hr><pre><a href="../">../</a>
<a href="bunsen-hydrogen/">bunsen-hydrogen/</a> 08-May-2017 20:31 -
<a href="jessie-backports/">jessie-backports/</a> 01-Jul-2017 15:58 -
<a href="unstable/">unstable/</a> 12-Aug-2017 19:32 -
</pre><hr></body>
</html>"""
links_check = ["../", "bunsen-hydrogen/", "jessie-backports/", "unstable/", "http://google.com"]
links = http_links_from_listing(html)
links.sort()
links_check.sort()
self.assertEqual(links, links_check)
def test_http_paths_from_listing(self):
html = """<html>
<head><title>Index of /debian/dists/</title></head>
<body bgcolor="white">
<h1>Index of /debian/dists/</h1><hr><pre><a href="../">../</a>
<a href="bunsen-hydrogen/">bunsen-hydrogen/</a> 08-May-2017 20:31 -
<a href="jessie-backports/">jessie-backports/</a> 01-Jul-2017 15:58 -
<a href="unstable/">unstable/</a> 12-Aug-2017 19:32 -
</pre><hr></body>
</html>"""
paths_check = ["bunsen-hydrogen/", "jessie-backports/", "unstable/"]
paths = http_paths_from_listing(html)
paths.sort()
paths_check.sort()
self.assertEqual(paths, paths_check)
class TestPackageFunctions(unittest.TestCase):
def test_deb_packages_extract_version(self):
packages = """Package: sendmailanalyzer
Version: 9.2-1.1
Architecture: all
Maintainer: Dominique Fournier <dominique@fournier38.fr>
Installed-Size: 749
Pre-Depends: perl
Depends: apache2
Homepage: http://sareport.darold.net/
Priority: optional
Section: mail
Filename: pool/main/s/sendmailanalyzer/sendmailanalyzer_9.2-1.1_all.deb
Size: 143576
SHA256: 0edcbde19a23333c8c894e27af32447582b38e3ccd84122ac07720fdaab8fa0c
SHA1: a7f4dcf42e850acf2c201bc4594cb6b765dced20
MD5sum: adb39196fc33a826b24e9d0e440cba25
Description: Perl Sendmail/Postfix log analyser
SendmailAnalyzer continuously read your mail log file to generate
periodical HTML and graph reports. All reports are shown through
a CGI web interface.
It reports all you ever wanted to know about email trafic on your network.
You can also use it in ISP environment with per domain report.
Package: tint2
Version: 0.14.6-1
Architecture: amd64
Maintainer: Jens John <dev@2ion.de>
Installed-Size: 1230
Depends: libatk1.0-0 (>= 1.12.4), libc6 (>= 2.15), libcairo2 (>= 1.2.4), libfontconfig1 (>= 2.11), libfreetype6 (>= 2.2.1), libgdk-pixbuf2.0-0 (>= 2.22.0), libglib2.0-0 (>= 2.35.9), libgtk2.0-0 (>= 2.14.0), libimlib2 (>= 1.4.5), libpango-1.0-0 (>= 1.20.0), libpangocairo-1.0-0 (>= 1.14.0), libpangoft2-1.0-0 (>= 1.14.0), librsvg2-2 (>= 2.14.4), libstartup-notification0 (>= 0.4), libx11-6, libxcomposite1 (>= 1:0.3-1), libxdamage1 (>= 1:1.1), libxfixes3, libxinerama1, libxrandr2 (>= 2:1.2.99.3), libxrender1
Homepage: https://gitlab.com/o9000/tint2/
Priority: optional
Section: x11
Filename: pool/main/t/tint2/tint2_0.14.6-1_amd64.deb
Size: 279638
SHA256: c96e745425a97828952e9e0277176fe68e2512056915560ac968a66c88a0a8b7
SHA1: 82edd60429a494bb127e6d8a10434fca0ee60f61
MD5sum: 65455638fb41503361560b25a70b33b7
Description: lightweight taskbar
Tint is a simple panel/taskbar intentionally made for openbox3, but should
also work with other window managers. The taskbar includes transparency and
color settings for the font, icons, border, and background. It also supports
multihead setups, customized mouse actions, and a built-in clock. Tint was
originally based on ttm code. Since then, support has also been added
for a battery monitor and system tray.
.
The goal is to keep a clean and unintrusive look with lightweight code and
compliance with freedesktop specification.
Package: xfce4-power-manager
Version: 1.4.4-4~bpo8+1
Architecture: amd64
Maintainer: Debian Xfce Maintainers <pkg-xfce-devel@lists.alioth.debian.org>
Installed-Size: 541
Depends: libc6 (>= 2.4), libcairo2 (>= 1.2.4), libdbus-1-3 (>= 1.0.2), libdbus-glib-1-2 (>= 0.88), libgdk-pixbuf2.0-0 (>= 2.22.0), libglib2.0-0 (>= 2.41.1), libgtk2.0-0 (>= 2.24.0), libnotify4 (>= 0.7.0), libpango-1.0-0 (>= 1.14.0), libpangocairo-1.0-0 (>= 1.14.0), libupower-glib3 (>= 0.99.0), libx11-6, libxext6, libxfce4ui-1-0 (>= 4.9.0), libxfce4util6 (>= 4.9.0), libxfconf-0-2 (>= 4.6.0), libxrandr2 (>= 2:1.2.99.2), upower (>= 0.99), xfce4-power-manager-data (= 1.4.4-4~bpo8+1)
Recommends: libpam-systemd, xfce4-power-manager-plugins
Homepage: http://goodies.xfce.org/projects/applications/xfce4-power-manager
Priority: optional
Section: xfce
Filename: pool/main/x/xfce4-power-manager/xfce4-power-manager_1.4.4-4~bpo8+1_amd64.deb
Size: 214122
SHA256: 992b606afe5e9934bce19a1df2b8d7067c98b9d64e23a9b63dbd0c4cf28b4ac9
SHA1: 6bfcd77071f31577a37abab063bf21a34f4d616c
MD5sum: fb777aecbbfe39742649b768eb22c697
Description: power manager for Xfce desktop
This power manager for the Xfce desktop enables laptop users to set up
a power profile for two different modes "on battery power" and "on ac
power" while still allowing desktop users to at least change the DPMS
settings and CPU frequency using the settings dialogue..
.
It features:
* battery monitoring
* cpu frequency settings
* monitor DPMS settings
* suspend/Hibernate
* LCD brightness control
* Lid, sleep and power switches control"""
version, maintainer = deb_packages_extract_version(packages, "tint2")
self.assertEqual(version, "0.14.6-1")
self.assertEqual(maintainer, "Jens John <dev@2ion.de>")
version, maintainer = deb_packages_extract_version(packages, "asdf")
self.assertEqual(version, None)
self.assertEqual(maintainer, None)
def test_arch_packages_extract_version(self):
pkgbuild = """# $Id$
# Maintainer: Alexander F Rødseth <xyproto@archlinux.org>
# Contributor: Blue Peppers <bluepeppers@archlinux.us>
# Contributor: Stefan Husmann <stefan-husmann@t-online.de>
# Contributor: Yannick LM <LMyannicklm1337@gmail.com>
pkgname=tint2
pkgver=0.14.6
pkgrel=2
pkgdesc='Basic, good-looking task manager for WMs'
arch=('x86_64' 'i686')
url='https://gitlab.com/o9000/tint2'
license=('GPL2')
depends=('gtk2' 'imlib2' 'startup-notification')
makedepends=('cmake' 'startup-notification' 'git' 'ninja' 'setconf')
source=("$pkgname-$pkgver.tar.bz2::https://gitlab.com/o9000/tint2/repository/archive.tar.bz2?ref=$pkgver")
sha256sums=('b40079fb187aa248cd3b6957076f138d040c723b309e1b254ac0c8ec9826a451')
prepare() {
mv "$pkgname-$pkgver-"* "$pkgname"
setconf "$pkgname/get_version.sh" VERSION "$pkgver"
}
build() {
mkdir -p build
cd build
cmake "../$pkgname" \
-DCMAKE_INSTALL_PREFIX=/usr \
-DENABLE_TINT2CONF=1 \
-GNinja
ninja
}
package() {
DESTDIR="$pkgdir" ninja -C build install
}
# getver: gitlab.com/o9000/tint2/blob/master/README.md
# vim: ts=2 sw=2 et:"""
version, maintainer = arch_pkgbuild_extract_version(pkgbuild)
self.assertEqual(version, "0.14.6")
self.assertEqual(maintainer, "Alexander F Rødseth <xyproto@archlinux.org>")
if __name__ == '__main__':
unittest.main()

View File

@@ -35,20 +35,25 @@ gboolean bat1_has_font;
PangoFontDescription *bat1_font_desc; PangoFontDescription *bat1_font_desc;
gboolean bat2_has_font; gboolean bat2_has_font;
PangoFontDescription *bat2_font_desc; PangoFontDescription *bat2_font_desc;
char *bat1_format;
char *bat2_format;
struct BatteryState battery_state; struct BatteryState battery_state;
gboolean battery_enabled; gboolean battery_enabled;
gboolean battery_tooltip_enabled; gboolean battery_tooltip_enabled;
int percentage_hide; int percentage_hide;
static timeout *battery_timeout; static timeout *battery_timeout;
static char buf_bat_percentage[10]; #define BATTERY_BUF_SIZE 256
static char buf_bat_time[20]; static char buf_bat_line1[BATTERY_BUF_SIZE];
static char buf_bat_line2[BATTERY_BUF_SIZE];
int8_t battery_low_status; int8_t battery_low_status;
gboolean battery_low_cmd_sent; gboolean battery_low_cmd_sent;
gboolean battery_full_cmd_sent;
char *ac_connected_cmd; char *ac_connected_cmd;
char *ac_disconnected_cmd; char *ac_disconnected_cmd;
char *battery_low_cmd; char *battery_low_cmd;
char *battery_full_cmd;
char *battery_lclick_command; char *battery_lclick_command;
char *battery_mclick_command; char *battery_mclick_command;
char *battery_rclick_command; char *battery_rclick_command;
@@ -56,7 +61,7 @@ char *battery_uwheel_command;
char *battery_dwheel_command; char *battery_dwheel_command;
gboolean battery_found; gboolean battery_found;
char *battery_sys_prefix = (char*)""; char *battery_sys_prefix = (char *)"";
void battery_init_fonts(); void battery_init_fonts();
char *battery_get_tooltip(void *obj); char *battery_get_tooltip(void *obj);
@@ -70,14 +75,18 @@ void default_battery()
battery_found = FALSE; battery_found = FALSE;
percentage_hide = 101; percentage_hide = 101;
battery_low_cmd_sent = FALSE; battery_low_cmd_sent = FALSE;
battery_full_cmd_sent = FALSE;
battery_timeout = NULL; battery_timeout = NULL;
bat1_has_font = FALSE; bat1_has_font = FALSE;
bat1_font_desc = NULL; bat1_font_desc = NULL;
bat1_format = NULL;
bat2_has_font = FALSE; bat2_has_font = FALSE;
bat2_font_desc = NULL; bat2_font_desc = NULL;
bat2_format = NULL;
ac_connected_cmd = NULL; ac_connected_cmd = NULL;
ac_disconnected_cmd = NULL; ac_disconnected_cmd = NULL;
battery_low_cmd = NULL; battery_low_cmd = NULL;
battery_full_cmd = NULL;
battery_lclick_command = NULL; battery_lclick_command = NULL;
battery_mclick_command = NULL; battery_mclick_command = NULL;
battery_rclick_command = NULL; battery_rclick_command = NULL;
@@ -98,6 +107,12 @@ void cleanup_battery()
bat2_font_desc = NULL; bat2_font_desc = NULL;
free(battery_low_cmd); free(battery_low_cmd);
battery_low_cmd = NULL; battery_low_cmd = NULL;
free(battery_full_cmd);
battery_full_cmd = NULL;
free(bat1_format);
bat1_format = NULL;
free(bat2_format);
bat2_format = NULL;
free(battery_lclick_command); free(battery_lclick_command);
battery_lclick_command = NULL; battery_lclick_command = NULL;
free(battery_mclick_command); free(battery_mclick_command);
@@ -119,6 +134,91 @@ void cleanup_battery()
battery_os_free(); battery_os_free();
} }
// Appends addendum to dest, and does not allow dest to grow over limit (including NULL terminator).
char *strnappend(char *dest, const char *addendum, size_t limit)
{
char *tmp = strdup(dest);
// Actually concatenate them.
snprintf(dest, limit, "%s%s", tmp, addendum);
free(tmp);
return dest;
}
void battery_update_text(char *dest, char *format)
{
if (!battery_enabled || !dest || !format)
return;
// We want to loop over the format specifier, replacing any known symbols with our battery data.
// First, erase anything already stored in the buffer.
// This ensures the string will always be null-terminated.
bzero(dest, BATTERY_BUF_SIZE);
for (size_t o = 0; o < strlen(format); o++) {
char buf[BATTERY_BUF_SIZE];
bzero(buf, BATTERY_BUF_SIZE);
char *c = &format[o];
// Format specification:
// %s : State (charging, discharging, full, unknown)
// %m : Minutes left (estimated).
// %h : Hours left (estimated).
// %t : Time left. This is equivalent to the old behaviour; i.e. "(plugged in)" or "hrs:mins" otherwise.
// %p : Percentage left. Includes the % sign.
if (*c == '%') {
c++;
o++; // Skip the format control character.
switch (*c) {
case 's':
// Append the appropriate status message to the string.
strnappend(dest,
(battery_state.state == BATTERY_CHARGING)
? "Charging"
: (battery_state.state == BATTERY_DISCHARGING)
? "Discharging"
: (battery_state.state == BATTERY_FULL)
? "Full"
: "Unknown",
BATTERY_BUF_SIZE);
break;
case 'm':
snprintf(buf, sizeof(buf), "%02d", battery_state.time.minutes);
strnappend(dest, buf, BATTERY_BUF_SIZE);
break;
case 'h':
snprintf(buf, sizeof(buf), "%02d", battery_state.time.hours);
strnappend(dest, buf, BATTERY_BUF_SIZE);
break;
case 'p':
snprintf(buf, sizeof(buf), "%d%%", battery_state.percentage);
strnappend(dest, buf, BATTERY_BUF_SIZE);
break;
case 't':
if (battery_state.state == BATTERY_FULL) {
snprintf(buf, sizeof(buf), "Full");
strnappend(dest, buf, BATTERY_BUF_SIZE);
} else if (battery_state.time.hours > 0 && battery_state.time.minutes > 0) {
snprintf(buf, sizeof(buf), "%02d:%02d", battery_state.time.hours, battery_state.time.minutes);
strnappend(dest, buf, BATTERY_BUF_SIZE);
}
break;
case '%':
case '\0':
strnappend(dest, "%", BATTERY_BUF_SIZE);
break;
default:
fprintf(stderr, "tint2: Battery: unrecognised format specifier '%%%c'.\n", *c);
buf[0] = *c;
strnappend(dest, buf, BATTERY_BUF_SIZE);
}
} else {
buf[0] = *c;
strnappend(dest, buf, BATTERY_BUF_SIZE);
}
}
}
void init_battery() void init_battery()
{ {
if (!battery_enabled) if (!battery_enabled)
@@ -126,7 +226,6 @@ void init_battery()
battery_found = battery_os_init(); battery_found = battery_os_init();
if (!battery_timeout)
battery_timeout = add_timeout(10, 30000, update_battery_tick, 0, &battery_timeout); battery_timeout = add_timeout(10, 30000, update_battery_tick, 0, &battery_timeout);
update_battery(); update_battery();
@@ -169,6 +268,11 @@ void init_battery_panel(void *p)
if (battery_tooltip_enabled) if (battery_tooltip_enabled)
battery->area._get_tooltip_text = battery_get_tooltip; battery->area._get_tooltip_text = battery_get_tooltip;
instantiate_area_gradients(&battery->area); instantiate_area_gradients(&battery->area);
if (!bat1_format && !bat2_format) {
bat1_format = strdup("%p");
bat2_format = strdup("%t");
}
} }
void battery_init_fonts() void battery_init_fonts()
@@ -227,14 +331,14 @@ void update_battery_tick(void *arg)
if (old_ac_connected != battery_state.ac_connected) { if (old_ac_connected != battery_state.ac_connected) {
if (battery_state.ac_connected) if (battery_state.ac_connected)
tint_exec(ac_connected_cmd); tint_exec_no_sn(ac_connected_cmd);
else else
tint_exec(ac_disconnected_cmd); tint_exec_no_sn(ac_disconnected_cmd);
} }
if (battery_state.percentage < battery_low_status && battery_state.state == BATTERY_DISCHARGING && if (battery_state.percentage < battery_low_status && battery_state.state == BATTERY_DISCHARGING &&
!battery_low_cmd_sent) { !battery_low_cmd_sent) {
tint_exec(battery_low_cmd); tint_exec_no_sn(battery_low_cmd);
battery_low_cmd_sent = TRUE; battery_low_cmd_sent = TRUE;
} }
if (battery_state.percentage > battery_low_status && battery_state.state == BATTERY_CHARGING && if (battery_state.percentage > battery_low_status && battery_state.state == BATTERY_CHARGING &&
@@ -242,6 +346,16 @@ void update_battery_tick(void *arg)
battery_low_cmd_sent = FALSE; battery_low_cmd_sent = FALSE;
} }
if ((battery_state.percentage >= 100 || battery_state.state == BATTERY_FULL) &&
!battery_full_cmd_sent) {
tint_exec_no_sn(battery_full_cmd);
battery_full_cmd_sent = TRUE;
}
if (battery_state.percentage < 100 && battery_state.state != BATTERY_FULL &&
battery_full_cmd_sent) {
battery_full_cmd_sent = FALSE;
}
for (int i = 0; i < num_panels; i++) { for (int i = 0; i < num_panels; i++) {
// Show/hide if needed // Show/hide if needed
if (!battery_found) { if (!battery_found) {
@@ -278,168 +392,51 @@ int update_battery()
battery_state.percentage = 100; battery_state.percentage = 100;
} }
battery_update_text(buf_bat_line1, bat1_format);
if (bat2_format != 0) {
battery_update_text(buf_bat_line2, bat2_format);
}
return err; return err;
} }
int battery_compute_desired_size(void *obj) int battery_compute_desired_size(void *obj)
{ {
Battery *battery = (Battery *)obj; Battery *battery = (Battery *)obj;
Panel *panel = (Panel *)battery->area.panel; return text_area_compute_desired_size(&battery->area, buf_bat_line1, buf_bat_line2, bat1_font_desc, bat2_font_desc);
int bat_percentage_height, bat_percentage_width, bat_percentage_height_ink;
int bat_time_height, bat_time_width, bat_time_height_ink;
snprintf(buf_bat_percentage, sizeof(buf_bat_percentage), "%d%%", battery_state.percentage);
if (battery_state.state == BATTERY_FULL) {
strcpy(buf_bat_time, "Full");
} else {
snprintf(buf_bat_time, sizeof(buf_bat_time), "%02d:%02d", battery_state.time.hours, battery_state.time.minutes);
}
get_text_size2(bat1_font_desc,
&bat_percentage_height_ink,
&bat_percentage_height,
&bat_percentage_width,
panel->area.height,
panel->area.width,
buf_bat_percentage,
strlen(buf_bat_percentage),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
get_text_size2(bat2_font_desc,
&bat_time_height_ink,
&bat_time_height,
&bat_time_width,
panel->area.height,
panel->area.width,
buf_bat_time,
strlen(buf_bat_time),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
if (panel_horizontal) {
int new_size = (bat_percentage_width > bat_time_width) ? bat_percentage_width : bat_time_width;
new_size += 2 * battery->area.paddingxlr + left_right_border_width(&battery->area);
return new_size;
} else {
int new_size = bat_percentage_height + bat_time_height + 2 * battery->area.paddingxlr +
top_bottom_border_width(&battery->area);
return new_size;
}
} }
gboolean resize_battery(void *obj) gboolean resize_battery(void *obj)
{ {
Battery *battery = (Battery *)obj; Battery *battery = (Battery *)obj;
Panel *panel = (Panel *)battery->area.panel; return resize_text_area(&battery->area,
int bat_percentage_height, bat_percentage_width, bat_percentage_height_ink; buf_bat_line1,
int bat_time_height, bat_time_width, bat_time_height_ink; buf_bat_line2,
int ret = 0; bat1_font_desc,
bat2_font_desc,
snprintf(buf_bat_percentage, sizeof(buf_bat_percentage), "%d%%", battery_state.percentage); &battery->bat1_posy,
if (battery_state.state == BATTERY_FULL) { &battery->bat2_posy);
strcpy(buf_bat_time, "Full");
} else {
snprintf(buf_bat_time, sizeof(buf_bat_time), "%02d:%02d", battery_state.time.hours, battery_state.time.minutes);
}
get_text_size2(bat1_font_desc,
&bat_percentage_height_ink,
&bat_percentage_height,
&bat_percentage_width,
panel->area.height,
panel->area.width,
buf_bat_percentage,
strlen(buf_bat_percentage),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
get_text_size2(bat2_font_desc,
&bat_time_height_ink,
&bat_time_height,
&bat_time_width,
panel->area.height,
panel->area.width,
buf_bat_time,
strlen(buf_bat_time),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
if (panel_horizontal) {
int new_size = (bat_percentage_width > bat_time_width) ? bat_percentage_width : bat_time_width;
new_size += 2 * battery->area.paddingxlr + left_right_border_width(&battery->area);
if (new_size > battery->area.width || new_size < battery->area.width - 2) {
// we try to limit the number of resize
battery->area.width = new_size;
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height) / 2;
battery->bat2_posy = battery->bat1_posy + bat_percentage_height;
ret = 1;
}
} else {
int new_size = bat_percentage_height + bat_time_height + 2 * battery->area.paddingxlr +
top_bottom_border_width(&battery->area);
if (new_size > battery->area.height || new_size < battery->area.height - 2) {
battery->area.height = new_size;
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height - 2) / 2;
battery->bat2_posy = battery->bat1_posy + bat_percentage_height + 2;
ret = 1;
}
}
schedule_redraw(&battery->area);
return ret;
} }
void draw_battery(void *obj, cairo_t *c) void draw_battery(void *obj, cairo_t *c)
{ {
Battery *battery = obj; Battery *battery = (Battery *)obj;
draw_text_area(&battery->area,
PangoLayout *layout = pango_cairo_create_layout(c); c,
pango_layout_set_font_description(layout, bat1_font_desc); buf_bat_line1,
pango_layout_set_width(layout, battery->area.width * PANGO_SCALE); buf_bat_line2,
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER); bat1_font_desc,
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR); bat2_font_desc,
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE); battery->bat1_posy,
pango_layout_set_text(layout, buf_bat_percentage, strlen(buf_bat_percentage)); battery->bat2_posy,
&battery->font_color);
cairo_set_source_rgba(c,
battery->font_color.rgb[0],
battery->font_color.rgb[1],
battery->font_color.rgb[2],
battery->font_color.alpha);
pango_cairo_update_layout(c, layout);
draw_text(layout, c, 0, battery->bat1_posy, &battery->font_color, ((Panel *)battery->area.panel)->font_shadow);
pango_layout_set_font_description(layout, bat2_font_desc);
pango_layout_set_indent(layout, 0);
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
pango_layout_set_text(layout, buf_bat_time, strlen(buf_bat_time));
pango_layout_set_width(layout, battery->area.width * PANGO_SCALE);
pango_cairo_update_layout(c, layout);
draw_text(layout, c, 0, battery->bat2_posy, &battery->font_color, ((Panel *)battery->area.panel)->font_shadow);
pango_cairo_show_layout(c, layout);
g_object_unref(layout);
} }
void battery_dump_geometry(void *obj, int indent) void battery_dump_geometry(void *obj, int indent)
{ {
Battery *battery = obj; Battery *battery = (Battery *)obj;
fprintf(stderr, fprintf(stderr, "tint2: %*sText 1: y = %d, text = %s\n", indent, "", battery->bat1_posy, buf_bat_line1);
"%*sText 1: y = %d, text = %s\n", fprintf(stderr, "tint2: %*sText 2: y = %d, text = %s\n", indent, "", battery->bat2_posy, buf_bat_line2);
indent,
"",
battery->bat1_posy,
buf_bat_percentage);
fprintf(stderr,
"%*sText 2: y = %d, text = %s\n",
indent,
"",
battery->bat2_posy,
buf_bat_time);
} }
char *battery_get_tooltip(void *obj) char *battery_get_tooltip(void *obj)
@@ -447,7 +444,7 @@ char *battery_get_tooltip(void *obj)
return battery_os_tooltip(); return battery_os_tooltip();
} }
void battery_action(int button) void battery_action(void *obj, int button, int x, int y, Time time)
{ {
char *command = NULL; char *command = NULL;
switch (button) { switch (button) {
@@ -467,5 +464,5 @@ void battery_action(int button)
command = battery_dwheel_command; command = battery_dwheel_command;
break; break;
} }
tint_exec(command); tint_exec(command, NULL, NULL, time, obj, x, y, FALSE, TRUE);
} }

View File

@@ -48,12 +48,15 @@ extern gboolean bat1_has_font;
extern PangoFontDescription *bat1_font_desc; extern PangoFontDescription *bat1_font_desc;
extern gboolean bat2_has_font; extern gboolean bat2_has_font;
extern PangoFontDescription *bat2_font_desc; extern PangoFontDescription *bat2_font_desc;
extern char *bat1_format;
extern char *bat2_format;
extern gboolean battery_enabled; extern gboolean battery_enabled;
extern gboolean battery_tooltip_enabled; extern gboolean battery_tooltip_enabled;
extern int percentage_hide; extern int percentage_hide;
extern int8_t battery_low_status; extern int8_t battery_low_status;
extern char *battery_low_cmd; extern char *battery_low_cmd;
extern char *battery_full_cmd;
extern char *ac_connected_cmd; extern char *ac_connected_cmd;
extern char *ac_disconnected_cmd; extern char *ac_disconnected_cmd;
@@ -108,7 +111,7 @@ void battery_default_font_changed();
gboolean resize_battery(void *obj); gboolean resize_battery(void *obj);
void battery_action(int button); void battery_action(void *obj, int button, int x, int y, Time time);
/* operating system specific functions */ /* operating system specific functions */
gboolean battery_os_init(); gboolean battery_os_init();

View File

@@ -58,7 +58,7 @@ int battery_os_update(BatteryState *state)
break; break;
} }
} else { } else {
fprintf(stderr, "power update: no such sysctl"); fprintf(stderr, "tint2: power update: no such sysctl");
err = -1; err = -1;
} }

View File

@@ -60,6 +60,20 @@ struct psy_mains {
gboolean online; gboolean online;
}; };
static gboolean is_file_non_empty(const char *path)
{
FILE *f = fopen(path, "r");
if (!f)
return FALSE;
char buffer[1024];
size_t count = fread(buffer, 1, sizeof(buffer), f);
fclose(f);
if (count > 0)
return TRUE;
else
return FALSE;
}
static void uevent_battery_update() static void uevent_battery_update()
{ {
update_battery_tick(NULL); update_battery_tick(NULL);
@@ -68,7 +82,7 @@ static struct uevent_notify psy_change = {UEVENT_CHANGE, "power_supply", NULL, u
static void uevent_battery_plug() static void uevent_battery_plug()
{ {
printf("reinitialize batteries after HW change\n"); fprintf(stderr, "tint2: reinitialize batteries after HW change\n");
reinit_battery(); reinit_battery();
} }
static struct uevent_notify psy_plug = {UEVENT_ADD | UEVENT_REMOVE, "power_supply", NULL, uevent_battery_plug}; static struct uevent_notify psy_plug = {UEVENT_ADD | UEVENT_REMOVE, "power_supply", NULL, uevent_battery_plug};
@@ -76,7 +90,7 @@ static struct uevent_notify psy_plug = {UEVENT_ADD | UEVENT_REMOVE, "power_suppl
#define RETURN_ON_ERROR(err) \ #define RETURN_ON_ERROR(err) \
if (err) { \ if (err) { \
g_error_free(err); \ g_error_free(err); \
fprintf(stderr, RED "%s:%d: errror" RESET "\n", __FILE__, __LINE__); \ fprintf(stderr, RED "tint2: %s:%d: errror" RESET "\n", __FILE__, __LINE__); \
return FALSE; \ return FALSE; \
} }
@@ -96,12 +110,13 @@ static enum psy_type power_supply_get_type(const gchar *entryname)
gsize typelen; gsize typelen;
g_file_get_contents(path_type, &type, &typelen, &error); g_file_get_contents(path_type, &type, &typelen, &error);
g_free(path_type);
if (error) { if (error) {
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__); fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, path_type);
g_free(path_type);
g_error_free(error); g_error_free(error);
return PSY_UNKNOWN; return PSY_UNKNOWN;
} }
g_free(path_type);
if (!g_strcmp0(type, "Battery\n")) { if (!g_strcmp0(type, "Battery\n")) {
g_free(type); g_free(type);
@@ -123,20 +138,20 @@ static gboolean init_linux_battery(struct psy_battery *bat)
const gchar *entryname = bat->name; const gchar *entryname = bat->name;
bat->path_present = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "present", NULL); bat->path_present = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "present", NULL);
if (!g_file_test(bat->path_present, G_FILE_TEST_EXISTS)) { if (!is_file_non_empty(bat->path_present)) {
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__); fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, bat->path_present);
goto err0; goto err0;
} }
bat->path_level_now = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "energy_now", NULL); bat->path_level_now =
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "energy_now", NULL);
bat->path_level_full = bat->path_level_full =
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "energy_full", NULL); g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "energy_full", NULL);
bat->path_rate_now = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "power_now", NULL); bat->path_rate_now = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "power_now", NULL);
bat->unit = 'W'; bat->unit = 'W';
if (!g_file_test(bat->path_level_now, G_FILE_TEST_EXISTS) || if (!is_file_non_empty(bat->path_level_now) ||
!g_file_test(bat->path_level_full, G_FILE_TEST_EXISTS) || !is_file_non_empty(bat->path_level_full)) {
!g_file_test(bat->path_rate_now, G_FILE_TEST_EXISTS)) {
g_free(bat->path_level_now); g_free(bat->path_level_now);
g_free(bat->path_level_full); g_free(bat->path_level_full);
g_free(bat->path_rate_now); g_free(bat->path_rate_now);
@@ -148,16 +163,18 @@ static gboolean init_linux_battery(struct psy_battery *bat)
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "current_now", NULL); g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "current_now", NULL);
bat->unit = 'A'; bat->unit = 'A';
} }
if (!g_file_test(bat->path_level_now, G_FILE_TEST_EXISTS) || if (!is_file_non_empty(bat->path_level_now)) {
!g_file_test(bat->path_level_full, G_FILE_TEST_EXISTS) || fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, bat->path_level_now);
!g_file_test(bat->path_rate_now, G_FILE_TEST_EXISTS)) { goto err1;
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__); }
if (!is_file_non_empty(bat->path_level_full)) {
fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, bat->path_level_full);
goto err1; goto err1;
} }
bat->path_status = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "status", NULL); bat->path_status = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "status", NULL);
if (!g_file_test(bat->path_status, G_FILE_TEST_EXISTS)) { if (!is_file_non_empty(bat->path_status)) {
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__); fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, bat->path_status);
goto err2; goto err2;
} }
@@ -180,8 +197,8 @@ static gboolean init_linux_mains(struct psy_mains *ac)
const gchar *entryname = ac->name; const gchar *entryname = ac->name;
ac->path_online = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "online", NULL); ac->path_online = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "online", NULL);
if (!g_file_test(ac->path_online, G_FILE_TEST_EXISTS)) { if (!is_file_non_empty(ac->path_online)) {
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__); fprintf(stderr, RED "tint2: %s:%d: read failed for %s" RESET "\n", __FILE__, __LINE__, ac->path_online);
g_free(ac->path_online); g_free(ac->path_online);
return FALSE; return FALSE;
} }
@@ -227,10 +244,10 @@ static void add_battery(const char *entryname)
if (init_linux_battery(bat)) { if (init_linux_battery(bat)) {
batteries = g_list_append(batteries, bat); batteries = g_list_append(batteries, bat);
fprintf(stdout, GREEN "Found battery \"%s\"" RESET "\n", bat->name); fprintf(stderr, GREEN "Found battery \"%s\"" RESET "\n", bat->name);
} else { } else {
g_free(bat); g_free(bat);
fprintf(stderr, RED "Failed to initialize battery \"%s\"" RESET "\n", entryname); fprintf(stderr, RED "tint2: Failed to initialize battery \"%s\"" RESET "\n", entryname);
} }
} }
@@ -241,10 +258,10 @@ static void add_mains(const char *entryname)
if (init_linux_mains(ac)) { if (init_linux_mains(ac)) {
mains = g_list_append(mains, ac); mains = g_list_append(mains, ac);
fprintf(stdout, GREEN "Found mains \"%s\"" RESET "\n", ac->name); fprintf(stderr, GREEN "Found mains \"%s\"" RESET "\n", ac->name);
} else { } else {
g_free(ac); g_free(ac);
fprintf(stderr, RED "Failed to initialize mains \"%s\"" RESET "\n", entryname); fprintf(stderr, RED "tint2: Failed to initialize mains \"%s\"" RESET "\n", entryname);
} }
} }
@@ -262,7 +279,7 @@ gboolean battery_os_init()
RETURN_ON_ERROR(error); RETURN_ON_ERROR(error);
while ((entryname = g_dir_read_name(directory))) { while ((entryname = g_dir_read_name(directory))) {
fprintf(stderr, GREEN "Found power device %s" RESET "\n", entryname); fprintf(stderr, GREEN "tint2: Found power device %s" RESET "\n", entryname);
enum psy_type type = power_supply_get_type(entryname); enum psy_type type = power_supply_get_type(entryname);
switch (type) { switch (type) {
@@ -445,6 +462,17 @@ int battery_os_update(BatteryState *state)
/* AC state */ /* AC state */
state->ac_connected = ac_connected; state->ac_connected = ac_connected;
if (state->state == BATTERY_UNKNOWN) {
if (ac_connected) {
if (total_rate_now == 0 && state->percentage >= 90)
state->state = BATTERY_FULL;
else
state->state = BATTERY_CHARGING;
} else {
state->state = BATTERY_DISCHARGING;
}
}
return 0; return 0;
} }

View File

@@ -52,11 +52,9 @@ void destroy_button(void *obj)
Button *button = (Button *)obj; Button *button = (Button *)obj;
if (button->frontend) { if (button->frontend) {
// This is a frontend element // This is a frontend element
if (button->frontend->icon) { free_icon(button->frontend->icon);
imlib_context_set_image(button->frontend->icon); free_icon(button->frontend->icon_hover);
imlib_free_image(); free_icon(button->frontend->icon_pressed);
button->frontend->icon = NULL;
}
button->backend->instances = g_list_remove_all(button->backend->instances, button); button->backend->instances = g_list_remove_all(button->backend->instances, button);
free_and_null(button->frontend); free_and_null(button->frontend);
remove_area(&button->area); remove_area(&button->area);
@@ -78,8 +76,8 @@ void destroy_button(void *obj)
free_and_null(button->backend->uwheel_command); free_and_null(button->backend->uwheel_command);
if (button->backend->instances) { if (button->backend->instances) {
fprintf(stderr, "Error: Attempt to destroy backend while there are still frontend instances!\n"); fprintf(stderr, "tint2: Error: Attempt to destroy backend while there are still frontend instances!\n");
exit(-1); exit(EXIT_FAILURE);
} }
free(button->backend); free(button->backend);
free(button); free(button);
@@ -309,8 +307,8 @@ int button_compute_desired_size(void *obj)
&txt_height, &txt_height,
&txt_width, &txt_width,
panel->area.height, panel->area.height,
button->area.width - icon_w - (icon_w ? interior_padding : 0) - button->area.width - icon_w - (icon_w ? interior_padding : 0) - 2 * horiz_padding -
2 * horiz_padding - left_right_border_width(&button->area), left_right_border_width(&button->area),
button->backend->text, button->backend->text,
strlen(button->backend->text), strlen(button->backend->text),
PANGO_WRAP_WORD_CHAR, PANGO_WRAP_WORD_CHAR,
@@ -337,6 +335,7 @@ gboolean resize_button(void *obj)
{ {
Button *button = (Button *)obj; Button *button = (Button *)obj;
Panel *panel = (Panel *)button->area.panel; Panel *panel = (Panel *)button->area.panel;
Area *area = &button->area;
int horiz_padding = (panel_horizontal ? button->area.paddingxlr : button->area.paddingy); int horiz_padding = (panel_horizontal ? button->area.paddingxlr : button->area.paddingy);
int vert_padding = (panel_horizontal ? button->area.paddingy : button->area.paddingxlr); int vert_padding = (panel_horizontal ? button->area.paddingy : button->area.paddingxlr);
int interior_padding = button->area.paddingx; int interior_padding = button->area.paddingx;
@@ -360,34 +359,29 @@ gboolean resize_button(void *obj)
if (button->frontend->icon_load_size != button->frontend->iconw) if (button->frontend->icon_load_size != button->frontend->iconw)
button_reload_icon(button); button_reload_icon(button);
int available_w, available_h;
if (panel_horizontal) {
available_w = panel->area.width;
available_h = area->height - 2 * area->paddingy - left_right_border_width(area);
} else {
available_w =
area->width - icon_w - (icon_w ? interior_padding : 0) - 2 * horiz_padding - left_right_border_width(area);
available_h = panel->area.height;
}
int txt_height_ink, txt_height, txt_width; int txt_height_ink, txt_height, txt_width;
if (button->backend->text) { if (button->backend->text) {
if (panel_horizontal) {
get_text_size2(button->backend->font_desc, get_text_size2(button->backend->font_desc,
&txt_height_ink, &txt_height_ink,
&txt_height, &txt_height,
&txt_width, &txt_width,
panel->area.height, available_h,
panel->area.width, available_w,
button->backend->text, button->backend->text,
strlen(button->backend->text), strlen(button->backend->text),
PANGO_WRAP_WORD_CHAR, PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE, PANGO_ELLIPSIZE_NONE,
FALSE); FALSE);
} else {
get_text_size2(button->backend->font_desc,
&txt_height_ink,
&txt_height,
&txt_width,
panel->area.height,
button->area.width - icon_w - (icon_w ? interior_padding : 0) -
2 * horiz_padding - left_right_border_width(&button->area),
button->backend->text,
strlen(button->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
}
} else { } else {
txt_height_ink = txt_height = txt_width = 0; txt_height_ink = txt_height = txt_width = 0;
} }
@@ -414,7 +408,8 @@ gboolean resize_button(void *obj)
if (button->backend->centered) { if (button->backend->centered) {
if (icon_w) { if (icon_w) {
button->frontend->icony = (button->area.height - icon_h) / 2; button->frontend->icony = (button->area.height - icon_h) / 2;
button->frontend->iconx = (button->area.width - txt_width - (txt_width ? interior_padding : 0) - icon_w) / 2; button->frontend->iconx =
(button->area.width - txt_width - (txt_width ? interior_padding : 0) - icon_w) / 2;
button->frontend->texty = (button->area.height - txt_height) / 2; button->frontend->texty = (button->area.height - txt_height) / 2;
button->frontend->textx = button->frontend->iconx + icon_w + interior_padding; button->frontend->textx = button->frontend->iconx + icon_w + interior_padding;
} else { } else {
@@ -491,7 +486,7 @@ void button_dump_geometry(void *obj, int indent)
Imlib_Image tmp = imlib_context_get_image(); Imlib_Image tmp = imlib_context_get_image();
imlib_context_set_image(button->frontend->icon); imlib_context_set_image(button->frontend->icon);
fprintf(stderr, fprintf(stderr,
"%*sIcon: x = %d, y = %d, w = %d, h = %d\n", "tint2: %*sIcon: x = %d, y = %d, w = %d, h = %d\n",
indent, indent,
"", "",
button->frontend->iconx, button->frontend->iconx,
@@ -502,7 +497,7 @@ void button_dump_geometry(void *obj, int indent)
imlib_context_set_image(tmp); imlib_context_set_image(tmp);
} }
fprintf(stderr, fprintf(stderr,
"%*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n", "tint2: %*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n",
indent, indent,
"", "",
button->frontend->textx, button->frontend->textx,
@@ -512,9 +507,9 @@ void button_dump_geometry(void *obj, int indent)
button->backend->text); button->backend->text);
} }
void button_action(void *obj, int mouse_button, int x, int y) void button_action(void *obj, int mouse_button, int x, int y, Time time)
{ {
Button *button = obj; Button *button = (Button *)obj;
char *command = NULL; char *command = NULL;
switch (mouse_button) { switch (mouse_button) {
case 1: case 1:
@@ -533,29 +528,7 @@ void button_action(void *obj, int mouse_button, int x, int y)
command = button->backend->dwheel_command; command = button->backend->dwheel_command;
break; break;
} }
if (command) { tint_exec(command, NULL, NULL, time, obj, x, y, FALSE, TRUE);
char *full_cmd = g_strdup_printf("export BUTTON_X=%d;"
"export BUTTON_Y=%d;"
"export BUTTON_W=%d;"
"export BUTTON_H=%d; %s",
x,
y,
button->area.width,
button->area.height,
command);
pid_t pid = fork();
if (pid < 0) {
fprintf(stderr, "Could not fork\n");
} else if (pid == 0) {
// Child process
// Allow children to exist after parent destruction
setsid();
// Run the command
execl("/bin/sh", "/bin/sh", "-c", full_cmd, NULL);
fprintf(stderr, "Failed to execlp %s\n", full_cmd);
exit(1);
}
}
} }
char *button_get_tooltip(void *obj) char *button_get_tooltip(void *obj)

View File

@@ -65,7 +65,8 @@ typedef struct Button {
} Button; } Button;
// Called before the config is read and panel_config/panels are created. // Called before the config is read and panel_config/panels are created.
// Afterwards, the config parsing code creates the array of Button in panel_config and populates the configuration fields // Afterwards, the config parsing code creates the array of Button in panel_config and populates the configuration
// fields
// in the backend. // in the backend.
// Probably does nothing. // Probably does nothing.
void default_button(); void default_button();
@@ -102,7 +103,7 @@ void draw_button(void *obj, cairo_t *c);
gboolean resize_button(void *obj); gboolean resize_button(void *obj);
// Called on mouse click event. // Called on mouse click event.
void button_action(void *obj, int button, int x, int y); void button_action(void *obj, int button, int x, int y, Time time);
void button_default_font_changed(); void button_default_font_changed();
void button_default_icon_theme_changed(); void button_default_icon_theme_changed();

View File

@@ -77,6 +77,9 @@ void default_clock()
time1_font_desc = NULL; time1_font_desc = NULL;
time2_has_font = FALSE; time2_has_font = FALSE;
time2_font_desc = NULL; time2_font_desc = NULL;
buf_time[0] = 0;
buf_date[0] = 0;
buf_tooltip[0] = 0;
} }
void cleanup_clock() void cleanup_clock()
@@ -111,31 +114,6 @@ void cleanup_clock()
clock_timeout = NULL; clock_timeout = NULL;
} }
void update_clocks_sec(void *arg)
{
gettimeofday(&time_clock, 0);
if (time1_format) {
for (int i = 0; i < num_panels; i++)
panels[i].clock.area.resize_needed = 1;
}
schedule_panel_redraw();
}
void update_clocks_min(void *arg)
{
// remember old_sec because after suspend/hibernate the clock should be updated directly, and not
// on next minute change
time_t old_sec = time_clock.tv_sec;
gettimeofday(&time_clock, 0);
if (time_clock.tv_sec % 60 == 0 || time_clock.tv_sec - old_sec > 60) {
if (time1_format) {
for (int i = 0; i < num_panels; i++)
panels[i].clock.area.resize_needed = 1;
}
schedule_panel_redraw();
}
}
struct tm *clock_gettime_for_tz(const char *timezone) struct tm *clock_gettime_for_tz(const char *timezone)
{ {
if (timezone) { if (timezone) {
@@ -152,6 +130,46 @@ struct tm *clock_gettime_for_tz(const char *timezone)
} }
} }
void update_clocks()
{
if (time1_format)
strftime(buf_time, sizeof(buf_time), time1_format, clock_gettime_for_tz(time1_timezone));
if (time2_format)
strftime(buf_date, sizeof(buf_date), time2_format, clock_gettime_for_tz(time2_timezone));
if (time1_format || time2_format) {
for (int i = 0; i < num_panels; i++)
panels[i].clock.area.resize_needed = 1;
}
schedule_panel_redraw();
}
int ms_until_second_change(struct timeval* tm)
{
long us_until_change = 1000000 - tm->tv_usec;
// compute ms, rounding up so we don't ask to wait too short
int ms = (us_until_change+999)/1000;
return ms;
}
void update_clocks_sec(void *arg)
{
gettimeofday(&time_clock, 0);
update_clocks();
clock_timeout = add_timeout(ms_until_second_change(&time_clock), 0, update_clocks_sec, 0, &clock_timeout);
}
void update_clocks_min(void *arg)
{
// remember old_sec because after suspend/hibernate the clock should be updated directly, and not
// on next minute change
static time_t old_sec = 0;
gettimeofday(&time_clock, 0);
if (time_clock.tv_sec % 60 == 0 || time_clock.tv_sec - old_sec > 60 || (time1_format && !buf_time[0]) || (time2_format && !buf_date[0]))
update_clocks();
old_sec = time_clock.tv_sec;
clock_timeout = add_timeout(ms_until_second_change(&time_clock), 0, update_clocks_min, 0, &clock_timeout);
}
gboolean time_format_needs_sec_ticks(char *time_format) gboolean time_format_needs_sec_ticks(char *time_format)
{ {
if (!time_format) if (!time_format)
@@ -170,14 +188,6 @@ void init_clock_panel(void *p)
Panel *panel = (Panel *)p; Panel *panel = (Panel *)p;
Clock *clock = &panel->clock; Clock *clock = &panel->clock;
if (!clock_timeout) {
if (time_format_needs_sec_ticks(time1_format) || time_format_needs_sec_ticks(time2_format)) {
clock_timeout = add_timeout(10, 1000, update_clocks_sec, 0, &clock_timeout);
} else {
clock_timeout = add_timeout(10, 1000, update_clocks_min, 0, &clock_timeout);
}
}
if (!clock->area.bg) if (!clock->area.bg)
clock->area.bg = &g_array_index(backgrounds, Background, 0); clock->area.bg = &g_array_index(backgrounds, Background, 0);
clock_init_fonts(); clock_init_fonts();
@@ -205,6 +215,14 @@ void init_clock_panel(void *p)
clock->area._get_tooltip_text = clock_get_tooltip; clock->area._get_tooltip_text = clock_get_tooltip;
strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone)); strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone));
} }
if (!clock_timeout) {
if (time_format_needs_sec_ticks(time1_format) || time_format_needs_sec_ticks(time2_format)) {
update_clocks_sec(NULL);
} else {
update_clocks_min(NULL);
}
}
} }
void clock_init_fonts() void clock_init_fonts()
@@ -243,7 +261,7 @@ void clock_default_font_changed()
schedule_panel_redraw(); schedule_panel_redraw();
} }
void clock_compute_text_geometry(Panel *panel, void clock_compute_text_geometry(Clock *clock,
int *time_height_ink, int *time_height_ink,
int *time_height, int *time_height,
int *time_width, int *time_width,
@@ -251,139 +269,61 @@ void clock_compute_text_geometry(Panel *panel,
int *date_height, int *date_height,
int *date_width) int *date_width)
{ {
*date_height = *date_width = 0; area_compute_text_geometry(&clock->area,
strftime(buf_time, sizeof(buf_time), time1_format, clock_gettime_for_tz(time1_timezone)); buf_time,
get_text_size2(time1_font_desc, time2_format ? buf_date : NULL,
time1_font_desc,
time2_font_desc,
time_height_ink, time_height_ink,
time_height, time_height,
time_width, time_width,
panel->area.height,
panel->area.width,
buf_time,
strlen(buf_time),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
if (time2_format) {
strftime(buf_date, sizeof(buf_date), time2_format, clock_gettime_for_tz(time2_timezone));
get_text_size2(time2_font_desc,
date_height_ink, date_height_ink,
date_height, date_height,
date_width, date_width);
panel->area.height,
panel->area.width,
buf_date,
strlen(buf_date),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
FALSE);
}
} }
int clock_compute_desired_size(void *obj) int clock_compute_desired_size(void *obj)
{ {
Clock *clock = (Clock *)obj; Clock *clock = (Clock *)obj;
Panel *panel = (Panel *)clock->area.panel; return text_area_compute_desired_size(&clock->area,
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width; buf_time,
clock_compute_text_geometry(panel, time2_format ? buf_date : NULL,
&time_height_ink, time1_font_desc,
&time_height, time2_font_desc);
&time_width,
&date_height_ink,
&date_height,
&date_width);
if (panel_horizontal) {
int new_size = (time_width > date_width) ? time_width : date_width;
new_size += 2 * clock->area.paddingxlr + left_right_border_width(&clock->area);
return new_size;
} else {
int new_size = time_height + date_height + 2 * clock->area.paddingxlr + top_bottom_border_width(&clock->area);
return new_size;
}
} }
gboolean resize_clock(void *obj) gboolean resize_clock(void *obj)
{ {
Clock *clock = (Clock *)obj; Clock *clock = (Clock *)obj;
Panel *panel = (Panel *)clock->area.panel; return resize_text_area(&clock->area,
gboolean result = FALSE; buf_time,
time2_format ? buf_date : NULL,
schedule_redraw(&clock->area); time1_font_desc,
time2_font_desc,
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width; &clock->time1_posy,
clock_compute_text_geometry(panel, &clock->time2_posy);
&time_height_ink,
&time_height,
&time_width,
&date_height_ink,
&date_height,
&date_width);
int new_size = clock_compute_desired_size(clock);
if (panel_horizontal) {
if (new_size > clock->area.width || new_size < (clock->area.width - 6)) {
// we try to limit the number of resizes
clock->area.width = new_size + 1;
clock->time1_posy = (clock->area.height - time_height) / 2;
if (time2_format) {
clock->time1_posy -= (date_height) / 2;
clock->time2_posy = clock->time1_posy + time_height;
}
result = TRUE;
}
} else {
if (new_size != clock->area.height) {
// we try to limit the number of resizes
clock->area.height = new_size;
clock->time1_posy = (clock->area.height - time_height) / 2;
if (time2_format) {
clock->time1_posy -= (date_height) / 2;
clock->time2_posy = clock->time1_posy + time_height;
}
result = TRUE;
}
}
return result;
} }
void draw_clock(void *obj, cairo_t *c) void draw_clock(void *obj, cairo_t *c)
{ {
Clock *clock = obj; Clock *clock = (Clock *)obj;
PangoLayout *layout = pango_cairo_create_layout(c); draw_text_area(&clock->area,
c,
pango_layout_set_font_description(layout, time1_font_desc); buf_time,
pango_layout_set_width(layout, clock->area.width * PANGO_SCALE); time2_format ? buf_date : NULL,
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER); time1_font_desc,
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR); time2_font_desc,
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE); clock->time1_posy,
pango_layout_set_text(layout, buf_time, strlen(buf_time)); clock->time2_posy,
&clock->font);
cairo_set_source_rgba(c, clock->font.rgb[0], clock->font.rgb[1], clock->font.rgb[2], clock->font.alpha);
pango_cairo_update_layout(c, layout);
draw_text(layout, c, 0, clock->time1_posy, &clock->font, ((Panel *)clock->area.panel)->font_shadow);
if (time2_format) {
pango_layout_set_font_description(layout, time2_font_desc);
pango_layout_set_indent(layout, 0);
pango_layout_set_text(layout, buf_date, strlen(buf_date));
pango_layout_set_width(layout, clock->area.width * PANGO_SCALE);
pango_cairo_update_layout(c, layout);
draw_text(layout, c, 0, clock->time2_posy, &clock->font, ((Panel *)clock->area.panel)->font_shadow);
}
g_object_unref(layout);
} }
void clock_dump_geometry(void *obj, int indent) void clock_dump_geometry(void *obj, int indent)
{ {
Clock *clock = (Clock *)obj; Clock *clock = (Clock *)obj;
fprintf(stderr, "%*sText 1: y = %d, text = %s\n", indent, "", clock->time1_posy, buf_time); fprintf(stderr, "tint2: %*sText 1: y = %d, text = %s\n", indent, "", clock->time1_posy, buf_time);
if (time2_format) { if (time2_format) {
fprintf(stderr, "%*sText 2: y = %d, text = %s\n", indent, "", clock->time2_posy, buf_date); fprintf(stderr, "tint2: %*sText 2: y = %d, text = %s\n", indent, "", clock->time2_posy, buf_date);
} }
} }
@@ -393,7 +333,7 @@ char *clock_get_tooltip(void *obj)
return strdup(buf_tooltip); return strdup(buf_tooltip);
} }
void clock_action(int button) void clock_action(void *obj, int button, int x, int y, Time time)
{ {
char *command = NULL; char *command = NULL;
switch (button) { switch (button) {
@@ -413,5 +353,5 @@ void clock_action(int button)
command = clock_dwheel_command; command = clock_dwheel_command;
break; break;
} }
tint_exec(command); tint_exec(command, NULL, NULL, time, obj, x, y, FALSE, TRUE);
} }

View File

@@ -54,6 +54,6 @@ void draw_clock(void *obj, cairo_t *c);
gboolean resize_clock(void *obj); gboolean resize_clock(void *obj);
void clock_action(int button); void clock_action(void *obj, int button, int x, int y, Time time);
#endif #endif

View File

@@ -62,8 +62,8 @@
#endif #endif
// global path // global path
char *config_path; char *config_path = NULL;
char *snapshot_path; char *snapshot_path = NULL;
#ifndef TINT2CONF #ifndef TINT2CONF
@@ -119,7 +119,7 @@ void get_action(char *event, MouseAction *action)
else if (strcmp(event, "prev_task") == 0) else if (strcmp(event, "prev_task") == 0)
*action = PREV_TASK; *action = PREV_TASK;
else else
fprintf(stderr, "Error: unrecognized action '%s'. Please fix your config file.\n", event); fprintf(stderr, "tint2: Error: unrecognized action '%s'. Please fix your config file.\n", event);
} }
int get_task_status(char *status) int get_task_status(char *status)
@@ -135,7 +135,16 @@ int get_task_status(char *status)
int config_get_monitor(char *monitor) int config_get_monitor(char *monitor)
{ {
if (strcmp(monitor, "all") != 0) { if (strcmp(monitor, "primary") == 0) {
for (int i = 0; i < server.num_monitors; ++i) {
if (server.monitors[i].primary)
return i;
}
return 0;
}
if (strcmp(monitor, "all") == 0) {
return -1;
}
char *endptr; char *endptr;
int ret_int = strtol(monitor, &endptr, 10); int ret_int = strtol(monitor, &endptr, 10);
if (*endptr == 0) if (*endptr == 0)
@@ -154,8 +163,8 @@ int config_get_monitor(char *monitor)
} }
} }
} }
}
// monitor == "all" or monitor not found or xrandr can't identify monitors // monitor not found or xrandr can't identify monitors => all
return -1; return -1;
} }
@@ -206,7 +215,7 @@ void load_launcher_app_dir(const char *path)
Separator *get_or_create_last_separator() Separator *get_or_create_last_separator()
{ {
if (!panel_config.separator_list) { if (!panel_config.separator_list) {
fprintf(stderr, "Warning: separator items should shart with 'separator = new'\n"); fprintf(stderr, "tint2: Warning: separator items should shart with 'separator = new'\n");
panel_config.separator_list = g_list_append(panel_config.separator_list, create_separator()); panel_config.separator_list = g_list_append(panel_config.separator_list, create_separator());
} }
return (Separator *)g_list_last(panel_config.separator_list)->data; return (Separator *)g_list_last(panel_config.separator_list)->data;
@@ -215,7 +224,7 @@ Separator *get_or_create_last_separator()
Execp *get_or_create_last_execp() Execp *get_or_create_last_execp()
{ {
if (!panel_config.execp_list) { if (!panel_config.execp_list) {
fprintf(stderr, "Warning: execp items should start with 'execp = new'\n"); fprintf(stderr, "tint2: Warning: execp items should start with 'execp = new'\n");
panel_config.execp_list = g_list_append(panel_config.execp_list, create_execp()); panel_config.execp_list = g_list_append(panel_config.execp_list, create_execp());
} }
return (Execp *)g_list_last(panel_config.execp_list)->data; return (Execp *)g_list_last(panel_config.execp_list)->data;
@@ -224,7 +233,7 @@ Execp *get_or_create_last_execp()
Button *get_or_create_last_button() Button *get_or_create_last_button()
{ {
if (!panel_config.button_list) { if (!panel_config.button_list) {
fprintf(stderr, "Warning: button items should start with 'button = new'\n"); fprintf(stderr, "tint2: Warning: button items should start with 'button = new'\n");
panel_config.button_list = g_list_append(panel_config.button_list, create_button()); panel_config.button_list = g_list_append(panel_config.button_list, create_button());
} }
return (Button *)g_list_last(panel_config.button_list)->data; return (Button *)g_list_last(panel_config.button_list)->data;
@@ -341,6 +350,12 @@ void add_entry(char *key, char *value)
id = (id < gradients->len && id >= 0) ? id : -1; id = (id < gradients->len && id >= 0) ? id : -1;
if (id >= 0) if (id >= 0)
bg->gradients[MOUSE_DOWN] = &g_array_index(gradients, GradientClass, id); bg->gradients[MOUSE_DOWN] = &g_array_index(gradients, GradientClass, id);
} else if (strcmp(key, "border_content_tint_weight") == 0) {
Background *bg = &g_array_index(backgrounds, Background, backgrounds->len - 1);
bg->border_content_tint_weight = MAX(0.0, MIN(1.0, atoi(value) / 100.));
} else if (strcmp(key, "background_content_tint_weight") == 0) {
Background *bg = &g_array_index(backgrounds, Background, backgrounds->len - 1);
bg->fill_content_tint_weight = MAX(0.0, MIN(1.0, atoi(value) / 100.));
} }
/* Gradients */ /* Gradients */
@@ -368,7 +383,7 @@ void add_entry(char *key, char *value)
} else if (strcmp(key, "color_stop") == 0) { } else if (strcmp(key, "color_stop") == 0) {
GradientClass *g = &g_array_index(gradients, GradientClass, gradients->len - 1); GradientClass *g = &g_array_index(gradients, GradientClass, gradients->len - 1);
extract_values(value, &value1, &value2, &value3); extract_values(value, &value1, &value2, &value3);
ColorStop *color_stop = (ColorStop *) calloc(1, sizeof(ColorStop)); ColorStop *color_stop = (ColorStop *)calloc(1, sizeof(ColorStop));
color_stop->offset = atof(value1) / 100.0; color_stop->offset = atof(value1) / 100.0;
get_color(value2, color_stop->color.rgb); get_color(value2, color_stop->color.rgb);
if (value3) if (value3)
@@ -381,8 +396,6 @@ void add_entry(char *key, char *value)
/* Panel */ /* Panel */
else if (strcmp(key, "panel_monitor") == 0) { else if (strcmp(key, "panel_monitor") == 0) {
panel_config.monitor = config_get_monitor(value); panel_config.monitor = config_get_monitor(value);
} else if (strcmp(key, "primary_monitor_first") == 0) {
primary_monitor_first = atoi(value);
} else if (strcmp(key, "panel_shrink") == 0) { } else if (strcmp(key, "panel_shrink") == 0) {
panel_shrink = atoi(value); panel_shrink = atoi(value);
} else if (strcmp(key, "panel_size") == 0) { } else if (strcmp(key, "panel_size") == 0) {
@@ -408,6 +421,7 @@ void add_entry(char *key, char *value)
} }
} else if (strcmp(key, "panel_items") == 0) { } else if (strcmp(key, "panel_items") == 0) {
new_config_file = TRUE; new_config_file = TRUE;
free_and_null(panel_items_order);
panel_items_order = strdup(value); panel_items_order = strdup(value);
systray_enabled = 0; systray_enabled = 0;
launcher_enabled = 0; launcher_enabled = 0;
@@ -425,7 +439,7 @@ void add_entry(char *key, char *value)
#ifdef ENABLE_BATTERY #ifdef ENABLE_BATTERY
battery_enabled = 1; battery_enabled = 1;
#else #else
fprintf(stderr, "tint2 is build without battery support\n"); fprintf(stderr, "tint2: tint2 has been compiled without battery support\n");
#endif #endif
} }
if (panel_items_order[j] == 'S') { if (panel_items_order[j] == 'S') {
@@ -544,6 +558,11 @@ void add_entry(char *key, char *value)
#ifdef ENABLE_BATTERY #ifdef ENABLE_BATTERY
if (strlen(value) > 0) if (strlen(value) > 0)
battery_low_cmd = strdup(value); battery_low_cmd = strdup(value);
#endif
} else if (strcmp(key, "battery_full_cmd") == 0) {
#ifdef ENABLE_BATTERY
if (strlen(value) > 0)
battery_full_cmd = strdup(value);
#endif #endif
} else if (strcmp(key, "ac_connected_cmd") == 0) { } else if (strcmp(key, "ac_connected_cmd") == 0) {
#ifdef ENABLE_BATTERY #ifdef ENABLE_BATTERY
@@ -564,6 +583,21 @@ void add_entry(char *key, char *value)
#ifdef ENABLE_BATTERY #ifdef ENABLE_BATTERY
bat2_font_desc = pango_font_description_from_string(value); bat2_font_desc = pango_font_description_from_string(value);
bat2_has_font = TRUE; bat2_has_font = TRUE;
#endif
} else if (strcmp(key, "bat1_format") == 0) {
#ifdef ENABLE_BATTERY
if (strlen(value) > 0) {
free(bat1_format);
bat1_format = strdup(value);
battery_enabled = 1;
}
#endif
} else if (strcmp(key, "bat2_format") == 0) {
#ifdef ENABLE_BATTERY
if (strlen(value) > 0) {
free(bat2_format);
bat2_format = strdup(value);
}
#endif #endif
} else if (strcmp(key, "battery_font_color") == 0) { } else if (strcmp(key, "battery_font_color") == 0) {
#ifdef ENABLE_BATTERY #ifdef ENABLE_BATTERY
@@ -626,7 +660,7 @@ void add_entry(char *key, char *value)
else if (g_str_equal(value, "dots")) else if (g_str_equal(value, "dots"))
separator->style = SEPARATOR_DOTS; separator->style = SEPARATOR_DOTS;
else else
fprintf(stderr, RED "Invalid separator_style value: %s" RESET "\n", value); fprintf(stderr, RED "tint2: Invalid separator_style value: %s" RESET "\n", value);
} else if (strcmp(key, "separator_size") == 0) { } else if (strcmp(key, "separator_size") == 0) {
Separator *separator = get_or_create_last_separator(); Separator *separator = get_or_create_last_separator();
separator->thickness = atoi(value); separator->thickness = atoi(value);
@@ -652,8 +686,8 @@ void add_entry(char *key, char *value)
Execp *execp = get_or_create_last_execp(); Execp *execp = get_or_create_last_execp();
execp->backend->interval = 0; execp->backend->interval = 0;
int v = atoi(value); int v = atoi(value);
if (v < 1) { if (v < 0) {
fprintf(stderr, "execp_interval must be an integer >= 1\n"); fprintf(stderr, "tint2: execp_interval must be an integer >= 0\n");
} else { } else {
execp->backend->interval = v; execp->backend->interval = v;
} }
@@ -673,6 +707,7 @@ void add_entry(char *key, char *value)
Execp *execp = get_or_create_last_execp(); Execp *execp = get_or_create_last_execp();
free_and_null(execp->backend->tooltip); free_and_null(execp->backend->tooltip);
execp->backend->tooltip = strdup(value); execp->backend->tooltip = strdup(value);
execp->backend->has_user_tooltip = TRUE;
} else if (strcmp(key, "execp_font") == 0) { } else if (strcmp(key, "execp_font") == 0) {
Execp *execp = get_or_create_last_execp(); Execp *execp = get_or_create_last_execp();
pango_font_description_free(execp->backend->font_desc); pango_font_description_free(execp->backend->font_desc);
@@ -708,7 +743,7 @@ void add_entry(char *key, char *value)
Execp *execp = get_or_create_last_execp(); Execp *execp = get_or_create_last_execp();
int v = atoi(value); int v = atoi(value);
if (v < 0) { if (v < 0) {
fprintf(stderr, "execp_icon_w must be an integer >= 0\n"); fprintf(stderr, "tint2: execp_icon_w must be an integer >= 0\n");
} else { } else {
execp->backend->icon_w = v; execp->backend->icon_w = v;
} }
@@ -716,7 +751,7 @@ void add_entry(char *key, char *value)
Execp *execp = get_or_create_last_execp(); Execp *execp = get_or_create_last_execp();
int v = atoi(value); int v = atoi(value);
if (v < 0) { if (v < 0) {
fprintf(stderr, "execp_icon_h must be an integer >= 0\n"); fprintf(stderr, "tint2: execp_icon_h must be an integer >= 0\n");
} else { } else {
execp->backend->icon_h = v; execp->backend->icon_h = v;
} }
@@ -750,17 +785,23 @@ void add_entry(char *key, char *value)
/* Button */ /* Button */
else if (strcmp(key, "button") == 0) { else if (strcmp(key, "button") == 0) {
panel_config.button_list = g_list_append(panel_config.button_list, create_button()); panel_config.button_list = g_list_append(panel_config.button_list, create_button());
} else if (strcmp(key, "button_icon") == 0 && strlen(value)) { } else if (strcmp(key, "button_icon") == 0) {
if (strlen(value)) {
Button *button = get_or_create_last_button(); Button *button = get_or_create_last_button();
button->backend->icon_name = strdup(value); button->backend->icon_name = strdup(value);
} else if (strcmp(key, "button_text") == 0 && strlen(value)) { }
} else if (strcmp(key, "button_text") == 0) {
if (strlen(value)) {
Button *button = get_or_create_last_button(); Button *button = get_or_create_last_button();
free_and_null(button->backend->text); free_and_null(button->backend->text);
button->backend->text = strdup(value); button->backend->text = strdup(value);
} else if (strcmp(key, "button_tooltip") == 0 && strlen(value)) { }
} else if (strcmp(key, "button_tooltip") == 0) {
if (strlen(value)) {
Button *button = get_or_create_last_button(); Button *button = get_or_create_last_button();
free_and_null(button->backend->tooltip); free_and_null(button->backend->tooltip);
button->backend->tooltip = strdup(value); button->backend->tooltip = strdup(value);
}
} else if (strcmp(key, "button_font") == 0) { } else if (strcmp(key, "button_font") == 0) {
Button *button = get_or_create_last_button(); Button *button = get_or_create_last_button();
pango_font_description_free(button->backend->font_desc); pango_font_description_free(button->backend->font_desc);
@@ -960,6 +1001,8 @@ void add_entry(char *key, char *value)
hide_inactive_tasks = atoi(value); hide_inactive_tasks = atoi(value);
} else if (strcmp(key, "taskbar_hide_different_monitor") == 0) { } else if (strcmp(key, "taskbar_hide_different_monitor") == 0) {
hide_task_diff_monitor = atoi(value); hide_task_diff_monitor = atoi(value);
} else if (strcmp(key, "taskbar_hide_different_desktop") == 0) {
hide_task_diff_desktop = atoi(value);
} else if (strcmp(key, "taskbar_hide_if_empty") == 0) { } else if (strcmp(key, "taskbar_hide_if_empty") == 0) {
hide_taskbar_if_empty = atoi(value); hide_taskbar_if_empty = atoi(value);
} else if (strcmp(key, "taskbar_always_show_all_desktop_tasks") == 0) { } else if (strcmp(key, "taskbar_always_show_all_desktop_tasks") == 0) {
@@ -1049,11 +1092,18 @@ void add_entry(char *key, char *value)
panel_config.g_task.config_background_mask |= (1 << status); panel_config.g_task.config_background_mask |= (1 << status);
if (status == TASK_NORMAL) if (status == TASK_NORMAL)
panel_config.g_task.area.bg = panel_config.g_task.background[TASK_NORMAL]; panel_config.g_task.area.bg = panel_config.g_task.background[TASK_NORMAL];
if (panel_config.g_task.background[status]->border_content_tint_weight > 0 ||
panel_config.g_task.background[status]->fill_content_tint_weight > 0)
panel_config.g_task.has_content_tint = TRUE;
} }
} }
// "tooltip" is deprecated but here for backwards compatibility // "tooltip" is deprecated but here for backwards compatibility
else if (strcmp(key, "task_tooltip") == 0 || strcmp(key, "tooltip") == 0) else if (strcmp(key, "task_tooltip") == 0 || strcmp(key, "tooltip") == 0)
panel_config.g_task.tooltip_enabled = atoi(value); panel_config.g_task.tooltip_enabled = atoi(value);
else if (strcmp(key, "task_thumbnail") == 0)
panel_config.g_task.thumbnail_enabled = atoi(value);
else if (strcmp(key, "task_thumbnail_size") == 0)
panel_config.g_task.thumbnail_width = MAX(8, atoi(value));
/* Systray */ /* Systray */
else if (strcmp(key, "systray_padding") == 0) { else if (strcmp(key, "systray_padding") == 0) {
@@ -1094,10 +1144,13 @@ void add_entry(char *key, char *value)
systray.saturation = atoi(value2); systray.saturation = atoi(value2);
systray.brightness = atoi(value3); systray.brightness = atoi(value3);
} else if (strcmp(key, "systray_monitor") == 0) { } else if (strcmp(key, "systray_monitor") == 0) {
systray_monitor = atoi(value) - 1; systray_monitor = MAX(0, config_get_monitor(value));
} else if (strcmp(key, "systray_name_filter") == 0) { } else if (strcmp(key, "systray_name_filter") == 0) {
if (systray_hide_name_filter) if (systray_hide_name_filter) {
fprintf(stderr, "tint2: Error: duplicate option 'systray_name_filter'. Please use it only once. See "
"https://gitlab.com/o9000/tint2/issues/652\n");
free(systray_hide_name_filter); free(systray_hide_name_filter);
}
systray_hide_name_filter = strdup(value); systray_hide_name_filter = strdup(value);
} }
@@ -1250,8 +1303,13 @@ void add_entry(char *key, char *value)
} }
} }
#endif #endif
else else if (strcmp(key, "primary_monitor_first") == 0) {
fprintf(stderr, "tint2 : invalid option \"%s\",\n upgrade tint2 or correct your config file\n", key); fprintf(stderr,
"tint2: deprecated config option \"%s\"\n"
" Please see the documentation regarding the alternatives.\n",
key);
} else
fprintf(stderr, "tint2: invalid option \"%s\",\n upgrade tint2 or correct your config file\n", key);
if (value1) if (value1)
free(value1); free(value1);
@@ -1263,11 +1321,13 @@ void add_entry(char *key, char *value)
gboolean config_read_file(const char *path) gboolean config_read_file(const char *path)
{ {
fprintf(stderr, "tint2: Loading config file: %s\n", path);
FILE *fp = fopen(path, "r"); FILE *fp = fopen(path, "r");
if (!fp) if (!fp)
return FALSE; return FALSE;
char* line = NULL; char *line = NULL;
size_t line_size = 0; size_t line_size = 0;
while (getline(&line, &line_size, fp) >= 0) { while (getline(&line, &line_size, fp) >= 0) {
char *key, *value; char *key, *value;
@@ -1331,11 +1391,12 @@ gboolean config_read_default_path()
// copy tint2rc from system directory to user directory // copy tint2rc from system directory to user directory
fprintf(stderr, "tint2 warning: could not find a config file! Creating a default one.\n"); fprintf(stderr, "tint2: could not find a config file! Creating a default one.\n");
// According to the XDG Base Directory Specification (https://specifications.freedesktop.org/basedir-spec/basedir-spec-0.6.html) // According to the XDG Base Directory Specification
// (https://specifications.freedesktop.org/basedir-spec/basedir-spec-0.6.html)
// if the user's config directory does not exist, we should create it with permissions set to 0700. // if the user's config directory does not exist, we should create it with permissions set to 0700.
if (!g_file_test(g_get_user_config_dir(), G_FILE_TEST_IS_DIR)) if (!g_file_test(g_get_user_config_dir(), G_FILE_TEST_IS_DIR))
g_mkdir(g_get_user_config_dir(), 0700); g_mkdir_with_parents(g_get_user_config_dir(), 0700);
gchar *path2 = 0; gchar *path2 = 0;
system_dirs = g_get_system_config_dirs(); system_dirs = g_get_system_config_dirs();
@@ -1352,7 +1413,7 @@ gboolean config_read_default_path()
// copy file in user directory (path1) // copy file in user directory (path1)
gchar *dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL); gchar *dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL);
if (!g_file_test(dir, G_FILE_TEST_IS_DIR)) if (!g_file_test(dir, G_FILE_TEST_IS_DIR))
g_mkdir(dir, 0700); g_mkdir_with_parents(dir, 0700);
g_free(dir); g_free(dir);
path1 = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL); path1 = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL);
@@ -1368,7 +1429,7 @@ gboolean config_read_default_path()
// generate config file // generate config file
gchar *dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL); gchar *dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL);
if (!g_file_test(dir, G_FILE_TEST_IS_DIR)) if (!g_file_test(dir, G_FILE_TEST_IS_DIR))
g_mkdir(dir, 0700); g_mkdir_with_parents(dir, 0700);
g_free(dir); g_free(dir);
path1 = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL); path1 = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL);

444
src/drag_and_drop.c Normal file
View File

@@ -0,0 +1,444 @@
/**************************************************************************
* Copyright (C) 2017 tint2 authors
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "drag_and_drop.h"
#include "panel.h"
#include "server.h"
#include "task.h"
// Drag and Drop state variables
static Window dnd_source_window;
static Window dnd_target_window;
static int dnd_version;
static Atom dnd_selection;
static Atom dnd_atom;
static int dnd_sent_request;
static LauncherIcon *dnd_launcher_icon;
gboolean debug_dnd = FALSE;
gboolean hidden_panel_shown_for_dnd;
// This fetches all the data from a property
struct Property dnd_read_property(Display *disp, Window w, Atom property)
{
Atom actual_type;
int actual_format;
unsigned long nitems;
unsigned long bytes_after;
unsigned char *ret = 0;
int read_bytes = 1024;
// Keep trying to read the property until there are no
// bytes unread.
do {
if (ret != 0)
XFree(ret);
XGetWindowProperty(disp,
w,
property,
0,
read_bytes,
False,
AnyPropertyType,
&actual_type,
&actual_format,
&nitems,
&bytes_after,
&ret);
read_bytes *= 2;
} while (bytes_after != 0);
if (debug_dnd)
fprintf(stderr, "tint2: DnD %s:%d: Property:\n", __FILE__, __LINE__);
fprintf(stderr, "tint2: DnD %s:%d: Actual type: %s\n", __FILE__, __LINE__, GetAtomName(disp, actual_type));
fprintf(stderr, "tint2: DnD %s:%d: Actual format: %d\n", __FILE__, __LINE__, actual_format);
fprintf(stderr, "tint2: DnD %s:%d: Number of items: %lu\n", __FILE__, __LINE__, nitems);
Property p;
p.data = ret;
p.format = actual_format;
p.nitems = nitems;
p.type = actual_type;
return p;
}
// This function takes a list of targets which can be converted to (atom_list, nitems)
// and a list of acceptable targets with prioritees (datatypes). It returns the highest
// entry in datatypes which is also in atom_list: ie it finds the best match.
Atom dnd_pick_target_from_list(Display *disp, Atom *atom_list, int nitems)
{
Atom to_be_requested = None;
int i;
for (i = 0; i < nitems; i++) {
const char *atom_name = GetAtomName(disp, atom_list[i]);
fprintf(stderr, "tint2: DnD %s:%d: Type %d = %s\n", __FILE__, __LINE__, i, atom_name);
// See if this data type is allowed and of higher priority (closer to zero)
// than the present one.
if (strcasecmp(atom_name, "STRING") == 0) {
to_be_requested = atom_list[i];
} else if (strcasecmp(atom_name, "text/uri-list") == 0 && !to_be_requested) {
to_be_requested = atom_list[i];
}
}
fprintf(stderr,
"tint2: DnD %s:%d: Accepting: Type %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, to_be_requested));
return to_be_requested;
}
// Finds the best target given up to three atoms provided (any can be None).
// Useful for part of the Xdnd protocol.
Atom dnd_pick_target_from_atoms(Display *disp, Atom t1, Atom t2, Atom t3)
{
Atom atoms[3];
int n = 0;
if (t1 != None)
atoms[n++] = t1;
if (t2 != None)
atoms[n++] = t2;
if (t3 != None)
atoms[n++] = t3;
return dnd_pick_target_from_list(disp, atoms, n);
}
// Finds the best target given a local copy of a property.
Atom dnd_pick_target_from_targets(Display *disp, Property p)
{
// The list of targets is a list of atoms, so it should have type XA_ATOM
// but it may have the type TARGETS instead.
if ((p.type != XA_ATOM && p.type != server.atom.TARGETS) || p.format != 32) {
// This would be really broken. Targets have to be an atom list
// and applications should support this. Nevertheless, some
// seem broken (MATLAB 7, for instance), so ask for STRING
// next instead as the lowest common denominator
return XA_STRING;
} else {
Atom *atom_list = (Atom *)p.data;
return dnd_pick_target_from_list(disp, atom_list, p.nitems);
}
}
void dnd_init()
{
dnd_source_window = 0;
dnd_target_window = 0;
dnd_version = 0;
dnd_selection = XInternAtom(server.display, "PRIMARY", 0);
dnd_atom = None;
dnd_sent_request = 0;
dnd_launcher_icon = NULL;
hidden_panel_shown_for_dnd = FALSE;
}
void handle_dnd_enter(XClientMessageEvent *e)
{
dnd_atom = None;
int more_than_3 = e->data.l[1] & 1;
dnd_source_window = e->data.l[0];
dnd_version = (e->data.l[1] >> 24);
if (debug_dnd) {
fprintf(stderr, "tint2: DnD %s:%d: DnDEnter\n", __FILE__, __LINE__);
fprintf(stderr,
"DnD %s:%d: DnDEnter. Supports > 3 types = %s\n",
__FILE__,
__LINE__,
more_than_3 ? "yes" : "no");
fprintf(stderr, "tint2: DnD %s:%d: Protocol version = %d\n", __FILE__, __LINE__, dnd_version);
fprintf(stderr,
"tint2: DnD %s:%d: Type 1 = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, e->data.l[2]));
fprintf(stderr,
"tint2: DnD %s:%d: Type 2 = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, e->data.l[3]));
fprintf(stderr,
"tint2: DnD %s:%d: Type 3 = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, e->data.l[4]));
}
// Query which conversions are available and pick the best
if (more_than_3) {
// Fetch the list of possible conversions
// Notice the similarity to TARGETS with paste.
Property p = dnd_read_property(server.display, dnd_source_window, server.atom.XdndTypeList);
dnd_atom = dnd_pick_target_from_targets(server.display, p);
XFree(p.data);
} else {
// Use the available list
dnd_atom = dnd_pick_target_from_atoms(server.display, e->data.l[2], e->data.l[3], e->data.l[4]);
}
if (debug_dnd)
fprintf(stderr,
"tint2: DnD %s:%d: Requested type = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, dnd_atom));
}
void handle_dnd_position(XClientMessageEvent *e)
{
dnd_target_window = e->window;
int accept = 0;
Panel *panel = get_panel(e->window);
int x, y, mapX, mapY;
Window child;
x = (e->data.l[2] >> 16) & 0xFFFF;
y = e->data.l[2] & 0xFFFF;
XTranslateCoordinates(server.display, server.root_win, e->window, x, y, &mapX, &mapY, &child);
Task *task = click_task(panel, mapX, mapY);
if (task) {
if (task->desktop != server.desktop)
change_desktop(task->desktop);
task_handle_mouse_event(task, TOGGLE);
} else {
LauncherIcon *icon = click_launcher_icon(panel, mapX, mapY);
if (icon) {
accept = 1;
dnd_launcher_icon = icon;
} else {
dnd_launcher_icon = NULL;
}
}
// send XdndStatus event to get more XdndPosition events
XClientMessageEvent se;
se.type = ClientMessage;
se.window = e->data.l[0];
se.message_type = server.atom.XdndStatus;
se.format = 32;
se.data.l[0] = e->window; // XID of the target window
se.data.l[1] = accept ? 1 : 0; // bit 0: accept drop bit 1: send XdndPosition events if inside rectangle
se.data.l[2] = 0; // Rectangle x,y for which no more XdndPosition events
se.data.l[3] = (1 << 16) | 1; // Rectangle w,h for which no more XdndPosition events
if (accept) {
se.data.l[4] = server.atom.XdndActionCopy;
} else {
se.data.l[4] = None; // None = drop will not be accepted
}
if (debug_dnd)
fprintf(stderr,
"tint2: DnD %s:%d: Accepted: %s\n",
__FILE__,
__LINE__,
accept ? GetAtomName(server.display, (Atom)se.data.l[4]) : "no");
XSendEvent(server.display, e->data.l[0], False, NoEventMask, (XEvent *)&se);
}
void handle_dnd_drop(XClientMessageEvent *e)
{
if (dnd_target_window && dnd_launcher_icon) {
if (dnd_version >= 1) {
XConvertSelection(server.display,
server.atom.XdndSelection,
dnd_atom,
dnd_selection,
dnd_target_window,
e->data.l[2]);
} else {
XConvertSelection(server.display,
server.atom.XdndSelection,
dnd_atom,
dnd_selection,
dnd_target_window,
CurrentTime);
}
} else {
// The source is sending anyway, despite instructions to the contrary.
// So reply that we're not interested.
XClientMessageEvent m;
memset(&m, 0, sizeof(m));
m.type = ClientMessage;
m.display = e->display;
m.window = e->data.l[0];
m.message_type = server.atom.XdndFinished;
m.format = 32;
m.data.l[0] = dnd_target_window;
m.data.l[1] = 0;
m.data.l[2] = None; // Failed.
XSendEvent(server.display, e->data.l[0], False, NoEventMask, (XEvent *)&m);
}
}
void handle_dnd_selection_notify(XSelectionEvent *e)
{
Atom target = e->target;
if (debug_dnd) {
fprintf(stderr, "tint2: DnD %s:%d: A selection notify has arrived!\n", __FILE__, __LINE__);
fprintf(stderr,
"DnD %s:%d: Selection atom = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, e->selection));
fprintf(stderr, "tint2: DnD %s:%d: Target atom = %s\n", __FILE__, __LINE__, GetAtomName(server.display, target));
fprintf(stderr,
"DnD %s:%d: Property atom = %s\n",
__FILE__,
__LINE__,
GetAtomName(server.display, e->property));
}
if (dnd_launcher_icon) {
Property prop = dnd_read_property(server.display, dnd_target_window, dnd_selection);
if (prop.data) {
// If we're being given a list of targets (possible conversions)
if (target == server.atom.TARGETS && !dnd_sent_request) {
dnd_sent_request = 1;
dnd_atom = dnd_pick_target_from_targets(server.display, prop);
if (dnd_atom == None) {
if (debug_dnd)
fprintf(stderr, "tint2: No matching datatypes.\n");
} else {
// Request the data type we are able to select
if (debug_dnd)
fprintf(stderr, "tint2: Now requsting type %s", GetAtomName(server.display, dnd_atom));
XConvertSelection(server.display,
dnd_selection,
dnd_atom,
dnd_selection,
dnd_target_window,
CurrentTime);
}
} else if (target == dnd_atom) {
// Dump the binary data
if (debug_dnd) {
fprintf(stderr, "tint2: DnD %s:%d: Received data:\n", __FILE__, __LINE__);
fprintf(stderr, "tint2: --------\n");
for (int i = 0; i < prop.nitems * prop.format / 8; i++)
fprintf(stderr, "%c", ((char *)prop.data)[i]);
fprintf(stderr, "tint2: --------\n");
}
// https://standards.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html#exec-variables
GString *cmd = g_string_new(dnd_launcher_icon->cmd);
const char *atom_name = GetAtomName(server.display, prop.type);
if (strcasecmp(atom_name, "STRING") == 0 || strcasecmp(atom_name, "text/uri-list") == 0) {
GString *url = g_string_new("");
GString *prev_url = g_string_new("");
gboolean must_unescape = strcasecmp(atom_name, "text/uri-list") == 0;
for (int i = 0; i < prop.nitems * prop.format / 8; i++) {
char c = ((char *)prop.data)[i];
if (c == '\n') {
if (must_unescape) {
char *raw = g_uri_unescape_string(url->str, NULL);
if (raw) {
g_string_assign(url, raw);
}
free(raw);
}
// Many programs cannot handle this prefix
tint2_g_string_replace(url, "file://", "");
// Some programs put duplicates in the list, we remove them
if (strcmp(url->str, prev_url->str) != 0) {
if (strstr(cmd->str, "%F")) {
GString *piece = g_string_new("");
g_string_append(piece, " \"");
g_string_append(piece, url->str);
g_string_append(piece, "\"");
g_string_append(piece, " %F");
tint2_g_string_replace(cmd, "%F", piece->str);
g_string_free(piece, TRUE);
} else if (strstr(cmd->str, "%f")) {
GString *piece = g_string_new("");
g_string_append(piece, " \"");
g_string_append(piece, url->str);
g_string_append(piece, "\"");
tint2_g_string_replace(cmd, "%f", piece->str);
g_string_free(piece, TRUE);
break;
} else {
g_string_append(cmd, " \"");
g_string_append(cmd, url->str);
g_string_append(cmd, "\"");
}
}
g_string_assign(prev_url, url->str);
g_string_assign(url, "");
} else if (c == '\r') {
// Nothing to do
} else {
if (c == '`' || c == '$' || c == '\\') {
g_string_append(url, "\\");
}
g_string_append_c(url, c);
}
}
g_string_free(url, TRUE);
g_string_free(prev_url, TRUE);
}
tint2_g_string_replace(cmd, "%F", "");
tint2_g_string_replace(cmd, "%f", "");
if (debug_dnd)
fprintf(stderr, "tint2: DnD %s:%d: Running command: %s\n", __FILE__, __LINE__, cmd->str);
tint_exec(cmd->str,
NULL,
NULL,
e->time,
NULL,
0,
0,
dnd_launcher_icon->start_in_terminal,
dnd_launcher_icon->startup_notification);
g_string_free(cmd, TRUE);
// Reply OK.
XClientMessageEvent m;
memset(&m, 0, sizeof(m));
m.type = ClientMessage;
m.display = server.display;
m.window = dnd_source_window;
m.message_type = server.atom.XdndFinished;
m.format = 32;
m.data.l[0] = dnd_target_window;
m.data.l[1] = 1;
m.data.l[2] = server.atom.XdndActionCopy; // We only ever copy.
XSendEvent(server.display, dnd_source_window, False, NoEventMask, (XEvent *)&m);
XSync(server.display, False);
}
XFree(prop.data);
}
}
}

22
src/drag_and_drop.h Normal file
View File

@@ -0,0 +1,22 @@
/**************************************************************************
* Copyright (C) 2017 tint2 authors
*
**************************************************************************/
#ifndef DRAG_AND_DROP_H
#define DRAG_AND_DROP_H
#include <X11/Xlib.h>
#include <glib.h>
extern gboolean hidden_panel_shown_for_dnd;
extern gboolean debug_dnd;
void dnd_init();
void handle_dnd_enter(XClientMessageEvent *e);
void handle_dnd_position(XClientMessageEvent *e);
void handle_dnd_drop(XClientMessageEvent *e);
void handle_dnd_selection_notify(XSelectionEvent *e);
#endif

View File

@@ -18,6 +18,8 @@
#include "timer.h" #include "timer.h"
#include "common.h" #include "common.h"
#define MAX_TOOLTIP_LEN 4096
void execp_timer_callback(void *arg); void execp_timer_callback(void *arg);
char *execp_get_tooltip(void *obj); char *execp_get_tooltip(void *obj);
void execp_init_fonts(); void execp_init_fonts();
@@ -30,9 +32,10 @@ void default_execp()
Execp *create_execp() Execp *create_execp()
{ {
Execp *execp = calloc(1, sizeof(Execp)); Execp *execp = (Execp *)calloc(1, sizeof(Execp));
execp->backend = calloc(1, sizeof(ExecpBackend)); execp->backend = (ExecpBackend *)calloc(1, sizeof(ExecpBackend));
execp->backend->child_pipe = -1; execp->backend->child_pipe_stdout = -1;
execp->backend->child_pipe_stderr = -1;
execp->backend->cmd_pids = g_tree_new(cmp_ptr); execp->backend->cmd_pids = g_tree_new(cmp_ptr);
execp->backend->interval = 30; execp->backend->interval = 30;
execp->backend->cache_icon = TRUE; execp->backend->cache_icon = TRUE;
@@ -45,10 +48,10 @@ gpointer create_execp_frontend(gconstpointer arg, gpointer data)
{ {
Execp *execp_backend = (Execp *)arg; Execp *execp_backend = (Execp *)arg;
Execp *execp_frontend = calloc(1, sizeof(Execp)); Execp *execp_frontend = (Execp *)calloc(1, sizeof(Execp));
execp_frontend->backend = execp_backend->backend; execp_frontend->backend = execp_backend->backend;
execp_backend->backend->instances = g_list_append(execp_backend->backend->instances, execp_frontend); execp_backend->backend->instances = g_list_append(execp_backend->backend->instances, execp_frontend);
execp_frontend->frontend = calloc(1, sizeof(ExecpFrontend)); execp_frontend->frontend = (ExecpFrontend *)calloc(1, sizeof(ExecpFrontend));
return execp_frontend; return execp_frontend;
} }
@@ -67,21 +70,22 @@ void destroy_execp(void *obj)
stop_timeout(execp->backend->timer); stop_timeout(execp->backend->timer);
execp->backend->timer = NULL; execp->backend->timer = NULL;
if (execp->backend->icon) { free_icon(execp->backend->icon);
imlib_context_set_image(execp->backend->icon); free_and_null(execp->backend->buf_stdout);
imlib_free_image(); free_and_null(execp->backend->buf_stderr);
execp->backend->icon = NULL;
}
free_and_null(execp->backend->buf_output);
free_and_null(execp->backend->text); free_and_null(execp->backend->text);
free_and_null(execp->backend->icon_path); free_and_null(execp->backend->icon_path);
if (execp->backend->child) { if (execp->backend->child) {
kill(-execp->backend->child, SIGHUP); kill(-execp->backend->child, SIGHUP);
execp->backend->child = 0; execp->backend->child = 0;
} }
if (execp->backend->child_pipe >= 0) { if (execp->backend->child_pipe_stdout >= 0) {
close(execp->backend->child_pipe); close(execp->backend->child_pipe_stdout);
execp->backend->child_pipe = -1; execp->backend->child_pipe_stdout = -1;
}
if (execp->backend->child_pipe_stderr >= 0) {
close(execp->backend->child_pipe_stderr);
execp->backend->child_pipe_stderr = -1;
} }
if (execp->backend->cmd_pids) { if (execp->backend->cmd_pids) {
g_tree_destroy(execp->backend->cmd_pids); g_tree_destroy(execp->backend->cmd_pids);
@@ -100,8 +104,8 @@ void destroy_execp(void *obj)
free_and_null(execp->backend->uwheel_command); free_and_null(execp->backend->uwheel_command);
if (execp->backend->instances) { if (execp->backend->instances) {
fprintf(stderr, "Error: Attempt to destroy backend while there are still frontend instances!\n"); fprintf(stderr, "tint2: Error: Attempt to destroy backend while there are still frontend instances!\n");
exit(-1); exit(EXIT_FAILURE);
} }
free(execp->backend); free(execp->backend);
free(execp); free(execp);
@@ -138,9 +142,11 @@ void init_execp()
// Set missing config options // Set missing config options
if (!execp->backend->bg) if (!execp->backend->bg)
execp->backend->bg = &g_array_index(backgrounds, Background, 0); execp->backend->bg = &g_array_index(backgrounds, Background, 0);
execp->backend->buf_capacity = 1024; execp->backend->buf_stdout_capacity = 1024;
execp->backend->buf_output = calloc(execp->backend->buf_capacity, 1); execp->backend->buf_stdout = calloc(execp->backend->buf_stdout_capacity, 1);
execp->backend->text = strdup(" "); execp->backend->buf_stderr_capacity = 1024;
execp->backend->buf_stderr = calloc(execp->backend->buf_stderr_capacity, 1);
execp->backend->text = strdup("");
execp->backend->icon_path = NULL; execp->backend->icon_path = NULL;
} }
} }
@@ -186,8 +192,9 @@ void init_execp_panel(void *p)
execp->area.on_screen = TRUE; execp->area.on_screen = TRUE;
instantiate_area_gradients(&execp->area); instantiate_area_gradients(&execp->area);
if (!execp->backend->timer)
execp->backend->timer = add_timeout(10, 0, execp_timer_callback, execp, &execp->backend->timer); execp->backend->timer = add_timeout(10, 0, execp_timer_callback, execp, &execp->backend->timer);
execp_update_post_read(execp);
} }
} }
@@ -290,156 +297,140 @@ gboolean reload_icon(Execp *execp)
return FALSE; return FALSE;
} }
int execp_compute_desired_size(void *obj) void execp_compute_icon_text_geometry(Execp *execp,
int *horiz_padding,
int *vert_padding,
int *interior_padding,
int *icon_w,
int *icon_h,
gboolean *text_next_line,
int *txt_height_ink,
int *txt_height,
int *txt_width,
int *new_size,
gboolean *resized)
{ {
Execp *execp = (Execp *)obj;
Panel *panel = (Panel *)execp->area.panel; Panel *panel = (Panel *)execp->area.panel;
int horiz_padding = (panel_horizontal ? execp->area.paddingxlr : execp->area.paddingy); Area *area = &execp->area;
int vert_padding = (panel_horizontal ? execp->area.paddingy : execp->area.paddingxlr); *horiz_padding = (panel_horizontal ? area->paddingxlr : area->paddingy);
int interior_padding = execp->area.paddingx; *vert_padding = (panel_horizontal ? area->paddingy : area->paddingxlr);
*interior_padding = area->paddingx;
int icon_w, icon_h;
if (reload_icon(execp)) { if (reload_icon(execp)) {
if (execp->backend->icon) { if (execp->backend->icon) {
imlib_context_set_image(execp->backend->icon); imlib_context_set_image(execp->backend->icon);
icon_w = imlib_image_get_width(); *icon_w = imlib_image_get_width();
icon_h = imlib_image_get_height(); *icon_h = imlib_image_get_height();
} else { } else {
icon_w = icon_h = 0; *icon_w = *icon_h = 0;
} }
} else { } else {
icon_w = icon_h = 0; *icon_w = *icon_h = 0;
} }
int text_next_line = !panel_horizontal && icon_w > execp->area.width / 2; *text_next_line = !panel_horizontal && *icon_w > area->width / 2;
int available_w, available_h;
if (panel_horizontal) {
available_w = panel->area.width;
available_h = area->height - 2 * area->paddingy - left_right_border_width(area);
} else {
available_w = !text_next_line
? area->width - *icon_w - (*icon_w ? *interior_padding : 0) - 2 * *horiz_padding -
left_right_border_width(area)
: area->width - 2 * *horiz_padding - left_right_border_width(area);
available_h = panel->area.height;
}
get_text_size2(execp->backend->font_desc,
txt_height_ink,
txt_height,
txt_width,
available_h,
available_w,
execp->backend->text,
strlen(execp->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
execp->backend->has_markup);
*resized = FALSE;
if (panel_horizontal) {
*new_size = *txt_width;
if (*icon_w)
*new_size += *interior_padding + *icon_w;
*new_size += 2 * *horiz_padding + left_right_border_width(area);
if (*new_size < area->width && abs(*new_size - area->width) < 6) {
// we try to limit the number of resizes
*new_size = area->width;
*resized = TRUE;
} else {
*resized = *new_size != area->width;
}
} else {
if (!*text_next_line) {
*new_size = *txt_height + 2 * *vert_padding + top_bottom_border_width(area);
*new_size = MAX(*new_size, *icon_h + 2 * *vert_padding + top_bottom_border_width(area));
} else {
*new_size = *icon_h + *interior_padding + *txt_height + 2 * *vert_padding + top_bottom_border_width(area);
}
if (*new_size != area->height) {
*resized = TRUE;
}
}
}
int execp_compute_desired_size(void *obj)
{
Execp *execp = (Execp *)obj;
int horiz_padding, vert_padding, interior_padding;
int icon_w, icon_h;
gboolean text_next_line;
int txt_height_ink, txt_height, txt_width; int txt_height_ink, txt_height, txt_width;
if (panel_horizontal) {
get_text_size2(execp->backend->font_desc,
&txt_height_ink,
&txt_height,
&txt_width,
panel->area.height,
panel->area.width,
execp->backend->text,
strlen(execp->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
execp->backend->has_markup);
} else {
get_text_size2(execp->backend->font_desc,
&txt_height_ink,
&txt_height,
&txt_width,
panel->area.height,
!text_next_line
? execp->area.width - icon_w - (icon_w ? interior_padding : 0) - 2 * horiz_padding -
left_right_border_width(&execp->area)
: execp->area.width - 2 * horiz_padding - left_right_border_width(&execp->area),
execp->backend->text,
strlen(execp->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
execp->backend->has_markup);
}
if (panel_horizontal) {
int new_size = txt_width;
if (icon_w)
new_size += interior_padding + icon_w;
new_size += 2 * horiz_padding + left_right_border_width(&execp->area);
return new_size;
} else {
int new_size; int new_size;
if (!text_next_line) { gboolean resized;
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area); execp_compute_icon_text_geometry(execp,
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&execp->area)); &horiz_padding,
} else { &vert_padding,
new_size = &interior_padding,
icon_h + interior_padding + txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area); &icon_w,
} &icon_h,
&text_next_line,
&txt_height_ink,
&txt_height,
&txt_width,
&new_size,
&resized);
return new_size; return new_size;
}
} }
gboolean resize_execp(void *obj) gboolean resize_execp(void *obj)
{ {
Execp *execp = (Execp *)obj; Execp *execp = (Execp *)obj;
Panel *panel = (Panel *)execp->area.panel; int horiz_padding, vert_padding, interior_padding;
int horiz_padding = (panel_horizontal ? execp->area.paddingxlr : execp->area.paddingy);
int vert_padding = (panel_horizontal ? execp->area.paddingy : execp->area.paddingxlr);
int interior_padding = execp->area.paddingx;
int icon_w, icon_h; int icon_w, icon_h;
if (reload_icon(execp)) { gboolean text_next_line;
if (execp->backend->icon) {
imlib_context_set_image(execp->backend->icon);
icon_w = imlib_image_get_width();
icon_h = imlib_image_get_height();
} else {
icon_w = icon_h = 0;
}
} else {
icon_w = icon_h = 0;
}
int text_next_line = !panel_horizontal && icon_w > execp->area.width / 2;
int txt_height_ink, txt_height, txt_width; int txt_height_ink, txt_height, txt_width;
if (panel_horizontal) {
get_text_size2(execp->backend->font_desc,
&txt_height_ink,
&txt_height,
&txt_width,
panel->area.height,
panel->area.width,
execp->backend->text,
strlen(execp->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
execp->backend->has_markup);
} else {
get_text_size2(execp->backend->font_desc,
&txt_height_ink,
&txt_height,
&txt_width,
panel->area.height,
!text_next_line
? execp->area.width - icon_w - (icon_w ? interior_padding : 0) - 2 * horiz_padding -
left_right_border_width(&execp->area)
: execp->area.width - 2 * horiz_padding - left_right_border_width(&execp->area),
execp->backend->text,
strlen(execp->backend->text),
PANGO_WRAP_WORD_CHAR,
PANGO_ELLIPSIZE_NONE,
execp->backend->has_markup);
}
gboolean result = FALSE;
if (panel_horizontal) {
int new_size = txt_width;
if (icon_w)
new_size += interior_padding + icon_w;
new_size += 2 * horiz_padding + left_right_border_width(&execp->area);
if (new_size > execp->area.width || new_size < (execp->area.width - 6)) {
// we try to limit the number of resize
execp->area.width = new_size + 1;
result = TRUE;
}
} else {
int new_size; int new_size;
if (!text_next_line) { gboolean resized;
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area); execp_compute_icon_text_geometry(execp,
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&execp->area)); &horiz_padding,
} else { &vert_padding,
new_size = &interior_padding,
icon_h + interior_padding + txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area); &icon_w,
} &icon_h,
if (new_size != execp->area.height) { &text_next_line,
&txt_height_ink,
&txt_height,
&txt_width,
&new_size,
&resized);
if (panel_horizontal)
execp->area.width = new_size;
else
execp->area.height = new_size; execp->area.height = new_size;
result = TRUE;
}
}
execp->frontend->textw = txt_width; execp->frontend->textw = txt_width;
execp->frontend->texth = txt_height; execp->frontend->texth = txt_height;
if (execp->backend->centered) { if (execp->backend->centered) {
@@ -479,13 +470,12 @@ gboolean resize_execp(void *obj)
} }
schedule_redraw(&execp->area); schedule_redraw(&execp->area);
return resized;
return result;
} }
void draw_execp(void *obj, cairo_t *c) void draw_execp(void *obj, cairo_t *c)
{ {
Execp *execp = obj; Execp *execp = (Execp *)obj;
PangoLayout *layout = pango_cairo_create_layout(c); PangoLayout *layout = pango_cairo_create_layout(c);
if (execp->backend->has_icon && execp->backend->icon) { if (execp->backend->has_icon && execp->backend->icon) {
@@ -518,13 +508,13 @@ void draw_execp(void *obj, cairo_t *c)
void execp_dump_geometry(void *obj, int indent) void execp_dump_geometry(void *obj, int indent)
{ {
Execp *execp = obj; Execp *execp = (Execp *)obj;
if (execp->backend->has_icon && execp->backend->icon) { if (execp->backend->has_icon && execp->backend->icon) {
Imlib_Image tmp = imlib_context_get_image(); Imlib_Image tmp = imlib_context_get_image();
imlib_context_set_image(execp->backend->icon); imlib_context_set_image(execp->backend->icon);
fprintf(stderr, fprintf(stderr,
"%*sIcon: x = %d, y = %d, w = %d, h = %d\n", "tint2: %*sIcon: x = %d, y = %d, w = %d, h = %d\n",
indent, indent,
"", "",
execp->frontend->iconx, execp->frontend->iconx,
@@ -535,7 +525,7 @@ void execp_dump_geometry(void *obj, int indent)
imlib_context_set_image(tmp); imlib_context_set_image(tmp);
} }
fprintf(stderr, fprintf(stderr,
"%*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n", "tint2: %*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n",
indent, indent,
"", "",
execp->frontend->textx, execp->frontend->textx,
@@ -547,7 +537,7 @@ void execp_dump_geometry(void *obj, int indent)
void execp_force_update(Execp *execp) void execp_force_update(Execp *execp)
{ {
if (execp->backend->child_pipe > 0) { if (execp->backend->child_pipe_stdout > 0) {
// Command currently running, nothing to do // Command currently running, nothing to do
} else { } else {
if (execp->backend->timer) if (execp->backend->timer)
@@ -557,9 +547,9 @@ void execp_force_update(Execp *execp)
} }
} }
void execp_action(void *obj, int button, int x, int y) void execp_action(void *obj, int button, int x, int y, Time time)
{ {
Execp *execp = obj; Execp *execp = (Execp *)obj;
char *command = NULL; char *command = NULL;
switch (button) { switch (button) {
case 1: case 1:
@@ -579,30 +569,17 @@ void execp_action(void *obj, int button, int x, int y)
break; break;
} }
if (command) { if (command) {
char *full_cmd = g_strdup_printf("export EXECP_X=%d;" setenvd("EXECP_X", x);
"export EXECP_Y=%d;" setenvd("EXECP_Y", y);
"export EXECP_W=%d;" setenvd("EXECP_W", execp->area.width);
"export EXECP_H=%d; %s", setenvd("EXECP_H", execp->area.height);
x, pid_t pid = tint_exec(command, NULL, NULL, time, obj, x, y, FALSE, TRUE);
y, unsetenv("EXECP_X");
execp->area.width, unsetenv("EXECP_Y");
execp->area.height, unsetenv("EXECP_W");
command); unsetenv("EXECP_H");
pid_t pid = fork(); if (pid > 0)
if (pid < 0) {
fprintf(stderr, "Could not fork\n");
} else if (pid == 0) {
// Child process
// Allow children to exist after parent destruction
setsid();
// Run the command
execl("/bin/sh", "/bin/sh", "-c", full_cmd, NULL);
fprintf(stderr, "Failed to execlp %s\n", full_cmd);
exit(1);
}
// Parent process
g_tree_insert(execp->backend->cmd_pids, GINT_TO_POINTER(pid), GINT_TO_POINTER(1)); g_tree_insert(execp->backend->cmd_pids, GINT_TO_POINTER(pid), GINT_TO_POINTER(1));
g_free(full_cmd);
} else { } else {
execp_force_update(execp); execp_force_update(execp);
} }
@@ -616,78 +593,93 @@ void execp_cmd_completed(Execp *execp, pid_t pid)
void execp_timer_callback(void *arg) void execp_timer_callback(void *arg)
{ {
Execp *execp = arg; Execp *execp = (Execp *)arg;
if (!execp->backend->command) if (!execp->backend->command)
return; return;
// Still running! // Still running!
if (execp->backend->child_pipe > 0) if (execp->backend->child_pipe_stdout > 0)
return; return;
int pipe_fd[2]; int pipe_fd_stdout[2];
if (pipe(pipe_fd)) { if (pipe(pipe_fd_stdout)) {
// TODO maybe write this in tooltip, but if this happens we're screwed anyways // TODO maybe write this in tooltip, but if this happens we're screwed anyways
fprintf(stderr, "Execp: Creating pipe failed!\n"); fprintf(stderr, "tint2: Execp: Creating pipe failed!\n");
return; return;
} }
fcntl(pipe_fd[0], F_SETFL, O_NONBLOCK | fcntl(pipe_fd[0], F_GETFL)); fcntl(pipe_fd_stdout[0], F_SETFL, O_NONBLOCK | fcntl(pipe_fd_stdout[0], F_GETFL));
int pipe_fd_stderr[2];
if (pipe(pipe_fd_stderr)) {
close(pipe_fd_stdout[1]);
close(pipe_fd_stdout[0]);
// TODO maybe write this in tooltip, but if this happens we're screwed anyways
fprintf(stderr, "tint2: Execp: Creating pipe failed!\n");
return;
}
fcntl(pipe_fd_stderr[0], F_SETFL, O_NONBLOCK | fcntl(pipe_fd_stderr[0], F_GETFL));
// Fork and run command, capturing stdout in pipe // Fork and run command, capturing stdout in pipe
pid_t child = fork(); pid_t child = fork();
if (child == -1) { if (child == -1) {
// TODO maybe write this in tooltip, but if this happens we're screwed anyways // TODO maybe write this in tooltip, but if this happens we're screwed anyways
fprintf(stderr, "Fork failed.\n"); fprintf(stderr, "tint2: Fork failed.\n");
close(pipe_fd[1]); close(pipe_fd_stdout[1]);
close(pipe_fd[0]); close(pipe_fd_stdout[0]);
close(pipe_fd_stderr[1]);
close(pipe_fd_stderr[0]);
return; return;
} else if (child == 0) { } else if (child == 0) {
fprintf(stderr, "Executing: %s\n", execp->backend->command); fprintf(stderr, "tint2: Executing: %s\n", execp->backend->command);
// We are in the child // We are in the child
close(pipe_fd[0]); close(pipe_fd_stdout[0]);
dup2(pipe_fd[1], 1); // 1 is stdout dup2(pipe_fd_stdout[1], 1); // 1 is stdout
close(pipe_fd[1]); close(pipe_fd_stdout[1]);
close(pipe_fd_stderr[0]);
dup2(pipe_fd_stderr[1], 2); // 2 is stderr
close(pipe_fd_stderr[1]);
close_all_fds();
setpgid(0, 0); setpgid(0, 0);
execl("/bin/sh", "/bin/sh", "-c", execp->backend->command, NULL); execl("/bin/sh", "/bin/sh", "-c", execp->backend->command, NULL);
// This should never happen! // This should never happen!
fprintf(stdout, "execl() failed\nexecl() failed\n"); fprintf(stderr, "execl() failed\nexecl() failed\n");
fflush(stdout);
exit(0); exit(0);
} }
close(pipe_fd[1]); close(pipe_fd_stdout[1]);
close(pipe_fd_stderr[1]);
execp->backend->child = child; execp->backend->child = child;
execp->backend->child_pipe = pipe_fd[0]; execp->backend->child_pipe_stdout = pipe_fd_stdout[0];
execp->backend->buf_length = 0; execp->backend->child_pipe_stderr = pipe_fd_stderr[0];
execp->backend->buf_output[execp->backend->buf_length] = '\0'; execp->backend->buf_stdout_length = 0;
execp->backend->buf_stdout[execp->backend->buf_stdout_length] = '\0';
execp->backend->buf_stderr_length = 0;
execp->backend->buf_stderr[execp->backend->buf_stderr_length] = '\0';
execp->backend->last_update_start_time = time(NULL); execp->backend->last_update_start_time = time(NULL);
} }
gboolean read_execp(void *obj) void read_from_pipe(int fd, char **buffer, ssize_t *buffer_length, ssize_t *buffer_capacity, gboolean *eof)
{ {
Execp *execp = (Execp *)obj; *eof = FALSE;
if (execp->backend->child_pipe < 0)
return FALSE;
gboolean command_finished = FALSE;
while (1) { while (1) {
// Make sure there is free space in the buffer // Make sure there is free space in the buffer
if (execp->backend->buf_capacity - execp->backend->buf_length < 1024) { if (*buffer_capacity - *buffer_length < 1024) {
execp->backend->buf_capacity *= 2; *buffer_capacity *= 2;
execp->backend->buf_output = realloc(execp->backend->buf_output, execp->backend->buf_capacity); *buffer = (char *)realloc(*buffer, *buffer_capacity);
} }
ssize_t count = read(execp->backend->child_pipe, ssize_t count = read(fd,
execp->backend->buf_output + execp->backend->buf_length, *buffer + *buffer_length,
execp->backend->buf_capacity - execp->backend->buf_length - 1); *buffer_capacity - *buffer_length - 1);
if (count > 0) { if (count > 0) {
// Successful read // Successful read
execp->backend->buf_length += count; *buffer_length += count;
execp->backend->buf_output[execp->backend->buf_length] = '\0'; (*buffer)[*buffer_length] = '\0';
continue; continue;
} else if (count == 0) { } else if (count == 0) {
// End of file // End of file
command_finished = TRUE; *eof = TRUE;
break; break;
} else if (errno == EAGAIN || errno == EWOULDBLOCK) { } else if (errno == EAGAIN || errno == EWOULDBLOCK) {
// No more data available at the moment // No more data available at the moment
@@ -697,28 +689,87 @@ gboolean read_execp(void *obj)
continue; continue;
} else { } else {
// Error // Error
command_finished = TRUE; *eof = TRUE;
break; break;
} }
break; break;
} }
}
gboolean starts_with(char *s, char *prefix)
{
char *p, *q;
for (p = s, q = prefix; *p && *q; p++, q++) {
if (*p != *q)
return FALSE;
}
return *q == '\0';
}
char *last_substring(char *s, char *sub)
{
char *result = NULL;
for (char *p = s; *p; p++) {
if (starts_with(p, sub))
result = p;
}
return result;
}
void rstrip(char *s)
{
size_t len = strlen(s);
while (len > 0) {
if (s[len-1] == ' ' || s[len-1] == '\n') {
s[len-1] = 0;
len--;
} else {
break;
}
}
}
gboolean read_execp(void *obj)
{
Execp *execp = (Execp *)obj;
if (execp->backend->child_pipe_stdout < 0)
return FALSE;
gboolean stdout_eof, stderr_eof;
read_from_pipe(execp->backend->child_pipe_stdout,
&execp->backend->buf_stdout,
&execp->backend->buf_stdout_length,
&execp->backend->buf_stdout_capacity,
&stdout_eof);
read_from_pipe(execp->backend->child_pipe_stderr,
&execp->backend->buf_stderr,
&execp->backend->buf_stderr_length,
&execp->backend->buf_stderr_capacity,
&stderr_eof);
gboolean command_finished = stdout_eof && stderr_eof;
if (command_finished) { if (command_finished) {
execp->backend->child = 0; execp->backend->child = 0;
close(execp->backend->child_pipe); close(execp->backend->child_pipe_stdout);
execp->backend->child_pipe = -1; execp->backend->child_pipe_stdout = -1;
close(execp->backend->child_pipe_stderr);
execp->backend->child_pipe_stderr = -1;
if (execp->backend->interval) if (execp->backend->interval)
execp->backend->timer = execp->backend->timer =
add_timeout(execp->backend->interval * 1000, 0, execp_timer_callback, execp, &execp->backend->timer); add_timeout(execp->backend->interval * 1000, 0, execp_timer_callback, execp, &execp->backend->timer);
} }
char *ansi_clear_screen = (char*)"\x1b[2J";
if (!execp->backend->continuous && command_finished) { if (!execp->backend->continuous && command_finished) {
// Handle stdout
free_and_null(execp->backend->text); free_and_null(execp->backend->text);
free_and_null(execp->backend->icon_path); free_and_null(execp->backend->icon_path);
if (!execp->backend->has_icon) { if (!execp->backend->has_icon) {
execp->backend->text = strdup(execp->backend->buf_output); execp->backend->text = strdup(execp->backend->buf_stdout);
} else { } else {
char *text = strchr(execp->backend->buf_output, '\n'); char *text = strchr(execp->backend->buf_stdout, '\n');
if (text) { if (text) {
*text = '\0'; *text = '\0';
text++; text++;
@@ -726,41 +777,75 @@ gboolean read_execp(void *obj)
} else { } else {
execp->backend->text = strdup(""); execp->backend->text = strdup("");
} }
execp->backend->icon_path = strdup(execp->backend->buf_output); execp->backend->icon_path = strdup(execp->backend->buf_stdout);
} }
int len = strlen(execp->backend->text); int len = strlen(execp->backend->text);
if (len > 0 && execp->backend->text[len - 1] == '\n') if (len > 0 && execp->backend->text[len - 1] == '\n')
execp->backend->text[len - 1] = '\0'; execp->backend->text[len - 1] = '\0';
execp->backend->buf_length = 0; execp->backend->buf_stdout_length = 0;
execp->backend->buf_output[execp->backend->buf_length] = '\0'; execp->backend->buf_stdout[execp->backend->buf_stdout_length] = '\0';
// Handle stderr
if (!execp->backend->has_user_tooltip) {
free_and_null(execp->backend->tooltip);
char *start = last_substring(execp->backend->buf_stderr, ansi_clear_screen);
if (start)
start += strlen(ansi_clear_screen);
else
start = execp->backend->buf_stderr;
if (*start) {
execp->backend->tooltip = strdup(start);
rstrip(execp->backend->tooltip);
if (strlen(execp->backend->tooltip) > MAX_TOOLTIP_LEN)
execp->backend->tooltip[MAX_TOOLTIP_LEN] = '\0';
}
}
execp->backend->buf_stderr_length = 0;
execp->backend->buf_stderr[execp->backend->buf_stderr_length] = '\0';
//
execp->backend->last_update_finish_time = time(NULL); execp->backend->last_update_finish_time = time(NULL);
execp->backend->last_update_duration = execp->backend->last_update_duration =
execp->backend->last_update_finish_time - execp->backend->last_update_start_time; execp->backend->last_update_finish_time - execp->backend->last_update_start_time;
return TRUE; return TRUE;
} else if (execp->backend->continuous > 0) { } else if (execp->backend->continuous > 0) {
// Handle stderr
if (!execp->backend->has_user_tooltip) {
free_and_null(execp->backend->tooltip);
char *start = last_substring(execp->backend->buf_stderr, ansi_clear_screen);
if (start) {
start += strlen(ansi_clear_screen);
memmove(execp->backend->buf_stderr, start, strlen(start) + 1);
execp->backend->buf_stderr_length = (ssize_t)strlen(execp->backend->buf_stderr);
}
if (execp->backend->buf_stderr_length > MAX_TOOLTIP_LEN) {
execp->backend->buf_stderr_length = MAX_TOOLTIP_LEN;
execp->backend->buf_stderr[execp->backend->buf_stderr_length] = '\0';
}
execp->backend->tooltip = strdup(execp->backend->buf_stderr);
rstrip(execp->backend->tooltip);
} else {
execp->backend->buf_stderr_length = 0;
execp->backend->buf_stderr[execp->backend->buf_stderr_length] = '\0';
}
// Handle stdout
// Count lines in buffer // Count lines in buffer
int num_lines = 0; int num_lines = 0;
char *last = execp->backend->buf_output;
char *end = NULL; char *end = NULL;
for (char *c = execp->backend->buf_output; *c; c++) { for (char *c = execp->backend->buf_stdout; *c; c++) {
if (*c == '\n') { if (*c == '\n') {
num_lines++; num_lines++;
if (num_lines == execp->backend->continuous) if (num_lines == execp->backend->continuous)
end = c; end = c;
} }
last = c;
} }
if (*last && *last != '\n')
num_lines++;
if (num_lines >= execp->backend->continuous) { if (num_lines >= execp->backend->continuous) {
if (end) if (end)
*end = '\0'; *end = '\0';
free_and_null(execp->backend->text); free_and_null(execp->backend->text);
free_and_null(execp->backend->icon_path); free_and_null(execp->backend->icon_path);
if (!execp->backend->has_icon) { if (!execp->backend->has_icon) {
execp->backend->text = strdup(execp->backend->buf_output); execp->backend->text = strdup(execp->backend->buf_stdout);
} else { } else {
char *text = strchr(execp->backend->buf_output, '\n'); char *text = strchr(execp->backend->buf_stdout, '\n');
if (text) { if (text) {
*text = '\0'; *text = '\0';
text++; text++;
@@ -768,23 +853,23 @@ gboolean read_execp(void *obj)
} else { } else {
execp->backend->text = strdup(""); execp->backend->text = strdup("");
} }
execp->backend->icon_path = strdup(execp->backend->buf_output); execp->backend->icon_path = strdup(execp->backend->buf_stdout);
} }
int len = strlen(execp->backend->text); size_t len = strlen(execp->backend->text);
if (len > 0 && execp->backend->text[len - 1] == '\n') if (len > 0 && execp->backend->text[len - 1] == '\n')
execp->backend->text[len - 1] = '\0'; execp->backend->text[len - 1] = '\0';
if (end) { if (end) {
char *next = end + 1; char *next = end + 1;
int copied = next - execp->backend->buf_output; ssize_t copied = next - execp->backend->buf_stdout;
int remaining = execp->backend->buf_length - copied; ssize_t remaining = execp->backend->buf_stdout_length - copied;
if (remaining > 0) { if (remaining > 0) {
memmove(execp->backend->buf_output, next, remaining); memmove(execp->backend->buf_stdout, next, (size_t)remaining);
execp->backend->buf_length = remaining; execp->backend->buf_stdout_length = remaining;
execp->backend->buf_output[execp->backend->buf_length] = '\0'; execp->backend->buf_stdout[execp->backend->buf_stdout_length] = '\0';
} else { } else {
execp->backend->buf_length = 0; execp->backend->buf_stdout_length = 0;
execp->backend->buf_output[execp->backend->buf_length] = '\0'; execp->backend->buf_stdout[execp->backend->buf_stdout_length] = '\0';
} }
} }
@@ -819,7 +904,7 @@ const char *time_to_string(int seconds, char *buffer)
char *execp_get_tooltip(void *obj) char *execp_get_tooltip(void *obj)
{ {
Execp *execp = obj; Execp *execp = (Execp *)obj;
if (execp->backend->tooltip) { if (execp->backend->tooltip) {
if (strlen(execp->backend->tooltip) > 0) if (strlen(execp->backend->tooltip) > 0)
@@ -833,7 +918,7 @@ char *execp_get_tooltip(void *obj)
char tmp_buf1[256]; char tmp_buf1[256];
char tmp_buf2[256]; char tmp_buf2[256];
char tmp_buf3[256]; char tmp_buf3[256];
if (execp->backend->child_pipe < 0) { if (execp->backend->child_pipe_stdout < 0) {
// Not executing command // Not executing command
if (execp->backend->last_update_finish_time) { if (execp->backend->last_update_finish_time) {
// We updated at least once // We updated at least once
@@ -871,3 +956,44 @@ char *execp_get_tooltip(void *obj)
} }
return strdup(execp->backend->tooltip_text); return strdup(execp->backend->tooltip_text);
} }
void execp_update_post_read(Execp *execp)
{
int icon_h, icon_w;
if (reload_icon(execp)) {
if (execp->backend->icon) {
imlib_context_set_image(execp->backend->icon);
icon_w = imlib_image_get_width();
icon_h = imlib_image_get_height();
} else {
icon_w = icon_h = 0;
}
} else {
icon_w = icon_h = 0;
}
if ((icon_h == 0 || icon_w == 0) && execp->backend->text[0] == 0) {
// Easy to test with bash -c 'R=$(( RANDOM % 2 )); [ $R -eq 0 ] && echo HELLO $R'
if (execp->area.on_screen)
hide(&execp->area);
} else {
if (!execp->area.on_screen)
show(&execp->area);
execp->area.resize_needed = TRUE;
schedule_panel_redraw();
}
}
void handle_execp_events()
{
for (GList *l = panel_config.execp_list; l; l = l->next) {
Execp *execp = (Execp *)l->data;
if (read_execp(execp)) {
GList *l_instance;
for (l_instance = execp->backend->instances; l_instance; l_instance = l_instance->next) {
Execp *instance = (Execp *)l_instance->data;
execp_update_post_read(instance);
}
}
}
}

View File

@@ -27,6 +27,7 @@ typedef struct ExecpBackend {
gboolean cache_icon; gboolean cache_icon;
int icon_w; int icon_w;
int icon_h; int icon_h;
gboolean has_user_tooltip;
char *tooltip; char *tooltip;
gboolean centered; gboolean centered;
gboolean has_font; gboolean has_font;
@@ -46,20 +47,24 @@ typedef struct ExecpBackend {
// Backend state: // Backend state:
timeout *timer; timeout *timer;
int child_pipe; int child_pipe_stdout;
int child_pipe_stderr;
pid_t child; pid_t child;
// Command output buffer // Command output buffer
char *buf_output; char *buf_stdout;
int buf_length; ssize_t buf_stdout_length;
int buf_capacity; ssize_t buf_stdout_capacity;
char *buf_stderr;
ssize_t buf_stderr_length;
ssize_t buf_stderr_capacity;
// Text extracted from the output buffer // Text extracted from the output buffer
char *text; char *text;
// Icon path extracted from the output buffer // Icon path extracted from the output buffer
char *icon_path; char *icon_path;
Imlib_Image icon; Imlib_Image icon;
char tooltip_text[512]; gchar tooltip_text[512];
// The time the last command was started // The time the last command was started
time_t last_update_start_time; time_t last_update_start_time;
@@ -129,7 +134,7 @@ void draw_execp(void *obj, cairo_t *c);
gboolean resize_execp(void *obj); gboolean resize_execp(void *obj);
// Called on mouse click event. // Called on mouse click event.
void execp_action(void *obj, int button, int x, int y); void execp_action(void *obj, int button, int x, int y, Time time);
void execp_cmd_completed(Execp *obj, pid_t pid); void execp_cmd_completed(Execp *obj, pid_t pid);
@@ -138,6 +143,11 @@ void execp_cmd_completed(Execp *obj, pid_t pid);
// Returns 1 if the output has been updated and a redraw is needed. // Returns 1 if the output has been updated and a redraw is needed.
gboolean read_execp(void *obj); gboolean read_execp(void *obj);
// Called for Execp front elements when the command output has changed.
void execp_update_post_read(Execp *execp);
void execp_default_font_changed(); void execp_default_font_changed();
void handle_execp_events();
#endif // EXECPLUGIN_H #endif // EXECPLUGIN_H

View File

@@ -42,7 +42,7 @@ void init_freespace_panel(void *p)
for (size_t k = 0; k < strlen(panel_items_order); k++) { for (size_t k = 0; k < strlen(panel_items_order); k++) {
if (panel_items_order[k] == 'F') { if (panel_items_order[k] == 'F') {
FreeSpace *freespace = (FreeSpace *) calloc(1, sizeof(FreeSpace)); FreeSpace *freespace = (FreeSpace *)calloc(1, sizeof(FreeSpace));
panel->freespace_list = g_list_append(panel->freespace_list, freespace); panel->freespace_list = g_list_append(panel->freespace_list, freespace);
if (!freespace->area.bg) if (!freespace->area.bg)
freespace->area.bg = &g_array_index(backgrounds, Background, 0); freespace->area.bg = &g_array_index(backgrounds, Background, 0);
@@ -98,7 +98,7 @@ int freespace_get_max_size(Panel *p)
int freespace_area_compute_desired_size(void *obj) int freespace_area_compute_desired_size(void *obj)
{ {
FreeSpace *freespace = (FreeSpace *) obj; FreeSpace *freespace = (FreeSpace *)obj;
return freespace_get_max_size((Panel *)freespace->area.panel); return freespace_get_max_size((Panel *)freespace->area.panel);
} }

294
src/init.c Normal file
View File

@@ -0,0 +1,294 @@
#include "init.h"
#include <locale.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xatom.h>
#include <X11/extensions/XShm.h>
#include "config.h"
#include "drag_and_drop.h"
#include "fps_distribution.h"
#include "panel.h"
#include "server.h"
#include "signals.h"
#include "tooltip.h"
#include "tracing.h"
#include "uevent.h"
#include "version.h"
void print_usage()
{
fprintf(stdout,
"Usage: tint2 [OPTION...]\n"
"\n"
"Options:\n"
" -c path_to_config_file Loads the configuration file from a\n"
" custom location.\n"
" -v, --version Prints version information and exits.\n"
" -h, --help Display this help and exits.\n"
"\n"
"For more information, run `man tint2` or visit the project page\n"
"<https://gitlab.com/o9000/tint2>.\n");
}
void handle_cli_arguments(int argc, char **argv)
{
// Read command line arguments
for (int i = 1; i < argc; ++i) {
gboolean error = FALSE;
if (strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0) {
print_usage();
exit(0);
} else if (strcmp(argv[i], "-v") == 0 || strcmp(argv[i], "--version") == 0) {
fprintf(stdout, "tint2 version %s\n", VERSION_STRING);
exit(0);
} else if (strcmp(argv[i], "-c") == 0) {
if (i + 1 < argc) {
i++;
config_path = strdup(argv[i]);
} else {
error = TRUE;
}
} else if (strcmp(argv[i], "-s") == 0) {
if (i + 1 < argc) {
i++;
snapshot_path = strdup(argv[i]);
} else {
error = TRUE;
}
} else if (i + 1 == argc) {
config_path = strdup(argv[i]);
}
#ifdef ENABLE_BATTERY
else if (strcmp(argv[i], "--battery-sys-prefix") == 0) {
if (i + 1 < argc) {
i++;
battery_sys_prefix = strdup(argv[i]);
} else {
error = TRUE;
}
}
#endif
else {
error = TRUE;
}
if (error) {
print_usage();
exit(EXIT_FAILURE);
}
}
}
void handle_env_vars()
{
debug_geometry = getenv("DEBUG_GEOMETRY") != NULL;
debug_gradients = getenv("DEBUG_GRADIENTS") != NULL;
debug_icons = getenv("DEBUG_ICONS") != NULL;
debug_fps = getenv("DEBUG_FPS") != NULL;
debug_frames = getenv("DEBUG_FRAMES") != NULL;
debug_dnd = getenv("DEBUG_DND") != NULL;
debug_thumbnails = getenv("DEBUG_THUMBNAILS") != NULL;
if (debug_fps) {
init_fps_distribution();
char *s = getenv("TRACING_FPS_THRESHOLD");
if (!s || sscanf(s, "%lf", &tracing_fps_threshold) != 1) {
tracing_fps_threshold = 60;
}
}
}
static timeout *detect_compositor_timer = NULL;
static int detect_compositor_timer_counter = 0;
void detect_compositor(void *arg)
{
if (server.composite_manager) {
stop_timeout(detect_compositor_timer);
return;
}
detect_compositor_timer_counter--;
if (detect_compositor_timer_counter < 0) {
stop_timeout(detect_compositor_timer);
return;
}
// No compositor, check for one
if (XGetSelectionOwner(server.display, server.atom._NET_WM_CM_S0) != None) {
stop_timeout(detect_compositor_timer);
// Restart tint2
fprintf(stderr, "tint2: Detected compositor, restarting tint2...\n");
kill(getpid(), SIGUSR1);
}
}
void start_detect_compositor()
{
// Already have a compositor, nothing to do
if (server.composite_manager)
return;
stop_timeout(detect_compositor_timer);
// Check every 0.5 seconds for up to 30 seconds
detect_compositor_timer_counter = 60;
detect_compositor_timer = add_timeout(500, 500, detect_compositor, 0, &detect_compositor_timer);
}
void create_default_elements()
{
default_timeout();
default_systray();
memset(&server, 0, sizeof(server));
#ifdef ENABLE_BATTERY
default_battery();
#endif
default_clock();
default_launcher();
default_taskbar();
default_tooltip();
default_execp();
default_button();
default_panel();
}
void init_post_config()
{
server_init_visual();
server_init_xdamage();
imlib_context_set_display(server.display);
imlib_context_set_visual(server.visual);
imlib_context_set_colormap(server.colormap);
init_signals_postconfig();
// load default icon
const gchar *const *data_dirs = g_get_system_data_dirs();
for (int i = 0; data_dirs[i] != NULL; i++) {
gchar *path = g_build_filename(data_dirs[i], "tint2", "default_icon.png", NULL);
if (g_file_test(path, G_FILE_TEST_EXISTS))
default_icon = load_image(path, TRUE);
g_free(path);
}
if (!default_icon) {
fprintf(stderr,
RED "Could not load default_icon.png. Please check that tint2 has been installed correctly!" RESET
"\n");
}
XSync(server.display, False);
}
void init_X11_pre_config()
{
server.display = XOpenDisplay(NULL);
if (!server.display) {
fprintf(stderr, "tint2: could not open display!\n");
exit(EXIT_FAILURE);
}
server.x11_fd = ConnectionNumber(server.display);
XSetErrorHandler((XErrorHandler)server_catch_error);
XSetIOErrorHandler((XIOErrorHandler)x11_io_error);
server_init_atoms();
server.screen = DefaultScreen(server.display);
server.root_win = RootWindow(server.display, server.screen);
server.desktop = get_current_desktop();
server.has_shm = XShmQueryExtension(server.display);
// Needed since the config file uses '.' as decimal separator
setlocale(LC_ALL, "");
setlocale(LC_NUMERIC, "POSIX");
/* Catch events */
XSelectInput(server.display, server.root_win, PropertyChangeMask | StructureNotifyMask);
// get monitor and desktop config
get_monitors();
get_desktops();
server.disable_transparency = 0;
xsettings_client = xsettings_client_new(server.display, server.screen, xsettings_notify_cb, NULL, NULL);
}
void init(int argc, char **argv)
{
setlinebuf(stdout);
setlinebuf(stderr);
default_config();
handle_cli_arguments(argc, argv);
create_default_elements();
init_signals();
handle_env_vars();
init_X11_pre_config();
if (!config_read()) {
fprintf(stderr, "tint2: Could not read config file.\n");
print_usage();
cleanup();
return;
}
init_post_config();
start_detect_compositor();
init_panel();
}
void cleanup()
{
#ifdef HAVE_SN
if (startup_notifications) {
sn_display_unref(server.sn_display);
server.sn_display = NULL;
}
#endif // HAVE_SN
cleanup_button();
cleanup_execp();
cleanup_systray();
cleanup_tooltip();
cleanup_clock();
cleanup_launcher();
#ifdef ENABLE_BATTERY
cleanup_battery();
#endif
cleanup_separator();
cleanup_panel();
cleanup_config();
if (default_icon) {
imlib_context_set_image(default_icon);
imlib_free_image();
default_icon = NULL;
}
imlib_context_disconnect_display();
xsettings_client_destroy(xsettings_client);
xsettings_client = NULL;
cleanup_server();
cleanup_timeout();
if (server.display)
XCloseDisplay(server.display);
server.display = NULL;
if (sigchild_pipe_valid) {
sigchild_pipe_valid = FALSE;
close(sigchild_pipe[1]);
close(sigchild_pipe[0]);
}
uevent_cleanup();
cleanup_fps_distribution();
#ifdef HAVE_TRACING
cleanup_tracing();
#endif
}

7
src/init.h Normal file
View File

@@ -0,0 +1,7 @@
#ifndef INIT_H
#define INIT_H
void init(int argc, char **argv);
void cleanup();
#endif

View File

@@ -96,6 +96,10 @@ void expand_exec(DesktopEntry *entry, const char *path)
q += strlen("''"); q += strlen("''");
q += strlen(path); q += strlen(path);
q--; // To balance the q++ in the for q--; // To balance the q++ in the for
} else if (*p == 'f' || *p == 'F') {
sprintf(q, "%c%c", '%', *p);
q += 2;
q--; // To balance the q++ in the for
} else { } else {
// We don't care about other expansions // We don't care about other expansions
q--; // Delete the last % from q q--; // Delete the last % from q
@@ -113,10 +117,12 @@ gboolean read_desktop_file_full_path(const char *path, DesktopEntry *entry)
{ {
entry->name = entry->generic_name = entry->icon = entry->exec = entry->cwd = NULL; entry->name = entry->generic_name = entry->icon = entry->exec = entry->cwd = NULL;
entry->hidden_from_menus = FALSE; entry->hidden_from_menus = FALSE;
entry->start_in_terminal = FALSE;
entry->startup_notification = TRUE;
FILE *fp = fopen(path, "rt"); FILE *fp = fopen(path, "rt");
if (fp == NULL) { if (fp == NULL) {
fprintf(stderr, "Could not open file %s\n", path); fprintf(stderr, "tint2: Could not open file %s\n", path);
return FALSE; return FALSE;
} }
@@ -126,14 +132,14 @@ gboolean read_desktop_file_full_path(const char *path, DesktopEntry *entry)
int lang_index_default = 1; int lang_index_default = 1;
#define LANG_DBG 0 #define LANG_DBG 0
if (LANG_DBG) if (LANG_DBG)
printf("Languages:"); fprintf(stderr, "tint2: Languages:");
for (int i = 0; languages[i]; i++) { for (int i = 0; languages[i]; i++) {
lang_index_default = i + 1; lang_index_default = i + 1;
if (LANG_DBG) if (LANG_DBG)
printf(" %s", languages[i]); fprintf(stderr, "tint2: %s", languages[i]);
} }
if (LANG_DBG) if (LANG_DBG)
printf("\n"); fprintf(stderr, "tint2: \n");
// we currently do not know about any Name key at all, so use an invalid index // we currently do not know about any Name key at all, so use an invalid index
int lang_index_name = lang_index_default + 1; int lang_index_name = lang_index_default + 1;
int lang_index_generic_name = lang_index_default + 1; int lang_index_generic_name = lang_index_default + 1;
@@ -192,6 +198,10 @@ gboolean read_desktop_file_full_path(const char *path, DesktopEntry *entry)
entry->icon = strdup(value); entry->icon = strdup(value);
} else if (strcmp(key, "NoDisplay") == 0) { } else if (strcmp(key, "NoDisplay") == 0) {
entry->hidden_from_menus = strcasecmp(value, "true") == 0; entry->hidden_from_menus = strcasecmp(value, "true") == 0;
} else if (strcmp(key, "Terminal") == 0) {
entry->start_in_terminal = strcasecmp(value, "true") == 0;
} else if (strcmp(key, "StartupNotify") == 0) {
entry->startup_notification = strcasecmp(value, "true") == 0;
} }
} }
} }
@@ -278,11 +288,11 @@ void free_desktop_entry(DesktopEntry *entry)
void test_read_desktop_file() void test_read_desktop_file()
{ {
fprintf(stdout, "\033[1;33m"); fprintf(stderr, YELLOW);
DesktopEntry entry; DesktopEntry entry;
read_desktop_file("/usr/share/applications/firefox.desktop", &entry); read_desktop_file("/usr/share/applications/firefox.desktop", &entry);
printf("Name:%s GenericName:%s Icon:%s Exec:%s\n", entry.name, entry.generic_name, entry.icon, entry.exec); fprintf(stderr, "tint2: Name:%s GenericName:%s Icon:%s Exec:%s\n", entry.name, entry.generic_name, entry.icon, entry.exec);
fprintf(stdout, "\033[0m"); fprintf(stderr, RESET);
} }
GSList *apps_locations = NULL; GSList *apps_locations = NULL;

View File

@@ -17,6 +17,8 @@ typedef struct DesktopEntry {
char *path; char *path;
char *cwd; char *cwd;
gboolean hidden_from_menus; gboolean hidden_from_menus;
gboolean start_in_terminal;
gboolean startup_notification;
} DesktopEntry; } DesktopEntry;
// Parses a line of the form "key = value". Modifies the line. // Parses a line of the form "key = value". Modifies the line.

View File

@@ -29,6 +29,8 @@
#include "common.h" #include "common.h"
#include "cache.h" #include "cache.h"
gboolean debug_icons = FALSE;
#define ICON_DIR_TYPE_SCALABLE 0 #define ICON_DIR_TYPE_SCALABLE 0
#define ICON_DIR_TYPE_FIXED 1 #define ICON_DIR_TYPE_FIXED 1
#define ICON_DIR_TYPE_THRESHOLD 2 #define ICON_DIR_TYPE_THRESHOLD 2
@@ -90,7 +92,7 @@ IconTheme *load_theme_from_index(const char *file_name, const char *name)
size_t line_size; size_t line_size;
if ((f = fopen(file_name, "rt")) == NULL) { if ((f = fopen(file_name, "rt")) == NULL) {
fprintf(stderr, "Could not open theme '%s'\n", file_name); fprintf(stderr, "tint2: Could not open theme '%s'\n", file_name);
return NULL; return NULL;
} }
@@ -325,22 +327,22 @@ void free_themes(IconThemeWrapper *wrapper)
void test_launcher_read_theme_file() void test_launcher_read_theme_file()
{ {
fprintf(stdout, "\033[1;33m"); fprintf(stdout, YELLOW);
IconTheme *theme = load_theme("oxygen"); IconTheme *theme = load_theme("oxygen");
if (!theme) { if (!theme) {
printf("Could not load theme\n"); fprintf(stderr, "tint2: Could not load theme\n");
return; return;
} }
printf("Loaded theme: %s\n", theme->name); fprintf(stderr, "tint2: Loaded theme: %s\n", theme->name);
GSList *item = theme->list_inherits; GSList *item = theme->list_inherits;
while (item != NULL) { while (item != NULL) {
printf("Inherits:%s\n", (char *)item->data); fprintf(stderr, "tint2: Inherits:%s\n", (char *)item->data);
item = g_slist_next(item); item = g_slist_next(item);
} }
item = theme->list_directories; item = theme->list_directories;
while (item != NULL) { while (item != NULL) {
IconThemeDir *dir = item->data; IconThemeDir *dir = item->data;
printf("Dir:%s Size=%d MinSize=%d MaxSize=%d Threshold=%d Type=%s\n", fprintf(stderr, "tint2: Dir:%s Size=%d MinSize=%d MaxSize=%d Threshold=%d Type=%s\n",
dir->name, dir->name,
dir->size, dir->size,
dir->min_size, dir->min_size,
@@ -352,7 +354,7 @@ void test_launcher_read_theme_file()
: "?????"); : "?????");
item = g_slist_next(item); item = g_slist_next(item);
} }
fprintf(stdout, "\033[0m"); fprintf(stdout, RESET);
} }
gboolean str_list_contains(const GSList *list, const char *value) gboolean str_list_contains(const GSList *list, const char *value)
@@ -379,7 +381,7 @@ void load_themes_helper(const char *name, GSList **themes, GSList **queued)
char *queued_name = queue->data; char *queued_name = queue->data;
queue = g_slist_remove(queue, queued_name); queue = g_slist_remove(queue, queued_name);
fprintf(stderr, " '%s',", queued_name); fprintf(stderr, "tint2: '%s',", queued_name);
IconTheme *theme = load_theme(queued_name); IconTheme *theme = load_theme(queued_name);
if (theme != NULL) { if (theme != NULL) {
*themes = g_slist_append(*themes, theme); *themes = g_slist_append(*themes, theme);
@@ -399,7 +401,7 @@ void load_themes_helper(const char *name, GSList **themes, GSList **queued)
free(queued_name); free(queued_name);
} }
fprintf(stderr, "\n"); fprintf(stderr, "tint2: \n");
// Free the queue // Free the queue
GSList *l; GSList *l;
@@ -413,7 +415,7 @@ void load_default_theme(IconThemeWrapper *wrapper)
if (wrapper->_themes_loaded) if (wrapper->_themes_loaded)
return; return;
fprintf(stderr, GREEN "Loading icon theme %s:" RESET "\n", wrapper->icon_theme_name); fprintf(stderr, GREEN "tint2: Loading icon theme %s:" RESET "\n", wrapper->icon_theme_name);
load_themes_helper(wrapper->icon_theme_name, &wrapper->themes, &wrapper->_queued); load_themes_helper(wrapper->icon_theme_name, &wrapper->themes, &wrapper->_queued);
load_themes_helper("hicolor", &wrapper->themes, &wrapper->_queued); load_themes_helper("hicolor", &wrapper->themes, &wrapper->_queued);
@@ -426,7 +428,7 @@ void load_fallbacks(IconThemeWrapper *wrapper)
if (wrapper->_fallback_loaded) if (wrapper->_fallback_loaded)
return; return;
fprintf(stderr, RED "Loading additional icon themes (this means your icon theme is incomplete)..." RESET "\n"); fprintf(stderr, RED "tint2: Loading additional icon themes (this means your icon theme is incomplete)..." RESET "\n");
// Load wrapper->themes_fallback // Load wrapper->themes_fallback
const GSList *location; const GSList *location;
@@ -459,7 +461,7 @@ void load_icon_cache(IconThemeWrapper *wrapper)
if (wrapper->_cache.loaded) if (wrapper->_cache.loaded)
return; return;
fprintf(stderr, GREEN "Loading icon theme cache..." RESET "\n"); fprintf(stderr, GREEN "tint2: Loading icon theme cache..." RESET "\n");
gchar *cache_path = get_icon_cache_path(); gchar *cache_path = get_icon_cache_path();
load_cache(&wrapper->_cache, cache_path); load_cache(&wrapper->_cache, cache_path);
@@ -471,7 +473,7 @@ void save_icon_cache(IconThemeWrapper *wrapper)
if (!wrapper || !wrapper->_cache.dirty) if (!wrapper || !wrapper->_cache.dirty)
return; return;
fprintf(stderr, GREEN "Saving icon theme cache..." RESET "\n"); fprintf(stderr, GREEN "tint2: Saving icon theme cache..." RESET "\n");
gchar *cache_path = get_icon_cache_path(); gchar *cache_path = get_icon_cache_path();
save_cache(&wrapper->_cache, cache_path); save_cache(&wrapper->_cache, cache_path);
g_free(cache_path); g_free(cache_path);
@@ -482,7 +484,7 @@ IconThemeWrapper *load_themes(const char *icon_theme_name)
IconThemeWrapper *wrapper = calloc(1, sizeof(IconThemeWrapper)); IconThemeWrapper *wrapper = calloc(1, sizeof(IconThemeWrapper));
if (!icon_theme_name) { if (!icon_theme_name) {
fprintf(stderr, "Missing icon_theme_name theme, default to 'hicolor'.\n"); fprintf(stderr, "tint2: Missing icon_theme_name theme, default to 'hicolor'.\n");
icon_theme_name = "hicolor"; icon_theme_name = "hicolor";
} }
@@ -533,7 +535,6 @@ gint compare_theme_directories(gconstpointer a, gconstpointer b, gpointer size_q
return abs(da->size - size) - abs(db->size - size); return abs(da->size - size) - abs(db->size - size);
} }
#define DEBUG_ICON_SEARCH 0
char *get_icon_path_helper(GSList *themes, const char *icon_name, int size) char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
{ {
if (icon_name == NULL) if (icon_name == NULL)
@@ -585,6 +586,8 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
char *file_name = calloc(file_name_size, 1); char *file_name = calloc(file_name_size, 1);
for (theme = themes; theme; theme = g_slist_next(theme)) { for (theme = themes; theme; theme = g_slist_next(theme)) {
if (debug_icons)
fprintf(stderr, "tint2: Searching theme: %s\n", ((IconTheme *)theme->data)->name);
((IconTheme *)theme->data)->list_directories = ((IconTheme *)theme->data)->list_directories =
g_slist_sort_with_data(((IconTheme *)theme->data)->list_directories, g_slist_sort_with_data(((IconTheme *)theme->data)->list_directories,
compare_theme_directories, compare_theme_directories,
@@ -600,6 +603,8 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
(!next_larger_theme ? 1 : theme == next_larger_theme)); (!next_larger_theme ? 1 : theme == next_larger_theme));
if (!possible) if (!possible)
continue; continue;
if (debug_icons)
fprintf(stderr, "tint2: Searching directory: %s\n", ((IconThemeDir *)dir->data)->name);
const GSList *base; const GSList *base;
for (base = basenames; base; base = g_slist_next(base)) { for (base = basenames; base; base = g_slist_next(base)) {
for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) { for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) {
@@ -617,11 +622,11 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
file_name[0] = 0; file_name[0] = 0;
// filename = directory/$(themename)/subdirectory/iconname.extension // filename = directory/$(themename)/subdirectory/iconname.extension
sprintf(file_name, "%s/%s/%s/%s%s", base_name, theme_name, dir_name, icon_name, extension); sprintf(file_name, "%s/%s/%s/%s%s", base_name, theme_name, dir_name, icon_name, extension);
if (DEBUG_ICON_SEARCH) if (debug_icons)
printf("checking %s\n", file_name); fprintf(stderr, "tint2: Checking %s\n", file_name);
if (g_file_test(file_name, G_FILE_TEST_EXISTS)) { if (g_file_test(file_name, G_FILE_TEST_EXISTS)) {
if (DEBUG_ICON_SEARCH) if (debug_icons)
printf("found: %s\n", file_name); fprintf(stderr, "tint2: Found potential match: %s\n", file_name);
// Closest match // Closest match
if (directory_size_distance((IconThemeDir *)dir->data, size) < minimal_size && if (directory_size_distance((IconThemeDir *)dir->data, size) < minimal_size &&
(!best_file_theme ? 1 : theme == best_file_theme)) { (!best_file_theme ? 1 : theme == best_file_theme)) {
@@ -632,8 +637,8 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
best_file_name = strdup(file_name); best_file_name = strdup(file_name);
minimal_size = directory_size_distance((IconThemeDir *)dir->data, size); minimal_size = directory_size_distance((IconThemeDir *)dir->data, size);
best_file_theme = theme; best_file_theme = theme;
if (DEBUG_ICON_SEARCH) if (debug_icons)
printf("best_file_name = %s; minimal_size = %d\n", best_file_name, minimal_size); fprintf(stderr, "tint2: best_file_name = %s; minimal_size = %d\n", best_file_name, minimal_size);
} }
// Next larger match // Next larger match
if (((IconThemeDir *)dir->data)->size >= size && if (((IconThemeDir *)dir->data)->size >= size &&
@@ -646,8 +651,8 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
next_larger = strdup(file_name); next_larger = strdup(file_name);
next_larger_size = ((IconThemeDir *)dir->data)->size; next_larger_size = ((IconThemeDir *)dir->data)->size;
next_larger_theme = theme; next_larger_theme = theme;
if (DEBUG_ICON_SEARCH) if (debug_icons)
printf("next_larger = %s; next_larger_size = %d\n", next_larger, next_larger_size); fprintf(stderr, "tint2: next_larger = %s; next_larger_size = %d\n", next_larger, next_larger_size);
} }
} }
} }
@@ -668,6 +673,8 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
// Look in unthemed icons // Look in unthemed icons
{ {
if (debug_icons)
fprintf(stderr, "tint2: Searching unthemed icons\n");
for (const GSList *base = basenames; base; base = g_slist_next(base)) { for (const GSList *base = basenames; base; base = g_slist_next(base)) {
for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) { for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) {
char *base_name = (char *)base->data; char *base_name = (char *)base->data;
@@ -675,9 +682,11 @@ char *get_icon_path_helper(GSList *themes, const char *icon_name, int size)
file_name = calloc(strlen(base_name) + strlen(icon_name) + strlen(extension) + 100, 1); file_name = calloc(strlen(base_name) + strlen(icon_name) + strlen(extension) + 100, 1);
// filename = directory/iconname.extension // filename = directory/iconname.extension
sprintf(file_name, "%s/%s%s", base_name, icon_name, extension); sprintf(file_name, "%s/%s%s", base_name, icon_name, extension);
if (DEBUG_ICON_SEARCH) if (debug_icons)
printf("checking %s\n", file_name); fprintf(stderr, "tint2: Checking %s\n", file_name);
if (g_file_test(file_name, G_FILE_TEST_EXISTS)) { if (g_file_test(file_name, G_FILE_TEST_EXISTS)) {
if (debug_icons)
fprintf(stderr, "tint2: Found %s\n", file_name);
g_slist_free(extensions); g_slist_free(extensions);
return file_name; return file_name;
} else { } else {
@@ -715,7 +724,7 @@ char *get_icon_path_from_cache(IconThemeWrapper *wrapper, const char *icon_name,
if (!g_file_test(value, G_FILE_TEST_EXISTS)) if (!g_file_test(value, G_FILE_TEST_EXISTS))
return NULL; return NULL;
// fprintf(stderr, "Icon path found in cache: theme = %s, icon = %s, size = %d, path = %s\n", // fprintf(stderr, "tint2: Icon path found in cache: theme = %s, icon = %s, size = %d, path = %s\n",
// wrapper->icon_theme_name, icon_name, size, value); // wrapper->icon_theme_name, icon_name, size, value);
return strdup(value); return strdup(value);
@@ -742,28 +751,44 @@ void add_icon_path_to_cache(IconThemeWrapper *wrapper, const char *icon_name, in
char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size, gboolean use_fallbacks) char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size, gboolean use_fallbacks)
{ {
if (!wrapper) if (debug_icons)
fprintf(stderr,
"Searching for icon %s with size %d, fallbacks %sallowed\n",
icon_name,
size,
use_fallbacks ? "" : "not ");
if (!wrapper) {
if (debug_icons)
fprintf(stderr,
"Icon search aborted, themes not loaded\n");
return NULL; return NULL;
}
if (!icon_name || strlen(icon_name) == 0) if (!icon_name || strlen(icon_name) == 0)
goto notfound; goto notfound;
char *path = get_icon_path_from_cache(wrapper, icon_name, size); char *path = get_icon_path_from_cache(wrapper, icon_name, size);
if (path) if (path) {
if (debug_icons)
fprintf(stderr,
"Icon found in cache: %s\n", path);
return path; return path;
}
load_default_theme(wrapper); load_default_theme(wrapper);
icon_name = icon_name ? icon_name : DEFAULT_ICON; icon_name = icon_name ? icon_name : DEFAULT_ICON;
path = get_icon_path_helper(wrapper->themes, icon_name, size); path = get_icon_path_helper(wrapper->themes, icon_name, size);
if (path) { if (path) {
if (debug_icons)
fprintf(stderr, "tint2: Icon found: %s\n", path);
add_icon_path_to_cache(wrapper, icon_name, size, path); add_icon_path_to_cache(wrapper, icon_name, size, path);
return path; return path;
} }
if (!use_fallbacks) if (!use_fallbacks)
goto notfound; goto notfound;
fprintf(stderr, YELLOW "Icon not found in default theme: %s" RESET "\n", icon_name); fprintf(stderr, YELLOW "tint2: Icon not found in default theme: %s" RESET "\n", icon_name);
load_fallbacks(wrapper); load_fallbacks(wrapper);
path = get_icon_path_helper(wrapper->themes_fallback, icon_name, size); path = get_icon_path_helper(wrapper->themes_fallback, icon_name, size);
@@ -773,7 +798,7 @@ char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size,
} }
notfound: notfound:
fprintf(stderr, RED "Could not find icon '%s', using default." RESET "\n", icon_name); fprintf(stderr, RED "tint2: Could not find icon '%s', using default." RESET "\n", icon_name);
path = get_icon_path_helper(wrapper->themes, DEFAULT_ICON, size); path = get_icon_path_helper(wrapper->themes, DEFAULT_ICON, size);
if (path) if (path)
return path; return path;

View File

@@ -57,4 +57,6 @@ char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size,
// Do not free the result, it is cached. // Do not free the result, it is cached.
const GSList *get_icon_locations(); const GSList *get_icon_locations();
extern gboolean debug_icons;
#endif #endif

View File

@@ -50,7 +50,6 @@ int launcher_brightness;
char *icon_theme_name_config; char *icon_theme_name_config;
char *icon_theme_name_xsettings; char *icon_theme_name_xsettings;
int launcher_icon_theme_override; int launcher_icon_theme_override;
int startup_notifications;
Background *launcher_icon_bg; Background *launcher_icon_bg;
GList *launcher_icon_gradients; GList *launcher_icon_gradients;
@@ -65,6 +64,8 @@ void launcher_reload_hidden_icons(Launcher *launcher);
void launcher_icon_on_change_layout(void *obj); void launcher_icon_on_change_layout(void *obj);
int launcher_compute_desired_size(void *obj); int launcher_compute_desired_size(void *obj);
void relayout_launcher();
void default_launcher() void default_launcher()
{ {
launcher_enabled = 0; launcher_enabled = 0;
@@ -96,6 +97,7 @@ void init_launcher_panel(void *p)
launcher->area._draw_foreground = NULL; launcher->area._draw_foreground = NULL;
launcher->area.size_mode = LAYOUT_FIXED; launcher->area.size_mode = LAYOUT_FIXED;
launcher->area._resize = resize_launcher; launcher->area._resize = resize_launcher;
launcher->area._on_change_layout = relayout_launcher;
launcher->area._compute_desired_size = launcher_compute_desired_size; launcher->area._compute_desired_size = launcher_compute_desired_size;
launcher->area.resize_needed = 1; launcher->area.resize_needed = 1;
schedule_redraw(&launcher->area); schedule_redraw(&launcher->area);
@@ -295,7 +297,7 @@ gboolean resize_launcher(void *obj)
launcherIcon->y = posy; launcherIcon->y = posy;
launcherIcon->x = posx; launcherIcon->x = posx;
launcher_icon_on_change_layout(launcherIcon); launcher_icon_on_change_layout(launcherIcon);
// printf("launcher %d : %d,%d\n", i, posx, posy); // fprintf(stderr, "tint2: launcher %d : %d,%d\n", i, posx, posy);
if (panel_horizontal) { if (panel_horizontal) {
if (i % icons_per_column) { if (i % icons_per_column) {
posy += launcher->icon_size + launcher->area.paddingx; posy += launcher->icon_size + launcher->area.paddingx;
@@ -326,6 +328,17 @@ gboolean resize_launcher(void *obj)
return TRUE; return TRUE;
} }
void relayout_launcher(void *obj)
{
Launcher *launcher = (Launcher *)obj;
for (GSList *l = launcher->list_icons; l; l = l->next) {
LauncherIcon *launcherIcon = (LauncherIcon *)l->data;
if (!launcherIcon->area.on_screen)
continue;
launcher_icon_on_change_layout(launcherIcon);
}
}
// Here we override the default layout of the icons; normally Area layouts its children // Here we override the default layout of the icons; normally Area layouts its children
// in a stack; we need to layout them in a kind of table // in a stack; we need to layout them in a kind of table
void launcher_icon_on_change_layout(void *obj) void launcher_icon_on_change_layout(void *obj)
@@ -372,7 +385,12 @@ void draw_launcher_icon(void *obj, cairo_t *c)
void launcher_icon_dump_geometry(void *obj, int indent) void launcher_icon_dump_geometry(void *obj, int indent)
{ {
LauncherIcon *launcherIcon = (LauncherIcon *)obj; LauncherIcon *launcherIcon = (LauncherIcon *)obj;
fprintf(stderr, "%*sIcon: w = h = %d, name = %s\n", indent, "", launcherIcon->icon_size, launcherIcon->icon_name); fprintf(stderr,
"tint2: %*sIcon: w = h = %d, name = %s\n",
indent,
"",
launcherIcon->icon_size,
launcherIcon->icon_name);
} }
Imlib_Image scale_icon(Imlib_Image original, int icon_size) Imlib_Image scale_icon(Imlib_Image original, int icon_size)
@@ -416,68 +434,28 @@ void free_icon(Imlib_Image icon)
} }
} }
void launcher_action(LauncherIcon *icon, XEvent *evt) void launcher_action(LauncherIcon *icon, XEvent *evt, int x, int y)
{ {
launcher_reload_icon((Launcher *)icon->area.parent, icon); launcher_reload_icon((Launcher *)icon->area.parent, icon);
launcher_reload_hidden_icons((Launcher *)icon->area.parent); launcher_reload_hidden_icons((Launcher *)icon->area.parent);
char *cmd = calloc(strlen(icon->cmd) + 10, 1);
sprintf(cmd, "(%s&)", icon->cmd);
#if HAVE_SN
SnLauncherContext *ctx = 0;
Time time;
if (startup_notifications) {
ctx = sn_launcher_context_new(server.sn_display, server.screen);
sn_launcher_context_set_name(ctx, icon->icon_tooltip);
sn_launcher_context_set_description(ctx, "Application launched from tint2");
sn_launcher_context_set_binary_name(ctx, icon->cmd);
// Get a timestamp from the X event
if (evt->type == ButtonPress || evt->type == ButtonRelease) { if (evt->type == ButtonPress || evt->type == ButtonRelease) {
time = evt->xbutton.time; GString *cmd = g_string_new(icon->cmd);
} else { tint2_g_string_replace(cmd, "%f", "");
fprintf(stderr, "Unknown X event: %d\n", evt->type); tint2_g_string_replace(cmd, "%F", "");
free(cmd); tint_exec(cmd->str,
return; icon->cwd,
icon->icon_tooltip,
evt->xbutton.time,
&icon->area,
x,
y,
icon->start_in_terminal,
icon->startup_notification);
g_string_free(cmd, TRUE);
} }
sn_launcher_context_initiate(ctx, "tint2", icon->cmd, time);
}
#endif /* HAVE_SN */
pid_t pid;
pid = fork();
if (pid < 0) {
fprintf(stderr, "Could not fork\n");
} else if (pid == 0) {
// Child process
#if HAVE_SN
if (startup_notifications) {
sn_launcher_context_setup_child_process(ctx);
}
#endif // HAVE_SN
// Allow children to exist after parent destruction
setsid();
// Run the command
if (icon->cwd)
chdir(icon->cwd);
execl("/bin/sh", "/bin/sh", "-c", icon->cmd, NULL);
fprintf(stderr, "Failed to execlp %s\n", icon->cmd);
#if HAVE_SN
if (startup_notifications) {
sn_launcher_context_unref(ctx);
}
#endif // HAVE_SN
exit(1);
} else {
// Parent process
#if HAVE_SN
if (startup_notifications) {
g_tree_insert(server.pids, GINT_TO_POINTER(pid), ctx);
}
#endif // HAVE_SN
}
free(cmd);
} }
// Populates the list_icons list from the list_apps list // Populates the list_icons list from the list_apps list
void launcher_load_icons(Launcher *launcher) void launcher_load_icons(Launcher *launcher)
{ {
@@ -530,6 +508,8 @@ void launcher_reload_icon(Launcher *launcher, LauncherIcon *launcherIcon)
launcherIcon->cwd = strdup(entry.cwd); launcherIcon->cwd = strdup(entry.cwd);
else else
launcherIcon->cwd = NULL; launcherIcon->cwd = NULL;
launcherIcon->start_in_terminal = entry.start_in_terminal;
launcherIcon->startup_notification = entry.startup_notification;
if (launcherIcon->icon_name) if (launcherIcon->icon_name)
free(launcherIcon->icon_name); free(launcherIcon->icon_name);
launcherIcon->icon_name = entry.icon ? strdup(entry.icon) : strdup(DEFAULT_ICON); launcherIcon->icon_name = entry.icon ? strdup(entry.icon) : strdup(DEFAULT_ICON);
@@ -585,7 +565,7 @@ void launcher_reload_icon_image(Launcher *launcher, LauncherIcon *launcherIcon)
free_icon(original); free_icon(original);
free(launcherIcon->icon_path); free(launcherIcon->icon_path);
launcherIcon->icon_path = new_icon_path; launcherIcon->icon_path = new_icon_path;
// fprintf(stderr, "launcher.c %d: Using icon %s\n", __LINE__, launcherIcon->icon_path); // fprintf(stderr, "tint2: launcher.c %d: Using icon %s\n", __LINE__, launcherIcon->icon_path);
if (panel_config.mouse_effects) { if (panel_config.mouse_effects) {
launcherIcon->image_hover = adjust_icon(launcherIcon->image, launcherIcon->image_hover = adjust_icon(launcherIcon->image,

View File

@@ -33,6 +33,8 @@ typedef struct LauncherIcon {
Imlib_Image image_pressed; Imlib_Image image_pressed;
char *cmd; char *cmd;
char *cwd; char *cwd;
gboolean start_in_terminal;
gboolean startup_notification;
char *icon_name; char *icon_name;
char *icon_path; char *icon_path;
char *icon_tooltip; char *icon_tooltip;
@@ -49,7 +51,6 @@ extern int launcher_brightness;
extern char *icon_theme_name_xsettings; // theme name extern char *icon_theme_name_xsettings; // theme name
extern char *icon_theme_name_config; extern char *icon_theme_name_config;
extern int launcher_icon_theme_override; extern int launcher_icon_theme_override;
extern int startup_notifications;
extern Background *launcher_icon_bg; extern Background *launcher_icon_bg;
extern GList *launcher_icon_gradients; extern GList *launcher_icon_gradients;
@@ -68,7 +69,7 @@ void launcher_default_icon_theme_changed();
// Populates the list_icons list // Populates the list_icons list
void launcher_load_icons(Launcher *launcher); void launcher_load_icons(Launcher *launcher);
void launcher_action(LauncherIcon *icon, XEvent *e); void launcher_action(LauncherIcon *icon, XEvent *e, int x, int y);
void test_launcher_read_desktop_file(); void test_launcher_read_desktop_file();
void test_launcher_read_theme_file(); void test_launcher_read_theme_file();

View File

@@ -48,7 +48,7 @@ void xsettings_notify_cb(const char *name, XSettingsAction action, XSettingsSett
{ {
if ((action == XSETTINGS_ACTION_NEW || action == XSETTINGS_ACTION_CHANGED) && name != NULL && setting != NULL) { if ((action == XSETTINGS_ACTION_NEW || action == XSETTINGS_ACTION_CHANGED) && name != NULL && setting != NULL) {
if (strcmp(name, "Net/IconThemeName") == 0 && setting->type == XSETTINGS_TYPE_STRING) { if (strcmp(name, "Net/IconThemeName") == 0 && setting->type == XSETTINGS_TYPE_STRING) {
fprintf(stderr, "xsettings: %s = %s\n", name, setting->data.v_string); fprintf(stderr, "tint2: xsettings: %s = %s\n", name, setting->data.v_string);
if (icon_theme_name_xsettings) { if (icon_theme_name_xsettings) {
if (strcmp(icon_theme_name_xsettings, setting->data.v_string) == 0) if (strcmp(icon_theme_name_xsettings, setting->data.v_string) == 0)
return; return;
@@ -57,7 +57,7 @@ void xsettings_notify_cb(const char *name, XSettingsAction action, XSettingsSett
icon_theme_name_xsettings = strdup(setting->data.v_string); icon_theme_name_xsettings = strdup(setting->data.v_string);
default_icon_theme_changed(); default_icon_theme_changed();
} else if (strcmp(name, "Gtk/FontName") == 0 && setting->type == XSETTINGS_TYPE_STRING) { } else if (strcmp(name, "Gtk/FontName") == 0 && setting->type == XSETTINGS_TYPE_STRING) {
fprintf(stderr, "xsettings: %s = %s\n", name, setting->data.v_string); fprintf(stderr, "tint2: xsettings: %s = %s\n", name, setting->data.v_string);
if (default_font) { if (default_font) {
if (strcmp(default_font, setting->data.v_string) == 0) if (strcmp(default_font, setting->data.v_string) == 0)
return; return;
@@ -191,7 +191,7 @@ static XSettingsList *parse_settings(unsigned char *data, size_t len)
result = fetch_card8(&buffer, (CARD8 *)&buffer.byte_order); result = fetch_card8(&buffer, (CARD8 *)&buffer.byte_order);
if (buffer.byte_order != MSBFirst && buffer.byte_order != LSBFirst) { if (buffer.byte_order != MSBFirst && buffer.byte_order != LSBFirst) {
fprintf(stderr, "Invalid byte order %x in XSETTINGS property\n", buffer.byte_order); fprintf(stderr, "tint2: Invalid byte order %x in XSETTINGS property\n", buffer.byte_order);
result = XSETTINGS_FAILED; result = XSETTINGS_FAILED;
goto out; goto out;
} }
@@ -312,13 +312,13 @@ out:
if (result != XSETTINGS_SUCCESS) { if (result != XSETTINGS_SUCCESS) {
switch (result) { switch (result) {
case XSETTINGS_NO_MEM: case XSETTINGS_NO_MEM:
fprintf(stderr, "Out of memory reading XSETTINGS property\n"); fprintf(stderr, "tint2: Out of memory reading XSETTINGS property\n");
break; break;
case XSETTINGS_ACCESS: case XSETTINGS_ACCESS:
fprintf(stderr, "Invalid XSETTINGS property (read off end)\n"); fprintf(stderr, "tint2: Invalid XSETTINGS property (read off end)\n");
break; break;
case XSETTINGS_DUPLICATE_ENTRY: case XSETTINGS_DUPLICATE_ENTRY:
fprintf(stderr, "Duplicate XSETTINGS entry for '%s'\n", setting->name); fprintf(stderr, "tint2: Duplicate XSETTINGS entry for '%s'\n", setting->name);
case XSETTINGS_FAILED: case XSETTINGS_FAILED:
case XSETTINGS_SUCCESS: case XSETTINGS_SUCCESS:
case XSETTINGS_NO_ENTRY: case XSETTINGS_NO_ENTRY:
@@ -365,7 +365,7 @@ static void read_settings(XSettingsClient *client)
if (result == Success && type == server.atom._XSETTINGS_SETTINGS) { if (result == Success && type == server.atom._XSETTINGS_SETTINGS) {
if (format != 8) { if (format != 8) {
fprintf(stderr, "Invalid format for XSETTINGS property %d", format); fprintf(stderr, "tint2: Invalid format for XSETTINGS property %d", format);
} else } else
client->settings = parse_settings(data, n_items); client->settings = parse_settings(data, n_items);
XFree(data); XFree(data);
@@ -420,7 +420,7 @@ XSettingsClient *xsettings_client_new(Display *display,
check_manager_window(client); check_manager_window(client);
if (client->manager_window == None) { if (client->manager_window == None) {
printf("No XSETTINGS manager, tint2 uses config option 'launcher_icon_theme'.\n"); fprintf(stderr, "tint2: No XSETTINGS manager, tint2 uses config option 'launcher_icon_theme'.\n");
free(client); free(client);
return NULL; return NULL;
} else { } else {

806
src/main.c Normal file
View File

@@ -0,0 +1,806 @@
/**************************************************************************
*
* Tint2 panel
*
* Copyright (C) 2007 Pål Staurland (staura@gmail.com)
* Modified (C) 2008 thierry lorthiois (lorthiois@bbsoft.fr) from Omega distribution
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**************************************************************************/
#include <errno.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include <X11/Xutil.h>
#include <X11/Xatom.h>
#include <X11/Xlocale.h>
#include <X11/extensions/Xdamage.h>
#include <Imlib2.h>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>
#include <pwd.h>
#include <time.h>
#include <sys/time.h>
#include <sys/wait.h>
#ifdef HAVE_SN
#include <libsn/sn.h>
#endif
#include "config.h"
#include "drag_and_drop.h"
#include "fps_distribution.h"
#include "init.h"
#include "launcher.h"
#include "mouse_actions.h"
#include "panel.h"
#include "server.h"
#include "signals.h"
#include "systraybar.h"
#include "task.h"
#include "taskbar.h"
#include "tooltip.h"
#include "timer.h"
#include "tracing.h"
#include "uevent.h"
#include "version.h"
#include "window.h"
#include "xsettings-client.h"
// Global process state variables
XSettingsClient *xsettings_client = NULL;
gboolean debug_fps = FALSE;
gboolean debug_frames = FALSE;
static int frame = 0;
double tracing_fps_threshold = 60;
static double ts_event_read;
static double ts_event_processed;
static double ts_render_finished;
static double ts_flush_finished;
static gboolean first_render;
void handle_event_property_notify(XEvent *e)
{
gboolean debug = FALSE;
Window win = e->xproperty.window;
Atom at = e->xproperty.atom;
if (xsettings_client)
xsettings_client_process_event(xsettings_client, e);
for (int i = 0; i < num_panels; i++) {
Panel *p = &panels[i];
if (win == p->main_win) {
if (at == server.atom._NET_WM_DESKTOP && get_window_desktop(p->main_win) != ALL_DESKTOPS)
replace_panel_all_desktops(p);
return;
}
}
if (win == server.root_win) {
if (!server.got_root_win) {
XSelectInput(server.display, server.root_win, PropertyChangeMask | StructureNotifyMask);
server.got_root_win = TRUE;
}
// Change name of desktops
else if (at == server.atom._NET_DESKTOP_NAMES) {
if (debug)
fprintf(stderr, "tint2: %s %d: win = root, atom = _NET_DESKTOP_NAMES\n", __func__, __LINE__);
update_desktop_names();
}
// Change desktops
else if (at == server.atom._NET_NUMBER_OF_DESKTOPS || at == server.atom._NET_DESKTOP_GEOMETRY ||
at == server.atom._NET_DESKTOP_VIEWPORT || at == server.atom._NET_WORKAREA ||
at == server.atom._NET_CURRENT_DESKTOP) {
if (debug)
fprintf(stderr, "tint2: %s %d: win = root, atom = ?? desktops changed\n", __func__, __LINE__);
if (!taskbar_enabled)
return;
int old_num_desktops = server.num_desktops;
int old_desktop = server.desktop;
server_get_number_of_desktops();
server.desktop = get_current_desktop();
if (old_num_desktops != server.num_desktops) { // If number of desktop changed
if (server.num_desktops <= server.desktop) {
server.desktop = server.num_desktops - 1;
}
cleanup_taskbar();
init_taskbar();
for (int i = 0; i < num_panels; i++) {
init_taskbar_panel(&panels[i]);
set_panel_items_order(&panels[i]);
panels[i].area.resize_needed = 1;
}
taskbar_refresh_tasklist();
reset_active_task();
update_all_taskbars_visibility();
if (old_desktop != server.desktop)
tooltip_trigger_hide();
schedule_panel_redraw();
} else if (old_desktop != server.desktop) {
tooltip_trigger_hide();
for (int i = 0; i < num_panels; i++) {
Panel *panel = &panels[i];
set_taskbar_state(&panel->taskbar[old_desktop], TASKBAR_NORMAL);
set_taskbar_state(&panel->taskbar[server.desktop], TASKBAR_ACTIVE);
// check ALL_DESKTOPS task => resize taskbar
Taskbar *taskbar;
if (server.num_desktops > old_desktop) {
taskbar = &panel->taskbar[old_desktop];
GList *l = taskbar->area.children;
if (taskbarname_enabled)
l = l->next;
for (; l; l = l->next) {
Task *task = l->data;
if (task->desktop == ALL_DESKTOPS) {
task->area.on_screen = always_show_all_desktop_tasks;
taskbar->area.resize_needed = 1;
schedule_panel_redraw();
if (taskbar_mode == MULTI_DESKTOP)
panel->area.resize_needed = 1;
}
}
}
taskbar = &panel->taskbar[server.desktop];
GList *l = taskbar->area.children;
if (taskbarname_enabled)
l = l->next;
for (; l; l = l->next) {
Task *task = l->data;
if (task->desktop == ALL_DESKTOPS) {
task->area.on_screen = TRUE;
taskbar->area.resize_needed = 1;
if (taskbar_mode == MULTI_DESKTOP)
panel->area.resize_needed = 1;
}
}
if (server.viewports) {
GList *need_update = NULL;
GHashTableIter iter;
gpointer key, value;
g_hash_table_iter_init(&iter, win_to_task);
while (g_hash_table_iter_next(&iter, &key, &value)) {
Window task_win = *(Window *)key;
Task *task = get_task(task_win);
if (task) {
int desktop = get_window_desktop(task_win);
if (desktop != task->desktop) {
need_update = g_list_append(need_update, task);
}
}
}
for (l = need_update; l; l = l->next) {
Task *task = l->data;
task_update_desktop(task);
}
}
}
}
}
// Window list
else if (at == server.atom._NET_CLIENT_LIST) {
if (debug)
fprintf(stderr, "tint2: %s %d: win = root, atom = _NET_CLIENT_LIST\n", __func__, __LINE__);
taskbar_refresh_tasklist();
update_all_taskbars_visibility();
schedule_panel_redraw();
}
// Change active
else if (at == server.atom._NET_ACTIVE_WINDOW) {
if (debug)
fprintf(stderr, "tint2: %s %d: win = root, atom = _NET_ACTIVE_WINDOW\n", __func__, __LINE__);
reset_active_task();
schedule_panel_redraw();
} else if (at == server.atom._XROOTPMAP_ID || at == server.atom._XROOTMAP_ID) {
if (debug)
fprintf(stderr, "tint2: %s %d: win = root, atom = _XROOTPMAP_ID\n", __func__, __LINE__);
// change Wallpaper
for (int i = 0; i < num_panels; i++) {
set_panel_background(&panels[i]);
}
schedule_panel_redraw();
}
} else {
TrayWindow *traywin = systray_find_icon(win);
if (traywin) {
systray_property_notify(traywin, e);
return;
}
Task *task = get_task(win);
if (debug) {
char *atom_name = XGetAtomName(server.display, at);
fprintf(stderr,
"%s %d: win = %ld, task = %s, atom = %s\n",
__func__,
__LINE__,
win,
task ? (task->title ? task->title : "??") : "null",
atom_name);
XFree(atom_name);
}
if (!task) {
if (debug)
fprintf(stderr, "tint2: %s %d\n", __func__, __LINE__);
if (at == server.atom._NET_WM_STATE) {
// xfce4 sends _NET_WM_STATE after minimized to tray, so we need to check if window is mapped
// if it is mapped and not set as skip_taskbar, we must add it to our task list
XWindowAttributes wa;
XGetWindowAttributes(server.display, win, &wa);
if (wa.map_state == IsViewable && !window_is_skip_taskbar(win)) {
if ((task = add_task(win)))
schedule_panel_redraw();
}
}
return;
}
// fprintf(stderr, "tint2: atom root_win = %s, %s\n", XGetAtomName(server.display, at), task->title);
// Window title changed
if (at == server.atom._NET_WM_VISIBLE_NAME || at == server.atom._NET_WM_NAME || at == server.atom.WM_NAME) {
if (task_update_title(task)) {
if (g_tooltip.mapped && (g_tooltip.area == (Area *)task)) {
tooltip_update_contents_for((Area *)task);
tooltip_update();
}
if (taskbar_sort_method == TASKBAR_SORT_TITLE)
sort_taskbar_for_win(win);
schedule_panel_redraw();
}
}
// Demand attention
else if (at == server.atom._NET_WM_STATE) {
if (debug) {
int count;
Atom *atom_state = server_get_property(win, server.atom._NET_WM_STATE, XA_ATOM, &count);
for (int j = 0; j < count; j++) {
char *atom_state_name = XGetAtomName(server.display, atom_state[j]);
fprintf(stderr, "tint2: %s %d: _NET_WM_STATE = %s\n", __func__, __LINE__, atom_state_name);
XFree(atom_state_name);
}
XFree(atom_state);
}
if (window_is_urgent(win)) {
add_urgent(task);
}
if (window_is_skip_taskbar(win)) {
remove_task(task);
schedule_panel_redraw();
}
} else if (at == server.atom.WM_STATE) {
// Iconic state
TaskState state = (active_task && task->win == active_task->win ? TASK_ACTIVE : TASK_NORMAL);
if (window_is_iconified(win))
state = TASK_ICONIFIED;
set_task_state(task, state);
schedule_panel_redraw();
}
// Window icon changed
else if (at == server.atom._NET_WM_ICON) {
task_update_icon(task);
schedule_panel_redraw();
}
// Window desktop changed
else if (at == server.atom._NET_WM_DESKTOP) {
int desktop = get_window_desktop(win);
// fprintf(stderr, "tint2: Window desktop changed %d, %d\n", task->desktop, desktop);
// bug in windowmaker : send unecessary 'desktop changed' when focus changed
if (desktop != task->desktop) {
task_update_desktop(task);
}
} else if (at == server.atom.WM_HINTS) {
XWMHints *wmhints = XGetWMHints(server.display, win);
if (wmhints && wmhints->flags & XUrgencyHint) {
add_urgent(task);
}
XFree(wmhints);
task_update_icon(task);
schedule_panel_redraw();
}
if (!server.got_root_win)
server.root_win = RootWindow(server.display, server.screen);
}
}
void handle_event_expose(XEvent *e)
{
Panel *panel;
panel = get_panel(e->xany.window);
if (!panel)
return;
// TODO : one panel_refresh per panel ?
schedule_panel_redraw();
}
void handle_event_configure_notify(XEvent *e)
{
Window win = e->xconfigure.window;
// change in root window (xrandr)
if (win == server.root_win) {
emit_self_restart("configuration change in the root window");
return;
}
TrayWindow *traywin = systray_find_icon(win);
if (traywin) {
systray_reconfigure_event(traywin, e);
return;
}
// 'win' move in another monitor
if (num_panels > 1 || hide_task_diff_monitor) {
Task *task = get_task(win);
if (task) {
Panel *p = task->area.panel;
int monitor = get_window_monitor(win);
if ((hide_task_diff_monitor && p->monitor != monitor && task->area.on_screen) ||
(hide_task_diff_monitor && p->monitor == monitor && !task->area.on_screen) ||
(p->monitor != monitor && num_panels > 1)) {
remove_task(task);
task = add_task(win);
if (win == get_active_window()) {
set_task_state(task, TASK_ACTIVE);
active_task = task;
}
schedule_panel_redraw();
}
}
}
if (server.viewports) {
Task *task = get_task(win);
if (task) {
int desktop = get_window_desktop(win);
if (task->desktop != desktop) {
task_update_desktop(task);
}
}
}
sort_taskbar_for_win(win);
}
gboolean handle_x_event_autohide(XEvent *e)
{
Panel *panel = get_panel(e->xany.window);
if (panel && panel_autohide) {
if (e->type == EnterNotify)
autohide_trigger_show(panel);
else if (e->type == LeaveNotify)
autohide_trigger_hide(panel);
if (panel->is_hidden) {
if (e->type == ClientMessage && e->xclient.message_type == server.atom.XdndPosition) {
hidden_panel_shown_for_dnd = TRUE;
autohide_show(panel);
} else {
// discard further processing of this event because the panel is not visible yet
return TRUE;
}
} else if (hidden_panel_shown_for_dnd && e->type == ClientMessage &&
e->xclient.message_type == server.atom.XdndLeave) {
hidden_panel_shown_for_dnd = FALSE;
autohide_hide(panel);
}
}
return FALSE;
}
void handle_x_event(XEvent *e)
{
#if HAVE_SN
if (startup_notifications)
sn_display_process_event(server.sn_display, e);
#endif // HAVE_SN
if (handle_x_event_autohide(e))
return;
Panel *panel = get_panel(e->xany.window);
switch (e->type) {
case ButtonPress: {
tooltip_hide(0);
handle_mouse_press_event(e);
Area *area = find_area_under_mouse(panel, e->xbutton.x, e->xbutton.y);
if (panel_config.mouse_effects)
mouse_over(area, TRUE);
break;
}
case ButtonRelease: {
handle_mouse_release_event(e);
Area *area = find_area_under_mouse(panel, e->xbutton.x, e->xbutton.y);
if (panel_config.mouse_effects)
mouse_over(area, FALSE);
break;
}
case MotionNotify: {
unsigned int button_mask = Button1Mask | Button2Mask | Button3Mask | Button4Mask | Button5Mask;
if (e->xmotion.state & button_mask)
handle_mouse_move_event(e);
Area *area = find_area_under_mouse(panel, e->xmotion.x, e->xmotion.y);
if (area->_get_tooltip_text)
tooltip_trigger_show(area, panel, e);
else
tooltip_trigger_hide();
if (panel_config.mouse_effects)
mouse_over(area, e->xmotion.state & button_mask);
break;
}
case LeaveNotify: {
tooltip_trigger_hide();
if (panel_config.mouse_effects)
mouse_out();
break;
}
case Expose:
handle_event_expose(e);
break;
case PropertyNotify:
handle_event_property_notify(e);
break;
case ConfigureNotify:
handle_event_configure_notify(e);
break;
case ConfigureRequest: {
TrayWindow *traywin = systray_find_icon(e->xany.window);
if (traywin)
systray_reconfigure_event(traywin, e);
break;
}
case ResizeRequest: {
TrayWindow *traywin = systray_find_icon(e->xany.window);
if (traywin)
systray_resize_request_event(traywin, e);
break;
}
case ReparentNotify: {
if (!systray_enabled)
break;
Panel *systray_panel = (Panel *)systray.area.panel;
if (e->xany.window == systray_panel->main_win) // don't care
break;
TrayWindow *traywin = systray_find_icon(e->xreparent.window);
if (traywin) {
if (traywin->win == e->xreparent.window) {
if (traywin->parent == e->xreparent.parent) {
embed_icon(traywin);
} else {
remove_icon(traywin);
}
break;
}
}
break;
}
case UnmapNotify:
break;
case DestroyNotify:
if (e->xany.window == server.composite_manager) {
// Stop real_transparency
emit_self_restart("compositor shutdown");
break;
}
if (e->xany.window == g_tooltip.window || !systray_enabled)
break;
for (GSList *it = systray.list_icons; it; it = g_slist_next(it)) {
if (((TrayWindow *)it->data)->win == e->xany.window) {
systray_destroy_event((TrayWindow *)it->data);
break;
}
}
break;
case ClientMessage: {
XClientMessageEvent *ev = &e->xclient;
if (ev->data.l[1] == server.atom._NET_WM_CM_S0) {
if (ev->data.l[2] == None) {
// Stop real_transparency
emit_self_restart("compositor changed");
} else {
// Start real_transparency
emit_self_restart("compositor changed");
}
}
if (systray_enabled && e->xclient.message_type == server.atom._NET_SYSTEM_TRAY_OPCODE &&
e->xclient.format == 32 && e->xclient.window == net_sel_win) {
handle_systray_event(&e->xclient);
} else if (e->xclient.message_type == server.atom.XdndEnter) {
handle_dnd_enter(&e->xclient);
} else if (e->xclient.message_type == server.atom.XdndPosition) {
handle_dnd_position(&e->xclient);
} else if (e->xclient.message_type == server.atom.XdndDrop) {
handle_dnd_drop(&e->xclient);
}
break;
}
case SelectionNotify: {
handle_dnd_selection_notify(&e->xselection);
break;
}
default:
if (e->type == server.xdamage_event_type) {
XDamageNotifyEvent *de = (XDamageNotifyEvent *)e;
TrayWindow *traywin = systray_find_icon(de->drawable);
if (traywin)
systray_render_icon(traywin);
}
}
}
void handle_x_events()
{
if (XPending(server.display) > 0) {
XEvent e;
XNextEvent(server.display, &e);
if (debug_fps)
ts_event_read = get_time();
handle_x_event(&e);
}
}
void prepare_fd_set(fd_set *set, int *max_fd)
{
FD_ZERO(set);
FD_SET(server.x11_fd, set);
*max_fd = server.x11_fd;
if (sigchild_pipe_valid) {
FD_SET(sigchild_pipe[0], set);
*max_fd = MAX(*max_fd, sigchild_pipe[0]);
}
for (GList *l = panel_config.execp_list; l; l = l->next) {
Execp *execp = (Execp *)l->data;
int fd = execp->backend->child_pipe_stdout;
if (fd > 0) {
FD_SET(fd, set);
*max_fd = MAX(*max_fd, fd);
}
fd = execp->backend->child_pipe_stderr;
if (fd > 0) {
FD_SET(fd, set);
*max_fd = MAX(*max_fd, fd);
}
}
if (uevent_fd > 0) {
FD_SET(uevent_fd, set);
*max_fd = MAX(*max_fd, uevent_fd);
}
}
void handle_panel_refresh()
{
if (debug_fps)
ts_event_processed = get_time();
panel_refresh = FALSE;
for (int i = 0; i < num_panels; i++) {
Panel *panel = &panels[i];
if (!first_render)
shrink_panel(panel);
if (!panel->is_hidden || panel->area.resize_needed) {
if (panel->temp_pmap)
XFreePixmap(server.display, panel->temp_pmap);
panel->temp_pmap = XCreatePixmap(server.display,
server.root_win,
panel->area.width,
panel->area.height,
server.depth);
render_panel(panel);
}
if (panel->is_hidden) {
if (!panel->hidden_pixmap) {
panel->hidden_pixmap = XCreatePixmap(server.display,
server.root_win,
panel->hidden_width,
panel->hidden_height,
server.depth);
int xoff = 0, yoff = 0;
if (panel_horizontal && panel_position & BOTTOM)
yoff = panel->area.height - panel->hidden_height;
else if (!panel_horizontal && panel_position & RIGHT)
xoff = panel->area.width - panel->hidden_width;
XCopyArea(server.display,
panel->area.pix,
panel->hidden_pixmap,
server.gc,
xoff,
yoff,
panel->hidden_width,
panel->hidden_height,
0,
0);
}
XCopyArea(server.display,
panel->hidden_pixmap,
panel->main_win,
server.gc,
0,
0,
panel->hidden_width,
panel->hidden_height,
0,
0);
XSetWindowBackgroundPixmap(server.display, panel->main_win, panel->hidden_pixmap);
} else {
XCopyArea(server.display,
panel->temp_pmap,
panel->main_win,
server.gc,
0,
0,
panel->area.width,
panel->area.height,
0,
0);
if (panel == (Panel *)systray.area.panel) {
if (refresh_systray && panel && !panel->is_hidden) {
refresh_systray = FALSE;
XSetWindowBackgroundPixmap(server.display, panel->main_win, panel->temp_pmap);
refresh_systray_icons();
}
}
}
}
if (first_render) {
first_render = FALSE;
if (panel_shrink)
schedule_panel_redraw();
}
if (debug_fps)
ts_render_finished = get_time();
XFlush(server.display);
if (debug_fps && ts_event_read > 0) {
ts_flush_finished = get_time();
double period = ts_flush_finished - ts_event_read;
double fps = 1.0 / period;
sample_fps(fps);
double proc_ratio = (ts_event_processed - ts_event_read) / period;
double render_ratio = (ts_render_finished - ts_event_processed) / period;
double flush_ratio = (ts_flush_finished - ts_render_finished) / period;
double fps_low, fps_median, fps_high, fps_samples;
fps_compute_stats(&fps_low, &fps_median, &fps_high, &fps_samples);
fprintf(stderr,
BLUE "frame %d: fps = %.0f (low %.0f, med %.0f, high %.0f, samples %.0f) : processing %.0f%%, "
"rendering %.0f%%, "
"flushing %.0f%%" RESET "\n",
frame,
fps,
fps_low,
fps_median,
fps_high,
fps_samples,
proc_ratio * 100,
render_ratio * 100,
flush_ratio * 100);
#ifdef HAVE_TRACING
stop_tracing();
if (fps <= tracing_fps_threshold) {
print_tracing_events();
}
#endif
}
if (debug_frames) {
for (int i = 0; i < num_panels; i++) {
char path[256];
sprintf(path, "tint2-%d-panel-%d-frame-%d.png", getpid(), i, frame);
save_panel_screenshot(&panels[i], path);
}
}
frame++;
}
void run_tint2_event_loop()
{
ts_event_read = 0;
ts_event_processed = 0;
ts_render_finished = 0;
ts_flush_finished = 0;
first_render = TRUE;
while (!get_signal_pending()) {
if (panel_refresh)
handle_panel_refresh();
fd_set fds;
int max_fd;
prepare_fd_set(&fds, &max_fd);
// Wait for an event and handle it
ts_event_read = 0;
if (XPending(server.display) > 0 || select(max_fd + 1, &fds, 0, 0, get_next_timeout()) >= 0) {
#ifdef HAVE_TRACING
start_tracing((void*)run_tint2_event_loop);
#endif
uevent_handler();
handle_sigchld_events();
handle_execp_events();
handle_x_events();
}
handle_expired_timers();
}
}
void tint2(int argc, char **argv, gboolean *restart)
{
init(argc, argv);
if (snapshot_path) {
save_screenshot(snapshot_path);
cleanup();
return;
}
dnd_init();
uevent_init();
run_tint2_event_loop();
if (get_signal_pending()) {
cleanup();
if (get_signal_pending() == SIGUSR1) {
fprintf(stderr, YELLOW "tint2: %s %d: restarting tint2..." RESET "\n", __FILE__, __LINE__);
*restart = TRUE;
return;
} else if (get_signal_pending() == SIGUSR2) {
fprintf(stderr, YELLOW "tint2: %s %d: reexecuting tint2..." RESET "\n", __FILE__, __LINE__);
if (execvp(argv[0], argv) == -1) {
fprintf(stderr, RED "tint2: %s %d: failed!" RESET "\n", __FILE__, __LINE__);
return;
}
} else {
// SIGINT, SIGTERM, SIGHUP etc.
return;
}
}
}
int main(int argc, char **argv)
{
gboolean restart;
do {
restart = FALSE;
tint2(argc, argv, &restart);
} while(restart);
return 0;
}

287
src/mouse_actions.c Normal file
View File

@@ -0,0 +1,287 @@
/**************************************************************************
* Copyright (C) 2017 tint2 authors
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "drag_and_drop.h"
#include "panel.h"
#include "server.h"
#include "task.h"
#include "window.h"
gboolean tint2_handles_click(Panel *panel, XButtonEvent *e)
{
Task *task = click_task(panel, e->x, e->y);
if (task) {
if ((e->button == 1 && mouse_left != 0) || (e->button == 2 && mouse_middle != 0) ||
(e->button == 3 && mouse_right != 0) || (e->button == 4 && mouse_scroll_up != 0) ||
(e->button == 5 && mouse_scroll_down != 0)) {
return TRUE;
} else
return FALSE;
}
LauncherIcon *icon = click_launcher_icon(panel, e->x, e->y);
if (icon) {
if (e->button == 1) {
return TRUE;
} else {
return FALSE;
}
}
// no launcher/task clicked --> check if taskbar clicked
Taskbar *taskbar = click_taskbar(panel, e->x, e->y);
if (taskbar && e->button == 1 && taskbar_mode == MULTI_DESKTOP)
return TRUE;
if (click_clock(panel, e->x, e->y)) {
if ((e->button == 1 && clock_lclick_command) || (e->button == 2 && clock_mclick_command) ||
(e->button == 3 && clock_rclick_command) || (e->button == 4 && clock_uwheel_command) ||
(e->button == 5 && clock_dwheel_command))
return TRUE;
else
return FALSE;
}
#ifdef ENABLE_BATTERY
if (click_battery(panel, e->x, e->y)) {
if ((e->button == 1 && battery_lclick_command) || (e->button == 2 && battery_mclick_command) ||
(e->button == 3 && battery_rclick_command) || (e->button == 4 && battery_uwheel_command) ||
(e->button == 5 && battery_dwheel_command))
return TRUE;
else
return FALSE;
}
#endif
if (click_execp(panel, e->x, e->y))
return TRUE;
if (click_button(panel, e->x, e->y))
return TRUE;
return FALSE;
}
void handle_mouse_press_event(XEvent *e)
{
Panel *panel = get_panel(e->xany.window);
if (!panel)
return;
if (wm_menu && !tint2_handles_click(panel, &e->xbutton)) {
forward_click(e);
return;
}
task_drag = click_task(panel, e->xbutton.x, e->xbutton.y);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
}
void handle_mouse_move_event(XEvent *e)
{
Panel *panel = get_panel(e->xany.window);
if (!panel || !task_drag)
return;
// Find the taskbar on the event's location
Taskbar *event_taskbar = click_taskbar(panel, e->xbutton.x, e->xbutton.y);
if (event_taskbar == NULL)
return;
// Find the task on the event's location
Task *event_task = click_task(panel, e->xbutton.x, e->xbutton.y);
// If the event takes place on the same taskbar as the task being dragged
if (&event_taskbar->area == task_drag->area.parent) {
if (taskbar_sort_method != TASKBAR_NOSORT) {
sort_tasks(event_taskbar);
} else {
// Swap the task_drag with the task on the event's location (if they differ)
if (event_task && event_task != task_drag) {
GList *drag_iter = g_list_find(event_taskbar->area.children, task_drag);
GList *task_iter = g_list_find(event_taskbar->area.children, event_task);
if (drag_iter && task_iter) {
gpointer temp = task_iter->data;
task_iter->data = drag_iter->data;
drag_iter->data = temp;
event_taskbar->area.resize_needed = 1;
schedule_panel_redraw();
task_dragged = 1;
}
}
}
} else { // The event is on another taskbar than the task being dragged
if (task_drag->desktop == ALL_DESKTOPS || taskbar_mode != MULTI_DESKTOP)
return;
Taskbar *drag_taskbar = (Taskbar *)task_drag->area.parent;
remove_area((Area *)task_drag);
if (event_taskbar->area.posx > drag_taskbar->area.posx || event_taskbar->area.posy > drag_taskbar->area.posy) {
int i = (taskbarname_enabled) ? 1 : 0;
event_taskbar->area.children = g_list_insert(event_taskbar->area.children, task_drag, i);
} else
event_taskbar->area.children = g_list_append(event_taskbar->area.children, task_drag);
// Move task to other desktop (but avoid the 'Window desktop changed' code in 'event_property_notify')
task_drag->area.parent = &event_taskbar->area;
task_drag->desktop = event_taskbar->desktop;
change_window_desktop(task_drag->win, event_taskbar->desktop);
if (hide_task_diff_desktop)
change_desktop(event_taskbar->desktop);
if (taskbar_sort_method != TASKBAR_NOSORT) {
sort_tasks(event_taskbar);
}
event_taskbar->area.resize_needed = 1;
drag_taskbar->area.resize_needed = 1;
task_dragged = 1;
schedule_panel_redraw();
panel->area.resize_needed = 1;
}
}
void handle_mouse_release_event(XEvent *e)
{
Panel *panel = get_panel(e->xany.window);
if (!panel)
return;
if (wm_menu && !tint2_handles_click(panel, &e->xbutton)) {
forward_click(e);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
MouseAction action = TOGGLE_ICONIFY;
switch (e->xbutton.button) {
case 1:
action = mouse_left;
break;
case 2:
action = mouse_middle;
break;
case 3:
action = mouse_right;
break;
case 4:
action = mouse_scroll_up;
break;
case 5:
action = mouse_scroll_down;
break;
case 6:
action = mouse_tilt_left;
break;
case 7:
action = mouse_tilt_right;
break;
}
Clock *clock = click_clock(panel, e->xbutton.x, e->xbutton.y);
if (clock) {
clock_action(clock, e->xbutton.button, e->xbutton.x - clock->area.posx, e->xbutton.y - clock->area.posy, e->xbutton.time);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
#ifdef ENABLE_BATTERY
Battery *battery = click_battery(panel, e->xbutton.x, e->xbutton.y);
if (battery) {
battery_action(battery, e->xbutton.button, e->xbutton.x - battery->area.posx, e->xbutton.y - battery->area.posy, e->xbutton.time);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
#endif
Execp *execp = click_execp(panel, e->xbutton.x, e->xbutton.y);
if (execp) {
execp_action(execp, e->xbutton.button, e->xbutton.x - execp->area.posx, e->xbutton.y - execp->area.posy, e->xbutton.time);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
Button *button = click_button(panel, e->xbutton.x, e->xbutton.y);
if (button) {
button_action(button, e->xbutton.button, e->xbutton.x - button->area.posx, e->xbutton.y - button->area.posy, e->xbutton.time);
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
if (e->xbutton.button == 1 && click_launcher(panel, e->xbutton.x, e->xbutton.y)) {
LauncherIcon *icon = click_launcher_icon(panel, e->xbutton.x, e->xbutton.y);
if (icon) {
launcher_action(icon, e, e->xbutton.x - icon->area.posx, e->xbutton.y - icon->area.posy);
}
task_drag = 0;
return;
}
Taskbar *taskbar;
if (!(taskbar = click_taskbar(panel, e->xbutton.x, e->xbutton.y))) {
// TODO: check better solution to keep window below
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
task_drag = 0;
return;
}
// drag and drop task
if (task_dragged) {
task_drag = 0;
task_dragged = 0;
return;
}
// switch desktop
if (taskbar_mode == MULTI_DESKTOP) {
gboolean diff_desktop = FALSE;
if (taskbar->desktop != server.desktop && action != CLOSE && action != DESKTOP_LEFT &&
action != DESKTOP_RIGHT) {
diff_desktop = TRUE;
change_desktop(taskbar->desktop);
}
Task *task = click_task(panel, e->xbutton.x, e->xbutton.y);
if (task) {
if (diff_desktop) {
if (action == TOGGLE_ICONIFY) {
if (!window_is_active(task->win))
activate_window(task->win);
} else {
task_handle_mouse_event(task, action);
}
} else {
task_handle_mouse_event(task, action);
}
}
} else {
task_handle_mouse_event(click_task(panel, e->xbutton.x, e->xbutton.y), action);
}
// to keep window below
if (panel_layer == BOTTOM_LAYER)
XLowerWindow(server.display, panel->main_win);
}

12
src/mouse_actions.h Normal file
View File

@@ -0,0 +1,12 @@
#ifndef MOUSE_ACTIONS_H
#define MOUSE_ACTIONS_H
#include "panel.h"
gboolean tint2_handles_click(Panel *panel, XButtonEvent *e);
void handle_mouse_press_event(XEvent *e);
void handle_mouse_move_event(XEvent *e);
void handle_mouse_release_event(XEvent *e);
#endif

View File

@@ -20,6 +20,8 @@
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <X11/Xlib.h> #include <X11/Xlib.h>
#include <X11/Xutil.h> #include <X11/Xutil.h>
#include <X11/Xatom.h> #include <X11/Xatom.h>
@@ -36,8 +38,6 @@
void panel_clear_background(void *obj); void panel_clear_background(void *obj);
int signal_pending;
MouseAction mouse_left; MouseAction mouse_left;
MouseAction mouse_middle; MouseAction mouse_middle;
MouseAction mouse_right; MouseAction mouse_right;
@@ -57,6 +57,8 @@ gboolean task_dragged;
char *panel_window_name = NULL; char *panel_window_name = NULL;
gboolean debug_geometry; gboolean debug_geometry;
gboolean debug_gradients; gboolean debug_gradients;
gboolean startup_notifications;
gboolean debug_thumbnails;
gboolean panel_autohide; gboolean panel_autohide;
int panel_autohide_show_timeout; int panel_autohide_show_timeout;
@@ -147,6 +149,8 @@ void cleanup_panel()
cleanup_freespace(p); cleanup_freespace(p);
} }
free_icon_themes();
free(panel_items_order); free(panel_items_order);
panel_items_order = NULL; panel_items_order = NULL;
free(panel_window_name); free(panel_window_name);
@@ -174,11 +178,11 @@ void init_panel()
{ {
if (panel_config.monitor > (server.num_monitors - 1)) { if (panel_config.monitor > (server.num_monitors - 1)) {
// server.num_monitors minimum value is 1 (see get_monitors()) // server.num_monitors minimum value is 1 (see get_monitors())
fprintf(stderr, "warning : monitor not found. tint2 default to all monitors.\n"); fprintf(stderr, "tint2: warning : monitor not found. tint2 default to all monitors.\n");
panel_config.monitor = 0; panel_config.monitor = 0;
} }
fprintf(stderr, "panel items: %s\n", panel_items_order); fprintf(stderr, "tint2: panel items: %s\n", panel_items_order);
icon_theme_wrapper = NULL; icon_theme_wrapper = NULL;
@@ -206,7 +210,7 @@ void init_panel()
} }
fprintf(stderr, fprintf(stderr,
"tint2 : nb monitor %d, nb monitor used %d, nb desktop %d\n", "tint2: nb monitors %d, nb monitors used %d, nb desktops %d\n",
server.num_monitors, server.num_monitors,
num_panels, num_panels,
server.num_desktops); server.num_desktops);
@@ -286,7 +290,7 @@ void init_panel()
XGCValues gcv; XGCValues gcv;
server.gc = XCreateGC(server.display, p->main_win, 0, &gcv); server.gc = XCreateGC(server.display, p->main_win, 0, &gcv);
} }
// printf("panel %d : %d, %d, %d, %d\n", i, p->posx, p->posy, p->area.width, p->area.height); // fprintf(stderr, "tint2: panel %d : %d, %d, %d, %d\n", i, p->posx, p->posy, p->area.width, p->area.height);
set_panel_properties(p); set_panel_properties(p);
set_panel_background(p); set_panel_background(p);
if (!snapshot_path) { if (!snapshot_path) {
@@ -319,7 +323,7 @@ void panel_compute_size(Panel *panel)
if (panel->fractional_height) if (panel->fractional_height)
panel->area.height = (server.monitors[panel->monitor].height - panel->marginy) * panel->area.height / 100; panel->area.height = (server.monitors[panel->monitor].height - panel->marginy) * panel->area.height / 100;
if (panel->area.bg->border.radius > panel->area.height / 2) { if (panel->area.bg->border.radius > panel->area.height / 2) {
printf("panel_background_id rounded is too big... please fix your tint2rc\n"); fprintf(stderr, "tint2: panel_background_id rounded is too big... please fix your tint2rc\n");
g_array_append_val(backgrounds, *panel->area.bg); g_array_append_val(backgrounds, *panel->area.bg);
panel->area.bg = &g_array_index(backgrounds, Background, backgrounds->len - 1); panel->area.bg = &g_array_index(backgrounds, Background, backgrounds->len - 1);
panel->area.bg->border.radius = panel->area.height / 2; panel->area.bg->border.radius = panel->area.height / 2;
@@ -345,7 +349,7 @@ void panel_compute_size(Panel *panel)
panel->area.width = old_panel_height; panel->area.width = old_panel_height;
if (panel->area.bg->border.radius > panel->area.width / 2) { if (panel->area.bg->border.radius > panel->area.width / 2) {
printf("panel_background_id rounded is too big... please fix your tint2rc\n"); fprintf(stderr, "tint2: panel_background_id rounded is too big... please fix your tint2rc\n");
g_array_append_val(backgrounds, *panel->area.bg); g_array_append_val(backgrounds, *panel->area.bg);
panel->area.bg = &g_array_index(backgrounds, Background, backgrounds->len - 1); panel->area.bg = &g_array_index(backgrounds, Background, backgrounds->len - 1);
panel->area.bg->border.radius = panel->area.width / 2; panel->area.bg->border.radius = panel->area.width / 2;
@@ -398,7 +402,7 @@ void panel_compute_position(Panel *panel)
panel->hidden_width = panel->area.width - diff; panel->hidden_width = panel->area.width - diff;
panel->hidden_height = panel->area.height; panel->hidden_height = panel->area.height;
} }
// printf("panel : posx %d, posy %d, width %d, height %d\n", panel->posx, panel->posy, panel->area.width, // fprintf(stderr, "tint2: panel : posx %d, posy %d, width %d, height %d\n", panel->posx, panel->posy, panel->area.width,
// panel->area.height); // panel->area.height);
} }
@@ -413,7 +417,7 @@ gboolean resize_panel(void *obj)
Panel *panel = (Panel *)obj; Panel *panel = (Panel *)obj;
relayout_with_constraint(&panel->area, 0); relayout_with_constraint(&panel->area, 0);
// printf("resize_panel\n"); // fprintf(stderr, "tint2: resize_panel\n");
if (taskbar_mode != MULTI_DESKTOP && taskbar_enabled) { if (taskbar_mode != MULTI_DESKTOP && taskbar_enabled) {
// propagate width/height on hidden taskbar // propagate width/height on hidden taskbar
int width = panel->taskbar[server.desktop].area.width; int width = panel->taskbar[server.desktop].area.width;
@@ -550,7 +554,7 @@ void update_strut(Panel *p)
long struts[12] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; long struts[12] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
if (panel_horizontal) { if (panel_horizontal) {
int height = p->area.height + p->marginy; int height = p->area.height + p->marginy;
if (panel_strut_policy == STRUT_MINIMUM || (panel_strut_policy == STRUT_FOLLOW_SIZE && panel_autohide)) if (panel_strut_policy == STRUT_MINIMUM || (panel_strut_policy == STRUT_FOLLOW_SIZE && panel_autohide && p->is_hidden))
height = p->hidden_height; height = p->hidden_height;
if (panel_position & TOP) { if (panel_position & TOP) {
struts[2] = height + monitor.y; struts[2] = height + monitor.y;
@@ -565,7 +569,7 @@ void update_strut(Panel *p)
} }
} else { } else {
int width = p->area.width + p->marginx; int width = p->area.width + p->marginx;
if (panel_strut_policy == STRUT_MINIMUM || (panel_strut_policy == STRUT_FOLLOW_SIZE && panel_autohide)) if (panel_strut_policy == STRUT_MINIMUM || (panel_strut_policy == STRUT_FOLLOW_SIZE && panel_autohide && p->is_hidden))
width = p->hidden_width; width = p->hidden_width;
if (panel_position & LEFT) { if (panel_position & LEFT) {
struts[0] = width + monitor.x; struts[0] = width + monitor.x;
@@ -813,15 +817,24 @@ void set_panel_properties(Panel *p)
g_free(name); g_free(name);
} }
long pid = getpid();
XChangeProperty(server.display,
p->main_win,
server.atom._NET_WM_PID,
XA_CARDINAL,
32,
PropModeReplace,
(unsigned char *)&pid,
1);
// Dock // Dock
long val = server.atom._NET_WM_WINDOW_TYPE_DOCK;
XChangeProperty(server.display, XChangeProperty(server.display,
p->main_win, p->main_win,
server.atom._NET_WM_WINDOW_TYPE, server.atom._NET_WM_WINDOW_TYPE,
XA_ATOM, XA_ATOM,
32, 32,
PropModeReplace, PropModeReplace,
(unsigned char *)&val, (unsigned char *)&server.atom._NET_WM_WINDOW_TYPE_DOCK,
1); 1);
place_panel_all_desktops(p); place_panel_all_desktops(p);
@@ -1083,6 +1096,7 @@ void shrink_panel(Panel *panel)
systray.area.resize_needed = TRUE; systray.area.resize_needed = TRUE;
schedule_redraw(&systray.area); schedule_redraw(&systray.area);
refresh_systray = TRUE; refresh_systray = TRUE;
update_minimized_icon_positions(panel);
} }
} }
@@ -1133,6 +1147,63 @@ void _schedule_panel_redraw(const char *file, const char *function, const int li
{ {
panel_refresh = TRUE; panel_refresh = TRUE;
if (debug_fps) { if (debug_fps) {
fprintf(stderr, YELLOW "%s %s %d: triggering panel redraw" RESET "\n", file, function, line); fprintf(stderr, YELLOW "tint2: %s %s %d: triggering panel redraw" RESET "\n", file, function, line);
} }
} }
void save_panel_screenshot(const Panel *panel, const char *path)
{
imlib_context_set_drawable(panel->temp_pmap);
Imlib_Image img = imlib_create_image_from_drawable(0, 0, 0, panel->area.width, panel->area.height, 1);
if (!img) {
XImage *ximg =
XGetImage(server.display, panel->temp_pmap, 0, 0, panel->area.width, panel->area.height, AllPlanes, ZPixmap);
if (ximg) {
DATA32 *pixels = (DATA32 *)calloc(panel->area.width * panel->area.height, sizeof(DATA32));
for (int x = 0; x < panel->area.width; x++) {
for (int y = 0; y < panel->area.height; y++) {
DATA32 xpixel = XGetPixel(ximg, x, y);
DATA32 r = (xpixel >> 16) & 0xff;
DATA32 g = (xpixel >> 8) & 0xff;
DATA32 b = (xpixel >> 0) & 0xff;
DATA32 a = 0x0;
DATA32 argb = (a << 24) | (r << 16) | (g << 8) | b;
pixels[y * panel->area.width + x] = argb;
}
}
XDestroyImage(ximg);
img = imlib_create_image_using_data(panel->area.width, panel->area.height, pixels);
}
}
if (img) {
imlib_context_set_image(img);
if (!panel_horizontal) {
// rotate 90° vertical panel
imlib_image_flip_horizontal();
imlib_image_flip_diagonal();
}
imlib_save_image(path);
imlib_free_image();
}
}
void save_screenshot(const char *path)
{
Panel *panel = &panels[0];
if (panel->area.width > server.monitors[0].width)
panel->area.width = server.monitors[0].width;
panel->temp_pmap =
XCreatePixmap(server.display, server.root_win, panel->area.width, panel->area.height, server.depth);
render_panel(panel);
XSync(server.display, False);
save_panel_screenshot(panel, path);
}

View File

@@ -29,7 +29,6 @@
#include "battery.h" #include "battery.h"
#endif #endif
extern int signal_pending;
// -------------------------------------------------- // --------------------------------------------------
// mouse events // mouse events
extern MouseAction mouse_left; extern MouseAction mouse_left;
@@ -90,9 +89,12 @@ extern Imlib_Image default_icon;
#define DEFAULT_FONT "sans 10" #define DEFAULT_FONT "sans 10"
extern char *default_font; extern char *default_font;
extern XSettingsClient *xsettings_client; extern XSettingsClient *xsettings_client;
extern gboolean startup_notifications;
extern gboolean debug_geometry; extern gboolean debug_geometry;
extern gboolean debug_fps; extern gboolean debug_fps;
extern double tracing_fps_threshold;
extern gboolean debug_frames; extern gboolean debug_frames;
extern gboolean debug_thumbnails;
typedef struct Panel { typedef struct Panel {
Area area; Area area;
@@ -164,7 +166,7 @@ gboolean resize_panel(void *obj);
void render_panel(Panel *panel); void render_panel(Panel *panel);
void shrink_panel(Panel *panel); void shrink_panel(Panel *panel);
void _schedule_panel_redraw(const char *file, const char *function, const int line); void _schedule_panel_redraw(const char *file, const char *function, const int line);
#define schedule_panel_redraw() _schedule_panel_redraw(__FILE__, __FUNCTION__, __LINE__) #define schedule_panel_redraw() _schedule_panel_redraw(__FILE__, __func__, __LINE__)
void set_panel_items_order(Panel *p); void set_panel_items_order(Panel *p);
void place_panel_all_desktops(Panel *p); void place_panel_all_desktops(Panel *p);
@@ -206,4 +208,7 @@ void default_font_changed();
void free_icon(Imlib_Image icon); void free_icon(Imlib_Image icon);
Imlib_Image scale_icon(Imlib_Image original, int icon_size); Imlib_Image scale_icon(Imlib_Image original, int icon_size);
void save_screenshot(const char *path);
void save_panel_screenshot(const Panel *panel, const char *path);
#endif #endif

View File

@@ -7,11 +7,7 @@
#include "common.h" #include "common.h"
#include "area.h" #include "area.h"
typedef enum SeparatorStyle { typedef enum SeparatorStyle { SEPARATOR_EMPTY = 0, SEPARATOR_LINE, SEPARATOR_DOTS } SeparatorStyle;
SEPARATOR_EMPTY = 0,
SEPARATOR_LINE,
SEPARATOR_DOTS
} SeparatorStyle;
typedef struct Separator { typedef struct Separator {
Area area; Area area;

View File

@@ -18,21 +18,24 @@
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**************************************************************************/ **************************************************************************/
#include <X11/extensions/Xdamage.h>
#include <X11/extensions/Xrender.h> #include <X11/extensions/Xrender.h>
#include <X11/extensions/Xrandr.h> #include <X11/extensions/Xrandr.h>
#include <stdio.h> #include <stdio.h>
#include <fcntl.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include <unistd.h> #include <unistd.h>
#include "server.h" #include "common.h"
#include "config.h" #include "config.h"
#include "server.h"
#include "signals.h"
#include "window.h" #include "window.h"
Server server; Server server;
gboolean primary_monitor_first = FALSE;
void server_catch_error(Display *d, XErrorEvent *ev) void server_catch_error(Display *d, XErrorEvent *ev)
{ {
} }
@@ -120,6 +123,14 @@ void server_init_atoms()
server.atom.TARGETS = XInternAtom(server.display, "TARGETS", False); server.atom.TARGETS = XInternAtom(server.display, "TARGETS", False);
} }
const char *GetAtomName(Display *disp, Atom a)
{
if (a == None)
return "None";
else
return XGetAtomName(disp, a);
}
void cleanup_server() void cleanup_server()
{ {
if (server.colormap) if (server.colormap)
@@ -250,7 +261,7 @@ void get_root_pixmap()
server.root_pmap = ret; server.root_pmap = ret;
if (server.root_pmap == None) { if (server.root_pmap == None) {
fprintf(stderr, "tint2 : pixmap background detection failed\n"); fprintf(stderr, "tint2: pixmap background detection failed\n");
} else { } else {
XGCValues gcv; XGCValues gcv;
gcv.ts_x_origin = 0; gcv.ts_x_origin = 0;
@@ -268,13 +279,6 @@ int compare_monitor_pos(const void *monitor1, const void *monitor2)
const Monitor *m1 = (const Monitor *)monitor1; const Monitor *m1 = (const Monitor *)monitor1;
const Monitor *m2 = (const Monitor *)monitor2; const Monitor *m2 = (const Monitor *)monitor2;
if (primary_monitor_first) {
if (m1->primary && !m2->primary)
return -1;
if (!m1->primary && m2->primary)
return 1;
}
if (m1->x < m2->x) { if (m1->x < m2->x) {
return -1; return -1;
} else if (m1->x > m2->x) { } else if (m1->x > m2->x) {
@@ -317,14 +321,14 @@ void get_monitors()
if (res && res->ncrtc >= num_monitors) { if (res && res->ncrtc >= num_monitors) {
// use xrandr to identify monitors (does not work with proprietery nvidia drivers) // use xrandr to identify monitors (does not work with proprietery nvidia drivers)
printf("xRandr: Found crtc's: %d\n", res->ncrtc); fprintf(stderr, "tint2: xRandr: Found crtc's: %d\n", res->ncrtc);
server.monitors = calloc(res->ncrtc, sizeof(Monitor)); server.monitors = calloc(res->ncrtc, sizeof(Monitor));
num_monitors = 0; num_monitors = 0;
for (int i = 0; i < res->ncrtc; ++i) { for (int i = 0; i < res->ncrtc; ++i) {
XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(server.display, res, res->crtcs[i]); XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(server.display, res, res->crtcs[i]);
// Ignore empty crtc // Ignore empty crtc
if (!crtc_info->width || !crtc_info->height) { if (!crtc_info->width || !crtc_info->height) {
printf("xRandr: crtc %d seems disabled\n", i); fprintf(stderr, "tint2: xRandr: crtc %d seems disabled\n", i);
XRRFreeCrtcInfo(crtc_info); XRRFreeCrtcInfo(crtc_info);
continue; continue;
} }
@@ -337,7 +341,7 @@ void get_monitors()
server.monitors[i_monitor].names = calloc((crtc_info->noutput + 1), sizeof(gchar *)); server.monitors[i_monitor].names = calloc((crtc_info->noutput + 1), sizeof(gchar *));
for (int j = 0; j < crtc_info->noutput; ++j) { for (int j = 0; j < crtc_info->noutput; ++j) {
XRROutputInfo *output_info = XRRGetOutputInfo(server.display, res, crtc_info->outputs[j]); XRROutputInfo *output_info = XRRGetOutputInfo(server.display, res, crtc_info->outputs[j]);
printf("xRandr: Linking output %s with crtc %d\n", output_info->name, i); fprintf(stderr, "tint2: xRandr: Linking output %s with crtc %d\n", output_info->name, i);
server.monitors[i_monitor].names[j] = g_strdup(output_info->name); server.monitors[i_monitor].names[j] = g_strdup(output_info->name);
XRRFreeOutputInfo(output_info); XRRFreeOutputInfo(output_info);
server.monitors[i_monitor].primary = crtc_info->outputs[j] == primary_output; server.monitors[i_monitor].primary = crtc_info->outputs[j] == primary_output;
@@ -394,7 +398,7 @@ void get_monitors()
void print_monitors() void print_monitors()
{ {
fprintf(stderr, "Number of monitors: %d\n", server.num_monitors); fprintf(stderr, "tint2: Number of monitors: %d\n", server.num_monitors);
for (int i = 0; i < server.num_monitors; i++) { for (int i = 0; i < server.num_monitors; i++) {
fprintf(stderr, fprintf(stderr,
"Monitor %d: x = %d, y = %d, w = %d, h = %d\n", "Monitor %d: x = %d, y = %d, w = %d, h = %d\n",
@@ -516,10 +520,10 @@ int get_current_desktop()
int ncols = x_screen_width / work_area_width; int ncols = x_screen_width / work_area_width;
// fprintf(stderr, "\n"); // fprintf(stderr, "tint2: \n");
// fprintf(stderr, "Work area size: %d x %d\n", work_area_width, work_area_height); // fprintf(stderr, "tint2: Work area size: %d x %d\n", work_area_width, work_area_height);
// fprintf(stderr, "Viewport pos: %d x %d\n", viewport_x, viewport_y); // fprintf(stderr, "tint2: Viewport pos: %d x %d\n", viewport_x, viewport_y);
// fprintf(stderr, "Viewport i: %d\n", (viewport_y / work_area_height) * ncols + viewport_x / work_area_width); // fprintf(stderr, "tint2: Viewport i: %d\n", (viewport_y / work_area_height) * ncols + viewport_x / work_area_width);
int result = (viewport_y / work_area_height) * ncols + viewport_x / work_area_width; int result = (viewport_y / work_area_height) * ncols + viewport_x / work_area_width;
return MAX(0, MIN(server.num_desktops - 1, result)); return MAX(0, MIN(server.num_desktops - 1, result));
@@ -550,7 +554,7 @@ void get_desktops()
} }
if (server.num_desktops == 0) { if (server.num_desktops == 0) {
server.num_desktops = 1; server.num_desktops = 1;
fprintf(stderr, "warning : WM doesn't respect NETWM specs. tint2 default to 1 desktop.\n"); fprintf(stderr, "tint2: warning : WM doesn't respect NETWM specs. tint2 default to 1 desktop.\n");
} }
} }
@@ -594,15 +598,77 @@ void server_init_visual()
server.real_transparency = TRUE; server.real_transparency = TRUE;
server.depth = 32; server.depth = 32;
printf("real transparency on... depth: %d\n", server.depth); fprintf(stderr, "tint2: real transparency on... depth: %d\n", server.depth);
server.colormap = XCreateColormap(server.display, server.root_win, visual, AllocNone); server.colormap = XCreateColormap(server.display, server.root_win, visual, AllocNone);
server.visual = visual; server.visual = visual;
} else { } else {
// no composite manager or snapshot mode => fake transparency // no composite manager or snapshot mode => fake transparency
server.real_transparency = FALSE; server.real_transparency = FALSE;
server.depth = DefaultDepth(server.display, server.screen); server.depth = DefaultDepth(server.display, server.screen);
printf("real transparency off.... depth: %d\n", server.depth); fprintf(stderr, "tint2: real transparency off.... depth: %d\n", server.depth);
server.colormap = DefaultColormap(server.display, server.screen); server.colormap = DefaultColormap(server.display, server.screen);
server.visual = DefaultVisual(server.display, server.screen); server.visual = DefaultVisual(server.display, server.screen);
} }
} }
void server_init_xdamage()
{
XDamageQueryExtension(server.display, &server.xdamage_event_type, &server.xdamage_event_error_type);
server.xdamage_event_type += XDamageNotify;
server.xdamage_event_error_type += XDamageNotify;
}
// Forward mouse click to the desktop window
void forward_click(XEvent *e)
{
// forward the click to the desktop window (thanks conky)
XUngrabPointer(server.display, e->xbutton.time);
e->xbutton.window = server.root_win;
// icewm doesn't open under the mouse.
// and xfce doesn't open at all.
e->xbutton.x = e->xbutton.x_root;
e->xbutton.y = e->xbutton.y_root;
// fprintf(stderr, "tint2: **** %d, %d\n", e->xbutton.x, e->xbutton.y);
// XSetInputFocus(server.display, e->xbutton.window, RevertToParent, e->xbutton.time);
XSendEvent(server.display, e->xbutton.window, False, ButtonPressMask, e);
}
void handle_crash(const char *reason)
{
#ifndef DISABLE_BACKTRACE
char path[4096];
sprintf(path, "%s/.tint2-crash.log", get_home_dir());
int log_fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0600);
log_string(log_fd, RED "tint2: crashed, reason: ");
log_string(log_fd, reason);
log_string(log_fd, RESET "\n");
dump_backtrace(log_fd);
log_string(log_fd, RED "Please create a bug report with this log output." RESET "\n");
close(log_fd);
#endif
}
void x11_io_error(Display *display)
{
handle_crash("X11 I/O error");
}
#ifdef HAVE_SN
static int error_trap_depth = 0;
void error_trap_push(SnDisplay *display, Display *xdisplay)
{
++error_trap_depth;
}
void error_trap_pop(SnDisplay *display, Display *xdisplay)
{
if (error_trap_depth == 0) {
fprintf(stderr, "tint2: Error trap underflow!\n");
return;
}
XSync(xdisplay, False); /* get all errors out of the queue */
--error_trap_depth;
}
#endif // HAVE_SN

View File

@@ -93,6 +93,15 @@ typedef struct Global_atom {
Atom TARGETS; Atom TARGETS;
} Global_atom; } Global_atom;
typedef struct Property {
unsigned char *data;
int format, nitems;
Atom type;
} Property;
// Returns the name of an Atom as string. Do not free the string.
const char *GetAtomName(Display *disp, Atom a);
typedef struct Monitor { typedef struct Monitor {
int x; int x;
int y; int y;
@@ -111,6 +120,7 @@ typedef struct Viewport {
typedef struct Server { typedef struct Server {
Display *display; Display *display;
int x11_fd;
Window root_win; Window root_win;
Window composite_manager; Window composite_manager;
gboolean real_transparency; gboolean real_transparency;
@@ -135,6 +145,9 @@ typedef struct Server {
Colormap colormap; Colormap colormap;
Colormap colormap32; Colormap colormap32;
Global_atom atom; Global_atom atom;
int xdamage_event_type;
int xdamage_event_error_type;
gboolean has_shm;
#ifdef HAVE_SN #ifdef HAVE_SN
SnDisplay *sn_display; SnDisplay *sn_display;
GTree *pids; GTree *pids;
@@ -153,6 +166,10 @@ Atom server_get_atom(char *atom_name);
void server_catch_error(Display *d, XErrorEvent *ev); void server_catch_error(Display *d, XErrorEvent *ev);
void server_init_atoms(); void server_init_atoms();
void server_init_visual(); void server_init_visual();
void server_init_xdamage();
void x11_io_error(Display *display);
void handle_crash(const char *reason);
// detect root background // detect root background
void get_root_pixmap(); void get_root_pixmap();
@@ -167,4 +184,12 @@ GSList *get_desktop_names();
int get_current_desktop(); int get_current_desktop();
void change_desktop(int desktop); void change_desktop(int desktop);
// Forward mouse click to the desktop window
void forward_click(XEvent *e);
#ifdef HAVE_SN
void error_trap_push(SnDisplay *display, Display *xdisplay);
void error_trap_pop(SnDisplay *display, Display *xdisplay);
#endif
#endif #endif

166
src/signals.c Normal file
View File

@@ -0,0 +1,166 @@
#include <errno.h>
#include <fcntl.h>
#include <glib.h>
#ifdef HAVE_SN
#include <libsn/sn.h>
#endif
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <unistd.h>
#include "common.h"
#include "panel.h"
#include "launcher.h"
#include "server.h"
#include "signals.h"
static sig_atomic_t signal_pending;
void signal_handler(int sig)
{
// signal handler is light as it should be
signal_pending = sig;
}
void reset_signals()
{
for (int sig = 1; sig < 32; sig++) {
signal(sig, SIG_DFL);
}
sigset_t signal_set;
sigemptyset(&signal_set);
sigprocmask(SIG_SETMASK, &signal_set, NULL);
}
void init_signals()
{
// Set signal handlers
signal_pending = 0;
reset_signals();
struct sigaction sa_chld = {.sa_handler = SIG_IGN};
sigaction(SIGCHLD, &sa_chld, 0);
struct sigaction sa = {.sa_handler = signal_handler, .sa_flags = SA_RESTART};
sigaction(SIGUSR1, &sa, 0);
sigaction(SIGUSR2, &sa, 0);
sigaction(SIGINT, &sa, 0);
sigaction(SIGTERM, &sa, 0);
sigaction(SIGHUP, &sa, 0);
#ifdef BACKTRACE_ON_SIGNAL
struct sigaction sa_crash = {.sa_handler = crash_handler};
sigaction(SIGSEGV, &sa_crash, 0);
sigaction(SIGFPE, &sa_crash, 0);
sigaction(SIGPIPE, &sa_crash, 0);
sigaction(SIGBUS, &sa_crash, 0);
sigaction(SIGABRT, &sa_crash, 0);
sigaction(SIGSYS, &sa_crash, 0);
#endif
}
#ifdef BACKTRACE_ON_SIGNAL
void crash_handler(int sig)
{
handle_crash(signal_name(sig));
struct sigaction sa = {.sa_handler = SIG_DFL};
sigaction(sig, &sa, 0);
raise(sig);
}
#endif
int sigchild_pipe_valid = FALSE;
int sigchild_pipe[2];
static void sigchld_handler(int sig)
{
if (!sigchild_pipe_valid)
return;
int savedErrno = errno;
ssize_t unused = write(sigchild_pipe[1], "x", 1);
(void)unused;
fsync(sigchild_pipe[1]);
errno = savedErrno;
}
void sigchld_handler_async()
{
// Wait for all dead processes
pid_t pid;
int status;
while ((pid = waitpid(-1, &status, WNOHANG)) != -1 && pid != 0) {
#ifdef HAVE_SN
if (startup_notifications) {
SnLauncherContext *ctx = (SnLauncherContext *)g_tree_lookup(server.pids, GINT_TO_POINTER(pid));
if (ctx) {
g_tree_remove(server.pids, GINT_TO_POINTER(pid));
sn_launcher_context_complete(ctx);
sn_launcher_context_unref(ctx);
}
}
#endif
for (GList *l = panel_config.execp_list; l; l = l->next) {
Execp *execp = (Execp *)l->data;
if (g_tree_lookup(execp->backend->cmd_pids, GINT_TO_POINTER(pid)))
execp_cmd_completed(execp, pid);
}
}
}
void handle_sigchld_events()
{
if (sigchild_pipe_valid) {
char buffer[1];
while (read(sigchild_pipe[0], buffer, sizeof(buffer)) > 0) {
sigchld_handler_async();
}
}
}
void init_signals_postconfig()
{
gboolean need_sigchld = FALSE;
#ifdef HAVE_SN
// Initialize startup-notification
if (startup_notifications) {
server.sn_display = sn_display_new(server.display, error_trap_push, error_trap_pop);
server.pids = g_tree_new(cmp_ptr);
need_sigchld = TRUE;
}
#endif // HAVE_SN
if (panel_config.execp_list)
need_sigchld = TRUE;
if (need_sigchld) {
// Setup a handler for child termination
if (pipe(sigchild_pipe) != 0) {
fprintf(stderr, "tint2: Creating pipe failed.\n");
} else {
fcntl(sigchild_pipe[0], F_SETFL, O_NONBLOCK | fcntl(sigchild_pipe[0], F_GETFL));
fcntl(sigchild_pipe[1], F_SETFL, O_NONBLOCK | fcntl(sigchild_pipe[1], F_GETFL));
sigchild_pipe_valid = 1;
struct sigaction act = {.sa_handler = sigchld_handler, .sa_flags = SA_RESTART};
if (sigaction(SIGCHLD, &act, 0)) {
perror("sigaction");
}
}
}
}
void emit_self_restart(const char *reason)
{
fprintf(stderr,
YELLOW "%s %d: triggering tint2 restart, reason: %s" RESET "\n",
__FILE__,
__LINE__,
reason);
signal_pending = SIGUSR1;
}
int get_signal_pending()
{
return signal_pending;
}

15
src/signals.h Normal file
View File

@@ -0,0 +1,15 @@
#ifndef SIGNALS_H
#define SIGNALS_H
void init_signals();
void init_signals_postconfig();
void emit_self_restart(const char *reason);
int get_signal_pending();
void reset_signals();
void handle_sigchld_events();
extern int sigchild_pipe_valid;
extern int sigchild_pipe[2];
#endif

View File

@@ -114,10 +114,10 @@ void init_systray()
return; return;
systray_composited = !server.disable_transparency && server.visual32 && server.colormap32; systray_composited = !server.disable_transparency && server.visual32 && server.colormap32;
fprintf(stderr, "Systray composited rendering %s\n", systray_composited ? "on" : "off"); fprintf(stderr, "tint2: Systray composited rendering %s\n", systray_composited ? "on" : "off");
if (!systray_composited) { if (!systray_composited) {
fprintf(stderr, "systray_asb forced to 100 0 0\n"); fprintf(stderr, "tint2: systray_asb forced to 100 0 0\n");
systray.alpha = 100; systray.alpha = 100;
systray.brightness = systray.saturation = 0; systray.brightness = systray.saturation = 0;
} }
@@ -142,8 +142,8 @@ void init_systray_panel(void *p)
void systray_compute_geometry(int *size) void systray_compute_geometry(int *size)
{ {
systray.icon_size = panel_horizontal ? systray.area.height : systray.area.width; systray.icon_size = panel_horizontal ? systray.area.height : systray.area.width;
systray.icon_size -= MAX(left_right_border_width(&systray.area), top_bottom_border_width(&systray.area)) + systray.icon_size -=
2 * systray.area.paddingy; MAX(left_right_border_width(&systray.area), top_bottom_border_width(&systray.area)) + 2 * systray.area.paddingy;
if (systray_max_icon_size > 0) if (systray_max_icon_size > 0)
systray.icon_size = MIN(systray.icon_size, systray_max_icon_size); systray.icon_size = MIN(systray.icon_size, systray_max_icon_size);
@@ -160,8 +160,7 @@ void systray_compute_geometry(int *size)
height - (systray.icons_per_column - 1) * (systray.icon_size + systray.area.paddingx) - systray.icon_size; height - (systray.icons_per_column - 1) * (systray.icon_size + systray.area.paddingx) - systray.icon_size;
systray.icons_per_row = count / systray.icons_per_column + (count % systray.icons_per_column != 0); systray.icons_per_row = count / systray.icons_per_column + (count % systray.icons_per_column != 0);
*size = left_right_border_width(&systray.area) + 2 * systray.area.paddingxlr + *size = left_right_border_width(&systray.area) + 2 * systray.area.paddingxlr +
(systray.icon_size * systray.icons_per_row) + (systray.icon_size * systray.icons_per_row) + ((systray.icons_per_row - 1) * systray.area.paddingx);
((systray.icons_per_row - 1) * systray.area.paddingx);
} else { } else {
int width = systray.area.width - left_right_border_width(&systray.area) - 2 * systray.area.paddingy; int width = systray.area.width - left_right_border_width(&systray.area) - 2 * systray.area.paddingy;
// here icons_per_row always higher than 0 // here icons_per_row always higher than 0
@@ -185,7 +184,7 @@ int systray_compute_desired_size(void *obj)
gboolean resize_systray(void *obj) gboolean resize_systray(void *obj)
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
int size; int size;
systray_compute_geometry(&size); systray_compute_geometry(&size);
@@ -228,7 +227,7 @@ gboolean resize_systray(void *obj)
void draw_systray(void *obj, cairo_t *c) void draw_systray(void *obj, cairo_t *c)
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, BLUE "[%f] %s:%d" RESET "\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, BLUE "tint2: [%f] %s:%d" RESET "\n", profiling_get_time(), __func__, __LINE__);
if (systray_composited) { if (systray_composited) {
if (render_background) if (render_background)
XFreePixmap(server.display, render_background); XFreePixmap(server.display, render_background);
@@ -253,15 +252,12 @@ void systray_dump_geometry(void *obj, int indent)
{ {
Systray *tray = (Systray *)obj; Systray *tray = (Systray *)obj;
fprintf(stderr, fprintf(stderr, "tint2: %*sIcons:\n", indent, "");
"%*sIcons:\n",
indent,
"");
indent += 2; indent += 2;
for (GSList *l = tray->list_icons; l; l = l->next) { for (GSList *l = tray->list_icons; l; l = l->next) {
TrayWindow *traywin = (TrayWindow *)l->data; TrayWindow *traywin = (TrayWindow *)l->data;
fprintf(stderr, fprintf(stderr,
"%*sIcon: x = %d, y = %d, w = %d, h = %d, name = %s\n", "tint2: %*sIcon: x = %d, y = %d, w = %d, h = %d, name = %s\n",
indent, indent,
"", "",
traywin->x, traywin->x,
@@ -275,7 +271,7 @@ void systray_dump_geometry(void *obj, int indent)
void on_change_systray(void *obj) void on_change_systray(void *obj)
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
if (systray.icons_per_column == 0 || systray.icons_per_row == 0) if (systray.icons_per_column == 0 || systray.icons_per_row == 0)
return; return;
@@ -305,7 +301,7 @@ void on_change_systray(void *obj)
if (systray_profile) if (systray_profile)
fprintf(stderr, fprintf(stderr,
"%s:%d win = %lu (%s), parent = %lu, x = %d, y = %d\n", "%s:%d win = %lu (%s), parent = %lu, x = %d, y = %d\n",
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name, traywin->name,
@@ -336,7 +332,7 @@ void on_change_systray(void *obj)
unsigned int width, height, depth; unsigned int width, height, depth;
Window root; Window root;
if (!XGetGeometry(server.display, traywin->parent, &root, &xpos, &ypos, &width, &height, &border_width, &depth)) { if (!XGetGeometry(server.display, traywin->parent, &root, &xpos, &ypos, &width, &height, &border_width, &depth)) {
fprintf(stderr, RED "Couldn't get geometry of window!" RESET "\n"); fprintf(stderr, RED "tint2: Couldn't get geometry of window!" RESET "\n");
} }
if (width != traywin->width || height != traywin->height || xpos != traywin->x || ypos != traywin->y) { if (width != traywin->width || height != traywin->height || xpos != traywin->x || ypos != traywin->y) {
if (systray_profile) if (systray_profile)
@@ -362,7 +358,7 @@ void on_change_systray(void *obj)
void start_net() void start_net()
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
if (net_sel_win) { if (net_sel_win) {
// protocol already started // protocol already started
if (!systray_enabled) if (!systray_enabled)
@@ -399,11 +395,11 @@ void start_net()
&bytes_after, &bytes_after,
&prop); &prop);
fprintf(stderr, RED "tint2 : another systray is running" RESET); fprintf(stderr, RED "tint2: another systray is running, cannot use systray" RESET);
if (ret == Success && prop) { if (ret == Success && prop) {
pid = prop[1] * 256; pid = prop[1] * 256;
pid += prop[0]; pid += prop[0];
fprintf(stderr, " pid=%d", pid); fprintf(stderr, "tint2: pid=%d", pid);
} }
fprintf(stderr, RESET "\n"); fprintf(stderr, RESET "\n");
return; return;
@@ -411,7 +407,7 @@ void start_net()
// init systray protocol // init systray protocol
net_sel_win = XCreateSimpleWindow(server.display, server.root_win, -1, -1, 1, 1, 0, 0, 0); net_sel_win = XCreateSimpleWindow(server.display, server.root_win, -1, -1, 1, 1, 0, 0, 0);
fprintf(stderr, "systray window %ld\n", net_sel_win); fprintf(stderr, "tint2: systray window %ld\n", net_sel_win);
// v0.3 trayer specification. tint2 always horizontal. // v0.3 trayer specification. tint2 always horizontal.
// Vertical panel will draw the systray horizontal. // Vertical panel will draw the systray horizontal.
@@ -462,13 +458,13 @@ void start_net()
XSetSelectionOwner(server.display, server.atom._NET_SYSTEM_TRAY_SCREEN, net_sel_win, CurrentTime); XSetSelectionOwner(server.display, server.atom._NET_SYSTEM_TRAY_SCREEN, net_sel_win, CurrentTime);
if (XGetSelectionOwner(server.display, server.atom._NET_SYSTEM_TRAY_SCREEN) != net_sel_win) { if (XGetSelectionOwner(server.display, server.atom._NET_SYSTEM_TRAY_SCREEN) != net_sel_win) {
stop_net(); stop_net();
fprintf(stderr, RED "tint2 : can't get systray manager" RESET "\n"); fprintf(stderr, RED "tint2: cannot find systray manager" RESET "\n");
return; return;
} }
fprintf(stderr, GREEN "tint2 : systray started" RESET "\n"); fprintf(stderr, GREEN "tint2: systray started" RESET "\n");
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
XClientMessageEvent ev; XClientMessageEvent ev;
ev.type = ClientMessage; ev.type = ClientMessage;
ev.window = server.root_win; ev.window = server.root_win;
@@ -482,10 +478,10 @@ void start_net()
XSendEvent(server.display, server.root_win, False, StructureNotifyMask, (XEvent *)&ev); XSendEvent(server.display, server.root_win, False, StructureNotifyMask, (XEvent *)&ev);
} }
void net_message(XClientMessageEvent *e) void handle_systray_event(XClientMessageEvent *e)
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
Window win; Window win;
unsigned long opcode = e->data.l[1]; unsigned long opcode = e->data.l[1];
@@ -503,9 +499,9 @@ void net_message(XClientMessageEvent *e)
default: default:
if (opcode == server.atom._NET_SYSTEM_TRAY_MESSAGE_DATA) if (opcode == server.atom._NET_SYSTEM_TRAY_MESSAGE_DATA)
fprintf(stderr, "message from dockapp: %s\n", e->data.b); fprintf(stderr, "tint2: message from dockapp: %s\n", e->data.b);
else else
fprintf(stderr, RED "SYSTEM_TRAY : unknown message type" RESET "\n"); fprintf(stderr, RED "tint2: SYSTEM_TRAY : unknown message type" RESET "\n");
break; break;
} }
} }
@@ -513,7 +509,7 @@ void net_message(XClientMessageEvent *e)
void stop_net() void stop_net()
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
if (systray.list_icons) { if (systray.list_icons) {
// remove_icon change systray.list_icons // remove_icon change systray.list_icons
while (systray.list_icons) while (systray.list_icons)
@@ -533,10 +529,10 @@ gboolean error;
int window_error_handler(Display *d, XErrorEvent *e) int window_error_handler(Display *d, XErrorEvent *e)
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, RED "[%f] %s:%d" RESET "\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, RED "tint2: [%f] %s:%d" RESET "\n", profiling_get_time(), __func__, __LINE__);
error = TRUE; error = TRUE;
if (e->error_code != BadWindow) { if (e->error_code != BadWindow) {
fprintf(stderr, RED "systray: error code %d" RESET "\n", e->error_code); fprintf(stderr, RED "tint2: systray: error code %d" RESET "\n", e->error_code);
} }
return 0; return 0;
} }
@@ -568,18 +564,18 @@ static gint compare_traywindows(gconstpointer a, gconstpointer b)
void print_icons() void print_icons()
{ {
fprintf(stderr, "systray.list_icons: \n"); fprintf(stderr, "tint2: systray.list_icons: \n");
for (GSList *l = systray.list_icons; l; l = l->next) { for (GSList *l = systray.list_icons; l; l = l->next) {
TrayWindow *t = l->data; TrayWindow *t = l->data;
fprintf(stderr, "%s\n", t->name); fprintf(stderr, "tint2: %s\n", t->name);
} }
fprintf(stderr, "systray.list_icons order: \n"); fprintf(stderr, "tint2: systray.list_icons order: \n");
for (GSList *l = systray.list_icons; l; l = l->next) { for (GSList *l = systray.list_icons; l; l = l->next) {
if (l->next) { if (l->next) {
TrayWindow *t = l->data; TrayWindow *t = l->data;
TrayWindow *u = l->next->data; TrayWindow *u = l->next->data;
int cmp = compare_traywindows(t, u); int cmp = compare_traywindows(t, u);
fprintf(stderr, "%s %s %s\n", t->name, cmp < 0 ? "<" : cmp == 0 ? "=" : ">", u->name); fprintf(stderr, "tint2: %s %s %s\n", t->name, cmp < 0 ? "<" : cmp == 0 ? "=" : ">", u->name);
} }
} }
} }
@@ -588,16 +584,16 @@ gboolean reject_icon(Window win)
{ {
if (systray_hide_name_filter && strlen(systray_hide_name_filter)) { if (systray_hide_name_filter && strlen(systray_hide_name_filter)) {
if (!systray_hide_name_regex) { if (!systray_hide_name_regex) {
systray_hide_name_regex = (regex_t *) calloc(1, sizeof(*systray_hide_name_regex)); systray_hide_name_regex = (regex_t *)calloc(1, sizeof(*systray_hide_name_regex));
if (regcomp(systray_hide_name_regex, systray_hide_name_filter, 0) != 0) { if (regcomp(systray_hide_name_regex, systray_hide_name_filter, 0) != 0) {
fprintf(stderr, RED "Could not compile regex %s" RESET "\n", systray_hide_name_filter); fprintf(stderr, RED "tint2: Could not compile regex %s" RESET "\n", systray_hide_name_filter);
free_and_null(systray_hide_name_regex); free_and_null(systray_hide_name_regex);
return FALSE; return FALSE;
} }
} }
char *name = get_window_name(win); char *name = get_window_name(win);
if (regexec(systray_hide_name_regex, name, 0, NULL, 0) == 0) { if (regexec(systray_hide_name_regex, name, 0, NULL, 0) == 0) {
fprintf(stderr, GREEN "Filtering out systray icon '%s'" RESET "\n", name); fprintf(stderr, GREEN "tint2: Filtering out systray icon '%s'" RESET "\n", name);
return TRUE; return TRUE;
} }
} }
@@ -627,7 +623,7 @@ gboolean add_icon(Window win)
char *name = get_window_name(win); char *name = get_window_name(win);
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d win = %lu (%s)\n", profiling_get_time(), __FUNCTION__, __LINE__, win, name); fprintf(stderr, "tint2: [%f] %s:%d win = %lu (%s)\n", profiling_get_time(), __func__, __LINE__, win, name);
Panel *panel = systray.area.panel; Panel *panel = systray.area.panel;
// Get the process ID of the application that created the window // Get the process ID of the application that created the window
@@ -659,7 +655,7 @@ gboolean add_icon(Window win)
// Create the parent window that will embed the icon // Create the parent window that will embed the icon
XWindowAttributes attr; XWindowAttributes attr;
if (systray_profile) if (systray_profile)
fprintf(stderr, "XGetWindowAttributes(server.display, win = %ld, &attr)\n", win); fprintf(stderr, "tint2: XGetWindowAttributes(server.display, win = %ld, &attr)\n", win);
if (XGetWindowAttributes(server.display, win, &attr) == False) { if (XGetWindowAttributes(server.display, win, &attr) == False) {
free(name); free(name);
XSelectInput(server.display, win, NoEventMask); XSelectInput(server.display, win, NoEventMask);
@@ -683,7 +679,7 @@ gboolean add_icon(Window win)
win, win,
name, name,
pid, pid,
attr.visual, (void*)attr.visual,
attr.colormap, attr.colormap,
attr.depth, attr.depth,
attr.width, attr.width,
@@ -710,7 +706,7 @@ gboolean add_icon(Window win)
} }
if (systray_profile) if (systray_profile)
fprintf(stderr, "XCreateWindow(...)\n"); fprintf(stderr, "tint2: XCreateWindow(...)\n");
Window parent = XCreateWindow(server.display, Window parent = XCreateWindow(server.display,
panel->main_win, panel->main_win,
0, 0,
@@ -747,19 +743,19 @@ gboolean add_icon(Window win)
if (!panel->is_hidden) { if (!panel->is_hidden) {
if (systray_profile) if (systray_profile)
fprintf(stderr, "XMapRaised(server.display, traywin->parent)\n"); fprintf(stderr, "tint2: XMapRaised(server.display, traywin->parent)\n");
XMapRaised(server.display, traywin->parent); XMapRaised(server.display, traywin->parent);
} }
if (systray_profile) if (systray_profile)
fprintf(stderr, "[%f] %s:%d\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, "tint2: [%f] %s:%d\n", profiling_get_time(), __func__, __LINE__);
// Resize and redraw the systray // Resize and redraw the systray
if (systray_profile) if (systray_profile)
fprintf(stderr, fprintf(stderr,
BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n", BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__); __LINE__);
systray.area.resize_needed = TRUE; systray.area.resize_needed = TRUE;
panel->area.resize_needed = TRUE; panel->area.resize_needed = TRUE;
@@ -774,7 +770,7 @@ gboolean reparent_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -812,7 +808,7 @@ gboolean reparent_icon(TrayWindow *traywin)
e.xclient.data.l[3] = traywin->parent; e.xclient.data.l[3] = traywin->parent;
e.xclient.data.l[4] = 0; e.xclient.data.l[4] = 0;
if (systray_profile) if (systray_profile)
fprintf(stderr, "XSendEvent(server.display, traywin->win, False, NoEventMask, &e)\n"); fprintf(stderr, "tint2: XSendEvent(server.display, traywin->win, False, NoEventMask, &e)\n");
XSendEvent(server.display, traywin->win, False, NoEventMask, &e); XSendEvent(server.display, traywin->win, False, NoEventMask, &e);
} }
@@ -836,7 +832,7 @@ gboolean reparent_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -850,7 +846,7 @@ gboolean embed_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -863,62 +859,13 @@ gboolean embed_icon(TrayWindow *traywin)
error = FALSE; error = FALSE;
XErrorHandler old = XSetErrorHandler(window_error_handler); XErrorHandler old = XSetErrorHandler(window_error_handler);
if (0) {
Atom acttype;
int actfmt;
unsigned long nbitem, bytes;
unsigned long *data = 0;
int ret;
if (systray_profile)
fprintf(stderr,
"XGetWindowProperty(server.display, traywin->win, server.atom._XEMBED_INFO, 0, 2, False, "
"server.atom._XEMBED_INFO, &acttype, &actfmt, &nbitem, &bytes, &data)\n");
ret = XGetWindowProperty(server.display,
traywin->win,
server.atom._XEMBED_INFO,
0,
2,
False,
server.atom._XEMBED_INFO,
&acttype,
&actfmt,
&nbitem,
&bytes,
(unsigned char **)&data);
if (ret == Success) {
if (data) {
if (nbitem >= 2) {
int hide = ((data[1] & XEMBED_MAPPED) == 0);
if (hide) {
// In theory we have to check the embedding with this and remove icons that refuse embedding.
// In practice we have no idea when the other application processes the event and accepts the
// embed so we cannot check now without a race.
// Race can be triggered with PyGtk(2) apps.
// We could defer this for later (if we set PropertyChangeMask in XSelectInput we get notified)
// but for some reason it breaks transparency for Qt icons. So we don't.
// fprintf(stderr, RED "tint2: window refused embedding" RESET "\n");
// remove_icon(traywin);
// XFree(data);
// return FALSE;
}
}
XFree(data);
}
} else {
fprintf(stderr, RED "tint2 : xembed error" RESET "\n");
remove_icon(traywin);
return FALSE;
}
}
// Redirect rendering when using compositing // Redirect rendering when using compositing
if (systray_composited) { if (systray_composited) {
if (systray_profile) if (systray_profile)
fprintf(stderr, "XDamageCreate(server.display, traywin->parent, XDamageReportRawRectangles)\n"); fprintf(stderr, "tint2: XDamageCreate(server.display, traywin->parent, XDamageReportRawRectangles)\n");
traywin->damage = XDamageCreate(server.display, traywin->parent, XDamageReportRawRectangles); traywin->damage = XDamageCreate(server.display, traywin->parent, XDamageReportRawRectangles);
if (systray_profile) if (systray_profile)
fprintf(stderr, "XCompositeRedirectWindow(server.display, traywin->parent, CompositeRedirectManual)\n"); fprintf(stderr, "tint2: XCompositeRedirectWindow(server.display, traywin->parent, CompositeRedirectManual)\n");
XCompositeRedirectWindow(server.display, traywin->parent, CompositeRedirectManual); XCompositeRedirectWindow(server.display, traywin->parent, CompositeRedirectManual);
} }
@@ -926,16 +873,16 @@ gboolean embed_icon(TrayWindow *traywin)
// Make the icon visible // Make the icon visible
if (systray_profile) if (systray_profile)
fprintf(stderr, "XMapWindow(server.display, traywin->win)\n"); fprintf(stderr, "tint2: XMapWindow(server.display, traywin->win)\n");
XMapWindow(server.display, traywin->win); XMapWindow(server.display, traywin->win);
if (!panel->is_hidden) { if (!panel->is_hidden) {
if (systray_profile) if (systray_profile)
fprintf(stderr, "XMapRaised(server.display, traywin->parent)\n"); fprintf(stderr, "tint2: XMapRaised(server.display, traywin->parent)\n");
XMapRaised(server.display, traywin->parent); XMapRaised(server.display, traywin->parent);
} }
if (systray_profile) if (systray_profile)
fprintf(stderr, "XSync(server.display, False)\n"); fprintf(stderr, "tint2: XSync(server.display, False)\n");
XSync(server.display, False); XSync(server.display, False);
XSetErrorHandler(old); XSetErrorHandler(old);
if (error != FALSE) { if (error != FALSE) {
@@ -956,7 +903,7 @@ gboolean embed_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -970,7 +917,7 @@ void remove_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -978,7 +925,7 @@ void remove_icon(TrayWindow *traywin)
// remove from our list // remove from our list
systray.list_icons = g_slist_remove(systray.list_icons, traywin); systray.list_icons = g_slist_remove(systray.list_icons, traywin);
fprintf(stderr, YELLOW "remove_icon: %lu (%s)" RESET "\n", traywin->win, traywin->name); fprintf(stderr, YELLOW "tint2: remove_icon: %lu (%s)" RESET "\n", traywin->win, traywin->name);
XSelectInput(server.display, traywin->win, NoEventMask); XSelectInput(server.display, traywin->win, NoEventMask);
if (traywin->damage) if (traywin->damage)
@@ -1016,7 +963,7 @@ void remove_icon(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n", BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__); __LINE__);
systray.area.resize_needed = TRUE; systray.area.resize_needed = TRUE;
panel->area.resize_needed = TRUE; panel->area.resize_needed = TRUE;
@@ -1139,7 +1086,7 @@ void systray_reconfigure_event(TrayWindow *traywin, XEvent *e)
fprintf(stderr, fprintf(stderr,
BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n", BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__); __LINE__);
schedule_panel_redraw(); schedule_panel_redraw();
refresh_systray = TRUE; refresh_systray = TRUE;
@@ -1218,7 +1165,7 @@ void systray_resize_request_event(TrayWindow *traywin, XEvent *e)
fprintf(stderr, fprintf(stderr,
BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n", BLUE "[%f] %s:%d trigger resize & redraw" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__); __LINE__);
schedule_panel_redraw(); schedule_panel_redraw();
refresh_systray = TRUE; refresh_systray = TRUE;
@@ -1230,7 +1177,7 @@ void systray_destroy_event(TrayWindow *traywin)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1265,7 +1212,7 @@ void systray_render_icon_composited(void *t)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1283,7 +1230,7 @@ void systray_render_icon_composited(void *t)
fprintf(stderr, fprintf(stderr,
YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n", YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1303,7 +1250,7 @@ void systray_render_icon_composited(void *t)
fprintf(stderr, fprintf(stderr,
YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n", YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1332,7 +1279,7 @@ void systray_render_icon_composited(void *t)
} else if (traywin->depth == 32) { } else if (traywin->depth == 32) {
f = XRenderFindStandardFormat(server.display, PictStandardARGB32); f = XRenderFindStandardFormat(server.display, PictStandardARGB32);
} else { } else {
fprintf(stderr, RED "Strange tray icon found with depth: %d" RESET "\n", traywin->depth); fprintf(stderr, RED "tint2: Strange tray icon found with depth: %d" RESET "\n", traywin->depth);
XFreePixmap(server.display, tmp_pmap); XFreePixmap(server.display, tmp_pmap);
return; return;
} }
@@ -1434,7 +1381,7 @@ void systray_render_icon_composited(void *t)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1468,14 +1415,14 @@ void systray_render_icon(void *t)
{ {
TrayWindow *traywin = t; TrayWindow *traywin = t;
if (!traywin->reparented || !traywin->embedded) { if (!traywin->reparented || !traywin->embedded) {
// if (systray_profile) // if (systray_profile)
// fprintf(stderr, // fprintf(stderr,
// YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n", // YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n",
// profiling_get_time(), // profiling_get_time(),
// __FUNCTION__, // __func__,
// __LINE__, // __LINE__,
// traywin->win, // traywin->win,
// traywin->name); // traywin->name);
stop_timeout(traywin->render_timeout); stop_timeout(traywin->render_timeout);
traywin->render_timeout = traywin->render_timeout =
add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout); add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout);
@@ -1486,12 +1433,11 @@ void systray_render_icon(void *t)
fprintf(stderr, fprintf(stderr,
"[%f] %s:%d win = %lu (%s)\n", "[%f] %s:%d win = %lu (%s)\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
if (systray_composited) { if (systray_composited) {
XSync(server.display, False); XSync(server.display, False);
error = FALSE; error = FALSE;
@@ -1503,7 +1449,7 @@ void systray_render_icon(void *t)
Window root; Window root;
if (!XGetGeometry(server.display, traywin->win, &root, &xpos, &ypos, &width, &height, &border_width, &depth)) { if (!XGetGeometry(server.display, traywin->win, &root, &xpos, &ypos, &width, &height, &border_width, &depth)) {
stop_timeout(traywin->render_timeout); stop_timeout(traywin->render_timeout);
if (!traywin->resize_timeout) if (!traywin->render_timeout)
traywin->render_timeout = traywin->render_timeout =
add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout); add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout);
systray_render_icon_from_image(traywin); systray_render_icon_from_image(traywin);
@@ -1512,7 +1458,7 @@ void systray_render_icon(void *t)
} else { } else {
if (xpos != 0 || ypos != 0 || width != traywin->width || height != traywin->height) { if (xpos != 0 || ypos != 0 || width != traywin->width || height != traywin->height) {
stop_timeout(traywin->render_timeout); stop_timeout(traywin->render_timeout);
if (!traywin->resize_timeout) if (!traywin->render_timeout)
traywin->render_timeout = traywin->render_timeout =
add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout); add_timeout(min_refresh_period, 0, systray_render_icon, traywin, &traywin->render_timeout);
systray_render_icon_from_image(traywin); systray_render_icon_from_image(traywin);
@@ -1520,7 +1466,7 @@ void systray_render_icon(void *t)
fprintf(stderr, fprintf(stderr,
YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n", YELLOW "[%f] %s:%d win = %lu (%s) delaying rendering" RESET "\n",
profiling_get_time(), profiling_get_time(),
__FUNCTION__, __func__,
__LINE__, __LINE__,
traywin->win, traywin->win,
traywin->name); traywin->name);
@@ -1532,7 +1478,7 @@ void systray_render_icon(void *t)
} }
if (systray_profile) if (systray_profile)
fprintf(stderr, "rendering tray icon\n"); fprintf(stderr, "tint2: rendering tray icon\n");
if (systray_composited) { if (systray_composited) {
systray_render_icon_composited(traywin); systray_render_icon_composited(traywin);
@@ -1554,7 +1500,7 @@ void systray_render_icon(void *t)
void refresh_systray_icons() void refresh_systray_icons()
{ {
if (systray_profile) if (systray_profile)
fprintf(stderr, BLUE "[%f] %s:%d" RESET "\n", profiling_get_time(), __FUNCTION__, __LINE__); fprintf(stderr, BLUE "tint2: [%f] %s:%d" RESET "\n", profiling_get_time(), __func__, __LINE__);
TrayWindow *traywin; TrayWindow *traywin;
GSList *l; GSList *l;
for (l = systray.list_icons; l; l = l->next) { for (l = systray.list_icons; l; l = l->next) {

View File

@@ -96,7 +96,7 @@ gboolean systray_on_monitor(int i_monitor, int num_panels);
// many tray icon doesn't manage stop/restart of the systray manager // many tray icon doesn't manage stop/restart of the systray manager
void start_net(); void start_net();
void stop_net(); void stop_net();
void net_message(XClientMessageEvent *e); void handle_systray_event(XClientMessageEvent *e);
gboolean add_icon(Window id); gboolean add_icon(Window id);
gboolean reparent_icon(TrayWindow *traywin); gboolean reparent_icon(TrayWindow *traywin);

View File

@@ -40,6 +40,8 @@ GSList *urgent_list;
void task_dump_geometry(void *obj, int indent); void task_dump_geometry(void *obj, int indent);
int task_compute_desired_size(void *obj); int task_compute_desired_size(void *obj);
void task_refresh_thumbnail(Task *task);
void task_get_content_color(void *obj, Color *color);
char *task_get_tooltip(void *obj) char *task_get_tooltip(void *obj)
{ {
@@ -47,6 +49,17 @@ char *task_get_tooltip(void *obj)
return strdup(t->title); return strdup(t->title);
} }
cairo_surface_t *task_get_thumbnail(void *obj)
{
if (!panel_config.g_task.thumbnail_enabled)
return NULL;
Task *t = (Task *)obj;
if (!t->thumbnail)
task_refresh_thumbnail(t);
taskbar_start_thumbnail_timer(THUMB_MODE_TOOLTIP_WINDOW);
return t->thumbnail;
}
Task *add_task(Window win) Task *add_task(Window win)
{ {
if (!win) if (!win)
@@ -73,6 +86,7 @@ Task *add_task(Window win)
task_template.area.has_mouse_press_effect = panel_config.mouse_effects; task_template.area.has_mouse_press_effect = panel_config.mouse_effects;
task_template.area._dump_geometry = task_dump_geometry; task_template.area._dump_geometry = task_dump_geometry;
task_template.area._is_under_mouse = full_width_area_is_under_mouse; task_template.area._is_under_mouse = full_width_area_is_under_mouse;
task_template.area._get_content_color = task_get_content_color;
task_template.win = win; task_template.win = win;
task_template.desktop = get_window_desktop(win); task_template.desktop = get_window_desktop(win);
task_template.area.panel = &panels[monitor]; task_template.area.panel = &panels[monitor];
@@ -93,10 +107,6 @@ Task *add_task(Window win)
(int)win, (int)win,
task_template.title ? task_template.title : "null"); task_template.title ? task_template.title : "null");
// fprintf(stderr, "%s %d: win = %ld, task = %s\n", __FUNCTION__, __LINE__, win, task_template.title ?
// task_template.title : "??");
// fprintf(stderr, "new task %s win %u: desktop %d, monitor %d\n", new_task.title, win, new_task.desktop, monitor);
GPtrArray *task_buttons = g_ptr_array_new(); GPtrArray *task_buttons = g_ptr_array_new();
for (int j = 0; j < panels[monitor].num_desktops; j++) { for (int j = 0; j < panels[monitor].num_desktops; j++) {
if (task_template.desktop != ALL_DESKTOPS && task_template.desktop != j) if (task_template.desktop != ALL_DESKTOPS && task_template.desktop != j)
@@ -110,6 +120,7 @@ Task *add_task(Window win)
task_instance->area._dump_geometry = task_dump_geometry; task_instance->area._dump_geometry = task_dump_geometry;
task_instance->area._is_under_mouse = full_width_area_is_under_mouse; task_instance->area._is_under_mouse = full_width_area_is_under_mouse;
task_instance->area._compute_desired_size = task_compute_desired_size; task_instance->area._compute_desired_size = task_compute_desired_size;
task_instance->area._get_content_color = task_get_content_color;
task_instance->win = task_template.win; task_instance->win = task_template.win;
task_instance->desktop = task_template.desktop; task_instance->desktop = task_template.desktop;
task_instance->win_x = task_template.win_x; task_instance->win_x = task_template.win_x;
@@ -118,12 +129,16 @@ Task *add_task(Window win)
task_instance->win_h = task_template.win_h; task_instance->win_h = task_template.win_h;
task_instance->current_state = TASK_UNDEFINED; // to update the current state later in set_task_state... task_instance->current_state = TASK_UNDEFINED; // to update the current state later in set_task_state...
if (task_instance->desktop == ALL_DESKTOPS && server.desktop != j) { if (task_instance->desktop == ALL_DESKTOPS && server.desktop != j) {
// fprintf(stderr, "%s %d: win = %ld hiding task: another desktop\n", __FUNCTION__, __LINE__, win);
task_instance->area.on_screen = always_show_all_desktop_tasks; task_instance->area.on_screen = always_show_all_desktop_tasks;
} }
task_instance->title = task_template.title; task_instance->title = task_template.title;
if (panels[monitor].g_task.tooltip_enabled) if (panels[monitor].g_task.tooltip_enabled) {
task_instance->area._get_tooltip_text = task_get_tooltip; task_instance->area._get_tooltip_text = task_get_tooltip;
task_instance->area._get_tooltip_image = task_get_thumbnail;
}
task_instance->icon_color = task_template.icon_color;
task_instance->icon_color_hover = task_template.icon_color_hover;
task_instance->icon_color_press = task_template.icon_color_press;
for (int k = 0; k < TASK_STATE_COUNT; ++k) { for (int k = 0; k < TASK_STATE_COUNT; ++k) {
task_instance->icon[k] = task_template.icon[k]; task_instance->icon[k] = task_template.icon[k];
task_instance->icon_hover[k] = task_template.icon_hover[k]; task_instance->icon_hover[k] = task_template.icon_hover[k];
@@ -158,26 +173,10 @@ Task *add_task(Window win)
return (Task *)g_ptr_array_index(task_buttons, 0); return (Task *)g_ptr_array_index(task_buttons, 0);
} }
void remove_task(Task *task) void task_remove_icon(Task *task)
{ {
if (!task) if (!task)
return; return;
// fprintf(stderr, "%s %d: win = %ld, task = %s\n", __FUNCTION__, __LINE__, task->win, task->title ? task->title :
// "??");
if (taskbar_mode == MULTI_DESKTOP) {
Panel *panel = task->area.panel;
panel->area.resize_needed = 1;
}
Window win = task->win;
// free title and icon just for the first task
// even with task_on_all_desktop and with task_on_all_panel
// printf("remove_task %s %d\n", task->title, task->desktop);
if (task->title)
free(task->title);
for (int k = 0; k < TASK_STATE_COUNT; ++k) { for (int k = 0; k < TASK_STATE_COUNT; ++k) {
if (task->icon[k]) { if (task->icon[k]) {
imlib_context_set_image(task->icon[k]); imlib_context_set_image(task->icon[k]);
@@ -195,6 +194,27 @@ void remove_task(Task *task)
task->icon_press[k] = 0; task->icon_press[k] = 0;
} }
} }
}
void remove_task(Task *task)
{
if (!task)
return;
if (taskbar_mode == MULTI_DESKTOP) {
Panel *panel = task->area.panel;
panel->area.resize_needed = 1;
}
Window win = task->win;
// free title and icon just for the first task
// even with task_on_all_desktop and with task_on_all_panel
if (task->title)
free(task->title);
if (task->thumbnail)
cairo_surface_destroy(task->thumbnail);
task_remove_icon(task);
GPtrArray *task_buttons = g_hash_table_lookup(win_to_task, &win); GPtrArray *task_buttons = g_hash_table_lookup(win_to_task, &win);
for (int i = 0; i < task_buttons->len; ++i) { for (int i = 0; i < task_buttons->len; ++i) {
@@ -205,6 +225,8 @@ void remove_task(Task *task)
task_drag = 0; task_drag = 0;
if (g_slist_find(urgent_list, task2)) if (g_slist_find(urgent_list, task2))
del_urgent(task2); del_urgent(task2);
if (g_tooltip.area == &task2->area)
tooltip_hide(NULL);
remove_area((Area *)task2); remove_area((Area *)task2);
free(task2); free(task2);
} }
@@ -259,43 +281,31 @@ gboolean task_update_title(Task *task)
return TRUE; return TRUE;
} }
void task_update_icon(Task *task) Imlib_Image task_get_icon(Window win, int icon_size)
{ {
Panel *panel = task->area.panel;
if (!panel->g_task.has_icon)
return;
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
if (task->icon[k]) {
imlib_context_set_image(task->icon[k]);
imlib_free_image();
task->icon[k] = 0;
}
}
Imlib_Image img = NULL; Imlib_Image img = NULL;
if (!img) { if (!img) {
int i; int len;
gulong *data = server_get_property(task->win, server.atom._NET_WM_ICON, XA_CARDINAL, &i); gulong *data = server_get_property(win, server.atom._NET_WM_ICON, XA_CARDINAL, &len);
if (data) { if (data) {
if (len > 0) {
// get ARGB icon // get ARGB icon
int w, h; int w, h;
gulong *tmp_data; gulong *tmp_data = get_best_icon(data, get_icon_count(data, len), len, &w, &h, icon_size);
if (tmp_data) {
tmp_data = get_best_icon(data, get_icon_count(data, i), i, &w, &h, panel->g_task.icon_size1);
DATA32 icon_data[w * h]; DATA32 icon_data[w * h];
for (int j = 0; j < w * h; ++j) for (int j = 0; j < w * h; ++j)
icon_data[j] = tmp_data[j]; icon_data[j] = tmp_data[j];
img = imlib_create_image_using_copied_data(w, h, icon_data); img = imlib_create_image_using_copied_data(w, h, icon_data);
if (img) }
fprintf(stderr, "%s: Got %dx%d icon via _NET_WM_ICON for %s\n", __FUNCTION__, w, h, task->title ? task->title : "task"); }
XFree(data); XFree(data);
} }
} }
if (!img) { if (!img) {
XWMHints *hints = XGetWMHints(server.display, task->win); XWMHints *hints = XGetWMHints(server.display, win);
if (hints) { if (hints) {
if (hints->flags & IconPixmapHint && hints->icon_pixmap != 0) { if (hints->flags & IconPixmapHint && hints->icon_pixmap != 0) {
// get width, height and depth for the pixmap // get width, height and depth for the pixmap
@@ -307,8 +317,6 @@ void task_update_icon(Task *task)
XGetGeometry(server.display, hints->icon_pixmap, &root, &icon_x, &icon_y, &w, &h, &border_width, &bpp); XGetGeometry(server.display, hints->icon_pixmap, &root, &icon_x, &icon_y, &w, &h, &border_width, &bpp);
imlib_context_set_drawable(hints->icon_pixmap); imlib_context_set_drawable(hints->icon_pixmap);
img = imlib_create_image_from_drawable(hints->icon_mask, 0, 0, w, h, 0); img = imlib_create_image_from_drawable(hints->icon_mask, 0, 0, w, h, 0);
if (img)
fprintf(stderr, "%s: Got %dx%d pixmap icon via WM_HINTS for %s\n", __FUNCTION__, w, h, task->title ? task->title : "task");
} }
XFree(hints); XFree(hints);
} }
@@ -317,9 +325,46 @@ void task_update_icon(Task *task)
if (img == NULL) { if (img == NULL) {
imlib_context_set_image(default_icon); imlib_context_set_image(default_icon);
img = imlib_clone_image(); img = imlib_clone_image();
fprintf(stderr, "%s: Using default icon for %s\n", __FUNCTION__, task->title ? task->title : "task");
} }
return img;
}
void task_set_icon_color(Task *task, Imlib_Image icon)
{
get_image_mean_color(icon, &task->icon_color);
if (panel_config.mouse_effects) {
task->icon_color_hover = task->icon_color;
adjust_color(&task->icon_color_hover,
panel_config.mouse_over_alpha,
panel_config.mouse_over_saturation,
panel_config.mouse_over_brightness);
task->icon_color_press = task->icon_color;
adjust_color(&task->icon_color_press,
panel_config.mouse_pressed_alpha,
panel_config.mouse_pressed_saturation,
panel_config.mouse_pressed_brightness);
}
}
void task_update_icon(Task *task)
{
Panel *panel = task->area.panel;
if (!panel->g_task.has_icon) {
if (panel_config.g_task.has_content_tint) {
Imlib_Image img = task_get_icon(task->win, panel->g_task.icon_size1);
task_set_icon_color(task, img);
imlib_context_set_image(img);
imlib_free_image();
}
return;
}
task_remove_icon(task);
Imlib_Image img = task_get_icon(task->win, panel->g_task.icon_size1);
task_set_icon_color(task, img);
// transform icons // transform icons
imlib_context_set_image(img); imlib_context_set_image(img);
imlib_image_set_has_alpha(1); imlib_image_set_has_alpha(1);
@@ -333,20 +378,10 @@ void task_update_icon(Task *task)
task->icon_width = imlib_image_get_width(); task->icon_width = imlib_image_get_width();
task->icon_height = imlib_image_get_height(); task->icon_height = imlib_image_get_height();
for (int k = 0; k < TASK_STATE_COUNT; ++k) { for (int k = 0; k < TASK_STATE_COUNT; ++k) {
imlib_context_set_image(orig_image); task->icon[k] = adjust_icon(orig_image,
task->icon[k] = imlib_clone_image(); panel->g_task.alpha[k],
imlib_context_set_image(task->icon[k]); panel->g_task.saturation[k],
DATA32 *data32; panel->g_task.brightness[k] != 0);
if (panel->g_task.alpha[k] != 100 || panel->g_task.saturation[k] != 0 || panel->g_task.brightness[k] != 0) {
data32 = imlib_image_get_data();
adjust_asb(data32,
task->icon_width,
task->icon_height,
panel->g_task.alpha[k] / 100.0,
panel->g_task.saturation[k] / 100.0,
panel->g_task.brightness[k] / 100.0);
imlib_image_put_back_data(data32);
}
if (panel_config.mouse_effects) { if (panel_config.mouse_effects) {
task->icon_hover[k] = adjust_icon(task->icon[k], task->icon_hover[k] = adjust_icon(task->icon[k],
panel_config.mouse_over_alpha, panel_config.mouse_over_alpha,
@@ -364,9 +399,12 @@ void task_update_icon(Task *task)
GPtrArray *task_buttons = get_task_buttons(task->win); GPtrArray *task_buttons = get_task_buttons(task->win);
if (task_buttons) { if (task_buttons) {
for (int i = 0; i < task_buttons->len; ++i) { for (int i = 0; i < task_buttons->len; ++i) {
Task *task2 = g_ptr_array_index(task_buttons, i); Task *task2 = (Task *)g_ptr_array_index(task_buttons, i);
task2->icon_width = task->icon_width; task2->icon_width = task->icon_width;
task2->icon_height = task->icon_height; task2->icon_height = task->icon_height;
task2->icon_color = task->icon_color;
task2->icon_color_hover = task->icon_color_hover;
task2->icon_color_press = task->icon_color_press;
for (int k = 0; k < TASK_STATE_COUNT; ++k) { for (int k = 0; k < TASK_STATE_COUNT; ++k) {
task2->icon[k] = task->icon[k]; task2->icon[k] = task->icon[k];
task2->icon_hover[k] = task->icon_hover[k]; task2->icon_hover[k] = task->icon_hover[k];
@@ -454,7 +492,7 @@ void task_dump_geometry(void *obj, int indent)
Panel *panel = (Panel *)task->area.panel; Panel *panel = (Panel *)task->area.panel;
fprintf(stderr, fprintf(stderr,
"%*sText: x = %d, y = %d, w = %d, h = %d, align = %s, text = %s\n", "tint2: %*sText: x = %d, y = %d, w = %d, h = %d, align = %s, text = %s\n",
indent, indent,
"", "",
(int)panel->g_task.text_posx, (int)panel->g_task.text_posx,
@@ -464,7 +502,7 @@ void task_dump_geometry(void *obj, int indent)
panel->g_task.centered ? "center" : "left", panel->g_task.centered ? "center" : "left",
task->title); task->title);
fprintf(stderr, fprintf(stderr,
"%*sIcon: x = %d, y = %d, w = h = %d\n", "tint2: %*sIcon: x = %d, y = %d, w = h = %d\n",
indent, indent,
"", "",
task->_icon_x, task->_icon_x,
@@ -472,6 +510,24 @@ void task_dump_geometry(void *obj, int indent)
panel->g_task.icon_size1); panel->g_task.icon_size1);
} }
void task_get_content_color(void *obj, Color *color)
{
Task *task = (Task *)obj;
Color *content_color = NULL;
if (panel_config.mouse_effects) {
if (task->area.mouse_state == MOUSE_OVER)
content_color = &task->icon_color_hover;
else if (task->area.mouse_state == MOUSE_DOWN)
content_color = &task->icon_color_press;
else
content_color = &task->icon_color;
} else {
content_color = &task->icon_color;
}
if (content_color)
*color = *content_color;
}
int task_compute_desired_size(void *obj) int task_compute_desired_size(void *obj)
{ {
Task *task = (Task *)obj; Task *task = (Task *)obj;
@@ -485,7 +541,11 @@ void on_change_task(void *obj)
Task *task = (Task *)obj; Task *task = (Task *)obj;
Panel *panel = (Panel *)task->area.panel; Panel *panel = (Panel *)task->area.panel;
long value[] = {panel->posx + task->area.posx, panel->posy + task->area.posy, task->area.width, task->area.height}; if (task->area.on_screen) {
long value[] = {panel->posx + task->area.posx,
panel->posy + task->area.posy,
task->area.width,
task->area.height};
XChangeProperty(server.display, XChangeProperty(server.display,
task->win, task->win,
server.atom._NET_WM_ICON_GEOMETRY, server.atom._NET_WM_ICON_GEOMETRY,
@@ -494,6 +554,9 @@ void on_change_task(void *obj)
PropModeReplace, PropModeReplace,
(unsigned char *)value, (unsigned char *)value,
4); 4);
} else {
XDeleteProperty(server.display, task->win, server.atom._NET_WM_ICON_GEOMETRY);
}
} }
Task *find_active_task(Task *current_task) Task *find_active_task(Task *current_task)
@@ -572,7 +635,6 @@ void reset_active_task()
} }
Window w1 = get_active_window(); Window w1 = get_active_window();
// printf("Change active task %ld\n", w1);
if (w1) { if (w1) {
if (!get_task_buttons(w1)) { if (!get_task_buttons(w1)) {
@@ -584,11 +646,49 @@ void reset_active_task()
} }
} }
void task_refresh_thumbnail(Task *task)
{
if (!panel_config.g_task.thumbnail_enabled)
return;
if (task->current_state == TASK_ICONIFIED)
return;
double now = get_time();
if (now - task->thumbnail_last_update < 0.1)
return;
if (debug_thumbnails)
fprintf(stderr, "tint2: thumbnail for window: %s" RESET "\n", task->title ? task->title : "");
cairo_surface_t *thumbnail = get_window_thumbnail(task->win, panel_config.g_task.thumbnail_width);
if (!thumbnail)
return;
if (task->thumbnail)
cairo_surface_destroy(task->thumbnail);
task->thumbnail = thumbnail;
task->thumbnail_last_update = get_time();
if (debug_thumbnails)
fprintf(stderr,
YELLOW "tint2: %s took %f ms (window: %s)" RESET "\n",
__func__,
1000 * (task->thumbnail_last_update - now),
task->title ? task->title : "");
if (g_tooltip.mapped && (g_tooltip.area == &task->area)) {
tooltip_update_contents_for(&task->area);
tooltip_update();
}
}
void set_task_state(Task *task, TaskState state) void set_task_state(Task *task, TaskState state)
{ {
if (!task || state == TASK_UNDEFINED || state >= TASK_STATE_COUNT) if (!task || state == TASK_UNDEFINED || state >= TASK_STATE_COUNT)
return; return;
if (!task->thumbnail)
task_refresh_thumbnail(task);
if (state == TASK_ACTIVE) {
// For active windows, we get the thumbnail twice with a small delay in between.
// This is because they sometimes redraw their windows slowly.
taskbar_start_thumbnail_timer(THUMB_MODE_ACTIVE_WINDOW);
}
if (state == TASK_ACTIVE && task->current_state != state) { if (state == TASK_ACTIVE && task->current_state != state) {
clock_gettime(CLOCK_MONOTONIC, &task->last_activation_time); clock_gettime(CLOCK_MONOTONIC, &task->last_activation_time);
if (taskbar_sort_method == TASKBAR_SORT_LRU || taskbar_sort_method == TASKBAR_SORT_MRU) { if (taskbar_sort_method == TASKBAR_SORT_LRU || taskbar_sort_method == TASKBAR_SORT_MRU) {
@@ -603,7 +703,7 @@ void set_task_state(Task *task, TaskState state)
} }
} }
if (task->current_state != state || hide_task_diff_monitor) { if (task->current_state != state || hide_task_diff_monitor || hide_task_diff_desktop) {
GPtrArray *task_buttons = get_task_buttons(task->win); GPtrArray *task_buttons = get_task_buttons(task->win);
if (task_buttons) { if (task_buttons) {
for (int i = 0; i < task_buttons->len; ++i) { for (int i = 0; i < task_buttons->len; ++i) {
@@ -627,6 +727,10 @@ void set_task_state(Task *task, TaskState state)
hide = TRUE; hide = TRUE;
} }
} }
if (hide_task_diff_desktop) {
if (taskbar->desktop != server.desktop)
hide = TRUE;
}
if (get_window_monitor(task->win) != ((Panel *)task->area.panel)->monitor && if (get_window_monitor(task->win) != ((Panel *)task->area.panel)->monitor &&
(hide_task_diff_monitor || num_panels > 1)) { (hide_task_diff_monitor || num_panels > 1)) {
hide = TRUE; hide = TRUE;
@@ -694,3 +798,75 @@ void del_urgent(Task *task)
urgent_timeout = NULL; urgent_timeout = NULL;
} }
} }
void task_handle_mouse_event(Task *task, MouseAction action)
{
if (!task)
return;
switch (action) {
case NONE:
break;
case CLOSE:
close_window(task->win);
break;
case TOGGLE:
activate_window(task->win);
break;
case ICONIFY:
XIconifyWindow(server.display, task->win, server.screen);
break;
case TOGGLE_ICONIFY:
if (active_task && task->win == active_task->win)
XIconifyWindow(server.display, task->win, server.screen);
else
activate_window(task->win);
break;
case SHADE:
toggle_window_shade(task->win);
break;
case MAXIMIZE_RESTORE:
toggle_window_maximized(task->win);
break;
case MAXIMIZE:
toggle_window_maximized(task->win);
break;
case RESTORE:
toggle_window_maximized(task->win);
break;
case DESKTOP_LEFT: {
if (task->desktop == 0)
break;
int desktop = task->desktop - 1;
change_window_desktop(task->win, desktop);
if (desktop == server.desktop)
activate_window(task->win);
break;
}
case DESKTOP_RIGHT: {
if (task->desktop == server.num_desktops)
break;
int desktop = task->desktop + 1;
change_window_desktop(task->win, desktop);
if (desktop == server.desktop)
activate_window(task->win);
break;
}
case NEXT_TASK: {
Task *task1 = next_task(find_active_task(task));
activate_window(task1->win);
} break;
case PREV_TASK: {
Task *task1 = prev_task(find_active_task(task));
activate_window(task1->win);
}
}
}
void task_update_desktop(Task *task)
{
Window win = task->win;
remove_task(task);
task = add_task(win);
reset_active_task();
schedule_panel_redraw();
}

View File

@@ -10,6 +10,7 @@
#include <X11/Xlib.h> #include <X11/Xlib.h>
#include <pango/pangocairo.h> #include <pango/pangocairo.h>
#include <Imlib2.h> #include <Imlib2.h>
#include "common.h" #include "common.h"
#include "timer.h" #include "timer.h"
@@ -41,10 +42,13 @@ typedef struct GlobalTask {
// starting position for text ~ task_padding + task_border + icon_size // starting position for text ~ task_padding + task_border + icon_size
double text_posx, text_height; double text_posx, text_height;
gboolean has_font; gboolean has_font;
gboolean has_content_tint;
PangoFontDescription *font_desc; PangoFontDescription *font_desc;
Color font[TASK_STATE_COUNT]; Color font[TASK_STATE_COUNT];
int config_font_mask; int config_font_mask;
gboolean tooltip_enabled; gboolean tooltip_enabled;
gboolean thumbnail_enabled;
int thumbnail_width;
} GlobalTask; } GlobalTask;
// Stores information about a task. // Stores information about a task.
@@ -60,6 +64,9 @@ typedef struct Task {
Imlib_Image icon_press[TASK_STATE_COUNT]; Imlib_Image icon_press[TASK_STATE_COUNT];
unsigned int icon_width; unsigned int icon_width;
unsigned int icon_height; unsigned int icon_height;
Color icon_color;
Color icon_color_hover;
Color icon_color_press;
char *title; char *title;
int urgent_tick; int urgent_tick;
// These may not be up-to-date // These may not be up-to-date
@@ -73,6 +80,8 @@ typedef struct Task {
double _text_posy; double _text_posy;
int _icon_x; int _icon_x;
int _icon_y; int _icon_y;
cairo_surface_t *thumbnail;
double thumbnail_last_update;
} Task; } Task;
extern timeout *urgent_timeout; extern timeout *urgent_timeout;
@@ -85,9 +94,12 @@ void draw_task(void *obj, cairo_t *c);
void on_change_task(void *obj); void on_change_task(void *obj);
void task_update_icon(Task *task); void task_update_icon(Task *task);
void task_update_desktop(Task *task);
gboolean task_update_title(Task *task); gboolean task_update_title(Task *task);
void reset_active_task(); void reset_active_task();
void set_task_state(Task *task, TaskState state); void set_task_state(Task *task, TaskState state);
void task_handle_mouse_event(Task *task, MouseAction action);
void task_refresh_thumbnail(Task *task);
// Given a pointer to the task that is currently under the mouse (current_task), // Given a pointer to the task that is currently under the mouse (current_task),
// returns a pointer to the Task for the active window on the same taskbar. // returns a pointer to the Task for the active window on the same taskbar.

View File

@@ -32,6 +32,7 @@
#include "window.h" #include "window.h"
#include "panel.h" #include "panel.h"
#include "strnatcmp.h" #include "strnatcmp.h"
#include "tooltip.h"
GHashTable *win_to_task; GHashTable *win_to_task;
@@ -39,12 +40,19 @@ Task *active_task;
Task *task_drag; Task *task_drag;
gboolean taskbar_enabled; gboolean taskbar_enabled;
gboolean taskbar_distribute_size; gboolean taskbar_distribute_size;
gboolean hide_task_diff_desktop;
gboolean hide_inactive_tasks; gboolean hide_inactive_tasks;
gboolean hide_task_diff_monitor; gboolean hide_task_diff_monitor;
gboolean hide_taskbar_if_empty; gboolean hide_taskbar_if_empty;
gboolean always_show_all_desktop_tasks; gboolean always_show_all_desktop_tasks;
TaskbarSortMethod taskbar_sort_method; TaskbarSortMethod taskbar_sort_method;
Alignment taskbar_alignment; Alignment taskbar_alignment;
static timeout *thumbnail_update_timer_all;
static timeout *thumbnail_update_timer_active;
static timeout *thumbnail_update_timer_tooltip;
static GList *taskbar_task_orderings = NULL;
static GList *taskbar_thumbnail_jobs_done = NULL;
void taskbar_init_fonts(); void taskbar_init_fonts();
int taskbar_compute_desired_size(void *obj); int taskbar_compute_desired_size(void *obj);
@@ -52,6 +60,8 @@ int taskbar_compute_desired_size(void *obj);
// Removes the task with &win = key. The other args are ignored. // Removes the task with &win = key. The other args are ignored.
void taskbar_remove_task(Window *win); void taskbar_remove_task(Window *win);
void taskbar_update_thumbnails(void *arg);
guint win_hash(gconstpointer key) guint win_hash(gconstpointer key)
{ {
return *((const Window *)key); return *((const Window *)key);
@@ -74,17 +84,61 @@ void default_taskbar()
urgent_list = NULL; urgent_list = NULL;
taskbar_enabled = FALSE; taskbar_enabled = FALSE;
taskbar_distribute_size = FALSE; taskbar_distribute_size = FALSE;
hide_task_diff_desktop = FALSE;
hide_inactive_tasks = FALSE; hide_inactive_tasks = FALSE;
hide_task_diff_monitor = FALSE; hide_task_diff_monitor = FALSE;
hide_taskbar_if_empty = FALSE; hide_taskbar_if_empty = FALSE;
always_show_all_desktop_tasks = FALSE; always_show_all_desktop_tasks = FALSE;
thumbnail_update_timer_all = NULL;
thumbnail_update_timer_active = NULL;
thumbnail_update_timer_tooltip = NULL;
taskbar_thumbnail_jobs_done = NULL;
taskbar_sort_method = TASKBAR_NOSORT; taskbar_sort_method = TASKBAR_NOSORT;
taskbar_alignment = ALIGN_LEFT; taskbar_alignment = ALIGN_LEFT;
default_taskbarname(); default_taskbarname();
} }
void taskbar_clear_orderings()
{
if (!taskbar_task_orderings)
return;
for (GList *order = taskbar_task_orderings; order; order = order->next) {
g_list_free_full((GList *)order->data, free);
}
g_list_free(taskbar_task_orderings);
taskbar_task_orderings = NULL;
}
void taskbar_save_orderings()
{
taskbar_clear_orderings();
taskbar_task_orderings = NULL;
for (int i = 0; i < num_panels; i++) {
Panel *panel = &panels[i];
for (int j = 0; j < panel->num_desktops; j++) {
Taskbar *taskbar = &panel->taskbar[j];
GList *task_order = NULL;
for (GList *c = (taskbar->area.children && taskbarname_enabled) ? taskbar->area.children->next
: taskbar->area.children;
c;
c = c->next) {
Task *t = (Task *)c->data;
Window *window = calloc(1, sizeof(Window));
*window = t->win;
task_order = g_list_append(task_order, window);
}
taskbar_task_orderings = g_list_append(taskbar_task_orderings, task_order);
}
}
}
void cleanup_taskbar() void cleanup_taskbar()
{ {
stop_timeout(thumbnail_update_timer_all);
stop_timeout(thumbnail_update_timer_active);
stop_timeout(thumbnail_update_timer_tooltip);
g_list_free(taskbar_thumbnail_jobs_done);
taskbar_save_orderings();
if (win_to_task) { if (win_to_task) {
while (g_hash_table_size(win_to_task)) { while (g_hash_table_size(win_to_task)) {
GHashTableIter iter; GHashTableIter iter;
@@ -134,6 +188,9 @@ void init_taskbar()
panel_config.g_task.has_text = panel_config.g_task.has_icon = 1; panel_config.g_task.has_text = panel_config.g_task.has_icon = 1;
} }
if (panel_config.g_task.thumbnail_width < 8)
panel_config.g_task.thumbnail_width = 210;
if (!win_to_task) if (!win_to_task)
win_to_task = g_hash_table_new_full(win_hash, win_compare, free, free_ptr_array); win_to_task = g_hash_table_new_full(win_hash, win_compare, free, free_ptr_array);
@@ -238,9 +295,9 @@ void init_taskbar_panel(void *p)
if ((panel->g_task.config_background_mask & (1 << TASK_URGENT)) == 0) if ((panel->g_task.config_background_mask & (1 << TASK_URGENT)) == 0)
panel->g_task.background[TASK_URGENT] = panel->g_task.background[TASK_ACTIVE]; panel->g_task.background[TASK_URGENT] = panel->g_task.background[TASK_ACTIVE];
if (!panel->g_task.maximum_width) if (!panel->g_task.maximum_width || !panel_horizontal)
panel->g_task.maximum_width = server.monitors[panel->monitor].width; panel->g_task.maximum_width = server.monitors[panel->monitor].width;
if (!panel->g_task.maximum_height) if (!panel->g_task.maximum_height || panel_horizontal)
panel->g_task.maximum_height = server.monitors[panel->monitor].height; panel->g_task.maximum_height = server.monitors[panel->monitor].height;
if (panel_horizontal) { if (panel_horizontal) {
@@ -265,7 +322,8 @@ void init_taskbar_panel(void *p)
if (!panel->g_task.background[j]) if (!panel->g_task.background[j])
panel->g_task.background[j] = &g_array_index(backgrounds, Background, 0); panel->g_task.background[j] = &g_array_index(backgrounds, Background, 0);
if (panel->g_task.background[j]->border.radius > panel->g_task.area.height / 2) { if (panel->g_task.background[j]->border.radius > panel->g_task.area.height / 2) {
printf("task%sbackground_id has a too large rounded value. Please fix your tint2rc\n", fprintf(stderr,
"tint2: task%sbackground_id has a too large rounded value. Please fix your tint2rc\n",
j == 0 ? "_" : j == 1 ? "_active_" : j == 2 ? "_iconified_" : "_urgent_"); j == 0 ? "_" : j == 1 ? "_active_" : j == 2 ? "_iconified_" : "_urgent_");
g_array_append_val(backgrounds, *panel->g_task.background[j]); g_array_append_val(backgrounds, *panel->g_task.background[j]);
panel->g_task.background[j] = &g_array_index(backgrounds, Background, backgrounds->len - 1); panel->g_task.background[j] = &g_array_index(backgrounds, Background, backgrounds->len - 1);
@@ -288,7 +346,8 @@ void init_taskbar_panel(void *p)
FALSE); FALSE);
panel->g_task.text_posx = left_bg_border_width(panel->g_task.background[0]) + panel->g_task.area.paddingxlr; panel->g_task.text_posx = left_bg_border_width(panel->g_task.background[0]) + panel->g_task.area.paddingxlr;
panel->g_task.text_height = panel->g_task.area.height - (2 * panel->g_task.area.paddingy); panel->g_task.text_height =
panel->g_task.area.height - (2 * panel->g_task.area.paddingy) - top_bottom_border_width(&panel->g_task.area);
if (panel->g_task.has_icon) { if (panel->g_task.has_icon) {
panel->g_task.icon_size1 = MIN(MIN(panel->g_task.maximum_width, panel->g_task.maximum_height), panel->g_task.icon_size1 = MIN(MIN(panel->g_task.maximum_width, panel->g_task.maximum_height),
MIN(panel->g_task.area.width, panel->g_task.area.height)) - MIN(panel->g_task.area.width, panel->g_task.area.height)) -
@@ -296,16 +355,6 @@ void init_taskbar_panel(void *p)
top_bottom_border_width(&panel->g_task.area)); top_bottom_border_width(&panel->g_task.area));
panel->g_task.text_posx += panel->g_task.icon_size1 + panel->g_task.area.paddingx; panel->g_task.text_posx += panel->g_task.icon_size1 + panel->g_task.area.paddingx;
panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2; panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2;
if (0)
printf("task: icon_size = %d, textx = %f, texth = %f, icony = %d, w = %d, h = %d, maxw = %d, maxh = %d\n",
panel->g_task.icon_size1,
panel->g_task.text_posx,
panel->g_task.text_height,
panel->g_task.icon_posy,
panel->g_task.area.width,
panel->g_task.area.height,
panel->g_task.maximum_width,
panel->g_task.maximum_height);
} }
Taskbar *taskbar; Taskbar *taskbar;
@@ -324,9 +373,23 @@ void init_taskbar_panel(void *p)
free_area_gradient_instances(&taskbar->area); free_area_gradient_instances(&taskbar->area);
instantiate_area_gradients(&taskbar->area); instantiate_area_gradients(&taskbar->area);
} }
} }
init_taskbarname_panel(panel); init_taskbarname_panel(panel);
taskbar_start_thumbnail_timer(THUMB_MODE_ALL);
}
void taskbar_start_thumbnail_timer(ThumbnailUpdateMode mode)
{
if (!panel_config.g_task.thumbnail_enabled)
return;
if (debug_thumbnails)
fprintf(stderr, BLUE "tint2: taskbar_start_thumbnail_timer %s" RESET "\n", mode == THUMB_MODE_ACTIVE_WINDOW ? "active" : mode == THUMB_MODE_TOOLTIP_WINDOW ? "tooltip" : "all");
change_timeout(mode == THUMB_MODE_ALL ? &thumbnail_update_timer_all :
mode == THUMB_MODE_ACTIVE_WINDOW ? &thumbnail_update_timer_active : &thumbnail_update_timer_tooltip,
mode == THUMB_MODE_TOOLTIP_WINDOW ? 1000 : 500,
mode == THUMB_MODE_ALL ? 10 * 1000 : 0,
taskbar_update_thumbnails,
(void *)(long)mode);
} }
void taskbar_init_fonts() void taskbar_init_fonts()
@@ -389,14 +452,67 @@ GPtrArray *get_task_buttons(Window win)
return NULL; return NULL;
} }
static Window *sort_windows = NULL;
int compare_windows(const void *a, const void *b)
{
if (!sort_windows)
return 0;
int ia = *(int *)a;
int ib = *(int *)b;
Window wina = sort_windows[ia];
Window winb = sort_windows[ib];
for (GList *order = taskbar_task_orderings; order; order = order->next) {
int posa = -1;
int posb = -1;
int pos = 0;
for (GList *item = (GList *)order->data; item; item = item->next, pos++) {
Window win = *(Window *)item->data;
if (win == wina)
posa = pos;
if (win == winb)
posb = pos;
}
if (posa >= 0 && posb >= 0) {
return posa - posb;
}
}
return ia - ib;
}
void sort_win_list(Window *windows, int count)
{
int *indices = (int *)calloc(count, sizeof(int));
for (int i = 0; i < count; i++)
indices[i] = i;
sort_windows = windows;
qsort(indices, count, sizeof(int), compare_windows);
Window *result = (Window *)calloc(count, sizeof(Window));
for (int i = 0; i < count; i++)
result[i] = windows[indices[i]];
memcpy(windows, result, count * sizeof(Window));
free(result);
free(indices);
sort_windows = NULL;
}
void taskbar_refresh_tasklist() void taskbar_refresh_tasklist()
{ {
if (!taskbar_enabled) if (!taskbar_enabled)
return; return;
// fprintf(stderr, "%s %d:\n", __FUNCTION__, __LINE__);
int num_results; int num_results;
Window *win = server_get_property(server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results); Window *win = server_get_property(server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results);
Window *sorted = (Window *)calloc(num_results, sizeof(Window));
memcpy(sorted, win, num_results * sizeof(Window));
if (taskbar_task_orderings) {
sort_win_list(sorted, num_results);
taskbar_clear_orderings();
}
if (!win) if (!win)
return; return;
@@ -404,7 +520,7 @@ void taskbar_refresh_tasklist()
for (GList *it = win_list; it; it = it->next) { for (GList *it = win_list; it; it = it->next) {
int i; int i;
for (i = 0; i < num_results; i++) for (i = 0; i < num_results; i++)
if (*((Window *)it->data) == win[i]) if (*((Window *)it->data) == sorted[i])
break; break;
if (i == num_results) if (i == num_results)
taskbar_remove_task(it->data); taskbar_remove_task(it->data);
@@ -413,10 +529,11 @@ void taskbar_refresh_tasklist()
// Add any new // Add any new
for (int i = 0; i < num_results; i++) for (int i = 0; i < num_results; i++)
if (!get_task(win[i])) if (!get_task(sorted[i]))
add_task(win[i]); add_task(sorted[i]);
XFree(win); XFree(win);
free(sorted);
} }
int taskbar_compute_desired_size(void *obj) int taskbar_compute_desired_size(void *obj)
@@ -440,7 +557,6 @@ gboolean resize_taskbar(void *obj)
Taskbar *taskbar = (Taskbar *)obj; Taskbar *taskbar = (Taskbar *)obj;
Panel *panel = (Panel *)taskbar->area.panel; Panel *panel = (Panel *)taskbar->area.panel;
// printf("resize_taskbar %d %d\n", taskbar->area.posx, taskbar->area.posy);
if (panel_horizontal) { if (panel_horizontal) {
relayout_with_constraint(&taskbar->area, panel->g_task.maximum_width); relayout_with_constraint(&taskbar->area, panel->g_task.maximum_width);
@@ -535,6 +651,15 @@ void set_taskbar_state(Taskbar *taskbar, TaskbarState state)
for (; l; l = l->next) for (; l; l = l->next)
schedule_redraw((Area *)l->data); schedule_redraw((Area *)l->data);
} }
if (taskbar_mode == MULTI_DESKTOP && hide_task_diff_desktop) {
GList *l = taskbar->area.children;
if (taskbarname_enabled)
l = l->next;
for (; l; l = l->next) {
Task *task = (Task *)l->data;
set_task_state(task, task->current_state);
}
}
} }
schedule_panel_redraw(); schedule_panel_redraw();
} }
@@ -674,3 +799,63 @@ void sort_taskbar_for_win(Window win)
} }
} }
} }
void update_minimized_icon_positions(void *p)
{
Panel *panel = (Panel *)p;
for (int i = 0; i < panel->num_desktops; i++) {
Taskbar *taskbar = &panel->taskbar[i];
if (!taskbar->area.on_screen)
continue;
for (GList *c = taskbar->area.children; c; c = c->next) {
Area *area = (Area *)c->data;
if (area->_on_change_layout)
area->_on_change_layout(area);
}
}
}
void taskbar_update_thumbnails(void *arg)
{
if (!panel_config.g_task.thumbnail_enabled)
return;
ThumbnailUpdateMode mode = (ThumbnailUpdateMode)(long)arg;
if (debug_thumbnails)
fprintf(stderr, BLUE "tint2: taskbar_update_thumbnails %s" RESET "\n", mode == THUMB_MODE_ACTIVE_WINDOW ? "active" : mode == THUMB_MODE_TOOLTIP_WINDOW ? "tooltip" : "all");
double start_time = get_time();
for (int i = 0; i < num_panels; i++) {
Panel *panel = &panels[i];
for (int j = 0; j < panel->num_desktops; j++) {
Taskbar *taskbar = &panel->taskbar[j];
for (GList *c = (taskbar->area.children && taskbarname_enabled) ? taskbar->area.children->next
: taskbar->area.children;
c;
c = c->next) {
Task *t = (Task *)c->data;
if ((mode == THUMB_MODE_ALL && t->current_state == TASK_ACTIVE && !g_list_find(taskbar_thumbnail_jobs_done, t)) || (mode == THUMB_MODE_ACTIVE_WINDOW && t->current_state == TASK_ACTIVE) ||
(mode == THUMB_MODE_TOOLTIP_WINDOW && g_tooltip.mapped && g_tooltip.area == &t->area)) {
task_refresh_thumbnail(t);
if (mode == THUMB_MODE_ALL)
taskbar_thumbnail_jobs_done = g_list_append(taskbar_thumbnail_jobs_done, t);
if (t->thumbnail && mode == THUMB_MODE_TOOLTIP_WINDOW) {
taskbar_start_thumbnail_timer(THUMB_MODE_TOOLTIP_WINDOW);
}
}
if (mode == THUMB_MODE_ALL) {
double now = get_time();
if (now - start_time > 0.030) {
change_timeout(&thumbnail_update_timer_all, 50, 10 * 1000, taskbar_update_thumbnails, arg);
return;
}
}
}
}
}
if (mode == THUMB_MODE_ALL) {
if (taskbar_thumbnail_jobs_done) {
g_list_free(taskbar_thumbnail_jobs_done);
taskbar_thumbnail_jobs_done = NULL;
change_timeout(&thumbnail_update_timer_all, 10 * 1000, 10 * 1000, taskbar_update_thumbnails, arg);
}
}
}

View File

@@ -25,6 +25,12 @@ typedef enum TaskbarSortMethod {
TASKBAR_SORT_MRU, TASKBAR_SORT_MRU,
} TaskbarSortMethod; } TaskbarSortMethod;
typedef enum ThumbnailUpdateMode {
THUMB_MODE_ACTIVE_WINDOW = 0,
THUMB_MODE_TOOLTIP_WINDOW,
THUMB_MODE_ALL
} ThumbnailUpdateMode;
typedef struct { typedef struct {
Area area; Area area;
gchar *name; gchar *name;
@@ -49,6 +55,7 @@ typedef struct GlobalTaskbar {
extern gboolean taskbar_enabled; extern gboolean taskbar_enabled;
extern gboolean taskbar_distribute_size; extern gboolean taskbar_distribute_size;
extern gboolean hide_task_diff_desktop;
extern gboolean hide_inactive_tasks; extern gboolean hide_inactive_tasks;
extern gboolean hide_task_diff_monitor; extern gboolean hide_task_diff_monitor;
extern gboolean hide_taskbar_if_empty; extern gboolean hide_taskbar_if_empty;
@@ -71,6 +78,7 @@ void init_taskbar_panel(void *p);
gboolean resize_taskbar(void *obj); gboolean resize_taskbar(void *obj);
void taskbar_default_font_changed(); void taskbar_default_font_changed();
void taskbar_start_thumbnail_timer(ThumbnailUpdateMode mode);
// Reloads the entire list of tasks from the window manager and recreates the task buttons. // Reloads the entire list of tasks from the window manager and recreates the task buttons.
void taskbar_refresh_tasklist(); void taskbar_refresh_tasklist();
@@ -88,6 +96,8 @@ void set_taskbar_state(Taskbar *taskbar, TaskbarState state);
// Updates the visibility of all taskbars // Updates the visibility of all taskbars
void update_all_taskbars_visibility(); void update_all_taskbars_visibility();
void update_minimized_icon_positions(void *p);
// Sorts the taskbar(s) on which the window is present. // Sorts the taskbar(s) on which the window is present.
void sort_taskbar_for_win(Window win); void sort_taskbar_for_win(Window win);

View File

@@ -214,3 +214,35 @@ void draw_taskbarname(void *obj, cairo_t *c)
g_object_unref(layout); g_object_unref(layout);
} }
void update_desktop_names()
{
if (!taskbarname_enabled)
return;
GSList *list = get_desktop_names();
for (int i = 0; i < num_panels; i++) {
int j;
GSList *l;
for (j = 0, l = list; j < panels[i].num_desktops; j++) {
gchar *name;
if (l) {
name = g_strdup(l->data);
l = l->next;
} else {
name = g_strdup_printf("%d", j + 1);
}
Taskbar *taskbar = &panels[i].taskbar[j];
if (strcmp(name, taskbar->bar_name.name) != 0) {
g_free(taskbar->bar_name.name);
taskbar->bar_name.name = name;
taskbar->bar_name.area.resize_needed = 1;
} else {
g_free(name);
}
}
}
for (GSList *l = list; l; l = l->next)
g_free(l->data);
g_slist_free(list);
schedule_panel_redraw();
}

View File

@@ -23,4 +23,6 @@ gboolean resize_taskbarname(void *obj);
void taskbarname_default_font_changed(); void taskbarname_default_font_changed();
void update_desktop_names();
#endif #endif

2168
src/tint.c

File diff suppressed because it is too large Load Diff

View File

@@ -22,6 +22,7 @@ include_directories( ../util
set(SOURCES ../util/common.c set(SOURCES ../util/common.c
../util/strnatcmp.c ../util/strnatcmp.c
../util/cache.c ../util/cache.c
../util/timer.c
../config.c ../config.c
../server.c ../server.c
../launcher/apps-common.c ../launcher/apps-common.c
@@ -63,7 +64,7 @@ target_link_libraries( tint2conf ${X11_T2C_LIBRARIES}
add_definitions( -DINSTALL_PREFIX=\"${CMAKE_INSTALL_PREFIX}\" ) add_definitions( -DINSTALL_PREFIX=\"${CMAKE_INSTALL_PREFIX}\" )
add_definitions( -DLOCALEDIR=\"${CMAKE_INSTALL_FULL_LOCALEDIR}\" ) add_definitions( -DLOCALEDIR=\"${CMAKE_INSTALL_FULL_LOCALEDIR}\" )
add_definitions( -DGETTEXT_PACKAGE=\"tint2conf\" ) add_definitions( -DGETTEXT_PACKAGE=\"tint2conf\" )
set_target_properties( tint2conf PROPERTIES COMPILE_FLAGS "-Wall -pthread -std=c11" ) set_target_properties( tint2conf PROPERTIES COMPILE_FLAGS "-Wall -pthread -std=c99" )
set_target_properties( tint2conf PROPERTIES LINK_FLAGS "-pthread" ) set_target_properties( tint2conf PROPERTIES LINK_FLAGS "-pthread" )
add_subdirectory(po) add_subdirectory(po)
@@ -71,3 +72,4 @@ add_subdirectory(po)
install( TARGETS tint2conf DESTINATION bin ) install( TARGETS tint2conf DESTINATION bin )
install( FILES tint2conf.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps ) install( FILES tint2conf.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps )
install( FILES tint2conf.desktop DESTINATION ${CMAKE_INSTALL_DATADIR}/applications ) install( FILES tint2conf.desktop DESTINATION ${CMAKE_INSTALL_DATADIR}/applications )
install( FILES tint2conf.xml DESTINATION ${CMAKE_INSTALL_DATADIR}/mime/packages )

View File

@@ -6,7 +6,7 @@ GtkWidget *current_background, *background_fill_color, *background_border_color,
*background_fill_color_over, *background_border_color_over, *background_gradient_over, *background_fill_color_press, *background_fill_color_over, *background_border_color_over, *background_gradient_over, *background_fill_color_press,
*background_border_color_press, *background_gradient_press, *background_border_width, *background_corner_radius, *background_border_color_press, *background_gradient_press, *background_border_width, *background_corner_radius,
*background_border_sides_top, *background_border_sides_bottom, *background_border_sides_left, *background_border_sides_top, *background_border_sides_bottom, *background_border_sides_left,
*background_border_sides_right; *background_border_sides_right, *background_border_content_tint_weight, *background_fill_content_tint_weight;
GtkWidget *create_background_combo(const char *label) GtkWidget *create_background_combo(const char *label)
{ {
@@ -115,7 +115,9 @@ void create_background(GtkWidget *parent)
GTK_TYPE_BOOL, GTK_TYPE_BOOL,
GTK_TYPE_BOOL, GTK_TYPE_BOOL,
GTK_TYPE_BOOL, GTK_TYPE_BOOL,
GTK_TYPE_BOOL); GTK_TYPE_BOOL,
GTK_TYPE_DOUBLE,
GTK_TYPE_DOUBLE);
GtkWidget *table, *label, *button; GtkWidget *table, *label, *button;
int row, col; int row, col;
@@ -175,6 +177,19 @@ void create_background(GtkWidget *parent)
col++; col++;
gtk_tooltips_set_tip(tooltips, background_fill_color, _("The fill color of the current background"), NULL); gtk_tooltips_set_tip(tooltips, background_fill_color, _("The fill color of the current background"), NULL);
row++, col = 2;
label = gtk_label_new(_("Fill tint"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_widget_show(label);
gtk_table_attach(GTK_TABLE(table), label, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++;
background_fill_content_tint_weight = gtk_spin_button_new_with_range(0, 100, 1);
gtk_widget_show(background_fill_content_tint_weight);
gtk_table_attach(GTK_TABLE(table), background_fill_content_tint_weight, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++;
gtk_tooltips_set_tip(tooltips, background_fill_content_tint_weight, _("How much the border color should be tinted with the content color"), NULL);
row++, col = 2; row++, col = 2;
label = gtk_label_new(_("Border color")); label = gtk_label_new(_("Border color"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0); gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
@@ -189,6 +204,19 @@ void create_background(GtkWidget *parent)
col++; col++;
gtk_tooltips_set_tip(tooltips, background_border_color, _("The border color of the current background"), NULL); gtk_tooltips_set_tip(tooltips, background_border_color, _("The border color of the current background"), NULL);
row++, col = 2;
label = gtk_label_new(_("Border tint"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_widget_show(label);
gtk_table_attach(GTK_TABLE(table), label, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++;
background_border_content_tint_weight = gtk_spin_button_new_with_range(0, 100, 1);
gtk_widget_show(background_border_content_tint_weight);
gtk_table_attach(GTK_TABLE(table), background_border_content_tint_weight, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++;
gtk_tooltips_set_tip(tooltips, background_border_content_tint_weight, _("How much the border color should be tinted with the content color"), NULL);
row++, col = 2; row++, col = 2;
label = gtk_label_new(_("Gradient")); label = gtk_label_new(_("Gradient"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0); gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
@@ -366,6 +394,8 @@ void create_background(GtkWidget *parent)
g_signal_connect(G_OBJECT(background_border_sides_bottom), "toggled", G_CALLBACK(background_update), NULL); g_signal_connect(G_OBJECT(background_border_sides_bottom), "toggled", G_CALLBACK(background_update), NULL);
g_signal_connect(G_OBJECT(background_border_sides_left), "toggled", G_CALLBACK(background_update), NULL); g_signal_connect(G_OBJECT(background_border_sides_left), "toggled", G_CALLBACK(background_update), NULL);
g_signal_connect(G_OBJECT(background_border_sides_right), "toggled", G_CALLBACK(background_update), NULL); g_signal_connect(G_OBJECT(background_border_sides_right), "toggled", G_CALLBACK(background_update), NULL);
g_signal_connect(G_OBJECT(background_border_content_tint_weight), "value-changed", G_CALLBACK(background_update), NULL);
g_signal_connect(G_OBJECT(background_fill_content_tint_weight), "value-changed", G_CALLBACK(background_update), NULL);
change_paragraph(parent); change_paragraph(parent);
} }
@@ -750,6 +780,9 @@ void background_update(GtkWidget *widget, gpointer data)
r = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_corner_radius)); r = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_corner_radius));
b = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_border_width)); b = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_border_width));
double fill_weight = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_fill_content_tint_weight));
double border_weight = gtk_spin_button_get_value(GTK_SPIN_BUTTON(background_border_content_tint_weight));
gboolean sideTop = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_top)); gboolean sideTop = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_top));
gboolean sideBottom = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_bottom)); gboolean sideBottom = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_bottom));
gboolean sideLeft = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_left)); gboolean sideLeft = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(background_border_sides_left));
@@ -836,6 +869,10 @@ void background_update(GtkWidget *widget, gpointer data)
sideLeft, sideLeft,
bgColBorderSidesRight, bgColBorderSidesRight,
sideRight, sideRight,
bgColFillWeight,
fill_weight,
bgColBorderWeight,
border_weight,
-1); -1);
background_update_image(index); background_update_image(index);
} }
@@ -862,6 +899,8 @@ void current_background_changed(GtkWidget *widget, gpointer data)
gtk_widget_set_sensitive(background_border_sides_left, index > 0); gtk_widget_set_sensitive(background_border_sides_left, index > 0);
gtk_widget_set_sensitive(background_border_sides_right, index > 0); gtk_widget_set_sensitive(background_border_sides_right, index > 0);
gtk_widget_set_sensitive(background_corner_radius, index > 0); gtk_widget_set_sensitive(background_corner_radius, index > 0);
gtk_widget_set_sensitive(background_border_content_tint_weight, index > 0);
gtk_widget_set_sensitive(background_fill_content_tint_weight, index > 0);
background_updates_disabled = TRUE; background_updates_disabled = TRUE;
@@ -875,6 +914,9 @@ void current_background_changed(GtkWidget *widget, gpointer data)
int r; int r;
int b; int b;
double fill_weight;
double border_weight;
gboolean sideTop; gboolean sideTop;
gboolean sideBottom; gboolean sideBottom;
gboolean sideLeft; gboolean sideLeft;
@@ -938,6 +980,10 @@ void current_background_changed(GtkWidget *widget, gpointer data)
&sideLeft, &sideLeft,
bgColBorderSidesRight, bgColBorderSidesRight,
&sideRight, &sideRight,
bgColFillWeight,
&fill_weight,
bgColBorderWeight,
&border_weight,
-1); -1);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(background_border_sides_top), sideTop); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(background_border_sides_top), sideTop);
@@ -968,6 +1014,9 @@ void current_background_changed(GtkWidget *widget, gpointer data)
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_border_width), b); gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_border_width), b);
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_corner_radius), r); gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_corner_radius), r);
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_fill_content_tint_weight), fill_weight);
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_border_content_tint_weight), border_weight);
g_boxed_free(GDK_TYPE_COLOR, fillColor); g_boxed_free(GDK_TYPE_COLOR, fillColor);
g_boxed_free(GDK_TYPE_COLOR, borderColor); g_boxed_free(GDK_TYPE_COLOR, borderColor);
g_boxed_free(GDK_TYPE_COLOR, fillColorOver); g_boxed_free(GDK_TYPE_COLOR, fillColorOver);

View File

@@ -5,7 +5,7 @@
GtkListStore *gradient_ids, *gradient_stop_ids; GtkListStore *gradient_ids, *gradient_stop_ids;
GList *gradients; GList *gradients;
GtkWidget *current_gradient, *gradient_combo_type, *gradient_start_color, *gradient_end_color, *current_gradient_stop, GtkWidget *current_gradient, *gradient_combo_type, *gradient_start_color, *gradient_end_color, *current_gradient_stop,
*gradient_stop_color, *gradient_stop_offset; *gradient_stop_color, *gradient_stop_offset;
int gradient_index_safe(int index) int gradient_index_safe(int index)
{ {
@@ -176,12 +176,12 @@ void create_gradient(GtkWidget *parent)
label = gtk_label_new(_("Position")); label = gtk_label_new(_("Position"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0); gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_widget_show(label); gtk_widget_show(label);
gtk_table_attach(GTK_TABLE(table), label, col, col+1, row, row+1, GTK_FILL, 0, 0, 0); gtk_table_attach(GTK_TABLE(table), label, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++; col++;
gradient_stop_offset = gtk_spin_button_new_with_range(0, 100, 1); gradient_stop_offset = gtk_spin_button_new_with_range(0, 100, 1);
gtk_widget_show(gradient_stop_offset); gtk_widget_show(gradient_stop_offset);
gtk_table_attach(GTK_TABLE(table), gradient_stop_offset, col, col+1, row, row+1, GTK_FILL, 0, 0, 0); gtk_table_attach(GTK_TABLE(table), gradient_stop_offset, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
col++; col++;
g_signal_connect(G_OBJECT(current_gradient), "changed", G_CALLBACK(current_gradient_changed), NULL); g_signal_connect(G_OBJECT(current_gradient), "changed", G_CALLBACK(current_gradient_changed), NULL);
@@ -205,10 +205,14 @@ void gradient_create_new(GradientConfigType t)
gradients = g_list_append(gradients, g); gradients = g_list_append(gradients, g);
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_append(gradient_ids, &iter); gtk_list_store_append(gradient_ids, &iter);
gtk_list_store_set(gradient_ids, &iter, gtk_list_store_set(gradient_ids,
grColPixbuf, NULL, &iter,
grColId, &index, grColPixbuf,
grColText, index == 0 ? _("None") : "", NULL,
grColId,
&index,
grColText,
index == 0 ? _("None") : "",
-1); -1);
gradient_update_image(index); gradient_update_image(index);
@@ -233,7 +237,7 @@ void gradient_duplicate(GtkWidget *widget, gpointer data)
return; return;
} }
GradientConfig *g_old = (GradientConfig*)g_list_nth(gradients, (guint)old_index)->data; GradientConfig *g_old = (GradientConfig *)g_list_nth(gradients, (guint)old_index)->data;
int index = get_model_length(GTK_TREE_MODEL(gradient_ids)); int index = get_model_length(GTK_TREE_MODEL(gradient_ids));
@@ -245,11 +249,7 @@ void gradient_duplicate(GtkWidget *widget, gpointer data)
gradients = g_list_append(gradients, g); gradients = g_list_append(gradients, g);
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_append(gradient_ids, &iter); gtk_list_store_append(gradient_ids, &iter);
gtk_list_store_set(gradient_ids, &iter, gtk_list_store_set(gradient_ids, &iter, grColPixbuf, NULL, grColId, &index, grColText, "", -1);
grColPixbuf, NULL,
grColId, &index,
grColText, "",
-1);
gradient_update_image(index); gradient_update_image(index);
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient), index); gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient), index);
@@ -289,8 +289,14 @@ void gradient_draw(cairo_t *c, GradientConfig *g, int w, int h, gboolean preserv
gpat = cairo_pattern_create_linear(0, 0, 0, h); gpat = cairo_pattern_create_linear(0, 0, 0, h);
else if (g->type == GRADIENT_CONFIG_HORIZONTAL) else if (g->type == GRADIENT_CONFIG_HORIZONTAL)
gpat = cairo_pattern_create_linear(0, 0, w, 0); gpat = cairo_pattern_create_linear(0, 0, w, 0);
else gpat = cairo_pattern_create_radial(w/2, h/2, 0, w/2, h/2, sqrt(w * w + h * h) / 2); else
cairo_pattern_add_color_stop_rgba(gpat, 0.0, g->start_color.color.rgb[0], g->start_color.color.rgb[1], g->start_color.color.rgb[2], g->start_color.color.alpha); gpat = cairo_pattern_create_radial(w / 2, h / 2, 0, w / 2, h / 2, sqrt(w * w + h * h) / 2);
cairo_pattern_add_color_stop_rgba(gpat,
0.0,
g->start_color.color.rgb[0],
g->start_color.color.rgb[1],
g->start_color.color.rgb[2],
g->start_color.color.alpha);
for (GList *l = g->extra_color_stops; l; l = l->next) { for (GList *l = g->extra_color_stops; l; l = l->next) {
GradientConfigColorStop *stop = (GradientConfigColorStop *)l->data; GradientConfigColorStop *stop = (GradientConfigColorStop *)l->data;
cairo_pattern_add_color_stop_rgba(gpat, cairo_pattern_add_color_stop_rgba(gpat,
@@ -300,7 +306,12 @@ void gradient_draw(cairo_t *c, GradientConfig *g, int w, int h, gboolean preserv
stop->color.rgb[2], stop->color.rgb[2],
stop->color.alpha); stop->color.alpha);
} }
cairo_pattern_add_color_stop_rgba(gpat, 1.0, g->end_color.color.rgb[0], g->end_color.color.rgb[1], g->end_color.color.rgb[2], g->end_color.color.alpha); cairo_pattern_add_color_stop_rgba(gpat,
1.0,
g->end_color.color.rgb[0],
g->end_color.color.rgb[1],
g->end_color.color.rgb[2],
g->end_color.color.alpha);
cairo_set_source(c, gpat); cairo_set_source(c, gpat);
cairo_rectangle(c, 0, 0, w, h); cairo_rectangle(c, 0, 0, w, h);
if (preserve) if (preserve)
@@ -336,9 +347,7 @@ void gradient_update_image(int index)
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_ids), &iter, path); gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_ids), &iter, path);
gtk_tree_path_free(path); gtk_tree_path_free(path);
gtk_list_store_set(gradient_ids, &iter, gtk_list_store_set(gradient_ids, &iter, grColPixbuf, pixbuf, -1);
grColPixbuf, pixbuf,
-1);
if (pixbuf) if (pixbuf)
g_object_unref(pixbuf); g_object_unref(pixbuf);
} }
@@ -381,11 +390,7 @@ void gradient_update(GtkWidget *widget, gpointer data)
gdkColor2CairoColor(color, &g->end_color.color.rgb[0], &g->end_color.color.rgb[1], &g->end_color.color.rgb[2]); gdkColor2CairoColor(color, &g->end_color.color.rgb[0], &g->end_color.color.rgb[1], &g->end_color.color.rgb[2]);
g->end_color.color.alpha = opacity / 100.0; g->end_color.color.alpha = opacity / 100.0;
gtk_list_store_set(gradient_ids, &iter, gtk_list_store_set(gradient_ids, &iter, grColPixbuf, NULL, grColId, &index, grColText, "", -1);
grColPixbuf, NULL,
grColId, &index,
grColText, "",
-1);
gradient_update_image(index); gradient_update_image(index);
background_update_for_gradient(index); background_update_for_gradient(index);
} }
@@ -428,9 +433,7 @@ void current_gradient_changed(GtkWidget *widget, gpointer data)
for (GList *l = g->extra_color_stops; l; l = l->next, stop_index++) { for (GList *l = g->extra_color_stops; l; l = l->next, stop_index++) {
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_append(gradient_stop_ids, &iter); gtk_list_store_append(gradient_stop_ids, &iter);
gtk_list_store_set(gradient_stop_ids, &iter, gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, NULL, -1);
grStopColPixbuf, NULL,
-1);
gradient_stop_update_image(stop_index); gradient_stop_update_image(stop_index);
} }
if (old_stop_index >= 0 && old_stop_index < stop_index) if (old_stop_index >= 0 && old_stop_index < stop_index)
@@ -468,9 +471,7 @@ void gradient_stop_create_new()
g->extra_color_stops = g_list_append(g->extra_color_stops, stop); g->extra_color_stops = g_list_append(g->extra_color_stops, stop);
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_append(gradient_stop_ids, &iter); gtk_list_store_append(gradient_stop_ids, &iter);
gtk_list_store_set(gradient_stop_ids, &iter, gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, NULL, -1);
grStopColPixbuf, NULL,
-1);
int stop_index = g_list_length(g->extra_color_stops) - 1; int stop_index = g_list_length(g->extra_color_stops) - 1;
gradient_stop_update_image(stop_index); gradient_stop_update_image(stop_index);
@@ -545,9 +546,7 @@ void gradient_stop_update_image(int index)
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_stop_ids), &iter, path); gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_stop_ids), &iter, path);
gtk_tree_path_free(path); gtk_tree_path_free(path);
gtk_list_store_set(gradient_stop_ids, &iter, gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, pixbuf, -1);
grStopColPixbuf, pixbuf,
-1);
if (pixbuf) if (pixbuf)
g_object_unref(pixbuf); g_object_unref(pixbuf);
} }

View File

@@ -74,7 +74,7 @@ gboolean endswith(const char *str, const char *suffix)
// Returns TRUE if the theme file is in ~/.config. // Returns TRUE if the theme file is in ~/.config.
gboolean theme_is_editable(const char *filepath) gboolean theme_is_editable(const char *filepath)
{ {
return startswith(filepath, g_get_user_config_dir()); return access(filepath, W_OK) == 0;
} }
// Returns TRUE if the theme file is ~/.config/tint2/tint2rc. // Returns TRUE if the theme file is ~/.config/tint2/tint2rc.
@@ -119,7 +119,7 @@ gchar *import_no_overwrite(const char *filepath)
if (!g_file_test(newpath, G_FILE_TEST_EXISTS)) { if (!g_file_test(newpath, G_FILE_TEST_EXISTS)) {
copy_file(filepath, newpath); copy_file(filepath, newpath);
theme_list_append(newpath); theme_list_append(newpath);
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} }
return newpath; return newpath;
@@ -138,7 +138,7 @@ void import_with_overwrite(const char *filepath, const char *newpath)
if (theme_is_editable(newpath)) { if (theme_is_editable(newpath)) {
if (!theme_existed) { if (!theme_existed) {
theme_list_append(newpath); theme_list_append(newpath);
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} else { } else {
int unused = system("killall -SIGUSR1 tint2 || pkill -SIGUSR1 -x tint2"); int unused = system("killall -SIGUSR1 tint2 || pkill -SIGUSR1 -x tint2");
(void)unused; (void)unused;
@@ -158,7 +158,7 @@ static void menuImportFile();
static void menuSaveAs(); static void menuSaveAs();
static void menuDelete(); static void menuDelete();
static void menuReset(); static void menuReset();
static void edit_theme(); static gboolean edit_theme(gpointer ignored);
static void make_selected_theme_default(); static void make_selected_theme_default();
static void menuAbout(); static void menuAbout();
static gboolean view_onPopupMenu(GtkWidget *treeview, gpointer userdata); static gboolean view_onPopupMenu(GtkWidget *treeview, gpointer userdata);
@@ -227,7 +227,7 @@ int main(int argc, char **argv)
{ {
gchar *tint2_config_dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL); gchar *tint2_config_dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL);
if (!g_file_test(tint2_config_dir, G_FILE_TEST_IS_DIR)) if (!g_file_test(tint2_config_dir, G_FILE_TEST_IS_DIR))
g_mkdir(tint2_config_dir, 0700); g_mkdir_with_parents(tint2_config_dir, 0700);
g_free(tint2_config_dir); g_free(tint2_config_dir);
} }
@@ -250,13 +250,43 @@ int main(int argc, char **argv)
// Menubar and toolbar entries // Menubar and toolbar entries
GtkActionEntry entries[] = GtkActionEntry entries[] =
{{"ThemeMenu", NULL, _("_Theme"), NULL, NULL, NULL}, {{"ThemeMenu", NULL, _("_Theme"), NULL, NULL, NULL},
{"ThemeImportFile", GTK_STOCK_ADD, _("_Import theme..."), "<Control>N", _("Import theme(s) from file system"), G_CALLBACK(menuImportFile)}, {"ThemeImportFile",
{"ThemeSaveAs", GTK_STOCK_SAVE_AS, _("_Save as..."), NULL, _("Save the theme with a new name"), G_CALLBACK(menuSaveAs)}, GTK_STOCK_ADD,
_("_Import theme..."),
"<Control>N",
_("Import theme(s) from file system"),
G_CALLBACK(menuImportFile)},
{"ThemeSaveAs",
GTK_STOCK_SAVE_AS,
_("_Save as..."),
NULL,
_("Save the theme with a new name"),
G_CALLBACK(menuSaveAs)},
{"ThemeDelete", GTK_STOCK_DELETE, _("_Delete"), NULL, _("Delete the selected theme"), G_CALLBACK(menuDelete)}, {"ThemeDelete", GTK_STOCK_DELETE, _("_Delete"), NULL, _("Delete the selected theme"), G_CALLBACK(menuDelete)},
{"ThemeReset", GTK_STOCK_REVERT_TO_SAVED, _("_Reset"), NULL, _("Reset the selected theme to default"), G_CALLBACK(menuReset)}, {"ThemeReset",
{"ThemeEdit", GTK_STOCK_PROPERTIES, _("_Edit theme..."), NULL, _("Edit the selected theme"), G_CALLBACK(edit_theme)}, GTK_STOCK_REVERT_TO_SAVED,
{"ThemeMakeDefault", GTK_STOCK_APPLY, _("_Make default"), NULL, _("Replace the default theme with the selected one"), G_CALLBACK(make_selected_theme_default)}, _("_Reset"),
{"ThemeRefresh", GTK_STOCK_REFRESH, _("Refresh"), NULL, _("Redraw the selected theme"), G_CALLBACK(refresh_current_theme)}, NULL,
_("Reset the selected theme to default"),
G_CALLBACK(menuReset)},
{"ThemeEdit",
GTK_STOCK_PROPERTIES,
_("_Edit theme..."),
NULL,
_("Edit the selected theme"),
G_CALLBACK(edit_theme)},
{"ThemeMakeDefault",
GTK_STOCK_APPLY,
_("_Make default"),
NULL,
_("Replace the default theme with the selected one"),
G_CALLBACK(make_selected_theme_default)},
{"ThemeRefresh",
GTK_STOCK_REFRESH,
_("Refresh"),
NULL,
_("Redraw the selected theme"),
G_CALLBACK(refresh_current_theme)},
{"RefreshAll", GTK_STOCK_REFRESH, _("Refresh all"), NULL, _("Redraw all themes"), G_CALLBACK(reload_all_themes)}, {"RefreshAll", GTK_STOCK_REFRESH, _("Refresh all"), NULL, _("Redraw all themes"), G_CALLBACK(reload_all_themes)},
{"Quit", GTK_STOCK_QUIT, _("_Quit"), "<control>Q", _("Quit"), G_CALLBACK(gtk_main_quit)}, {"Quit", GTK_STOCK_QUIT, _("_Quit"), "<control>Q", _("Quit"), G_CALLBACK(gtk_main_quit)},
{"HelpMenu", NULL, _("_Help"), NULL, NULL, NULL}, {"HelpMenu", NULL, _("_Help"), NULL, NULL, NULL},
@@ -309,7 +339,11 @@ int main(int argc, char **argv)
argc--, argv++; argc--, argv++;
if (argc > 0) { if (argc > 0) {
load_specific_themes(argv, argc); load_specific_themes(argv, argc);
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)edit_theme, NULL); g_timeout_add(SNAPSHOT_TICK, edit_theme, NULL);
} else if (getenv("TINT2_CONFIG")) {
char *themes[2] = {getenv("TINT2_CONFIG"), NULL};
load_specific_themes(themes, 1);
g_timeout_add(SNAPSHOT_TICK, edit_theme, NULL);
} }
gtk_widget_show_all(g_window); gtk_widget_show_all(g_window);
@@ -409,7 +443,8 @@ gchar *get_selected_theme_or_warn()
return filepath; return filepath;
} }
// For the selected theme, shows save dialog (with overwrite confirmation) and copies to the selected file with overwrite. // For the selected theme, shows save dialog (with overwrite confirmation) and copies to the selected file with
// overwrite.
// Shows error box if no theme is selected. // Shows error box if no theme is selected.
static void menuSaveAs() static void menuSaveAs()
{ {
@@ -643,11 +678,11 @@ void select_theme(const char *given_path)
} }
// Edits the selected theme. If it is read-only, it copies first to ~. // Edits the selected theme. If it is read-only, it copies first to ~.
static void edit_theme() static gboolean edit_theme(gpointer ignored)
{ {
gchar *filepath = get_selected_theme_or_warn(); gchar *filepath = get_selected_theme_or_warn();
if (!filepath) if (!filepath)
return; return FALSE;
gboolean editable = theme_is_editable(filepath); gboolean editable = theme_is_editable(filepath);
if (!editable) { if (!editable) {
@@ -665,6 +700,8 @@ static void edit_theme()
g_free(filepath); g_free(filepath);
destroy_please_wait(); destroy_please_wait();
return FALSE;
} }
static void make_selected_theme_default() static void make_selected_theme_default()
@@ -680,7 +717,8 @@ static void make_selected_theme_default()
return; return;
} }
GtkWidget *w = gtk_message_dialog_new(GTK_WINDOW(g_window), GtkWidget *w =
gtk_message_dialog_new(GTK_WINDOW(g_window),
GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_QUESTION, GTK_MESSAGE_QUESTION,
GTK_BUTTONS_YES_NO, GTK_BUTTONS_YES_NO,
@@ -702,7 +740,7 @@ static void make_selected_theme_default()
static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data) static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data)
{ {
edit_theme(); edit_theme(NULL);
} }
// ====== Theme load/reload ====== // ====== Theme load/reload ======
@@ -834,7 +872,7 @@ static void load_all_themes()
have_iter = gtk_tree_model_iter_next(model, &iter); have_iter = gtk_tree_model_iter_next(model, &iter);
} }
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} }
} }
@@ -865,7 +903,7 @@ static void reload_all_themes()
have_iter = gtk_tree_model_iter_next(model, &iter); have_iter = gtk_tree_model_iter_next(model, &iter);
} }
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} }
} }
@@ -878,7 +916,7 @@ static void load_specific_themes(char **paths, int count)
// Load configs // Load configs
const char *file_name = paths[0]; const char *file_name = paths[0];
paths++, count--; paths++, count--;
if (g_file_test(file_name, G_FILE_TEST_IS_REGULAR) || g_file_test(file_name, G_FILE_TEST_IS_SYMLINK) ) { if (g_file_test(file_name, G_FILE_TEST_IS_REGULAR) || g_file_test(file_name, G_FILE_TEST_IS_SYMLINK)) {
theme_list_append(file_name); theme_list_append(file_name);
if (!found_themes) { if (!found_themes) {
select_theme(file_name); select_theme(file_name);
@@ -899,7 +937,7 @@ static void load_specific_themes(char **paths, int count)
have_iter = gtk_tree_model_iter_next(model, &iter); have_iter = gtk_tree_model_iter_next(model, &iter);
} }
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} }
} }
@@ -910,7 +948,7 @@ void refresh_current_theme()
GtkTreeModel *model; GtkTreeModel *model;
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) { if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
gtk_list_store_set(theme_list_store, &iter, COL_SNAPSHOT, NULL, -1); gtk_list_store_set(theme_list_store, &iter, COL_SNAPSHOT, NULL, -1);
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
} }
} }
@@ -925,7 +963,7 @@ void refresh_theme(const char *given_path)
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &filepath, -1); gtk_tree_model_get(model, &iter, COL_THEME_FILE, &filepath, -1);
if (g_str_equal(filepath, given_path)) { if (g_str_equal(filepath, given_path)) {
gtk_list_store_set(theme_list_store, &iter, COL_SNAPSHOT, NULL, -1); gtk_list_store_set(theme_list_store, &iter, COL_SNAPSHOT, NULL, -1);
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL); g_timeout_add(SNAPSHOT_TICK, update_snapshot, NULL);
g_free(filepath); g_free(filepath);
break; break;
} }

View File

@@ -15,7 +15,7 @@
#endif #endif
#define SNAPSHOT_TICK 190 #define SNAPSHOT_TICK 190
gboolean update_snapshot(); gboolean update_snapshot(gpointer ignored);
void menuApply(); void menuApply();
void refresh_current_theme(); void refresh_current_theme();
extern GtkWidget *g_window; extern GtkWidget *g_window;

View File

@@ -72,9 +72,9 @@ static inline u32 H(u32 x, u32 y, u32 z)
return x ^ y ^ z; return x ^ y ^ z;
} }
#define ROUND1(a,b,c,d,k,s) (a = lshift(a + F(b,c,d) + k, s)) #define ROUND1(a, b, c, d, k, s) (a = lshift(a + F(b, c, d) + k, s))
#define ROUND2(a,b,c,d,k,s) (a = lshift(a + G(b,c,d) + k + (u32)0x5A827999,s)) #define ROUND2(a, b, c, d, k, s) (a = lshift(a + G(b, c, d) + k + (u32)0x5A827999, s))
#define ROUND3(a,b,c,d,k,s) (a = lshift(a + H(b,c,d) + k + (u32)0x6ED9EBA1,s)) #define ROUND3(a, b, c, d, k, s) (a = lshift(a + H(b, c, d) + k + (u32)0x6ED9EBA1, s))
/* XXX: this stuff can be optimized */ /* XXX: this stuff can be optimized */
static inline void le32_to_cpu_array(u32 *buf, unsigned int words) static inline void le32_to_cpu_array(u32 *buf, unsigned int words)
@@ -119,7 +119,7 @@ static void md4_transform(u32 *hash, u32 const *in)
ROUND1(c, d, a, b, in[14], 11); ROUND1(c, d, a, b, in[14], 11);
ROUND1(b, c, d, a, in[15], 19); ROUND1(b, c, d, a, in[15], 19);
ROUND2(a, b, c, d,in[ 0], 3); ROUND2(a, b, c, d, in[0], 3);
ROUND2(d, a, b, c, in[4], 5); ROUND2(d, a, b, c, in[4], 5);
ROUND2(c, d, a, b, in[8], 9); ROUND2(c, d, a, b, in[8], 9);
ROUND2(b, c, d, a, in[12], 13); ROUND2(b, c, d, a, in[12], 13);
@@ -136,7 +136,7 @@ static void md4_transform(u32 *hash, u32 const *in)
ROUND2(c, d, a, b, in[11], 9); ROUND2(c, d, a, b, in[11], 9);
ROUND2(b, c, d, a, in[15], 13); ROUND2(b, c, d, a, in[15], 13);
ROUND3(a, b, c, d,in[ 0], 3); ROUND3(a, b, c, d, in[0], 3);
ROUND3(d, a, b, c, in[8], 9); ROUND3(d, a, b, c, in[8], 9);
ROUND3(c, d, a, b, in[4], 11); ROUND3(c, d, a, b, in[4], 11);
ROUND3(b, c, d, a, in[12], 15); ROUND3(b, c, d, a, in[12], 15);
@@ -186,13 +186,11 @@ static int md4_update(struct shash_desc *desc, const u8 *data, unsigned int len)
mctx->byte_count += len; mctx->byte_count += len;
if (avail > len) { if (avail > len) {
memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), data, len);
data, len);
return 0; return 0;
} }
memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), data, avail);
data, avail);
md4_transform_helper(mctx); md4_transform_helper(mctx);
data += avail; data += avail;
@@ -219,7 +217,7 @@ static int md4_final(struct shash_desc *desc, u8 *out)
*p++ = 0x80; *p++ = 0x80;
if (padding < 0) { if (padding < 0) {
memset(p, 0x00, padding + sizeof (u64)); memset(p, 0x00, padding + sizeof(u64));
md4_transform_helper(mctx); md4_transform_helper(mctx);
p = (char *)mctx->block; p = (char *)mctx->block;
padding = 56; padding = 56;
@@ -228,8 +226,7 @@ static int md4_final(struct shash_desc *desc, u8 *out)
memset(p, 0, padding); memset(p, 0, padding);
mctx->block[14] = mctx->byte_count << 3; mctx->block[14] = mctx->byte_count << 3;
mctx->block[15] = mctx->byte_count >> 29; mctx->block[15] = mctx->byte_count >> 29;
le32_to_cpu_array(mctx->block, (sizeof(mctx->block) - le32_to_cpu_array(mctx->block, (sizeof(mctx->block) - sizeof(u64)) / sizeof(u32));
sizeof(u64)) / sizeof(u32));
md4_transform(mctx->hash, mctx->block); md4_transform(mctx->hash, mctx->block);
cpu_to_le32_array(mctx->hash, ARRAY_SIZE(mctx->hash)); cpu_to_le32_array(mctx->hash, ARRAY_SIZE(mctx->hash));
memcpy(out, mctx->hash, sizeof(mctx->hash)); memcpy(out, mctx->hash, sizeof(mctx->hash));
@@ -267,8 +264,8 @@ void md4hexf(const char *path, char *hash)
md4_final(&mctx, out); md4_final(&mctx, out);
for (int i = 0; i < MD4_DIGEST_SIZE; i++) { for (int i = 0; i < MD4_DIGEST_SIZE; i++) {
hash[2*i+0] = to_hex(out[i] >> 4); hash[2 * i + 0] = to_hex(out[i] >> 4);
hash[2*i+1] = to_hex(out[i] & 0xf); hash[2 * i + 1] = to_hex(out[i] & 0xf);
} }
hash[2*MD4_DIGEST_SIZE] = 0; hash[2 * MD4_DIGEST_SIZE] = 0;
} }

File diff suppressed because it is too large Load Diff

2418
src/tint2conf/po/es.po Normal file

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,36 @@
#!/bin/bash
set -e
set -x
find .. -name '*.c' | sort -r | xargs xgettext --keyword=_ --language=C --output=tint2conf.pot -
sed -i "s/PACKAGE VERSION/tint2conf $(../../../get_version.sh | head -n1)/g" tint2conf.pot
sed -i "s/CHARSET/UTF-8/g" tint2conf.pot
for f in *.po
do
lang=$(basename $f .po)
echo $lang
msgmerge -i -o $lang.pox $lang.po tint2conf.pot
cat ${lang}.pox > ${lang}.po
rm ${lang}.pox
done
set +e
set +x
echo "Status:"
for f in *.po
do
lang=$(basename $f .po)
fuzzy=$(cat ${lang}.po | grep -A2 "#, fuzzy")
missing=$(cat ${lang}.po | grep -B1 'msgstr ""')
if [ -z "$fuzzy" ] && [ -z "$missing" ]
then
echo $lang ": Up to date"
else
count=$(( $(echo -e "$fuzzy" "\n" "$missing" | grep "^--$" | wc -l) + 1))
echo "${lang}: Translation incomplete: ${count} strings to be updated. See ${lang}.todo"
echo "$fuzzy" > ${lang}.todo
echo "$missing" >> ${lang}.todo
fi
done

File diff suppressed because it is too large Load Diff

View File

@@ -8,22 +8,20 @@
#include "../launcher/icon-theme-common.h" #include "../launcher/icon-theme-common.h"
// panel // panel
extern GtkWidget *panel_width, *panel_height, *panel_margin_x, *panel_margin_y, *panel_padding_x, *panel_padding_y, *panel_spacing; extern GtkWidget *panel_width, *panel_height, *panel_margin_x, *panel_margin_y, *panel_padding_x, *panel_padding_y,
extern GtkWidget *panel_wm_menu, *panel_dock, *panel_autohide, *panel_autohide_show_time, *panel_autohide_hide_time, *panel_autohide_size; *panel_spacing;
extern GtkWidget *panel_combo_strut_policy, *panel_combo_layer, *panel_combo_width_type, *panel_combo_height_type, *panel_combo_monitor; extern GtkWidget *panel_wm_menu, *panel_dock, *panel_autohide, *panel_autohide_show_time, *panel_autohide_hide_time,
*panel_autohide_size;
extern GtkWidget *panel_combo_strut_policy, *panel_combo_layer, *panel_combo_width_type, *panel_combo_height_type,
*panel_combo_monitor;
extern GtkWidget *panel_window_name, *disable_transparency; extern GtkWidget *panel_window_name, *disable_transparency;
extern GtkWidget *panel_mouse_effects; extern GtkWidget *panel_mouse_effects;
extern GtkWidget *mouse_hover_icon_opacity, *mouse_hover_icon_saturation, *mouse_hover_icon_brightness; extern GtkWidget *mouse_hover_icon_opacity, *mouse_hover_icon_saturation, *mouse_hover_icon_brightness;
extern GtkWidget *mouse_pressed_icon_opacity, *mouse_pressed_icon_saturation, *mouse_pressed_icon_brightness; extern GtkWidget *mouse_pressed_icon_opacity, *mouse_pressed_icon_saturation, *mouse_pressed_icon_brightness;
extern GtkWidget *panel_primary_monitor_first, *panel_shrink; extern GtkWidget *panel_shrink;
enum { enum { itemsColName = 0, itemsColValue, itemsNumCols };
itemsColName = 0,
itemsColValue,
itemsNumCols
};
extern GtkListStore *panel_items, *all_items; extern GtkListStore *panel_items, *all_items;
extern GtkWidget *panel_items_view, *all_items_view; extern GtkWidget *panel_items_view, *all_items_view;
char *get_panel_items(); char *get_panel_items();
@@ -51,44 +49,34 @@ extern GtkWidget *panel_background;
// taskbar // taskbar
extern GtkWidget *taskbar_show_desktop, *taskbar_show_name, *taskbar_padding_x, *taskbar_padding_y, *taskbar_spacing; extern GtkWidget *taskbar_show_desktop, *taskbar_show_name, *taskbar_padding_x, *taskbar_padding_y, *taskbar_spacing;
extern GtkWidget *taskbar_hide_inactive_tasks, *taskbar_hide_diff_monitor; extern GtkWidget *taskbar_hide_inactive_tasks, *taskbar_hide_diff_monitor, *taskbar_hide_diff_desktop;
extern GtkWidget *taskbar_name_padding_x, *taskbar_name_padding_y, *taskbar_name_inactive_color, *taskbar_name_active_color; extern GtkWidget *taskbar_name_padding_x, *taskbar_name_padding_y, *taskbar_name_inactive_color,
*taskbar_name_active_color;
extern GtkWidget *taskbar_name_font, *taskbar_name_font_set; extern GtkWidget *taskbar_name_font, *taskbar_name_font_set;
extern GtkWidget *taskbar_active_background, *taskbar_inactive_background; extern GtkWidget *taskbar_active_background, *taskbar_inactive_background;
extern GtkWidget *taskbar_name_active_background, *taskbar_name_inactive_background; extern GtkWidget *taskbar_name_active_background, *taskbar_name_inactive_background;
extern GtkWidget *taskbar_distribute_size, *taskbar_sort_order, *taskbar_alignment, *taskbar_always_show_all_desktop_tasks; extern GtkWidget *taskbar_distribute_size, *taskbar_sort_order, *taskbar_alignment,
*taskbar_always_show_all_desktop_tasks;
extern GtkWidget *taskbar_hide_empty; extern GtkWidget *taskbar_hide_empty;
// task // task
extern GtkWidget *task_mouse_left, *task_mouse_middle, *task_mouse_right, *task_mouse_scroll_up, *task_mouse_scroll_down; extern GtkWidget *task_mouse_left, *task_mouse_middle, *task_mouse_right, *task_mouse_scroll_up,
*task_mouse_scroll_down;
extern GtkWidget *task_show_icon, *task_show_text, *task_align_center, *font_shadow; extern GtkWidget *task_show_icon, *task_show_text, *task_align_center, *font_shadow;
extern GtkWidget *task_maximum_width, *task_maximum_height, *task_padding_x, *task_padding_y, *task_spacing; extern GtkWidget *task_maximum_width, *task_maximum_height, *task_padding_x, *task_padding_y, *task_spacing;
extern GtkWidget *task_font, *task_font_set; extern GtkWidget *task_font, *task_font_set;
extern GtkWidget *task_default_color, *task_default_color_set, extern GtkWidget *task_default_color, *task_default_color_set, *task_default_icon_opacity, *task_default_icon_osb_set,
*task_default_icon_opacity, *task_default_icon_osb_set, *task_default_icon_saturation, *task_default_icon_brightness, *task_default_background,
*task_default_icon_saturation, *task_default_background_set;
*task_default_icon_brightness, extern GtkWidget *task_normal_color, *task_normal_color_set, *task_normal_icon_opacity, *task_normal_icon_osb_set,
*task_default_background, *task_default_background_set; *task_normal_icon_saturation, *task_normal_icon_brightness, *task_normal_background, *task_normal_background_set;
extern GtkWidget *task_normal_color, *task_normal_color_set, extern GtkWidget *task_active_color, *task_active_color_set, *task_active_icon_opacity, *task_active_icon_osb_set,
*task_normal_icon_opacity, *task_normal_icon_osb_set, *task_active_icon_saturation, *task_active_icon_brightness, *task_active_background, *task_active_background_set;
*task_normal_icon_saturation, extern GtkWidget *task_urgent_color, *task_urgent_color_set, *task_urgent_icon_opacity, *task_urgent_icon_osb_set,
*task_normal_icon_brightness, *task_urgent_icon_saturation, *task_urgent_icon_brightness, *task_urgent_background, *task_urgent_background_set;
*task_normal_background, *task_normal_background_set;
extern GtkWidget *task_active_color, *task_active_color_set,
*task_active_icon_opacity, *task_active_icon_osb_set,
*task_active_icon_saturation,
*task_active_icon_brightness,
*task_active_background, *task_active_background_set;
extern GtkWidget *task_urgent_color, *task_urgent_color_set,
*task_urgent_icon_opacity, *task_urgent_icon_osb_set,
*task_urgent_icon_saturation,
*task_urgent_icon_brightness,
*task_urgent_background, *task_urgent_background_set;
extern GtkWidget *task_urgent_blinks; extern GtkWidget *task_urgent_blinks;
extern GtkWidget *task_iconified_color, *task_iconified_color_set, extern GtkWidget *task_iconified_color, *task_iconified_color_set, *task_iconified_icon_opacity,
*task_iconified_icon_opacity, *task_iconified_icon_osb_set, *task_iconified_icon_osb_set, *task_iconified_icon_saturation, *task_iconified_icon_brightness,
*task_iconified_icon_saturation,
*task_iconified_icon_brightness,
*task_iconified_background, *task_iconified_background_set; *task_iconified_background, *task_iconified_background_set;
// clock // clock
@@ -100,12 +88,14 @@ extern GtkWidget *clock_font_line1, *clock_font_line1_set, *clock_font_line2, *c
extern GtkWidget *clock_background; extern GtkWidget *clock_background;
// battery // battery
extern GtkWidget *battery_hide_if_higher, *battery_alert_if_lower, *battery_alert_cmd; extern GtkWidget *battery_hide_if_higher, *battery_alert_if_lower, *battery_alert_cmd, *battery_alert_full_cmd;
extern GtkWidget *battery_padding_x, *battery_padding_y; extern GtkWidget *battery_padding_x, *battery_padding_y;
extern GtkWidget *battery_font_line1, *battery_font_line1_set, *battery_font_line2, *battery_font_line2_set, *battery_font_color; extern GtkWidget *battery_font_line1, *battery_font_line1_set, *battery_font_line2, *battery_font_line2_set,
*battery_font_color, *battery_format1, *battery_format2;
extern GtkWidget *battery_background; extern GtkWidget *battery_background;
extern GtkWidget *battery_tooltip; extern GtkWidget *battery_tooltip;
extern GtkWidget *battery_left_command, *battery_mclick_command, *battery_right_command, *battery_uwheel_command, *battery_dwheel_command; extern GtkWidget *battery_left_command, *battery_mclick_command, *battery_right_command, *battery_uwheel_command,
*battery_dwheel_command;
extern GtkWidget *ac_connected_cmd, *ac_disconnected_cmd; extern GtkWidget *ac_connected_cmd, *ac_disconnected_cmd;
// systray // systray
@@ -115,7 +105,7 @@ extern GtkWidget *systray_background, *systray_monitor, *systray_name_filter;
// tooltip // tooltip
extern GtkWidget *tooltip_padding_x, *tooltip_padding_y, *tooltip_font, *tooltip_font_set, *tooltip_font_color; extern GtkWidget *tooltip_padding_x, *tooltip_padding_y, *tooltip_font, *tooltip_font_set, *tooltip_font_color;
extern GtkWidget *tooltip_task_show, *tooltip_show_after, *tooltip_hide_after; extern GtkWidget *tooltip_task_show, *tooltip_show_after, *tooltip_hide_after, *tooltip_task_thumbnail, *tooltip_task_thumbnail_size;
extern GtkWidget *clock_format_tooltip, *clock_tmz_tooltip; extern GtkWidget *clock_format_tooltip, *clock_tmz_tooltip;
extern GtkWidget *tooltip_background; extern GtkWidget *tooltip_background;
@@ -160,7 +150,8 @@ typedef struct Button {
GtkWidget *button_icon, *button_text, *button_tooltip; GtkWidget *button_icon, *button_text, *button_tooltip;
GtkWidget *button_left_command, *button_right_command; GtkWidget *button_left_command, *button_right_command;
GtkWidget *button_mclick_command, *button_rclick_command, *button_uwheel_command, *button_dwheel_command; GtkWidget *button_mclick_command, *button_rclick_command, *button_uwheel_command, *button_dwheel_command;
GtkWidget *button_font, *button_font_set, *button_font_color, *button_padding_x, *button_padding_y, *button_centered; GtkWidget *button_font, *button_font_set, *button_font_color, *button_padding_x, *button_padding_y,
*button_centered;
GtkWidget *button_background, *button_max_icon_size; GtkWidget *button_background, *button_max_icon_size;
} Button; } Button;
@@ -168,13 +159,7 @@ extern GArray *buttons;
// launcher // launcher
enum { enum { appsColIcon = 0, appsColIconName, appsColText, appsColPath, appsNumCols };
appsColIcon = 0,
appsColIconName,
appsColText,
appsColPath,
appsNumCols
};
extern GtkListStore *launcher_apps, *all_apps; extern GtkListStore *launcher_apps, *all_apps;
extern GtkWidget *launcher_apps_view, *all_apps_view; extern GtkWidget *launcher_apps_view, *all_apps_view;
@@ -218,50 +203,28 @@ enum {
bgColBorderSidesBottom, bgColBorderSidesBottom,
bgColBorderSidesLeft, bgColBorderSidesLeft,
bgColBorderSidesRight, bgColBorderSidesRight,
bgColFillWeight,
bgColBorderWeight,
bgNumCols bgNumCols
}; };
extern GtkListStore *backgrounds; extern GtkListStore *backgrounds;
extern GtkWidget *current_background, extern GtkWidget *current_background, *background_fill_color, *background_border_color, *background_gradient,
*background_fill_color, *background_fill_color_over, *background_border_color_over, *background_gradient_over, *background_fill_color_press,
*background_border_color, *background_border_color_press, *background_gradient_press, *background_border_width, *background_border_sides_top,
*background_gradient, *background_border_sides_bottom, *background_border_sides_left, *background_border_sides_right,
*background_fill_color_over, *background_corner_radius, *background_border_content_tint_weight, *background_fill_content_tint_weight;
*background_border_color_over,
*background_gradient_over,
*background_fill_color_press,
*background_border_color_press,
*background_gradient_press,
*background_border_width,
*background_border_sides_top,
*background_border_sides_bottom,
*background_border_sides_left,
*background_border_sides_right,
*background_corner_radius;
// gradients // gradients
enum { enum { grColPixbuf = 0, grColId, grColText, grNumCols };
grColPixbuf = 0,
grColId,
grColText,
grNumCols
};
// gradient color stops // gradient color stops
enum { enum { grStopColPixbuf = 0, grStopNumCols };
grStopColPixbuf = 0,
grStopNumCols
};
extern GtkListStore *gradient_ids, *gradient_stop_ids; extern GtkListStore *gradient_ids, *gradient_stop_ids;
extern GList *gradients; extern GList *gradients;
extern GtkWidget *current_gradient, extern GtkWidget *current_gradient, *gradient_combo_type, *gradient_start_color, *gradient_end_color,
*gradient_combo_type, *current_gradient_stop, *gradient_stop_color, *gradient_stop_offset;
*gradient_start_color,
*gradient_end_color,
*current_gradient_stop,
*gradient_stop_color,
*gradient_stop_offset;
void background_create_new(); void background_create_new();
void background_force_update(); void background_force_update();

View File

@@ -49,7 +49,7 @@ void config_read_file(const char *path)
FILE *fp = fopen(path, "r"); FILE *fp = fopen(path, "r");
if (fp) { if (fp) {
char* line = NULL; char *line = NULL;
size_t line_size = 0; size_t line_size = 0;
while (getline(&line, &line_size, fp) >= 0) { while (getline(&line, &line_size, fp) >= 0) {
char *key, *value; char *key, *value;
@@ -108,9 +108,15 @@ void config_write_gradients(FILE *fp)
int opacity; int opacity;
fprintf(fp, "# Gradient %d\n", index); fprintf(fp, "# Gradient %d\n", index);
fprintf(fp, "gradient = %s\n", g->type == GRADIENT_CONFIG_HORIZONTAL ? "horizontal" : g->type == GRADIENT_CONFIG_VERTICAL ? "vertical" : "radial"); fprintf(fp,
"gradient = %s\n",
g->type == GRADIENT_CONFIG_HORIZONTAL ? "horizontal" : g->type == GRADIENT_CONFIG_VERTICAL ? "vertical"
: "radial");
cairoColor2GdkColor(g->start_color.color.rgb[0], g->start_color.color.rgb[1], g->start_color.color.rgb[2], &color); cairoColor2GdkColor(g->start_color.color.rgb[0],
g->start_color.color.rgb[1],
g->start_color.color.rgb[2],
&color);
opacity = g->start_color.color.alpha * 100; opacity = g->start_color.color.alpha * 100;
config_write_color(fp, "start_color", color, opacity); config_write_color(fp, "start_color", color, opacity);
@@ -123,7 +129,13 @@ void config_write_gradients(FILE *fp)
// color_stop = percentage #rrggbb opacity // color_stop = percentage #rrggbb opacity
cairoColor2GdkColor(stop->color.rgb[0], stop->color.rgb[1], stop->color.rgb[2], &color); cairoColor2GdkColor(stop->color.rgb[0], stop->color.rgb[1], stop->color.rgb[2], &color);
opacity = stop->color.alpha * 100; opacity = stop->color.alpha * 100;
fprintf(fp, "color_stop = %f #%02x%02x%02x %d\n", stop->offset * 100, color.red >> 8, color.green >> 8, color.blue >> 8, opacity); fprintf(fp,
"color_stop = %f #%02x%02x%02x %d\n",
stop->offset * 100,
color.red >> 8,
color.green >> 8,
color.blue >> 8,
opacity);
} }
fprintf(fp, "\n"); fprintf(fp, "\n");
} }
@@ -149,6 +161,8 @@ void config_write_backgrounds(FILE *fp)
int r; int r;
int b; int b;
double fill_weight;
double border_weight;
gboolean sideTop; gboolean sideTop;
gboolean sideBottom; gboolean sideBottom;
gboolean sideLeft; gboolean sideLeft;
@@ -216,6 +230,10 @@ void config_write_backgrounds(FILE *fp)
&sideLeft, &sideLeft,
bgColBorderSidesRight, bgColBorderSidesRight,
&sideRight, &sideRight,
bgColFillWeight,
&fill_weight,
bgColBorderWeight,
&border_weight,
-1); -1);
fprintf(fp, "# Background %d: %s\n", index, text ? text : ""); fprintf(fp, "# Background %d: %s\n", index, text ? text : "");
fprintf(fp, "rounded = %d\n", r); fprintf(fp, "rounded = %d\n", r);
@@ -233,6 +251,9 @@ void config_write_backgrounds(FILE *fp)
strcat(sides, "R"); strcat(sides, "R");
fprintf(fp, "border_sides = %s\n", sides); fprintf(fp, "border_sides = %s\n", sides);
fprintf(fp, "border_content_tint_weight = %d\n", (int)(border_weight));
fprintf(fp, "background_content_tint_weight = %d\n", (int)(fill_weight));
config_write_color(fp, "background_color", *fillColor, fillOpacity); config_write_color(fp, "background_color", *fillColor, fillOpacity);
config_write_color(fp, "border_color", *borderColor, borderOpacity); config_write_color(fp, "border_color", *borderColor, borderOpacity);
if (gradient_id >= 0) if (gradient_id >= 0)
@@ -314,20 +335,16 @@ void config_write_panel(FILE *fp)
fprintf(fp, "\n"); fprintf(fp, "\n");
fprintf(fp, "panel_monitor = "); fprintf(fp, "panel_monitor = ");
if (gtk_combo_box_get_active(GTK_COMBO_BOX(panel_combo_monitor)) == 0) { if (gtk_combo_box_get_active(GTK_COMBO_BOX(panel_combo_monitor)) <= 0) {
fprintf(fp, "all"); fprintf(fp, "all");
} else if (gtk_combo_box_get_active(GTK_COMBO_BOX(panel_combo_monitor)) == 1) {
fprintf(fp, "primary");
} else { } else {
fprintf(fp, "%d", gtk_combo_box_get_active(GTK_COMBO_BOX(panel_combo_monitor))); fprintf(fp, "%d", gtk_combo_box_get_active(GTK_COMBO_BOX(panel_combo_monitor)) - 1);
} }
fprintf(fp, "\n"); fprintf(fp, "\n");
fprintf(fp, fprintf(fp, "panel_shrink = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(panel_shrink)) ? 1 : 0);
"primary_monitor_first = %d\n",
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(panel_primary_monitor_first)) ? 1 : 0);
fprintf(fp,
"panel_shrink = %d\n",
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(panel_shrink)) ? 1 : 0);
fprintf(fp, "autohide = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(panel_autohide)) ? 1 : 0); fprintf(fp, "autohide = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(panel_autohide)) ? 1 : 0);
fprintf(fp, "autohide_show_timeout = %g\n", gtk_spin_button_get_value(GTK_SPIN_BUTTON(panel_autohide_show_time))); fprintf(fp, "autohide_show_timeout = %g\n", gtk_spin_button_get_value(GTK_SPIN_BUTTON(panel_autohide_show_time)));
@@ -391,6 +408,9 @@ void config_write_taskbar(FILE *fp)
fprintf(fp, fprintf(fp,
"taskbar_hide_different_monitor = %d\n", "taskbar_hide_different_monitor = %d\n",
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_monitor)) ? 1 : 0); gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_monitor)) ? 1 : 0);
fprintf(fp,
"taskbar_hide_different_desktop = %d\n",
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_desktop)) ? 1 : 0);
fprintf(fp, fprintf(fp,
"taskbar_always_show_all_desktop_tasks = %d\n", "taskbar_always_show_all_desktop_tasks = %d\n",
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(taskbar_always_show_all_desktop_tasks)) ? 1 : 0); gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(taskbar_always_show_all_desktop_tasks)) ? 1 : 0);
@@ -506,6 +526,11 @@ void config_write_task(FILE *fp)
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(task_font_set))) if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(task_font_set)))
fprintf(fp, "task_font = %s\n", gtk_font_button_get_font_name(GTK_FONT_BUTTON(task_font))); fprintf(fp, "task_font = %s\n", gtk_font_button_get_font_name(GTK_FONT_BUTTON(task_font)));
fprintf(fp, "task_tooltip = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(tooltip_task_show)) ? 1 : 0); fprintf(fp, "task_tooltip = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(tooltip_task_show)) ? 1 : 0);
fprintf(fp, "task_thumbnail = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(tooltip_task_thumbnail)) ? 1 : 0);
fprintf(fp,
"task_thumbnail_size = %d\n",
(int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(tooltip_task_thumbnail_size)));
// same for: "" _normal _active _urgent _iconified // same for: "" _normal _active _urgent _iconified
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(task_default_color_set))) { if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(task_default_color_set))) {
@@ -619,7 +644,11 @@ void config_write_systray(FILE *fp)
(int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(systray_icon_brightness))); (int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(systray_icon_brightness)));
fprintf(fp, "systray_monitor = "); fprintf(fp, "systray_monitor = ");
fprintf(fp, "%d", MAX(1, 1 + gtk_combo_box_get_active(GTK_COMBO_BOX(systray_monitor)))); if (gtk_combo_box_get_active(GTK_COMBO_BOX(systray_monitor)) <= 0) {
fprintf(fp, "primary");
} else {
fprintf(fp, "%d", MAX(1, gtk_combo_box_get_active(GTK_COMBO_BOX(systray_monitor))));
}
fprintf(fp, "\n"); fprintf(fp, "\n");
fprintf(fp, "systray_name_filter = %s\n", gtk_entry_get_text(GTK_ENTRY(systray_name_filter))); fprintf(fp, "systray_name_filter = %s\n", gtk_entry_get_text(GTK_ENTRY(systray_name_filter)));
@@ -742,6 +771,7 @@ void config_write_battery(FILE *fp)
fprintf(fp, "battery_tooltip = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_tooltip)) ? 1 : 0); fprintf(fp, "battery_tooltip = %d\n", gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_tooltip)) ? 1 : 0);
fprintf(fp, "battery_low_status = %g\n", gtk_spin_button_get_value(GTK_SPIN_BUTTON(battery_alert_if_lower))); fprintf(fp, "battery_low_status = %g\n", gtk_spin_button_get_value(GTK_SPIN_BUTTON(battery_alert_if_lower)));
fprintf(fp, "battery_low_cmd = %s\n", gtk_entry_get_text(GTK_ENTRY(battery_alert_cmd))); fprintf(fp, "battery_low_cmd = %s\n", gtk_entry_get_text(GTK_ENTRY(battery_alert_cmd)));
fprintf(fp, "battery_full_cmd = %s\n", gtk_entry_get_text(GTK_ENTRY(battery_alert_full_cmd)));
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_font_line1_set))) if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_font_line1_set)))
fprintf(fp, "bat1_font = %s\n", gtk_font_button_get_font_name(GTK_FONT_BUTTON(battery_font_line1))); fprintf(fp, "bat1_font = %s\n", gtk_font_button_get_font_name(GTK_FONT_BUTTON(battery_font_line1)));
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_font_line2_set))) if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(battery_font_line2_set)))
@@ -752,6 +782,8 @@ void config_write_battery(FILE *fp)
"battery_font_color", "battery_font_color",
color, color,
gtk_color_button_get_alpha(GTK_COLOR_BUTTON(battery_font_color)) * 100 / 0xffff); gtk_color_button_get_alpha(GTK_COLOR_BUTTON(battery_font_color)) * 100 / 0xffff);
fprintf(fp, "bat1_format = %s\n", gtk_entry_get_text(GTK_ENTRY(battery_format1)));
fprintf(fp, "bat2_format = %s\n", gtk_entry_get_text(GTK_ENTRY(battery_format2)));
fprintf(fp, fprintf(fp,
"battery_padding = %d %d\n", "battery_padding = %d %d\n",
(int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(battery_padding_x)), (int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(battery_padding_x)),
@@ -908,7 +940,6 @@ void config_write_button(FILE *fp)
"button_max_icon_size = %d\n", "button_max_icon_size = %d\n",
(int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(button->button_max_icon_size))); (int)gtk_spin_button_get_value(GTK_SPIN_BUTTON(button->button_max_icon_size)));
fprintf(fp, "\n"); fprintf(fp, "\n");
} }
} }
@@ -964,7 +995,7 @@ unsigned short checksum_txt(FILE *f)
void config_save_file(const char *path) void config_save_file(const char *path)
{ {
printf("config_save_file : %s\n", path); fprintf(stderr, "tint2: config_save_file : %s\n", path);
FILE *fp; FILE *fp;
if ((fp = fopen(path, "w+t")) == NULL) if ((fp = fopen(path, "w+t")) == NULL)
@@ -1108,7 +1139,7 @@ void add_entry(char *key, char *value)
} else if (strcmp(key, "color_stop") == 0) { } else if (strcmp(key, "color_stop") == 0) {
GradientConfig *g = (GradientConfig *)g_list_last(gradients)->data; GradientConfig *g = (GradientConfig *)g_list_last(gradients)->data;
extract_values(value, &value1, &value2, &value3); extract_values(value, &value1, &value2, &value3);
GradientConfigColorStop *color_stop = (GradientConfigColorStop *) calloc(1, sizeof(GradientConfigColorStop)); GradientConfigColorStop *color_stop = (GradientConfigColorStop *)calloc(1, sizeof(GradientConfigColorStop));
color_stop->offset = atof(value1) / 100.0; color_stop->offset = atof(value1) / 100.0;
get_color(value2, color_stop->color.rgb); get_color(value2, color_stop->color.rgb);
if (value3) if (value3)
@@ -1208,6 +1239,12 @@ void add_entry(char *key, char *value)
int id = gradient_index_safe(atoi(value)); int id = gradient_index_safe(atoi(value));
gtk_combo_box_set_active(GTK_COMBO_BOX(background_gradient_press), id); gtk_combo_box_set_active(GTK_COMBO_BOX(background_gradient_press), id);
background_force_update(); background_force_update();
} else if (strcmp(key, "border_content_tint_weight") == 0) {
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_border_content_tint_weight), atoi(value));
background_force_update();
} else if (strcmp(key, "background_content_tint_weight") == 0) {
gtk_spin_button_set_value(GTK_SPIN_BUTTON(background_fill_content_tint_weight), atoi(value));
background_force_update();
} }
/* Panel */ /* Panel */
@@ -1342,20 +1379,20 @@ void add_entry(char *key, char *value)
} else if (strcmp(key, "panel_monitor") == 0) { } else if (strcmp(key, "panel_monitor") == 0) {
if (strcmp(value, "all") == 0) if (strcmp(value, "all") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 0); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 0);
else if (strcmp(value, "1") == 0) else if (strcmp(value, "primary") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 1); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 1);
else if (strcmp(value, "2") == 0) else if (strcmp(value, "1") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 2); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 2);
else if (strcmp(value, "3") == 0) else if (strcmp(value, "2") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 3); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 3);
else if (strcmp(value, "4") == 0) else if (strcmp(value, "3") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 4); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 4);
else if (strcmp(value, "5") == 0) else if (strcmp(value, "4") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 5); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 5);
else if (strcmp(value, "6") == 0) else if (strcmp(value, "5") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 6); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 6);
} else if (strcmp(key, "primary_monitor_first") == 0) { else if (strcmp(value, "6") == 0)
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(panel_primary_monitor_first), atoi(value)); gtk_combo_box_set_active(GTK_COMBO_BOX(panel_combo_monitor), 7);
} else if (strcmp(key, "panel_shrink") == 0) { } else if (strcmp(key, "panel_shrink") == 0) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(panel_shrink), atoi(value)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(panel_shrink), atoi(value));
} }
@@ -1394,12 +1431,18 @@ void add_entry(char *key, char *value)
gtk_spin_button_set_value(GTK_SPIN_BUTTON(battery_alert_if_lower), atof(value)); gtk_spin_button_set_value(GTK_SPIN_BUTTON(battery_alert_if_lower), atof(value));
} else if (strcmp(key, "battery_low_cmd") == 0) { } else if (strcmp(key, "battery_low_cmd") == 0) {
gtk_entry_set_text(GTK_ENTRY(battery_alert_cmd), value); gtk_entry_set_text(GTK_ENTRY(battery_alert_cmd), value);
} else if (strcmp(key, "battery_full_cmd") == 0) {
gtk_entry_set_text(GTK_ENTRY(battery_alert_full_cmd), value);
} else if (strcmp(key, "bat1_font") == 0) { } else if (strcmp(key, "bat1_font") == 0) {
gtk_font_button_set_font_name(GTK_FONT_BUTTON(battery_font_line1), value); gtk_font_button_set_font_name(GTK_FONT_BUTTON(battery_font_line1), value);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(battery_font_line1_set), TRUE); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(battery_font_line1_set), TRUE);
} else if (strcmp(key, "bat2_font") == 0) { } else if (strcmp(key, "bat2_font") == 0) {
gtk_font_button_set_font_name(GTK_FONT_BUTTON(battery_font_line2), value); gtk_font_button_set_font_name(GTK_FONT_BUTTON(battery_font_line2), value);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(battery_font_line2_set), TRUE); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(battery_font_line2_set), TRUE);
} else if (strcmp(key, "bat1_format") == 0) {
gtk_entry_set_text(GTK_ENTRY(battery_format1), value);
} else if (strcmp(key, "bat2_format") == 0) {
gtk_entry_set_text(GTK_ENTRY(battery_format2), value);
} else if (strcmp(key, "battery_font_color") == 0) { } else if (strcmp(key, "battery_font_color") == 0) {
extract_values(value, &value1, &value2, &value3); extract_values(value, &value1, &value2, &value3);
GdkColor col; GdkColor col;
@@ -1542,6 +1585,8 @@ void add_entry(char *key, char *value)
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_hide_inactive_tasks), atoi(value)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_hide_inactive_tasks), atoi(value));
} else if (strcmp(key, "taskbar_hide_different_monitor") == 0) { } else if (strcmp(key, "taskbar_hide_different_monitor") == 0) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_monitor), atoi(value)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_monitor), atoi(value));
} else if (strcmp(key, "taskbar_hide_different_desktop") == 0) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_hide_diff_desktop), atoi(value));
} else if (strcmp(key, "taskbar_always_show_all_desktop_tasks") == 0) { } else if (strcmp(key, "taskbar_always_show_all_desktop_tasks") == 0) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_always_show_all_desktop_tasks), atoi(value)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(taskbar_always_show_all_desktop_tasks), atoi(value));
} else if (strcmp(key, "taskbar_name_padding") == 0) { } else if (strcmp(key, "taskbar_name_padding") == 0) {
@@ -1707,6 +1752,10 @@ void add_entry(char *key, char *value)
else if (strcmp(key, "task_tooltip") == 0 || strcmp(key, "tooltip") == 0) { else if (strcmp(key, "task_tooltip") == 0 || strcmp(key, "tooltip") == 0) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(tooltip_task_show), atoi(value)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(tooltip_task_show), atoi(value));
} }
else if (strcmp(key, "task_thumbnail") == 0)
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(tooltip_task_thumbnail), atoi(value));
else if (strcmp(key, "task_thumbnail_size") == 0)
gtk_spin_button_set_value(GTK_SPIN_BUTTON(tooltip_task_thumbnail_size), MAX(8, atoi(value)));
/* Systray */ /* Systray */
else if (strcmp(key, "systray") == 0) { else if (strcmp(key, "systray") == 0) {
@@ -1738,18 +1787,20 @@ void add_entry(char *key, char *value)
} else if (strcmp(key, "systray_icon_size") == 0) { } else if (strcmp(key, "systray_icon_size") == 0) {
gtk_spin_button_set_value(GTK_SPIN_BUTTON(systray_icon_size), atoi(value)); gtk_spin_button_set_value(GTK_SPIN_BUTTON(systray_icon_size), atoi(value));
} else if (strcmp(key, "systray_monitor") == 0) { } else if (strcmp(key, "systray_monitor") == 0) {
if (strcmp(value, "1") == 0) if (strcmp(value, "primary") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 0); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 0);
else if (strcmp(value, "2") == 0) else if (strcmp(value, "1") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 1); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 1);
else if (strcmp(value, "3") == 0) else if (strcmp(value, "2") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 2); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 2);
else if (strcmp(value, "4") == 0) else if (strcmp(value, "3") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 3); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 3);
else if (strcmp(value, "5") == 0) else if (strcmp(value, "4") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 4); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 4);
else if (strcmp(value, "6") == 0) else if (strcmp(value, "5") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 5); gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 5);
else if (strcmp(value, "6") == 0)
gtk_combo_box_set_active(GTK_COMBO_BOX(systray_monitor), 6);
} else if (strcmp(key, "systray_icon_asb") == 0) { } else if (strcmp(key, "systray_icon_asb") == 0) {
extract_values(value, &value1, &value2, &value3); extract_values(value, &value1, &value2, &value3);
gtk_spin_button_set_value(GTK_SPIN_BUTTON(systray_icon_opacity), atoi(value1)); gtk_spin_button_set_value(GTK_SPIN_BUTTON(systray_icon_opacity), atoi(value1));

View File

@@ -5,7 +5,7 @@
char *get_current_theme_path(); char *get_current_theme_path();
gboolean config_is_manual(const char *path); gboolean config_is_manual(const char *path);
void config_read_file (const char *path); void config_read_file(const char *path);
void config_save_file(const char *path); void config_save_file(const char *path);
#endif #endif

View File

@@ -37,8 +37,13 @@ GtkWidget *create_view()
GtkTreeViewColumn *col; GtkTreeViewColumn *col;
GtkCellRenderer *renderer; GtkCellRenderer *renderer;
theme_list_store = theme_list_store = gtk_list_store_new(NB_COL,
gtk_list_store_new(NB_COL, G_TYPE_STRING, G_TYPE_STRING, GDK_TYPE_PIXBUF, G_TYPE_INT, G_TYPE_INT, G_TYPE_BOOLEAN); G_TYPE_STRING,
G_TYPE_STRING,
GDK_TYPE_PIXBUF,
G_TYPE_INT,
G_TYPE_INT,
G_TYPE_BOOLEAN);
GtkWidget *view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(theme_list_store)); GtkWidget *view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(theme_list_store));
gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(view), TRUE); gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(view), TRUE);
@@ -140,25 +145,39 @@ void theme_list_append(const gchar *path)
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_append(theme_list_store, &iter); gtk_list_store_append(theme_list_store, &iter);
gchar *name = strrchr(path, '/') + 1; gchar *name, *dir;
gchar *dir = g_strdup(path); if (strchr(path, '/')) {
name = strrchr(path, '/') + 1;
dir = g_strdup(path);
strrchr(dir, '/')[0] = 0; strrchr(dir, '/')[0] = 0;
} else {
name = (gchar*)path;
dir = g_strdup(".");
}
char *suffix = contract_tilde(dir); char *suffix = contract_tilde(dir);
g_free(dir); g_free(dir);
gchar *display_name = g_strdup_printf("%s\n(%s)", name, suffix); gchar *display_name = g_strdup_printf("%s\n(%s)", name, suffix);
gtk_list_store_set(theme_list_store, &iter, COL_THEME_FILE, path, COL_THEME_NAME, display_name, COL_FORCE_REFRESH, FALSE, -1); gtk_list_store_set(theme_list_store,
&iter,
COL_THEME_FILE,
path,
COL_THEME_NAME,
display_name,
COL_FORCE_REFRESH,
FALSE,
-1);
g_free(display_name); g_free(display_name);
g_free(suffix); g_free(suffix);
} }
gboolean update_snapshot() gboolean update_snapshot(gpointer ignored)
{ {
{ {
gchar *tint2_cache_dir = g_build_filename(g_get_user_cache_dir(), "tint2", NULL); gchar *tint2_cache_dir = g_build_filename(g_get_user_cache_dir(), "tint2", NULL);
if (!g_file_test(tint2_cache_dir, G_FILE_TEST_IS_DIR)) if (!g_file_test(tint2_cache_dir, G_FILE_TEST_IS_DIR))
g_mkdir(tint2_cache_dir, 0700); g_mkdir_with_parents(tint2_cache_dir, 0700);
g_free(tint2_cache_dir); g_free(tint2_cache_dir);
} }

Some files were not shown because too many files have changed in this diff Show More