Compare commits
809 Commits
debian-tes
...
v0.14.3
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f351b6fd8e | ||
|
|
a7e4cf42e2 | ||
|
|
a4542e907e | ||
|
|
12e3fd64bb | ||
|
|
e5b3a2d7f2 | ||
|
|
b973cafce5 | ||
|
|
162de7fc2b | ||
|
|
b2cc11dd5e | ||
|
|
c53274a576 | ||
|
|
3120cb9cae | ||
|
|
91f7384fd2 | ||
|
|
2e832a988f | ||
|
|
ae60e8f1e7 | ||
|
|
e7dc61565d | ||
|
|
4b53bc2a22 | ||
|
|
354e23da8f | ||
|
|
f6d121570b | ||
|
|
eddc29a696 | ||
|
|
eca3820e48 | ||
|
|
ef73ff9024 | ||
|
|
b7316ed24a | ||
|
|
49b41667ad | ||
|
|
5646dca8b7 | ||
|
|
701b57e113 | ||
|
|
affac4cfa6 | ||
|
|
c1e5a7b172 | ||
|
|
8f67884119 | ||
|
|
0932cc1084 | ||
|
|
e7375a186f | ||
|
|
8917a4d15b | ||
|
|
87ca207107 | ||
|
|
982767ce2e | ||
|
|
fc5385aa38 | ||
|
|
39480f48ac | ||
|
|
1c1ba904a4 | ||
|
|
6cde7bef1e | ||
|
|
b9ab0ae58e | ||
|
|
2c90c5640e | ||
|
|
cb43e76660 | ||
|
|
0fc22b5cd6 | ||
|
|
a2c5553f4f | ||
|
|
63dd4a3e6c | ||
|
|
a7fc86cfbb | ||
|
|
d72aa2edfb | ||
|
|
54d361cf42 | ||
|
|
153de1aa45 | ||
|
|
fc883a5dd8 | ||
|
|
dfd735956c | ||
|
|
a77b728110 | ||
|
|
5cfffd7c8e | ||
|
|
a73c6928c3 | ||
|
|
d5dfda296f | ||
|
|
89d57f893c | ||
|
|
46371fe816 | ||
|
|
bfe4873204 | ||
|
|
acd1ed5768 | ||
|
|
b6a4fe03df | ||
|
|
b25ad07c30 | ||
|
|
ffcd53e989 | ||
|
|
4e22e0f4e7 | ||
|
|
d6a40c7523 | ||
|
|
4ee2f4e7de | ||
|
|
1567e56a09 | ||
|
|
639ccbf16c | ||
|
|
4d12abdb42 | ||
|
|
23109b8beb | ||
|
|
9882603188 | ||
|
|
aee90e5b99 | ||
|
|
5aa907feeb | ||
|
|
98c5dac781 | ||
|
|
5fab41887a | ||
|
|
c5f6e64a83 | ||
|
|
d758fd167c | ||
|
|
0f96cb66f1 | ||
|
|
3baa3d38a3 | ||
|
|
cc74af00ba | ||
|
|
4a5f0a7d83 | ||
|
|
468bc16b0f | ||
|
|
e6a472e6b7 | ||
|
|
be710ff488 | ||
|
|
aa40473481 | ||
|
|
4170b5d878 | ||
|
|
3c9642c5fc | ||
|
|
e833b27533 | ||
|
|
d17b2d8015 | ||
|
|
93c9830c23 | ||
|
|
51211fa626 | ||
|
|
1fc417e24e | ||
|
|
2e49b66ac2 | ||
|
|
4534bd6aaa | ||
|
|
de982b9ada | ||
|
|
6931a37e95 | ||
|
|
a4ed5c6e38 | ||
|
|
b94cb3eb7e | ||
|
|
8ec7f2dd9c | ||
|
|
330f1093bb | ||
|
|
8e8046af6d | ||
|
|
94e2b5edb1 | ||
|
|
c2d6faabc7 | ||
|
|
d7f294d7c2 | ||
|
|
cb72aa7996 | ||
|
|
661ef0cbe3 | ||
|
|
d47924492a | ||
|
|
e594cf4560 | ||
|
|
4c45538f6e | ||
|
|
e3e81505b9 | ||
|
|
d2c36cb85a | ||
|
|
207129447d | ||
|
|
15150fb577 | ||
|
|
993fd2563d | ||
|
|
7c0df616b9 | ||
|
|
5e9791079d | ||
|
|
4c103423ba | ||
|
|
f438ef712b | ||
|
|
0392c34e71 | ||
|
|
5041f74aad | ||
|
|
6f58ea9e21 | ||
|
|
b0cfb4e933 | ||
|
|
290ab2dfed | ||
|
|
21326dca12 | ||
|
|
6f49df9612 | ||
|
|
fca752849b | ||
|
|
8f2a28f56d | ||
|
|
17beb40283 | ||
|
|
cd76bd311d | ||
|
|
c37d61c862 | ||
|
|
d21507cd4b | ||
|
|
48078ffb25 | ||
| ef4cc5ea56 | |||
| 30469d37f0 | |||
|
|
07ef35aa4a | ||
|
|
040f6479bf | ||
|
|
ce0acdbd5b | ||
|
|
1d15c41c1b | ||
|
|
0d861ebd97 | ||
|
|
7c54f2acf7 | ||
|
|
278a915ba4 | ||
|
|
2c42a78725 | ||
|
|
07592355cb | ||
|
|
61a80b996f | ||
|
|
8c7f4cc825 | ||
|
|
7dad1c4e33 | ||
|
|
f8ceb5a849 | ||
|
|
d45af83eaa | ||
|
|
cbde035f36 | ||
|
|
0f3f6eebd8 | ||
|
|
c5845a7f70 | ||
|
|
b53aea76bd | ||
|
|
a5434a362b | ||
|
|
1922cafa14 | ||
|
|
b4c2b3e6af | ||
|
|
822b149419 | ||
|
|
edc5a02efe | ||
|
|
0867010841 | ||
|
|
031b6b3240 | ||
|
|
502c2f03ac | ||
|
|
55de4c27b8 | ||
|
|
d49ecee3a7 | ||
|
|
62e0ee6a3a | ||
|
|
6304715df3 | ||
|
|
7dbc894d2e | ||
|
|
c3a86e185d | ||
|
|
ce8061573b | ||
|
|
d0506f07a1 | ||
|
|
1d076ee6a4 | ||
|
|
0c36e79996 | ||
|
|
d0e8450225 | ||
|
|
924464423d | ||
|
|
1b2a751de7 | ||
|
|
047c59b53f | ||
|
|
863ef0259c | ||
|
|
d765190e3d | ||
|
|
8976f35c5f | ||
|
|
008eebe4b1 | ||
|
|
cbb76849ea | ||
|
|
cf930fa42e | ||
|
|
da51d37322 | ||
|
|
a09e1a0e45 | ||
|
|
1d4f64d745 | ||
|
|
2857b96c4d | ||
|
|
8b08930268 | ||
|
|
2922708c18 | ||
|
|
5c6385b4c3 | ||
|
|
e9d6ec9f23 | ||
|
|
46d7c79359 | ||
|
|
d3f16e00b5 | ||
|
|
d95a5a02cf | ||
|
|
22cb7d6d7f | ||
|
|
fc56e48781 | ||
|
|
95a9e8a82e | ||
|
|
717f4ac30a | ||
|
|
ee9207e84a | ||
|
|
9cbcca04df | ||
|
|
a5528a4bc1 | ||
|
|
e284e7c74a | ||
|
|
7a419df211 | ||
|
|
2edf6aaeba | ||
|
|
ce15c53c04 | ||
|
|
231a97a631 | ||
|
|
ed802d7602 | ||
|
|
77d2a74865 | ||
|
|
a5374cbc4a | ||
|
|
428cbc69e3 | ||
|
|
4538617d96 | ||
|
|
dc008ae89d | ||
|
|
3615571e39 | ||
|
|
ed3fcd63fc | ||
|
|
d49dac2e05 | ||
|
|
754a697099 | ||
|
|
285afd22ad | ||
|
|
50c2f72a66 | ||
|
|
42d95f2930 | ||
|
|
78bc215677 | ||
|
|
78bc8b5c74 | ||
|
|
5a5d8fd978 | ||
|
|
e09c4a0642 | ||
|
|
c1eb404095 | ||
|
|
7b3769da2a | ||
|
|
cc49e4026e | ||
|
|
385f4fd25b | ||
|
|
6c00321262 | ||
|
|
aa77f05a98 | ||
|
|
ccd590e397 | ||
|
|
4ee1e8f2fa | ||
|
|
690f30308f | ||
|
|
fea91746a4 | ||
|
|
9d8350dabc | ||
|
|
a6879ea2a5 | ||
|
|
d2b32d528f | ||
|
|
ad675495e6 | ||
|
|
48e7e38b11 | ||
|
|
dc191668fe | ||
|
|
955be3658b | ||
|
|
dcc163ec0b | ||
|
|
14716d41b9 | ||
|
|
9018313fb2 | ||
|
|
92b391a1b2 | ||
|
|
b023229600 | ||
|
|
82b71ec5b6 | ||
|
|
1edaf21bde | ||
|
|
76bd4baa9c | ||
|
|
0bd49f4227 | ||
|
|
580c3e765f | ||
|
|
7dc24b8ee6 | ||
|
|
6e6dff2ca3 | ||
|
|
3e03e81dbf | ||
|
|
9150a180fa | ||
|
|
7b6ce97940 | ||
|
|
d398795202 | ||
|
|
5f82ef4d3b | ||
|
|
d1b959818e | ||
|
|
219b24cbdb | ||
|
|
7b36910be7 | ||
|
|
4656f7fc94 | ||
|
|
1dcf9c676d | ||
|
|
9768b326aa | ||
|
|
b608cd5960 | ||
|
|
fc7aa2d342 | ||
|
|
3000740213 | ||
|
|
c5c8786363 | ||
|
|
c8250e5200 | ||
|
|
5a08d028ce | ||
|
|
6c3ec392e8 | ||
|
|
e665efb3a0 | ||
|
|
d79b48e3f7 | ||
|
|
9af0f83c89 | ||
|
|
b62138beca | ||
|
|
0c54c36c71 | ||
|
|
9b6d68f966 | ||
|
|
d36d2a5e0b | ||
|
|
e13a03ff74 | ||
|
|
c66f9a0fd4 | ||
|
|
38ef80637f | ||
|
|
7f9ac5c584 | ||
|
|
856ac5d821 | ||
|
|
3811929a6e | ||
|
|
85a97a9cab | ||
|
|
f5236f1d21 | ||
|
|
b0f172a8e1 | ||
|
|
71abe9a7e1 | ||
|
|
b4677426a4 | ||
|
|
745108dc6d | ||
|
|
448feae732 | ||
|
|
3b5ecb631d | ||
|
|
50e6278327 | ||
|
|
4dfe411bf4 | ||
|
|
5a9dc31fbb | ||
|
|
734af1c025 | ||
|
|
1000cbf491 | ||
|
|
df241dfd89 | ||
|
|
a2f02194ee | ||
|
|
fcbc006e43 | ||
|
|
edbf9f4437 | ||
|
|
e8eae27029 | ||
|
|
c4d1614cc3 | ||
|
|
8f40060583 | ||
|
|
46e386a6f8 | ||
|
|
805693f67f | ||
|
|
08c13c10c8 | ||
|
|
f72a7b88ce | ||
|
|
5c54fcb55c | ||
|
|
2f95d31092 | ||
|
|
0a77293f7d | ||
|
|
b038b58015 | ||
|
|
2d57168905 | ||
|
|
1d28475a0c | ||
|
|
7c70ae365e | ||
|
|
6ce441ecf1 | ||
|
|
995a257b2c | ||
|
|
354f679d72 | ||
|
|
e103918050 | ||
|
|
f18345ede6 | ||
|
|
4ab996e796 | ||
|
|
0668e722b6 | ||
|
|
67057adeaf | ||
|
|
e4ced531ad | ||
|
|
7bdae24ddf | ||
|
|
107bfc85a3 | ||
|
|
85536065bc | ||
|
|
efd28f38d9 | ||
|
|
4b26a5ef8a | ||
|
|
b196bb1c00 | ||
|
|
402713d4b1 | ||
|
|
f0de284dd9 | ||
|
|
9d9d6aa302 | ||
|
|
117ffb2bfd | ||
|
|
a8e7f4137a | ||
|
|
27715a5dbb | ||
|
|
7d0b0b85bc | ||
|
|
ec7e9e18ad | ||
|
|
5966b198b6 | ||
|
|
8af45bf3e6 | ||
|
|
71f8a01d48 | ||
|
|
418119a1ab | ||
|
|
1ecfdf5289 | ||
|
|
c31a2856f9 | ||
|
|
eef3d6a85a | ||
|
|
81e21a8277 | ||
|
|
011acc270a | ||
|
|
d7deafb1b9 | ||
|
|
6ec48de43f | ||
|
|
471ebafaa6 | ||
|
|
78fb4f5254 | ||
|
|
285a53e1a3 | ||
|
|
c806b868f0 | ||
|
|
9a52416b34 | ||
|
|
97ea370bf6 | ||
|
|
23e740c7f1 | ||
|
|
10ca7f0762 | ||
|
|
0f43318798 | ||
|
|
ac5aefb52b | ||
|
|
6ae4c0a9a8 | ||
|
|
abbf44bab7 | ||
|
|
af1fe52725 | ||
|
|
11b285d045 | ||
|
|
650806525a | ||
|
|
92bd9e9184 | ||
|
|
d08606dc93 | ||
|
|
5450f89612 | ||
|
|
dbf065ab12 | ||
|
|
79fa4b3ad5 | ||
|
|
228d20a72c | ||
|
|
29a34083c9 | ||
|
|
b3127478d9 | ||
|
|
3994ddf1b4 | ||
|
|
b921440d9f | ||
|
|
4fb93d1af6 | ||
|
|
5835eb6f57 | ||
|
|
88e1d56a98 | ||
|
|
1ae20fa3eb | ||
|
|
84df026878 | ||
|
|
b8a7304f4b | ||
|
|
4bf57ce1b0 | ||
|
|
1013791b35 | ||
|
|
90521c492f | ||
|
|
cd863c2819 | ||
|
|
69f6f65db6 | ||
|
|
6fc608f099 | ||
|
|
07256c0ed3 | ||
|
|
7079d70e8b | ||
|
|
da0369492e | ||
|
|
0a035c9c3b | ||
|
|
1ff868fdb7 | ||
|
|
0992662396 | ||
|
|
9d407eebf0 | ||
|
|
34c29ea88b | ||
|
|
b210675893 | ||
|
|
fa5137cad8 | ||
|
|
76a68fb071 | ||
|
|
f511ad2a67 | ||
|
|
106471225f | ||
|
|
59e56e5310 | ||
|
|
a6da0186e0 | ||
|
|
a38b90dbab | ||
|
|
54f2b34623 | ||
|
|
82dc07d7c3 | ||
|
|
737150d7e9 | ||
|
|
80794109ae | ||
|
|
277e406952 | ||
|
|
db44e0824a | ||
|
|
f8e4cdb56f | ||
|
|
453f875314 | ||
|
|
d57ec3cab1 | ||
|
|
6a285a628d | ||
|
|
8c3017634f | ||
|
|
7582b9f960 | ||
|
|
da0c52ecff | ||
|
|
c56ca2fe61 | ||
|
|
dd371c468c | ||
|
|
21bb019434 | ||
|
|
3790723483 | ||
|
|
7025cc399e | ||
|
|
ea236e94a0 | ||
|
|
e3fcff7f21 | ||
|
|
9a85f7f2c5 | ||
|
|
1b545f6bbe | ||
|
|
2d5f2dd794 | ||
|
|
bdf3d945c7 | ||
|
|
b1374f0cb8 | ||
|
|
c125e73402 | ||
|
|
9437c91bd8 | ||
|
|
e4c5509061 | ||
|
|
17f94205b4 | ||
|
|
ad50d32f54 | ||
|
|
ea82926b3f | ||
|
|
3f84d5d14c | ||
|
|
33645f9b5a | ||
|
|
8df91bd475 | ||
|
|
5c474f4256 | ||
|
|
333ed7a207 | ||
|
|
29b3c590cd | ||
|
|
eb044da8bc | ||
|
|
4a6937826c | ||
|
|
c0e62e2e79 | ||
|
|
5f41544089 | ||
|
|
e8d78f4656 | ||
|
|
3960ab31da | ||
|
|
e185e3fb47 | ||
|
|
095b311edc | ||
|
|
96b9a46c07 | ||
|
|
3037d8b507 | ||
|
|
df9df99c23 | ||
|
|
7d972e53d7 | ||
|
|
6061b76331 | ||
|
|
0900c088ee | ||
|
|
68d3799c47 | ||
|
|
1b554ebc1e | ||
|
|
732b9d3910 | ||
|
|
34b341767c | ||
|
|
070eb7ba4d | ||
|
|
57c35549f3 | ||
|
|
f2922d7dbd | ||
|
|
61016a9318 | ||
|
|
9e2fbec797 | ||
|
|
294fdee57f | ||
|
|
fe06ff5075 | ||
|
|
73ed55f288 | ||
|
|
2ff9ee9eba | ||
|
|
3b434c0ef5 | ||
|
|
6d59e55d8d | ||
|
|
d2636c8c57 | ||
|
|
b52f193804 | ||
|
|
23ce5df057 | ||
|
|
da06691292 | ||
|
|
8b6aad3a41 | ||
|
|
985c557dcd | ||
|
|
6937aa8b9d | ||
|
|
2299132cd3 | ||
|
|
005e5a49c0 | ||
|
|
85b3b8a749 | ||
|
|
42d1ba2b20 | ||
|
|
b887d50409 | ||
|
|
66b38cc7d6 | ||
|
|
dca5c2586c | ||
|
|
89f26595b6 | ||
|
|
3d39da9330 | ||
|
|
eb86d5ac00 | ||
|
|
5b01670a22 | ||
|
|
1f7f1971ec | ||
|
|
7109bac4a0 | ||
|
|
00a1803348 | ||
|
|
03dd8b8483 | ||
|
|
553adbb946 | ||
|
|
2fb556031b | ||
|
|
9e85b6dcfe | ||
|
|
0d0b1249c7 | ||
|
|
46a6d2c2ad | ||
|
|
1051be9815 | ||
|
|
f241c4546b | ||
|
|
00b88c2725 | ||
|
|
10b006ffb2 | ||
|
|
133bbc911e | ||
|
|
28272621b0 | ||
|
|
61d786cf4f | ||
|
|
9d2e62f724 | ||
|
|
6e77b59ef8 | ||
|
|
3a733d7353 | ||
|
|
8df9ed5977 | ||
|
|
cf748d6d41 | ||
|
|
8528a6a4a4 | ||
|
|
169278c9d8 | ||
|
|
db490247e0 | ||
|
|
3c45cf29c7 | ||
|
|
66acd8ed38 | ||
|
|
d1c22762c4 | ||
|
|
441c420773 | ||
|
|
edad9bb7f5 | ||
|
|
f954be3e87 | ||
|
|
cbd52d1a48 | ||
|
|
aa47642161 | ||
|
|
57c38c462c | ||
|
|
995ae3c72b | ||
|
|
9df55d5ef7 | ||
|
|
ffd659208a | ||
|
|
76b69b86dc | ||
|
|
a8e1c9d3aa | ||
|
|
3dbd13aa77 | ||
|
|
291be57cde | ||
|
|
ae375ae526 | ||
|
|
84f9f6d0cd | ||
|
|
ed24d0bd4c | ||
|
|
8eca71ac95 | ||
|
|
84c58ce9af | ||
|
|
184703998b | ||
|
|
e345c0ddcd | ||
|
|
c874e76343 | ||
|
|
687f5f2a1b | ||
|
|
c7aa70f078 | ||
|
|
690fe3f4d8 | ||
|
|
c1d0a42bed | ||
|
|
e5cd73f4a7 | ||
|
|
7ffc220891 | ||
|
|
2799fe9346 | ||
|
|
e2023b7172 | ||
|
|
9224971407 | ||
|
|
d492f80468 | ||
|
|
78bc330448 | ||
|
|
3805adc9f3 | ||
|
|
4e3989f9f5 | ||
|
|
9624b5d558 | ||
|
|
754d9187df | ||
|
|
a4996c9d3e | ||
|
|
b99c5c204b | ||
|
|
7162c5dea1 | ||
|
|
e3939a56e1 | ||
|
|
ceafe9b281 | ||
|
|
263d1ab15f | ||
|
|
e76202b355 | ||
|
|
f68eabcf35 | ||
|
|
e218b3fa96 | ||
|
|
50e21b4077 | ||
|
|
b6a1a1c0f6 | ||
|
|
15e12142ca | ||
|
|
b8c1c29df2 | ||
|
|
4a1880ead0 | ||
|
|
57b878d5e4 | ||
|
|
46aab61bf6 | ||
|
|
227dc8e48a | ||
|
|
68c3205b53 | ||
|
|
61f0a4ec85 | ||
|
|
c606a1a35a | ||
|
|
9933399dc4 | ||
|
|
5ce8023ef7 | ||
|
|
6438c75faf | ||
|
|
712097ba45 | ||
|
|
9a5cb749ed | ||
|
|
d730f23027 | ||
|
|
9f161f2baf | ||
|
|
2570ae2cf6 | ||
|
|
8d5c2d8cbb | ||
|
|
28bf0a437c | ||
|
|
fd78e6d886 | ||
|
|
e84d963ab6 | ||
|
|
c4fbc2962e | ||
|
|
a0af851d9d | ||
|
|
477d08b5da | ||
|
|
50a07df5d2 | ||
|
|
b3cdde78b1 | ||
|
|
8207b16598 | ||
|
|
f61f2af2f4 | ||
|
|
90805cb526 | ||
|
|
04fe80b498 | ||
|
|
60657c9809 | ||
|
|
3150b91f80 | ||
|
|
7b854d49e3 | ||
|
|
81a48b8f79 | ||
|
|
655ea6f06f | ||
|
|
efe9aadb39 | ||
|
|
c0e1709d57 | ||
|
|
ea8a4fbb1e | ||
|
|
093963dadd | ||
|
|
df429e8c34 | ||
|
|
d3c45adf34 | ||
|
|
043a0f7ab3 | ||
|
|
574b6bc214 | ||
|
|
89e4a5a460 | ||
|
|
2ebc75944e | ||
|
|
acedd48708 | ||
|
|
9fd9d51010 | ||
|
|
9f8423f2eb | ||
|
|
8242c969e3 | ||
|
|
7cd5c16fca | ||
|
|
290ddc1fea | ||
|
|
47a7013029 | ||
|
|
a4de005a54 | ||
|
|
7ace3a5956 | ||
|
|
5abbf168a9 | ||
|
|
4d69a0bc33 | ||
|
|
222eca4099 | ||
|
|
ea75558ac2 | ||
|
|
876cae8239 | ||
|
|
7c3951e3a0 | ||
|
|
552f060612 | ||
|
|
e7cfa35288 | ||
|
|
4326cd1380 | ||
|
|
19b40bc59e | ||
|
|
8c9a9a111e | ||
|
|
b13540123b | ||
|
|
f5b2de62b6 | ||
|
|
1ba023fc83 | ||
|
|
36e9e92df1 | ||
|
|
0f1dac50f1 | ||
|
|
eb0346e02a | ||
|
|
cae621918e | ||
|
|
f28983e5a7 | ||
|
|
b35cdb70df | ||
|
|
80361a121e | ||
|
|
b4ff007e42 | ||
|
|
ae379d865c | ||
|
|
0a3876dc8c | ||
|
|
e2d95acfa1 | ||
|
|
5ac1a6a653 | ||
|
|
481c409617 | ||
|
|
e8869b4d87 | ||
|
|
b1b95e086d | ||
|
|
df1f2d4e4b | ||
|
|
18b0ef2d1b | ||
|
|
860c100a04 | ||
|
|
9040764cf1 | ||
|
|
d8d4bc76cd | ||
|
|
fbcfa54061 | ||
|
|
b235f77f39 | ||
|
|
1b4b0a8d8d | ||
|
|
7ea0dc241c | ||
|
|
39e67df44d | ||
|
|
cb17e5a662 | ||
|
|
ea67f24051 | ||
| 7e2e1d5adf | |||
|
|
c49d231875 | ||
| d839d80c2e | |||
|
|
cc826cbcb5 | ||
|
|
f59702054d | ||
|
|
06dfcf70be | ||
|
|
44785a49da | ||
|
|
e20448d6b5 | ||
|
|
6014b51850 | ||
|
|
5acda6a0c0 | ||
|
|
f4f028a773 | ||
|
|
66c98bc820 | ||
|
|
de1d255088 | ||
|
|
b381ad2905 | ||
|
|
a4894b8257 | ||
|
|
b7dbb919a9 | ||
|
|
a08491c122 | ||
|
|
8795f50bb8 | ||
|
|
3ab42f9b72 | ||
|
|
7bce19452e | ||
|
|
1ff028e99f | ||
|
|
5450dcca0b | ||
|
|
d2ee0a481f | ||
|
|
1d95e01882 | ||
|
|
ebc356d1c7 | ||
|
|
54708a6167 | ||
|
|
60f01f7fff | ||
|
|
9f8860f003 | ||
|
|
be19d1eeef | ||
|
|
1d02b8588b | ||
|
|
487774eac9 | ||
|
|
d710eb9cea | ||
|
|
0385ca46dc | ||
|
|
e7c3f99e28 | ||
|
|
14d5c4e43d | ||
|
|
a9330b424b | ||
|
|
5486c93f4c | ||
|
|
b9508450b7 | ||
|
|
7384fd8270 | ||
|
|
7ea2452ae4 | ||
|
|
c3b60f1b4d | ||
|
|
193b3b5a81 | ||
|
|
9bd4257fcf | ||
|
|
5bf8efac03 | ||
|
|
bd9e12b993 | ||
|
|
78595a3d1f | ||
|
|
71df0bc553 | ||
|
|
c4c07fc9cc | ||
|
|
da5d1c5979 | ||
|
|
95fa0bbf83 | ||
|
|
afd4a806a6 | ||
|
|
e6612d2893 | ||
|
|
298f60f1b0 | ||
|
|
7794272f8e | ||
|
|
8e5a1a3848 | ||
|
|
3810080be2 | ||
|
|
74dd8e11a3 | ||
|
|
a2efc493b2 | ||
|
|
0d7dfc082d | ||
|
|
aecd658d2b | ||
|
|
89e2057a8e | ||
|
|
2ecb1c4ef4 | ||
|
|
dff7355f3b | ||
|
|
5d11fe5cc1 | ||
|
|
233157930b | ||
|
|
ae76f82ee0 | ||
|
|
00930b217d | ||
|
|
0e322b3563 | ||
|
|
2745eefec7 | ||
|
|
2357e31965 | ||
|
|
3467a44761 | ||
|
|
f6e4820fdb | ||
|
|
0070fd6b34 | ||
|
|
665ce3e5ff | ||
|
|
aed6caca92 | ||
|
|
f762a975b1 | ||
|
|
e6112a7921 | ||
|
|
53c8ee5e9e | ||
|
|
c2f661eba6 | ||
|
|
ad01f1a481 | ||
|
|
60b56599cb | ||
|
|
b4f15db397 | ||
|
|
6ba25fa945 | ||
|
|
aa9e1afdf2 | ||
|
|
167bb31084 | ||
|
|
b9f8bf30f7 | ||
|
|
64c1fc03f9 | ||
|
|
b78c854700 | ||
|
|
eef4987b0b | ||
|
|
66cae4bb7c | ||
|
|
778b9f0ebf | ||
|
|
f88b887074 | ||
|
|
45d6463315 | ||
|
|
d95fbdc2b2 | ||
|
|
d7e48187b1 | ||
|
|
7d70189bee | ||
|
|
eb1244a415 | ||
|
|
e49e05ad69 | ||
|
|
75f8a07ca4 | ||
|
|
af003d0e19 | ||
|
|
321ccc0794 | ||
|
|
cab31cb726 | ||
|
|
40d8d05be7 | ||
|
|
5e0e2c4af9 | ||
|
|
e539c6536f | ||
|
|
67d5bfcfce | ||
|
|
82fe74743c | ||
|
|
5376e09963 | ||
|
|
1a41159142 | ||
|
|
e113080a0e | ||
|
|
5faf063f96 | ||
|
|
c900bc24b2 | ||
|
|
a2de8d6e53 | ||
|
|
f6b78ad094 | ||
|
|
83b6d1ac72 | ||
|
|
f5a18cc4f4 | ||
|
|
c583ea42eb | ||
|
|
4c16f97e73 | ||
|
|
b8a4766773 | ||
|
|
a8600a598b | ||
|
|
da7efb27a9 | ||
|
|
ba40b0752f | ||
|
|
0d1b78d808 | ||
|
|
32db15ae50 | ||
|
|
bab686d8c4 | ||
|
|
51fe905b47 | ||
|
|
a11becfb2e | ||
|
|
f4af2a352a | ||
|
|
7b698e1fe1 | ||
|
|
f527a2639b | ||
|
|
5bd4bcf727 | ||
|
|
89c8338067 | ||
|
|
3f42584fbd | ||
|
|
1272ce4bb9 | ||
|
|
f0e2e72efc | ||
|
|
c392cdf257 | ||
|
|
856385d5c0 | ||
|
|
b70e6b500e | ||
|
|
5a789c2d7f | ||
|
|
0fa10d83b9 | ||
|
|
28d726626d | ||
|
|
45cc690167 | ||
|
|
4e76c2bb5d | ||
|
|
88e0e4fed8 | ||
|
|
d584d04691 | ||
|
|
1a3ba21245 | ||
|
|
ec80e09b0b | ||
|
|
de539e218a | ||
|
|
1cb0c16ff4 | ||
|
|
2432a2ed51 | ||
|
|
9a1ec1884f | ||
|
|
b6efa12bef | ||
|
|
c59f239999 | ||
|
|
9ac902b62b | ||
|
|
42469038a8 | ||
|
|
de5045d88e | ||
|
|
54154e8298 | ||
|
|
364d0cf703 | ||
|
|
f70415d7db | ||
|
|
45a2156c12 | ||
|
|
fc4b29631f | ||
|
|
dc137ef1a6 |
66
.clang-format
Normal file
66
.clang-format
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
# works with clang-format-3.7
|
||||||
|
#
|
||||||
|
# Most controversial fields are UseTab, IndentWidth, ContinuationIndentWidth, ColumnLimit, BreakBeforeBraces
|
||||||
|
|
||||||
|
#BasedOnStyle: Google
|
||||||
|
AccessModifierOffset: 0
|
||||||
|
AlignAfterOpenBracket: true
|
||||||
|
AlignConsecutiveAssignments: false
|
||||||
|
#AlignConsecutiveDeclarations: false
|
||||||
|
AlignEscapedNewlinesLeft: true
|
||||||
|
AlignOperands: true
|
||||||
|
AlignTrailingComments: true
|
||||||
|
AllowAllParametersOfDeclarationOnNextLine: false
|
||||||
|
AllowShortBlocksOnASingleLine: false
|
||||||
|
AllowShortCaseLabelsOnASingleLine: false
|
||||||
|
AllowShortFunctionsOnASingleLine: Empty
|
||||||
|
AllowShortIfStatementsOnASingleLine: false
|
||||||
|
AllowShortLoopsOnASingleLine: false
|
||||||
|
AlwaysBreakAfterDefinitionReturnType: None
|
||||||
|
AlwaysBreakBeforeMultilineStrings: false
|
||||||
|
AlwaysBreakTemplateDeclarations: false
|
||||||
|
BinPackArguments: false
|
||||||
|
BinPackParameters: false
|
||||||
|
BreakBeforeBinaryOperators: None
|
||||||
|
BreakBeforeBraces: Linux
|
||||||
|
BreakBeforeTernaryOperators: true
|
||||||
|
BreakConstructorInitializersBeforeComma: false
|
||||||
|
ColumnLimit: 120
|
||||||
|
CommentPragmas: ''
|
||||||
|
ConstructorInitializerAllOnOneLineOrOnePerLine: true
|
||||||
|
ConstructorInitializerIndentWidth: 0
|
||||||
|
ContinuationIndentWidth: 4
|
||||||
|
Cpp11BracedListStyle: true
|
||||||
|
DerivePointerAlignment: false
|
||||||
|
DerivePointerBinding: false
|
||||||
|
DisableFormat: false
|
||||||
|
ForEachMacros: ['foreach']
|
||||||
|
IndentCaseLabels: false
|
||||||
|
IndentFunctionDeclarationAfterType: false
|
||||||
|
IndentWidth: 4
|
||||||
|
IndentWrappedFunctionNames: false
|
||||||
|
Language: Cpp
|
||||||
|
MaxEmptyLinesToKeep: 1
|
||||||
|
NamespaceIndentation: None
|
||||||
|
ObjCSpaceAfterProperty: true
|
||||||
|
ObjCSpaceBeforeProtocolList: true
|
||||||
|
PenaltyBreakBeforeFirstCallParameter: 1000
|
||||||
|
PenaltyBreakComment: 100
|
||||||
|
PenaltyBreakFirstLessLess: 0
|
||||||
|
PenaltyBreakString: 100
|
||||||
|
PenaltyExcessCharacter: 100
|
||||||
|
PenaltyReturnTypeOnItsOwnLine: 2000
|
||||||
|
PointerAlignment: Right
|
||||||
|
SpaceAfterCStyleCast: false
|
||||||
|
SpaceBeforeAssignmentOperators: true
|
||||||
|
SpaceBeforeParens: ControlStatements
|
||||||
|
SpaceInEmptyParentheses: false
|
||||||
|
SpacesBeforeTrailingComments: 1
|
||||||
|
SpacesInAngles: false
|
||||||
|
SpacesInCStyleCastParentheses: false
|
||||||
|
SpacesInContainerLiterals: false
|
||||||
|
SpacesInParentheses: false
|
||||||
|
SpacesInSquareBrackets: false
|
||||||
|
Standard: Cpp11
|
||||||
|
TabWidth: 4
|
||||||
|
UseTab: Never
|
||||||
3
.gitignore
vendored
Normal file
3
.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
build
|
||||||
|
*.user
|
||||||
|
version.h
|
||||||
33
AUTHORS
33
AUTHORS
@@ -1,14 +1,15 @@
|
|||||||
|
|
||||||
tint2 is developped by :
|
tint2 is developped by :
|
||||||
- Thierry Lorthiois <lorthiois@bbsoft.fr> from Omega distribution
|
- Thierry Lorthiois <lorthiois@bbsoft.fr> from Omega distribution
|
||||||
- Andreas Fink <andreas.fink85@googlemail.com>
|
- Andreas Fink <andreas.fink85@googlemail.com>
|
||||||
- Euan Freeman <euan04@gmail.com> (tintwizard)
|
- Euan Freeman <euan04@gmail.com> (tintwizard)
|
||||||
- Christian Ruppert <Spooky85@gmail.com> (autotools build system)
|
- Christian Ruppert <Spooky85@gmail.com> (autotools build system)
|
||||||
|
- Ovidiu M <mrovi9000 at gmail.com> : launcher, bug fixes
|
||||||
|
- Mishael A Sibiryakov (death@junki.org) : freespace
|
||||||
|
- Sebastian Reichel <sre@ring0.de> : battery, various fixes, debian package maintainer
|
||||||
|
|
||||||
tint2 is based on ttm source code (http://code.google.com/p/ttm/)
|
tint2 is based on the ttm source code (http://code.google.com/p/ttm/)
|
||||||
- 2007-2008 Pål Staurland <staura@gmail.com>
|
- 2007-2008 Pål Staurland <staura@gmail.com>
|
||||||
|
|
||||||
|
|
||||||
Contributors:
|
Contributors:
|
||||||
Kwaku Yeboah <kwakuyeboah@gmail.com> : wiki page
|
Kwaku Yeboah <kwakuyeboah@gmail.com> : wiki page
|
||||||
Daniel Moerner <dmoerner@gmail.com> : man page and debian package
|
Daniel Moerner <dmoerner@gmail.com> : man page and debian package
|
||||||
@@ -16,5 +17,29 @@ Contributors:
|
|||||||
James Buren <ryuo@frugalware.org> : Frugalware package
|
James Buren <ryuo@frugalware.org> : Frugalware package
|
||||||
Pierre-Emmanuel Andre <pea@raveland.org> : openbsd port
|
Pierre-Emmanuel Andre <pea@raveland.org> : openbsd port
|
||||||
Redroar : arch package
|
Redroar : arch package
|
||||||
|
Rene Garcia <garciamx@gmail.com> : launcher SVG support
|
||||||
|
Marcelo Vianna : taskbar sorting
|
||||||
|
Xico Atelo : startup notifications
|
||||||
|
Craig Oakes : WM flags, issue tracker organization
|
||||||
|
Jeff Blake (https://gitlab.com/u/berkley4) : more mouse event handlers
|
||||||
|
Vladimir <vladimir-csp@yandex.ru> : translations, bug reports
|
||||||
|
Christophe D. <stophe72.d@gmail.com> : non-rectangular borders
|
||||||
|
Benoit Averty : taskbar enhancements
|
||||||
|
Justin Jacobs : tint2conf fixes
|
||||||
|
Oskari Rauta : separator plugin, gradients
|
||||||
|
Michael Messmore : Support for Path in .desktop files
|
||||||
|
Matthew Otnel : config option systray_name_filter
|
||||||
|
|
||||||
|
Translations:
|
||||||
|
Bosnian:
|
||||||
|
Dino Duratović <dinomol@mail.com>
|
||||||
|
Croatian:
|
||||||
|
Dino Duratović <dinomol@mail.com>
|
||||||
|
French:
|
||||||
|
Jocelyn <anechampenois@solydxk.com>
|
||||||
|
The Bento team <meets@gmx.fr>
|
||||||
|
Wagner <https://gitlab.com/u/wagmic>
|
||||||
|
Polish:
|
||||||
|
Daniel Napora <napcok@gmail.com>
|
||||||
|
Serbian:
|
||||||
|
Dino Duratović <dinomol@mail.com>
|
||||||
|
|||||||
210
CMakeLists.txt
210
CMakeLists.txt
@@ -1,8 +1,29 @@
|
|||||||
project( tint2 )
|
project( tint2 )
|
||||||
cmake_minimum_required( VERSION 2.6 )
|
cmake_minimum_required( VERSION 2.8.5 )
|
||||||
|
|
||||||
|
option( ENABLE_BATTERY "Enable battery status plugin" ON )
|
||||||
|
option( ENABLE_TINT2CONF "Enable tint2conf build, a GTK+2 theme configurator for tint2" ON )
|
||||||
|
option( ENABLE_EXTRA_THEMES "Install additional tint2 themes" ON )
|
||||||
|
option( ENABLE_RSVG "Rsvg support (launcher only)" ON )
|
||||||
|
option( ENABLE_SN "Startup notification support" ON )
|
||||||
|
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_ON_SIGNAL "Dump a backtrace also when receiving signals such as SIGSEGV" OFF )
|
||||||
|
if( CMAKE_SYSTEM_NAME STREQUAL "Linux" )
|
||||||
|
option( ENABLE_UEVENT "Kernel event handling support" ON )
|
||||||
|
endif( CMAKE_SYSTEM_NAME STREQUAL "Linux" )
|
||||||
|
|
||||||
|
include( GNUInstallDirs )
|
||||||
|
if(NOT docdir)
|
||||||
|
set(docdir ${CMAKE_INSTALL_DOCDIR})
|
||||||
|
endif()
|
||||||
|
if(NOT htmldir)
|
||||||
|
set(htmldir ${docdir}/html)
|
||||||
|
endif()
|
||||||
|
|
||||||
include( FindPkgConfig )
|
include( FindPkgConfig )
|
||||||
include( CheckLibraryExists )
|
include( CheckLibraryExists )
|
||||||
|
include( CheckCSourceCompiles )
|
||||||
pkg_check_modules( X11 REQUIRED x11 xcomposite xdamage xinerama xrender xrandr>=1.3 )
|
pkg_check_modules( X11 REQUIRED x11 xcomposite xdamage xinerama 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 )
|
||||||
@@ -10,20 +31,65 @@ pkg_check_modules( CAIRO REQUIRED cairo )
|
|||||||
pkg_check_modules( GLIB2 REQUIRED glib-2.0 )
|
pkg_check_modules( GLIB2 REQUIRED glib-2.0 )
|
||||||
pkg_check_modules( GOBJECT2 REQUIRED gobject-2.0 )
|
pkg_check_modules( GOBJECT2 REQUIRED gobject-2.0 )
|
||||||
pkg_check_modules( IMLIB2 REQUIRED imlib2>=1.4.2 )
|
pkg_check_modules( IMLIB2 REQUIRED imlib2>=1.4.2 )
|
||||||
|
|
||||||
|
if(ENABLE_BACKTRACE)
|
||||||
|
check_c_source_compiles(
|
||||||
|
"#include <stdlib.h>\n#include <execinfo.h>\nint main () { backtrace(NULL, 0); }"
|
||||||
|
BACKTRACE_LIBC)
|
||||||
|
|
||||||
|
if(BACKTRACE_LIBC)
|
||||||
|
set(BACKTRACE_LIBC_FOUND TRUE)
|
||||||
|
set(BACKTRACE_L_FLAGS "-rdynamic")
|
||||||
|
else()
|
||||||
|
pkg_check_modules( UNWIND libunwind )
|
||||||
|
find_library(EXECINFO_LIBRARIES NAMES execinfo)
|
||||||
|
if(EXECINFO_LIBRARIES OR EXECINFO_LIBRARIES_FOUND)
|
||||||
|
set(EXECINFO_FOUND TRUE)
|
||||||
|
set(EXECINFO_LIBRARIES "-lexecinfo")
|
||||||
|
set(BACKTRACE_L_FLAGS "-rdynamic")
|
||||||
|
else()
|
||||||
|
set(EXECINFO_LIBRARIES "")
|
||||||
|
set(BACKTRACE_L_FLAGS "")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if( NOT BACKTRACE_LIBC_FOUND AND NOT UNWIND_FOUND AND NOT EXECINFO_FOUND )
|
||||||
|
message( WARNING "Backtrace support not available. You can enable it by installing libexecinfo or libunwind." )
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
set(EXECINFO_LIBRARIES "")
|
||||||
|
set(BACKTRACE_L_FLAGS "")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if( ENABLE_RSVG )
|
||||||
|
pkg_check_modules( RSVG librsvg-2.0>=2.14.0 )
|
||||||
|
endif( ENABLE_RSVG )
|
||||||
|
|
||||||
|
if( ENABLE_SN )
|
||||||
pkg_check_modules( SN libstartup-notification-1.0>=0.12 )
|
pkg_check_modules( SN libstartup-notification-1.0>=0.12 )
|
||||||
|
endif(ENABLE_SN)
|
||||||
|
|
||||||
find_library( RT_LIBRARY rt )
|
find_library( RT_LIBRARY rt )
|
||||||
|
|
||||||
if( NOT X11_FOUND OR NOT PANGOCAIRO_FOUND OR NOT PANGO_FOUND OR NOT CAIRO_FOUND OR NOT GLIB2_FOUND OR NOT GOBJECT2_FOUND OR NOT IMLIB2_FOUND )
|
if( NOT X11_FOUND OR NOT PANGOCAIRO_FOUND OR NOT PANGO_FOUND OR NOT CAIRO_FOUND OR NOT GLIB2_FOUND OR NOT GOBJECT2_FOUND OR NOT IMLIB2_FOUND )
|
||||||
message( FATAL_ERROR "Not all dependencies fulfilled. See http://code.google.com/p/tint2/wiki/Install" )
|
message( FATAL_ERROR "Not all dependencies fulfilled. See https://gitlab.com/o9000/tint2/wikis/Install" )
|
||||||
endif( NOT X11_FOUND OR NOT PANGOCAIRO_FOUND OR NOT PANGO_FOUND OR NOT CAIRO_FOUND OR NOT GLIB2_FOUND OR NOT GOBJECT2_FOUND OR NOT IMLIB2_FOUND )
|
endif( NOT X11_FOUND OR NOT PANGOCAIRO_FOUND OR NOT PANGO_FOUND OR NOT CAIRO_FOUND OR NOT GLIB2_FOUND OR NOT GOBJECT2_FOUND OR NOT IMLIB2_FOUND )
|
||||||
|
|
||||||
string( REPLACE ";" " " FLAGS_REPLACED "${IMLIB2_LDFLAGS}" )
|
string( REPLACE ";" " " FLAGS_REPLACED "${IMLIB2_LDFLAGS}" )
|
||||||
set( CMAKE_REQUIRED_FLAGS "${FLAGS_REPLACED}" )
|
set( CMAKE_REQUIRED_FLAGS "${FLAGS_REPLACED}" )
|
||||||
check_library_exists( "${IMLIB2_LIBRARIES}" "imlib_context_set_display" "${IMLIB2_LIBRARY_DIRS}" IMLIB_BUILD_WITH_X )
|
check_library_exists( "${IMLIB2_LIBRARIES}" "imlib_context_set_display" "${IMLIB2_LIBRARY_DIRS}" IMLIB_BUILD_WITH_X )
|
||||||
if( NOT IMLIB_BUILD_WITH_X )
|
if( NOT IMLIB_BUILD_WITH_X )
|
||||||
message( FATAL_ERROR "Imlib is not build with x support" )
|
message( FATAL_ERROR "Imlib is not built with X support" )
|
||||||
endif( NOT IMLIB_BUILD_WITH_X )
|
endif( NOT IMLIB_BUILD_WITH_X )
|
||||||
|
|
||||||
|
|
||||||
|
if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
add_definitions( -D_POSIX_C_SOURCE=200809L -D_BSD_SOURCE -D_DEFAULT_SOURCE )
|
||||||
|
else(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
add_definitions( -D_WITH_GETLINE )
|
||||||
|
endif(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
|
||||||
|
|
||||||
include_directories( ${PROJECT_BINARY_DIR}
|
include_directories( ${PROJECT_BINARY_DIR}
|
||||||
src
|
src
|
||||||
src/battery
|
src/battery
|
||||||
@@ -33,6 +99,10 @@ include_directories( ${PROJECT_BINARY_DIR}
|
|||||||
src/launcher
|
src/launcher
|
||||||
src/tooltip
|
src/tooltip
|
||||||
src/util
|
src/util
|
||||||
|
src/execplugin
|
||||||
|
src/button
|
||||||
|
src/freespace
|
||||||
|
src/separator
|
||||||
${X11_INCLUDE_DIRS}
|
${X11_INCLUDE_DIRS}
|
||||||
${PANGOCAIRO_INCLUDE_DIRS}
|
${PANGOCAIRO_INCLUDE_DIRS}
|
||||||
${PANGO_INCLUDE_DIRS}
|
${PANGO_INCLUDE_DIRS}
|
||||||
@@ -40,7 +110,8 @@ include_directories( ${PROJECT_BINARY_DIR}
|
|||||||
${GLIB2_INCLUDE_DIRS}
|
${GLIB2_INCLUDE_DIRS}
|
||||||
${GOBJECT2_INCLUDE_DIRS}
|
${GOBJECT2_INCLUDE_DIRS}
|
||||||
${IMLIB2_INCLUDE_DIRS}
|
${IMLIB2_INCLUDE_DIRS}
|
||||||
${SN_INCLUDE_DIRS} )
|
${RSVG_INCLUDE_DIRS}
|
||||||
|
${SN_INCLUDE_DIRS} )
|
||||||
|
|
||||||
set( SOURCES src/config.c
|
set( SOURCES src/config.c
|
||||||
src/panel.c
|
src/panel.c
|
||||||
@@ -49,36 +120,88 @@ set( SOURCES src/config.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
|
||||||
|
src/launcher/apps-common.c
|
||||||
|
src/launcher/icon-theme-common.c
|
||||||
src/launcher/xsettings-client.c
|
src/launcher/xsettings-client.c
|
||||||
src/launcher/xsettings-common.c
|
src/launcher/xsettings-common.c
|
||||||
src/taskbar/task.c
|
src/taskbar/task.c
|
||||||
src/taskbar/taskbar.c
|
src/taskbar/taskbar.c
|
||||||
src/taskbar/taskbarname.c
|
src/taskbar/taskbarname.c
|
||||||
src/tooltip/tooltip.c
|
src/tooltip/tooltip.c
|
||||||
|
src/execplugin/execplugin.c
|
||||||
|
src/button/button.c
|
||||||
|
src/freespace/freespace.c
|
||||||
|
src/separator/separator.c
|
||||||
|
src/tint2rc.c
|
||||||
src/util/area.c
|
src/util/area.c
|
||||||
src/util/common.c
|
src/util/common.c
|
||||||
|
src/util/strnatcmp.c
|
||||||
src/util/timer.c
|
src/util/timer.c
|
||||||
|
src/util/cache.c
|
||||||
|
src/util/color.c
|
||||||
|
src/util/gradient.c
|
||||||
src/util/window.c )
|
src/util/window.c )
|
||||||
|
|
||||||
option( ENABLE_BATTERY "Enable battery status plugin" ON )
|
|
||||||
option( ENABLE_TINT2CONF "Enable tint2conf build, a GTK+2 theme switcher for tint2" ON )
|
|
||||||
option( ENABLE_EXAMPLES "Install additional tin2rc examples" OFF )
|
|
||||||
option( ENABLE_SN "Startup notification support" ON )
|
|
||||||
if( ENABLE_SN )
|
|
||||||
if( SN_FOUND )
|
|
||||||
add_definitions( -DHAVE_SN -DSN_API_NOT_YET_FROZEN )
|
|
||||||
endif( SN_FOUND )
|
|
||||||
endif( ENABLE_SN)
|
|
||||||
|
|
||||||
if( ENABLE_BATTERY )
|
if( ENABLE_BATTERY )
|
||||||
set( SOURCES ${SOURCES} src/battery/battery.c )
|
set( SOURCES ${SOURCES} src/battery/battery.c)
|
||||||
|
|
||||||
|
if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/linux.c)
|
||||||
|
elseif(CMAKE_SYSTEM_NAME STREQUAL "DragonFly")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/freebsd.c)
|
||||||
|
elseif(CMAKE_SYSTEM_NAME STREQUAL "FreeBSD")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/freebsd.c)
|
||||||
|
elseif(CMAKE_SYSTEM_NAME STREQUAL "OpenBSD")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/openbsd.c)
|
||||||
|
elseif(CMAKE_SYSTEM_NAME STREQUAL "NetBSD")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/openbsd.c)
|
||||||
|
else(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
set( SOURCES ${SOURCES} src/battery/dummy.c)
|
||||||
|
endif(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||||||
|
|
||||||
add_definitions( -DENABLE_BATTERY )
|
add_definitions( -DENABLE_BATTERY )
|
||||||
endif( ENABLE_BATTERY )
|
endif( ENABLE_BATTERY )
|
||||||
|
|
||||||
set( MANDIR share/man CACHE PATH "Directory for man pages" )
|
if( ENABLE_RSVG )
|
||||||
set( DATADIR share CACHE PATH "Directory for shared data" )
|
if( RSVG_FOUND )
|
||||||
set( SYSCONFDIR /etc CACHE PATH "Directory for configuration files" )
|
add_definitions( -DHAVE_RSVG )
|
||||||
set( DOCDIR share/doc/tint2 CACHE PATH "Directory for documentation files" )
|
else()
|
||||||
|
message( FATAL_ERROR "SVG support enabled yet dependency not fulfilled: librsvg-2.0" )
|
||||||
|
endif( RSVG_FOUND )
|
||||||
|
endif( ENABLE_RSVG )
|
||||||
|
|
||||||
|
if( ENABLE_SN )
|
||||||
|
if( SN_FOUND )
|
||||||
|
add_definitions( -DHAVE_SN -DSN_API_NOT_YET_FROZEN )
|
||||||
|
else()
|
||||||
|
message( FATAL_ERROR "Startup notification support enabled yet dependency not fulfilled: libstartup-notification-1.0" )
|
||||||
|
endif( SN_FOUND )
|
||||||
|
endif( ENABLE_SN)
|
||||||
|
|
||||||
|
if( ENABLE_UEVENT )
|
||||||
|
add_definitions( -DENABLE_UEVENT )
|
||||||
|
set( SOURCES ${SOURCES} src/util/uevent.c)
|
||||||
|
endif( ENABLE_UEVENT )
|
||||||
|
|
||||||
|
if(ENABLE_BACKTRACE)
|
||||||
|
if(BACKTRACE_LIBC_FOUND)
|
||||||
|
add_definitions( -DENABLE_EXECINFO )
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if( UNWIND_FOUND )
|
||||||
|
add_definitions( -DENABLE_LIBUNWIND )
|
||||||
|
endif( UNWIND_FOUND )
|
||||||
|
|
||||||
|
if( EXECINFO_FOUND )
|
||||||
|
add_definitions( -DENABLE_EXECINFO )
|
||||||
|
endif( EXECINFO_FOUND )
|
||||||
|
|
||||||
|
if(ENABLE_BACKTRACE_ON_SIGNAL)
|
||||||
|
add_definitions( -DBACKTRACE_ON_SIGNAL )
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
add_definitions( -DDISABLE_BACKTRACE )
|
||||||
|
endif()
|
||||||
|
|
||||||
if( ENABLE_TINT2CONF )
|
if( ENABLE_TINT2CONF )
|
||||||
add_definitions( -DHAVE_VERSION_H )
|
add_definitions( -DHAVE_VERSION_H )
|
||||||
@@ -86,7 +209,15 @@ if( ENABLE_TINT2CONF )
|
|||||||
add_dependencies( tint2conf version )
|
add_dependencies( tint2conf version )
|
||||||
endif( ENABLE_TINT2CONF )
|
endif( ENABLE_TINT2CONF )
|
||||||
|
|
||||||
add_custom_target( version ALL "${PROJECT_SOURCE_DIR}/get_svnrev.sh" "\"${PROJECT_SOURCE_DIR}\"" )
|
if( ENABLE_ASAN )
|
||||||
|
SET(ASAN_C_FLAGS " -O0 -g3 -gdwarf-2 -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 ")
|
||||||
|
else()
|
||||||
|
SET(ASAN_C_FLAGS "")
|
||||||
|
SET(ASAN_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}
|
||||||
@@ -95,7 +226,8 @@ link_directories( ${X11_LIBRARY_DIRS}
|
|||||||
${GLIB2_LIBRARY_DIRS}
|
${GLIB2_LIBRARY_DIRS}
|
||||||
${GOBJECT2_LIBRARY_DIRS}
|
${GOBJECT2_LIBRARY_DIRS}
|
||||||
${IMLIB2_LIBRARY_DIRS}
|
${IMLIB2_LIBRARY_DIRS}
|
||||||
${SN_LIBRARY_DIRS} )
|
${RSVG_LIBRARY_DIRS}
|
||||||
|
${SN_LIBRARY_DIRS} )
|
||||||
add_executable(tint2 ${SOURCES})
|
add_executable(tint2 ${SOURCES})
|
||||||
target_link_libraries( tint2 ${X11_LIBRARIES}
|
target_link_libraries( tint2 ${X11_LIBRARIES}
|
||||||
${PANGOCAIRO_LIBRARIES}
|
${PANGOCAIRO_LIBRARIES}
|
||||||
@@ -103,22 +235,34 @@ target_link_libraries( tint2 ${X11_LIBRARIES}
|
|||||||
${CAIRO_LIBRARIES}
|
${CAIRO_LIBRARIES}
|
||||||
${GLIB2_LIBRARIES}
|
${GLIB2_LIBRARIES}
|
||||||
${GOBJECT2_LIBRARIES}
|
${GOBJECT2_LIBRARIES}
|
||||||
${IMLIB2_LIBRARIES}
|
${IMLIB2_LIBRARIES}
|
||||||
${SN_LIBRARIES} )
|
${UNWIND_LIBRARIES}
|
||||||
|
${EXECINFO_LIBRARIES} )
|
||||||
|
if( ENABLE_RSVG )
|
||||||
|
target_link_libraries( tint2 ${RSVG_LIBRARIES} )
|
||||||
|
endif( ENABLE_RSVG )
|
||||||
|
if( ENABLE_SN )
|
||||||
|
target_link_libraries( tint2 ${SN_LIBRARIES} )
|
||||||
|
endif( ENABLE_SN )
|
||||||
if( RT_LIBRARY )
|
if( RT_LIBRARY )
|
||||||
target_link_libraries( tint2 ${RT_LIBRARY} )
|
target_link_libraries( tint2 ${RT_LIBRARY} )
|
||||||
endif( RT_LIBRARY )
|
endif( RT_LIBRARY )
|
||||||
|
|
||||||
|
target_link_libraries( tint2 m )
|
||||||
|
|
||||||
add_dependencies( tint2 version )
|
add_dependencies( tint2 version )
|
||||||
set_target_properties( tint2 PROPERTIES COMPILE_FLAGS "-Wall -pthread" )
|
set_target_properties( tint2 PROPERTIES COMPILE_FLAGS "-Wall -Wpointer-arith -fno-strict-aliasing -pthread -std=c99 ${ASAN_C_FLAGS}" )
|
||||||
set_target_properties(tint2 PROPERTIES LINK_FLAGS "-pthread" )
|
set_target_properties( tint2 PROPERTIES LINK_FLAGS "-pthread -fno-strict-aliasing ${ASAN_L_FLAGS} ${BACKTRACE_L_FLAGS}" )
|
||||||
|
|
||||||
install( TARGETS tint2 DESTINATION bin )
|
install( TARGETS tint2 DESTINATION bin )
|
||||||
install( FILES sample/tint2rc DESTINATION ${SYSCONFDIR}/xdg/tint2 )
|
install( FILES tint2.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps )
|
||||||
install( FILES default_icon.png DESTINATION ${DATADIR}/tint2 )
|
install( FILES tint2.desktop DESTINATION ${CMAKE_INSTALL_DATADIR}/applications )
|
||||||
install( FILES AUTHORS ChangeLog README DESTINATION ${DOCDIR} )
|
install( FILES themes/tint2rc DESTINATION /etc/xdg/tint2 )
|
||||||
install( FILES doc/tint2.1 DESTINATION ${MANDIR}/man1 )
|
install( FILES default_icon.png DESTINATION ${CMAKE_INSTALL_DATADIR}/tint2 )
|
||||||
if( ENABLE_EXAMPLES )
|
install( FILES AUTHORS ChangeLog README.md doc/tint2.md DESTINATION ${docdir} )
|
||||||
file( GLOB SAMPLEFILES sample/*.tint2rc )
|
install( FILES doc/manual.html doc/readme.html DESTINATION ${htmldir} )
|
||||||
install( FILES ${SAMPLEFILES} DESTINATION ${DOCDIR}/examples )
|
install( DIRECTORY doc/images DESTINATION ${htmldir} )
|
||||||
endif( ENABLE_EXAMPLES )
|
install( FILES doc/tint2.1 DESTINATION ${CMAKE_INSTALL_MANDIR}/man1 )
|
||||||
|
if( ENABLE_EXTRA_THEMES )
|
||||||
|
add_subdirectory(themes)
|
||||||
|
endif( ENABLE_EXTRA_THEMES )
|
||||||
|
|||||||
1
CONTRIBUTING.md
Normal file
1
CONTRIBUTING.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Please read https://gitlab.com/o9000/tint2/wikis/Development
|
||||||
303
ChangeLog
303
ChangeLog
@@ -1,3 +1,304 @@
|
|||||||
|
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
|
||||||
|
- Enhancements:
|
||||||
|
- New plugin: button.
|
||||||
|
|
||||||
|
2017-03-25 0.13.3
|
||||||
|
- Fixes:
|
||||||
|
- Fixed autohide for non-bottom panels (issue #632)
|
||||||
|
- Translations updated (contributed by Vladimir)
|
||||||
|
|
||||||
|
2017-03-19 0.13.2
|
||||||
|
- Fixes:
|
||||||
|
- Fixed compilation under FreeBSD
|
||||||
|
|
||||||
|
2017-03-12 0.13.1
|
||||||
|
- Fixes:
|
||||||
|
- Fixed compilation with new glibc (issue #625)
|
||||||
|
- Fixed regression in distributing size between taskbars (issue #628)
|
||||||
|
- Create ~/.config dir if it does not exist (issue #629)
|
||||||
|
- Enhancements:
|
||||||
|
- New config option systray_name_filter to hide hide specific apps from the system tray (contributed by Matthew Otnel)
|
||||||
|
- Tint2conf: minor improvements
|
||||||
|
|
||||||
|
2017-03-04 0.13
|
||||||
|
- Fixes:
|
||||||
|
- Ignore monitors with size 0, fixing crash (issue #618)
|
||||||
|
- Battery: support Asus Chromebook Flip C100PA (issue #616)
|
||||||
|
- Panel: do not change struts (available screen size) when shown in autohide mode (issue #619)
|
||||||
|
- tint2conf: executor tooltips are now correctly disabled when text config value is empty (contributed by Justin Jacobs)
|
||||||
|
- Enhancements:
|
||||||
|
- Desktop files (shortcuts) used in launcher are reloaded on click, in case the file has changed
|
||||||
|
- New config option taskbar_hide_if_empty to hide an empty taskbar in multi_desktop mode (contributed by Benoit Averty)
|
||||||
|
- Gradient backgrounds (contributed by Oskari Rauta)
|
||||||
|
- New option: panel_shrink (fixes issue #333)
|
||||||
|
- Support for Path in .desktop files (contributed by Michael Messmore)
|
||||||
|
- Tint2conf start up is much faster
|
||||||
|
- New plugin: separator (contributed by Oskari Rauta)
|
||||||
|
|
||||||
|
2016-08-02 0.12.12
|
||||||
|
- Fixes:
|
||||||
|
- Set task maximum height equal to width if not specified, instead of hardcoding 30 (issue #583; thanks @VastOne)
|
||||||
|
- tint2conf
|
||||||
|
- Process background selection events correctly (issue #582)
|
||||||
|
- Menus accessible with keyboard shortcuts (issue #590)
|
||||||
|
- Enhancements:
|
||||||
|
- Borders can now be drawn on only some sides; configurable with the option border_sides (issue #580; thanks @stophe)
|
||||||
|
- Updated man page
|
||||||
|
- The geometry of panel items is printed to stderr if the environment variable DEBUG_GEOMETRY is set
|
||||||
|
|
||||||
|
2016-05-14 0.12.11
|
||||||
|
- Fixes:
|
||||||
|
- tint2conf:
|
||||||
|
- Fixed crash in tint2conf when adding background
|
||||||
|
- Add correct extension to file name in tint2conf for 'Save as'
|
||||||
|
- Changed main window title in tint2conf
|
||||||
|
|
||||||
|
2016-05-07 0.12.10
|
||||||
|
- Fixes:
|
||||||
|
- Fixed crash in systray with non-Latin languagess (thanks zcodes)
|
||||||
|
- Invalidate cached pixmaps on resize/move (issue #576)
|
||||||
|
- Battery: do not show negative durations when the sensors return garbage
|
||||||
|
- Proper workaround for issue #555
|
||||||
|
|
||||||
|
2016-04-02 0.12.9
|
||||||
|
- Fixes:
|
||||||
|
- Regression: Do not detect empty areas as clickable (issue #572)
|
||||||
|
- Regression: Position and resize task icon correctly (issue #575)
|
||||||
|
- Use imlib2 for taking screenshot; if it fails, use Xlib (issue #574)
|
||||||
|
- Battery: lower sample frequency when there is no support for reading current/power
|
||||||
|
|
||||||
|
2016-03-25 0.12.8
|
||||||
|
- Major changes (see details below):
|
||||||
|
- Icon cache and lazy icon theme loading to improve performance in tint2 and tint2conf
|
||||||
|
- Extra tint2rc themes are installed to /usr/share/tint2 and available in tint2conf
|
||||||
|
- Tint2conf GUI improvements
|
||||||
|
- Config options with changed behavior:
|
||||||
|
- The launcher now also allows launcher_item_app entries without a full path.
|
||||||
|
In this case the .desktop file is searched in the standard application directories (issue #565).
|
||||||
|
- If the panel size is given as a percentage and a non-zero margin is also specified,
|
||||||
|
the size is now computed as a fraction of the available size (i.e. monitor size - margin).
|
||||||
|
Before it was computed as a fraction of the monitor size first, then the margin was subtracted from the value, which
|
||||||
|
was not intuitive (issue #559).
|
||||||
|
- Fixes:
|
||||||
|
- Taskbar icons are now resized correctly for certain geometries (issue #560)
|
||||||
|
- Fix get_version.sh so that it returns the correct version when .git is missing
|
||||||
|
- Fix build on powerpc
|
||||||
|
- Temporary files are now created in /tmp
|
||||||
|
- The XDG paths are now used in the icon and application lookup in addition to the hardcoded defaults
|
||||||
|
- Brigtness adjustments no longer distort colors
|
||||||
|
- Fixed race in sorting systray icons by name
|
||||||
|
- Fixed desktop entry parsing (issue #570)
|
||||||
|
- tint2conf:
|
||||||
|
- Applications are now sorted correctly
|
||||||
|
- Avoid duplicate icon themes due to symlinks
|
||||||
|
- Avoid loading desktop files marked as NoDisplay
|
||||||
|
- Enhancements:
|
||||||
|
- Launcher icon paths are now cached, which greatly improves loading time for tint2 and tint2conf.
|
||||||
|
The correct icon should be found even if you change the icon theme or install a new theme.
|
||||||
|
If this is not the case, delete the file ~/.cache/tint2/icon.cache, restart tint2 and please file a bug report
|
||||||
|
indicating the application name and the icon theme name.
|
||||||
|
- Fallback icon themes are loaded lazily to speed up tint2 and tint2conf startup
|
||||||
|
- A better Name and GenericName is used in the tint2 and tint2conf .desktop files
|
||||||
|
- tint2conf:
|
||||||
|
- Allows selection of themes from /usr/share
|
||||||
|
- Menu reorganization (thanks @Vladimir-csp for feedback)
|
||||||
|
- Icon themes are now sorted in the displayed list
|
||||||
|
- Updated ru translation (thanks @Vladimir-csp)
|
||||||
|
|
||||||
|
2016-01-29 0.12.7
|
||||||
|
- Fixes:
|
||||||
|
- Fix crash caused by race when reading inconsistent values for _NET_CURRENT_DESKTOP and _NET_NUMBER_OF_DESKTOPS
|
||||||
|
- Fix regression (all desktop tasks not working)
|
||||||
|
- Fix small memory leak in launcher
|
||||||
|
|
||||||
|
2016-01-25 0.12.6
|
||||||
|
- Fixes:
|
||||||
|
- Fix crash on 32-bit systems (issue #546)
|
||||||
|
- Fix compilation on Slackware (issue #547)
|
||||||
|
- Terminal color reset code moved to the same line to prevent interference with logging (issue #545)
|
||||||
|
- Enhancements:
|
||||||
|
- Executor now sends click coordinates via environment variables (issue #544)
|
||||||
|
|
||||||
|
2016-01-24 0.12.5
|
||||||
|
- Fixes:
|
||||||
|
- Fix rendering corruption triggered occasionally when the compositor is disabled (regression in 0.12.4)
|
||||||
|
|
||||||
|
2016-01-23 0.12.4
|
||||||
|
- Enhancements:
|
||||||
|
- Support for NETWM viewports (as used by Compiz or Unity) (issue #94)
|
||||||
|
- The default desktop font (Gtk/FontName from XSettings) is used when a font is not specified in the config file
|
||||||
|
- When no fonts are specified in the config, tint2 picks up the default font of the desktop environment via XSettings
|
||||||
|
- Tint2 is now able to dump a stack trace to stderr and ~/.tint2-crash.log in case of a crash.
|
||||||
|
If the cmake flag ENABLE_BACKTRACE is set, stack traces are dumped on X11 I/O errors.
|
||||||
|
If the cmake flag ENABLE_BACKTRACE_ON_SIGNAL is set, stack traces are dumped also on signals such as SIGSEGV.
|
||||||
|
Both ENABLE_BACKTRACE and ENABLE_BACKTRACE_ON_SIGNAL are disabled by default.
|
||||||
|
- Pixmaps are cached instead of always redrawn between different button states when mouse effects are enabled,
|
||||||
|
to improve performance
|
||||||
|
- Tooltips are no longer very wide
|
||||||
|
- Default config looks nicer
|
||||||
|
- New config options:
|
||||||
|
- Executor: new plugin that displays the output of a command in the panel (issue #161)
|
||||||
|
- New taskbar sort order options: least-recently-used (lru), most-recently-used (mru) (issue #532)
|
||||||
|
- Place the primary monitor before all the other monitors (primary_monitor_first, issue #538)
|
||||||
|
- taskbar_always_show_all_desktop_tasks (issue #279)
|
||||||
|
- Config options with changed behavior:
|
||||||
|
- Mouse effects are enabled by default
|
||||||
|
- Fixes:
|
||||||
|
- Battery changes visibility correctly (issue #531)
|
||||||
|
- Fixed rendering corruption (issue #543)
|
||||||
|
- Reverted the window flags back to the value in 0.11 to make sure the window manager detects that tint2 is a panel
|
||||||
|
(affected kwin, bspwm; fixes issue #537, breaks issue #455)
|
||||||
|
- Fixed a rare crash caused by the system tray code
|
||||||
|
- The system tray now reorders icons correctly when the icon name changes and the sorting order is configured as
|
||||||
|
ascending or descending (affected GTK icons, which sometimes set their name late after creating the icon)
|
||||||
|
- Tint2conf no longer copies sample themes to ~/.config
|
||||||
|
- New optional dependencies:
|
||||||
|
- libexecinfo on the *BSDs (only if the cmake flag ENABLE_BACKTRACE is set)
|
||||||
|
|
||||||
|
2015-11-12 0.12.3
|
||||||
|
- Enhancements:
|
||||||
|
- Battery: Multiple batteries are now supported under Linux (issue #139;
|
||||||
|
thanks to Sebastian Reichel)
|
||||||
|
- Battery: The time left for charging/discharging the battery is now estimated
|
||||||
|
when the system is not able to read current data from sensors (issue #522;
|
||||||
|
thanks to Sebastian Reichel)
|
||||||
|
- Battery: Reacts to plug/unplug events (thanks to Sebastian Reichel)
|
||||||
|
- tint2conf: Backgrounds now have a text label showing which panel component
|
||||||
|
uses them, to make them easier to identify; the label is saved as a comment
|
||||||
|
in the config file (issue #521)
|
||||||
|
- New config options:
|
||||||
|
- Mouse over effects (mouse_effects, background_color_hover, border_color_hover)
|
||||||
|
- Launcher icon background (launcher_icon_background_id)
|
||||||
|
- Enable/disable battery tooltips (battery_tooltip_enabled)
|
||||||
|
- Fixes:
|
||||||
|
- Updated French translation
|
||||||
|
- The correct icon is now used in tint2.desktop (issue #523)
|
||||||
|
- The font setting for the desktop name is no longer lost on tint2 restart
|
||||||
|
|
||||||
|
2015-08-11 0.12.2
|
||||||
|
- Fixes:
|
||||||
|
- Systray: do not move empty icons to the side, as it breaks GTK2 StatusIcon blinking (issue #515)
|
||||||
|
- tint2conf: Fix read of panel_monitor (issue #520)
|
||||||
|
- Fix command line argument processing (issue #516)
|
||||||
|
- Fix battery option parsing
|
||||||
|
|
||||||
|
2015-08-01 0.12.1
|
||||||
|
- Fixes:
|
||||||
|
- Config:
|
||||||
|
- Read config correctly when panel_items is at the end of the config file (issue #511)
|
||||||
|
- Panel:
|
||||||
|
- Do not use nested event loops (related: issue #509)
|
||||||
|
- System tray:
|
||||||
|
- Set _NET_SYSTEM_TRAY_ICON_SIZE and _NET_SYSTEM_TRAY_PADDING
|
||||||
|
- Throttle repeated resizes (workaround for issue #509)
|
||||||
|
- Taskbar:
|
||||||
|
- Use consistent visibility for sticky (all desktop) windows (related: issue #279)
|
||||||
|
- Compute monitor correctly for windows when Openbox animations are enabled (issue #511)
|
||||||
|
- tint2conf:
|
||||||
|
- Bad read of option panel_margin
|
||||||
|
- New config options:
|
||||||
|
- Battery mouse actions and clock middle click and wheel actions (thanks to Jeff Blake)
|
||||||
|
|
||||||
|
2015-07-12 0.12
|
||||||
|
- Note: the changes listed here are based on the previous release tint2 0.11, however some distributions (e.g. Debian)
|
||||||
|
offered packages using newer commits and/or patches; thus from the user's perspective some of these features are
|
||||||
|
already present. They are marked with '(already released by distros)'.
|
||||||
|
- Major changes:
|
||||||
|
- Panel:
|
||||||
|
- Checks if a compositor is launched during the first 30 seconds after startup and if found, restarts the panel with
|
||||||
|
transparency enabled.
|
||||||
|
- Launcher:
|
||||||
|
- The launcher is now considered stable
|
||||||
|
- Enhancement: SVG icon support
|
||||||
|
- Enhancement: more thorough search for icons
|
||||||
|
- Configuration GUI: tint2conf
|
||||||
|
- Experimental, testing/feedback needed
|
||||||
|
- Icons (system tray, task buttons, launcher):
|
||||||
|
- Changed rendering method to fix icon corruptions (please report any problems)
|
||||||
|
- System tray:
|
||||||
|
- Icon rendering is faster
|
||||||
|
- Several kinds of graphical corruptions have been fixed
|
||||||
|
- Added workaround for misbehaving applications leaving empty tray icons
|
||||||
|
- Many bugfixes
|
||||||
|
- New config options (see https://gitlab.com/o9000/tint2/wikis/Configure):
|
||||||
|
- Panel:
|
||||||
|
- panel_window_name
|
||||||
|
- panel_items (already released by distros)
|
||||||
|
- disable_transparency
|
||||||
|
- Taskbar:
|
||||||
|
- taskbar_distribute_size
|
||||||
|
- taskbar_hide_different_monitor
|
||||||
|
- taskbar_hide_inactive_tasks
|
||||||
|
- taskbar_sort_order
|
||||||
|
- taskbar_name (already released by distros)
|
||||||
|
- task_align (already released by distros)
|
||||||
|
- Launcher:
|
||||||
|
- launcher* (already released by distros)
|
||||||
|
- launcher_apps_dir (previously patched in by some distros)
|
||||||
|
- startup_notifications
|
||||||
|
- launcher_icon_theme_override
|
||||||
|
- System tray:
|
||||||
|
- systray_monitor
|
||||||
|
- Freespace (already released by distros)
|
||||||
|
- Config options with changed behavior:
|
||||||
|
- Panel:
|
||||||
|
- panel_dock: previously, 'panel_dock = 1' was actually not placing the panel into the dock. This option now
|
||||||
|
functions correctly. Due to the fact that OpenBox forcefully draws a border around dock windows, you might want to
|
||||||
|
set it to zero (or change the border color/style to match tint2). If you set it to zero, make sure you do not have
|
||||||
|
reserved space at the edge of the screen in the OpenBox config.
|
||||||
|
Reason for change: issues 257, 394, 461, 465, 481.
|
||||||
|
- panel_layer: previously, 'panel_layer = normal' was not functioning correctly. Now it does (it requires panel_dock = 0).
|
||||||
|
Note that in this case some compositors will draw shadows of other windows behind tint2. This can be avoided in compton
|
||||||
|
using the option shadow-exclude-reg = "x35+0-0" where 35 should be replaced with the size of the panel.
|
||||||
|
- font_shadow: shadows are thicker and softer, and are now applied to all text elements, not just the taskbar.
|
||||||
|
Reason for change: legibility improved for transparent panels.
|
||||||
|
- Launcher:
|
||||||
|
- launcher_item_app: now it expands leading ~ to the path to the user's home directory.
|
||||||
|
- Known issues:
|
||||||
|
- tint2 might close unexpectedly and/or cause X to freeze on certain ATI graphics cards.
|
||||||
|
Workaround: set 'disable_transparency = 1' in the config.
|
||||||
|
Reference: https://gitlab.com/o9000/tint2/issues/497
|
||||||
|
- Project hosting:
|
||||||
|
- Migrated from https://code.google.com/p/tint2 to https://gitlab.com/o9000/tint2 and switched from svn to git
|
||||||
|
- Bugfixes:
|
||||||
|
- task switching from tint2 using the mouse wheel (mouse_scroll_up = prev_task and mouse_scroll_down = next_task) with
|
||||||
|
multiple desktops and taskbar_mode = multi_desktop when hovering over the first/last window in the taskbar was broken;
|
||||||
|
also when a window is set to appear on all desktops;
|
||||||
|
- the launcher was not loading an icon if the icon theme was not providing it in a size close to the value of launcher_icon_size;
|
||||||
|
- the launcher was not loading an icon if the extension was incorrect (e.g. file contents were PNG but the file name
|
||||||
|
had extension SVG);
|
||||||
|
- some launcher icons (PNG files saved with an indexed colormap and alpha mask) were sometimes rendered with white instead of
|
||||||
|
transparency;
|
||||||
|
- icons looked washed out when the panel was very transparent;
|
||||||
|
- the panel border was rendered in the background of every systray icon; sometimes systray icons were rendered in the
|
||||||
|
background of other icons;
|
||||||
|
- a crash caused by a change that prevents Google Chrome (and possibly other misbehaving applications) from leaving a large
|
||||||
|
number of empty icons in the system tray;
|
||||||
|
- a crash caused by the system tray code on computers with ATI cards;
|
||||||
|
- the panel window had a shadow;
|
||||||
|
- some ATI drivers return an extra monitor with size zero which should be ignored when using panel_monitor = all;
|
||||||
|
- the battery applet was sometimes working incorrectly with certain laptop models (the battery applet stopped updating after
|
||||||
|
suspend with the battery unplugged);
|
||||||
|
- crashed caused by incomplete or malformed config files;
|
||||||
|
- with the fvwm window manager, the window title was sometimes not displayed correctly if it changed right after an
|
||||||
|
application was started;
|
||||||
|
- seconds were not updating in the clock;
|
||||||
|
- tint2 hang triggered when an application started from the launcher was closed and startup notifications were enabled;
|
||||||
|
- tint2 may fail to process X events for a few seconds when a new icon is added to the system tray;
|
||||||
|
- the code that loads SVG icons was using a lot of memory;
|
||||||
|
- fixed occasional race at startup in detecting screen resolution changes;
|
||||||
|
- modified timer code to prevent some rare double-frees or duplicate timers.
|
||||||
|
|
||||||
2010-06-26
|
2010-06-26
|
||||||
- unhide tint2 panel when dragging something
|
- unhide tint2 panel when dragging something
|
||||||
- battery FreeBSD uses the new ACPI API (thx to yamagi.burmeister)
|
- battery FreeBSD uses the new ACPI API (thx to yamagi.burmeister)
|
||||||
@@ -583,5 +884,3 @@ 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'.
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
11
INSTALL.txt
11
INSTALL.txt
@@ -1,11 +0,0 @@
|
|||||||
To build and install tint2 you need CMake.
|
|
||||||
These steps should be enough for building tint2:
|
|
||||||
|
|
||||||
mkdir build
|
|
||||||
cd build
|
|
||||||
cmake -DCMAKE_INSTALL_PREFIX=/usr ../
|
|
||||||
make
|
|
||||||
sudo make install
|
|
||||||
|
|
||||||
To see additional options you can do after the cmake step a 'cmake -L ../'
|
|
||||||
|
|
||||||
10
README
10
README
@@ -1,10 +0,0 @@
|
|||||||
---------------------------------------------------------
|
|
||||||
execute "tint2"
|
|
||||||
or "tint2 -c path_to_config_file"
|
|
||||||
|
|
||||||
|
|
||||||
check http://code.google.com/p/tint2/
|
|
||||||
for latest release, documentation and sample config file.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
95
README.md
Normal file
95
README.md
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
# Latest stable release: 0.14.3
|
||||||
|
Changes: https://gitlab.com/o9000/tint2/blob/0.14.3/ChangeLog
|
||||||
|
|
||||||
|
Documentation: [doc/tint2.md](doc/tint2.md)
|
||||||
|
|
||||||
|
Compile it with (after you install the [dependencies](https://gitlab.com/o9000/tint2/wikis/Install#dependencies)):
|
||||||
|
|
||||||
|
```
|
||||||
|
git clone https://gitlab.com/o9000/tint2.git
|
||||||
|
cd tint2
|
||||||
|
git checkout 0.14.3
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake ..
|
||||||
|
make -j4
|
||||||
|
```
|
||||||
|
|
||||||
|
To install, run (as root):
|
||||||
|
|
||||||
|
```
|
||||||
|
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`.
|
||||||
|
|
||||||
|
Please report any problems to https://gitlab.com/o9000/tint2/issues. Your feedback is much appreciated.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
# What is tint2?
|
||||||
|
|
||||||
|
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 https://code.google.com/p/ttm/.
|
||||||
|
|
||||||
|
# Features
|
||||||
|
|
||||||
|
* Panel with taskbar, system tray, clock and launcher icons;
|
||||||
|
* Easy to customize: color/transparency on fonts, icons, borders and backgrounds;
|
||||||
|
* Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;
|
||||||
|
* Multi-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;
|
||||||
|
* Customizable mouse events.
|
||||||
|
|
||||||
|
# Goals
|
||||||
|
|
||||||
|
* Be unintrusive and light (in terms of memory, CPU and aesthetic);
|
||||||
|
* Follow the freedesktop.org specifications;
|
||||||
|
* Make certain workflows, such as multi-desktop and multi-monitor, easy to use.
|
||||||
|
|
||||||
|
# I want it!
|
||||||
|
|
||||||
|
* [Install tint2](https://gitlab.com/o9000/tint2/wikis/Install)
|
||||||
|
|
||||||
|
# How do I ...
|
||||||
|
|
||||||
|
* [Install](https://gitlab.com/o9000/tint2/wikis/Install)
|
||||||
|
* [Configure](https://gitlab.com/o9000/tint2/blob/master/doc/tint2.md)
|
||||||
|
* [Add applet not supported by tint2](https://gitlab.com/o9000/tint2/wikis/ThirdPartyApplets)
|
||||||
|
* [Other frequently asked questions](https://gitlab.com/o9000/tint2/wikis/FAQ)
|
||||||
|
* [Obtain a stack trace when tint2 crashes](https://gitlab.com/o9000/tint2/wikis/Debug)
|
||||||
|
|
||||||
|
# Known issues
|
||||||
|
|
||||||
|
* Graphic glitches on Intel graphics cards can be avoided by changing the acceleration method to UXA ([issue 595](https://gitlab.com/o9000/tint2/issues/595))
|
||||||
|
* Window managers that do not follow exactly the EWMH specification might not interact well with tint2 (known issues for [awesome](https://gitlab.com/o9000/tint2/issues/385), [bspwm](https://gitlab.com/o9000/tint2/issues/524). [openbox-multihead](https://gitlab.com/o9000/tint2/issues/456))
|
||||||
|
* Full transparency requires a compositor such as Compton (if not provided already by the window manager, as in Compiz/Unity, KDE or XFCE)
|
||||||
|
|
||||||
|
# How can I help out?
|
||||||
|
|
||||||
|
* Report bugs and ask questions on the [issue tracker](https://gitlab.com/o9000/tint2/issues);
|
||||||
|
* Contribute to the development by helping us fix bugs and suggesting new features. Please read the contribution guide: [CONTRIBUTING.md](CONTRIBUTING.md)
|
||||||
|
|
||||||
|
# Links
|
||||||
|
* Home page: https://gitlab.com/o9000/tint2
|
||||||
|
* Git repository: https://gitlab.com/o9000/tint2.git
|
||||||
|
* Documentation: https://gitlab.com/o9000/tint2/wikis/home
|
||||||
|
* Downloads: https://gitlab.com/o9000/tint2-archive/tree/master or https://code.google.com/p/tint2/downloads/list
|
||||||
|
* Old project location (inactive): https://code.google.com/p/tint2
|
||||||
|
|
||||||
|
# Screenshots
|
||||||
|
|
||||||
|
## Default config:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Various configs:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Demos
|
||||||
|
|
||||||
|
* [Compact panel, separator, color gradients](https://gitlab.com/o9000/tint2/wikis/whats-new-0.13.0.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)
|
||||||
|
* [Distribute size between taskbars, freespace](https://gitlab.com/o9000/tint2/wikis/whats-new-0.12.gif)
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
DEPENDENCIES:
|
|
||||||
cairo (whit X support), pango, glib2, libX11, libXinerama, libXrandr, libXrender, libXcomposite, libXdamage imlib2 (with X support)
|
|
||||||
you might need -dev packages on debian
|
|
||||||
|
|
||||||
2
doc/footer.html
Normal file
2
doc/footer.html
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
</body>
|
||||||
|
</html>
|
||||||
16
doc/generate-doc.sh
Executable file
16
doc/generate-doc.sh
Executable file
@@ -0,0 +1,16 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# You can install md2man with gem install md2man. You need gem and ruby-dev.
|
||||||
|
|
||||||
|
set -e
|
||||||
|
set -x
|
||||||
|
|
||||||
|
md2man-roff tint2.md > tint2.1
|
||||||
|
|
||||||
|
cat header.html > manual.html
|
||||||
|
cat tint2.md | sed 's/^# TINT2 .*$/# TINT2/g' | md2man-html >> manual.html
|
||||||
|
cat footer.html >> manual.html
|
||||||
|
|
||||||
|
cat header.html > readme.html
|
||||||
|
cat ../README.md | sed 's|doc/tint2.md|manual.html|g' | md2man-html >> readme.html
|
||||||
|
cat footer.html >> readme.html
|
||||||
201
doc/header.html
Normal file
201
doc/header.html
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<head>
|
||||||
|
<style type="text/css">
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
color: #333;
|
||||||
|
font-family: Helvetica, Arial, sans-serif;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 1.6;
|
||||||
|
word-wrap: break-word;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
font-size: 1.2rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h4 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h5 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
h6 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
code,
|
||||||
|
kbd,
|
||||||
|
pre {
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-spacing: 0;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
td,
|
||||||
|
th {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
asdff {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #46c;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover,
|
||||||
|
a:active {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:not([href]) {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
box-sizing: content-box;
|
||||||
|
height: 0;
|
||||||
|
margin: 15px 0;
|
||||||
|
overflow: hidden;
|
||||||
|
background: transparent;
|
||||||
|
border: 0;
|
||||||
|
border-bottom: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::before {
|
||||||
|
display: table;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::after {
|
||||||
|
display: table;
|
||||||
|
clear: both;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0.7em;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
font-family: "Liberation Mono", monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
font: 1em "Liberation Mono", monospace;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
padding: 0 15px;
|
||||||
|
color: #777;
|
||||||
|
border-left: 4px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
overflow: auto;
|
||||||
|
word-break: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th,
|
||||||
|
table td {
|
||||||
|
padding: 6px 13px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr {
|
||||||
|
background-color: #fff;
|
||||||
|
border-top: 1px solid #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr:nth-child(2n) {
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
padding: 0.2em;
|
||||||
|
margin: 0;
|
||||||
|
font-size: 0.95em;
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
word-wrap: normal;
|
||||||
|
background-color: #eee;
|
||||||
|
margin: 1em 0;
|
||||||
|
padding: 0.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
BIN
doc/images/panel_padding.jpg
Normal file
BIN
doc/images/panel_padding.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 22 KiB |
BIN
doc/images/panel_size_margin.jpg
Normal file
BIN
doc/images/panel_size_margin.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 13 KiB |
BIN
doc/images/task_padding.jpg
Normal file
BIN
doc/images/task_padding.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 12 KiB |
BIN
doc/images/taskbar_padding.jpg
Normal file
BIN
doc/images/taskbar_padding.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 11 KiB |
688
doc/manual.html
Normal file
688
doc/manual.html
Normal file
@@ -0,0 +1,688 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<head>
|
||||||
|
<style type="text/css">
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
color: #333;
|
||||||
|
font-family: Helvetica, Arial, sans-serif;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 1.6;
|
||||||
|
word-wrap: break-word;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
font-size: 1.2rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h4 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h5 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
h6 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
code,
|
||||||
|
kbd,
|
||||||
|
pre {
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-spacing: 0;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
td,
|
||||||
|
th {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
asdff {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #46c;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover,
|
||||||
|
a:active {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:not([href]) {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
box-sizing: content-box;
|
||||||
|
height: 0;
|
||||||
|
margin: 15px 0;
|
||||||
|
overflow: hidden;
|
||||||
|
background: transparent;
|
||||||
|
border: 0;
|
||||||
|
border-bottom: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::before {
|
||||||
|
display: table;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::after {
|
||||||
|
display: table;
|
||||||
|
clear: both;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0.7em;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
font-family: "Liberation Mono", monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
font: 1em "Liberation Mono", monospace;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
padding: 0 15px;
|
||||||
|
color: #777;
|
||||||
|
border-left: 4px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
overflow: auto;
|
||||||
|
word-break: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th,
|
||||||
|
table td {
|
||||||
|
padding: 6px 13px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr {
|
||||||
|
background-color: #fff;
|
||||||
|
border-top: 1px solid #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr:nth-child(2n) {
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
padding: 0.2em;
|
||||||
|
margin: 0;
|
||||||
|
font-size: 0.95em;
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
word-wrap: normal;
|
||||||
|
background-color: #eee;
|
||||||
|
margin: 1em 0;
|
||||||
|
padding: 0.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<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="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>
|
||||||
|
<ul>
|
||||||
|
<li>Panel with taskbar, system tray, clock and launcher icons;</li>
|
||||||
|
<li>Easy to customize: color/transparency on fonts, icons, borders and backgrounds;</li>
|
||||||
|
<li>Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;</li>
|
||||||
|
<li>Multi-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;</li>
|
||||||
|
<li>Customizable mouse events.</li>
|
||||||
|
</ul>
|
||||||
|
<p>Goals:</p>
|
||||||
|
<ul>
|
||||||
|
<li>Be unintrusive and light (in terms of memory, CPU and aesthetic);</li>
|
||||||
|
<li>Follow the freedesktop.org specifications;</li>
|
||||||
|
<li>Make certain workflows, such as multi-desktop and multi-monitor, easy to use.</li>
|
||||||
|
</ul>
|
||||||
|
<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>
|
||||||
|
<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="#gradients">Gradients</a></p></li>
|
||||||
|
<li><p><a href="#panel">Panel</a></p></li>
|
||||||
|
<li><p><a href="#launcher">Launcher</a></p></li>
|
||||||
|
<li><p><a href="#taskbar-pager">Taskbar/Pager</a></p></li>
|
||||||
|
<li><p><a href="#taskbar-buttons">Taskbar buttons</a></p></li>
|
||||||
|
<li><p><a href="#mouse-actions-for-taskbar-buttons">Mouse actions for taskbar buttons</a></p></li>
|
||||||
|
<li><p><a href="#system-tray">System tray</a></p></li>
|
||||||
|
<li><p><a href="#clock">Clock</a></p></li>
|
||||||
|
<li><p><a href="#tooltip">Tooltip</a></p></li>
|
||||||
|
<li><p><a href="#battery">Battery</a></p></li>
|
||||||
|
<li><p><a href="#executor">Executor</a></p></li>
|
||||||
|
<li><p><a href="#button">Button</a></p></li>
|
||||||
|
<li><p><a href="#separator">Separator</a></p></li>
|
||||||
|
<li><p><a href="#example-configuration">Example configuration</a></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="introduction">Introduction<a name="introduction" href="#introduction" class="md2man-permalink" title="permalink"></a></h3><p>These are instructions for configuring tint2 directly by editing its config file.
|
||||||
|
You may also use instead the graphical interface <code>tint2conf</code>.</p><p>The first time you run tint2, it will create the config file in <code>$HOME/.config/tint2/tint2rc</code> (This applies 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 also specify another file on the command line with the -c option, e.g.: <code>tint2 -c $HOME/tint2.conf</code>. This can be used to run multiple instances of tint2 that use different settings.</p><p>If you change the config file while tint2 is running, the command <code>killall -SIGUSR1 tint2</code> will force tint2 to reload it.</p><p>All the configuration options supported in the config file are listed below.
|
||||||
|
Try to respect as much as possible the order of the options as given below.</p><h3 id="backgrounds-and-borders">Backgrounds and borders<a name="backgrounds-and-borders" href="#backgrounds-and-borders" class="md2man-permalink" title="permalink"></a></h3><p>The tint2 config file starts with the options defining background elements with borders:</p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>rounded = number_of_pixels</code> : the corner radius</p></li>
|
||||||
|
<li><p><code>border_width = integer</code> : the border width in pixels</p></li>
|
||||||
|
<li><p><code>border_sides = LRTB</code> : the sides to draw the border on (left, right, top, bottom). If not specified, all sides are used. <em>(since 0.12.12)</em></p></li>
|
||||||
|
<li><p><code>background_color = color opacity</code></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz).</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color = color opacity</code></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>background_color_hover = color opacity</code> (default: same as <code>background_color</code>) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color_hover = color opacity</code> (default: same as <code>border_color</code>) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>background_color_pressed = color opacity</code> (default: same as <code>background_color_hover</code>) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color_pressed = color opacity</code> (default: same as <code>border_color_hover</code>) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
</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
|
||||||
|
border_width = 0
|
||||||
|
background_color = #282828 100
|
||||||
|
border_color = #000000 0
|
||||||
|
|
||||||
|
rounded = 1
|
||||||
|
border_width = 0
|
||||||
|
background_color = #f6b655 90
|
||||||
|
border_color = #cccccc 40
|
||||||
|
</code></pre>
|
||||||
|
<p>tint2 automatically identifies each background with a number starting from 1 (1, 2, ...).
|
||||||
|
Afterwards, you can apply a background to objects (panel, taskbar, task, clock, systray) using the background id, for example:</p><pre class="highlight plaintext"><code>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
|
||||||
|
</code></pre>
|
||||||
|
<p>Identifier 0 refers to a special background which is fully transparent, identifier 1 applies the first background defined in the config file etc.</p><h3 id="gradients">Gradients<a name="gradients" href="#gradients" class="md2man-permalink" title="permalink"></a></h3><p>(Available since 0.13.0)</p><p>Backgrounds also allow specifying gradient layers
|
||||||
|
that are drawn on top of the solid color background.</p><p>First the user must define one or more gradients in the config file,
|
||||||
|
each starting with <code>gradient = TYPE</code>. These must be added before backgrounds.</p><p>Then gradients can be added by index to backgrounds,
|
||||||
|
using the <code>gradient_id = INDEX</code>, <code>gradient_id_hover = INDEX</code> and
|
||||||
|
<code>gradient_id_pressed = INDEX</code>, where <code>INDEX</code> is
|
||||||
|
the gradient index, starting from 1.</p><h4 id="gradient-types">Gradient types<a name="gradient-types" href="#gradient-types" class="md2man-permalink" title="permalink"></a></h4><p>Gradients vary the color between fixed control points:
|
||||||
|
* vertical gradients: top-to-bottom;
|
||||||
|
* horizontal gradients: left-to-right;
|
||||||
|
* radial gradients: center-to-corners.</p><p>The user must specify the start and end colors, and can optionally add extra color stops in between
|
||||||
|
using the <code>color_stop</code> option, as explained below.</p><h5 id="vertical-gradient-with-color-varying-from-the-top-edge-to-the-bottom-edge-two-colors">Vertical gradient, with color varying from the top edge to the bottom edge, two colors<a name="vertical-gradient-with-color-varying-from-the-top-edge-to-the-bottom-edge-two-colors" href="#vertical-gradient-with-color-varying-from-the-top-edge-to-the-bottom-edge-two-colors" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>gradient = vertical
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="horizontal-gradient-with-color-varying-from-the-left-edge-to-the-right-edge-two-colors">Horizontal gradient, with color varying from the left edge to the right edge, two colors<a name="horizontal-gradient-with-color-varying-from-the-left-edge-to-the-right-edge-two-colors" href="#horizontal-gradient-with-color-varying-from-the-left-edge-to-the-right-edge-two-colors" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>gradient = horizontal
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="radial-gradient-with-color-varying-from-the-center-to-the-corner-two-colors">Radial gradient, with color varying from the center to the corner, two colors:<a name="radial-gradient-with-color-varying-from-the-center-to-the-corner-two-colors" href="#radial-gradient-with-color-varying-from-the-center-to-the-corner-two-colors" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>gradient = radial
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="adding-extra-color-stops-0-and-100-remain-fixed-more-colors-at-x-between-the-start-and-end-control-points">Adding extra color stops (0% and 100% remain fixed, more colors at x% between the start and end control points)<a name="adding-extra-color-stops-0-and-100-remain-fixed-more-colors-at-x-between-the-start-and-end-control-points" href="#adding-extra-color-stops-0-and-100-remain-fixed-more-colors-at-x-between-the-start-and-end-control-points" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>color_stop = percentage #rrggbb opacity
|
||||||
|
</code></pre>
|
||||||
|
<h4 id="gradient-examples">Gradient examples<a name="gradient-examples" href="#gradient-examples" class="md2man-permalink" title="permalink"></a></h4><pre class="highlight plaintext"><code># Gradient 1: thin film effect
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111122 30
|
||||||
|
end_color = #112211 30
|
||||||
|
color_stop = 60 #221111 30
|
||||||
|
|
||||||
|
# Gradient 2: radial glow
|
||||||
|
gradient = radial
|
||||||
|
start_color = #ffffff 20
|
||||||
|
end_color = #ffffff 0
|
||||||
|
|
||||||
|
# Gradient 3: elegant black
|
||||||
|
gradient = vertical
|
||||||
|
start_color = #444444 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Gradient 4: elegant black
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111111 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Background 1: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
border_sides = TBLR
|
||||||
|
background_color = #555555 10
|
||||||
|
border_color = #ffffff 60
|
||||||
|
background_color_hover = #555555 10
|
||||||
|
border_color_hover = #ffffff 60
|
||||||
|
background_color_pressed = #555555 10
|
||||||
|
border_color_pressed = #ffffff 60
|
||||||
|
gradient_id = 3
|
||||||
|
gradient_id_hover = 4
|
||||||
|
gradient_id_pressed = 2
|
||||||
|
|
||||||
|
[...]
|
||||||
|
</code></pre>
|
||||||
|
<h3 id="panel">Panel<a name="panel" href="#panel" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>panel_items = LTSBC</code> defines the items tint2 will show and the order of those items. Each letter refers to an item, defined as:</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>L</code> shows the Launcher</li>
|
||||||
|
<li><code>T</code> shows the Taskbar</li>
|
||||||
|
<li><code>S</code> shows the Systray (also called notification area)</li>
|
||||||
|
<li><code>B</code> shows the Battery status</li>
|
||||||
|
<li><code>C</code> shows the Clock</li>
|
||||||
|
<li><code>F</code> adds an extensible spacer (freespace). You can specify more than one. Has no effect if <code>T</code> is also present. <em>(since 0.12)</em></li>
|
||||||
|
<li><code>E</code> adds an executor plugin. You can specify more than one. <em>(since 0.12.4)</em></li>
|
||||||
|
<li><code>P</code> adds a push button. You can specify more than one. <em>(since 0.14)</em></li>
|
||||||
|
<li><code>:</code> adds a separator. You can specify more than one. <em>(since 0.13.0)</em></li>
|
||||||
|
</ul>
|
||||||
|
<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>
|
||||||
|
<ul>
|
||||||
|
<li>The first monitor is <code>1</code></li>
|
||||||
|
<li>Use <code>panel_monitor = all</code> to get a separate panel per monitor</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>
|
||||||
|
</ul>
|
||||||
|
<p><img src="images/panel_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>panel_position = vertical_position horizontal_position orientation</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>vertical_position</code> is one of: <code>bottom</code>, <code>top</code>, <code>center</code></li>
|
||||||
|
<li><code>horizontal_position</code> is one of: <code>left</code>, <code>right</code>, <code>center</code></li>
|
||||||
|
<li><code>orientation</code> is one of: <code>horizontal</code>, <code>vertical</code></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>panel_size = width height</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>width</code> and <code>height</code> can be specified without units (e.g. <code>123</code>) as pixels, or followed by <code>%</code> as percentages of the monitor size (e.g. <code>50%</code>). Use <code>100%</code> for full monitor width/height.
|
||||||
|
Example:</li>
|
||||||
|
</ul></li>
|
||||||
|
</ul>
|
||||||
|
<pre class="highlight plaintext"><code># The panel's width is 94% the size of the monitor, the height is 30 pixels:
|
||||||
|
panel_size = 94% 30
|
||||||
|
</code></pre>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><p><code>panel_shrink = boolean (0 or 1)</code> : If set to 1, the panel will shrink to a compact size dynamically. <em>(since 0.13)</em></p></li>
|
||||||
|
<li><p><code>panel_margin = horizontal_margin vertical_margin</code> : The margins define the distance between the panel and the horizontal/vertical monitor edge. Use <code>0</code> to obtain a panel with the same size as the edge of the monitor (no margin).</p></li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="images/panel_size_margin.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>panel_padding = horizontal_padding vertical_padding spacing</code> : Please refer to the image below.</li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="images/panel_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>font_shadow = boolean (0 or 1)</code></p></li>
|
||||||
|
<li><p><code>panel_background_id = integer</code> : Which background to use for the panel.</p></li>
|
||||||
|
<li><p><code>wm_menu = boolean (0 or 1)</code> : Defines if tint2 forwards unhandled mouse events to your window manager. Useful for window managers such as openbox, which display the start menu if you right click on the desktop.</p></li>
|
||||||
|
<li><p><code>panel_dock = boolean (0 or 1)</code> : Defines if tint2 is placed into the window manager's dock. For the openbox window manager it is advised to also use a modifier for the moveButton option, otherwise the mouse click is not forwarded to tint2 (in ~/.config/openbox/rc.xml).</p></li>
|
||||||
|
<li><p><code>panel_layer = bottom/normal/top</code> : Places tint2 into the bottom/normal/top layer. This is helpful for specifying if the panel can be covered by other windows or not. The default is the bottom layer, but with real transparency normal or top layer may be a nice alternative.</p></li>
|
||||||
|
<li><p><code>strut_policy = follow_size/minimum/none</code> : STRUTs are used by the window manager to decide the size of maximized windows. Note: on multi-monitor (Xinerama) setups, the panel must be placed at the edge (not in the middle) of the virtual screen for this to work correctly.</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>follow_size</code> means that the maximized windows always resize to have a common edge with tint2.</li>
|
||||||
|
<li><code>minimum</code> means that the maximized windows always expand to have a common edge with the hidden panel. This is useful if the <code>autohide</code> option is enabled.</li>
|
||||||
|
<li><code>none</code> means that the maximized windows use the full screen size.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>panel_window_name = string</code> : Defines the name of the panel's window. Default: 'tint2'. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>disable_transparency = boolean (0 or 1)</code> : Whether to disable transparency instead of detecting if it is supported. Useful on broken graphics stacks. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>mouse_effects = boolean (0 or 1)</code> : Whether to enable mouse hover effects for clickable items. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>mouse_hover_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency on mouse hover (works only when mouse_effects = 1).` <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>mouse_pressed_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency on mouse press (works only when mouse_effects = 1).` <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>autohide = boolean (0 or 1)</code> : Whether to enable panel hiding when the mouse cursor exists the panel.</p></li>
|
||||||
|
<li><p><code>autohide_show_timeout = float</code> : Show timeout in seconds after the mouse cursor enters the panel. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>autohide_hide_timeout = float</code> : Hide timeout in seconds after the mouse cursor exits the panel. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>autohide_height = integer</code> : panel height (width for vertical panels) in hidden mode.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="launcher">Launcher<a name="launcher" href="#launcher" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>launcher_item_app = path_to_application</code> : Each <code>launcher_item_app</code> must be a file path to a .desktop file following the freedesktop.org <a href="http://standards.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html">specification</a>. The paths may begin with <code>~</code>, which is expanded to the path of the user's home directory. If only a file name is specified, the file is search in the standard application directories (<code>$XDG_DATA_HOME/applications</code>, <code>~/.local/share/applications</code>, <code>$XDG_DATA_DIRS/applications</code>, <code>/usr/local/share/applications</code>, <code>/usr/share/applications</code>, <code>/opt/share/applications</code>).</p></li>
|
||||||
|
<li><p><code>launcher_apps_dir = path_to_directory</code> : Specifies a path to a directory from which the launcher is loading all .desktop files (all subdirectories are explored recursively). Can be used multiple times. The path may begin with <code>~</code>, which is expanded to the path of the user's home directory. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>launcher_background_id = integer</code> : Defines which background to use.</p></li>
|
||||||
|
<li><p><code>launcher_icon_background_id = integer</code> : Defines which background to use for icons.</p></li>
|
||||||
|
<li><p><code>launcher_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>launcher_icon_size = integer</code> : The launcher icon size, in pixels.</p></li>
|
||||||
|
<li><p><code>launcher_icon_theme = name_of_theme</code> : (Optional) Uses the specified icon theme to display shortcut icons. Note that tint2 will detect and use the icon theme of your desktop if you have an XSETTINGS manager running (which you probably do), unless <code>launcher_icon_theme_override = 1</code>.</p></li>
|
||||||
|
<li><p><code>launcher_icon_theme_override = boolean (0 or 1)</code> : Whether <code>launcher_icon_theme</code> overrides the value obtained from the XSETTINGS manager. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>launcher_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency.</p></li>
|
||||||
|
<li><p><code>launcher_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for the launcher icons.</p></li>
|
||||||
|
<li><p><code>startup_notifications = boolean (0 or 1)</code> : Whether to show startup notifications when starting applications from the launcher. <em>(since 0.12)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="taskbar-pager">Taskbar / Pager<a name="taskbar-pager" href="#taskbar-pager" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>taskbar_mode = single_desktop/multi_desktop</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>single_desktop</code> : Shows a normal taskbar listing the tasks running on the current virtual desktop (also known as 'workspace');</li>
|
||||||
|
<li><code>multi_desktop</code> : Pager like capability. Shows multiple taskbars, one per virtual desktop, with which:
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li>You can drag-and-drop tasks between virtual desktops;</li>
|
||||||
|
<li>You can switch between virtual desktops.</li>
|
||||||
|
</ul></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>taskbar_hide_if_empty = boolean (0 or 1)</code> : If enabled, in multi-desktop mode the taskbars corresponding to empty desktops different from the current desktop are hidden. <em>(since 0.13)</em></p></li>
|
||||||
|
<li><p><code>taskbar_distribute_size = boolean (0 or 1)</code> : If enabled, in multi-desktop mode distributes between taskbars the available size proportionally to the number of tasks. Default: disabled. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>taskbar_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="images/taskbar_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>taskbar_background_id = integer</code> : Which background to use</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_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_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>
|
||||||
|
<ul>
|
||||||
|
<li><code>none</code> : No sorting. New tasks are simply appended at the end of the taskbar when they appear.</li>
|
||||||
|
<li><code>title</code> : Sorts the tasks by title.</li>
|
||||||
|
<li><code>center</code> : Sorts the tasks by their window centers.</li>
|
||||||
|
<li><code>mru</code> : Shows the most recently used tasks first. <em>(since 0.12.4)</em></li>
|
||||||
|
<li><code>lru</code> : Shows the most recently used tasks last. <em>(since 0.12.4)</em></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>task_align = left/center/right</code> : Specifies the alignment of the tasks on the taskbar. Default: left.</p></li>
|
||||||
|
<li><p><code>taskbar_name = boolean (0 or 1)</code> : Whether to show the virtual desktop name in the taskbar.</p></li>
|
||||||
|
<li><p><code>taskbar_name_padding = padding</code> : Padding for the virtual desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_background_id = integer</code> : Which background to use for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code> : Font configuration for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_font_color = color opacity (0 to 100)</code> : Font color for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_active_background_id = integer</code> : Which background to use for the name of the current desktop.</p></li>
|
||||||
|
<li><p><code>taskbar_name_active_font_color = color opacity (0 to 100)</code> : Font color for the name of the current desktop.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="taskbar-buttons">Taskbar buttons<a name="taskbar-buttons" href="#taskbar-buttons" class="md2man-permalink" title="permalink"></a></h1><p>The following options configure the task buttons in the taskbar:</p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_icon = boolean (0 or 1)</code> : Whether to display the task icon. There is no explicit option to control the task icon size; it depends on the vertical padding set with <code>task_padding</code>.</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_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for tasks.</p></li>
|
||||||
|
<li><p><code>task_maximum_size = width height</code></p>
|
||||||
|
<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>height</code> is used with vertical panels.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>task_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>urgent_nb_of_blink = integer</code> : Number of blinks on 'get attention' events.</p></li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="images/task_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>task_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>task_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjust the task icon's color and transparency.</p></li>
|
||||||
|
<li><p><code>task_background_id = integer</code> : Which background to use for non selected tasks</p></li>
|
||||||
|
</ul>
|
||||||
|
<dl><dt>For the next 3 options STATUS can be <code>active</code> / <code>iconified</code> / <code>urgent</code>:</dt><dd>* <code>task_STATUS_font_color = color opacity (0 to 100)</code></dd></dl>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_STATUS_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the task icon's color and transparency.</p></li>
|
||||||
|
<li><p><code>task_STATUS_background_id = integer</code> : Which background to use for the task.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="mouse-actions-for-taskbar-buttons">Mouse actions for taskbar buttons<a name="mouse-actions-for-taskbar-buttons" href="#mouse-actions-for-taskbar-buttons" class="md2man-permalink" title="permalink"></a></h3><p>The possible mouse events are: <code>left, middle, right, scroll_up, scroll_down</code>.</p><p>The possible mouse actions are: <code>none, close, toggle, iconify, shade, toggle_iconify, maximize_restore, desktop_left, desktop_right, next_task, prev_task</code>.</p><p>Use <code>mouse_event = action</code> to customize mouse actions. Example:
|
||||||
|
<code>
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = toggle
|
||||||
|
mouse_scroll_down = iconify
|
||||||
|
</code></p><dl><dt>The action semantics:</dt><dd>* <code>none</code> : If <code>wm_menu = 1</code> is set, the mouse event is forwarded to the window manager. Otherwise it is ignored.
|
||||||
|
* <code>close</code> : close the task
|
||||||
|
* <code>toggle</code> : toggle the task
|
||||||
|
* <code>iconify</code> : iconify (minimize) the task
|
||||||
|
* <code>toggle_iconify</code> : toggle or iconify the task
|
||||||
|
* <code>maximize_restore</code> : maximized or minimized the task
|
||||||
|
* <code>shade</code> : shades (collapses) the task
|
||||||
|
* <code>desktop_left</code> : send the task to the desktop on the left
|
||||||
|
* <code>desktop_right</code> : send the task to the desktop on the right
|
||||||
|
* <code>next_task</code> : send the focus to next task
|
||||||
|
* <code>prev_task</code> : send the focus to previous task</dd></dl><h3 id="system-tray">System Tray<a name="system-tray" href="#system-tray" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>systray_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>systray_background_id = integer</code> : Which background to use.</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_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_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>
|
||||||
|
<h3 id="clock">Clock<a name="clock" href="#clock" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>time1_format = %H:%M</code> : The format used by the first line of the clock.</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>time1_format</code>, <code>time2_format</code> and <code>clock_tooltip</code> use the 'strftime' syntax. More info can be found here: <a href="http://www.manpagez.com/man/3/strftime/">http://www.manpagez.com/man/3/strftime/</a></li>
|
||||||
|
<li>To hide the clock, comment <code>time1_format</code> and <code>time2_format</code>.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>time1_timezone = :US/Hawaii</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>time1_timezone</code>, <code>time2_timezone</code> and <code>clock_tooltip_timezone</code> 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 <code>/usr/share/zoneinfo</code>. If your timezones are in a different directory, you need to specify the absolute path, e.g. <code>time1_timezone = :/different/zoneinfo/dir/US/Hawaii</code> Always prepend the timezone with a ':'</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>time1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>time2_format = %A %d %B</code></p></li>
|
||||||
|
<li><p><code>time2_timezone = :Europe/Berlin</code></p></li>
|
||||||
|
<li><p><code>time2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>clock_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>clock_padding = horizontal_padding vertical_padding</code></p></li>
|
||||||
|
<li><p><code>clock_background_id = integer</code> : Which background to use</p></li>
|
||||||
|
<li><p><code>clock_tooltip = %a, %d. %b %Y</code> : Format for the clock's tooltip.</p></li>
|
||||||
|
<li><p><code>clock_tooltip_timezone = :UTC</code></p></li>
|
||||||
|
<li><p><code>clock_lclick_command = text</code> : Command to execute on left click.</p></li>
|
||||||
|
<li><p><code>clock_rclick_command = text</code> : Command to execute on right click.</p></li>
|
||||||
|
<li><p><code>clock_mclick_command = text</code> : Command to execute on middle click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>clock_uwheel_command = text</code> : Command to execute on wheel scroll up. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>clock_dwheel_command = text</code> : Command to execute on wheel scroll down. <em>(since 0.12.1)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="tooltip">Tooltip<a name="tooltip" href="#tooltip" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>tooltip_padding = horizontal_padding vertical_padding</code></p></li>
|
||||||
|
<li><p><code>tooltip_show_timeout = float</code> : Delay to show the tooltip in seconds. Use <code>.</code> as decimal separator.</p></li>
|
||||||
|
<li><p><code>tooltip_hide_timeout = float</code> : Delay to hide the tooltip in seconds. Use <code>.</code> as decimal separator.</p></li>
|
||||||
|
<li><p><code>tooltip_background_id = integer</code> : Which background to use for tooltips. Note that with fake transparency the alpha channel and corner radius options are not respected.</p></li>
|
||||||
|
<li><p><code>tooltip_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>tooltip_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="battery">Battery<a name="battery" href="#battery" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<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_cmd = notify-send "battery low"</code> : Command to execute when the battery is low.</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>battery_font_color = color opacity (0 to 100)</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_tooltip_enabled = boolean (0 or 1)</code> : Enable/disable battery tooltips. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>battery_lclick_command = text</code> : Command to execute on left click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_rclick_command = text</code> : Command to execute on right click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_mclick_command = text</code> : Command to execute on middle click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_uwheel_command = text</code> : Command to execute on wheel scroll up. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_dwheel_command = text</code> : Command to execute on wheel scroll down. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>ac_connected_cmd = text</code> : Command to execute when the power adapter is plugged in. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>ac_disconnected_cmd = text</code> : Command to execute when the power adapter is unplugged. <em>(since 0.12.3)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="executor">Executor<a name="executor" href="#executor" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>execp = new</code> : Begins the configuration of a new executor plugin. Multiple such plugins are supported; just use multiple <code>E</code>s in <code>panel_items</code>. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_command = text</code> : Command to execute. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_interval = integer</code> : The command is executed again after <code>execp_interval</code> seconds from the moment it exits. If zero, the command is executed only once. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_continuous = integer</code> : If non-zero, the last <code>execp_continuous</code> lines from the output of the command are displayed, every <code>execp_continuous</code> lines; this is useful for showing the output of commands that run indefinitely, such as <code>ping 127.0.0.1</code>. If zero, the output of the command is displayed after it finishes executing. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_has_icon = boolean (0 or 1)</code> : If <code>execp_has_icon = 1</code>, the first line printed by the command is interpreted as a path to an image file. <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_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_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_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_background_id = integer</code> : Which background to use. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_centered = boolean (0 or 1)</code> : Whether to center the text. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_padding = horizontal_padding vertical_padding spacing_between_icon_and_text</code> <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_lclick_command = text</code> : Command to execute on left click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_mclick_command = text</code> : Command to execute on right click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_rclick_command = text</code> : Command to execute on middle click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_uwheel_command = text</code> : Command to execute on wheel scroll up. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_dwheel_command = text</code> : Command to execute on wheel scroll down. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h4 id="executor-samples">Executor samples<a name="executor-samples" href="#executor-samples" class="md2man-permalink" title="permalink"></a></h4><h5 id="print-the-hostname">Print the hostname<a name="print-the-hostname" href="#print-the-hostname" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = hostname
|
||||||
|
execp_interval = 0
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="print-disk-usage-for-the-root-partition-every-10-seconds">Print disk usage for the root partition every 10 seconds<a name="print-disk-usage-for-the-root-partition-every-10-seconds" href="#print-disk-usage-for-the-root-partition-every-10-seconds" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = df -h | awk '/\/$/ { print $6 ": " $2 " " $5}'
|
||||||
|
execp_interval = 10
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="button-with-icon-and-rich-text-executes-command-when-clicked">Button with icon and rich text, executes command when clicked<a name="button-with-icon-and-rich-text-executes-command-when-clicked" href="#button-with-icon-and-rich-text-executes-command-when-clicked" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = echo /usr/share/icons/elementary-xfce/emblems/24/emblem-colors-blue.png; echo '<span foreground="#7f7">Click</span> <span foreground="#77f">me</span> <span foreground="#f77">pls</span>'
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_interval = 0
|
||||||
|
execp_centered = 1
|
||||||
|
execp_font = sans 9
|
||||||
|
execp_markup = 1
|
||||||
|
execp_font_color = #aaffaa 100
|
||||||
|
execp_padding = 2 0
|
||||||
|
execp_tooltip = I will tell you a secret...
|
||||||
|
execp_lclick_command = zenity --info "--text=$(uname -sr)"
|
||||||
|
execp_background_id = 2
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="desktop-pager-with-text">Desktop pager with text<a name="desktop-pager-with-text" href="#desktop-pager-with-text" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = xprop -root -spy | awk '/^_NET_CURRENT_DESKTOP/ { print "Workspace " ($3 + 1) ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_continuous = 1
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="desktop-pager-with-icon">Desktop pager with icon<a name="desktop-pager-with-icon" href="#desktop-pager-with-icon" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp_command = xprop -root -spy | awk -v home="$HOME" '/^_NET_CURRENT_DESKTOP/ { print home "/.config/myPager/" ($3 + 1) ".png\n" ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_cache_icon = 1
|
||||||
|
execp_continuous = 2
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="round-trip-time-to-the-gateway-refreshed-every-second">Round-trip time to the gateway, refreshed every second<a name="round-trip-time-to-the-gateway-refreshed-every-second" href="#round-trip-time-to-the-gateway-refreshed-every-second" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = ping -i 1 -c 1 -W 1 -O -D -n $(ip route | grep default | grep via | grep -o '[0-9]*\.[0-9]*\.[0-9]*\.[0-9]*') | awk '/no/ { print "<span foreground=\"#faa\">timeout</span>"; fflush(); }; /time=/ { gsub(/time=/, "", $8); printf "<span foreground=\"#7af\">%3.0f %s</span>\n", $8, $9; fflush(); } '
|
||||||
|
execp_continuous = 0
|
||||||
|
execp_interval = 1
|
||||||
|
execp_markup = 1
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="memory-usage">Memory usage<a name="memory-usage" href="#memory-usage" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = free | awk '/^-/ { printf "Mem: '$(free -h | awk '/^Mem:/ { print $2 }')' %.0f%%\n", 100*$3/($3+$4); fflush(stdout) }'
|
||||||
|
execp_interval = 5
|
||||||
|
execp_continuous = 0
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="network-load">Network load<a name="network-load" href="#network-load" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code># Note the use of "stdbuf -oL" to force the program to flush the output line by line.
|
||||||
|
execp = new
|
||||||
|
execp_command = stdbuf -oL bwm-ng -o csv -t 1000 | awk -F ';' '/total/ { printf "Net: %.0f Mb/s\n", ($5*8/1.0e6) }; fflush(stdout)'
|
||||||
|
execp_continuous = 1
|
||||||
|
execp_interval = 1
|
||||||
|
</code></pre>
|
||||||
|
<h3 id="button">Button<a name="button" href="#button" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>button = new</code> : Begins the configuration of a new button. Multiple such plugins are supported; just use multiple <code>P</code>s in <code>panel_items</code>. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_icon = text</code> : Name or path of icon (or empty). <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_text = text</code> : Text to display (or empty). <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_tooltip = text</code> : The tooltip (or empty). <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code> : The font used to draw the text. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_font_color = color opacity</code> : The font color. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_background_id = integer</code> : Which background to use. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_centered = boolean (0 or 1)</code> : Whether to center the text. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_padding = horizontal_padding vertical_padding spacing_between_icon_and_text</code> <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_max_icon_size = integer</code> : Sets a limit to the icon size. Otherwise, the icon will expand to the edges. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_lclick_command = text</code> : Command to execute on left click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_mclick_command = text</code> : Command to execute on right click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_rclick_command = text</code> : Command to execute on middle click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_uwheel_command = text</code> : Command to execute on wheel scroll up. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.14)</em></p></li>
|
||||||
|
<li><p><code>button_dwheel_command = text</code> : Command to execute on wheel scroll down. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.14)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="separator">Separator<a name="separator" href="#separator" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>separator = new</code> : Begins the configuration of a new separator. Multiple such plugins are supported; just use multiple <code>:</code>s in <code>panel_items</code>. <em>(since 0.13.0)</em></p></li>
|
||||||
|
<li><p><code>separator_background_id = integer</code> : Which background to use. <em>(since 0.13.0)</em></p></li>
|
||||||
|
<li><p><code>separator_color = color opacity</code> : The foreground color. <em>(since 0.13.0)</em></p></li>
|
||||||
|
<li><p><code>separator_style = [empty | line | dots]</code> : The separator style. <em>(since 0.13.0)</em></p></li>
|
||||||
|
<li><p><code>separator_size = integer</code> : The thickness of the separator. Does not include the border and padding. For example, if the style is <code>line</code>, this is the line thickness; if the style is <code>dots</code>, this is the dot's diameter. <em>(since 0.13.0)</em></p></li>
|
||||||
|
<li><p><code>separator_padding = side_padding cap_padding</code> : The padding to add to the sides of the separator, in pixels. <em>(since 0.13.0)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="example-configuration">Example configuration<a name="example-configuration" href="#example-configuration" class="md2man-permalink" title="permalink"></a></h3><p>See /etc/xdg/tint2/tint2rc.</p><h2 id="author">AUTHOR<a name="author" href="#author" class="md2man-permalink" title="permalink"></a></h2><p>tint2 was written by Thierry Lorthiois <a href="mailto:lorthiois@bbsoft.fr">lorthiois@bbsoft.fr</a>.
|
||||||
|
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>
|
||||||
|
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's default location
|
||||||
|
for documentation files, usually <code>/usr/share/doc/tint2</code> or <code>/usr/local/share/doc/tint2</code>.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
267
doc/readme.html
Normal file
267
doc/readme.html
Normal file
@@ -0,0 +1,267 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<head>
|
||||||
|
<style type="text/css">
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
color: #333;
|
||||||
|
font-family: Helvetica, Arial, sans-serif;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 1.6;
|
||||||
|
word-wrap: break-word;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
font-size: 1.2rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h4 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
h5 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
h6 {
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
code,
|
||||||
|
kbd,
|
||||||
|
pre {
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-spacing: 0;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
td,
|
||||||
|
th {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
asdff {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #46c;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover,
|
||||||
|
a:active {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:not([href]) {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
box-sizing: content-box;
|
||||||
|
height: 0;
|
||||||
|
margin: 15px 0;
|
||||||
|
overflow: hidden;
|
||||||
|
background: transparent;
|
||||||
|
border: 0;
|
||||||
|
border-bottom: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::before {
|
||||||
|
display: table;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
hr::after {
|
||||||
|
display: table;
|
||||||
|
clear: both;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0.7em;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
font-family: "Liberation Mono", monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
font: 1em "Liberation Mono", monospace;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
padding: 0 15px;
|
||||||
|
color: #777;
|
||||||
|
border-left: 4px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote>:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
overflow: auto;
|
||||||
|
word-break: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th,
|
||||||
|
table td {
|
||||||
|
padding: 6px 13px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr {
|
||||||
|
background-color: #fff;
|
||||||
|
border-top: 1px solid #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tr:nth-child(2n) {
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
padding: 0.2em;
|
||||||
|
margin: 0;
|
||||||
|
font-size: 0.95em;
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
word-wrap: normal;
|
||||||
|
background-color: #eee;
|
||||||
|
margin: 1em 0;
|
||||||
|
padding: 0.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1 id="latest-stable-release-0-14-3"><span class="md2man-title">Latest</span> <span class="md2man-section">stable</span> <span class="md2man-date">release:</span> <span class="md2man-source">0.14.3</span><a name="latest-stable-release-0-14-3" href="#latest-stable-release-0-14-3" class="md2man-permalink" title="permalink"></a></h1><p>Changes: <a href="https://gitlab.com/o9000/tint2/blob/0.14.3/ChangeLog">https://gitlab.com/o9000/tint2/blob/0.14.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
|
||||||
|
git checkout 0.14.3
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake ..
|
||||||
|
make -j4
|
||||||
|
</code></pre>
|
||||||
|
<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>
|
||||||
|
<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>
|
||||||
|
<li>Panel with taskbar, system tray, clock and launcher icons;</li>
|
||||||
|
<li>Easy to customize: color/transparency on fonts, icons, borders and backgrounds;</li>
|
||||||
|
<li>Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;</li>
|
||||||
|
<li>Multi-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;</li>
|
||||||
|
<li>Customizable mouse events.</li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="goals">Goals<a name="goals" href="#goals" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li>Be unintrusive and light (in terms of memory, CPU and aesthetic);</li>
|
||||||
|
<li>Follow the freedesktop.org specifications;</li>
|
||||||
|
<li>Make certain workflows, such as multi-desktop and multi-monitor, easy to use.</li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="i-want-it">I want it!<a name="i-want-it" href="#i-want-it" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/wikis/Install">Install tint2</a></li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="how-do-i">How do I ...<a name="how-do-i" href="#how-do-i" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/wikis/Install">Install</a></li>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/blob/master/manual.html">Configure</a></li>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/wikis/ThirdPartyApplets">Add applet not supported by tint2</a></li>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/wikis/FAQ">Other frequently asked questions</a></li>
|
||||||
|
<li><a href="https://gitlab.com/o9000/tint2/wikis/Debug">Obtain a stack trace when tint2 crashes</a></li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="known-issues">Known issues<a name="known-issues" href="#known-issues" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li>Graphic glitches on Intel graphics cards can be avoided by changing the acceleration method to UXA (<a href="https://gitlab.com/o9000/tint2/issues/595">issue 595</a>)</li>
|
||||||
|
<li>Window managers that do not follow exactly the EWMH specification might not interact well with tint2 (known issues for <a href="https://gitlab.com/o9000/tint2/issues/385">awesome</a>, <a href="https://gitlab.com/o9000/tint2/issues/524">bspwm</a>. <a href="https://gitlab.com/o9000/tint2/issues/456">openbox-multihead</a>)</li>
|
||||||
|
<li>Full transparency requires a compositor such as Compton (if not provided already by the window manager, as in Compiz/Unity, KDE or XFCE)</li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="how-can-i-help-out">How can I help out?<a name="how-can-i-help-out" href="#how-can-i-help-out" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li>Report bugs and ask questions on the <a href="https://gitlab.com/o9000/tint2/issues">issue tracker</a>;</li>
|
||||||
|
<li>Contribute to the development by helping us fix bugs and suggesting new features. Please read the contribution guide: <a href="CONTRIBUTING.md">CONTRIBUTING.md</a></li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="links">Links<a name="links" href="#links" class="md2man-permalink" title="permalink"></a></h1>
|
||||||
|
<ul>
|
||||||
|
<li>Home page: <a href="https://gitlab.com/o9000/tint2">https://gitlab.com/o9000/tint2</a></li>
|
||||||
|
<li>Git repository: <a href="https://gitlab.com/o9000/tint2.git">https://gitlab.com/o9000/tint2.git</a></li>
|
||||||
|
<li>Documentation: <a href="https://gitlab.com/o9000/tint2/wikis/home">https://gitlab.com/o9000/tint2/wikis/home</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>
|
||||||
|
</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>
|
||||||
|
<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.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.gif">Distribute size between taskbars, freespace</a></li>
|
||||||
|
</ul>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
910
doc/tint2.1
910
doc/tint2.1
@@ -1,51 +1,877 @@
|
|||||||
.\" Hey, EMACS: -*- nroff -*-
|
.TH TINT2 1 "2017\-04\-23" 0.14.3
|
||||||
.\" First parameter, NAME, should be all caps
|
|
||||||
.\" Second parameter, SECTION, should be 1-8, maybe w/ subsection
|
|
||||||
.\" other parameters are allowed: see man(7), man(1)
|
|
||||||
.TH TINT2 1 "2009-01-17"
|
|
||||||
.\" Please adjust this date whenever revising the manpage.
|
|
||||||
.\"
|
|
||||||
.\" Some roff macros, for reference:
|
|
||||||
.\" .nh disable hyphenation
|
|
||||||
.\" .hy enable hyphenation
|
|
||||||
.\" .ad l left justify
|
|
||||||
.\" .ad b justify to both left and right margins
|
|
||||||
.\" .nf disable filling
|
|
||||||
.\" .fi enable filling
|
|
||||||
.\" .br insert line break
|
|
||||||
.\" .sp <n> insert n+1 empty lines
|
|
||||||
.\" for manpage-specific macros, see man(7)
|
|
||||||
.SH NAME
|
.SH NAME
|
||||||
|
.PP
|
||||||
tint2 \- lightweight panel/taskbar
|
tint2 \- lightweight panel/taskbar
|
||||||
.SH SYNOPSIS
|
|
||||||
.B tint2
|
|
||||||
.br
|
|
||||||
.B tint2
|
|
||||||
.RI -c
|
|
||||||
.IR /path_to_config_file
|
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
This manual page documents briefly the
|
|
||||||
.B tint2
|
|
||||||
command.
|
|
||||||
.PP
|
.PP
|
||||||
.\" TeX users may be more comfortable with the \fB<whatever>\fP and
|
tint2 is a simple panel/taskbar made for modern X window managers.
|
||||||
.\" \fI<whatever>\fP escape sequences to invode bold face and italics,
|
It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.).
|
||||||
.\" respectively.
|
|
||||||
\fBtint2\fP is a simple panel/taskbar intentionally made for openbox3, but
|
|
||||||
should also work with other window managers. It's based on ttm code. The goal is to keep a clean and unintrusive look with code lightweight and
|
|
||||||
compliance with freedesktop specifications.
|
|
||||||
.PP
|
.PP
|
||||||
On the first startup tint2 creates a config file in $HOME/.config/tint2/tint2rc.
|
Features:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
Panel with taskbar, system tray, clock and launcher icons;
|
||||||
|
.IP \(bu 2
|
||||||
|
Easy to customize: color/transparency on fonts, icons, borders and backgrounds;
|
||||||
|
.IP \(bu 2
|
||||||
|
Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;
|
||||||
|
.IP \(bu 2
|
||||||
|
Multi\-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;
|
||||||
|
.IP \(bu 2
|
||||||
|
Customizable mouse events.
|
||||||
|
.RE
|
||||||
.PP
|
.PP
|
||||||
See wiki page on http://code.google.com/p/tint2/wiki/Welcome for more information.
|
Goals:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
Be unintrusive and light (in terms of memory, CPU and aesthetic);
|
||||||
|
.IP \(bu 2
|
||||||
|
Follow the freedesktop.org specifications;
|
||||||
|
.IP \(bu 2
|
||||||
|
Make certain workflows, such as multi\-desktop and multi\-monitor, easy to use.
|
||||||
|
.RE
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.PP
|
||||||
|
\fB\fCtint2 [OPTION...]\fR
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
.TP
|
.TP
|
||||||
.B \-c config-file
|
\fB\fC\-c path_to_config_file\fR
|
||||||
Specify which configuration file to use instead of the default.
|
Specifies which configuration file to use instead of the default.
|
||||||
.SH AUTHOR
|
.TP
|
||||||
tint2 was written by Thierry Lorthiois <lorthiois@bbsoft.fr>. It is based on
|
\fB\fC\-v, \-\-version\fR
|
||||||
ttm, originally written by Pål Staurland <staura@gmail.com>
|
Prints version information and exits.
|
||||||
|
.TP
|
||||||
|
\fB\fC\-h, \-\-help\fR
|
||||||
|
Display this help and exits.
|
||||||
|
.SH CONFIGURATION
|
||||||
|
.SS Table of contents
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
Introduction \[la]#introduction\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Backgrounds and borders \[la]#backgrounds-and-borders\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Gradients \[la]#gradients\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Panel \[la]#panel\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Launcher \[la]#launcher\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Taskbar/Pager \[la]#taskbar-pager\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Taskbar buttons \[la]#taskbar-buttons\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Mouse actions for taskbar buttons \[la]#mouse-actions-for-taskbar-buttons\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
System tray \[la]#system-tray\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Clock \[la]#clock\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Tooltip \[la]#tooltip\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Battery \[la]#battery\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Executor \[la]#executor\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Button \[la]#button\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Separator \[la]#separator\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
Example configuration \[la]#example-configuration\[ra]
|
||||||
|
.RE
|
||||||
|
.SS Introduction
|
||||||
.PP
|
.PP
|
||||||
This manual page was written by Daniel Moerner <dmoerner@gmail.com>,
|
These are instructions for configuring tint2 directly by editing its config file.
|
||||||
for the Debian project (but may be used by others). It was adopted from the
|
You may also use instead the graphical interface \fB\fCtint2conf\fR\&.
|
||||||
tint2 docs.
|
.PP
|
||||||
|
The first time you run tint2, it will create the config file in \fB\fC$HOME/.config/tint2/tint2rc\fR (This applies if you have done a clean install. Running tint2 in the source directory without doing 'make install' will not create the config file.)
|
||||||
|
.PP
|
||||||
|
You can also specify another file on the command line with the \-c option, e.g.: \fB\fCtint2 \-c $HOME/tint2.conf\fR\&. This can be used to run multiple instances of tint2 that use different settings.
|
||||||
|
.PP
|
||||||
|
If you change the config file while tint2 is running, the command \fB\fCkillall \-SIGUSR1 tint2\fR will force tint2 to reload it.
|
||||||
|
.PP
|
||||||
|
All the configuration options supported in the config file are listed below.
|
||||||
|
Try to respect as much as possible the order of the options as given below.
|
||||||
|
.SS Backgrounds and borders
|
||||||
|
.PP
|
||||||
|
The tint2 config file starts with the options defining background elements with borders:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCrounded = number_of_pixels\fR : the corner radius
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCborder_width = integer\fR : the border width in pixels
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCborder_sides = LRTB\fR : the sides to draw the border on (left, right, top, bottom). If not specified, all sides are used. \fI(since 0.12.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbackground_color = color opacity\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz).
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCborder_color = color opacity\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbackground_color_hover = color opacity\fR (default: same as \fB\fCbackground_color\fR) \fI(since 0.12.3)\fP
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCborder_color_hover = color opacity\fR (default: same as \fB\fCborder_color\fR) \fI(since 0.12.3)\fP
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbackground_color_pressed = color opacity\fR (default: same as \fB\fCbackground_color_hover\fR) \fI(since 0.12.3)\fP
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCborder_color_pressed = color opacity\fR (default: same as \fB\fCborder_color_hover\fR) \fI(since 0.12.3)\fP
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcolor\fR is specified in hex RGB, e.g. #ff0000 is red
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCopacity\fR varies from (0 to 100), where 0 is fully transparent, 100 is fully opaque
|
||||||
|
.RE
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
You can define as many backgrounds as you want. For example, the following config defines two backgrounds:
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
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
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
tint2 automatically identifies each background with a number starting from 1 (1, 2, ...).
|
||||||
|
Afterwards, you can apply a background to objects (panel, taskbar, task, clock, systray) using the background id, for example:
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
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
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
Identifier 0 refers to a special background which is fully transparent, identifier 1 applies the first background defined in the config file etc.
|
||||||
|
.SS Gradients
|
||||||
|
.PP
|
||||||
|
(Available since 0.13.0)
|
||||||
|
.PP
|
||||||
|
Backgrounds also allow specifying gradient layers
|
||||||
|
that are drawn on top of the solid color background.
|
||||||
|
.PP
|
||||||
|
First the user must define one or more gradients in the config file,
|
||||||
|
each starting with \fB\fCgradient = TYPE\fR\&. These must be added before backgrounds.
|
||||||
|
.PP
|
||||||
|
Then gradients can be added by index to backgrounds,
|
||||||
|
using the \fB\fCgradient_id = INDEX\fR, \fB\fCgradient_id_hover = INDEX\fR and
|
||||||
|
\fB\fCgradient_id_pressed = INDEX\fR, where \fB\fCINDEX\fR is
|
||||||
|
the gradient index, starting from 1.
|
||||||
|
.SS Gradient types
|
||||||
|
.PP
|
||||||
|
Gradients vary the color between fixed control points:
|
||||||
|
* vertical gradients: top\-to\-bottom;
|
||||||
|
* horizontal gradients: left\-to\-right;
|
||||||
|
* radial gradients: center\-to\-corners.
|
||||||
|
.PP
|
||||||
|
The user must specify the start and end colors, and can optionally add extra color stops in between
|
||||||
|
using the \fB\fCcolor_stop\fR option, as explained below.
|
||||||
|
.SS Vertical gradient, with color varying from the top edge to the bottom edge, two colors
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
gradient = vertical
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Horizontal gradient, with color varying from the left edge to the right edge, two colors
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Radial gradient, with color varying from the center to the corner, two colors:
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
gradient = radial
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Adding extra color stops (0% and 100% remain fixed, more colors at x% between the start and end control points)
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
color_stop = percentage #rrggbb opacity
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Gradient examples
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
# Gradient 1: thin film effect
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111122 30
|
||||||
|
end_color = #112211 30
|
||||||
|
color_stop = 60 #221111 30
|
||||||
|
|
||||||
|
# Gradient 2: radial glow
|
||||||
|
gradient = radial
|
||||||
|
start_color = #ffffff 20
|
||||||
|
end_color = #ffffff 0
|
||||||
|
|
||||||
|
# Gradient 3: elegant black
|
||||||
|
gradient = vertical
|
||||||
|
start_color = #444444 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Gradient 4: elegant black
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111111 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Background 1: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
border_sides = TBLR
|
||||||
|
background_color = #555555 10
|
||||||
|
border_color = #ffffff 60
|
||||||
|
background_color_hover = #555555 10
|
||||||
|
border_color_hover = #ffffff 60
|
||||||
|
background_color_pressed = #555555 10
|
||||||
|
border_color_pressed = #ffffff 60
|
||||||
|
gradient_id = 3
|
||||||
|
gradient_id_hover = 4
|
||||||
|
gradient_id_pressed = 2
|
||||||
|
|
||||||
|
[...]
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Panel
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_items = LTSBC\fR defines the items tint2 will show and the order of those items. Each letter refers to an item, defined as:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCL\fR shows the Launcher
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCT\fR shows the Taskbar
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCS\fR shows the Systray (also called notification area)
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCB\fR shows the Battery status
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCC\fR shows the Clock
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCF\fR adds an extensible spacer (freespace). You can specify more than one. Has no effect if \fB\fCT\fR is also present. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCE\fR adds an executor plugin. You can specify more than one. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCP\fR adds a push button. You can specify more than one. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fC:\fR adds a separator. You can specify more than one. \fI(since 0.13.0)\fP
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
For example, \fB\fCpanel_items = STC\fR will show the systray, the taskbar and the clock (from left to right).
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_monitor = monitor (all or 1 or 2 or ...)\fR : Which monitor tint2 draws the panel on
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
The first monitor is \fB\fC1\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
Use \fB\fCpanel_monitor = all\fR to get a separate panel per monitor
|
||||||
|
.RE
|
||||||
|
.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
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
[](images/panel_padding.jpg)
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_position = vertical_position horizontal_position orientation\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCvertical_position\fR is one of: \fB\fCbottom\fR, \fB\fCtop\fR, \fB\fCcenter\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fChorizontal_position\fR is one of: \fB\fCleft\fR, \fB\fCright\fR, \fB\fCcenter\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCorientation\fR is one of: \fB\fChorizontal\fR, \fB\fCvertical\fR
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_size = width height\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCwidth\fR and \fB\fCheight\fR can be specified without units (e.g. \fB\fC123\fR) as pixels, or followed by \fB\fC%\fR as percentages of the monitor size (e.g. \fB\fC50%\fR). Use \fB\fC100%\fR for full monitor width/height.
|
||||||
|
Example:
|
||||||
|
.RE
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
# The panel's width is 94% the size of the monitor, the height is 30 pixels:
|
||||||
|
panel_size = 94% 30
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_shrink = boolean (0 or 1)\fR : If set to 1, the panel will shrink to a compact size dynamically. \fI(since 0.13)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_margin = horizontal_margin vertical_margin\fR : The margins define the distance between the panel and the horizontal/vertical monitor edge. Use \fB\fC0\fR to obtain a panel with the same size as the edge of the monitor (no margin).
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
[](images/panel\fIsize\fPmargin.jpg)
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_padding = horizontal_padding vertical_padding spacing\fR : Please refer to the image below.
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
[](images/panel_padding.jpg)
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCfont_shadow = boolean (0 or 1)\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_background_id = integer\fR : Which background to use for the panel.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCwm_menu = boolean (0 or 1)\fR : Defines if tint2 forwards unhandled mouse events to your window manager. Useful for window managers such as openbox, which display the start menu if you right click on the desktop.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_dock = boolean (0 or 1)\fR : Defines if tint2 is placed into the window manager's dock. For the openbox window manager it is advised to also use a modifier for the moveButton option, otherwise the mouse click is not forwarded to tint2 (in ~/.config/openbox/rc.xml).
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_layer = bottom/normal/top\fR : Places tint2 into the bottom/normal/top layer. This is helpful for specifying if the panel can be covered by other windows or not. The default is the bottom layer, but with real transparency normal or top layer may be a nice alternative.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCstrut_policy = follow_size/minimum/none\fR : STRUTs are used by the window manager to decide the size of maximized windows. Note: on multi\-monitor (Xinerama) setups, the panel must be placed at the edge (not in the middle) of the virtual screen for this to work correctly.
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCfollow_size\fR means that the maximized windows always resize to have a common edge with tint2.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCminimum\fR means that the maximized windows always expand to have a common edge with the hidden panel. This is useful if the \fB\fCautohide\fR option is enabled.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCnone\fR means that the maximized windows use the full screen size.
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCpanel_window_name = string\fR : Defines the name of the panel's window. Default: 'tint2'. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCdisable_transparency = boolean (0 or 1)\fR : Whether to disable transparency instead of detecting if it is supported. Useful on broken graphics stacks. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCmouse_effects = boolean (0 or 1)\fR : Whether to enable mouse hover effects for clickable items. \fI(since 0.12.3)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCmouse_hover_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjusts the icon color and transparency on mouse hover (works only when mouse_effects = 1).` \fI(since 0.12.3)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCmouse_pressed_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjusts the icon color and transparency on mouse press (works only when mouse_effects = 1).` \fI(since 0.12.3)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCautohide = boolean (0 or 1)\fR : Whether to enable panel hiding when the mouse cursor exists the panel.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCautohide_show_timeout = float\fR : Show timeout in seconds after the mouse cursor enters the panel. Use '.' as decimal separator.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCautohide_hide_timeout = float\fR : Hide timeout in seconds after the mouse cursor exits the panel. Use '.' as decimal separator.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCautohide_height = integer\fR : panel height (width for vertical panels) in hidden mode.
|
||||||
|
.RE
|
||||||
|
.SS Launcher
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_item_app = path_to_application\fR : Each \fB\fClauncher_item_app\fR must be a file path to a .desktop file following the freedesktop.org specification \[la]http://standards.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html\[ra]\&. The paths may begin with \fB\fC~\fR, which is expanded to the path of the user's home directory. If only a file name is specified, the file is search in the standard application directories (\fB\fC$XDG_DATA_HOME/applications\fR, \fB\fC~/.local/share/applications\fR, \fB\fC$XDG_DATA_DIRS/applications\fR, \fB\fC/usr/local/share/applications\fR, \fB\fC/usr/share/applications\fR, \fB\fC/opt/share/applications\fR).
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_apps_dir = path_to_directory\fR : Specifies a path to a directory from which the launcher is loading all .desktop files (all subdirectories are explored recursively). Can be used multiple times. The path may begin with \fB\fC~\fR, which is expanded to the path of the user's home directory. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_background_id = integer\fR : Defines which background to use.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_icon_background_id = integer\fR : Defines which background to use for icons.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_padding = horizontal_padding vertical_padding spacing\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_icon_size = integer\fR : The launcher icon size, in pixels.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_icon_theme = name_of_theme\fR : (Optional) Uses the specified icon theme to display shortcut icons. Note that tint2 will detect and use the icon theme of your desktop if you have an XSETTINGS manager running (which you probably do), unless \fB\fClauncher_icon_theme_override = 1\fR\&.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_icon_theme_override = boolean (0 or 1)\fR : Whether \fB\fClauncher_icon_theme\fR overrides the value obtained from the XSETTINGS manager. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjusts the icon color and transparency.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClauncher_tooltip = boolean (0 or 1)\fR : Whether to show tooltips for the launcher icons.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCstartup_notifications = boolean (0 or 1)\fR : Whether to show startup notifications when starting applications from the launcher. \fI(since 0.12)\fP
|
||||||
|
.RE
|
||||||
|
.SS Taskbar / Pager
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_mode = single_desktop/multi_desktop\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCsingle_desktop\fR : Shows a normal taskbar listing the tasks running on the current virtual desktop (also known as 'workspace');
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCmulti_desktop\fR : Pager like capability. Shows multiple taskbars, one per virtual desktop, with which:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
You can drag\-and\-drop tasks between virtual desktops;
|
||||||
|
.IP \(bu 2
|
||||||
|
You can switch between virtual desktops.
|
||||||
|
.RE
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_hide_if_empty = boolean (0 or 1)\fR : If enabled, in multi\-desktop mode the taskbars corresponding to empty desktops different from the current desktop are hidden. \fI(since 0.13)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_distribute_size = boolean (0 or 1)\fR : If enabled, in multi\-desktop mode distributes between taskbars the available size proportionally to the number of tasks. Default: disabled. \fI(since 0.12)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_padding = horizontal_padding vertical_padding spacing\fR
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
[](images/taskbar_padding.jpg)
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_background_id = integer\fR : Which background to use
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_active_background_id = integer\fR : Which background to use for the taskbar of the current virtual desktop.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_hide_inactive_tasks = boolean (0 or 1)\fR : If enabled, the taskbar shows only the active task. \fI(since 0.12)\fP
|
||||||
|
.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
|
||||||
|
.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
|
||||||
|
.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
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCnone\fR : No sorting. New tasks are simply appended at the end of the taskbar when they appear.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtitle\fR : Sorts the tasks by title.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCcenter\fR : Sorts the tasks by their window centers.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCmru\fR : Shows the most recently used tasks first. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fClru\fR : Shows the most recently used tasks last. \fI(since 0.12.4)\fP
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_align = left/center/right\fR : Specifies the alignment of the tasks on the taskbar. Default: left.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name = boolean (0 or 1)\fR : Whether to show the virtual desktop name in the taskbar.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_padding = padding\fR : Padding for the virtual desktop name.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_background_id = integer\fR : Which background to use for the desktop name.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR : Font configuration for the desktop name.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_font_color = color opacity (0 to 100)\fR : Font color for the desktop name.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_active_background_id = integer\fR : Which background to use for the name of the current desktop.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtaskbar_name_active_font_color = color opacity (0 to 100)\fR : Font color for the name of the current desktop.
|
||||||
|
.RE
|
||||||
|
.SH Taskbar buttons
|
||||||
|
.PP
|
||||||
|
The following options configure the task buttons in the taskbar:
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_icon = boolean (0 or 1)\fR : Whether to display the task icon. There is no explicit option to control the task icon size; it depends on the vertical padding set with \fB\fCtask_padding\fR\&.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_text = boolean (0 or 1)\fR : Whether to display the task text.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_centered = boolean (0 or 1)\fR : Whether the task text is centered.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_tooltip = boolean (0 or 1)\fR : Whether to show tooltips for tasks.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_maximum_size = width height\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCwidth\fR is used with horizontal panels to limit the size of the tasks. Use \fB\fCwidth = 0\fR to get full taskbar width.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCheight\fR is used with vertical panels.
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_padding = horizontal_padding vertical_padding spacing\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCurgent_nb_of_blink = integer\fR : Number of blinks on 'get attention' events.
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
|
[](images/task_padding.jpg)
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_font_color = color opacity (0 to 100)\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjust the task icon's color and transparency.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_background_id = integer\fR : Which background to use for non selected tasks
|
||||||
|
.RE
|
||||||
|
.TP
|
||||||
|
For the next 3 options STATUS can be \fB\fCactive\fR / \fB\fCiconified\fR / \fB\fCurgent\fR:
|
||||||
|
* \fB\fCtask_STATUS_font_color = color opacity (0 to 100)\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_STATUS_icon_asb = alpha (0 to 100) saturation (\-100 to 100) brightness (\-100 to 100)\fR : Adjusts the task icon's color and transparency.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtask_STATUS_background_id = integer\fR : Which background to use for the task.
|
||||||
|
.RE
|
||||||
|
.SS Mouse actions for taskbar buttons
|
||||||
|
.PP
|
||||||
|
The possible mouse events are: \fB\fCleft, middle, right, scroll_up, scroll_down\fR\&.
|
||||||
|
.PP
|
||||||
|
The possible mouse actions are: \fB\fCnone, close, toggle, iconify, shade, toggle_iconify, maximize_restore, desktop_left, desktop_right, next_task, prev_task\fR\&.
|
||||||
|
.PP
|
||||||
|
Use \fB\fCmouse_event = action\fR to customize mouse actions. Example:
|
||||||
|
\fB\fC
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = toggle
|
||||||
|
mouse_scroll_down = iconify
|
||||||
|
\fR
|
||||||
|
.TP
|
||||||
|
The action semantics:
|
||||||
|
* \fB\fCnone\fR : If \fB\fCwm_menu = 1\fR is set, the mouse event is forwarded to the window manager. Otherwise it is ignored.
|
||||||
|
* \fB\fCclose\fR : close the task
|
||||||
|
* \fB\fCtoggle\fR : toggle the task
|
||||||
|
* \fB\fCiconify\fR : iconify (minimize) the task
|
||||||
|
* \fB\fCtoggle_iconify\fR : toggle or iconify the task
|
||||||
|
* \fB\fCmaximize_restore\fR : maximized or minimized the task
|
||||||
|
* \fB\fCshade\fR : shades (collapses) the task
|
||||||
|
* \fB\fCdesktop_left\fR : send the task to the desktop on the left
|
||||||
|
* \fB\fCdesktop_right\fR : send the task to the desktop on the right
|
||||||
|
* \fB\fCnext_task\fR : send the focus to next task
|
||||||
|
* \fB\fCprev_task\fR : send the focus to previous task
|
||||||
|
.SS System Tray
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCsystray_padding = horizontal_padding vertical_padding spacing\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCsystray_background_id = integer\fR : Which background to use.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCsystray_sort = ascending/descending/left2right/right2left\fR : 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 \fB\fCleft2right\fR or \fB\fCright2left\fR the order can be different on panel restart).
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCsystray_icon_size = max_icon_size\fR : Set the maximum system tray icon size to \fB\fCnumber\fR\&. Set to \fB\fC0\fR for automatic icon sizing.
|
||||||
|
.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.
|
||||||
|
.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
|
||||||
|
.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
|
||||||
|
.RE
|
||||||
|
.SS Clock
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime1_format = %H:%M\fR : The format used by the first line of the clock.
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime1_format\fR, \fB\fCtime2_format\fR and \fB\fCclock_tooltip\fR use the 'strftime' syntax. More info can be found here: \[la]http://www.manpagez.com/man/3/strftime/\[ra]
|
||||||
|
.IP \(bu 2
|
||||||
|
To hide the clock, comment \fB\fCtime1_format\fR and \fB\fCtime2_format\fR\&.
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime1_timezone = :US/Hawaii\fR
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime1_timezone\fR, \fB\fCtime2_timezone\fR and \fB\fCclock_tooltip_timezone\fR 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 \fB\fC/usr/share/zoneinfo\fR\&. If your timezones are in a different directory, you need to specify the absolute path, e.g. \fB\fCtime1_timezone = :/different/zoneinfo/dir/US/Hawaii\fR Always prepend the timezone with a ':'
|
||||||
|
.RE
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime1_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime2_format = %A %d %B\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime2_timezone = :Europe/Berlin\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtime2_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_font_color = color opacity (0 to 100)\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_padding = horizontal_padding vertical_padding\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_background_id = integer\fR : Which background to use
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_tooltip = %a, %d. %b %Y\fR : Format for the clock's tooltip.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_tooltip_timezone = :UTC\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_lclick_command = text\fR : Command to execute on left click.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_rclick_command = text\fR : Command to execute on right click.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_mclick_command = text\fR : Command to execute on middle click. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_uwheel_command = text\fR : Command to execute on wheel scroll up. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCclock_dwheel_command = text\fR : Command to execute on wheel scroll down. \fI(since 0.12.1)\fP
|
||||||
|
.RE
|
||||||
|
.SS Tooltip
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_padding = horizontal_padding vertical_padding\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_show_timeout = float\fR : Delay to show the tooltip in seconds. Use \fB\fC\&.\fR as decimal separator.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_hide_timeout = float\fR : Delay to hide the tooltip in seconds. Use \fB\fC\&.\fR as decimal separator.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_background_id = integer\fR : Which background to use for tooltips. Note that with fake transparency the alpha channel and corner radius options are not respected.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_font_color = color opacity (0 to 100)\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCtooltip_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.RE
|
||||||
|
.SS Battery
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_hide = never/integer (0 to 100)\fR : At what battery percentage the battery item is hidden.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_low_status = integer\fR: At what battery percentage the low command is executed.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_low_cmd = notify\-send "battery low"\fR : Command to execute when the battery is low.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbat1_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbat2_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_font_color = color opacity (0 to 100)\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_padding = horizontal_padding vertical_padding\fR
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_background_id = integer\fR : Which background to use for the battery.
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_tooltip_enabled = boolean (0 or 1)\fR : Enable/disable battery tooltips. \fI(since 0.12.3)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_lclick_command = text\fR : Command to execute on left click. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_rclick_command = text\fR : Command to execute on right click. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_mclick_command = text\fR : Command to execute on middle click. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_uwheel_command = text\fR : Command to execute on wheel scroll up. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbattery_dwheel_command = text\fR : Command to execute on wheel scroll down. \fI(since 0.12.1)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCac_connected_cmd = text\fR : Command to execute when the power adapter is plugged in. \fI(since 0.12.3)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCac_disconnected_cmd = text\fR : Command to execute when the power adapter is unplugged. \fI(since 0.12.3)\fP
|
||||||
|
.RE
|
||||||
|
.SS Executor
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp = new\fR : Begins the configuration of a new executor plugin. Multiple such plugins are supported; just use multiple \fB\fCE\fRs in \fB\fCpanel_items\fR\&. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_command = text\fR : Command to execute. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_interval = integer\fR : The command is executed again after \fB\fCexecp_interval\fR seconds from the moment it exits. If zero, the command is executed only once. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_continuous = integer\fR : If non\-zero, the last \fB\fCexecp_continuous\fR lines from the output of the command are displayed, every \fB\fCexecp_continuous\fR lines; this is useful for showing the output of commands that run indefinitely, such as \fB\fCping 127.0.0.1\fR\&. If zero, the output of the command is displayed after it finishes executing. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_has_icon = boolean (0 or 1)\fR : If \fB\fCexecp_has_icon = 1\fR, the first line printed by the command is interpreted as a path to an image file. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_cache_icon = boolean (0 or 1)\fR : If \fB\fCexecp_cache_icon = 0\fR, the image is reloaded each time the command is executed (useful if the image file is changed on disk by the program executed by \fB\fCexecp_command\fR). \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_icon_w = integer\fR : You can use \fB\fCexecp_icon_w\fR and \fB\fCexecp_icon_h\fR 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. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_icon_h = integer\fR : See \fB\fCexecp_icon_w\fR\&. \fI(since 0.12.4)\fP
|
||||||
|
.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
|
||||||
|
.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
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_font_color = color opacity\fR : The font color. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_markup = boolean (0 or 1)\fR : If non\-zero, the output of the command is treated as Pango markup, which allows rich text formatting. The format is documented here \[la]https://developer.gnome.org/pygtk/stable/pango-markup-language.html\[ra]\&. Note that using this with commands that print data downloaded from the Internet is a possible security risk. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_background_id = integer\fR : Which background to use. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_centered = boolean (0 or 1)\fR : Whether to center the text. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_padding = horizontal_padding vertical_padding spacing_between_icon_and_text\fR \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_lclick_command = text\fR : Command to execute on left click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_mclick_command = text\fR : Command to execute on right click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_rclick_command = text\fR : Command to execute on middle click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_uwheel_command = text\fR : Command to execute on wheel scroll up. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.12.4)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCexecp_dwheel_command = text\fR : Command to execute on wheel scroll down. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.12.4)\fP
|
||||||
|
.RE
|
||||||
|
.SS Executor samples
|
||||||
|
.SS Print the hostname
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = hostname
|
||||||
|
execp_interval = 0
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Print disk usage for the root partition every 10 seconds
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = df \-h | awk '/\\/$/ { print $6 ": " $2 " " $5}'
|
||||||
|
execp_interval = 10
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Button with icon and rich text, executes command when clicked
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = echo /usr/share/icons/elementary\-xfce/emblems/24/emblem\-colors\-blue.png; echo '<span foreground="#7f7">Click</span> <span foreground="#77f">me</span> <span foreground="#f77">pls</span>'
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_interval = 0
|
||||||
|
execp_centered = 1
|
||||||
|
execp_font = sans 9
|
||||||
|
execp_markup = 1
|
||||||
|
execp_font_color = #aaffaa 100
|
||||||
|
execp_padding = 2 0
|
||||||
|
execp_tooltip = I will tell you a secret...
|
||||||
|
execp_lclick_command = zenity \-\-info "\-\-text=$(uname \-sr)"
|
||||||
|
execp_background_id = 2
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Desktop pager with text
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = xprop \-root \-spy | awk '/^_NET_CURRENT_DESKTOP/ { print "Workspace " ($3 + 1) ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_continuous = 1
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Desktop pager with icon
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp_command = xprop \-root \-spy | awk \-v home="$HOME" '/^_NET_CURRENT_DESKTOP/ { print home "/.config/myPager/" ($3 + 1) ".png\\n" ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_cache_icon = 1
|
||||||
|
execp_continuous = 2
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Round\-trip time to the gateway, refreshed every second
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = ping \-i 1 \-c 1 \-W 1 \-O \-D \-n $(ip route | grep default | grep via | grep \-o '[0\-9]*\\.[0\-9]*\\.[0\-9]*\\.[0\-9]*') | awk '/no/ { print "<span foreground=\\"#faa\\">timeout</span>"; fflush(); }; /time=/ { gsub(/time=/, "", $8); printf "<span foreground=\\"#7af\\">%3.0f %s</span>\\n", $8, $9; fflush(); } '
|
||||||
|
execp_continuous = 0
|
||||||
|
execp_interval = 1
|
||||||
|
execp_markup = 1
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Memory usage
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
execp = new
|
||||||
|
execp_command = free | awk '/^\-/ { printf "Mem: '$(free \-h | awk '/^Mem:/ { print $2 }')' %.0f%%\\n", 100*$3/($3+$4); fflush(stdout) }'
|
||||||
|
execp_interval = 5
|
||||||
|
execp_continuous = 0
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Network load
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
# Note the use of "stdbuf \-oL" to force the program to flush the output line by line.
|
||||||
|
execp = new
|
||||||
|
execp_command = stdbuf \-oL bwm\-ng \-o csv \-t 1000 | awk \-F ';' '/total/ { printf "Net: %.0f Mb/s\\n", ($5*8/1.0e6) }; fflush(stdout)'
|
||||||
|
execp_continuous = 1
|
||||||
|
execp_interval = 1
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
|
.SS Button
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton = new\fR : Begins the configuration of a new button. Multiple such plugins are supported; just use multiple \fB\fCP\fRs in \fB\fCpanel_items\fR\&. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_icon = text\fR : Name or path of icon (or empty). \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_text = text\fR : Text to display (or empty). \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_tooltip = text\fR : The tooltip (or empty). \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_font = [FAMILY\-LIST] [STYLE\-OPTIONS] [SIZE]\fR : The font used to draw the text. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_font_color = color opacity\fR : The font color. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_background_id = integer\fR : Which background to use. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_centered = boolean (0 or 1)\fR : Whether to center the text. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_padding = horizontal_padding vertical_padding spacing_between_icon_and_text\fR \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_max_icon_size = integer\fR : Sets a limit to the icon size. Otherwise, the icon will expand to the edges. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_lclick_command = text\fR : Command to execute on left click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_mclick_command = text\fR : Command to execute on right click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_rclick_command = text\fR : Command to execute on middle click. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_uwheel_command = text\fR : Command to execute on wheel scroll up. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.14)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCbutton_dwheel_command = text\fR : Command to execute on wheel scroll down. If not defined, \fB\fCexecp_command\fR is executed immediately, unless it is currently running. \fI(since 0.14)\fP
|
||||||
|
.RE
|
||||||
|
.SS Separator
|
||||||
|
.RS
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator = new\fR : Begins the configuration of a new separator. Multiple such plugins are supported; just use multiple \fB\fC:\fRs in \fB\fCpanel_items\fR\&. \fI(since 0.13.0)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator_background_id = integer\fR : Which background to use. \fI(since 0.13.0)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator_color = color opacity\fR : The foreground color. \fI(since 0.13.0)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator_style = [empty | line | dots]\fR : The separator style. \fI(since 0.13.0)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator_size = integer\fR : The thickness of the separator. Does not include the border and padding. For example, if the style is \fB\fCline\fR, this is the line thickness; if the style is \fB\fCdots\fR, this is the dot's diameter. \fI(since 0.13.0)\fP
|
||||||
|
.IP \(bu 2
|
||||||
|
\fB\fCseparator_padding = side_padding cap_padding\fR : The padding to add to the sides of the separator, in pixels. \fI(since 0.13.0)\fP
|
||||||
|
.RE
|
||||||
|
.SS Example configuration
|
||||||
|
.PP
|
||||||
|
See /etc/xdg/tint2/tint2rc.
|
||||||
|
.SH AUTHOR
|
||||||
|
.PP
|
||||||
|
tint2 was written by Thierry Lorthiois \[la]lorthiois@bbsoft.fr\[ra]\&.
|
||||||
|
It is based on ttm, originally written by Pål Staurland \[la]staura@gmail.com\[ra]\&.
|
||||||
|
.PP
|
||||||
|
This manual page was originally written by Daniel Moerner \[la]dmoerner@gmail.com\[ra], for the Debian project (but may be used by others).
|
||||||
|
It was adopted from the tint2 docs.
|
||||||
|
.SH SEE ALSO
|
||||||
|
.PP
|
||||||
|
The main website \[la]https://gitlab.com/o9000/tint2\[ra]
|
||||||
|
and the wiki page at \[la]https://gitlab.com/o9000/tint2/wikis/home\[ra]\&.
|
||||||
|
.PP
|
||||||
|
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\&.
|
||||||
|
|||||||
524
doc/tint2.html
Normal file
524
doc/tint2.html
Normal file
@@ -0,0 +1,524 @@
|
|||||||
|
<h1 id="tint2-1-2016-05-22"><span class="md2man-title">TINT2</span> <span class="md2man-section">1</span> <span class="md2man-date">2016-05-22</span><a name="tint2-1-2016-05-22" href="#tint2-1-2016-05-22" 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.
|
||||||
|
It was specifically made for Openbox but it should also work with other window managers (GNOME, KDE, XFCE etc.).</p><p>Features:</p>
|
||||||
|
<ul>
|
||||||
|
<li>Panel with taskbar, system tray, clock and launcher icons;</li>
|
||||||
|
<li>Easy to customize: color/transparency on fonts, icons, borders and backgrounds;</li>
|
||||||
|
<li>Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;</li>
|
||||||
|
<li>Multi-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;</li>
|
||||||
|
<li>Customizable mouse events.</li>
|
||||||
|
</ul>
|
||||||
|
<p>Goals:</p>
|
||||||
|
<ul>
|
||||||
|
<li>Be unintrusive and light (in terms of memory, CPU and aesthetic);</li>
|
||||||
|
<li>Follow the freedesktop.org specifications;</li>
|
||||||
|
<li>Make certain workflows, such as multi-desktop and multi-monitor, easy to use.</li>
|
||||||
|
</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>
|
||||||
|
<ul>
|
||||||
|
<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="#panel">Panel</a></p></li>
|
||||||
|
<li><p><a href="#launcher">Launcher</a></p></li>
|
||||||
|
<li><p><a href="#taskbar-pager">Taskbar/Pager</a></p></li>
|
||||||
|
<li><p><a href="#taskbar-buttons">Taskbar buttons</a></p></li>
|
||||||
|
<li><p><a href="#mouse-actions-for-taskbar-buttons">Mouse actions for taskbar buttons</a></p></li>
|
||||||
|
<li><p><a href="#system-tray">System tray</a></p></li>
|
||||||
|
<li><p><a href="#clock">Clock</a></p></li>
|
||||||
|
<li><p><a href="#tooltip">Tooltip</a></p></li>
|
||||||
|
<li><p><a href="#battery">Battery</a></p></li>
|
||||||
|
<li><p><a href="#executor">Executor</a></p></li>
|
||||||
|
<li><p><a href="#example-configuration">Example configuration</a></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="introduction">Introduction<a name="introduction" href="#introduction" class="md2man-permalink" title="permalink"></a></h3><p>These are instructions for configuring tint2 directly by editing its config file.
|
||||||
|
You may also use instead the graphical interface <code>tint2conf</code>.</p><p>The first time you run tint2, it will create the config file in <code>$HOME/.config/tint2/tint2rc</code> (This applies 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 also specify another file on the command line with the -c option, e.g.: <code>tint2 -c $HOME/tint2.conf</code>. This can be used to run multiple instances of tint2 that use different settings.</p><p>If you change the config file while tint2 is running, the command <code>killall -SIGUSR1 tint2</code> will force tint2 to reload it.</p><p>All the configuration options supported in the config file are listed below.
|
||||||
|
Try to respect as much as possible the order of the options as given below.</p><h3 id="backgrounds-and-borders">Backgrounds and borders<a name="backgrounds-and-borders" href="#backgrounds-and-borders" class="md2man-permalink" title="permalink"></a></h3><p>The tint2 config file starts with the options defining background elements with borders:</p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>rounded = number_of_pixels</code> : the corner radius</p></li>
|
||||||
|
<li><p><code>border_width = integer</code> : the border width in pixels</p></li>
|
||||||
|
<li><p><code>border_sides = LRTB</code> : the sides to draw the border on (left, right, top, bottom). If not specified, all sides are used.</p></li>
|
||||||
|
<li><p><code>background_color = color opacity</code></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz).</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color = color opacity</code></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>background_color_hover = color opacity</code> (default: same as background_color) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color_hover = color opacity</code> (default: same as border_color) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>background_color_pressed = color opacity</code> (default: same as background<em>color</em>hover) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>border_color_pressed = color opacity</code> (default: same as border<em>color</em>hover) <em>(since 0.12.3)</em></p>
|
||||||
|
<ul>
|
||||||
|
<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>
|
||||||
|
</ul></li>
|
||||||
|
</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
|
||||||
|
border_width = 0
|
||||||
|
background_color = #282828 100
|
||||||
|
border_color = #000000 0
|
||||||
|
|
||||||
|
rounded = 1
|
||||||
|
border_width = 0
|
||||||
|
background_color = #f6b655 90
|
||||||
|
border_color = #cccccc 40
|
||||||
|
</code></pre>
|
||||||
|
<p>tint2 automatically identifies each background with a number starting from 1 (1, 2, ...).
|
||||||
|
Afterwards, you can apply a background to objects (panel, taskbar, task, clock, systray) using the background id, for example:</p><pre class="highlight plaintext"><code>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
|
||||||
|
</code></pre>
|
||||||
|
<p>Identifier 0 refers to a special background which is fully transparent, identifier 1 applies the first background defined in the config file etc.</p><h3 id="panel">Panel<a name="panel" href="#panel" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>panel_items = LTSBC</code> defines the items tint2 will show and the order of those items. Each letter refers to an item, defined as:</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>L</code> shows the Launcher</li>
|
||||||
|
<li><code>T</code> shows the Taskbar</li>
|
||||||
|
<li><code>S</code> shows the Systray (also called notification area)</li>
|
||||||
|
<li><code>B</code> shows the Battery status</li>
|
||||||
|
<li><code>C</code> shows the Clock</li>
|
||||||
|
<li><code>F</code> adds an extensible spacer (freespace). Has no effect if <code>T</code> is also present. <em>(since 0.12)</em></li>
|
||||||
|
<li><code>E</code> adds an executor plugin. You can specify more than one. <em>(since 0.12.4)</em></li>
|
||||||
|
</ul>
|
||||||
|
<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>
|
||||||
|
<ul>
|
||||||
|
<li>The first monitor is <code>1</code></li>
|
||||||
|
<li>Use <code>panel_monitor = all</code> to get a separate panel per monitor</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>
|
||||||
|
</ul>
|
||||||
|
<p><img src="panel_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>panel_position = vertical_position horizontal_position orientation</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>vertical_position</code> is one of: <code>bottom</code>, <code>top</code>, <code>center</code></li>
|
||||||
|
<li><code>horizontal_position</code> is one of: <code>left</code>, <code>right</code>, <code>center</code></li>
|
||||||
|
<li><code>orientation</code> is one of: <code>horizontal</code>, <code>vertical</code></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>panel_size = width height</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>width</code> and <code>height</code> can be specified without units (e.g. <code>123</code>) as pixels, or followed by <code>%</code> as percentages of the monitor size (e.g. <code>50%</code>). Use <code>100%</code> for full monitor width/height.
|
||||||
|
Example:</li>
|
||||||
|
</ul></li>
|
||||||
|
</ul>
|
||||||
|
<pre class="highlight plaintext"><code># The panel's width is 94% the size of the monitor, the height is 30 pixels:
|
||||||
|
panel_size = 94% 30
|
||||||
|
</code></pre>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><code>panel_margin = horizontal_margin vertical_margin</code> : The margins define the distance between the panel and the horizontal/vertical monitor edge. Use <code>0</code> to obtain a panel with the same size as the edge of the monitor (no margin).</li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="panel_size_margin.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>panel_padding = horizontal_padding vertical_padding spacing</code> : Please refer to the image below.</li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="panel_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>font_shadow = boolean (0 or 1)</code></p></li>
|
||||||
|
<li><p><code>panel_background_id = integer</code> : Which background to use for the panel.</p></li>
|
||||||
|
<li><p><code>wm_menu = boolean (0 or 1)</code> : Defines if tint2 forwards unhandled mouse events to your window manager. Useful for window managers such as openbox, which display the start menu if you right click on the desktop.</p></li>
|
||||||
|
<li><p><code>panel_dock = boolean (0 or 1)</code> : Defines if tint2 is placed into the window manager's dock. For the openbox window manager it is advised to also use a modifier for the moveButton option, otherwise the mouse click is not forwarded to tint2 (in ~/.config/openbox/rc.xml).</p></li>
|
||||||
|
<li><p><code>panel_layer = bottom/normal/top</code> : Places tint2 into the bottom/normal/top layer. This is helpful for specifying if the panel can be covered by other windows or not. The default is the bottom layer, but with real transparency normal or top layer may be a nice alternative.</p></li>
|
||||||
|
<li><p><code>strut_policy = follow_size/minimum/none</code> : STRUTs are used by the window manager to decide the size of maximized windows. Note: on multi-monitor (Xinerama) setups, the panel must be placed at the edge (not in the middle) of the virtual screen for this to work correctly.</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>follow_size</code> means that the maximized windows always resize to have a common edge with tint2.</li>
|
||||||
|
<li><code>minimum</code> means that the maximized windows always expand to have a common edge with the hidden panel. This is useful if the <code>autohide</code> option is enabled.</li>
|
||||||
|
<li><code>none</code> means that the maximized windows use the full screen size.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>panel_window_name = string</code> : Defines the name of the panel's window. Default: 'tint2'. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>disable_transparency = boolean (0 or 1)</code> : Whether to disable transparency instead of detecting if it is supported. Useful on broken graphics stacks. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>mouse_effects = boolean (0 or 1)</code> : Whether to enable mouse hover effects for clickable items. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>mouse_hover_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency on mouse hover (works only when mouse_effects = 1).` <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>mouse_pressed_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency on mouse press (works only when mouse_effects = 1).` <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>autohide = boolean (0 or 1)</code> : Whether to enable panel hiding when the mouse cursor exists the panel.</p></li>
|
||||||
|
<li><p><code>autohide_show_timeout = float</code> : Show timeout in seconds after the mouse cursor enters the panel. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>autohide_hide_timeout = float</code> : Hide timeout in seconds after the mouse cursor exits the panel. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>autohide_height = integer</code> : panel height (width for vertical panels) in hidden mode.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="launcher">Launcher<a name="launcher" href="#launcher" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>launcher_item_app = path_to_application</code> : Each <code>launcher_item_app</code> must be a file path to a .desktop file following the freedesktop.org <a href="http://standards.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html">specification</a>. The paths may begin with <code>~</code>, which is expanded to the path of the user's home directory. If only a file name is specified, the file is search in the standard application directories (<code>$XDG_DATA_HOME/applications</code>, <code>~/.local/share/applications</code>, <code>$XDG_DATA_DIRS/applications</code>, <code>/usr/local/share/applications</code>, <code>/usr/share/applications</code>, <code>/opt/share/applications</code>).</p></li>
|
||||||
|
<li><p><code>launcher_apps_dir = path_to_directory</code> : Specifies a path to a directory from which the launcher is loading all .desktop files (all subdirectories are explored recursively). Can be used multiple times. The path may begin with <code>~</code>, which is expanded to the path of the user's home directory. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>launcher_background_id = integer</code> : Defines which background to use.</p></li>
|
||||||
|
<li><p><code>launcher_icon_background_id = integer</code> : Defines which background to use for icons.</p></li>
|
||||||
|
<li><p><code>launcher_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>launcher_icon_size = integer</code> : The launcher icon size, in pixels.</p></li>
|
||||||
|
<li><p><code>launcher_icon_theme = name_of_theme</code> : (Optional) Uses the specified icon theme to display shortcut icons. Note that tint2 will detect and use the icon theme of your desktop if you have an XSETTINGS manager running (which you probably do), unless <code>launcher_icon_theme_override = 1</code>.</p></li>
|
||||||
|
<li><p><code>launcher_icon_theme_override = boolean (0 or 1)</code> : Whether <code>launcher_icon_theme</code> overrides the value obtained from the XSETTINGS manager. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>launcher_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the icon color and transparency.</p></li>
|
||||||
|
<li><p><code>launcher_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for the launcher icons.</p></li>
|
||||||
|
<li><p><code>startup_notifications = boolean (0 or 1)</code> : Whether to show startup notifications when starting applications from the launcher. <em>(since 0.12)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="taskbar-pager">Taskbar / Pager<a name="taskbar-pager" href="#taskbar-pager" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>taskbar_mode = single_desktop/multi_desktop</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>single_desktop</code> : Shows a normal taskbar listing the tasks running on the current virtual desktop (also known as 'workspace');</li>
|
||||||
|
<li><code>multi_desktop</code> : Pager like capability. Shows multiple taskbars, one per virtual desktop, with which:
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li>You can drag-and-drop tasks between virtual desktops;</li>
|
||||||
|
<li>You can switch between virtual desktops.</li>
|
||||||
|
</ul></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>taskbar_distribute_size = boolean (0 or 1)</code> : If enabled, in multi-desktop mode distributes between taskbars the available size proportionally to the number of tasks. Default: disabled. <em>(since 0.12)</em></p></li>
|
||||||
|
<li><p><code>taskbar_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="taskbar_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>taskbar_background_id = integer</code> : Which background to use</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_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_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>
|
||||||
|
<ul>
|
||||||
|
<li><code>none</code> : No sorting. New tasks are simply appended at the end of the taskbar when they appear.</li>
|
||||||
|
<li><code>title</code> : Sorts the tasks by title.</li>
|
||||||
|
<li><code>center</code> : Sorts the tasks by their window centers.</li>
|
||||||
|
<li><code>mru</code> : Shows the most recently used tasks first. <em>(since 0.12.4)</em></li>
|
||||||
|
<li><code>lru</code> : Shows the most recently used tasks last. <em>(since 0.12.4)</em></li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>task_align = left/center/right</code> : Specifies the alignment of the tasks on the taskbar. Default: left.</p></li>
|
||||||
|
<li><p><code>taskbar_name = boolean (0 or 1)</code> : Whether to show the virtual desktop name in the taskbar.</p></li>
|
||||||
|
<li><p><code>taskbar_name_padding = padding</code> : Padding for the virtual desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_background_id = integer</code> : Which background to use for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code> : Font configuration for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_font_color = color opacity (0 to 100)</code> : Font color for the desktop name.</p></li>
|
||||||
|
<li><p><code>taskbar_name_active_background_id = integer</code> : Which background to use for the name of the current desktop.</p></li>
|
||||||
|
<li><p><code>taskbar_name_active_font_color = color opacity (0 to 100)</code> : Font color for the name of the current desktop.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h1 id="taskbar-buttons">Taskbar buttons<a name="taskbar-buttons" href="#taskbar-buttons" class="md2man-permalink" title="permalink"></a></h1><p>The following options configure the task buttons in the taskbar:</p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_icon = boolean (0 or 1)</code> : Whether to display the task icon.</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_tooltip = boolean (0 or 1)</code> : Whether to show tooltips for tasks.</p></li>
|
||||||
|
<li><p><code>task_maximum_size = width height</code></p>
|
||||||
|
<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>height</code> is used with vertical panels.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>task_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>urgent_nb_of_blink = integer</code> : Number of blinks on 'get attention' events.</p></li>
|
||||||
|
</ul>
|
||||||
|
<p><img src="task_padding.jpg" alt=""></p>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>task_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>task_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjust the task icon's color and transparency.</p></li>
|
||||||
|
<li><p><code>task_background_id = integer</code> : Which background to use for non selected tasks</p></li>
|
||||||
|
</ul>
|
||||||
|
<dl><dt>For the next 3 options STATUS can be <code>active</code> / <code>iconified</code> / <code>urgent</code>:</dt><dd>* <code>task_STATUS_font_color = color opacity (0 to 100)</code></dd></dl>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>task_STATUS_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)</code> : Adjusts the task icon's color and transparency.</p></li>
|
||||||
|
<li><p><code>task_STATUS_background_id = integer</code> : Which background to use for the task.</p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="mouse-actions-for-taskbar-buttons">Mouse actions for taskbar buttons<a name="mouse-actions-for-taskbar-buttons" href="#mouse-actions-for-taskbar-buttons" class="md2man-permalink" title="permalink"></a></h3><p>The possible mouse events are: <code>left, middle, right, scroll_up, scroll_down</code>.</p><p>The possible mouse actions are: <code>none, close, toggle, iconify, shade, toggle_iconify, maximize_restore, desktop_left, desktop_right, next_task, prev_task</code>.</p><p>Use <code>mouse_event = action</code> to customize mouse actions. Example:
|
||||||
|
<code>
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = toggle
|
||||||
|
mouse_scroll_down = iconify
|
||||||
|
</code></p><dl><dt>The action semantics:</dt><dd>* <code>none</code> : If <code>wm_menu = 1</code> is set, the mouse event is forwarded to the window manager. Otherwise it is ignored.
|
||||||
|
* <code>close</code> : close the task
|
||||||
|
* <code>toggle</code> : toggle the task
|
||||||
|
* <code>iconify</code> : iconify (minimize) the task
|
||||||
|
* <code>toggle_iconify</code> : toggle or iconify the task
|
||||||
|
* <code>maximize_restore</code> : maximized or minimized the task
|
||||||
|
* <code>shade</code> : shades (collapses) the task
|
||||||
|
* <code>desktop_left</code> : send the task to the desktop on the left
|
||||||
|
* <code>desktop_right</code> : send the task to the desktop on the right
|
||||||
|
* <code>next_task</code> : send the focus to next task
|
||||||
|
* <code>prev_task</code> : send the focus to previous task</dd></dl><h3 id="system-tray">System Tray<a name="system-tray" href="#system-tray" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>systray_padding = horizontal_padding vertical_padding spacing</code></p></li>
|
||||||
|
<li><p><code>systray_background_id = integer</code> : Which background to use.</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_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>
|
||||||
|
</ul>
|
||||||
|
<h3 id="clock">Clock<a name="clock" href="#clock" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>time1_format = %H:%M</code> : The format used by the first line of the clock.</p>
|
||||||
|
<ul>
|
||||||
|
<li><code>time1_format</code>, <code>time2_format</code> and <code>clock_tooltip</code> use the 'strftime' syntax. More info can be found here: <a href="http://www.manpagez.com/man/3/strftime/">http://www.manpagez.com/man/3/strftime/</a></li>
|
||||||
|
<li>To hide the clock, comment <code>time1_format</code> and <code>time2_format</code>.</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>time1_timezone = :US/Hawaii</code></p>
|
||||||
|
<ul>
|
||||||
|
<li><code>time1_timezone</code>, <code>time2_timezone</code> and <code>clock_tooltip_timezone</code> 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 <code>/usr/share/zoneinfo</code>. If your timezones are in a different directory, you need to specify the absolute path, e.g. <code>time1_timezone = :/different/zoneinfo/dir/US/Hawaii</code> Always prepend the timezone with a ':'</li>
|
||||||
|
</ul></li>
|
||||||
|
<li><p><code>time1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>time2_format = %A %d %B</code></p></li>
|
||||||
|
<li><p><code>time2_timezone = :Europe/Berlin</code></p></li>
|
||||||
|
<li><p><code>time2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
<li><p><code>clock_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>clock_padding = horizontal_padding vertical_padding</code></p></li>
|
||||||
|
<li><p><code>clock_background_id = integer</code> : Which background to use</p></li>
|
||||||
|
<li><p><code>clock_tooltip = %a, %d. %b %Y</code> : Format for the clock's tooltip.</p></li>
|
||||||
|
<li><p><code>clock_tooltip_timezone = :UTC</code></p></li>
|
||||||
|
<li><p><code>clock_lclick_command = text</code> : Command to execute on left click.</p></li>
|
||||||
|
<li><p><code>clock_rclick_command = text</code> : Command to execute on right click.</p></li>
|
||||||
|
<li><p><code>clock_mclick_command = text</code> : Command to execute on middle click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>clock_uwheel_command = text</code> : Command to execute on wheel scroll up. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>clock_dwheel_command = text</code> : Command to execute on wheel scroll down. <em>(since 0.12.1)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="tooltip">Tooltip<a name="tooltip" href="#tooltip" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>tooltip_padding = horizontal_padding vertical_padding</code></p></li>
|
||||||
|
<li><p><code>tooltip_show_timeout = float</code> : Delay to show the tooltip in seconds. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>tooltip_hide_timeout = float</code> : Delay to hide the tooltip in seconds. Use '.' as decimal separator.</p></li>
|
||||||
|
<li><p><code>tooltip_background_id = integer</code> : Which background to use for tooltips. Note that with fake transparency the alpha channel and corner radius options are not respected.</p></li>
|
||||||
|
<li><p><code>tooltip_font_color = color opacity (0 to 100)</code></p></li>
|
||||||
|
<li><p><code>tooltip_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]</code></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="battery">Battery<a name="battery" href="#battery" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<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_cmd = notify-send "battery low"</code> : Command to execute when the battery is low.</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>battery_font_color = color opacity (0 to 100)</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_tooltip_enabled = boolean (0 or 1)</code> : Enable/disable battery tooltips. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>battery_lclick_command = text</code> : Command to execute on left click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_rclick_command = text</code> : Command to execute on right click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_mclick_command = text</code> : Command to execute on middle click. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_uwheel_command = text</code> : Command to execute on wheel scroll up. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>battery_dwheel_command = text</code> : Command to execute on wheel scroll down. <em>(since 0.12.1)</em></p></li>
|
||||||
|
<li><p><code>ac_connected_cmd = text</code> : Command to execute when the power adapter is plugged in. <em>(since 0.12.3)</em></p></li>
|
||||||
|
<li><p><code>ac_disconnected_cmd = text</code> : Command to execute when the power adapter is unplugged. <em>(since 0.12.3)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h3 id="executor">Executor<a name="executor" href="#executor" class="md2man-permalink" title="permalink"></a></h3>
|
||||||
|
<ul>
|
||||||
|
<li><p><code>execp = new</code> : Begins the configuration of a new executor plugin. Multiple such plugins are supported; just use multiple <code>E</code>s in <code>panel_items</code>. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_command = text</code> : Command to execute. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_interval = integer</code> : The command is executed again after <code>execp_interval</code> seconds from the moment it exits. If zero, the command is executed only once. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_continuous = integer</code> : If non-zero, the last <code>execp_continuous</code> lines from the output of the command are displayed, every <code>execp_continuous</code> lines; this is useful for showing the output of commands that run indefinitely, such as <code>ping 127.0.0.1</code>. If zero, the output of the command is displayed after it finishes executing. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_has_icon = boolean (0 or 1)</code> : If <code>execp_has_icon = 1</code>, the first line printed by the command is interpreted as a path to an image file. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_cache_icon = boolean (0 or 1)</code> : If execp<em>cache</em>icon = 0, 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 execp<em>icon</em>w and execp<em>icon</em>h 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_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_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_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_background_id = integer</code> : Which background to use. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_centered = boolean (0 or 1)</code> : Whether to center the text. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_padding = horizontal_padding vertical_padding spacing_between_icon_and_text</code> <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_lclick_command = text</code> : Command to execute on left click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_mclick_command = text</code> : Command to execute on right click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_rclick_command = text</code> : Command to execute on middle click. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_uwheel_command = text</code> : Command to execute on wheel scroll up. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
<li><p><code>execp_dwheel_command = text</code> : Command to execute on wheel scroll down. If not defined, <code>execp_command</code> is executed immediately, unless it is currently running. <em>(since 0.12.4)</em></p></li>
|
||||||
|
</ul>
|
||||||
|
<h4 id="executor-samples">Executor samples<a name="executor-samples" href="#executor-samples" class="md2man-permalink" title="permalink"></a></h4><h5 id="print-the-hostname">Print the hostname<a name="print-the-hostname" href="#print-the-hostname" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = hostname
|
||||||
|
execp_interval = 0
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="print-disk-usage-for-the-root-partition-every-10-seconds">Print disk usage for the root partition every 10 seconds<a name="print-disk-usage-for-the-root-partition-every-10-seconds" href="#print-disk-usage-for-the-root-partition-every-10-seconds" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = df -h | awk '/\/$/ { print $6 ": " $2 " " $5}'
|
||||||
|
execp_interval = 10
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="button-with-icon-and-rich-text-executes-command-when-clicked">Button with icon and rich text, executes command when clicked<a name="button-with-icon-and-rich-text-executes-command-when-clicked" href="#button-with-icon-and-rich-text-executes-command-when-clicked" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = echo /usr/share/icons/elementary-xfce/emblems/24/emblem-colors-blue.png; echo '<span foreground="#7f7">Click</span> <span foreground="#77f">me</span> <span foreground="#f77">pls</span>'
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_interval = 0
|
||||||
|
execp_centered = 1
|
||||||
|
execp_font = sans 9
|
||||||
|
execp_markup = 1
|
||||||
|
execp_font_color = #aaffaa 100
|
||||||
|
execp_padding = 2 0
|
||||||
|
execp_tooltip = I will tell you a secret...
|
||||||
|
execp_lclick_command = zenity --info "--text=$(uname -sr)"
|
||||||
|
execp_background_id = 2
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="desktop-pager-with-text">Desktop pager with text<a name="desktop-pager-with-text" href="#desktop-pager-with-text" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = xprop -root -spy | awk '/^_NET_CURRENT_DESKTOP/ { print "Workspace " ($3 + 1) ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_continuous = 1
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="desktop-pager-with-icon">Desktop pager with icon<a name="desktop-pager-with-icon" href="#desktop-pager-with-icon" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp_command = xprop -root -spy | awk -v home="$HOME" '/^_NET_CURRENT_DESKTOP/ { print home "/.config/myPager/" ($3 + 1) ".png\n" ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_cache_icon = 1
|
||||||
|
execp_continuous = 2
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="round-trip-time-to-the-gateway-refreshed-every-second">Round-trip time to the gateway, refreshed every second<a name="round-trip-time-to-the-gateway-refreshed-every-second" href="#round-trip-time-to-the-gateway-refreshed-every-second" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = ping -i 1 -W 1 -O -D -n $(ip route | grep default | grep via | grep -o '[0-9]*\.[0-9]*\.[0-9]*\.[0-9]*') | awk '/no/ { print "<span foreground=\"#faa\">timeout</span>"; fflush(); }; /time=/ { gsub(/time=/, "", $8); printf "<span foreground=\"#7af\">%3.0f %s</span>\n", $8, $9; fflush(); } '
|
||||||
|
execp_continuous = 1
|
||||||
|
execp_interval = 1
|
||||||
|
execp_markup = 1
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="memory-usage">Memory usage<a name="memory-usage" href="#memory-usage" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code>execp = new
|
||||||
|
execp_command = free -s 2 | awk '/^-/ { printf "Mem: '$(free -h | awk '/^Mem:/ { print $2 }')' %.0f%\n", 100*$3/($3+$4); fflush(stdout) }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_continuous = 1
|
||||||
|
</code></pre>
|
||||||
|
<h5 id="network-load">Network load<a name="network-load" href="#network-load" class="md2man-permalink" title="permalink"></a></h5><pre class="highlight plaintext"><code># Note the use of "stdbuf -oL" to force the program to flush the output line by line.
|
||||||
|
execp = new
|
||||||
|
execp_command = stdbuf -oL bwm-ng -o csv -t 1000 | awk -F ';' '/total/ { printf "Net: %.0f Mb/s\n", ($5*8/1.0e6) }; fflush(stdout)'
|
||||||
|
execp_continuous = 1
|
||||||
|
execp_interval = 1
|
||||||
|
</code></pre>
|
||||||
|
<h3 id="example-configuration">Example configuration<a name="example-configuration" href="#example-configuration" class="md2man-permalink" title="permalink"></a></h3><pre class="highlight plaintext"><code>#---------------------------------------------
|
||||||
|
## 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
|
||||||
|
</code></pre>
|
||||||
|
<h2 id="author">AUTHOR<a name="author" href="#author" class="md2man-permalink" title="permalink"></a></h2><p>tint2 was written by Thierry Lorthiois <a href="mailto:lorthiois@bbsoft.fr">lorthiois@bbsoft.fr</a>.
|
||||||
|
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 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>
|
||||||
|
and the wiki page at <a href="https://gitlab.com/o9000/tint2/wikis/home">https://gitlab.com/o9000/tint2/wikis/home</a>.</p>
|
||||||
741
doc/tint2.md
Normal file
741
doc/tint2.md
Normal file
@@ -0,0 +1,741 @@
|
|||||||
|
# TINT2 1 "2017-04-23" 0.14.3
|
||||||
|
|
||||||
|
## NAME
|
||||||
|
tint2 - lightweight panel/taskbar
|
||||||
|
|
||||||
|
## DESCRIPTION
|
||||||
|
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.).
|
||||||
|
|
||||||
|
Features:
|
||||||
|
|
||||||
|
* Panel with taskbar, system tray, clock and launcher icons;
|
||||||
|
* Easy to customize: color/transparency on fonts, icons, borders and backgrounds;
|
||||||
|
* Pager like capability: move tasks between workspaces (virtual desktops), switch between workspaces;
|
||||||
|
* Multi-monitor capability: create one panel per monitor, showing only the tasks from the current monitor;
|
||||||
|
* Customizable mouse events.
|
||||||
|
|
||||||
|
Goals:
|
||||||
|
|
||||||
|
* Be unintrusive and light (in terms of memory, CPU and aesthetic);
|
||||||
|
* Follow the freedesktop.org specifications;
|
||||||
|
* Make certain workflows, such as multi-desktop and multi-monitor, easy to use.
|
||||||
|
|
||||||
|
## SYNOPSIS
|
||||||
|
`tint2 [OPTION...]`
|
||||||
|
|
||||||
|
## OPTIONS
|
||||||
|
`-c path_to_config_file`
|
||||||
|
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
|
||||||
|
|
||||||
|
### Table of contents
|
||||||
|
|
||||||
|
* [Introduction](#introduction)
|
||||||
|
|
||||||
|
* [Backgrounds and borders](#backgrounds-and-borders)
|
||||||
|
|
||||||
|
* [Gradients](#gradients)
|
||||||
|
|
||||||
|
* [Panel](#panel)
|
||||||
|
|
||||||
|
* [Launcher](#launcher)
|
||||||
|
|
||||||
|
* [Taskbar/Pager](#taskbar-pager)
|
||||||
|
|
||||||
|
* [Taskbar buttons](#taskbar-buttons)
|
||||||
|
|
||||||
|
* [Mouse actions for taskbar buttons](#mouse-actions-for-taskbar-buttons)
|
||||||
|
|
||||||
|
* [System tray](#system-tray)
|
||||||
|
|
||||||
|
* [Clock](#clock)
|
||||||
|
|
||||||
|
* [Tooltip](#tooltip)
|
||||||
|
|
||||||
|
* [Battery](#battery)
|
||||||
|
|
||||||
|
* [Executor](#executor)
|
||||||
|
|
||||||
|
* [Button](#button)
|
||||||
|
|
||||||
|
* [Separator](#separator)
|
||||||
|
|
||||||
|
* [Example configuration](#example-configuration)
|
||||||
|
|
||||||
|
### Introduction
|
||||||
|
|
||||||
|
These are instructions for configuring tint2 directly by editing its config file.
|
||||||
|
You may also use instead the graphical interface `tint2conf`.
|
||||||
|
|
||||||
|
The first time you run tint2, it will create the config file in `$HOME/.config/tint2/tint2rc` (This applies if you have done a clean install. Running tint2 in the source directory without doing 'make install' will not create the config file.)
|
||||||
|
|
||||||
|
You can also specify another file on the command line with the -c option, e.g.: `tint2 -c $HOME/tint2.conf`. This can be used to run multiple instances of tint2 that use different settings.
|
||||||
|
|
||||||
|
If you change the config file while tint2 is running, the command `killall -SIGUSR1 tint2` will force tint2 to reload it.
|
||||||
|
|
||||||
|
All the configuration options supported in the config file are listed below.
|
||||||
|
Try to respect as much as possible the order of the options as given below.
|
||||||
|
|
||||||
|
### Backgrounds and borders
|
||||||
|
|
||||||
|
The tint2 config file starts with the options defining background elements with borders:
|
||||||
|
|
||||||
|
* `rounded = number_of_pixels` : the corner radius
|
||||||
|
|
||||||
|
* `border_width = integer` : the border width in pixels
|
||||||
|
|
||||||
|
* `border_sides = LRTB` : the sides to draw the border on (left, right, top, bottom). If not specified, all sides are used. *(since 0.12.12)*
|
||||||
|
|
||||||
|
* `background_color = color opacity`
|
||||||
|
* `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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz).
|
||||||
|
|
||||||
|
* `border_color = color opacity`
|
||||||
|
* `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
|
||||||
|
|
||||||
|
* `background_color_hover = color opacity` (default: same as `background_color`) *(since 0.12.3)*
|
||||||
|
* `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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)
|
||||||
|
|
||||||
|
* `border_color_hover = color opacity` (default: same as `border_color`) *(since 0.12.3)*
|
||||||
|
* `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
|
||||||
|
|
||||||
|
* `background_color_pressed = color opacity` (default: same as `background_color_hover`) *(since 0.12.3)*
|
||||||
|
* `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. Note that for a transparent panel you need to enable a desktop compositor (such as compton or compiz)
|
||||||
|
|
||||||
|
* `border_color_pressed = color opacity` (default: same as `border_color_hover`) *(since 0.12.3)*
|
||||||
|
* `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
|
||||||
|
|
||||||
|
You can define as many backgrounds as you want. For example, the following config defines two backgrounds:
|
||||||
|
|
||||||
|
```
|
||||||
|
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
|
||||||
|
```
|
||||||
|
|
||||||
|
tint2 automatically identifies each background with a number starting from 1 (1, 2, ...).
|
||||||
|
Afterwards, you can apply a background to objects (panel, taskbar, task, clock, systray) using the background id, for example:
|
||||||
|
|
||||||
|
```
|
||||||
|
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
|
||||||
|
```
|
||||||
|
|
||||||
|
Identifier 0 refers to a special background which is fully transparent, identifier 1 applies the first background defined in the config file etc.
|
||||||
|
|
||||||
|
### Gradients
|
||||||
|
|
||||||
|
(Available since 0.13.0)
|
||||||
|
|
||||||
|
Backgrounds also allow specifying gradient layers
|
||||||
|
that are drawn on top of the solid color background.
|
||||||
|
|
||||||
|
First the user must define one or more gradients in the config file,
|
||||||
|
each starting with `gradient = TYPE`. These must be added before backgrounds.
|
||||||
|
|
||||||
|
Then gradients can be added by index to backgrounds,
|
||||||
|
using the `gradient_id = INDEX`, `gradient_id_hover = INDEX` and
|
||||||
|
`gradient_id_pressed = INDEX`, where `INDEX` is
|
||||||
|
the gradient index, starting from 1.
|
||||||
|
|
||||||
|
#### Gradient types
|
||||||
|
|
||||||
|
Gradients vary the color between fixed control points:
|
||||||
|
* vertical gradients: top-to-bottom;
|
||||||
|
* horizontal gradients: left-to-right;
|
||||||
|
* radial gradients: center-to-corners.
|
||||||
|
|
||||||
|
The user must specify the start and end colors, and can optionally add extra color stops in between
|
||||||
|
using the `color_stop` option, as explained below.
|
||||||
|
|
||||||
|
##### Vertical gradient, with color varying from the top edge to the bottom edge, two colors
|
||||||
|
|
||||||
|
```
|
||||||
|
gradient = vertical
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Horizontal gradient, with color varying from the left edge to the right edge, two colors
|
||||||
|
|
||||||
|
```
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Radial gradient, with color varying from the center to the corner, two colors:
|
||||||
|
|
||||||
|
```
|
||||||
|
gradient = radial
|
||||||
|
start_color = #rrggbb opacity
|
||||||
|
end_color = #rrggbb opacity
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Adding extra color stops (0% and 100% remain fixed, more colors at x% between the start and end control points)
|
||||||
|
|
||||||
|
```
|
||||||
|
color_stop = percentage #rrggbb opacity
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Gradient examples
|
||||||
|
|
||||||
|
```
|
||||||
|
# Gradient 1: thin film effect
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111122 30
|
||||||
|
end_color = #112211 30
|
||||||
|
color_stop = 60 #221111 30
|
||||||
|
|
||||||
|
# Gradient 2: radial glow
|
||||||
|
gradient = radial
|
||||||
|
start_color = #ffffff 20
|
||||||
|
end_color = #ffffff 0
|
||||||
|
|
||||||
|
# Gradient 3: elegant black
|
||||||
|
gradient = vertical
|
||||||
|
start_color = #444444 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Gradient 4: elegant black
|
||||||
|
gradient = horizontal
|
||||||
|
start_color = #111111 100
|
||||||
|
end_color = #222222 100
|
||||||
|
|
||||||
|
# Background 1: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
border_sides = TBLR
|
||||||
|
background_color = #555555 10
|
||||||
|
border_color = #ffffff 60
|
||||||
|
background_color_hover = #555555 10
|
||||||
|
border_color_hover = #ffffff 60
|
||||||
|
background_color_pressed = #555555 10
|
||||||
|
border_color_pressed = #ffffff 60
|
||||||
|
gradient_id = 3
|
||||||
|
gradient_id_hover = 4
|
||||||
|
gradient_id_pressed = 2
|
||||||
|
|
||||||
|
[...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Panel
|
||||||
|
|
||||||
|
* `panel_items = LTSBC` defines the items tint2 will show and the order of those items. Each letter refers to an item, defined as:
|
||||||
|
* `L` shows the Launcher
|
||||||
|
* `T` shows the Taskbar
|
||||||
|
* `S` shows the Systray (also called notification area)
|
||||||
|
* `B` shows the Battery status
|
||||||
|
* `C` shows the Clock
|
||||||
|
* `F` adds an extensible spacer (freespace). You can specify more than one. Has no effect if `T` is also present. *(since 0.12)*
|
||||||
|
* `E` adds an executor plugin. You can specify more than one. *(since 0.12.4)*
|
||||||
|
* `P` adds a push button. You can specify more than one. *(since 0.14)*
|
||||||
|
* `:` adds a separator. You can specify more than one. *(since 0.13.0)*
|
||||||
|
|
||||||
|
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
|
||||||
|
* The first monitor is `1`
|
||||||
|
* 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)*
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* `panel_position = vertical_position horizontal_position orientation`
|
||||||
|
* `vertical_position` is one of: `bottom`, `top`, `center`
|
||||||
|
* `horizontal_position` is one of: `left`, `right`, `center`
|
||||||
|
* `orientation` is one of: `horizontal`, `vertical`
|
||||||
|
|
||||||
|
* `panel_size = width height`
|
||||||
|
* `width` and `height` can be specified without units (e.g. `123`) as pixels, or followed by `%` as percentages of the monitor size (e.g. `50%`). Use `100%` for full monitor width/height.
|
||||||
|
Example:
|
||||||
|
|
||||||
|
```
|
||||||
|
# The panel's width is 94% the size of the monitor, the height is 30 pixels:
|
||||||
|
panel_size = 94% 30
|
||||||
|
```
|
||||||
|
|
||||||
|
* `panel_shrink = boolean (0 or 1)` : If set to 1, the panel will shrink to a compact size dynamically. *(since 0.13)*
|
||||||
|
|
||||||
|
* `panel_margin = horizontal_margin vertical_margin` : The margins define the distance between the panel and the horizontal/vertical monitor edge. Use `0` to obtain a panel with the same size as the edge of the monitor (no margin).
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* `panel_padding = horizontal_padding vertical_padding spacing` : Please refer to the image below.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* `font_shadow = boolean (0 or 1)`
|
||||||
|
|
||||||
|
* `panel_background_id = integer` : Which background to use for the panel.
|
||||||
|
|
||||||
|
* `wm_menu = boolean (0 or 1)` : Defines if tint2 forwards unhandled mouse events to your window manager. Useful for window managers such as openbox, which display the start menu if you right click on the desktop.
|
||||||
|
|
||||||
|
* `panel_dock = boolean (0 or 1)` : Defines if tint2 is placed into the window manager's dock. For the openbox window manager it is advised to also use a modifier for the moveButton option, otherwise the mouse click is not forwarded to tint2 (in ~/.config/openbox/rc.xml).
|
||||||
|
|
||||||
|
* `panel_layer = bottom/normal/top` : Places tint2 into the bottom/normal/top layer. This is helpful for specifying if the panel can be covered by other windows or not. The default is the bottom layer, but with real transparency normal or top layer may be a nice alternative.
|
||||||
|
|
||||||
|
* `strut_policy = follow_size/minimum/none` : STRUTs are used by the window manager to decide the size of maximized windows. Note: on multi-monitor (Xinerama) setups, the panel must be placed at the edge (not in the middle) of the virtual screen for this to work correctly.
|
||||||
|
* `follow_size` means that the maximized windows always resize to have a common edge with tint2.
|
||||||
|
* `minimum` means that the maximized windows always expand to have a common edge with the hidden panel. This is useful if the `autohide` option is enabled.
|
||||||
|
* `none` means that the maximized windows use the full screen size.
|
||||||
|
|
||||||
|
* `panel_window_name = string` : Defines the name of the panel's window. Default: 'tint2'. *(since 0.12)*
|
||||||
|
|
||||||
|
* `disable_transparency = boolean (0 or 1)` : Whether to disable transparency instead of detecting if it is supported. Useful on broken graphics stacks. *(since 0.12)*
|
||||||
|
|
||||||
|
* `mouse_effects = boolean (0 or 1)` : Whether to enable mouse hover effects for clickable items. *(since 0.12.3)*
|
||||||
|
|
||||||
|
* `mouse_hover_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjusts the icon color and transparency on mouse hover (works only when mouse_effects = 1).` *(since 0.12.3)*
|
||||||
|
|
||||||
|
* `mouse_pressed_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjusts the icon color and transparency on mouse press (works only when mouse_effects = 1).` *(since 0.12.3)*
|
||||||
|
|
||||||
|
* `autohide = boolean (0 or 1)` : Whether to enable panel hiding when the mouse cursor exists the panel.
|
||||||
|
|
||||||
|
* `autohide_show_timeout = float` : Show timeout in seconds after the mouse cursor enters the panel. Use '.' as decimal separator.
|
||||||
|
|
||||||
|
* `autohide_hide_timeout = float` : Hide timeout in seconds after the mouse cursor exits the panel. Use '.' as decimal separator.
|
||||||
|
|
||||||
|
* `autohide_height = integer` : panel height (width for vertical panels) in hidden mode.
|
||||||
|
|
||||||
|
### Launcher
|
||||||
|
* `launcher_item_app = path_to_application` : Each `launcher_item_app` must be a file path to a .desktop file following the freedesktop.org [specification](http://standards.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html). The paths may begin with `~`, which is expanded to the path of the user's home directory. If only a file name is specified, the file is search in the standard application directories (`$XDG_DATA_HOME/applications`, `~/.local/share/applications`, `$XDG_DATA_DIRS/applications`, `/usr/local/share/applications`, `/usr/share/applications`, `/opt/share/applications`).
|
||||||
|
|
||||||
|
* `launcher_apps_dir = path_to_directory` : Specifies a path to a directory from which the launcher is loading all .desktop files (all subdirectories are explored recursively). Can be used multiple times. The path may begin with `~`, which is expanded to the path of the user's home directory. *(since 0.12)*
|
||||||
|
|
||||||
|
* `launcher_background_id = integer` : Defines which background to use.
|
||||||
|
|
||||||
|
* `launcher_icon_background_id = integer` : Defines which background to use for icons.
|
||||||
|
|
||||||
|
* `launcher_padding = horizontal_padding vertical_padding spacing`
|
||||||
|
|
||||||
|
* `launcher_icon_size = integer` : The launcher icon size, in pixels.
|
||||||
|
|
||||||
|
* `launcher_icon_theme = name_of_theme` : (Optional) Uses the specified icon theme to display shortcut icons. Note that tint2 will detect and use the icon theme of your desktop if you have an XSETTINGS manager running (which you probably do), unless `launcher_icon_theme_override = 1`.
|
||||||
|
|
||||||
|
* `launcher_icon_theme_override = boolean (0 or 1)` : Whether `launcher_icon_theme` overrides the value obtained from the XSETTINGS manager. *(since 0.12)*
|
||||||
|
|
||||||
|
* `launcher_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjusts the icon color and transparency.
|
||||||
|
|
||||||
|
* `launcher_tooltip = boolean (0 or 1)` : Whether to show tooltips for the launcher icons.
|
||||||
|
|
||||||
|
* `startup_notifications = boolean (0 or 1)` : Whether to show startup notifications when starting applications from the launcher. *(since 0.12)*
|
||||||
|
|
||||||
|
### Taskbar / Pager
|
||||||
|
|
||||||
|
* `taskbar_mode = single_desktop/multi_desktop`
|
||||||
|
* `single_desktop` : Shows a normal taskbar listing the tasks running on the current virtual desktop (also known as 'workspace');
|
||||||
|
* `multi_desktop` : Pager like capability. Shows multiple taskbars, one per virtual desktop, with which:
|
||||||
|
* You can drag-and-drop tasks between virtual desktops;
|
||||||
|
* You can switch between virtual desktops.
|
||||||
|
|
||||||
|
* `taskbar_hide_if_empty = boolean (0 or 1)` : If enabled, in multi-desktop mode the taskbars corresponding to empty desktops different from the current desktop are hidden. *(since 0.13)*
|
||||||
|
|
||||||
|
* `taskbar_distribute_size = boolean (0 or 1)` : If enabled, in multi-desktop mode distributes between taskbars the available size proportionally to the number of tasks. Default: disabled. *(since 0.12)*
|
||||||
|
|
||||||
|
* `taskbar_padding = horizontal_padding vertical_padding spacing`
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* `taskbar_background_id = integer` : Which background to use
|
||||||
|
|
||||||
|
* `taskbar_active_background_id = integer` : Which background to use for the taskbar of the current virtual desktop.
|
||||||
|
|
||||||
|
* `taskbar_hide_inactive_tasks = boolean (0 or 1)` : If enabled, the taskbar shows only the active task. *(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_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)*
|
||||||
|
* `none` : No sorting. New tasks are simply appended at the end of the taskbar when they appear.
|
||||||
|
* `title` : Sorts the tasks by title.
|
||||||
|
* `center` : Sorts the tasks by their window centers.
|
||||||
|
* `mru` : Shows the most recently used tasks first. *(since 0.12.4)*
|
||||||
|
* `lru` : Shows the most recently used tasks last. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `task_align = left/center/right` : Specifies the alignment of the tasks on the taskbar. Default: left.
|
||||||
|
|
||||||
|
* `taskbar_name = boolean (0 or 1)` : Whether to show the virtual desktop name in the taskbar.
|
||||||
|
|
||||||
|
* `taskbar_name_padding = padding` : Padding for the virtual desktop name.
|
||||||
|
|
||||||
|
* `taskbar_name_background_id = integer` : Which background to use for the desktop name.
|
||||||
|
|
||||||
|
* `taskbar_name_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` : Font configuration for the desktop name.
|
||||||
|
|
||||||
|
* `taskbar_name_font_color = color opacity (0 to 100)` : Font color for the desktop name.
|
||||||
|
|
||||||
|
* `taskbar_name_active_background_id = integer` : Which background to use for the name of the current desktop.
|
||||||
|
|
||||||
|
* `taskbar_name_active_font_color = color opacity (0 to 100)` : Font color for the name of the current desktop.
|
||||||
|
|
||||||
|
# Taskbar buttons
|
||||||
|
|
||||||
|
The following options configure the task buttons in the taskbar:
|
||||||
|
|
||||||
|
* `task_icon = boolean (0 or 1)` : Whether to display the task icon. There is no explicit option to control the task icon size; it depends on the vertical padding set with `task_padding`.
|
||||||
|
|
||||||
|
* `task_text = boolean (0 or 1)` : Whether to display the task text.
|
||||||
|
|
||||||
|
* `task_centered = boolean (0 or 1)` : Whether the task text is centered.
|
||||||
|
|
||||||
|
* `task_tooltip = boolean (0 or 1)` : Whether to show tooltips for tasks.
|
||||||
|
|
||||||
|
* `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.
|
||||||
|
* `height` is used with vertical panels.
|
||||||
|
|
||||||
|
* `task_padding = horizontal_padding vertical_padding spacing`
|
||||||
|
|
||||||
|
* `urgent_nb_of_blink = integer` : Number of blinks on 'get attention' events.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
* `task_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
* `task_font_color = color opacity (0 to 100)`
|
||||||
|
|
||||||
|
* `task_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjust the task icon's color and transparency.
|
||||||
|
|
||||||
|
* `task_background_id = integer` : Which background to use for non selected tasks
|
||||||
|
|
||||||
|
For the next 3 options STATUS can be `active` / `iconified` / `urgent`:
|
||||||
|
* `task_STATUS_font_color = color opacity (0 to 100)`
|
||||||
|
|
||||||
|
* `task_STATUS_icon_asb = alpha (0 to 100) saturation (-100 to 100) brightness (-100 to 100)` : Adjusts the task icon's color and transparency.
|
||||||
|
|
||||||
|
* `task_STATUS_background_id = integer` : Which background to use for the task.
|
||||||
|
|
||||||
|
### Mouse actions for taskbar buttons
|
||||||
|
|
||||||
|
The possible mouse events are: `left, middle, right, scroll_up, scroll_down`.
|
||||||
|
|
||||||
|
The possible mouse actions are: `none, close, toggle, iconify, shade, toggle_iconify, maximize_restore, desktop_left, desktop_right, next_task, prev_task`.
|
||||||
|
|
||||||
|
Use `mouse_event = action` to customize mouse actions. Example:
|
||||||
|
```
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = toggle
|
||||||
|
mouse_scroll_down = iconify
|
||||||
|
```
|
||||||
|
|
||||||
|
The action semantics:
|
||||||
|
* `none` : If `wm_menu = 1` is set, the mouse event is forwarded to the window manager. Otherwise it is ignored.
|
||||||
|
* `close` : close the task
|
||||||
|
* `toggle` : toggle the task
|
||||||
|
* `iconify` : iconify (minimize) the task
|
||||||
|
* `toggle_iconify` : toggle or iconify the task
|
||||||
|
* `maximize_restore` : maximized or minimized the task
|
||||||
|
* `shade` : shades (collapses) 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
|
||||||
|
|
||||||
|
### System Tray
|
||||||
|
|
||||||
|
* `systray_padding = horizontal_padding vertical_padding spacing`
|
||||||
|
|
||||||
|
* `systray_background_id = integer` : Which background to use.
|
||||||
|
|
||||||
|
* `systray_sort = ascending/descending/left2right/right2left` : 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 `left2right` or `right2left` the order can be different on panel restart).
|
||||||
|
|
||||||
|
* `systray_icon_size = max_icon_size` : Set the maximum system tray icon size to `number`. Set to `0` for automatic icon sizing.
|
||||||
|
|
||||||
|
* `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_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)*
|
||||||
|
|
||||||
|
### Clock
|
||||||
|
|
||||||
|
* `time1_format = %H:%M` : The format used by the first line of the clock.
|
||||||
|
* `time1_format`, `time2_format` and `clock_tooltip` use the 'strftime' syntax. More info can be found here: http://www.manpagez.com/man/3/strftime/
|
||||||
|
* To hide the clock, comment `time1_format` and `time2_format`.
|
||||||
|
|
||||||
|
* `time1_timezone = :US/Hawaii`
|
||||||
|
* `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, e.g. `time1_timezone = :/different/zoneinfo/dir/US/Hawaii` Always prepend the timezone with a ':'
|
||||||
|
|
||||||
|
* `time1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
* `time2_format = %A %d %B`
|
||||||
|
|
||||||
|
* `time2_timezone = :Europe/Berlin`
|
||||||
|
|
||||||
|
* `time2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
* `clock_font_color = color opacity (0 to 100)`
|
||||||
|
|
||||||
|
* `clock_padding = horizontal_padding vertical_padding`
|
||||||
|
|
||||||
|
* `clock_background_id = integer` : Which background to use
|
||||||
|
|
||||||
|
* `clock_tooltip = %a, %d. %b %Y` : Format for the clock's tooltip.
|
||||||
|
|
||||||
|
* `clock_tooltip_timezone = :UTC`
|
||||||
|
|
||||||
|
* `clock_lclick_command = text` : Command to execute on left click.
|
||||||
|
|
||||||
|
* `clock_rclick_command = text` : Command to execute on right click.
|
||||||
|
|
||||||
|
* `clock_mclick_command = text` : Command to execute on middle click. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `clock_uwheel_command = text` : Command to execute on wheel scroll up. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `clock_dwheel_command = text` : Command to execute on wheel scroll down. *(since 0.12.1)*
|
||||||
|
|
||||||
|
### Tooltip
|
||||||
|
|
||||||
|
* `tooltip_padding = horizontal_padding vertical_padding`
|
||||||
|
|
||||||
|
* `tooltip_show_timeout = float` : Delay to show the tooltip in seconds. Use `.` as decimal separator.
|
||||||
|
|
||||||
|
* `tooltip_hide_timeout = float` : Delay to hide the tooltip in seconds. Use `.` as decimal separator.
|
||||||
|
|
||||||
|
* `tooltip_background_id = integer` : Which background to use for tooltips. Note that with fake transparency the alpha channel and corner radius options are not respected.
|
||||||
|
|
||||||
|
* `tooltip_font_color = color opacity (0 to 100)`
|
||||||
|
|
||||||
|
* `tooltip_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
### Battery
|
||||||
|
|
||||||
|
* `battery_hide = never/integer (0 to 100)` : At what battery percentage the battery item is hidden.
|
||||||
|
|
||||||
|
* `battery_low_status = integer`: At what battery percentage the low command is executed.
|
||||||
|
|
||||||
|
* `battery_low_cmd = notify-send "battery low"` : Command to execute when the battery is low.
|
||||||
|
|
||||||
|
* `bat1_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
* `bat2_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]`
|
||||||
|
|
||||||
|
* `battery_font_color = color opacity (0 to 100)`
|
||||||
|
|
||||||
|
* `battery_padding = horizontal_padding vertical_padding`
|
||||||
|
|
||||||
|
* `battery_background_id = integer` : Which background to use for the battery.
|
||||||
|
|
||||||
|
* `battery_tooltip_enabled = boolean (0 or 1)` : Enable/disable battery tooltips. *(since 0.12.3)*
|
||||||
|
|
||||||
|
* `battery_lclick_command = text` : Command to execute on left click. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `battery_rclick_command = text` : Command to execute on right click. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `battery_mclick_command = text` : Command to execute on middle click. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `battery_uwheel_command = text` : Command to execute on wheel scroll up. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `battery_dwheel_command = text` : Command to execute on wheel scroll down. *(since 0.12.1)*
|
||||||
|
|
||||||
|
* `ac_connected_cmd = text` : Command to execute when the power adapter is plugged in. *(since 0.12.3)*
|
||||||
|
|
||||||
|
* `ac_disconnected_cmd = text` : Command to execute when the power adapter is unplugged. *(since 0.12.3)*
|
||||||
|
|
||||||
|
### Executor
|
||||||
|
|
||||||
|
* `execp = new` : Begins the configuration of a new executor plugin. Multiple such plugins are supported; just use multiple `E`s in `panel_items`. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_command = text` : Command to execute. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_interval = integer` : The command is executed again after `execp_interval` seconds from the moment it exits. If zero, the command is executed only once. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_continuous = integer` : If non-zero, the last `execp_continuous` lines from the output of the command are displayed, every `execp_continuous` lines; this is useful for showing the output of commands that run indefinitely, such as `ping 127.0.0.1`. If zero, the output of the command is displayed after it finishes executing. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_has_icon = boolean (0 or 1)` : If `execp_has_icon = 1`, the first line printed by the command is interpreted as a path to an image file. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_cache_icon = boolean (0 or 1)` : If `execp_cache_icon = 0`, the image is reloaded each time the command is executed (useful if the image file is changed on disk by the program executed by `execp_command`). *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_icon_w = integer` : You can use `execp_icon_w` and `execp_icon_h` 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. *(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_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` : The font used to draw the text. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_font_color = color opacity` : The font color. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_markup = boolean (0 or 1)` : If non-zero, the output of the command is treated as Pango markup, which allows rich text formatting. The format is [documented here](https://developer.gnome.org/pygtk/stable/pango-markup-language.html). Note that using this with commands that print data downloaded from the Internet is a possible security risk. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_background_id = integer` : Which background to use. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_centered = boolean (0 or 1)` : Whether to center the text. *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_padding = horizontal_padding vertical_padding spacing_between_icon_and_text` *(since 0.12.4)*
|
||||||
|
|
||||||
|
* `execp_lclick_command = text` : Command to execute on left click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.12.4)*
|
||||||
|
* `execp_mclick_command = text` : Command to execute on right click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.12.4)*
|
||||||
|
* `execp_rclick_command = text` : Command to execute on middle click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.12.4)*
|
||||||
|
* `execp_uwheel_command = text` : Command to execute on wheel scroll up. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.12.4)*
|
||||||
|
* `execp_dwheel_command = text` : Command to execute on wheel scroll down. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.12.4)*
|
||||||
|
|
||||||
|
#### Executor samples
|
||||||
|
|
||||||
|
##### Print the hostname
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = hostname
|
||||||
|
execp_interval = 0
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Print disk usage for the root partition every 10 seconds
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = df -h | awk '/\/$/ { print $6 ": " $2 " " $5}'
|
||||||
|
execp_interval = 10
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Button with icon and rich text, executes command when clicked
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = echo /usr/share/icons/elementary-xfce/emblems/24/emblem-colors-blue.png; echo '<span foreground="#7f7">Click</span> <span foreground="#77f">me</span> <span foreground="#f77">pls</span>'
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_interval = 0
|
||||||
|
execp_centered = 1
|
||||||
|
execp_font = sans 9
|
||||||
|
execp_markup = 1
|
||||||
|
execp_font_color = #aaffaa 100
|
||||||
|
execp_padding = 2 0
|
||||||
|
execp_tooltip = I will tell you a secret...
|
||||||
|
execp_lclick_command = zenity --info "--text=$(uname -sr)"
|
||||||
|
execp_background_id = 2
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Desktop pager with text
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = xprop -root -spy | awk '/^_NET_CURRENT_DESKTOP/ { print "Workspace " ($3 + 1) ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_continuous = 1
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Desktop pager with icon
|
||||||
|
|
||||||
|
```
|
||||||
|
execp_command = xprop -root -spy | awk -v home="$HOME" '/^_NET_CURRENT_DESKTOP/ { print home "/.config/myPager/" ($3 + 1) ".png\n" ; fflush(); }'
|
||||||
|
execp_interval = 1
|
||||||
|
execp_has_icon = 1
|
||||||
|
execp_cache_icon = 1
|
||||||
|
execp_continuous = 2
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Round-trip time to the gateway, refreshed every second
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = ping -i 1 -c 1 -W 1 -O -D -n $(ip route | grep default | grep via | grep -o '[0-9]*\.[0-9]*\.[0-9]*\.[0-9]*') | awk '/no/ { print "<span foreground=\"#faa\">timeout</span>"; fflush(); }; /time=/ { gsub(/time=/, "", $8); printf "<span foreground=\"#7af\">%3.0f %s</span>\n", $8, $9; fflush(); } '
|
||||||
|
execp_continuous = 0
|
||||||
|
execp_interval = 1
|
||||||
|
execp_markup = 1
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Memory usage
|
||||||
|
|
||||||
|
```
|
||||||
|
execp = new
|
||||||
|
execp_command = free | awk '/^-/ { printf "Mem: '$(free -h | awk '/^Mem:/ { print $2 }')' %.0f%%\n", 100*$3/($3+$4); fflush(stdout) }'
|
||||||
|
execp_interval = 5
|
||||||
|
execp_continuous = 0
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Network load
|
||||||
|
|
||||||
|
```
|
||||||
|
# Note the use of "stdbuf -oL" to force the program to flush the output line by line.
|
||||||
|
execp = new
|
||||||
|
execp_command = stdbuf -oL bwm-ng -o csv -t 1000 | awk -F ';' '/total/ { printf "Net: %.0f Mb/s\n", ($5*8/1.0e6) }; fflush(stdout)'
|
||||||
|
execp_continuous = 1
|
||||||
|
execp_interval = 1
|
||||||
|
```
|
||||||
|
|
||||||
|
### Button
|
||||||
|
|
||||||
|
* `button = new` : Begins the configuration of a new button. Multiple such plugins are supported; just use multiple `P`s in `panel_items`. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_icon = text` : Name or path of icon (or empty). *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_text = text` : Text to display (or empty). *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_tooltip = text` : The tooltip (or empty). *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_font = [FAMILY-LIST] [STYLE-OPTIONS] [SIZE]` : The font used to draw the text. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_font_color = color opacity` : The font color. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_background_id = integer` : Which background to use. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_centered = boolean (0 or 1)` : Whether to center the text. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_padding = horizontal_padding vertical_padding spacing_between_icon_and_text` *(since 0.14)*
|
||||||
|
* `button_max_icon_size = integer` : Sets a limit to the icon size. Otherwise, the icon will expand to the edges. *(since 0.14)*
|
||||||
|
|
||||||
|
* `button_lclick_command = text` : Command to execute on left click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.14)*
|
||||||
|
* `button_mclick_command = text` : Command to execute on right click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.14)*
|
||||||
|
* `button_rclick_command = text` : Command to execute on middle click. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.14)*
|
||||||
|
* `button_uwheel_command = text` : Command to execute on wheel scroll up. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.14)*
|
||||||
|
* `button_dwheel_command = text` : Command to execute on wheel scroll down. If not defined, `execp_command` is executed immediately, unless it is currently running. *(since 0.14)*
|
||||||
|
|
||||||
|
### Separator
|
||||||
|
|
||||||
|
* `separator = new` : Begins the configuration of a new separator. Multiple such plugins are supported; just use multiple `:`s in `panel_items`. *(since 0.13.0)*
|
||||||
|
|
||||||
|
* `separator_background_id = integer` : Which background to use. *(since 0.13.0)*
|
||||||
|
|
||||||
|
* `separator_color = color opacity` : The foreground color. *(since 0.13.0)*
|
||||||
|
|
||||||
|
* `separator_style = [empty | line | dots]` : The separator style. *(since 0.13.0)*
|
||||||
|
|
||||||
|
* `separator_size = integer` : The thickness of the separator. Does not include the border and padding. For example, if the style is `line`, this is the line thickness; if the style is `dots`, this is the dot's diameter. *(since 0.13.0)*
|
||||||
|
|
||||||
|
* `separator_padding = side_padding cap_padding` : The padding to add to the sides of the separator, in pixels. *(since 0.13.0)*
|
||||||
|
|
||||||
|
### Example configuration
|
||||||
|
|
||||||
|
See /etc/xdg/tint2/tint2rc.
|
||||||
|
|
||||||
|
## AUTHOR
|
||||||
|
tint2 was written by Thierry Lorthiois <lorthiois@bbsoft.fr>.
|
||||||
|
It is based on ttm, originally written by Pål Staurland <staura@gmail.com>.
|
||||||
|
|
||||||
|
This manual page was originally written by Daniel Moerner <dmoerner@gmail.com>, for the Debian project (but may be used by others).
|
||||||
|
It was adopted from the tint2 docs.
|
||||||
|
|
||||||
|
## SEE ALSO
|
||||||
|
The main website https://gitlab.com/o9000/tint2
|
||||||
|
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
|
||||||
|
for documentation files, usually `/usr/share/doc/tint2` or `/usr/local/share/doc/tint2`.
|
||||||
3
format-code.sh
Executable file
3
format-code.sh
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
find . '(' -name '*.h' -o -name '*.c' ')' -exec clang-format-3.7 -style=file -i '{}' \;
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
FALLBACK=\"0.11-svn\"
|
|
||||||
|
|
||||||
if [ $# -eq 0 ]; then
|
|
||||||
DIR=.
|
|
||||||
else
|
|
||||||
DIR=$1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f version.h ]; then
|
|
||||||
REV_OLD=$(cat version.h | cut -d" " -f3)
|
|
||||||
else
|
|
||||||
REV_OLD=\"\"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -x "$(which svnversion 2>/dev/null)" -a -d "${DIR}/.svn" ] ; then
|
|
||||||
REV=\"$(svnversion -n ${DIR})\"
|
|
||||||
else
|
|
||||||
REV=${FALLBACK}
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ${REV_OLD} != ${REV} ]; then
|
|
||||||
echo "Building new version.h"
|
|
||||||
echo "Rev_old: ${REV_OLD} Rev: ${REV}"
|
|
||||||
echo "#define VERSION_STRING ${REV}" > version.h
|
|
||||||
fi
|
|
||||||
47
get_version.sh
Executable file
47
get_version.sh
Executable file
@@ -0,0 +1,47 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
MAJOR=0.14
|
||||||
|
DIRTY=""
|
||||||
|
|
||||||
|
if 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
|
||||||
|
VERSION=$(git describe --exact-match 2>/dev/null || echo "$MAJOR-git$(git show -s --pretty=format:%ci | cut -d ' ' -f 1 | tr -d '-').$(git show -s --pretty=format:%h)")$DIRTY
|
||||||
|
else
|
||||||
|
VERSION=$(head -n 1 ChangeLog || head -n 1 ../ChangeLog | cut -d ' ' -f 2)
|
||||||
|
if [ $VERSION = "master" ]
|
||||||
|
then
|
||||||
|
VERSION=$VERSION-$(head -n 1 ChangeLog || head -n 1 ../ChangeLog | cut -d ' ' -f 1)
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
VERSION=$(echo "$VERSION" | sed 's/^v//')
|
||||||
|
|
||||||
|
echo '#define VERSION_STRING "'$VERSION'"' > version.h
|
||||||
|
echo $VERSION
|
||||||
@@ -1,23 +1,19 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
# usage: ./make_release.sh RELEASE_VERSION_NUMBER
|
# Usage: ./make_release.sh
|
||||||
|
# Creates a tar.gz archive of the current tree.
|
||||||
|
|
||||||
if [[ $# -ne 1 ]]; then
|
VERSION=$(./get_version.sh --strict)
|
||||||
echo "usage: $0 RELEASE_VERSION_NUMBER"
|
if [ ! $? -eq 0 ]
|
||||||
exit
|
then
|
||||||
|
echo >&2 "Error: get_version.sh failed!"
|
||||||
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
DIR=tint2-${1}
|
ARCHIVE=tint2-$VERSION.tar.gz
|
||||||
echo "Making release ${DIR}"
|
|
||||||
rm -Rf ${DIR}
|
|
||||||
svn export . ${DIR} > /dev/null
|
|
||||||
|
|
||||||
# delete unneeded files
|
echo "Making release tint2-$VERSION"
|
||||||
rm -f ${DIR}/configure ${DIR}/make_release.sh
|
git archive --format=tar.gz --prefix=tint2-$VERSION/ v$VERSION >$ARCHIVE
|
||||||
|
|
||||||
# replace get_svnrev.sh by a simple echo command
|
sha1sum -b $ARCHIVE
|
||||||
echo "echo \"#define VERSION_STRING \\\"${1}\\\"\" > version.h" > ${DIR}/get_svnrev.sh
|
sha256sum -b $ARCHIVE
|
||||||
|
|
||||||
# create tarball and remove the exported directory
|
|
||||||
tar -cjf ${DIR}.tar.bz2 ${DIR}
|
|
||||||
rm -Rf ${DIR}
|
|
||||||
|
|||||||
240
new-release.py
Executable file
240
new-release.py
Executable file
@@ -0,0 +1,240 @@
|
|||||||
|
#!/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, major=False, minor=False, rc=False):
|
||||||
|
if "-rc" in v:
|
||||||
|
# v4.0-rc7 -> v4.0-rc8 or v4.0
|
||||||
|
if minor:
|
||||||
|
return v.split("-rc")[0]
|
||||||
|
else:
|
||||||
|
parts = v.split("-rc")
|
||||||
|
parts[-1] = str(int(parts[-1]) + 1)
|
||||||
|
return "-rc".join(parts)
|
||||||
|
else:
|
||||||
|
# v4.11 = v4, 11, 0 -> v4.11.1 or v4.12 or v.4.12-rc1 or v5.0 or v5.0-rc1
|
||||||
|
# v4.11.7 = v4, 11, 7 -> ...
|
||||||
|
parts = v.split(".")
|
||||||
|
while len(parts) < 3:
|
||||||
|
parts.append("0")
|
||||||
|
assert len(parts) == 3
|
||||||
|
if major:
|
||||||
|
parts[-3] = "v" + str(int(parts[-3].replace("v", "")) + 1)
|
||||||
|
parts[-2] = "0"
|
||||||
|
if rc:
|
||||||
|
parts[-2] += "-rc1"
|
||||||
|
parts[-1] = ""
|
||||||
|
elif minor or rc:
|
||||||
|
parts[-2] = str(int(parts[-2]) + 1)
|
||||||
|
if rc:
|
||||||
|
parts[-2] += "-rc1"
|
||||||
|
parts[-1] = ""
|
||||||
|
else:
|
||||||
|
parts[-1] = str(int(parts[-1]) + 1)
|
||||||
|
assert not rc
|
||||||
|
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("v1.0"), "v1.0.1")
|
||||||
|
assert_equal(inc_version("v1.0.1"), "v1.0.2")
|
||||||
|
assert_equal(inc_version("v1.0.2"), "v1.0.3")
|
||||||
|
assert_equal(inc_version("v1.0.10"), "v1.0.11")
|
||||||
|
assert_equal(inc_version("v1.1.10"), "v1.1.11")
|
||||||
|
assert_equal(inc_version("v1.1.10"), "v1.1.11")
|
||||||
|
# rc
|
||||||
|
assert_equal(inc_version("v1.0", False, False, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.1", False, False, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.2", False, False, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.10", False, False, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, False, True), "v1.2-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, False, True), "v1.2-rc1")
|
||||||
|
# minor
|
||||||
|
assert_equal(inc_version("v1.0", False, True, False), "v1.1")
|
||||||
|
assert_equal(inc_version("v1.0.1", False, True, False), "v1.1")
|
||||||
|
assert_equal(inc_version("v1.0.2", False, True, False), "v1.1")
|
||||||
|
assert_equal(inc_version("v1.0.10", False, True, False), "v1.1")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, True, False), "v1.2")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, True, False), "v1.2")
|
||||||
|
# minor rc
|
||||||
|
assert_equal(inc_version("v1.0", False, True, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.1", False, True, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.2", False, True, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.10", False, True, True), "v1.1-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, True, True), "v1.2-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", False, True, True), "v1.2-rc1")
|
||||||
|
# major rc
|
||||||
|
assert_equal(inc_version("v1.0", True, False, True), "v2.0-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.1", True, False, True), "v2.0-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.2", True, False, True), "v2.0-rc1")
|
||||||
|
assert_equal(inc_version("v1.0.10", True, False, True), "v2.0-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", True, False, True), "v2.0-rc1")
|
||||||
|
assert_equal(inc_version("v1.1.10", True, False, True), "v2.0-rc1")
|
||||||
|
# major
|
||||||
|
assert_equal(inc_version("v1.0", True), "v2.0")
|
||||||
|
assert_equal(inc_version("v1.0.1", True), "v2.0")
|
||||||
|
assert_equal(inc_version("v1.0.2", True), "v2.0")
|
||||||
|
assert_equal(inc_version("v1.0.10", True), "v2.0")
|
||||||
|
assert_equal(inc_version("v1.1.10", True), "v2.0")
|
||||||
|
assert_equal(inc_version("v1.1.10", True), "v2.0")
|
||||||
|
# rc auto
|
||||||
|
assert_equal(inc_version("v1.0-rc1"), "v1.0-rc2")
|
||||||
|
assert_equal(inc_version("v1.1-rc2"), "v1.1-rc3")
|
||||||
|
# rc minor
|
||||||
|
assert_equal(inc_version("v1.0-rc1", False, True), "v1.0")
|
||||||
|
assert_equal(inc_version("v1.1-rc2", False, True), "v1.1")
|
||||||
|
|
||||||
|
|
||||||
|
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("--major", action="store_true")
|
||||||
|
parser.add_argument("--minor", action="store_true")
|
||||||
|
parser.add_argument("--rc", 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.major, args.minor, args.rc)
|
||||||
|
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")
|
||||||
244
packaging/debian/changelog
Normal file
244
packaging/debian/changelog
Normal file
@@ -0,0 +1,244 @@
|
|||||||
|
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
|
||||||
|
|
||||||
|
* New upstream release
|
||||||
|
* Update Debian Standards Version to 3.9.7
|
||||||
|
* Drop -dbg package (Debian will build -dbgsym package automatically)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Wed, 10 Feb 2016 00:42:23 +0100
|
||||||
|
|
||||||
|
tint2 (0.12.3-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New upstream release
|
||||||
|
* Fix privacy-breach-generic in documentation
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Thu, 26 Nov 2015 22:49:26 +0100
|
||||||
|
|
||||||
|
tint2 (0.12.2-1) unstable; urgency=medium
|
||||||
|
|
||||||
|
* New upstream release
|
||||||
|
* Drop all patches (applied upstream)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Fri, 11 Sep 2015 18:41:22 +0200
|
||||||
|
|
||||||
|
tint2 (0.12.1-2) unstable; urgency=medium
|
||||||
|
|
||||||
|
* Import upstream patch fixing bug with blinking systray icon
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Tue, 04 Aug 2015 01:16:08 +0200
|
||||||
|
|
||||||
|
tint2 (0.12.1-1) unstable; urgency=medium
|
||||||
|
|
||||||
|
* New upstream release
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Mon, 03 Aug 2015 17:03:13 +0200
|
||||||
|
|
||||||
|
tint2 (0.12-1) unstable; urgency=medium
|
||||||
|
|
||||||
|
* New upstream release (Closes: #793797)
|
||||||
|
- Drop 04-freespace.patch (applied upstream)
|
||||||
|
- Drop 05-task-align.patch (needs rework)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Wed, 29 Jul 2015 18:53:03 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20121014-3) unstable; urgency=low
|
||||||
|
|
||||||
|
* Add some patches from upstream's bug tracker. Those are already
|
||||||
|
used by the Debian based VSIDO distribution for some time.
|
||||||
|
- 03-launcher_apps_dir.patch
|
||||||
|
Add support for loading multiple launcher icons by specifing
|
||||||
|
the directory for their *.desktop files.
|
||||||
|
- 04-freespace.patch
|
||||||
|
Add support for a separator panel element.
|
||||||
|
- 05-task-align.patch, 06-sample-task-align.patch
|
||||||
|
Add support for task alignment.
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Thu, 05 Jun 2014 17:39:43 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20121014-2) unstable; urgency=low
|
||||||
|
|
||||||
|
* Reintroduce tint2conf (Closes: #720200)
|
||||||
|
* Add patch fixing argv evaluation (Closes: #716390)
|
||||||
|
* Add patch fixing incorrect function declaration (Closes: #748171)
|
||||||
|
* Bump Debian Standards Version to 3.9.5
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Sun, 01 Jun 2014 17:13:06 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20121014-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New upstream checkout (Closes: #678918)
|
||||||
|
- Fix for no task cycling if all windows are minimized
|
||||||
|
- Fix for incorrect task cycling when windows visible on
|
||||||
|
all desktops are present
|
||||||
|
- Adding startup-notification support
|
||||||
|
- Drag and drop support for launchers
|
||||||
|
- Localize launcher tooltips
|
||||||
|
- Launcher: add icon lookup in ~/.local and /usr/local
|
||||||
|
- Disable mouse hover events when the launcher tooltip
|
||||||
|
is disabled
|
||||||
|
* Do not install tint2conf (Closes: #672840)
|
||||||
|
* Add libstartup-notification dependency
|
||||||
|
* Bump Debian Standards Version to 3.9.4
|
||||||
|
* Bump compat level to 9
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Thu, 25 Oct 2012 13:35:15 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20111022-3) unstable; urgency=low
|
||||||
|
|
||||||
|
* Update debian/copyright to conform with Debian copyright format 1.0
|
||||||
|
* Update Debian Standards Version to 3.9.3
|
||||||
|
* Add dump of tint2's FAQ and Configure wiki page as offline
|
||||||
|
documentation (Closes: #658226)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Wed, 07 Mar 2012 10:09:20 +0100
|
||||||
|
|
||||||
|
tint2 (0.11+svn20111022-2) unstable; urgency=low
|
||||||
|
|
||||||
|
* Add inform-about-unsupported-saving.patch (Closes: #646055)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Sun, 23 Oct 2011 10:57:35 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20111022-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New upstream checkout
|
||||||
|
- Fix double free of launcher configuration (Closes: #645208)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Sat, 22 Oct 2011 08:17:18 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20111011-2) unstable; urgency=low
|
||||||
|
|
||||||
|
* build debug symbols
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Fri, 21 Oct 2011 16:30:50 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20111011-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New upstream snapshot
|
||||||
|
- misc. fixes
|
||||||
|
- added alpha, saturation, brightness control for launcher icons
|
||||||
|
- launcher tooltips configurable, default off
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Wed, 19 Oct 2011 19:49:54 +0200
|
||||||
|
|
||||||
|
tint2 (0.11+svn20110307-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New upstream snapshot (Closes: #624792)
|
||||||
|
* Remove power_now support patch (applied upstream)
|
||||||
|
* Do not install tintwizard (it does not support config of SVN snapshot)
|
||||||
|
* Also build a tint2-dbg package
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Sun, 01 May 2011 19:35:32 +0200
|
||||||
|
|
||||||
|
tint2 (0.11-2) unstable; urgency=low
|
||||||
|
|
||||||
|
* Recommend python-gtk2 instead of python-gtk
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Sun, 05 Dec 2010 05:05:46 +0100
|
||||||
|
|
||||||
|
tint2 (0.11-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* Make me the maintainer of this package (Closes: #598688)
|
||||||
|
* New Upstream Version (Closes: #591008)
|
||||||
|
* drop patch (applied upstream)
|
||||||
|
* Update watch file
|
||||||
|
+ use googlecode.debian.net
|
||||||
|
+ new files are release as tar.bz2
|
||||||
|
* Update Debian Standards Version to 3.9.1
|
||||||
|
* Use DEP5 for debian/copyright
|
||||||
|
* Added build dependencies: cmake, libgtk2.0-dev
|
||||||
|
* Add patch renaming tintwizard.py to tintwizard
|
||||||
|
* Add patch adding power_now support
|
||||||
|
* Recommend python (needed by tintwizard)
|
||||||
|
|
||||||
|
-- Sebastian Reichel <sre@debian.org> Fri, 26 Nov 2010 22:01:43 +0100
|
||||||
|
|
||||||
|
tint2 (0.9-2) unstable; urgency=low
|
||||||
|
|
||||||
|
* debian/patches/update-systray-clock-every-second.diff: Cherry-pick
|
||||||
|
patch from upstream to keep systray clock in sync with system time.
|
||||||
|
(Closes: #572227)
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Wed, 03 Mar 2010 23:40:02 -0800
|
||||||
|
|
||||||
|
tint2 (0.9-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New Upstream Version
|
||||||
|
* debian/watch: Upstream uses both _beta and -rc1 for version strings,
|
||||||
|
update uversionmangle to handle both.
|
||||||
|
* debian/control:
|
||||||
|
- Bump to Version 3.8.4, no changes.
|
||||||
|
- Add libxdamage-dev and libxcomposite-dev to Build-Depends.
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Wed, 24 Feb 2010 10:21:01 -0800
|
||||||
|
|
||||||
|
tint2 (0.8-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New Upstream Version
|
||||||
|
* debian/examples: Upstream has renamed the sample tint2rc files.
|
||||||
|
* debian/source: Switch to dpkg-source 3.0 (quilt) format.
|
||||||
|
* debian/copyright: Update with new upstream authors.
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Tue, 12 Jan 2010 11:48:49 -0800
|
||||||
|
|
||||||
|
tint2 (0.7.1-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New Upstream Version
|
||||||
|
* debian/watch: add uversionmangle line to ignore upstream's beta
|
||||||
|
releases.
|
||||||
|
* debian/control: Update to Debian Policy 3.8.3, fix description typo.
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Sun, 06 Sep 2009 18:57:06 -0700
|
||||||
|
|
||||||
|
tint2 (0.7-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* New Upstream Version. (LP: #319436)
|
||||||
|
- Update watch file to point to tint2 instead of tint.
|
||||||
|
- Remove debian/tint.1: integrated upstream (Closes: #532307)
|
||||||
|
- Remove dpatch and patches/01-remove-strip-from-makefile.dpatch:
|
||||||
|
integrated upstream.
|
||||||
|
- Refresh debian/examples and remove debian/docs
|
||||||
|
- tint2 is now licensed under the GPL-2.
|
||||||
|
* Added Vcs-Git and Vcs-Browser fields to debian/control.
|
||||||
|
* Updated to Standards Version 3.8.2
|
||||||
|
* Switched to tiny dh 7 rules file with overrides.
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Wed, 01 Jul 2009 02:01:27 -0700
|
||||||
|
|
||||||
|
tint2 (0.6.0-1) unstable; urgency=low
|
||||||
|
|
||||||
|
* Initial release (closes: #491596)
|
||||||
|
* Wrote manpage from upstream docs
|
||||||
|
* 01-remove-strip-from-makefile.dpatch: dh_strip will strip the binary instead
|
||||||
|
|
||||||
|
-- Daniel Moerner <dmoerner@gmail.com> Sat, 06 Sep 2008 07:54:24 -0700
|
||||||
1
packaging/debian/compat
Normal file
1
packaging/debian/compat
Normal file
@@ -0,0 +1 @@
|
|||||||
|
9
|
||||||
36
packaging/debian/control
Normal file
36
packaging/debian/control
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
Source: tint2
|
||||||
|
Section: x11
|
||||||
|
Priority: optional
|
||||||
|
Maintainer: Sebastian Reichel <sre@debian.org>
|
||||||
|
Build-Depends: cmake,
|
||||||
|
debhelper (>= 9),
|
||||||
|
libcairo2-dev,
|
||||||
|
libglib2.0-dev,
|
||||||
|
libgtk2.0-dev,
|
||||||
|
libimlib2-dev,
|
||||||
|
libpango1.0-dev,
|
||||||
|
librsvg2-dev,
|
||||||
|
libstartup-notification0-dev,
|
||||||
|
libxcomposite-dev,
|
||||||
|
libxdamage-dev,
|
||||||
|
libxinerama-dev,
|
||||||
|
libxrandr-dev
|
||||||
|
Standards-Version: 3.9.7
|
||||||
|
Vcs-Git: https://alioth.debian.org/anonscm/git/collab-maint/tint2.git
|
||||||
|
Vcs-Browser: https://anonscm.debian.org/gitweb/?p=collab-maint/tint2.git
|
||||||
|
Homepage: https://gitlab.com/o9000/tint2/
|
||||||
|
|
||||||
|
Package: tint2
|
||||||
|
Architecture: any
|
||||||
|
Depends: ${shlibs:Depends},
|
||||||
|
${misc:Depends}
|
||||||
|
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.
|
||||||
61
packaging/debian/copyright
Normal file
61
packaging/debian/copyright
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
Format: http://www.debian.org/doc/packaging-manuals/copyright-format/1.0/
|
||||||
|
Source: https://gitlab.com/o9000/tint2.git
|
||||||
|
|
||||||
|
Files: *
|
||||||
|
Copyright: 2007-2008 Pål Staurland <staura@gmail.com>
|
||||||
|
2008-2009 Thierry Lorthiois <lorthiois@bbsoft.fr>
|
||||||
|
2009 Andreas Fink <andreas.fink85@googlemail.com>
|
||||||
|
2011-2015 Ovidiu M <mrovi9000@gmail.com>
|
||||||
|
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/*
|
||||||
|
Copyright: 2009 Sebastian Reichel <sre@debian.org>
|
||||||
|
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: debian/*
|
||||||
|
Copyright: 2008-2010 Daniel Moerner <dmoerner@gmail.com>
|
||||||
|
2010-2012 Sebastian Reichel <sre@debian.org>
|
||||||
|
License: ISC
|
||||||
|
Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
purpose with or without fee is hereby granted, provided that the above
|
||||||
|
copyright notice and this permission notice appear in all copies.
|
||||||
|
.
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
1
packaging/debian/examples
Normal file
1
packaging/debian/examples
Normal file
@@ -0,0 +1 @@
|
|||||||
|
sample/*.tint2rc
|
||||||
13
packaging/debian/rules
Executable file
13
packaging/debian/rules
Executable file
@@ -0,0 +1,13 @@
|
|||||||
|
#!/usr/bin/make -f
|
||||||
|
|
||||||
|
%:
|
||||||
|
dh $@
|
||||||
|
|
||||||
|
override_dh_auto_configure:
|
||||||
|
dh_auto_configure -- -DCMAKE_BUILD_TYPE=RelWithDebInfo
|
||||||
|
|
||||||
|
override_dh_auto_install:
|
||||||
|
dh_auto_install --destdir=$(CURDIR)/debian/tmp
|
||||||
|
|
||||||
|
override_dh_installdocs:
|
||||||
|
dh_installdocs --link-doc=tint2
|
||||||
1
packaging/debian/source/format
Normal file
1
packaging/debian/source/format
Normal file
@@ -0,0 +1 @@
|
|||||||
|
3.0 (quilt)
|
||||||
0
packaging/debian/source/include-binaries
Normal file
0
packaging/debian/source/include-binaries
Normal file
12
packaging/debian/tint2.install
Normal file
12
packaging/debian/tint2.install
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
/usr/bin/tint2
|
||||||
|
/usr/bin/tint2conf
|
||||||
|
/etc/xdg/tint2/tint2rc
|
||||||
|
/usr/share/applications/tint2.desktop
|
||||||
|
/usr/share/applications/tint2conf.desktop
|
||||||
|
/usr/share/man/man1/tint2.1
|
||||||
|
/usr/share/doc/tint2/*
|
||||||
|
/usr/share/icons/hicolor/scalable/apps/tint2.svg
|
||||||
|
/usr/share/icons/hicolor/scalable/apps/tint2conf.svg
|
||||||
|
/usr/share/locale/*
|
||||||
|
/usr/share/tint2/*
|
||||||
|
/usr/share/mime/packages/*
|
||||||
1
packaging/debian/tint2.manpages
Normal file
1
packaging/debian/tint2.manpages
Normal file
@@ -0,0 +1 @@
|
|||||||
|
debian/tint2conf.1
|
||||||
15
packaging/debian/tint2conf.1
Normal file
15
packaging/debian/tint2conf.1
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
.TH TINT2CONF 1 "April 25, 2011"
|
||||||
|
.\" Please adjust this date whenever revising the manpage.
|
||||||
|
.SH NAME
|
||||||
|
tint2conf \- tint configuration manager
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.B tint2conf
|
||||||
|
.SH DESCRIPTION
|
||||||
|
tint2conf is a GTK based configuration previewer for tint2.
|
||||||
|
.SH OPTIONS
|
||||||
|
tint2conf takes no parameters.
|
||||||
|
.SH AUTHOR
|
||||||
|
tint2conf was written by the tint2 team.
|
||||||
|
.PP
|
||||||
|
This manual page was written by Sebastian Reichel <sre@debian.org>,
|
||||||
|
for the Debian project (but may be used by others).
|
||||||
4
packaging/debian/watch
Normal file
4
packaging/debian/watch
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
version=3
|
||||||
|
|
||||||
|
opts=filenamemangle=s/.*\.tar\.gz\?ref=v?(\d\S*)/tint2-$1\.tar\.gz/g,uversionmangle=s/\-rc(\d)/\~rc$1/ \
|
||||||
|
https://gitlab.com/o9000/tint2/tags .*archive\.tar\.gz\?ref=v?(\d\S*)
|
||||||
79
packaging/make_ubuntu.sh
Executable file
79
packaging/make_ubuntu.sh
Executable file
@@ -0,0 +1,79 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Requirements: devscripts
|
||||||
|
|
||||||
|
set -x
|
||||||
|
|
||||||
|
rm -rf tint2* 2>/dev/null || true
|
||||||
|
|
||||||
|
if [ ! -z "$1" ]
|
||||||
|
then
|
||||||
|
MINOR="$1"
|
||||||
|
else
|
||||||
|
MINOR="1"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Get version (and check that the repository is clean)
|
||||||
|
VERSION=$(../get_version.sh --strict)
|
||||||
|
if [ ! $? -eq 0 ]
|
||||||
|
then
|
||||||
|
echo >&2 "Error: get_version.sh failed!"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
rm -f version.h
|
||||||
|
VERSION=$(git describe --exact-match 2>/dev/null)
|
||||||
|
if [ $? -eq 0 ]
|
||||||
|
then
|
||||||
|
VERSION=$(echo "$VERSION" | sed 's/^v//')
|
||||||
|
REPO="tint2"
|
||||||
|
else
|
||||||
|
VERSION="$(git show -s --pretty=format:%cI.%ct.%h | tr -d ':' | tr -d '-' | tr '.' '-' | sed 's/T[0-9\+]*//g').$MINOR"
|
||||||
|
REPO="tint2-git"
|
||||||
|
fi
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Export repository contents to source directory
|
||||||
|
DIR=tint2-$VERSION
|
||||||
|
echo "Making release $DIR"
|
||||||
|
|
||||||
|
pushd .
|
||||||
|
cd ..
|
||||||
|
git checkout-index --prefix=packaging/$DIR/ -a
|
||||||
|
popd
|
||||||
|
|
||||||
|
# Update version file in source directory
|
||||||
|
rm -f $DIR/make_release.sh
|
||||||
|
echo "echo \"#define VERSION_STRING \\\"$VERSION\\\"\" > version.h" > $DIR/get_version.sh
|
||||||
|
|
||||||
|
# Copy the debian files into the source directory
|
||||||
|
cp -r debian $DIR/debian
|
||||||
|
|
||||||
|
for DISTRO in precise trusty xenial yakkety zesty
|
||||||
|
do
|
||||||
|
# Cleanup from previous builds
|
||||||
|
rm -rf tint2_$VERSION-*
|
||||||
|
|
||||||
|
# Update debian package changelog if necessary
|
||||||
|
echo -e "tint2 ($VERSION-$DISTRO-1) $DISTRO; urgency=medium\n\n$(git log --pretty=format:' * %h %an (%ci) %s %d')\n -- o9000 <mrovi9000@gmail.com> $(date -R)\n" > $DIR/debian/changelog
|
||||||
|
|
||||||
|
# Create source tarball
|
||||||
|
ARCHIVE=tint2_$VERSION-$DISTRO.orig.tar.gz
|
||||||
|
rm -rf $ARCHIVE
|
||||||
|
tar -czf $ARCHIVE $DIR
|
||||||
|
|
||||||
|
# Build package
|
||||||
|
KEY=$(gpg --list-secret-keys | awk '/^sec/ { print $2 }' | cut -d / -f 2)
|
||||||
|
|
||||||
|
pushd .
|
||||||
|
cd $DIR
|
||||||
|
debuild -S -k$KEY
|
||||||
|
popd
|
||||||
|
|
||||||
|
# Upload package
|
||||||
|
dput ppa:o9000/$REPO tint2_$VERSION-$DISTRO-1_source.changes
|
||||||
|
done
|
||||||
|
|
||||||
|
# Cleanup
|
||||||
|
rm -rf $DIR $ARCHIVE
|
||||||
|
rm -rf tint2_$VERSION-*
|
||||||
118
sample/tint2rc
118
sample/tint2rc
@@ -1,118 +0,0 @@
|
|||||||
# Tint2 config file
|
|
||||||
# Generated by tintwizard (http://code.google.com/p/tintwizard/)
|
|
||||||
# For information on manually configuring tint2 see http://code.google.com/p/tint2/wiki/Configure
|
|
||||||
|
|
||||||
# Background definitions
|
|
||||||
# ID 1
|
|
||||||
rounded = 7
|
|
||||||
border_width = 2
|
|
||||||
background_color = #000000 60
|
|
||||||
border_color = #FFFFFF 16
|
|
||||||
|
|
||||||
# ID 2
|
|
||||||
rounded = 5
|
|
||||||
border_width = 0
|
|
||||||
background_color = #FFFFFF 40
|
|
||||||
border_color = #FFFFFF 48
|
|
||||||
|
|
||||||
# ID 3
|
|
||||||
rounded = 5
|
|
||||||
border_width = 0
|
|
||||||
background_color = #FFFFFF 16
|
|
||||||
border_color = #FFFFFF 68
|
|
||||||
|
|
||||||
# Panel
|
|
||||||
panel_monitor = all
|
|
||||||
panel_position = bottom center horizontal
|
|
||||||
panel_size = 94% 30
|
|
||||||
panel_margin = 0 0
|
|
||||||
panel_padding = 7 0 7
|
|
||||||
panel_dock = 0
|
|
||||||
wm_menu = 0
|
|
||||||
panel_layer = top
|
|
||||||
panel_background_id = 1
|
|
||||||
|
|
||||||
# Panel Autohide
|
|
||||||
autohide = 0
|
|
||||||
autohide_show_timeout = 0.3
|
|
||||||
autohide_hide_timeout = 2
|
|
||||||
autohide_height = 2
|
|
||||||
strut_policy = follow_size
|
|
||||||
|
|
||||||
# Taskbar
|
|
||||||
taskbar_mode = single_desktop
|
|
||||||
taskbar_padding = 2 3 2
|
|
||||||
taskbar_background_id = 0
|
|
||||||
taskbar_active_background_id = 0
|
|
||||||
|
|
||||||
# Tasks
|
|
||||||
urgent_nb_of_blink = 8
|
|
||||||
task_icon = 1
|
|
||||||
task_text = 1
|
|
||||||
task_centered = 1
|
|
||||||
task_maximum_size = 140 35
|
|
||||||
task_padding = 6 2
|
|
||||||
task_background_id = 3
|
|
||||||
task_active_background_id = 2
|
|
||||||
task_urgent_background_id = 2
|
|
||||||
task_iconified_background_id = 3
|
|
||||||
task_tooltip = 0
|
|
||||||
|
|
||||||
# Task Icons
|
|
||||||
task_icon_asb = 70 0 0
|
|
||||||
task_active_icon_asb = 100 0 0
|
|
||||||
task_urgent_icon_asb = 100 0 0
|
|
||||||
task_iconified_icon_asb = 70 0 0
|
|
||||||
|
|
||||||
# Fonts
|
|
||||||
task_font = sans 7
|
|
||||||
task_font_color = #FFFFFF 68
|
|
||||||
task_active_font_color = #FFFFFF 83
|
|
||||||
task_urgent_font_color = #FFFFFF 83
|
|
||||||
task_iconified_font_color = #FFFFFF 68
|
|
||||||
font_shadow = 0
|
|
||||||
|
|
||||||
# System Tray
|
|
||||||
systray = 1
|
|
||||||
systray_padding = 0 4 5
|
|
||||||
systray_sort = ascending
|
|
||||||
systray_background_id = 0
|
|
||||||
systray_icon_size = 16
|
|
||||||
systray_icon_asb = 70 0 0
|
|
||||||
|
|
||||||
# Clock
|
|
||||||
time1_format = %H:%M
|
|
||||||
time1_font = sans 8
|
|
||||||
time2_format = %A %d %B
|
|
||||||
time2_font = sans 6
|
|
||||||
clock_font_color = #FFFFFF 74
|
|
||||||
clock_padding = 1 0
|
|
||||||
clock_background_id = 0
|
|
||||||
clock_rclick_command = orage
|
|
||||||
|
|
||||||
# Tooltips
|
|
||||||
tooltip_padding = 2 2
|
|
||||||
tooltip_show_timeout = 0.7
|
|
||||||
tooltip_hide_timeout = 0.3
|
|
||||||
tooltip_background_id = 1
|
|
||||||
tooltip_font = sans 10
|
|
||||||
tooltip_font_color = #000000 80
|
|
||||||
|
|
||||||
# Mouse
|
|
||||||
mouse_middle = none
|
|
||||||
mouse_right = close
|
|
||||||
mouse_scroll_up = toggle
|
|
||||||
mouse_scroll_down = iconify
|
|
||||||
|
|
||||||
# Battery
|
|
||||||
battery = 0
|
|
||||||
battery_low_status = 10
|
|
||||||
battery_low_cmd = notify-send "battery low"
|
|
||||||
battery_hide = 98
|
|
||||||
bat1_font = sans 8
|
|
||||||
bat2_font = sans 6
|
|
||||||
battery_font_color = #FFFFFF 74
|
|
||||||
battery_padding = 1 0
|
|
||||||
battery_background_id = 0
|
|
||||||
|
|
||||||
# End of config
|
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
*
|
*
|
||||||
* Tint2 : battery
|
* Tint2 : Generic battery
|
||||||
*
|
*
|
||||||
* Copyright (C) 2009 Sebastian Reichel <elektranox@gmail.com>
|
* Copyright (C) 2009-2015 Sebastian Reichel <sre@ring0.de>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or
|
* This program is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License version 2
|
* modify it under the terms of the GNU General Public License version 2
|
||||||
@@ -24,18 +24,6 @@
|
|||||||
#include <cairo-xlib.h>
|
#include <cairo-xlib.h>
|
||||||
#include <pango/pangocairo.h>
|
#include <pango/pangocairo.h>
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
|
||||||
#include <machine/apmvar.h>
|
|
||||||
#include <err.h>
|
|
||||||
#include <sys/ioctl.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(__FreeBSD__)
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/sysctl.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "window.h"
|
#include "window.h"
|
||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "panel.h"
|
#include "panel.h"
|
||||||
@@ -43,440 +31,431 @@
|
|||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
|
gboolean bat1_has_font;
|
||||||
PangoFontDescription *bat1_font_desc;
|
PangoFontDescription *bat1_font_desc;
|
||||||
|
gboolean bat2_has_font;
|
||||||
PangoFontDescription *bat2_font_desc;
|
PangoFontDescription *bat2_font_desc;
|
||||||
struct batstate battery_state;
|
struct BatteryState battery_state;
|
||||||
int battery_enabled;
|
gboolean battery_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];
|
static char buf_bat_percentage[10];
|
||||||
static char buf_bat_time[20];
|
static char buf_bat_time[20];
|
||||||
|
|
||||||
int8_t battery_low_status;
|
int8_t battery_low_status;
|
||||||
unsigned char battery_low_cmd_send;
|
gboolean battery_low_cmd_sent;
|
||||||
|
char *ac_connected_cmd;
|
||||||
|
char *ac_disconnected_cmd;
|
||||||
char *battery_low_cmd;
|
char *battery_low_cmd;
|
||||||
char *path_energy_now;
|
char *battery_lclick_command;
|
||||||
char *path_energy_full;
|
char *battery_mclick_command;
|
||||||
char *path_current_now;
|
char *battery_rclick_command;
|
||||||
char *path_status;
|
char *battery_uwheel_command;
|
||||||
|
char *battery_dwheel_command;
|
||||||
|
gboolean battery_found;
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
char *battery_sys_prefix = (char *)"";
|
||||||
int apm_fd;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void update_batterys(void* arg)
|
void battery_init_fonts();
|
||||||
{
|
char *battery_get_tooltip(void *obj);
|
||||||
int old_percentage = battery_state.percentage;
|
int battery_compute_desired_size(void *obj);
|
||||||
int16_t old_hours = battery_state.time.hours;
|
void battery_dump_geometry(void *obj, int indent);
|
||||||
int8_t old_minutes = battery_state.time.minutes;
|
|
||||||
|
|
||||||
update_battery();
|
|
||||||
if (old_percentage == battery_state.percentage && old_hours == battery_state.time.hours && old_minutes == battery_state.time.minutes)
|
|
||||||
return;
|
|
||||||
|
|
||||||
int i;
|
|
||||||
for (i=0 ; i < nb_panel ; i++) {
|
|
||||||
if (battery_state.percentage >= percentage_hide) {
|
|
||||||
if (panel1[i].battery.area.on_screen == 1) {
|
|
||||||
hide(&panel1[i].battery.area);
|
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (panel1[i].battery.area.on_screen == 0) {
|
|
||||||
show(&panel1[i].battery.area);
|
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (panel1[i].battery.area.on_screen == 1) {
|
|
||||||
panel1[i].battery.area.resize = 1;
|
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void default_battery()
|
void default_battery()
|
||||||
{
|
{
|
||||||
battery_enabled = 0;
|
battery_enabled = FALSE;
|
||||||
percentage_hide = 101;
|
battery_tooltip_enabled = TRUE;
|
||||||
battery_low_cmd_send = 0;
|
battery_found = FALSE;
|
||||||
battery_timeout = 0;
|
percentage_hide = 101;
|
||||||
bat1_font_desc = 0;
|
battery_low_cmd_sent = FALSE;
|
||||||
bat2_font_desc = 0;
|
battery_timeout = NULL;
|
||||||
battery_low_cmd = 0;
|
bat1_has_font = FALSE;
|
||||||
path_energy_now = 0;
|
bat1_font_desc = NULL;
|
||||||
path_energy_full = 0;
|
bat2_has_font = FALSE;
|
||||||
path_current_now = 0;
|
bat2_font_desc = NULL;
|
||||||
path_status = 0;
|
ac_connected_cmd = NULL;
|
||||||
battery_state.percentage = 0;
|
ac_disconnected_cmd = NULL;
|
||||||
battery_state.time.hours = 0;
|
battery_low_cmd = NULL;
|
||||||
battery_state.time.minutes = 0;
|
battery_lclick_command = NULL;
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
battery_mclick_command = NULL;
|
||||||
apm_fd = -1;
|
battery_rclick_command = NULL;
|
||||||
#endif
|
battery_uwheel_command = NULL;
|
||||||
|
battery_dwheel_command = NULL;
|
||||||
|
battery_state.percentage = 0;
|
||||||
|
battery_state.time.hours = 0;
|
||||||
|
battery_state.time.minutes = 0;
|
||||||
|
battery_state.time.seconds = 0;
|
||||||
|
battery_state.state = BATTERY_UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cleanup_battery()
|
void cleanup_battery()
|
||||||
{
|
{
|
||||||
if (bat1_font_desc) pango_font_description_free(bat1_font_desc);
|
pango_font_description_free(bat1_font_desc);
|
||||||
if (bat2_font_desc) pango_font_description_free(bat2_font_desc);
|
bat1_font_desc = NULL;
|
||||||
if (path_energy_now) g_free(path_energy_now);
|
pango_font_description_free(bat2_font_desc);
|
||||||
if (path_energy_full) g_free(path_energy_full);
|
bat2_font_desc = NULL;
|
||||||
if (path_current_now) g_free(path_current_now);
|
free(battery_low_cmd);
|
||||||
if (path_status) g_free(path_status);
|
battery_low_cmd = NULL;
|
||||||
if (battery_low_cmd) g_free(battery_low_cmd);
|
free(battery_lclick_command);
|
||||||
if (battery_timeout) stop_timeout(battery_timeout);
|
battery_lclick_command = NULL;
|
||||||
|
free(battery_mclick_command);
|
||||||
|
battery_mclick_command = NULL;
|
||||||
|
free(battery_rclick_command);
|
||||||
|
battery_rclick_command = NULL;
|
||||||
|
free(battery_uwheel_command);
|
||||||
|
battery_uwheel_command = NULL;
|
||||||
|
free(battery_dwheel_command);
|
||||||
|
battery_dwheel_command = NULL;
|
||||||
|
free(ac_connected_cmd);
|
||||||
|
ac_connected_cmd = NULL;
|
||||||
|
free(ac_disconnected_cmd);
|
||||||
|
ac_disconnected_cmd = NULL;
|
||||||
|
stop_timeout(battery_timeout);
|
||||||
|
battery_timeout = NULL;
|
||||||
|
battery_found = FALSE;
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
battery_os_free();
|
||||||
if ((apm_fd != -1) && (close(apm_fd) == -1))
|
|
||||||
warn("cannot close /dev/apm");
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_battery()
|
void init_battery()
|
||||||
{
|
{
|
||||||
if (!battery_enabled) return;
|
if (!battery_enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
battery_found = battery_os_init();
|
||||||
apm_fd = open("/dev/apm", O_RDONLY);
|
|
||||||
if (apm_fd < 0) {
|
|
||||||
warn("init_battery: failed to open /dev/apm.");
|
|
||||||
battery_enabled = 0;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
#elif !defined(__FreeBSD__)
|
if (!battery_timeout)
|
||||||
// check battery
|
battery_timeout = add_timeout(10, 30000, update_battery_tick, 0, &battery_timeout);
|
||||||
GDir *directory = 0;
|
|
||||||
GError *error = NULL;
|
|
||||||
const char *entryname;
|
|
||||||
char *battery_dir = 0;
|
|
||||||
|
|
||||||
directory = g_dir_open("/sys/class/power_supply", 0, &error);
|
update_battery();
|
||||||
if (error)
|
|
||||||
g_error_free(error);
|
|
||||||
else {
|
|
||||||
while ((entryname=g_dir_read_name(directory))) {
|
|
||||||
if (strncmp(entryname,"AC", 2) == 0) continue;
|
|
||||||
|
|
||||||
char *path1 = g_build_filename("/sys/class/power_supply", entryname, "present", NULL);
|
|
||||||
if (g_file_test (path1, G_FILE_TEST_EXISTS)) {
|
|
||||||
g_free(path1);
|
|
||||||
battery_dir = g_build_filename("/sys/class/power_supply", entryname, NULL);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
g_free(path1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (directory)
|
|
||||||
g_dir_close(directory);
|
|
||||||
if (!battery_dir) {
|
|
||||||
fprintf(stderr, "ERROR: battery applet can't found power_supply\n");
|
|
||||||
default_battery();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
char *path1 = g_build_filename(battery_dir, "energy_now", NULL);
|
|
||||||
if (g_file_test (path1, G_FILE_TEST_EXISTS)) {
|
|
||||||
path_energy_now = g_build_filename(battery_dir, "energy_now", NULL);
|
|
||||||
path_energy_full = g_build_filename(battery_dir, "energy_full", NULL);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
char *path2 = g_build_filename(battery_dir, "charge_now", NULL);
|
|
||||||
if (g_file_test (path2, G_FILE_TEST_EXISTS)) {
|
|
||||||
path_energy_now = g_build_filename(battery_dir, "charge_now", NULL);
|
|
||||||
path_energy_full = g_build_filename(battery_dir, "charge_full", NULL);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
fprintf(stderr, "ERROR: can't found energy_* or charge_*\n");
|
|
||||||
}
|
|
||||||
g_free(path2);
|
|
||||||
}
|
|
||||||
|
|
||||||
path_current_now = g_build_filename(battery_dir, "power_now", NULL);
|
|
||||||
if (!g_file_test (path_current_now, G_FILE_TEST_EXISTS)) {
|
|
||||||
g_free(path_current_now);
|
|
||||||
path_current_now = g_build_filename(battery_dir, "current_now", NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (path_energy_now && path_energy_full) {
|
|
||||||
path_status = g_build_filename(battery_dir, "status", NULL);
|
|
||||||
|
|
||||||
// check file
|
|
||||||
FILE *fp1, *fp2, *fp3, *fp4;
|
|
||||||
fp1 = fopen(path_energy_now, "r");
|
|
||||||
fp2 = fopen(path_energy_full, "r");
|
|
||||||
fp3 = fopen(path_current_now, "r");
|
|
||||||
fp4 = fopen(path_status, "r");
|
|
||||||
if (fp1 == NULL || fp2 == NULL || fp3 == NULL || fp4 == NULL) {
|
|
||||||
cleanup_battery();
|
|
||||||
default_battery();
|
|
||||||
fprintf(stderr, "ERROR: battery applet can't open energy_now\n");
|
|
||||||
}
|
|
||||||
fclose(fp1);
|
|
||||||
fclose(fp2);
|
|
||||||
fclose(fp3);
|
|
||||||
fclose(fp4);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_free(path1);
|
|
||||||
g_free(battery_dir);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (battery_enabled && battery_timeout==0)
|
|
||||||
battery_timeout = add_timeout(10, 10000, update_batterys, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void reinit_battery()
|
||||||
|
{
|
||||||
|
battery_os_free();
|
||||||
|
battery_found = battery_os_init();
|
||||||
|
update_battery();
|
||||||
|
}
|
||||||
|
|
||||||
void init_battery_panel(void *p)
|
void init_battery_panel(void *p)
|
||||||
{
|
{
|
||||||
Panel *panel = (Panel*)p;
|
Panel *panel = (Panel *)p;
|
||||||
Battery *battery = &panel->battery;
|
Battery *battery = &panel->battery;
|
||||||
|
|
||||||
if (!battery_enabled)
|
if (!battery_enabled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (battery->area.bg == 0)
|
battery_init_fonts();
|
||||||
battery->area.bg = &g_array_index(backgrounds, Background, 0);
|
|
||||||
|
|
||||||
battery->area.parent = p;
|
if (!battery->area.bg)
|
||||||
battery->area.panel = p;
|
battery->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
battery->area._draw_foreground = draw_battery;
|
|
||||||
battery->area.size_mode = SIZE_BY_CONTENT;
|
battery->area.parent = p;
|
||||||
battery->area._resize = resize_battery;
|
battery->area.panel = p;
|
||||||
battery->area.on_screen = 1;
|
snprintf(battery->area.name, sizeof(battery->area.name), "Battery");
|
||||||
battery->area.resize = 1;
|
battery->area._draw_foreground = draw_battery;
|
||||||
|
battery->area.size_mode = LAYOUT_FIXED;
|
||||||
|
battery->area._resize = resize_battery;
|
||||||
|
battery->area._compute_desired_size = battery_compute_desired_size;
|
||||||
|
battery->area._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
|
battery->area.on_screen = TRUE;
|
||||||
|
battery->area.resize_needed = 1;
|
||||||
|
battery->area.has_mouse_over_effect =
|
||||||
|
panel_config.mouse_effects && (battery_lclick_command || battery_mclick_command || battery_rclick_command ||
|
||||||
|
battery_uwheel_command || battery_dwheel_command);
|
||||||
|
battery->area.has_mouse_press_effect = battery->area.has_mouse_over_effect;
|
||||||
|
if (battery_tooltip_enabled)
|
||||||
|
battery->area._get_tooltip_text = battery_get_tooltip;
|
||||||
|
instantiate_area_gradients(&battery->area);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void battery_init_fonts()
|
||||||
void update_battery() {
|
|
||||||
#if !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__FreeBSD__)
|
|
||||||
// unused on OpenBSD, silence compiler warnings
|
|
||||||
FILE *fp;
|
|
||||||
char tmp[25];
|
|
||||||
int64_t current_now = 0;
|
|
||||||
#endif
|
|
||||||
#if defined(__FreeBSD__)
|
|
||||||
int sysctl_out = 0;
|
|
||||||
size_t len = 0;
|
|
||||||
#endif
|
|
||||||
int64_t energy_now = 0, energy_full = 0;
|
|
||||||
int seconds = 0;
|
|
||||||
int8_t new_percentage = 0;
|
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
|
||||||
struct apm_power_info info;
|
|
||||||
if (ioctl(apm_fd, APM_IOC_GETPOWER, &(info)) < 0)
|
|
||||||
warn("power update: APM_IOC_GETPOWER");
|
|
||||||
|
|
||||||
// best attempt at mapping to linux battery states
|
|
||||||
battery_state.state = BATTERY_UNKNOWN;
|
|
||||||
switch (info.battery_state) {
|
|
||||||
case APM_BATT_CHARGING:
|
|
||||||
battery_state.state = BATTERY_CHARGING;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
battery_state.state = BATTERY_DISCHARGING;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (info.battery_life == 100)
|
|
||||||
battery_state.state = BATTERY_FULL;
|
|
||||||
|
|
||||||
// no mapping for openbsd really
|
|
||||||
energy_full = 0;
|
|
||||||
energy_now = 0;
|
|
||||||
|
|
||||||
if (info.minutes_left != -1)
|
|
||||||
seconds = info.minutes_left * 60;
|
|
||||||
else
|
|
||||||
seconds = -1;
|
|
||||||
|
|
||||||
new_percentage = info.battery_life;
|
|
||||||
|
|
||||||
#elif defined(__FreeBSD__)
|
|
||||||
len = sizeof(sysctl_out);
|
|
||||||
|
|
||||||
if (sysctlbyname("hw.acpi.battery.state", &sysctl_out, &len, NULL, 0) != 0)
|
|
||||||
fprintf(stderr, "power update: no such sysctl");
|
|
||||||
|
|
||||||
// attemp to map the battery state to linux
|
|
||||||
battery_state.state = BATTERY_UNKNOWN;
|
|
||||||
|
|
||||||
switch(sysctl_out) {
|
|
||||||
case 1:
|
|
||||||
battery_state.state = BATTERY_DISCHARGING;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
battery_state.state = BATTERY_CHARGING;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
battery_state.state = BATTERY_FULL;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// no mapping for freebsd
|
|
||||||
energy_full = 0;
|
|
||||||
energy_now = 0;
|
|
||||||
|
|
||||||
if (sysctlbyname("hw.acpi.battery.time", &sysctl_out, &len, NULL, 0) != 0)
|
|
||||||
seconds = -1;
|
|
||||||
else
|
|
||||||
seconds = sysctl_out * 60;
|
|
||||||
|
|
||||||
// charging or error
|
|
||||||
if (seconds < 0)
|
|
||||||
seconds = 0;
|
|
||||||
|
|
||||||
if (sysctlbyname("hw.acpi.battery.life", &sysctl_out, &len, NULL, 0) != 0)
|
|
||||||
new_percentage = -1;
|
|
||||||
else
|
|
||||||
new_percentage = sysctl_out;
|
|
||||||
|
|
||||||
#else
|
|
||||||
fp = fopen(path_status, "r");
|
|
||||||
if(fp != NULL) {
|
|
||||||
if (fgets(tmp, sizeof tmp, fp)) {
|
|
||||||
battery_state.state = BATTERY_UNKNOWN;
|
|
||||||
if(strcasecmp(tmp, "Charging\n")==0) battery_state.state = BATTERY_CHARGING;
|
|
||||||
if(strcasecmp(tmp, "Discharging\n")==0) battery_state.state = BATTERY_DISCHARGING;
|
|
||||||
if(strcasecmp(tmp, "Full\n")==0) battery_state.state = BATTERY_FULL;
|
|
||||||
}
|
|
||||||
fclose(fp);
|
|
||||||
}
|
|
||||||
|
|
||||||
fp = fopen(path_energy_now, "r");
|
|
||||||
if(fp != NULL) {
|
|
||||||
if (fgets(tmp, sizeof tmp, fp)) energy_now = atoi(tmp);
|
|
||||||
fclose(fp);
|
|
||||||
}
|
|
||||||
|
|
||||||
fp = fopen(path_energy_full, "r");
|
|
||||||
if(fp != NULL) {
|
|
||||||
if (fgets(tmp, sizeof tmp, fp)) energy_full = atoi(tmp);
|
|
||||||
fclose(fp);
|
|
||||||
}
|
|
||||||
|
|
||||||
fp = fopen(path_current_now, "r");
|
|
||||||
if(fp != NULL) {
|
|
||||||
if (fgets(tmp, sizeof tmp, fp)) current_now = atoi(tmp);
|
|
||||||
fclose(fp);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(current_now > 0) {
|
|
||||||
switch(battery_state.state) {
|
|
||||||
case BATTERY_CHARGING:
|
|
||||||
seconds = 3600 * (energy_full - energy_now) / current_now;
|
|
||||||
break;
|
|
||||||
case BATTERY_DISCHARGING:
|
|
||||||
seconds = 3600 * energy_now / current_now;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
seconds = 0;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else seconds = 0;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
battery_state.time.hours = seconds / 3600;
|
|
||||||
seconds -= 3600 * battery_state.time.hours;
|
|
||||||
battery_state.time.minutes = seconds / 60;
|
|
||||||
seconds -= 60 * battery_state.time.minutes;
|
|
||||||
battery_state.time.seconds = seconds;
|
|
||||||
|
|
||||||
if(energy_full > 0)
|
|
||||||
new_percentage = (energy_now*100)/energy_full;
|
|
||||||
|
|
||||||
if(battery_low_status > new_percentage && battery_state.state == BATTERY_DISCHARGING && !battery_low_cmd_send) {
|
|
||||||
tint_exec(battery_low_cmd);
|
|
||||||
battery_low_cmd_send = 1;
|
|
||||||
}
|
|
||||||
if(battery_low_status < new_percentage && battery_state.state == BATTERY_CHARGING && battery_low_cmd_send) {
|
|
||||||
battery_low_cmd_send = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
battery_state.percentage = new_percentage;
|
|
||||||
|
|
||||||
// clamp percentage to 100 in case battery is misreporting that its current charge is more than its max
|
|
||||||
if(battery_state.percentage > 100) {
|
|
||||||
battery_state.percentage = 100;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void draw_battery (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Battery *battery = obj;
|
if (!bat1_font_desc) {
|
||||||
PangoLayout *layout;
|
bat1_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
pango_font_description_set_size(bat1_font_desc, pango_font_description_get_size(bat1_font_desc) - PANGO_SCALE);
|
||||||
layout = pango_cairo_create_layout (c);
|
}
|
||||||
|
if (!bat2_font_desc) {
|
||||||
// draw layout
|
bat2_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
pango_layout_set_font_description(layout, bat1_font_desc);
|
pango_font_description_set_size(bat2_font_desc, pango_font_description_get_size(bat2_font_desc) - PANGO_SCALE);
|
||||||
pango_layout_set_width(layout, battery->area.width * PANGO_SCALE);
|
}
|
||||||
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
|
|
||||||
pango_layout_set_text(layout, buf_bat_percentage, strlen(buf_bat_percentage));
|
|
||||||
|
|
||||||
cairo_set_source_rgba(c, battery->font.color[0], battery->font.color[1], battery->font.color[2], battery->font.alpha);
|
|
||||||
|
|
||||||
pango_cairo_update_layout(c, layout);
|
|
||||||
cairo_move_to(c, 0, battery->bat1_posy);
|
|
||||||
pango_cairo_show_layout(c, layout);
|
|
||||||
|
|
||||||
pango_layout_set_font_description(layout, bat2_font_desc);
|
|
||||||
pango_layout_set_indent(layout, 0);
|
|
||||||
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);
|
|
||||||
cairo_move_to(c, 0, battery->bat2_posy);
|
|
||||||
pango_cairo_show_layout(c, layout);
|
|
||||||
|
|
||||||
g_object_unref(layout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void battery_default_font_changed()
|
||||||
int resize_battery(void *obj)
|
|
||||||
{
|
{
|
||||||
Battery *battery = obj;
|
if (!battery_enabled)
|
||||||
Panel *panel = battery->area.panel;
|
return;
|
||||||
int bat_percentage_height, bat_percentage_width, bat_percentage_height_ink;
|
if (bat1_has_font && bat2_has_font)
|
||||||
int bat_time_height, bat_time_width, bat_time_height_ink;
|
return;
|
||||||
int ret = 0;
|
if (!bat1_has_font) {
|
||||||
|
pango_font_description_free(bat1_font_desc);
|
||||||
battery->area.redraw = 1;
|
bat1_font_desc = NULL;
|
||||||
|
}
|
||||||
snprintf(buf_bat_percentage, sizeof(buf_bat_percentage), "%d%%", battery_state.percentage);
|
if (!bat2_has_font) {
|
||||||
if(battery_state.state == BATTERY_FULL) {
|
pango_font_description_free(bat2_font_desc);
|
||||||
strcpy(buf_bat_time, "Full");
|
bat2_font_desc = NULL;
|
||||||
} else {
|
}
|
||||||
snprintf(buf_bat_time, sizeof(buf_bat_time), "%02d:%02d", battery_state.time.hours, battery_state.time.minutes);
|
battery_init_fonts();
|
||||||
}
|
for (int i = 0; i < num_panels; i++) {
|
||||||
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));
|
panels[i].battery.area.resize_needed = TRUE;
|
||||||
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));
|
schedule_redraw(&panels[i].battery.area);
|
||||||
|
}
|
||||||
if (panel_horizontal) {
|
schedule_panel_redraw();
|
||||||
int new_size = (bat_percentage_width > bat_time_width) ? bat_percentage_width : bat_time_width;
|
|
||||||
new_size += (2*battery->area.paddingxlr) + (2*battery->area.bg->border.width);
|
|
||||||
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 + battery->area.bg->border.width));
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void update_battery_tick(void *arg)
|
||||||
|
{
|
||||||
|
if (!battery_enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
gboolean old_found = battery_found;
|
||||||
|
int old_percentage = battery_state.percentage;
|
||||||
|
gboolean old_ac_connected = battery_state.ac_connected;
|
||||||
|
int16_t old_hours = battery_state.time.hours;
|
||||||
|
int8_t old_minutes = battery_state.time.minutes;
|
||||||
|
|
||||||
|
if (!battery_found) {
|
||||||
|
init_battery();
|
||||||
|
old_ac_connected = battery_state.ac_connected;
|
||||||
|
}
|
||||||
|
if (update_battery() != 0) {
|
||||||
|
// Try to reconfigure on failed update
|
||||||
|
init_battery();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (old_ac_connected != battery_state.ac_connected) {
|
||||||
|
if (battery_state.ac_connected)
|
||||||
|
tint_exec_no_sn(ac_connected_cmd);
|
||||||
|
else
|
||||||
|
tint_exec_no_sn(ac_disconnected_cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (battery_state.percentage < battery_low_status && battery_state.state == BATTERY_DISCHARGING &&
|
||||||
|
!battery_low_cmd_sent) {
|
||||||
|
tint_exec_no_sn(battery_low_cmd);
|
||||||
|
battery_low_cmd_sent = TRUE;
|
||||||
|
}
|
||||||
|
if (battery_state.percentage > battery_low_status && battery_state.state == BATTERY_CHARGING &&
|
||||||
|
battery_low_cmd_sent) {
|
||||||
|
battery_low_cmd_sent = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
// Show/hide if needed
|
||||||
|
if (!battery_found) {
|
||||||
|
hide(&panels[i].battery.area);
|
||||||
|
} else {
|
||||||
|
if (battery_state.percentage >= percentage_hide)
|
||||||
|
hide(&panels[i].battery.area);
|
||||||
|
else
|
||||||
|
show(&panels[i].battery.area);
|
||||||
|
}
|
||||||
|
// Redraw if needed
|
||||||
|
if (panels[i].battery.area.on_screen) {
|
||||||
|
if (old_found != battery_found || old_percentage != battery_state.percentage ||
|
||||||
|
old_hours != battery_state.time.hours || old_minutes != battery_state.time.minutes) {
|
||||||
|
panels[i].battery.area.resize_needed = TRUE;
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int update_battery()
|
||||||
|
{
|
||||||
|
// Reset
|
||||||
|
battery_state.state = BATTERY_UNKNOWN;
|
||||||
|
battery_state.percentage = 0;
|
||||||
|
battery_state.ac_connected = FALSE;
|
||||||
|
battery_state_set_time(&battery_state, 0);
|
||||||
|
|
||||||
|
int err = battery_os_update(&battery_state);
|
||||||
|
|
||||||
|
// Clamp percentage to 100 in case battery is misreporting that its current charge is more than its max
|
||||||
|
if (battery_state.percentage > 100) {
|
||||||
|
battery_state.percentage = 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
int battery_compute_desired_size(void *obj)
|
||||||
|
{
|
||||||
|
Battery *battery = (Battery *)obj;
|
||||||
|
Panel *panel = (Panel *)battery->area.panel;
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
Battery *battery = (Battery *)obj;
|
||||||
|
Panel *panel = (Panel *)battery->area.panel;
|
||||||
|
int bat_percentage_height, bat_percentage_width, bat_percentage_height_ink;
|
||||||
|
int bat_time_height, bat_time_width, bat_time_height_ink;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
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);
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
Battery *battery = obj;
|
||||||
|
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
pango_layout_set_font_description(layout, bat1_font_desc);
|
||||||
|
pango_layout_set_width(layout, battery->area.width * PANGO_SCALE);
|
||||||
|
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
|
||||||
|
pango_layout_set_text(layout, buf_bat_percentage, strlen(buf_bat_percentage));
|
||||||
|
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
Battery *battery = obj;
|
||||||
|
fprintf(stderr, "%*sText 1: y = %d, text = %s\n", 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)
|
||||||
|
{
|
||||||
|
return battery_os_tooltip();
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_action(int button, Time time)
|
||||||
|
{
|
||||||
|
char *command = NULL;
|
||||||
|
switch (button) {
|
||||||
|
case 1:
|
||||||
|
command = battery_lclick_command;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
command = battery_mclick_command;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
command = battery_rclick_command;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
command = battery_uwheel_command;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
command = battery_dwheel_command;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
tint_exec(command, NULL, NULL, time);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
* Copyright (C) 2009 Sebastian Reichel <elektranox@gmail.com>
|
* Copyright (C) 2009-2015 Sebastian Reichel <sre@ring0.de>
|
||||||
*
|
*
|
||||||
* Battery with functional data (percentage, time to life) and drawing data
|
* Battery with functional data (percentage, time to life) and drawing data
|
||||||
* (area, font, ...). Each panel use his own drawing data.
|
* (area, font, ...). Each panel use his own drawing data.
|
||||||
* Need kernel > 2.6.23.
|
|
||||||
*
|
*
|
||||||
**************************************************************************/
|
**************************************************************************/
|
||||||
|
|
||||||
@@ -17,45 +16,79 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "area.h"
|
#include "area.h"
|
||||||
|
|
||||||
|
|
||||||
// battery drawing parameter (per panel)
|
|
||||||
typedef struct Battery {
|
typedef struct Battery {
|
||||||
// always start with area
|
Area area;
|
||||||
Area area;
|
Color font_color;
|
||||||
|
int bat1_posy;
|
||||||
Color font;
|
int bat2_posy;
|
||||||
int bat1_posy;
|
|
||||||
int bat2_posy;
|
|
||||||
} Battery;
|
} Battery;
|
||||||
|
|
||||||
enum chargestate {
|
typedef enum ChargeState {
|
||||||
BATTERY_UNKNOWN,
|
BATTERY_UNKNOWN = 0,
|
||||||
BATTERY_CHARGING,
|
BATTERY_CHARGING,
|
||||||
BATTERY_DISCHARGING,
|
BATTERY_DISCHARGING,
|
||||||
BATTERY_FULL
|
BATTERY_FULL,
|
||||||
};
|
} ChargeState;
|
||||||
|
|
||||||
typedef struct battime {
|
typedef struct BatteryTime {
|
||||||
int16_t hours;
|
int16_t hours;
|
||||||
int8_t minutes;
|
int8_t minutes;
|
||||||
int8_t seconds;
|
int8_t seconds;
|
||||||
} battime;
|
} BatteryTime;
|
||||||
|
|
||||||
typedef struct batstate {
|
typedef struct BatteryState {
|
||||||
int percentage;
|
int percentage;
|
||||||
struct battime time;
|
BatteryTime time;
|
||||||
enum chargestate state;
|
ChargeState state;
|
||||||
} batstate;
|
gboolean ac_connected;
|
||||||
|
} BatteryState;
|
||||||
|
|
||||||
extern struct batstate battery_state;
|
extern struct BatteryState battery_state;
|
||||||
|
extern gboolean bat1_has_font;
|
||||||
extern PangoFontDescription *bat1_font_desc;
|
extern PangoFontDescription *bat1_font_desc;
|
||||||
|
extern gboolean bat2_has_font;
|
||||||
extern PangoFontDescription *bat2_font_desc;
|
extern PangoFontDescription *bat2_font_desc;
|
||||||
extern int battery_enabled;
|
extern gboolean battery_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 *path_energy_now, *path_energy_full, *path_current_now, *path_status;
|
|
||||||
|
extern char *ac_connected_cmd;
|
||||||
|
extern char *ac_disconnected_cmd;
|
||||||
|
|
||||||
|
extern char *battery_lclick_command;
|
||||||
|
extern char *battery_mclick_command;
|
||||||
|
extern char *battery_rclick_command;
|
||||||
|
extern char *battery_uwheel_command;
|
||||||
|
extern char *battery_dwheel_command;
|
||||||
|
|
||||||
|
extern char *battery_sys_prefix;
|
||||||
|
|
||||||
|
static inline gchar *chargestate2str(ChargeState state)
|
||||||
|
{
|
||||||
|
switch (state) {
|
||||||
|
case BATTERY_CHARGING:
|
||||||
|
return "Charging";
|
||||||
|
case BATTERY_DISCHARGING:
|
||||||
|
return "Discharging";
|
||||||
|
case BATTERY_FULL:
|
||||||
|
return "Full";
|
||||||
|
case BATTERY_UNKNOWN:
|
||||||
|
default:
|
||||||
|
return "Unknown";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void battery_state_set_time(BatteryState *state, int seconds)
|
||||||
|
{
|
||||||
|
state->time.hours = seconds / 3600;
|
||||||
|
seconds -= 3600 * state->time.hours;
|
||||||
|
state->time.minutes = seconds / 60;
|
||||||
|
seconds -= 60 * state->time.minutes;
|
||||||
|
state->time.seconds = seconds;
|
||||||
|
}
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_battery();
|
void default_battery();
|
||||||
@@ -63,14 +96,24 @@ void default_battery();
|
|||||||
// freed memory
|
// freed memory
|
||||||
void cleanup_battery();
|
void cleanup_battery();
|
||||||
|
|
||||||
// initialize clock : y position, ...
|
void update_battery_tick(void *arg);
|
||||||
void update_battery();
|
int update_battery();
|
||||||
|
|
||||||
void init_battery();
|
void init_battery();
|
||||||
void init_battery_panel(void *panel);
|
void init_battery_panel(void *panel);
|
||||||
|
|
||||||
|
void reinit_battery();
|
||||||
void draw_battery(void *obj, cairo_t *c);
|
void draw_battery(void *obj, cairo_t *c);
|
||||||
|
void battery_default_font_changed();
|
||||||
|
|
||||||
int resize_battery(void *obj);
|
gboolean resize_battery(void *obj);
|
||||||
|
|
||||||
|
void battery_action(int button, Time time);
|
||||||
|
|
||||||
|
/* operating system specific functions */
|
||||||
|
gboolean battery_os_init();
|
||||||
|
void battery_os_free();
|
||||||
|
int battery_os_update(BatteryState *state);
|
||||||
|
char *battery_os_tooltip();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
44
src/battery/dummy.c
Normal file
44
src/battery/dummy.c
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* Tint2 : Dummy battery (non-functional)
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 Sebastian Reichel <sre@ring0.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2
|
||||||
|
* or any later version 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 <string.h>
|
||||||
|
#include "common.h"
|
||||||
|
#include "battery.h"
|
||||||
|
|
||||||
|
#warning tint2 has no battery support for this operating system!
|
||||||
|
|
||||||
|
gboolean battery_os_init()
|
||||||
|
{
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_os_free()
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int battery_os_update(BatteryState *state)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *battery_os_tooltip()
|
||||||
|
{
|
||||||
|
return strdup("Operating System not supported");
|
||||||
|
}
|
||||||
102
src/battery/freebsd.c
Normal file
102
src/battery/freebsd.c
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* Tint2 : FreeBSD battery
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2
|
||||||
|
* or any later version 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.
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__) || defined(__DragonFly__)
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/sysctl.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "battery.h"
|
||||||
|
|
||||||
|
gboolean battery_os_init()
|
||||||
|
{
|
||||||
|
int sysctl_out = 0;
|
||||||
|
size_t len = sizeof(sysctl_out);
|
||||||
|
|
||||||
|
return (sysctlbyname("hw.acpi.battery.state", &sysctl_out, &len, NULL, 0) == 0) ||
|
||||||
|
(sysctlbyname("hw.acpi.battery.time", &sysctl_out, &len, NULL, 0) == 0) ||
|
||||||
|
(sysctlbyname("hw.acpi.battery.life", &sysctl_out, &len, NULL, 0) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_os_free()
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int battery_os_update(BatteryState *state)
|
||||||
|
{
|
||||||
|
int sysctl_out = 0;
|
||||||
|
size_t len = sizeof(sysctl_out);
|
||||||
|
gboolean err = 0;
|
||||||
|
|
||||||
|
if (sysctlbyname("hw.acpi.battery.state", &sysctl_out, &len, NULL, 0) == 0) {
|
||||||
|
switch (sysctl_out) {
|
||||||
|
case 1:
|
||||||
|
state->state = BATTERY_DISCHARGING;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
state->state = BATTERY_CHARGING;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
state->state = BATTERY_FULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "power update: no such sysctl");
|
||||||
|
err = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sysctlbyname("hw.acpi.battery.time", &sysctl_out, &len, NULL, 0) == 0)
|
||||||
|
battery_state_set_time(state, sysctl_out * 60);
|
||||||
|
else
|
||||||
|
err = -1;
|
||||||
|
|
||||||
|
if (sysctlbyname("hw.acpi.battery.life", &sysctl_out, &len, NULL, 0) == 0)
|
||||||
|
state->percentage = sysctl_out;
|
||||||
|
else
|
||||||
|
err = -1;
|
||||||
|
|
||||||
|
if (sysctlbyname("hw.acpi.acline", &sysctl_out, &len, NULL, 0) == 0)
|
||||||
|
state->ac_connected = sysctl_out;
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *battery_os_tooltip()
|
||||||
|
{
|
||||||
|
GString *tooltip = g_string_new("");
|
||||||
|
gchar *result;
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "Battery\n");
|
||||||
|
|
||||||
|
gchar *state = (battery_state.state == BATTERY_UNKNOWN) ? "Level" : chargestate2str(battery_state.state);
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "\t%s: %d%%", state, battery_state.percentage);
|
||||||
|
|
||||||
|
g_string_append_c(tooltip, '\n');
|
||||||
|
g_string_append_printf(tooltip, "AC\n");
|
||||||
|
g_string_append_printf(tooltip, battery_state.ac_connected ? "\tConnected" : "\tDisconnected");
|
||||||
|
|
||||||
|
result = tooltip->str;
|
||||||
|
g_string_free(tooltip, FALSE);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
539
src/battery/linux.c
Normal file
539
src/battery/linux.c
Normal file
@@ -0,0 +1,539 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* Tint2 : Linux battery
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 Sebastian Reichel <sre@ring0.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2
|
||||||
|
* or any later version 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.
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifdef __linux__
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "battery.h"
|
||||||
|
#include "uevent.h"
|
||||||
|
|
||||||
|
enum psy_type {
|
||||||
|
PSY_UNKNOWN,
|
||||||
|
PSY_BATTERY,
|
||||||
|
PSY_MAINS,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct psy_battery {
|
||||||
|
/* generic properties */
|
||||||
|
gchar *name;
|
||||||
|
/* monotonic time, in microseconds */
|
||||||
|
gint64 timestamp;
|
||||||
|
/* sysfs files */
|
||||||
|
gchar *path_present;
|
||||||
|
gchar *path_level_now;
|
||||||
|
gchar *path_level_full;
|
||||||
|
gchar *path_rate_now;
|
||||||
|
gchar *path_status;
|
||||||
|
/* values */
|
||||||
|
gboolean present;
|
||||||
|
gint level_now;
|
||||||
|
gint level_full;
|
||||||
|
gint rate_now;
|
||||||
|
gchar unit;
|
||||||
|
ChargeState status;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct psy_mains {
|
||||||
|
/* generic properties */
|
||||||
|
gchar *name;
|
||||||
|
/* sysfs files */
|
||||||
|
gchar *path_online;
|
||||||
|
/* values */
|
||||||
|
gboolean online;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void uevent_battery_update()
|
||||||
|
{
|
||||||
|
update_battery_tick(NULL);
|
||||||
|
}
|
||||||
|
static struct uevent_notify psy_change = {UEVENT_CHANGE, "power_supply", NULL, uevent_battery_update};
|
||||||
|
|
||||||
|
static void uevent_battery_plug()
|
||||||
|
{
|
||||||
|
printf("reinitialize batteries after HW change\n");
|
||||||
|
reinit_battery();
|
||||||
|
}
|
||||||
|
static struct uevent_notify psy_plug = {UEVENT_ADD | UEVENT_REMOVE, "power_supply", NULL, uevent_battery_plug};
|
||||||
|
|
||||||
|
#define RETURN_ON_ERROR(err) \
|
||||||
|
if (err) { \
|
||||||
|
g_error_free(err); \
|
||||||
|
fprintf(stderr, RED "%s:%d: errror" RESET "\n", __FILE__, __LINE__); \
|
||||||
|
return FALSE; \
|
||||||
|
}
|
||||||
|
|
||||||
|
static GList *batteries = NULL;
|
||||||
|
static GList *mains = NULL;
|
||||||
|
|
||||||
|
static guint8 level_to_percent(gint level_now, gint level_full)
|
||||||
|
{
|
||||||
|
return 0.5 + ((level_now <= level_full ? level_now : level_full) * 100.0) / level_full;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum psy_type power_supply_get_type(const gchar *entryname)
|
||||||
|
{
|
||||||
|
gchar *path_type = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "type", NULL);
|
||||||
|
GError *error = NULL;
|
||||||
|
gchar *type;
|
||||||
|
gsize typelen;
|
||||||
|
|
||||||
|
g_file_get_contents(path_type, &type, &typelen, &error);
|
||||||
|
g_free(path_type);
|
||||||
|
if (error) {
|
||||||
|
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__);
|
||||||
|
g_error_free(error);
|
||||||
|
return PSY_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!g_strcmp0(type, "Battery\n")) {
|
||||||
|
g_free(type);
|
||||||
|
return PSY_BATTERY;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!g_strcmp0(type, "Mains\n")) {
|
||||||
|
g_free(type);
|
||||||
|
return PSY_MAINS;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_free(type);
|
||||||
|
|
||||||
|
return PSY_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean init_linux_battery(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
const gchar *entryname = bat->name;
|
||||||
|
|
||||||
|
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)) {
|
||||||
|
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__);
|
||||||
|
goto err0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bat->path_level_now =
|
||||||
|
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "energy_now", NULL);
|
||||||
|
bat->path_level_full =
|
||||||
|
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->unit = 'W';
|
||||||
|
|
||||||
|
if (!g_file_test(bat->path_level_now, G_FILE_TEST_EXISTS) ||
|
||||||
|
!g_file_test(bat->path_level_full, G_FILE_TEST_EXISTS) ||
|
||||||
|
!g_file_test(bat->path_rate_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_free(bat->path_level_now);
|
||||||
|
g_free(bat->path_level_full);
|
||||||
|
g_free(bat->path_rate_now);
|
||||||
|
bat->path_level_now =
|
||||||
|
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "charge_now", NULL);
|
||||||
|
bat->path_level_full =
|
||||||
|
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "charge_full", NULL);
|
||||||
|
bat->path_rate_now =
|
||||||
|
g_build_filename(battery_sys_prefix, "/sys/class/power_supply", entryname, "current_now", NULL);
|
||||||
|
bat->unit = 'A';
|
||||||
|
}
|
||||||
|
if (!g_file_test(bat->path_level_now, G_FILE_TEST_EXISTS) ||
|
||||||
|
!g_file_test(bat->path_level_full, G_FILE_TEST_EXISTS) ||
|
||||||
|
!g_file_test(bat->path_rate_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__);
|
||||||
|
goto err1;
|
||||||
|
}
|
||||||
|
|
||||||
|
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)) {
|
||||||
|
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__);
|
||||||
|
goto err2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
err2:
|
||||||
|
g_free(bat->path_status);
|
||||||
|
err1:
|
||||||
|
g_free(bat->path_level_now);
|
||||||
|
g_free(bat->path_level_full);
|
||||||
|
g_free(bat->path_rate_now);
|
||||||
|
err0:
|
||||||
|
g_free(bat->path_present);
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean init_linux_mains(struct psy_mains *ac)
|
||||||
|
{
|
||||||
|
const gchar *entryname = ac->name;
|
||||||
|
|
||||||
|
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)) {
|
||||||
|
fprintf(stderr, RED "%s:%d: read failed" RESET "\n", __FILE__, __LINE__);
|
||||||
|
g_free(ac->path_online);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void psy_battery_free(gpointer data)
|
||||||
|
{
|
||||||
|
struct psy_battery *bat = data;
|
||||||
|
g_free(bat->name);
|
||||||
|
g_free(bat->path_status);
|
||||||
|
g_free(bat->path_rate_now);
|
||||||
|
g_free(bat->path_level_full);
|
||||||
|
g_free(bat->path_level_now);
|
||||||
|
g_free(bat->path_present);
|
||||||
|
g_free(bat);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void psy_mains_free(gpointer data)
|
||||||
|
{
|
||||||
|
struct psy_mains *ac = data;
|
||||||
|
g_free(ac->name);
|
||||||
|
g_free(ac->path_online);
|
||||||
|
g_free(ac);
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_os_free()
|
||||||
|
{
|
||||||
|
uevent_unregister_notifier(&psy_change);
|
||||||
|
uevent_unregister_notifier(&psy_plug);
|
||||||
|
|
||||||
|
g_list_free_full(batteries, psy_battery_free);
|
||||||
|
batteries = NULL;
|
||||||
|
g_list_free_full(mains, psy_mains_free);
|
||||||
|
mains = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void add_battery(const char *entryname)
|
||||||
|
{
|
||||||
|
struct psy_battery *bat = g_malloc0(sizeof(*bat));
|
||||||
|
bat->name = g_strdup(entryname);
|
||||||
|
|
||||||
|
if (init_linux_battery(bat)) {
|
||||||
|
batteries = g_list_append(batteries, bat);
|
||||||
|
fprintf(stdout, GREEN "Found battery \"%s\"" RESET "\n", bat->name);
|
||||||
|
} else {
|
||||||
|
g_free(bat);
|
||||||
|
fprintf(stderr, RED "Failed to initialize battery \"%s\"" RESET "\n", entryname);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void add_mains(const char *entryname)
|
||||||
|
{
|
||||||
|
struct psy_mains *ac = g_malloc0(sizeof(*ac));
|
||||||
|
ac->name = g_strdup(entryname);
|
||||||
|
|
||||||
|
if (init_linux_mains(ac)) {
|
||||||
|
mains = g_list_append(mains, ac);
|
||||||
|
fprintf(stdout, GREEN "Found mains \"%s\"" RESET "\n", ac->name);
|
||||||
|
} else {
|
||||||
|
g_free(ac);
|
||||||
|
fprintf(stderr, RED "Failed to initialize mains \"%s\"" RESET "\n", entryname);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean battery_os_init()
|
||||||
|
{
|
||||||
|
GDir *directory = 0;
|
||||||
|
GError *error = NULL;
|
||||||
|
const char *entryname;
|
||||||
|
|
||||||
|
battery_os_free();
|
||||||
|
|
||||||
|
gchar *dir_path = g_build_filename(battery_sys_prefix, "/sys/class/power_supply", NULL);
|
||||||
|
directory = g_dir_open(dir_path, 0, &error);
|
||||||
|
g_free(dir_path);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
|
||||||
|
while ((entryname = g_dir_read_name(directory))) {
|
||||||
|
fprintf(stderr, GREEN "Found power device %s" RESET "\n", entryname);
|
||||||
|
enum psy_type type = power_supply_get_type(entryname);
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case PSY_BATTERY:
|
||||||
|
add_battery(entryname);
|
||||||
|
break;
|
||||||
|
case PSY_MAINS:
|
||||||
|
add_mains(entryname);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
g_dir_close(directory);
|
||||||
|
|
||||||
|
uevent_register_notifier(&psy_change);
|
||||||
|
uevent_register_notifier(&psy_plug);
|
||||||
|
|
||||||
|
return batteries != NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gint estimate_rate_usage(struct psy_battery *bat, gint old_level_now, gint64 old_timestamp)
|
||||||
|
{
|
||||||
|
gint64 diff_level = ABS(bat->level_now - old_level_now);
|
||||||
|
gint64 diff_time = bat->timestamp - old_timestamp;
|
||||||
|
|
||||||
|
/* µW = (µWh * 3600) / (µs / 1000000) */
|
||||||
|
gint rate = diff_level * 3600 * 1000000 / MAX(1, diff_time);
|
||||||
|
|
||||||
|
return rate;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean update_linux_battery(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
GError *error = NULL;
|
||||||
|
gchar *data;
|
||||||
|
gsize datalen;
|
||||||
|
|
||||||
|
gint64 old_timestamp = bat->timestamp;
|
||||||
|
int old_level_now = bat->level_now;
|
||||||
|
gint old_rate_now = bat->rate_now;
|
||||||
|
|
||||||
|
/* reset values */
|
||||||
|
bat->present = 0;
|
||||||
|
bat->status = BATTERY_UNKNOWN;
|
||||||
|
bat->level_now = 0;
|
||||||
|
bat->level_full = 0;
|
||||||
|
bat->rate_now = 0;
|
||||||
|
bat->timestamp = g_get_monotonic_time();
|
||||||
|
|
||||||
|
/* present */
|
||||||
|
g_file_get_contents(bat->path_present, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
bat->present = (atoi(data) == 1);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* we are done, if battery is not present */
|
||||||
|
if (!bat->present)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
/* status */
|
||||||
|
bat->status = BATTERY_UNKNOWN;
|
||||||
|
g_file_get_contents(bat->path_status, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
if (!g_strcmp0(data, "Charging\n")) {
|
||||||
|
bat->status = BATTERY_CHARGING;
|
||||||
|
} else if (!g_strcmp0(data, "Discharging\n")) {
|
||||||
|
bat->status = BATTERY_DISCHARGING;
|
||||||
|
} else if (!g_strcmp0(data, "Full\n")) {
|
||||||
|
bat->status = BATTERY_FULL;
|
||||||
|
}
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* level now */
|
||||||
|
g_file_get_contents(bat->path_level_now, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
bat->level_now = atoi(data);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* level full */
|
||||||
|
g_file_get_contents(bat->path_level_full, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
bat->level_full = atoi(data);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* rate now */
|
||||||
|
g_file_get_contents(bat->path_rate_now, &data, &datalen, &error);
|
||||||
|
if (g_error_matches(error, G_FILE_ERROR, G_FILE_ERROR_NODEV)) {
|
||||||
|
/* some hardware does not support reading current rate consumption */
|
||||||
|
g_error_free(error);
|
||||||
|
bat->rate_now = estimate_rate_usage(bat, old_level_now, old_timestamp);
|
||||||
|
if (bat->rate_now == 0 && bat->status != BATTERY_FULL) {
|
||||||
|
/* If the hardware updates the level slower than our sampling period,
|
||||||
|
* we need to sample more rarely */
|
||||||
|
bat->rate_now = old_rate_now;
|
||||||
|
bat->timestamp = old_timestamp;
|
||||||
|
}
|
||||||
|
} else if (error) {
|
||||||
|
g_error_free(error);
|
||||||
|
return FALSE;
|
||||||
|
} else {
|
||||||
|
bat->rate_now = atoi(data);
|
||||||
|
g_free(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean update_linux_mains(struct psy_mains *ac)
|
||||||
|
{
|
||||||
|
GError *error = NULL;
|
||||||
|
gchar *data;
|
||||||
|
gsize datalen;
|
||||||
|
ac->online = FALSE;
|
||||||
|
|
||||||
|
/* online */
|
||||||
|
g_file_get_contents(ac->path_online, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
ac->online = (atoi(data) == 1);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
int battery_os_update(BatteryState *state)
|
||||||
|
{
|
||||||
|
GList *l;
|
||||||
|
|
||||||
|
gint64 total_level_now = 0;
|
||||||
|
gint64 total_level_full = 0;
|
||||||
|
gint64 total_rate_now = 0;
|
||||||
|
gint seconds = 0;
|
||||||
|
|
||||||
|
gboolean charging = FALSE;
|
||||||
|
gboolean discharging = FALSE;
|
||||||
|
gboolean full = FALSE;
|
||||||
|
gboolean ac_connected = FALSE;
|
||||||
|
|
||||||
|
for (l = batteries; l != NULL; l = l->next) {
|
||||||
|
struct psy_battery *bat = l->data;
|
||||||
|
update_linux_battery(bat);
|
||||||
|
|
||||||
|
total_level_now += bat->level_now;
|
||||||
|
total_level_full += bat->level_full;
|
||||||
|
total_rate_now += bat->rate_now;
|
||||||
|
|
||||||
|
charging |= (bat->status == BATTERY_CHARGING);
|
||||||
|
discharging |= (bat->status == BATTERY_DISCHARGING);
|
||||||
|
full |= (bat->status == BATTERY_FULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (l = mains; l != NULL; l = l->next) {
|
||||||
|
struct psy_mains *ac = l->data;
|
||||||
|
update_linux_mains(ac);
|
||||||
|
ac_connected |= (ac->online);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* build global state */
|
||||||
|
if (charging && !discharging)
|
||||||
|
state->state = BATTERY_CHARGING;
|
||||||
|
else if (!charging && discharging)
|
||||||
|
state->state = BATTERY_DISCHARGING;
|
||||||
|
else if (!charging && !discharging && full)
|
||||||
|
state->state = BATTERY_FULL;
|
||||||
|
|
||||||
|
/* calculate seconds */
|
||||||
|
if (total_rate_now > 0) {
|
||||||
|
if (state->state == BATTERY_CHARGING)
|
||||||
|
seconds = 3600 * (total_level_full - total_level_now) / total_rate_now;
|
||||||
|
else if (state->state == BATTERY_DISCHARGING)
|
||||||
|
seconds = 3600 * total_level_now / total_rate_now;
|
||||||
|
seconds = MAX(0, seconds);
|
||||||
|
}
|
||||||
|
battery_state_set_time(state, seconds);
|
||||||
|
|
||||||
|
/* calculate percentage */
|
||||||
|
state->percentage = level_to_percent(total_level_now, total_level_full);
|
||||||
|
|
||||||
|
/* AC state */
|
||||||
|
state->ac_connected = ac_connected;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gchar *level_human_readable(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
gint now = bat->level_now;
|
||||||
|
gint full = bat->level_full;
|
||||||
|
|
||||||
|
if (full >= 1000000) {
|
||||||
|
return g_strdup_printf("%d.%d / %d.%d %ch",
|
||||||
|
now / 1000000,
|
||||||
|
(now % 1000000) / 100000,
|
||||||
|
full / 1000000,
|
||||||
|
(full % 1000000) / 100000,
|
||||||
|
bat->unit);
|
||||||
|
} else if (full >= 1000) {
|
||||||
|
return g_strdup_printf("%d.%d / %d.%d m%ch",
|
||||||
|
now / 1000,
|
||||||
|
(now % 1000) / 100,
|
||||||
|
full / 1000,
|
||||||
|
(full % 1000) / 100,
|
||||||
|
bat->unit);
|
||||||
|
} else {
|
||||||
|
return g_strdup_printf("%d / %d µ%ch", now, full, bat->unit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static gchar *rate_human_readable(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
gint rate = bat->rate_now;
|
||||||
|
gchar unit = bat->unit;
|
||||||
|
|
||||||
|
if (rate >= 1000000) {
|
||||||
|
return g_strdup_printf("%d.%d %c", rate / 1000000, (rate % 1000000) / 100000, unit);
|
||||||
|
} else if (rate >= 1000) {
|
||||||
|
return g_strdup_printf("%d.%d m%c", rate / 1000, (rate % 1000) / 100, unit);
|
||||||
|
} else if (rate > 0) {
|
||||||
|
return g_strdup_printf("%d µ%c", rate, unit);
|
||||||
|
} else {
|
||||||
|
return g_strdup_printf("0 %c", unit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char *battery_os_tooltip()
|
||||||
|
{
|
||||||
|
GList *l;
|
||||||
|
GString *tooltip = g_string_new("");
|
||||||
|
gchar *result;
|
||||||
|
|
||||||
|
for (l = batteries; l != NULL; l = l->next) {
|
||||||
|
struct psy_battery *bat = l->data;
|
||||||
|
|
||||||
|
if (tooltip->len)
|
||||||
|
g_string_append_c(tooltip, '\n');
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "%s\n", bat->name);
|
||||||
|
|
||||||
|
if (!bat->present) {
|
||||||
|
g_string_append_printf(tooltip, "\tnot connected");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
gchar *rate = rate_human_readable(bat);
|
||||||
|
gchar *level = level_human_readable(bat);
|
||||||
|
gchar *state = (bat->status == BATTERY_UNKNOWN) ? "energy" : chargestate2str(bat->status);
|
||||||
|
|
||||||
|
guint8 percentage = level_to_percent(bat->level_now, bat->level_full);
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "\t%s: %s (%u %%)\n\trate: %s", state, level, percentage, rate);
|
||||||
|
|
||||||
|
g_free(rate);
|
||||||
|
g_free(level);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (l = mains; l != NULL; l = l->next) {
|
||||||
|
struct psy_mains *ac = l->data;
|
||||||
|
|
||||||
|
if (tooltip->len)
|
||||||
|
g_string_append_c(tooltip, '\n');
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "%s\n", ac->name);
|
||||||
|
g_string_append_printf(tooltip, ac->online ? "\tConnected" : "\tDisconnected");
|
||||||
|
}
|
||||||
|
|
||||||
|
result = tooltip->str;
|
||||||
|
g_string_free(tooltip, FALSE);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
108
src/battery/openbsd.c
Normal file
108
src/battery/openbsd.c
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* Tint2 : OpenBSD & NetBSD battery
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2
|
||||||
|
* or any later version 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.
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <err.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <machine/apmvar.h>
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "battery.h"
|
||||||
|
|
||||||
|
int apm_fd = -1;
|
||||||
|
|
||||||
|
gboolean battery_os_init()
|
||||||
|
{
|
||||||
|
if (apm_fd > 0)
|
||||||
|
close(apm_fd);
|
||||||
|
|
||||||
|
apm_fd = open("/dev/apm", O_RDONLY);
|
||||||
|
|
||||||
|
if (apm_fd < 0) {
|
||||||
|
warn("ERROR: battery applet cannot open /dev/apm.");
|
||||||
|
return FALSE;
|
||||||
|
} else {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_os_free()
|
||||||
|
{
|
||||||
|
if ((apm_fd != -1) && (close(apm_fd) == -1))
|
||||||
|
warn("cannot close /dev/apm");
|
||||||
|
apm_fd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int battery_os_update(BatteryState *state)
|
||||||
|
{
|
||||||
|
struct apm_power_info info;
|
||||||
|
|
||||||
|
if (apm_fd > 0 && ioctl(apm_fd, APM_IOC_GETPOWER, &(info)) == 0) {
|
||||||
|
// best attempt at mapping to Linux battery states
|
||||||
|
switch (info.battery_state) {
|
||||||
|
case APM_BATT_CHARGING:
|
||||||
|
state->state = BATTERY_CHARGING;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
state->state = BATTERY_DISCHARGING;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info.battery_life > 100)
|
||||||
|
info.battery_life = 100;
|
||||||
|
if (info.battery_life == 100)
|
||||||
|
state->state = BATTERY_FULL;
|
||||||
|
|
||||||
|
state->percentage = info.battery_life;
|
||||||
|
if (info.minutes_left != -1)
|
||||||
|
battery_state_set_time(state, info.minutes_left * 60);
|
||||||
|
|
||||||
|
state->ac_connected = info.ac_state == APM_AC_ON;
|
||||||
|
} else {
|
||||||
|
warn("power update: APM_IOC_GETPOWER");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *battery_os_tooltip()
|
||||||
|
{
|
||||||
|
GString *tooltip = g_string_new("");
|
||||||
|
gchar *result;
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "Battery\n");
|
||||||
|
|
||||||
|
gchar *state = (battery_state.state == BATTERY_UNKNOWN) ? "Level" : chargestate2str(battery_state.state);
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "\t%s: %d%%", state, battery_state.percentage);
|
||||||
|
|
||||||
|
g_string_append_c(tooltip, '\n');
|
||||||
|
g_string_append_printf(tooltip, "AC\n");
|
||||||
|
g_string_append_printf(tooltip, battery_state.ac_connected ? "\tConnected" : "\tDisconnected");
|
||||||
|
|
||||||
|
result = tooltip->str;
|
||||||
|
g_string_free(tooltip, FALSE);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
641
src/button/button.c
Normal file
641
src/button/button.c
Normal file
@@ -0,0 +1,641 @@
|
|||||||
|
#include "button.h"
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <cairo.h>
|
||||||
|
#include <cairo-xlib.h>
|
||||||
|
#include <math.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
|
||||||
|
#include "window.h"
|
||||||
|
#include "server.h"
|
||||||
|
#include "panel.h"
|
||||||
|
#include "timer.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
char *button_get_tooltip(void *obj);
|
||||||
|
void button_init_fonts();
|
||||||
|
int button_compute_desired_size(void *obj);
|
||||||
|
void button_dump_geometry(void *obj, int indent);
|
||||||
|
|
||||||
|
void default_button()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Button *create_button()
|
||||||
|
{
|
||||||
|
Button *button = calloc(1, sizeof(Button));
|
||||||
|
button->backend = calloc(1, sizeof(ButtonBackend));
|
||||||
|
button->backend->centered = TRUE;
|
||||||
|
button->backend->font_color.alpha = 0.5;
|
||||||
|
return button;
|
||||||
|
}
|
||||||
|
|
||||||
|
gpointer create_button_frontend(gconstpointer arg, gpointer data)
|
||||||
|
{
|
||||||
|
Button *button_backend = (Button *)arg;
|
||||||
|
|
||||||
|
Button *button_frontend = calloc(1, sizeof(Button));
|
||||||
|
button_frontend->backend = button_backend->backend;
|
||||||
|
button_backend->backend->instances = g_list_append(button_backend->backend->instances, button_frontend);
|
||||||
|
button_frontend->frontend = calloc(1, sizeof(ButtonFrontend));
|
||||||
|
return button_frontend;
|
||||||
|
}
|
||||||
|
|
||||||
|
void destroy_button(void *obj)
|
||||||
|
{
|
||||||
|
Button *button = (Button *)obj;
|
||||||
|
if (button->frontend) {
|
||||||
|
// This is a frontend element
|
||||||
|
if (button->frontend->icon) {
|
||||||
|
imlib_context_set_image(button->frontend->icon);
|
||||||
|
imlib_free_image();
|
||||||
|
button->frontend->icon = NULL;
|
||||||
|
}
|
||||||
|
button->backend->instances = g_list_remove_all(button->backend->instances, button);
|
||||||
|
free_and_null(button->frontend);
|
||||||
|
remove_area(&button->area);
|
||||||
|
free_area(&button->area);
|
||||||
|
free_and_null(button);
|
||||||
|
} else {
|
||||||
|
// This is a backend element
|
||||||
|
free_and_null(button->backend->text);
|
||||||
|
free_and_null(button->backend->icon_name);
|
||||||
|
free_and_null(button->backend->tooltip);
|
||||||
|
|
||||||
|
button->backend->bg = NULL;
|
||||||
|
pango_font_description_free(button->backend->font_desc);
|
||||||
|
button->backend->font_desc = NULL;
|
||||||
|
free_and_null(button->backend->lclick_command);
|
||||||
|
free_and_null(button->backend->mclick_command);
|
||||||
|
free_and_null(button->backend->rclick_command);
|
||||||
|
free_and_null(button->backend->dwheel_command);
|
||||||
|
free_and_null(button->backend->uwheel_command);
|
||||||
|
|
||||||
|
if (button->backend->instances) {
|
||||||
|
fprintf(stderr, "Error: Attempt to destroy backend while there are still frontend instances!\n");
|
||||||
|
exit(-1);
|
||||||
|
}
|
||||||
|
free(button->backend);
|
||||||
|
free(button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_button()
|
||||||
|
{
|
||||||
|
GList *to_remove = panel_config.button_list;
|
||||||
|
for (int k = 0; k < strlen(panel_items_order) && to_remove; k++) {
|
||||||
|
if (panel_items_order[k] == 'P') {
|
||||||
|
to_remove = to_remove->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (to_remove) {
|
||||||
|
if (to_remove == panel_config.button_list) {
|
||||||
|
g_list_free_full(to_remove, destroy_button);
|
||||||
|
panel_config.button_list = NULL;
|
||||||
|
} else {
|
||||||
|
// Cut panel_config.button_list
|
||||||
|
if (to_remove->prev)
|
||||||
|
to_remove->prev->next = NULL;
|
||||||
|
to_remove->prev = NULL;
|
||||||
|
// Remove all elements of to_remove and to_remove itself
|
||||||
|
g_list_free_full(to_remove, destroy_button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
button_init_fonts();
|
||||||
|
for (GList *l = panel_config.button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
|
||||||
|
// Set missing config options
|
||||||
|
if (!button->backend->bg)
|
||||||
|
button->backend->bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_button_panel(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
|
||||||
|
// Make sure this is only done once if there are multiple items
|
||||||
|
if (panel->button_list && ((Button *)panel->button_list->data)->frontend)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// panel->button_list is now a copy of the pointer panel_config.button_list
|
||||||
|
// We make it a deep copy
|
||||||
|
panel->button_list = g_list_copy_deep(panel_config.button_list, create_button_frontend, NULL);
|
||||||
|
|
||||||
|
load_icon_themes();
|
||||||
|
|
||||||
|
for (GList *l = panel->button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
button->area.bg = button->backend->bg;
|
||||||
|
button->area.paddingx = button->backend->paddingx;
|
||||||
|
button->area.paddingy = button->backend->paddingy;
|
||||||
|
button->area.paddingxlr = button->backend->paddingxlr;
|
||||||
|
button->area.parent = panel;
|
||||||
|
button->area.panel = panel;
|
||||||
|
button->area._dump_geometry = button_dump_geometry;
|
||||||
|
button->area._compute_desired_size = button_compute_desired_size;
|
||||||
|
snprintf(button->area.name, sizeof(button->area.name), "Button");
|
||||||
|
button->area._draw_foreground = draw_button;
|
||||||
|
button->area.size_mode = LAYOUT_FIXED;
|
||||||
|
button->area._resize = resize_button;
|
||||||
|
button->area._get_tooltip_text = button_get_tooltip;
|
||||||
|
button->area._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
|
button->area.has_mouse_press_effect =
|
||||||
|
panel_config.mouse_effects &&
|
||||||
|
(button->area.has_mouse_over_effect = button->backend->lclick_command || button->backend->mclick_command ||
|
||||||
|
button->backend->rclick_command || button->backend->uwheel_command ||
|
||||||
|
button->backend->dwheel_command);
|
||||||
|
|
||||||
|
button->area.resize_needed = TRUE;
|
||||||
|
button->area.on_screen = TRUE;
|
||||||
|
instantiate_area_gradients(&button->area);
|
||||||
|
|
||||||
|
button_reload_icon(button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_init_fonts()
|
||||||
|
{
|
||||||
|
for (GList *l = panel_config.button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
if (!button->backend->font_desc)
|
||||||
|
button->backend->font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_default_font_changed()
|
||||||
|
{
|
||||||
|
gboolean needs_update = FALSE;
|
||||||
|
for (GList *l = panel_config.button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
|
||||||
|
if (!button->backend->has_font) {
|
||||||
|
pango_font_description_free(button->backend->font_desc);
|
||||||
|
button->backend->font_desc = NULL;
|
||||||
|
needs_update = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!needs_update)
|
||||||
|
return;
|
||||||
|
|
||||||
|
button_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
for (GList *l = panels[i].button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
|
||||||
|
if (!button->backend->has_font) {
|
||||||
|
button->area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&button->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_reload_icon(Button *button)
|
||||||
|
{
|
||||||
|
free_icon(button->frontend->icon);
|
||||||
|
free_icon(button->frontend->icon_hover);
|
||||||
|
free_icon(button->frontend->icon_pressed);
|
||||||
|
button->frontend->icon = NULL;
|
||||||
|
button->frontend->icon_hover = NULL;
|
||||||
|
button->frontend->icon_pressed = NULL;
|
||||||
|
|
||||||
|
button->frontend->icon_load_size = button->frontend->iconw;
|
||||||
|
|
||||||
|
if (!button->backend->icon_name)
|
||||||
|
return;
|
||||||
|
|
||||||
|
char *new_icon_path = get_icon_path(icon_theme_wrapper, button->backend->icon_name, button->frontend->iconw, TRUE);
|
||||||
|
if (new_icon_path)
|
||||||
|
button->frontend->icon = load_image(new_icon_path, TRUE);
|
||||||
|
free(new_icon_path);
|
||||||
|
// On loading error, fallback to default
|
||||||
|
if (!button->frontend->icon) {
|
||||||
|
new_icon_path = get_icon_path(icon_theme_wrapper, DEFAULT_ICON, button->frontend->iconw, TRUE);
|
||||||
|
if (new_icon_path)
|
||||||
|
button->frontend->icon = load_image(new_icon_path, TRUE);
|
||||||
|
free(new_icon_path);
|
||||||
|
}
|
||||||
|
Imlib_Image original = button->frontend->icon;
|
||||||
|
button->frontend->icon = scale_icon(button->frontend->icon, button->frontend->iconw);
|
||||||
|
free_icon(original);
|
||||||
|
|
||||||
|
if (panel_config.mouse_effects) {
|
||||||
|
button->frontend->icon_hover = adjust_icon(button->frontend->icon,
|
||||||
|
panel_config.mouse_over_alpha,
|
||||||
|
panel_config.mouse_over_saturation,
|
||||||
|
panel_config.mouse_over_brightness);
|
||||||
|
button->frontend->icon_pressed = adjust_icon(button->frontend->icon,
|
||||||
|
panel_config.mouse_pressed_alpha,
|
||||||
|
panel_config.mouse_pressed_saturation,
|
||||||
|
panel_config.mouse_pressed_brightness);
|
||||||
|
}
|
||||||
|
schedule_redraw(&button->area);
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_default_icon_theme_changed()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
for (GList *l = panels[i].button_list; l; l = l->next) {
|
||||||
|
Button *button = l->data;
|
||||||
|
button_reload_icon(button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup_button()
|
||||||
|
{
|
||||||
|
// Cleanup frontends
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
g_list_free_full(panels[i].button_list, destroy_button);
|
||||||
|
panels[i].button_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup backends
|
||||||
|
g_list_free_full(panel_config.button_list, destroy_button);
|
||||||
|
panel_config.button_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int button_compute_desired_size(void *obj)
|
||||||
|
{
|
||||||
|
Button *button = (Button *)obj;
|
||||||
|
Panel *panel = (Panel *)button->area.panel;
|
||||||
|
int horiz_padding = (panel_horizontal ? button->area.paddingxlr : button->area.paddingy);
|
||||||
|
int vert_padding = (panel_horizontal ? button->area.paddingy : button->area.paddingxlr);
|
||||||
|
int interior_padding = button->area.paddingx;
|
||||||
|
|
||||||
|
int icon_w, icon_h;
|
||||||
|
if (button->backend->icon_name) {
|
||||||
|
if (panel_horizontal)
|
||||||
|
icon_h = icon_w = button->area.height - top_bottom_border_width(&button->area) - 2 * vert_padding;
|
||||||
|
else
|
||||||
|
icon_h = icon_w = button->area.width - left_right_border_width(&button->area) - 2 * horiz_padding;
|
||||||
|
if (button->backend->max_icon_size) {
|
||||||
|
icon_w = MIN(icon_w, button->backend->max_icon_size);
|
||||||
|
icon_h = MIN(icon_h, button->backend->max_icon_size);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
icon_h = icon_w = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int txt_height_ink, txt_height, txt_width;
|
||||||
|
if (button->backend->text) {
|
||||||
|
if (panel_horizontal) {
|
||||||
|
get_text_size2(button->backend->font_desc,
|
||||||
|
&txt_height_ink,
|
||||||
|
&txt_height,
|
||||||
|
&txt_width,
|
||||||
|
panel->area.height,
|
||||||
|
panel->area.width,
|
||||||
|
button->backend->text,
|
||||||
|
strlen(button->backend->text),
|
||||||
|
PANGO_WRAP_WORD_CHAR,
|
||||||
|
PANGO_ELLIPSIZE_NONE,
|
||||||
|
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 {
|
||||||
|
txt_height_ink = txt_height = txt_width = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
int new_size = txt_width + icon_w + (txt_width && icon_w ? interior_padding : 0);
|
||||||
|
new_size += 2 * horiz_padding + left_right_border_width(&button->area);
|
||||||
|
return new_size;
|
||||||
|
} else {
|
||||||
|
int new_size;
|
||||||
|
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&button->area);
|
||||||
|
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&button->area));
|
||||||
|
return new_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean resize_button(void *obj)
|
||||||
|
{
|
||||||
|
Button *button = (Button *)obj;
|
||||||
|
Panel *panel = (Panel *)button->area.panel;
|
||||||
|
int horiz_padding = (panel_horizontal ? button->area.paddingxlr : button->area.paddingy);
|
||||||
|
int vert_padding = (panel_horizontal ? button->area.paddingy : button->area.paddingxlr);
|
||||||
|
int interior_padding = button->area.paddingx;
|
||||||
|
|
||||||
|
int icon_w, icon_h;
|
||||||
|
if (button->backend->icon_name) {
|
||||||
|
if (panel_horizontal)
|
||||||
|
icon_h = icon_w = button->area.height - top_bottom_border_width(&button->area) - 2 * vert_padding;
|
||||||
|
else
|
||||||
|
icon_h = icon_w = button->area.width - left_right_border_width(&button->area) - 2 * horiz_padding;
|
||||||
|
if (button->backend->max_icon_size) {
|
||||||
|
icon_w = MIN(icon_w, button->backend->max_icon_size);
|
||||||
|
icon_h = MIN(icon_h, button->backend->max_icon_size);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
icon_h = icon_w = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
button->frontend->iconw = icon_w;
|
||||||
|
button->frontend->iconh = icon_h;
|
||||||
|
if (button->frontend->icon_load_size != button->frontend->iconw)
|
||||||
|
button_reload_icon(button);
|
||||||
|
|
||||||
|
int txt_height_ink, txt_height, txt_width;
|
||||||
|
if (button->backend->text) {
|
||||||
|
if (panel_horizontal) {
|
||||||
|
get_text_size2(button->backend->font_desc,
|
||||||
|
&txt_height_ink,
|
||||||
|
&txt_height,
|
||||||
|
&txt_width,
|
||||||
|
panel->area.height,
|
||||||
|
panel->area.width,
|
||||||
|
button->backend->text,
|
||||||
|
strlen(button->backend->text),
|
||||||
|
PANGO_WRAP_WORD_CHAR,
|
||||||
|
PANGO_ELLIPSIZE_NONE,
|
||||||
|
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 {
|
||||||
|
txt_height_ink = txt_height = txt_width = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean result = FALSE;
|
||||||
|
if (panel_horizontal) {
|
||||||
|
int new_size = txt_width + icon_w + (txt_width && icon_w ? interior_padding : 0);
|
||||||
|
new_size += 2 * horiz_padding + left_right_border_width(&button->area);
|
||||||
|
if (new_size != button->area.width) {
|
||||||
|
button->area.width = new_size;
|
||||||
|
result = TRUE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
int new_size;
|
||||||
|
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&button->area);
|
||||||
|
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&button->area));
|
||||||
|
if (new_size != button->area.height) {
|
||||||
|
button->area.height = new_size;
|
||||||
|
result = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
button->frontend->textw = txt_width;
|
||||||
|
button->frontend->texth = txt_height;
|
||||||
|
if (button->backend->centered) {
|
||||||
|
if (icon_w) {
|
||||||
|
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->texty = (button->area.height - txt_height) / 2;
|
||||||
|
button->frontend->textx = button->frontend->iconx + icon_w + interior_padding;
|
||||||
|
} else {
|
||||||
|
button->frontend->texty = (button->area.height - txt_height) / 2;
|
||||||
|
button->frontend->textx = (button->area.width - txt_width) / 2;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (icon_w) {
|
||||||
|
button->frontend->icony = (button->area.height - icon_h) / 2;
|
||||||
|
button->frontend->iconx = left_border_width(&button->area) + horiz_padding;
|
||||||
|
button->frontend->texty = (button->area.height - txt_height) / 2;
|
||||||
|
button->frontend->textx = button->frontend->iconx + icon_w + interior_padding;
|
||||||
|
} else {
|
||||||
|
button->frontend->texty = (button->area.height - txt_height) / 2;
|
||||||
|
button->frontend->textx = left_border_width(&button->area) + horiz_padding;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
schedule_redraw(&button->area);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_button(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
Button *button = obj;
|
||||||
|
|
||||||
|
if (button->frontend->icon) {
|
||||||
|
// Render icon
|
||||||
|
Imlib_Image image;
|
||||||
|
if (panel_config.mouse_effects) {
|
||||||
|
if (button->area.mouse_state == MOUSE_OVER)
|
||||||
|
image = button->frontend->icon_hover ? button->frontend->icon_hover : button->frontend->icon;
|
||||||
|
else if (button->area.mouse_state == MOUSE_DOWN)
|
||||||
|
image = button->frontend->icon_pressed ? button->frontend->icon_pressed : button->frontend->icon;
|
||||||
|
else
|
||||||
|
image = button->frontend->icon;
|
||||||
|
} else {
|
||||||
|
image = button->frontend->icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
imlib_context_set_image(image);
|
||||||
|
render_image(button->area.pix, button->frontend->iconx, button->frontend->icony);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Render text
|
||||||
|
if (button->backend->text) {
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
|
||||||
|
pango_layout_set_font_description(layout, button->backend->font_desc);
|
||||||
|
pango_layout_set_width(layout, button->frontend->textw * PANGO_SCALE);
|
||||||
|
pango_layout_set_alignment(layout, button->backend->centered ? PANGO_ALIGN_CENTER : PANGO_ALIGN_LEFT);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
|
||||||
|
pango_layout_set_text(layout, button->backend->text, strlen(button->backend->text));
|
||||||
|
|
||||||
|
pango_cairo_update_layout(c, layout);
|
||||||
|
draw_text(layout,
|
||||||
|
c,
|
||||||
|
button->frontend->textx,
|
||||||
|
button->frontend->texty,
|
||||||
|
&button->backend->font_color,
|
||||||
|
panel_config.font_shadow);
|
||||||
|
|
||||||
|
g_object_unref(layout);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_dump_geometry(void *obj, int indent)
|
||||||
|
{
|
||||||
|
Button *button = obj;
|
||||||
|
|
||||||
|
if (button->frontend->icon) {
|
||||||
|
Imlib_Image tmp = imlib_context_get_image();
|
||||||
|
imlib_context_set_image(button->frontend->icon);
|
||||||
|
fprintf(stderr,
|
||||||
|
"%*sIcon: x = %d, y = %d, w = %d, h = %d\n",
|
||||||
|
indent,
|
||||||
|
"",
|
||||||
|
button->frontend->iconx,
|
||||||
|
button->frontend->icony,
|
||||||
|
imlib_image_get_width(),
|
||||||
|
imlib_image_get_height());
|
||||||
|
if (tmp)
|
||||||
|
imlib_context_set_image(tmp);
|
||||||
|
}
|
||||||
|
fprintf(stderr,
|
||||||
|
"%*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n",
|
||||||
|
indent,
|
||||||
|
"",
|
||||||
|
button->frontend->textx,
|
||||||
|
button->frontend->texty,
|
||||||
|
button->frontend->textw,
|
||||||
|
button->backend->centered ? "center" : "left",
|
||||||
|
button->backend->text);
|
||||||
|
}
|
||||||
|
|
||||||
|
void button_action(void *obj, int mouse_button, int x, int y, Time time)
|
||||||
|
{
|
||||||
|
Button *button = (Button *)obj;
|
||||||
|
Panel *panel = (Panel *)button->area.panel;
|
||||||
|
char *command = NULL;
|
||||||
|
switch (mouse_button) {
|
||||||
|
case 1:
|
||||||
|
command = button->backend->lclick_command;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
command = button->backend->mclick_command;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
command = button->backend->rclick_command;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
command = button->backend->uwheel_command;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
command = button->backend->dwheel_command;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (command) {
|
||||||
|
int aligned_x, aligned_y, aligned_x1, aligned_y1, aligned_x2, aligned_y2;
|
||||||
|
int panel_x1, panel_x2, panel_y1, panel_y2;
|
||||||
|
if (panel_horizontal) {
|
||||||
|
if (area_is_first(button))
|
||||||
|
aligned_x1 = panel->posx;
|
||||||
|
else
|
||||||
|
aligned_x1 = panel->posx + button->area.posx;
|
||||||
|
|
||||||
|
if (area_is_last(button))
|
||||||
|
aligned_x2 = panel->posx + panel->area.width;
|
||||||
|
else
|
||||||
|
aligned_x2 = panel->posx + button->area.posx + button->area.width;
|
||||||
|
|
||||||
|
if (area_is_first(button))
|
||||||
|
aligned_x = aligned_x1;
|
||||||
|
else if (area_is_last(button))
|
||||||
|
aligned_x = aligned_x2;
|
||||||
|
else
|
||||||
|
aligned_x = aligned_x1;
|
||||||
|
|
||||||
|
if (panel_position & BOTTOM)
|
||||||
|
aligned_y = panel->posy;
|
||||||
|
else
|
||||||
|
aligned_y = panel->posy + panel->area.height;
|
||||||
|
|
||||||
|
aligned_y1 = aligned_y2 = aligned_y;
|
||||||
|
|
||||||
|
panel_x1 = panel->posx;
|
||||||
|
panel_x2 = panel->posx + panel->area.width;
|
||||||
|
panel_y1 = panel_y2 = aligned_y;
|
||||||
|
} else {
|
||||||
|
if (area_is_first(button))
|
||||||
|
aligned_y1 = panel->posy;
|
||||||
|
else
|
||||||
|
aligned_y1 = panel->posy + button->area.posy;
|
||||||
|
|
||||||
|
if (area_is_last(button))
|
||||||
|
aligned_y2 = panel->posy + panel->area.height;
|
||||||
|
else
|
||||||
|
aligned_y2 = panel->posy + button->area.posy + button->area.height;
|
||||||
|
|
||||||
|
if (area_is_first(button))
|
||||||
|
aligned_y = aligned_y1;
|
||||||
|
else if (area_is_last(button))
|
||||||
|
aligned_y = aligned_y2;
|
||||||
|
else
|
||||||
|
aligned_y = aligned_y1;
|
||||||
|
|
||||||
|
if (panel_position & RIGHT)
|
||||||
|
aligned_x = panel->posx;
|
||||||
|
else
|
||||||
|
aligned_x = panel->posx + panel->area.width;
|
||||||
|
|
||||||
|
aligned_x1 = aligned_x2 = aligned_x;
|
||||||
|
|
||||||
|
panel_x1 = panel_x2 = aligned_x;
|
||||||
|
panel_y1 = panel->posy;
|
||||||
|
panel_y2 = panel->posy + panel->area.height;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *full_cmd = g_strdup_printf("export TINT2_BUTTON_X=%d;"
|
||||||
|
"export TINT2_BUTTON_Y=%d;"
|
||||||
|
"export TINT2_BUTTON_W=%d;"
|
||||||
|
"export TINT2_BUTTON_H=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_X=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_Y=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_X1=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_Y1=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_X2=%d;"
|
||||||
|
"export TINT2_BUTTON_ALIGNED_Y2=%d;"
|
||||||
|
"export TINT2_BUTTON_PANEL_X1=%d;"
|
||||||
|
"export TINT2_BUTTON_PANEL_Y1=%d;"
|
||||||
|
"export TINT2_BUTTON_PANEL_X2=%d;"
|
||||||
|
"export TINT2_BUTTON_PANEL_Y2=%d;"
|
||||||
|
"%s",
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
button->area.width,
|
||||||
|
button->area.height,
|
||||||
|
aligned_x,
|
||||||
|
aligned_y,
|
||||||
|
aligned_x1,
|
||||||
|
aligned_y1,
|
||||||
|
aligned_x2,
|
||||||
|
aligned_y2,
|
||||||
|
panel_x1,
|
||||||
|
panel_y1,
|
||||||
|
panel_x2,
|
||||||
|
panel_y2,
|
||||||
|
command);
|
||||||
|
tint_exec(full_cmd, NULL, NULL, time);
|
||||||
|
g_free(full_cmd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char *button_get_tooltip(void *obj)
|
||||||
|
{
|
||||||
|
Button *button = obj;
|
||||||
|
|
||||||
|
if (button->backend->tooltip && strlen(button->backend->tooltip) > 0)
|
||||||
|
return strdup(button->backend->tooltip);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
113
src/button/button.h
Normal file
113
src/button/button.h
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
#ifndef BUTTON_H
|
||||||
|
#define BUTTON_H
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
|
||||||
|
#include "area.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "timer.h"
|
||||||
|
|
||||||
|
// Architecture:
|
||||||
|
// Panel panel_config contains an array of Button, each storing all config options and all the state variables.
|
||||||
|
// Only these run commands.
|
||||||
|
//
|
||||||
|
// Tint2 maintains an array of Panels, one for each monitor. Each stores an array of Button which was initially copied
|
||||||
|
// from panel_config. Each works as a frontend to the corresponding Button in panel_config as backend, using the
|
||||||
|
// backend's config and state variables.
|
||||||
|
|
||||||
|
typedef struct ButtonBackend {
|
||||||
|
// Config:
|
||||||
|
char *icon_name;
|
||||||
|
char *text;
|
||||||
|
char *tooltip;
|
||||||
|
gboolean centered;
|
||||||
|
int max_icon_size;
|
||||||
|
gboolean has_font;
|
||||||
|
PangoFontDescription *font_desc;
|
||||||
|
Color font_color;
|
||||||
|
char *lclick_command;
|
||||||
|
char *mclick_command;
|
||||||
|
char *rclick_command;
|
||||||
|
char *uwheel_command;
|
||||||
|
char *dwheel_command;
|
||||||
|
// paddingxlr = horizontal padding left/right
|
||||||
|
// paddingx = horizontal padding between childs
|
||||||
|
int paddingxlr, paddingx, paddingy;
|
||||||
|
Background *bg;
|
||||||
|
|
||||||
|
// List of Button which are frontends for this backend, one for each panel
|
||||||
|
GList *instances;
|
||||||
|
} ButtonBackend;
|
||||||
|
|
||||||
|
typedef struct ButtonFrontend {
|
||||||
|
// Frontend state:
|
||||||
|
Imlib_Image icon;
|
||||||
|
Imlib_Image icon_hover;
|
||||||
|
Imlib_Image icon_pressed;
|
||||||
|
int icon_load_size;
|
||||||
|
int iconx;
|
||||||
|
int icony;
|
||||||
|
int iconw;
|
||||||
|
int iconh;
|
||||||
|
int textx;
|
||||||
|
int texty;
|
||||||
|
int textw;
|
||||||
|
int texth;
|
||||||
|
} ButtonFrontend;
|
||||||
|
|
||||||
|
typedef struct Button {
|
||||||
|
Area area;
|
||||||
|
// All elements have the backend pointer set. However only backend elements have ownership.
|
||||||
|
ButtonBackend *backend;
|
||||||
|
// Set only for frontend Button items.
|
||||||
|
ButtonFrontend *frontend;
|
||||||
|
} Button;
|
||||||
|
|
||||||
|
// 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
|
||||||
|
// in the backend.
|
||||||
|
// Probably does nothing.
|
||||||
|
void default_button();
|
||||||
|
|
||||||
|
// Creates a new Button item with only the backend field set. The state is NOT initialized. The config is initialized to
|
||||||
|
// the default values.
|
||||||
|
// This will be used by the config code to populate its backedn config fields.
|
||||||
|
Button *create_button();
|
||||||
|
|
||||||
|
void destroy_button(void *obj);
|
||||||
|
|
||||||
|
// Called after the config is read and panel_config is populated, but before panels are created.
|
||||||
|
// Initializes the state of the backend items.
|
||||||
|
// panel_config.panel_items is used to determine which backend items are enabled. The others should be destroyed and
|
||||||
|
// removed from panel_config.button_list.
|
||||||
|
void init_button();
|
||||||
|
|
||||||
|
// Called after each on-screen panel is created, with a pointer to the panel.
|
||||||
|
// Initializes the state of the frontend items. Also adds a pointer to it in backend->instances.
|
||||||
|
// At this point the Area has not been added yet to the GUI tree, but it will be added right away.
|
||||||
|
void init_button_panel(void *panel);
|
||||||
|
|
||||||
|
// Called just before the panels are destroyed. Afterwards, tint2 exits or restarts and reads the config again.
|
||||||
|
// Releases all frontends and then all the backends.
|
||||||
|
// The frontend items are not freed by this function, only their members. The items are Areas which are freed in the
|
||||||
|
// GUI element tree cleanup function (remove_area).
|
||||||
|
void cleanup_button();
|
||||||
|
|
||||||
|
// Called on draw, obj = pointer to the front-end Button item.
|
||||||
|
void draw_button(void *obj, cairo_t *c);
|
||||||
|
|
||||||
|
// Called on resize, obj = pointer to the front-end Button item.
|
||||||
|
// Returns 1 if the new size is different than the previous size.
|
||||||
|
gboolean resize_button(void *obj);
|
||||||
|
|
||||||
|
// Called on mouse click event.
|
||||||
|
void button_action(void *obj, int button, int x, int y, Time time);
|
||||||
|
|
||||||
|
void button_default_font_changed();
|
||||||
|
void button_default_icon_theme_changed();
|
||||||
|
|
||||||
|
void button_reload_icon(Button *button);
|
||||||
|
|
||||||
|
#endif // BUTTON_H
|
||||||
@@ -31,7 +31,6 @@
|
|||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
|
|
||||||
char *time1_format;
|
char *time1_format;
|
||||||
char *time1_timezone;
|
char *time1_timezone;
|
||||||
char *time2_format;
|
char *time2_format;
|
||||||
@@ -39,225 +38,380 @@ char *time2_timezone;
|
|||||||
char *time_tooltip_format;
|
char *time_tooltip_format;
|
||||||
char *time_tooltip_timezone;
|
char *time_tooltip_timezone;
|
||||||
char *clock_lclick_command;
|
char *clock_lclick_command;
|
||||||
|
char *clock_mclick_command;
|
||||||
char *clock_rclick_command;
|
char *clock_rclick_command;
|
||||||
|
char *clock_uwheel_command;
|
||||||
|
char *clock_dwheel_command;
|
||||||
struct timeval time_clock;
|
struct timeval time_clock;
|
||||||
|
gboolean time1_has_font;
|
||||||
PangoFontDescription *time1_font_desc;
|
PangoFontDescription *time1_font_desc;
|
||||||
|
gboolean time2_has_font;
|
||||||
PangoFontDescription *time2_font_desc;
|
PangoFontDescription *time2_font_desc;
|
||||||
static char buf_time[256];
|
static char buf_time[256];
|
||||||
static char buf_date[256];
|
static char buf_date[256];
|
||||||
static char buf_tooltip[512];
|
static char buf_tooltip[512];
|
||||||
int clock_enabled;
|
int clock_enabled;
|
||||||
static timeout* clock_timeout;
|
static timeout *clock_timeout;
|
||||||
|
|
||||||
|
void clock_init_fonts();
|
||||||
|
char *clock_get_tooltip(void *obj);
|
||||||
|
int clock_compute_desired_size(void *obj);
|
||||||
|
void clock_dump_geometry(void *obj, int indent);
|
||||||
|
|
||||||
void default_clock()
|
void default_clock()
|
||||||
{
|
{
|
||||||
clock_enabled = 0;
|
clock_enabled = 0;
|
||||||
clock_timeout = 0;
|
clock_timeout = NULL;
|
||||||
time1_format = 0;
|
time1_format = NULL;
|
||||||
time1_timezone = 0;
|
time1_timezone = NULL;
|
||||||
time2_format = 0;
|
time2_format = NULL;
|
||||||
time2_timezone = 0;
|
time2_timezone = NULL;
|
||||||
time_tooltip_format = 0;
|
time_tooltip_format = NULL;
|
||||||
time_tooltip_timezone = 0;
|
time_tooltip_timezone = NULL;
|
||||||
clock_lclick_command = 0;
|
clock_lclick_command = NULL;
|
||||||
clock_rclick_command = 0;
|
clock_mclick_command = NULL;
|
||||||
time1_font_desc = 0;
|
clock_rclick_command = NULL;
|
||||||
time2_font_desc = 0;
|
clock_uwheel_command = NULL;
|
||||||
|
clock_dwheel_command = NULL;
|
||||||
|
time1_has_font = FALSE;
|
||||||
|
time1_font_desc = NULL;
|
||||||
|
time2_has_font = FALSE;
|
||||||
|
time2_font_desc = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cleanup_clock()
|
void cleanup_clock()
|
||||||
{
|
{
|
||||||
if (time1_font_desc) pango_font_description_free(time1_font_desc);
|
pango_font_description_free(time1_font_desc);
|
||||||
if (time2_font_desc) pango_font_description_free(time2_font_desc);
|
time1_font_desc = NULL;
|
||||||
if (time1_format) g_free(time1_format);
|
pango_font_description_free(time2_font_desc);
|
||||||
if (time2_format) g_free(time2_format);
|
time2_font_desc = NULL;
|
||||||
if (time_tooltip_format) g_free(time_tooltip_format);
|
free(time1_format);
|
||||||
if (time1_timezone) g_free(time1_timezone);
|
time1_format = NULL;
|
||||||
if (time2_timezone) g_free(time2_timezone);
|
free(time2_format);
|
||||||
if (time_tooltip_timezone) g_free(time_tooltip_timezone);
|
time2_format = NULL;
|
||||||
if (clock_lclick_command) g_free(clock_lclick_command);
|
free(time_tooltip_format);
|
||||||
if (clock_rclick_command) g_free(clock_rclick_command);
|
time_tooltip_format = NULL;
|
||||||
if (clock_timeout) stop_timeout(clock_timeout);
|
free(time1_timezone);
|
||||||
|
time1_timezone = NULL;
|
||||||
|
free(time2_timezone);
|
||||||
|
time2_timezone = NULL;
|
||||||
|
free(time_tooltip_timezone);
|
||||||
|
time_tooltip_timezone = NULL;
|
||||||
|
free(clock_lclick_command);
|
||||||
|
clock_lclick_command = NULL;
|
||||||
|
free(clock_mclick_command);
|
||||||
|
clock_mclick_command = NULL;
|
||||||
|
free(clock_rclick_command);
|
||||||
|
clock_rclick_command = NULL;
|
||||||
|
free(clock_uwheel_command);
|
||||||
|
clock_uwheel_command = NULL;
|
||||||
|
free(clock_dwheel_command);
|
||||||
|
clock_dwheel_command = NULL;
|
||||||
|
stop_timeout(clock_timeout);
|
||||||
|
clock_timeout = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void update_clocks_sec(void *arg)
|
||||||
void update_clocks_sec(void* arg)
|
|
||||||
{
|
{
|
||||||
gettimeofday(&time_clock, 0);
|
gettimeofday(&time_clock, 0);
|
||||||
int i;
|
if (time1_format) {
|
||||||
if (time1_format) {
|
for (int i = 0; i < num_panels; i++)
|
||||||
for (i=0 ; i < nb_panel ; i++)
|
panels[i].clock.area.resize_needed = 1;
|
||||||
panel1[i].clock.area.resize = 1;
|
}
|
||||||
}
|
schedule_panel_redraw();
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void update_clocks_min(void* arg)
|
void update_clocks_min(void *arg)
|
||||||
{
|
{
|
||||||
// remember old_sec because after suspend/hibernate the clock should be updated directly, and not
|
// remember old_sec because after suspend/hibernate the clock should be updated directly, and not
|
||||||
// on next minute change
|
// on next minute change
|
||||||
time_t old_sec = time_clock.tv_sec;
|
time_t old_sec = time_clock.tv_sec;
|
||||||
gettimeofday(&time_clock, 0);
|
gettimeofday(&time_clock, 0);
|
||||||
if (time_clock.tv_sec % 60 == 0 || time_clock.tv_sec - old_sec > 60) {
|
if (time_clock.tv_sec % 60 == 0 || time_clock.tv_sec - old_sec > 60) {
|
||||||
int i;
|
if (time1_format) {
|
||||||
if (time1_format) {
|
for (int i = 0; i < num_panels; i++)
|
||||||
for (i=0 ; i < nb_panel ; i++)
|
panels[i].clock.area.resize_needed = 1;
|
||||||
panel1[i].clock.area.resize = 1;
|
}
|
||||||
}
|
schedule_panel_redraw();
|
||||||
panel_refresh = 1;
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct tm* clock_gettime_for_tz(const char* timezone) {
|
struct tm *clock_gettime_for_tz(const char *timezone)
|
||||||
if (timezone) {
|
|
||||||
const char* old_tz = getenv("TZ");
|
|
||||||
setenv("TZ", timezone, 1);
|
|
||||||
struct tm* result = localtime(&time_clock.tv_sec);
|
|
||||||
if (old_tz) setenv("TZ", old_tz, 1);
|
|
||||||
else unsetenv("TZ");
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
else return localtime(&time_clock.tv_sec);
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* clock_get_tooltip(void* obj)
|
|
||||||
{
|
{
|
||||||
strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone));
|
if (timezone) {
|
||||||
return buf_tooltip;
|
const char *old_tz = getenv("TZ");
|
||||||
|
setenv("TZ", timezone, 1);
|
||||||
|
struct tm *result = localtime(&time_clock.tv_sec);
|
||||||
|
if (old_tz)
|
||||||
|
setenv("TZ", old_tz, 1);
|
||||||
|
else
|
||||||
|
unsetenv("TZ");
|
||||||
|
return result;
|
||||||
|
} else {
|
||||||
|
return localtime(&time_clock.tv_sec);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean time_format_needs_sec_ticks(char *time_format)
|
||||||
|
{
|
||||||
|
if (!time_format)
|
||||||
|
return FALSE;
|
||||||
|
if (strchr(time_format, 'S') || strchr(time_format, 'T') || strchr(time_format, 'r'))
|
||||||
|
return TRUE;
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
void init_clock()
|
void init_clock()
|
||||||
{
|
{
|
||||||
if(time1_format && clock_timeout==0) {
|
|
||||||
if (strchr(time1_format, 'S') || strchr(time1_format, 'T') || strchr(time1_format, 'r'))
|
|
||||||
clock_timeout = add_timeout(10, 1000, update_clocks_sec, 0);
|
|
||||||
else
|
|
||||||
clock_timeout = add_timeout(10, 1000, update_clocks_min, 0);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_clock_panel(void *p)
|
void init_clock_panel(void *p)
|
||||||
{
|
{
|
||||||
Panel *panel =(Panel*)p;
|
Panel *panel = (Panel *)p;
|
||||||
Clock *clock = &panel->clock;
|
Clock *clock = &panel->clock;
|
||||||
|
|
||||||
if (clock->area.bg == 0)
|
|
||||||
clock->area.bg = &g_array_index(backgrounds, Background, 0);
|
|
||||||
clock->area.parent = p;
|
|
||||||
clock->area.panel = p;
|
|
||||||
clock->area._draw_foreground = draw_clock;
|
|
||||||
clock->area.size_mode = SIZE_BY_CONTENT;
|
|
||||||
clock->area._resize = resize_clock;
|
|
||||||
// check consistency
|
|
||||||
if (time1_format == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
clock->area.resize = 1;
|
if (!clock_timeout) {
|
||||||
clock->area.on_screen = 1;
|
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 (time_tooltip_format) {
|
if (!clock->area.bg)
|
||||||
clock->area._get_tooltip_text = clock_get_tooltip;
|
clock->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone));
|
clock_init_fonts();
|
||||||
}
|
clock->area.parent = p;
|
||||||
|
clock->area.panel = p;
|
||||||
|
snprintf(clock->area.name, sizeof(clock->area.name), "Clock");
|
||||||
|
clock->area._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
|
clock->area.has_mouse_press_effect = clock->area.has_mouse_over_effect =
|
||||||
|
panel_config.mouse_effects && (clock_lclick_command || clock_mclick_command || clock_rclick_command ||
|
||||||
|
clock_uwheel_command || clock_dwheel_command);
|
||||||
|
clock->area._draw_foreground = draw_clock;
|
||||||
|
clock->area.size_mode = LAYOUT_FIXED;
|
||||||
|
clock->area._resize = resize_clock;
|
||||||
|
clock->area._compute_desired_size = clock_compute_desired_size;
|
||||||
|
clock->area._dump_geometry = clock_dump_geometry;
|
||||||
|
// check consistency
|
||||||
|
if (!time1_format)
|
||||||
|
return;
|
||||||
|
|
||||||
|
clock->area.resize_needed = 1;
|
||||||
|
clock->area.on_screen = TRUE;
|
||||||
|
instantiate_area_gradients(&clock->area);
|
||||||
|
|
||||||
|
if (time_tooltip_format) {
|
||||||
|
clock->area._get_tooltip_text = clock_get_tooltip;
|
||||||
|
strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void clock_init_fonts()
|
||||||
void draw_clock (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Clock *clock = obj;
|
if (!time1_font_desc) {
|
||||||
PangoLayout *layout;
|
time1_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
pango_font_description_set_weight(time1_font_desc, PANGO_WEIGHT_BOLD);
|
||||||
layout = pango_cairo_create_layout (c);
|
pango_font_description_set_size(time1_font_desc, pango_font_description_get_size(time1_font_desc));
|
||||||
|
}
|
||||||
// draw layout
|
if (!time2_font_desc) {
|
||||||
pango_layout_set_font_description (layout, time1_font_desc);
|
time2_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
pango_layout_set_width (layout, clock->area.width * PANGO_SCALE);
|
pango_font_description_set_size(time2_font_desc,
|
||||||
pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
pango_font_description_get_size(time2_font_desc) - PANGO_SCALE);
|
||||||
pango_layout_set_text (layout, buf_time, strlen(buf_time));
|
}
|
||||||
|
|
||||||
cairo_set_source_rgba (c, clock->font.color[0], clock->font.color[1], clock->font.color[2], clock->font.alpha);
|
|
||||||
|
|
||||||
pango_cairo_update_layout (c, layout);
|
|
||||||
cairo_move_to (c, 0, clock->time1_posy);
|
|
||||||
pango_cairo_show_layout (c, layout);
|
|
||||||
|
|
||||||
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);
|
|
||||||
cairo_move_to (c, 0, clock->time2_posy);
|
|
||||||
pango_cairo_show_layout (c, layout);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_object_unref (layout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void clock_default_font_changed()
|
||||||
int resize_clock (void *obj)
|
|
||||||
{
|
{
|
||||||
Clock *clock = obj;
|
if (!clock_enabled)
|
||||||
Panel *panel = clock->area.panel;
|
return;
|
||||||
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width, ret = 0;
|
if (time1_has_font && time2_has_font)
|
||||||
|
return;
|
||||||
clock->area.redraw = 1;
|
if (!time1_has_font) {
|
||||||
|
pango_font_description_free(time1_font_desc);
|
||||||
date_height = date_width = 0;
|
time1_font_desc = NULL;
|
||||||
strftime(buf_time, sizeof(buf_time), time1_format, clock_gettime_for_tz(time1_timezone));
|
}
|
||||||
get_text_size2(time1_font_desc, &time_height_ink, &time_height, &time_width, panel->area.height, panel->area.width, buf_time, strlen(buf_time));
|
if (!time2_has_font) {
|
||||||
if (time2_format) {
|
pango_font_description_free(time2_font_desc);
|
||||||
strftime(buf_date, sizeof(buf_date), time2_format, clock_gettime_for_tz(time2_timezone));
|
time2_font_desc = NULL;
|
||||||
get_text_size2(time2_font_desc, &date_height_ink, &date_height, &date_width, panel->area.height, panel->area.width, buf_date, strlen(buf_date));
|
}
|
||||||
}
|
clock_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
if (panel_horizontal) {
|
panels[i].clock.area.resize_needed = TRUE;
|
||||||
int new_size = (time_width > date_width) ? time_width : date_width;
|
schedule_redraw(&panels[i].clock.area);
|
||||||
new_size += (2*clock->area.paddingxlr) + (2*clock->area.bg->border.width);
|
}
|
||||||
if (new_size > clock->area.width || new_size < (clock->area.width-6)) {
|
schedule_panel_redraw();
|
||||||
// we try to limit the number of resize
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
int new_size = time_height + date_height + (2 * (clock->area.paddingxlr + clock->area.bg->border.width));
|
|
||||||
if (new_size != clock->area.height) {
|
|
||||||
// we try to limit the number of resize
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void clock_compute_text_geometry(Panel *panel,
|
||||||
void clock_action(int button)
|
int *time_height_ink,
|
||||||
|
int *time_height,
|
||||||
|
int *time_width,
|
||||||
|
int *date_height_ink,
|
||||||
|
int *date_height,
|
||||||
|
int *date_width)
|
||||||
{
|
{
|
||||||
char *command = 0;
|
*date_height = *date_width = 0;
|
||||||
switch (button) {
|
strftime(buf_time, sizeof(buf_time), time1_format, clock_gettime_for_tz(time1_timezone));
|
||||||
case 1:
|
get_text_size2(time1_font_desc,
|
||||||
command = clock_lclick_command;
|
time_height_ink,
|
||||||
break;
|
time_height,
|
||||||
case 3:
|
time_width,
|
||||||
command = clock_rclick_command;
|
panel->area.height,
|
||||||
break;
|
panel->area.width,
|
||||||
}
|
buf_time,
|
||||||
tint_exec(command);
|
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,
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
Clock *clock = (Clock *)obj;
|
||||||
|
Panel *panel = (Panel *)clock->area.panel;
|
||||||
|
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width;
|
||||||
|
clock_compute_text_geometry(panel,
|
||||||
|
&time_height_ink,
|
||||||
|
&time_height,
|
||||||
|
&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)
|
||||||
|
{
|
||||||
|
Clock *clock = (Clock *)obj;
|
||||||
|
Panel *panel = (Panel *)clock->area.panel;
|
||||||
|
gboolean result = FALSE;
|
||||||
|
|
||||||
|
schedule_redraw(&clock->area);
|
||||||
|
|
||||||
|
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width;
|
||||||
|
clock_compute_text_geometry(panel,
|
||||||
|
&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)
|
||||||
|
{
|
||||||
|
Clock *clock = obj;
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
|
||||||
|
pango_layout_set_font_description(layout, time1_font_desc);
|
||||||
|
pango_layout_set_width(layout, clock->area.width * PANGO_SCALE);
|
||||||
|
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
|
||||||
|
pango_layout_set_text(layout, buf_time, strlen(buf_time));
|
||||||
|
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
Clock *clock = (Clock *)obj;
|
||||||
|
fprintf(stderr, "%*sText 1: y = %d, text = %s\n", indent, "", clock->time1_posy, buf_time);
|
||||||
|
if (time2_format) {
|
||||||
|
fprintf(stderr, "%*sText 2: y = %d, text = %s\n", indent, "", clock->time2_posy, buf_date);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char *clock_get_tooltip(void *obj)
|
||||||
|
{
|
||||||
|
strftime(buf_tooltip, sizeof(buf_tooltip), time_tooltip_format, clock_gettime_for_tz(time_tooltip_timezone));
|
||||||
|
return strdup(buf_tooltip);
|
||||||
|
}
|
||||||
|
|
||||||
|
void clock_action(int button, Time time)
|
||||||
|
{
|
||||||
|
char *command = NULL;
|
||||||
|
switch (button) {
|
||||||
|
case 1:
|
||||||
|
command = clock_lclick_command;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
command = clock_mclick_command;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
command = clock_rclick_command;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
command = clock_uwheel_command;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
command = clock_dwheel_command;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
tint_exec(command, NULL, NULL, time);
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,29 +13,31 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "area.h"
|
#include "area.h"
|
||||||
|
|
||||||
|
|
||||||
typedef struct Clock {
|
typedef struct Clock {
|
||||||
// always start with area
|
// always start with area
|
||||||
Area area;
|
Area area;
|
||||||
|
|
||||||
Color font;
|
Color font;
|
||||||
int time1_posy;
|
int time1_posy;
|
||||||
int time2_posy;
|
int time2_posy;
|
||||||
} Clock;
|
} Clock;
|
||||||
|
|
||||||
|
|
||||||
extern char *time1_format;
|
extern char *time1_format;
|
||||||
extern char *time1_timezone;
|
extern char *time1_timezone;
|
||||||
extern char *time2_format;
|
extern char *time2_format;
|
||||||
extern char *time2_timezone;
|
extern char *time2_timezone;
|
||||||
extern char *time_tooltip_format;
|
extern char *time_tooltip_format;
|
||||||
extern char *time_tooltip_timezone;
|
extern char *time_tooltip_timezone;
|
||||||
|
extern gboolean time1_has_font;
|
||||||
extern PangoFontDescription *time1_font_desc;
|
extern PangoFontDescription *time1_font_desc;
|
||||||
|
extern gboolean time2_has_font;
|
||||||
extern PangoFontDescription *time2_font_desc;
|
extern PangoFontDescription *time2_font_desc;
|
||||||
extern char *clock_lclick_command;
|
extern char *clock_lclick_command;
|
||||||
|
extern char *clock_mclick_command;
|
||||||
extern char *clock_rclick_command;
|
extern char *clock_rclick_command;
|
||||||
extern int clock_enabled;
|
extern char *clock_uwheel_command;
|
||||||
|
extern char *clock_dwheel_command;
|
||||||
|
extern gboolean clock_enabled;
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_clock();
|
void default_clock();
|
||||||
@@ -46,11 +48,12 @@ void cleanup_clock();
|
|||||||
// initialize clock : y position, precision, ...
|
// initialize clock : y position, precision, ...
|
||||||
void init_clock();
|
void init_clock();
|
||||||
void init_clock_panel(void *panel);
|
void init_clock_panel(void *panel);
|
||||||
|
void clock_default_font_changed();
|
||||||
|
|
||||||
void draw_clock (void *obj, cairo_t *c);
|
void draw_clock(void *obj, cairo_t *c);
|
||||||
|
|
||||||
int resize_clock (void *obj);
|
gboolean resize_clock(void *obj);
|
||||||
|
|
||||||
void clock_action(int button);
|
void clock_action(int button, Time time);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
1920
src/config.c
1920
src/config.c
File diff suppressed because it is too large
Load Diff
@@ -9,6 +9,8 @@
|
|||||||
#ifndef CONFIG_H
|
#ifndef CONFIG_H
|
||||||
#define CONFIG_H
|
#define CONFIG_H
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
extern char *config_path;
|
extern char *config_path;
|
||||||
extern char *snapshot_path;
|
extern char *snapshot_path;
|
||||||
|
|
||||||
@@ -18,8 +20,6 @@ void default_config();
|
|||||||
// freed memory
|
// freed memory
|
||||||
void cleanup_config();
|
void cleanup_config();
|
||||||
|
|
||||||
int config_read_file (const char *path);
|
gboolean config_read();
|
||||||
int config_read ();
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
860
src/execplugin/execplugin.c
Normal file
860
src/execplugin/execplugin.c
Normal file
@@ -0,0 +1,860 @@
|
|||||||
|
#include "execplugin.h"
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <cairo.h>
|
||||||
|
#include <cairo-xlib.h>
|
||||||
|
#include <math.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
|
||||||
|
#include "window.h"
|
||||||
|
#include "server.h"
|
||||||
|
#include "panel.h"
|
||||||
|
#include "timer.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
void execp_timer_callback(void *arg);
|
||||||
|
char *execp_get_tooltip(void *obj);
|
||||||
|
void execp_init_fonts();
|
||||||
|
int execp_compute_desired_size(void *obj);
|
||||||
|
void execp_dump_geometry(void *obj, int indent);
|
||||||
|
|
||||||
|
void default_execp()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Execp *create_execp()
|
||||||
|
{
|
||||||
|
Execp *execp = calloc(1, sizeof(Execp));
|
||||||
|
execp->backend = calloc(1, sizeof(ExecpBackend));
|
||||||
|
execp->backend->child_pipe = -1;
|
||||||
|
execp->backend->cmd_pids = g_tree_new(cmp_ptr);
|
||||||
|
execp->backend->interval = 30;
|
||||||
|
execp->backend->cache_icon = TRUE;
|
||||||
|
execp->backend->centered = TRUE;
|
||||||
|
execp->backend->font_color.alpha = 0.5;
|
||||||
|
return execp;
|
||||||
|
}
|
||||||
|
|
||||||
|
gpointer create_execp_frontend(gconstpointer arg, gpointer data)
|
||||||
|
{
|
||||||
|
Execp *execp_backend = (Execp *)arg;
|
||||||
|
|
||||||
|
Execp *execp_frontend = calloc(1, sizeof(Execp));
|
||||||
|
execp_frontend->backend = execp_backend->backend;
|
||||||
|
execp_backend->backend->instances = g_list_append(execp_backend->backend->instances, execp_frontend);
|
||||||
|
execp_frontend->frontend = calloc(1, sizeof(ExecpFrontend));
|
||||||
|
return execp_frontend;
|
||||||
|
}
|
||||||
|
|
||||||
|
void destroy_execp(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = (Execp *)obj;
|
||||||
|
if (execp->frontend) {
|
||||||
|
// This is a frontend element
|
||||||
|
execp->backend->instances = g_list_remove_all(execp->backend->instances, execp);
|
||||||
|
free_and_null(execp->frontend);
|
||||||
|
remove_area(&execp->area);
|
||||||
|
free_area(&execp->area);
|
||||||
|
free_and_null(execp);
|
||||||
|
} else {
|
||||||
|
// This is a backend element
|
||||||
|
stop_timeout(execp->backend->timer);
|
||||||
|
execp->backend->timer = NULL;
|
||||||
|
|
||||||
|
if (execp->backend->icon) {
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
imlib_free_image();
|
||||||
|
execp->backend->icon = NULL;
|
||||||
|
}
|
||||||
|
free_and_null(execp->backend->buf_output);
|
||||||
|
free_and_null(execp->backend->text);
|
||||||
|
free_and_null(execp->backend->icon_path);
|
||||||
|
if (execp->backend->child) {
|
||||||
|
kill(-execp->backend->child, SIGHUP);
|
||||||
|
execp->backend->child = 0;
|
||||||
|
}
|
||||||
|
if (execp->backend->child_pipe >= 0) {
|
||||||
|
close(execp->backend->child_pipe);
|
||||||
|
execp->backend->child_pipe = -1;
|
||||||
|
}
|
||||||
|
if (execp->backend->cmd_pids) {
|
||||||
|
g_tree_destroy(execp->backend->cmd_pids);
|
||||||
|
execp->backend->cmd_pids = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
execp->backend->bg = NULL;
|
||||||
|
pango_font_description_free(execp->backend->font_desc);
|
||||||
|
execp->backend->font_desc = NULL;
|
||||||
|
free_and_null(execp->backend->command);
|
||||||
|
free_and_null(execp->backend->tooltip);
|
||||||
|
free_and_null(execp->backend->lclick_command);
|
||||||
|
free_and_null(execp->backend->mclick_command);
|
||||||
|
free_and_null(execp->backend->rclick_command);
|
||||||
|
free_and_null(execp->backend->dwheel_command);
|
||||||
|
free_and_null(execp->backend->uwheel_command);
|
||||||
|
|
||||||
|
if (execp->backend->instances) {
|
||||||
|
fprintf(stderr, "Error: Attempt to destroy backend while there are still frontend instances!\n");
|
||||||
|
exit(-1);
|
||||||
|
}
|
||||||
|
free(execp->backend);
|
||||||
|
free(execp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_execp()
|
||||||
|
{
|
||||||
|
GList *to_remove = panel_config.execp_list;
|
||||||
|
for (int k = 0; k < strlen(panel_items_order) && to_remove; k++) {
|
||||||
|
if (panel_items_order[k] == 'E') {
|
||||||
|
to_remove = to_remove->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (to_remove) {
|
||||||
|
if (to_remove == panel_config.execp_list) {
|
||||||
|
g_list_free_full(to_remove, destroy_execp);
|
||||||
|
panel_config.execp_list = NULL;
|
||||||
|
} else {
|
||||||
|
// Cut panel_config.execp_list
|
||||||
|
if (to_remove->prev)
|
||||||
|
to_remove->prev->next = NULL;
|
||||||
|
to_remove->prev = NULL;
|
||||||
|
// Remove all elements of to_remove and to_remove itself
|
||||||
|
g_list_free_full(to_remove, destroy_execp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
execp_init_fonts();
|
||||||
|
for (GList *l = panel_config.execp_list; l; l = l->next) {
|
||||||
|
Execp *execp = l->data;
|
||||||
|
|
||||||
|
// Set missing config options
|
||||||
|
if (!execp->backend->bg)
|
||||||
|
execp->backend->bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
execp->backend->buf_capacity = 1024;
|
||||||
|
execp->backend->buf_output = calloc(execp->backend->buf_capacity, 1);
|
||||||
|
execp->backend->text = strdup(" ");
|
||||||
|
execp->backend->icon_path = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_execp_panel(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
|
||||||
|
// Make sure this is only done once if there are multiple items
|
||||||
|
if (panel->execp_list && ((Execp *)panel->execp_list->data)->frontend)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// panel->execp_list is now a copy of the pointer panel_config.execp_list
|
||||||
|
// We make it a deep copy
|
||||||
|
panel->execp_list = g_list_copy_deep(panel_config.execp_list, create_execp_frontend, NULL);
|
||||||
|
|
||||||
|
for (GList *l = panel->execp_list; l; l = l->next) {
|
||||||
|
Execp *execp = l->data;
|
||||||
|
execp->area.bg = execp->backend->bg;
|
||||||
|
execp->area.paddingx = execp->backend->paddingx;
|
||||||
|
execp->area.paddingy = execp->backend->paddingy;
|
||||||
|
execp->area.paddingxlr = execp->backend->paddingxlr;
|
||||||
|
execp->area.parent = panel;
|
||||||
|
execp->area.panel = panel;
|
||||||
|
execp->area._dump_geometry = execp_dump_geometry;
|
||||||
|
execp->area._compute_desired_size = execp_compute_desired_size;
|
||||||
|
snprintf(execp->area.name,
|
||||||
|
sizeof(execp->area.name),
|
||||||
|
"Execp %s",
|
||||||
|
execp->backend->command ? execp->backend->command : "null");
|
||||||
|
execp->area._draw_foreground = draw_execp;
|
||||||
|
execp->area.size_mode = LAYOUT_FIXED;
|
||||||
|
execp->area._resize = resize_execp;
|
||||||
|
execp->area._get_tooltip_text = execp_get_tooltip;
|
||||||
|
execp->area._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
|
execp->area.has_mouse_press_effect =
|
||||||
|
panel_config.mouse_effects &&
|
||||||
|
(execp->area.has_mouse_over_effect = execp->backend->lclick_command || execp->backend->mclick_command ||
|
||||||
|
execp->backend->rclick_command || execp->backend->uwheel_command ||
|
||||||
|
execp->backend->dwheel_command);
|
||||||
|
|
||||||
|
execp->area.resize_needed = TRUE;
|
||||||
|
execp->area.on_screen = TRUE;
|
||||||
|
instantiate_area_gradients(&execp->area);
|
||||||
|
|
||||||
|
if (!execp->backend->timer)
|
||||||
|
execp->backend->timer = add_timeout(10, 0, execp_timer_callback, execp, &execp->backend->timer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_init_fonts()
|
||||||
|
{
|
||||||
|
for (GList *l = panel_config.execp_list; l; l = l->next) {
|
||||||
|
Execp *execp = l->data;
|
||||||
|
if (!execp->backend->font_desc)
|
||||||
|
execp->backend->font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_default_font_changed()
|
||||||
|
{
|
||||||
|
gboolean needs_update = FALSE;
|
||||||
|
for (GList *l = panel_config.execp_list; l; l = l->next) {
|
||||||
|
Execp *execp = l->data;
|
||||||
|
|
||||||
|
if (!execp->backend->has_font) {
|
||||||
|
pango_font_description_free(execp->backend->font_desc);
|
||||||
|
execp->backend->font_desc = NULL;
|
||||||
|
needs_update = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!needs_update)
|
||||||
|
return;
|
||||||
|
|
||||||
|
execp_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
for (GList *l = panels[i].execp_list; l; l = l->next) {
|
||||||
|
Execp *execp = l->data;
|
||||||
|
|
||||||
|
if (!execp->backend->has_font) {
|
||||||
|
execp->area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&execp->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup_execp()
|
||||||
|
{
|
||||||
|
// Cleanup frontends
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
g_list_free_full(panels[i].execp_list, destroy_execp);
|
||||||
|
panels[i].execp_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup backends
|
||||||
|
g_list_free_full(panel_config.execp_list, destroy_execp);
|
||||||
|
panel_config.execp_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called from backend functions.
|
||||||
|
gboolean reload_icon(Execp *execp)
|
||||||
|
{
|
||||||
|
char *icon_path = execp->backend->icon_path;
|
||||||
|
|
||||||
|
if (execp->backend->has_icon && icon_path) {
|
||||||
|
if (execp->backend->icon) {
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
imlib_free_image();
|
||||||
|
}
|
||||||
|
execp->backend->icon = load_image(icon_path, execp->backend->cache_icon);
|
||||||
|
if (execp->backend->icon) {
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
int w = imlib_image_get_width();
|
||||||
|
int h = imlib_image_get_height();
|
||||||
|
if (w && h) {
|
||||||
|
if (execp->backend->icon_w) {
|
||||||
|
if (!execp->backend->icon_h) {
|
||||||
|
h = (int)(0.5 + h * execp->backend->icon_w / (float)(w));
|
||||||
|
w = execp->backend->icon_w;
|
||||||
|
} else {
|
||||||
|
w = execp->backend->icon_w;
|
||||||
|
h = execp->backend->icon_h;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (execp->backend->icon_h) {
|
||||||
|
w = (int)(0.5 + w * execp->backend->icon_h / (float)(h));
|
||||||
|
h = execp->backend->icon_h;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (w < 1)
|
||||||
|
w = 1;
|
||||||
|
if (h < 1)
|
||||||
|
h = 1;
|
||||||
|
}
|
||||||
|
if (w != imlib_image_get_width() || h != imlib_image_get_height()) {
|
||||||
|
Imlib_Image icon_scaled =
|
||||||
|
imlib_create_cropped_scaled_image(0, 0, imlib_image_get_width(), imlib_image_get_height(), w, h);
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
imlib_free_image();
|
||||||
|
execp->backend->icon = icon_scaled;
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
int execp_compute_desired_size(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = (Execp *)obj;
|
||||||
|
Panel *panel = (Panel *)execp->area.panel;
|
||||||
|
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;
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
int text_next_line = !panel_horizontal && icon_w > execp->area.width / 2;
|
||||||
|
|
||||||
|
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;
|
||||||
|
if (!text_next_line) {
|
||||||
|
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area);
|
||||||
|
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&execp->area));
|
||||||
|
} else {
|
||||||
|
new_size =
|
||||||
|
icon_h + interior_padding + txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area);
|
||||||
|
}
|
||||||
|
return new_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean resize_execp(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = (Execp *)obj;
|
||||||
|
Panel *panel = (Panel *)execp->area.panel;
|
||||||
|
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;
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
int text_next_line = !panel_horizontal && icon_w > execp->area.width / 2;
|
||||||
|
|
||||||
|
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;
|
||||||
|
if (!text_next_line) {
|
||||||
|
new_size = txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area);
|
||||||
|
new_size = MAX(new_size, icon_h + 2 * vert_padding + top_bottom_border_width(&execp->area));
|
||||||
|
} else {
|
||||||
|
new_size =
|
||||||
|
icon_h + interior_padding + txt_height + 2 * vert_padding + top_bottom_border_width(&execp->area);
|
||||||
|
}
|
||||||
|
if (new_size != execp->area.height) {
|
||||||
|
execp->area.height = new_size;
|
||||||
|
result = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
execp->frontend->textw = txt_width;
|
||||||
|
execp->frontend->texth = txt_height;
|
||||||
|
if (execp->backend->centered) {
|
||||||
|
if (icon_w) {
|
||||||
|
if (!text_next_line) {
|
||||||
|
execp->frontend->icony = (execp->area.height - icon_h) / 2;
|
||||||
|
execp->frontend->iconx = (execp->area.width - txt_width - interior_padding - icon_w) / 2;
|
||||||
|
execp->frontend->texty = (execp->area.height - txt_height) / 2;
|
||||||
|
execp->frontend->textx = execp->frontend->iconx + icon_w + interior_padding;
|
||||||
|
} else {
|
||||||
|
execp->frontend->icony = (execp->area.height - icon_h - interior_padding - txt_height) / 2;
|
||||||
|
execp->frontend->iconx = (execp->area.width - icon_w) / 2;
|
||||||
|
execp->frontend->texty = execp->frontend->icony + icon_h + interior_padding;
|
||||||
|
execp->frontend->textx = (execp->area.width - txt_width) / 2;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
execp->frontend->texty = (execp->area.height - txt_height) / 2;
|
||||||
|
execp->frontend->textx = (execp->area.width - txt_width) / 2;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (icon_w) {
|
||||||
|
if (!text_next_line) {
|
||||||
|
execp->frontend->icony = (execp->area.height - icon_h) / 2;
|
||||||
|
execp->frontend->iconx = left_border_width(&execp->area) + horiz_padding;
|
||||||
|
execp->frontend->texty = (execp->area.height - txt_height) / 2;
|
||||||
|
execp->frontend->textx = execp->frontend->iconx + icon_w + interior_padding;
|
||||||
|
} else {
|
||||||
|
execp->frontend->icony = (execp->area.height - icon_h - interior_padding - txt_height) / 2;
|
||||||
|
execp->frontend->iconx = left_border_width(&execp->area) + horiz_padding;
|
||||||
|
execp->frontend->texty = execp->frontend->icony + icon_h + interior_padding;
|
||||||
|
execp->frontend->textx = execp->frontend->iconx;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
execp->frontend->texty = (execp->area.height - txt_height) / 2;
|
||||||
|
execp->frontend->textx = left_border_width(&execp->area) + horiz_padding;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
schedule_redraw(&execp->area);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_execp(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
Execp *execp = obj;
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
|
||||||
|
if (execp->backend->has_icon && execp->backend->icon) {
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
// Render icon
|
||||||
|
render_image(execp->area.pix, execp->frontend->iconx, execp->frontend->icony);
|
||||||
|
}
|
||||||
|
|
||||||
|
// draw layout
|
||||||
|
pango_layout_set_font_description(layout, execp->backend->font_desc);
|
||||||
|
pango_layout_set_width(layout, execp->frontend->textw * PANGO_SCALE);
|
||||||
|
pango_layout_set_alignment(layout, execp->backend->centered ? PANGO_ALIGN_CENTER : PANGO_ALIGN_LEFT);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
|
||||||
|
if (!execp->backend->has_markup)
|
||||||
|
pango_layout_set_text(layout, execp->backend->text, strlen(execp->backend->text));
|
||||||
|
else
|
||||||
|
pango_layout_set_markup(layout, execp->backend->text, strlen(execp->backend->text));
|
||||||
|
|
||||||
|
pango_cairo_update_layout(c, layout);
|
||||||
|
draw_text(layout,
|
||||||
|
c,
|
||||||
|
execp->frontend->textx,
|
||||||
|
execp->frontend->texty,
|
||||||
|
&execp->backend->font_color,
|
||||||
|
panel_config.font_shadow);
|
||||||
|
|
||||||
|
g_object_unref(layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_dump_geometry(void *obj, int indent)
|
||||||
|
{
|
||||||
|
Execp *execp = obj;
|
||||||
|
|
||||||
|
if (execp->backend->has_icon && execp->backend->icon) {
|
||||||
|
Imlib_Image tmp = imlib_context_get_image();
|
||||||
|
imlib_context_set_image(execp->backend->icon);
|
||||||
|
fprintf(stderr,
|
||||||
|
"%*sIcon: x = %d, y = %d, w = %d, h = %d\n",
|
||||||
|
indent,
|
||||||
|
"",
|
||||||
|
execp->frontend->iconx,
|
||||||
|
execp->frontend->icony,
|
||||||
|
imlib_image_get_width(),
|
||||||
|
imlib_image_get_height());
|
||||||
|
if (tmp)
|
||||||
|
imlib_context_set_image(tmp);
|
||||||
|
}
|
||||||
|
fprintf(stderr,
|
||||||
|
"%*sText: x = %d, y = %d, w = %d, align = %s, text = %s\n",
|
||||||
|
indent,
|
||||||
|
"",
|
||||||
|
execp->frontend->textx,
|
||||||
|
execp->frontend->texty,
|
||||||
|
execp->frontend->textw,
|
||||||
|
execp->backend->centered ? "center" : "left",
|
||||||
|
execp->backend->text);
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_force_update(Execp *execp)
|
||||||
|
{
|
||||||
|
if (execp->backend->child_pipe > 0) {
|
||||||
|
// Command currently running, nothing to do
|
||||||
|
} else {
|
||||||
|
if (execp->backend->timer)
|
||||||
|
stop_timeout(execp->backend->timer);
|
||||||
|
// Run command right away
|
||||||
|
execp->backend->timer = add_timeout(10, 0, execp_timer_callback, execp, &execp->backend->timer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_action(void *obj, int button, int x, int y, Time time)
|
||||||
|
{
|
||||||
|
Execp *execp = obj;
|
||||||
|
char *command = NULL;
|
||||||
|
switch (button) {
|
||||||
|
case 1:
|
||||||
|
command = execp->backend->lclick_command;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
command = execp->backend->mclick_command;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
command = execp->backend->rclick_command;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
command = execp->backend->uwheel_command;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
command = execp->backend->dwheel_command;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (command) {
|
||||||
|
char *full_cmd = g_strdup_printf("export EXECP_X=%d;"
|
||||||
|
"export EXECP_Y=%d;"
|
||||||
|
"export EXECP_W=%d;"
|
||||||
|
"export EXECP_H=%d; %s",
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
execp->area.width,
|
||||||
|
execp->area.height,
|
||||||
|
command);
|
||||||
|
tint_exec(full_cmd, NULL, NULL, time);
|
||||||
|
g_free(full_cmd);
|
||||||
|
} else {
|
||||||
|
execp_force_update(execp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_cmd_completed(Execp *execp, pid_t pid)
|
||||||
|
{
|
||||||
|
g_tree_remove(execp->backend->cmd_pids, GINT_TO_POINTER(pid));
|
||||||
|
execp_force_update(execp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void execp_timer_callback(void *arg)
|
||||||
|
{
|
||||||
|
Execp *execp = arg;
|
||||||
|
|
||||||
|
if (!execp->backend->command)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Still running!
|
||||||
|
if (execp->backend->child_pipe > 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int pipe_fd[2];
|
||||||
|
if (pipe(pipe_fd)) {
|
||||||
|
// TODO maybe write this in tooltip, but if this happens we're screwed anyways
|
||||||
|
fprintf(stderr, "Execp: Creating pipe failed!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fcntl(pipe_fd[0], F_SETFL, O_NONBLOCK | fcntl(pipe_fd[0], F_GETFL));
|
||||||
|
|
||||||
|
// Fork and run command, capturing stdout in pipe
|
||||||
|
pid_t child = fork();
|
||||||
|
if (child == -1) {
|
||||||
|
// TODO maybe write this in tooltip, but if this happens we're screwed anyways
|
||||||
|
fprintf(stderr, "Fork failed.\n");
|
||||||
|
close(pipe_fd[1]);
|
||||||
|
close(pipe_fd[0]);
|
||||||
|
return;
|
||||||
|
} else if (child == 0) {
|
||||||
|
fprintf(stderr, "Executing: %s\n", execp->backend->command);
|
||||||
|
// We are in the child
|
||||||
|
close(pipe_fd[0]);
|
||||||
|
dup2(pipe_fd[1], 1); // 1 is stdout
|
||||||
|
close(pipe_fd[1]);
|
||||||
|
setpgid(0, 0);
|
||||||
|
execl("/bin/sh", "/bin/sh", "-c", execp->backend->command, NULL);
|
||||||
|
// This should never happen!
|
||||||
|
fprintf(stdout, "execl() failed\nexecl() failed\n");
|
||||||
|
fflush(stdout);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
close(pipe_fd[1]);
|
||||||
|
execp->backend->child = child;
|
||||||
|
execp->backend->child_pipe = pipe_fd[0];
|
||||||
|
execp->backend->buf_length = 0;
|
||||||
|
execp->backend->buf_output[execp->backend->buf_length] = '\0';
|
||||||
|
execp->backend->last_update_start_time = time(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean read_execp(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = (Execp *)obj;
|
||||||
|
|
||||||
|
if (execp->backend->child_pipe < 0)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
gboolean command_finished = FALSE;
|
||||||
|
while (1) {
|
||||||
|
// Make sure there is free space in the buffer
|
||||||
|
if (execp->backend->buf_capacity - execp->backend->buf_length < 1024) {
|
||||||
|
execp->backend->buf_capacity *= 2;
|
||||||
|
execp->backend->buf_output = realloc(execp->backend->buf_output, execp->backend->buf_capacity);
|
||||||
|
}
|
||||||
|
ssize_t count = read(execp->backend->child_pipe,
|
||||||
|
execp->backend->buf_output + execp->backend->buf_length,
|
||||||
|
execp->backend->buf_capacity - execp->backend->buf_length - 1);
|
||||||
|
if (count > 0) {
|
||||||
|
// Successful read
|
||||||
|
execp->backend->buf_length += count;
|
||||||
|
execp->backend->buf_output[execp->backend->buf_length] = '\0';
|
||||||
|
continue;
|
||||||
|
} else if (count == 0) {
|
||||||
|
// End of file
|
||||||
|
command_finished = TRUE;
|
||||||
|
break;
|
||||||
|
} else if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
|
// No more data available at the moment
|
||||||
|
break;
|
||||||
|
} else if (errno == EINTR) {
|
||||||
|
// Harmless interruption by signal
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
// Error
|
||||||
|
command_finished = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (command_finished) {
|
||||||
|
execp->backend->child = 0;
|
||||||
|
close(execp->backend->child_pipe);
|
||||||
|
execp->backend->child_pipe = -1;
|
||||||
|
if (execp->backend->interval)
|
||||||
|
execp->backend->timer =
|
||||||
|
add_timeout(execp->backend->interval * 1000, 0, execp_timer_callback, execp, &execp->backend->timer);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!execp->backend->continuous && command_finished) {
|
||||||
|
free_and_null(execp->backend->text);
|
||||||
|
free_and_null(execp->backend->icon_path);
|
||||||
|
if (!execp->backend->has_icon) {
|
||||||
|
execp->backend->text = strdup(execp->backend->buf_output);
|
||||||
|
} else {
|
||||||
|
char *text = strchr(execp->backend->buf_output, '\n');
|
||||||
|
if (text) {
|
||||||
|
*text = '\0';
|
||||||
|
text++;
|
||||||
|
execp->backend->text = strdup(text);
|
||||||
|
} else {
|
||||||
|
execp->backend->text = strdup("");
|
||||||
|
}
|
||||||
|
execp->backend->icon_path = strdup(execp->backend->buf_output);
|
||||||
|
}
|
||||||
|
int len = strlen(execp->backend->text);
|
||||||
|
if (len > 0 && execp->backend->text[len - 1] == '\n')
|
||||||
|
execp->backend->text[len - 1] = '\0';
|
||||||
|
execp->backend->buf_length = 0;
|
||||||
|
execp->backend->buf_output[execp->backend->buf_length] = '\0';
|
||||||
|
execp->backend->last_update_finish_time = time(NULL);
|
||||||
|
execp->backend->last_update_duration =
|
||||||
|
execp->backend->last_update_finish_time - execp->backend->last_update_start_time;
|
||||||
|
return TRUE;
|
||||||
|
} else if (execp->backend->continuous > 0) {
|
||||||
|
// Count lines in buffer
|
||||||
|
int num_lines = 0;
|
||||||
|
char *last = execp->backend->buf_output;
|
||||||
|
char *end = NULL;
|
||||||
|
for (char *c = execp->backend->buf_output; *c; c++) {
|
||||||
|
if (*c == '\n') {
|
||||||
|
num_lines++;
|
||||||
|
if (num_lines == execp->backend->continuous)
|
||||||
|
end = c;
|
||||||
|
}
|
||||||
|
last = c;
|
||||||
|
}
|
||||||
|
if (*last && *last != '\n')
|
||||||
|
num_lines++;
|
||||||
|
if (num_lines >= execp->backend->continuous) {
|
||||||
|
if (end)
|
||||||
|
*end = '\0';
|
||||||
|
free_and_null(execp->backend->text);
|
||||||
|
free_and_null(execp->backend->icon_path);
|
||||||
|
if (!execp->backend->has_icon) {
|
||||||
|
execp->backend->text = strdup(execp->backend->buf_output);
|
||||||
|
} else {
|
||||||
|
char *text = strchr(execp->backend->buf_output, '\n');
|
||||||
|
if (text) {
|
||||||
|
*text = '\0';
|
||||||
|
text++;
|
||||||
|
execp->backend->text = strdup(text);
|
||||||
|
} else {
|
||||||
|
execp->backend->text = strdup("");
|
||||||
|
}
|
||||||
|
execp->backend->icon_path = strdup(execp->backend->buf_output);
|
||||||
|
}
|
||||||
|
int len = strlen(execp->backend->text);
|
||||||
|
if (len > 0 && execp->backend->text[len - 1] == '\n')
|
||||||
|
execp->backend->text[len - 1] = '\0';
|
||||||
|
|
||||||
|
if (end) {
|
||||||
|
char *next = end + 1;
|
||||||
|
int copied = next - execp->backend->buf_output;
|
||||||
|
int remaining = execp->backend->buf_length - copied;
|
||||||
|
if (remaining > 0) {
|
||||||
|
memmove(execp->backend->buf_output, next, remaining);
|
||||||
|
execp->backend->buf_length = remaining;
|
||||||
|
execp->backend->buf_output[execp->backend->buf_length] = '\0';
|
||||||
|
} else {
|
||||||
|
execp->backend->buf_length = 0;
|
||||||
|
execp->backend->buf_output[execp->backend->buf_length] = '\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
execp->backend->last_update_finish_time = time(NULL);
|
||||||
|
execp->backend->last_update_duration =
|
||||||
|
execp->backend->last_update_finish_time - execp->backend->last_update_start_time;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *time_to_string(int seconds, char *buffer)
|
||||||
|
{
|
||||||
|
if (seconds < 60) {
|
||||||
|
sprintf(buffer, "%ds", seconds);
|
||||||
|
} else if (seconds < 60 * 60) {
|
||||||
|
int m = seconds / 60;
|
||||||
|
seconds = seconds % 60;
|
||||||
|
int s = seconds;
|
||||||
|
sprintf(buffer, "%d:%ds", m, s);
|
||||||
|
} else {
|
||||||
|
int h = seconds / (60 * 60);
|
||||||
|
seconds = seconds % (60 * 60);
|
||||||
|
int m = seconds / 60;
|
||||||
|
seconds = seconds % 60;
|
||||||
|
int s = seconds;
|
||||||
|
sprintf(buffer, "%d:%d:%ds", h, m, s);
|
||||||
|
}
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *execp_get_tooltip(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = obj;
|
||||||
|
|
||||||
|
if (execp->backend->tooltip) {
|
||||||
|
if (strlen(execp->backend->tooltip) > 0)
|
||||||
|
return strdup(execp->backend->tooltip);
|
||||||
|
else
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
time_t now = time(NULL);
|
||||||
|
|
||||||
|
char tmp_buf1[256];
|
||||||
|
char tmp_buf2[256];
|
||||||
|
char tmp_buf3[256];
|
||||||
|
if (execp->backend->child_pipe < 0) {
|
||||||
|
// Not executing command
|
||||||
|
if (execp->backend->last_update_finish_time) {
|
||||||
|
// We updated at least once
|
||||||
|
if (execp->backend->interval > 0) {
|
||||||
|
sprintf(execp->backend->tooltip_text,
|
||||||
|
"Last update finished %s ago (took %s). Next update starting in %s.",
|
||||||
|
time_to_string((int)(now - execp->backend->last_update_finish_time), tmp_buf1),
|
||||||
|
time_to_string((int)execp->backend->last_update_duration, tmp_buf2),
|
||||||
|
time_to_string((int)(execp->backend->interval - (now - execp->backend->last_update_finish_time)),
|
||||||
|
tmp_buf3));
|
||||||
|
} else {
|
||||||
|
sprintf(execp->backend->tooltip_text,
|
||||||
|
"Last update finished %s ago (took %s).",
|
||||||
|
time_to_string((int)(now - execp->backend->last_update_finish_time), tmp_buf1),
|
||||||
|
time_to_string((int)execp->backend->last_update_duration, tmp_buf2));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// we never requested an update
|
||||||
|
sprintf(execp->backend->tooltip_text, "Never updated. No update scheduled.");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Currently executing command
|
||||||
|
if (execp->backend->last_update_finish_time) {
|
||||||
|
// we finished updating at least once
|
||||||
|
sprintf(execp->backend->tooltip_text,
|
||||||
|
"Last update finished %s ago. Update in progress (started %s ago).",
|
||||||
|
time_to_string((int)(now - execp->backend->last_update_finish_time), tmp_buf1),
|
||||||
|
time_to_string((int)(now - execp->backend->last_update_start_time), tmp_buf3));
|
||||||
|
} else {
|
||||||
|
// we never finished an update
|
||||||
|
sprintf(execp->backend->tooltip_text,
|
||||||
|
"First update in progress (started %s seconds ago).",
|
||||||
|
time_to_string((int)(now - execp->backend->last_update_start_time), tmp_buf1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return strdup(execp->backend->tooltip_text);
|
||||||
|
}
|
||||||
143
src/execplugin/execplugin.h
Normal file
143
src/execplugin/execplugin.h
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
#ifndef EXECPLUGIN_H
|
||||||
|
#define EXECPLUGIN_H
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
|
||||||
|
#include "area.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "timer.h"
|
||||||
|
|
||||||
|
// Architecture:
|
||||||
|
// Panel panel_config contains an array of Execp, each storing all config options and all the state variables.
|
||||||
|
// Only these run commands.
|
||||||
|
//
|
||||||
|
// Tint2 maintains an array of Panels, one for each monitor. Each stores an array of Execp which was initially copied
|
||||||
|
// from panel_config. Each works as a frontend to the corresponding Execp in panel_config as backend, using the
|
||||||
|
// backend's config and state variables.
|
||||||
|
|
||||||
|
typedef struct ExecpBackend {
|
||||||
|
// Config:
|
||||||
|
// Command to execute at a specified interval
|
||||||
|
char *command;
|
||||||
|
// Interval in seconds
|
||||||
|
int interval;
|
||||||
|
// 1 if first line of output is an icon path
|
||||||
|
gboolean has_icon;
|
||||||
|
gboolean cache_icon;
|
||||||
|
int icon_w;
|
||||||
|
int icon_h;
|
||||||
|
char *tooltip;
|
||||||
|
gboolean centered;
|
||||||
|
gboolean has_font;
|
||||||
|
PangoFontDescription *font_desc;
|
||||||
|
Color font_color;
|
||||||
|
int continuous;
|
||||||
|
gboolean has_markup;
|
||||||
|
char *lclick_command;
|
||||||
|
char *mclick_command;
|
||||||
|
char *rclick_command;
|
||||||
|
char *uwheel_command;
|
||||||
|
char *dwheel_command;
|
||||||
|
// paddingxlr = horizontal padding left/right
|
||||||
|
// paddingx = horizontal padding between childs
|
||||||
|
int paddingxlr, paddingx, paddingy;
|
||||||
|
Background *bg;
|
||||||
|
|
||||||
|
// Backend state:
|
||||||
|
timeout *timer;
|
||||||
|
int child_pipe;
|
||||||
|
pid_t child;
|
||||||
|
|
||||||
|
// Command output buffer
|
||||||
|
char *buf_output;
|
||||||
|
int buf_length;
|
||||||
|
int buf_capacity;
|
||||||
|
|
||||||
|
// Text extracted from the output buffer
|
||||||
|
char *text;
|
||||||
|
// Icon path extracted from the output buffer
|
||||||
|
char *icon_path;
|
||||||
|
Imlib_Image icon;
|
||||||
|
char tooltip_text[512];
|
||||||
|
|
||||||
|
// The time the last command was started
|
||||||
|
time_t last_update_start_time;
|
||||||
|
// The time the last output was obtained
|
||||||
|
time_t last_update_finish_time;
|
||||||
|
// The time it took to execute last command
|
||||||
|
time_t last_update_duration;
|
||||||
|
|
||||||
|
// List of Execp which are frontends for this backend, one for each panel
|
||||||
|
GList *instances;
|
||||||
|
GTree *cmd_pids;
|
||||||
|
} ExecpBackend;
|
||||||
|
|
||||||
|
typedef struct ExecpFrontend {
|
||||||
|
// Frontend state:
|
||||||
|
int iconx;
|
||||||
|
int icony;
|
||||||
|
int textx;
|
||||||
|
int texty;
|
||||||
|
int textw;
|
||||||
|
int texth;
|
||||||
|
} ExecpFrontend;
|
||||||
|
|
||||||
|
typedef struct Execp {
|
||||||
|
Area area;
|
||||||
|
// All elements have the backend pointer set. However only backend elements have ownership.
|
||||||
|
ExecpBackend *backend;
|
||||||
|
// Set only for frontend Execp items.
|
||||||
|
ExecpFrontend *frontend;
|
||||||
|
} Execp;
|
||||||
|
|
||||||
|
// Called before the config is read and panel_config/panels are created.
|
||||||
|
// Afterwards, the config parsing code creates the array of Execp in panel_config and populates the configuration fields
|
||||||
|
// in the backend.
|
||||||
|
// Probably does nothing.
|
||||||
|
void default_execp();
|
||||||
|
|
||||||
|
// Creates a new Execp item with only the backend field set. The state is NOT initialized. The config is initialized to
|
||||||
|
// the default values.
|
||||||
|
// This will be used by the config code to populate its backedn config fields.
|
||||||
|
Execp *create_execp();
|
||||||
|
|
||||||
|
void destroy_execp(void *obj);
|
||||||
|
|
||||||
|
// Called after the config is read and panel_config is populated, but before panels are created.
|
||||||
|
// Initializes the state of the backend items.
|
||||||
|
// panel_config.panel_items is used to determine which backend items are enabled. The others should be destroyed and
|
||||||
|
// removed from panel_config.execp_list.
|
||||||
|
void init_execp();
|
||||||
|
|
||||||
|
// Called after each on-screen panel is created, with a pointer to the panel.
|
||||||
|
// Initializes the state of the frontend items. Also adds a pointer to it in backend->instances.
|
||||||
|
// At this point the Area has not been added yet to the GUI tree, but it will be added right away.
|
||||||
|
void init_execp_panel(void *panel);
|
||||||
|
|
||||||
|
// Called just before the panels are destroyed. Afterwards, tint2 exits or restarts and reads the config again.
|
||||||
|
// Releases all frontends and then all the backends.
|
||||||
|
// The frontend items are not freed by this function, only their members. The items are Areas which are freed in the
|
||||||
|
// GUI element tree cleanup function (remove_area).
|
||||||
|
void cleanup_execp();
|
||||||
|
|
||||||
|
// Called on draw, obj = pointer to the front-end Execp item.
|
||||||
|
void draw_execp(void *obj, cairo_t *c);
|
||||||
|
|
||||||
|
// Called on resize, obj = pointer to the front-end Execp item.
|
||||||
|
// Returns 1 if the new size is different than the previous size.
|
||||||
|
gboolean resize_execp(void *obj);
|
||||||
|
|
||||||
|
// Called on mouse click event.
|
||||||
|
void execp_action(void *obj, int button, int x, int y, Time time);
|
||||||
|
|
||||||
|
void execp_cmd_completed(Execp *obj, pid_t pid);
|
||||||
|
|
||||||
|
// Called to check if new output from the command can be read.
|
||||||
|
// No command might be running.
|
||||||
|
// Returns 1 if the output has been updated and a redraw is needed.
|
||||||
|
gboolean read_execp(void *obj);
|
||||||
|
|
||||||
|
void execp_default_font_changed();
|
||||||
|
|
||||||
|
#endif // EXECPLUGIN_H
|
||||||
126
src/freespace/freespace.c
Normal file
126
src/freespace/freespace.c
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* Tint2 : freespace
|
||||||
|
*
|
||||||
|
* Copyright (C) 2011 Mishael A Sibiryakov (death@junki.org)
|
||||||
|
*
|
||||||
|
* 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 <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <cairo.h>
|
||||||
|
#include <cairo-xlib.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "window.h"
|
||||||
|
#include "server.h"
|
||||||
|
#include "panel.h"
|
||||||
|
#include "freespace.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
int freespace_area_compute_desired_size(void *obj);
|
||||||
|
|
||||||
|
void init_freespace_panel(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
|
||||||
|
// Make sure this is only done once if there are multiple items
|
||||||
|
if (panel->freespace_list)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (size_t k = 0; k < strlen(panel_items_order); k++) {
|
||||||
|
if (panel_items_order[k] == 'F') {
|
||||||
|
FreeSpace *freespace = (FreeSpace *)calloc(1, sizeof(FreeSpace));
|
||||||
|
panel->freespace_list = g_list_append(panel->freespace_list, freespace);
|
||||||
|
if (!freespace->area.bg)
|
||||||
|
freespace->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
freespace->area.parent = p;
|
||||||
|
freespace->area.panel = p;
|
||||||
|
snprintf(freespace->area.name, sizeof(freespace->area.name), "Freespace");
|
||||||
|
freespace->area.size_mode = LAYOUT_FIXED;
|
||||||
|
freespace->area.resize_needed = 1;
|
||||||
|
freespace->area.on_screen = TRUE;
|
||||||
|
freespace->area._resize = resize_freespace;
|
||||||
|
freespace->area._compute_desired_size = freespace_area_compute_desired_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup_freespace(Panel *panel)
|
||||||
|
{
|
||||||
|
if (panel->freespace_list)
|
||||||
|
g_list_free_full(panel->freespace_list, free);
|
||||||
|
panel->freespace_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int freespace_get_max_size(Panel *p)
|
||||||
|
{
|
||||||
|
if (panel_shrink)
|
||||||
|
return 0;
|
||||||
|
// Get space used by every element except the freespace
|
||||||
|
int size = 0;
|
||||||
|
int spacers = 0;
|
||||||
|
for (GList *walk = p->area.children; walk; walk = g_list_next(walk)) {
|
||||||
|
Area *a = (Area *)walk->data;
|
||||||
|
|
||||||
|
if (!a->on_screen)
|
||||||
|
continue;
|
||||||
|
if (a->_resize == resize_freespace) {
|
||||||
|
spacers++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (panel_horizontal)
|
||||||
|
size += a->width + p->area.paddingx;
|
||||||
|
else
|
||||||
|
size += a->height + p->area.paddingy;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (panel_horizontal)
|
||||||
|
size = p->area.width - size - left_right_border_width(&p->area) - p->area.paddingxlr;
|
||||||
|
else
|
||||||
|
size = p->area.height - size - top_bottom_border_width(&p->area) - p->area.paddingxlr;
|
||||||
|
|
||||||
|
return size / spacers;
|
||||||
|
}
|
||||||
|
|
||||||
|
int freespace_area_compute_desired_size(void *obj)
|
||||||
|
{
|
||||||
|
FreeSpace *freespace = (FreeSpace *)obj;
|
||||||
|
return freespace_get_max_size((Panel *)freespace->area.panel);
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean resize_freespace(void *obj)
|
||||||
|
{
|
||||||
|
FreeSpace *freespace = (FreeSpace *)obj;
|
||||||
|
Panel *panel = (Panel *)freespace->area.panel;
|
||||||
|
if (!freespace->area.on_screen)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
int old_size = panel_horizontal ? freespace->area.width : freespace->area.height;
|
||||||
|
int size = freespace_get_max_size(panel);
|
||||||
|
if (old_size == size)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
freespace->area.width = size;
|
||||||
|
} else {
|
||||||
|
freespace->area.height = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
schedule_redraw(&freespace->area);
|
||||||
|
schedule_panel_redraw();
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
22
src/freespace/freespace.h
Normal file
22
src/freespace/freespace.h
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Copyright (C) 2011 Mishael A Sibiryakov (death@junki.org)
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef FREESPACE_H
|
||||||
|
#define FREESPACE_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "area.h"
|
||||||
|
|
||||||
|
typedef struct FreeSpace {
|
||||||
|
Area area;
|
||||||
|
} FreeSpace;
|
||||||
|
|
||||||
|
struct Panel;
|
||||||
|
|
||||||
|
void cleanup_freespace(struct Panel *panel);
|
||||||
|
void init_freespace_panel(void *panel);
|
||||||
|
|
||||||
|
gboolean resize_freespace(void *obj);
|
||||||
|
|
||||||
|
#endif
|
||||||
309
src/launcher/apps-common.c
Normal file
309
src/launcher/apps-common.c
Normal file
@@ -0,0 +1,309 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Tint2 : .desktop file handling
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
/* http://standards.freedesktop.org/desktop-entry-spec/ */
|
||||||
|
|
||||||
|
#include "apps-common.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "strnatcmp.h"
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <strings.h>
|
||||||
|
|
||||||
|
static gint compare_strings(gconstpointer a, gconstpointer b)
|
||||||
|
{
|
||||||
|
return strnatcasecmp((const char *)a, (const char *)b);
|
||||||
|
}
|
||||||
|
|
||||||
|
int parse_dektop_line(char *line, char **key, char **value)
|
||||||
|
{
|
||||||
|
char *p;
|
||||||
|
int found = 0;
|
||||||
|
*key = line;
|
||||||
|
for (p = line; *p; p++) {
|
||||||
|
if (*p == '=') {
|
||||||
|
*value = p + 1;
|
||||||
|
*p = 0;
|
||||||
|
found = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!found)
|
||||||
|
return 0;
|
||||||
|
if (found && (strlen(*key) == 0 || strlen(*value) == 0))
|
||||||
|
return 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void expand_exec(DesktopEntry *entry, const char *path)
|
||||||
|
{
|
||||||
|
// Expand % in exec
|
||||||
|
// %i -> --icon Icon
|
||||||
|
// %c -> Name
|
||||||
|
// %k -> path
|
||||||
|
if (entry->exec) {
|
||||||
|
char *exec2 = calloc(strlen(entry->exec) + (entry->name ? strlen(entry->name) : 1) +
|
||||||
|
(entry->icon ? strlen(entry->icon) : 1) + 100,
|
||||||
|
1);
|
||||||
|
char *p, *q;
|
||||||
|
// p will never point to an escaped char
|
||||||
|
for (p = entry->exec, q = exec2; *p; p++, q++) {
|
||||||
|
*q = *p; // Copy
|
||||||
|
if (*p == '\\') {
|
||||||
|
p++, q++;
|
||||||
|
// Copy the escaped char
|
||||||
|
if (*p == '%') // For % we delete the backslash, i.e. write % over it
|
||||||
|
q--;
|
||||||
|
*q = *p;
|
||||||
|
if (!*p)
|
||||||
|
break;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (*p == '%') {
|
||||||
|
p++;
|
||||||
|
if (!*p)
|
||||||
|
break;
|
||||||
|
if (*p == 'i' && entry->icon != NULL) {
|
||||||
|
sprintf(q, "--icon '%s'", entry->icon);
|
||||||
|
q += strlen("--icon ''");
|
||||||
|
q += strlen(entry->icon);
|
||||||
|
q--; // To balance the q++ in the for
|
||||||
|
} else if (*p == 'c' && entry->name != NULL) {
|
||||||
|
sprintf(q, "'%s'", entry->name);
|
||||||
|
q += strlen("''");
|
||||||
|
q += strlen(entry->name);
|
||||||
|
q--; // To balance the q++ in the for
|
||||||
|
} else if (*p == 'c') {
|
||||||
|
sprintf(q, "'%s'", path);
|
||||||
|
q += strlen("''");
|
||||||
|
q += strlen(path);
|
||||||
|
q--; // To balance the q++ in the for
|
||||||
|
} else {
|
||||||
|
// We don't care about other expansions
|
||||||
|
q--; // Delete the last % from q
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*q = '\0';
|
||||||
|
free(entry->exec);
|
||||||
|
entry->exec = exec2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean read_desktop_file_full_path(const char *path, DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
entry->name = entry->generic_name = entry->icon = entry->exec = entry->cwd = NULL;
|
||||||
|
entry->hidden_from_menus = FALSE;
|
||||||
|
|
||||||
|
FILE *fp = fopen(path, "rt");
|
||||||
|
if (fp == NULL) {
|
||||||
|
fprintf(stderr, "Could not open file %s\n", path);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const gchar **languages = (const gchar **)g_get_language_names();
|
||||||
|
// lang_index is the index of the language for the best Name key in the language vector
|
||||||
|
// lang_index_default is a constant that encodes the Name key without a language
|
||||||
|
int lang_index_default = 1;
|
||||||
|
#define LANG_DBG 0
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf("Languages:");
|
||||||
|
for (int i = 0; languages[i]; i++) {
|
||||||
|
lang_index_default = i + 1;
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf(" %s", languages[i]);
|
||||||
|
}
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf("\n");
|
||||||
|
// 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_generic_name = lang_index_default + 1;
|
||||||
|
|
||||||
|
gboolean inside_desktop_entry = 0;
|
||||||
|
char *line = NULL;
|
||||||
|
size_t line_size;
|
||||||
|
while (getline(&line, &line_size, fp) >= 0) {
|
||||||
|
int len = strlen(line);
|
||||||
|
if (len == 0)
|
||||||
|
continue;
|
||||||
|
if (line[len - 1] == '\n')
|
||||||
|
line[len - 1] = '\0';
|
||||||
|
if (line[0] == '[') {
|
||||||
|
inside_desktop_entry = (strcmp(line, "[Desktop Entry]") == 0);
|
||||||
|
}
|
||||||
|
char *key, *value;
|
||||||
|
if (inside_desktop_entry && parse_dektop_line(line, &key, &value)) {
|
||||||
|
if (strstr(key, "Name") == key) {
|
||||||
|
if (strcmp(key, "Name") == 0 && lang_index_name > lang_index_default) {
|
||||||
|
entry->name = strdup(value);
|
||||||
|
lang_index_name = lang_index_default;
|
||||||
|
} else {
|
||||||
|
for (int i = 0; languages[i] && i < lang_index_name; i++) {
|
||||||
|
gchar *localized_key = g_strdup_printf("Name[%s]", languages[i]);
|
||||||
|
if (strcmp(key, localized_key) == 0) {
|
||||||
|
if (entry->name)
|
||||||
|
free(entry->name);
|
||||||
|
entry->name = strdup(value);
|
||||||
|
lang_index_name = i;
|
||||||
|
}
|
||||||
|
g_free(localized_key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (strstr(key, "GenericName") == key) {
|
||||||
|
if (strcmp(key, "GenericName") == 0 && lang_index_generic_name > lang_index_default) {
|
||||||
|
entry->generic_name = strdup(value);
|
||||||
|
lang_index_generic_name = lang_index_default;
|
||||||
|
} else {
|
||||||
|
for (int i = 0; languages[i] && i < lang_index_generic_name; i++) {
|
||||||
|
gchar *localized_key = g_strdup_printf("GenericName[%s]", languages[i]);
|
||||||
|
if (strcmp(key, localized_key) == 0) {
|
||||||
|
if (entry->generic_name)
|
||||||
|
free(entry->generic_name);
|
||||||
|
entry->generic_name = strdup(value);
|
||||||
|
lang_index_generic_name = i;
|
||||||
|
}
|
||||||
|
g_free(localized_key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (!entry->exec && strcmp(key, "Exec") == 0) {
|
||||||
|
entry->exec = strdup(value);
|
||||||
|
} else if (!entry->cwd && strcmp(key, "Path") == 0) {
|
||||||
|
entry->cwd = strdup(value);
|
||||||
|
} else if (!entry->icon && strcmp(key, "Icon") == 0) {
|
||||||
|
entry->icon = strdup(value);
|
||||||
|
} else if (strcmp(key, "NoDisplay") == 0) {
|
||||||
|
entry->hidden_from_menus = strcasecmp(value, "true") == 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose(fp);
|
||||||
|
// From this point:
|
||||||
|
// entry->name, entry->generic_name, entry->icon, entry->exec will never be empty strings (can be NULL though)
|
||||||
|
|
||||||
|
expand_exec(entry, entry->path);
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
return entry->exec != NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean read_desktop_file_from_dir(const char *path, const char *file_name, DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
gchar *full_path = g_build_filename(path, file_name, NULL);
|
||||||
|
if (read_desktop_file_full_path(full_path, entry)) {
|
||||||
|
g_free(full_path);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
free_and_null(entry->name);
|
||||||
|
free_and_null(entry->generic_name);
|
||||||
|
free_and_null(entry->icon);
|
||||||
|
free_and_null(entry->exec);
|
||||||
|
free_and_null(entry->cwd);
|
||||||
|
|
||||||
|
GList *subdirs = NULL;
|
||||||
|
|
||||||
|
GDir *d = g_dir_open(path, 0, NULL);
|
||||||
|
if (d) {
|
||||||
|
const gchar *name;
|
||||||
|
while ((name = g_dir_read_name(d))) {
|
||||||
|
gchar *child = g_build_filename(path, name, NULL);
|
||||||
|
if (g_file_test(child, G_FILE_TEST_IS_DIR)) {
|
||||||
|
subdirs = g_list_append(subdirs, child);
|
||||||
|
} else {
|
||||||
|
g_free(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
g_dir_close(d);
|
||||||
|
}
|
||||||
|
|
||||||
|
subdirs = g_list_sort(subdirs, compare_strings);
|
||||||
|
gboolean found = FALSE;
|
||||||
|
for (GList *l = subdirs; l; l = g_list_next(l)) {
|
||||||
|
if (read_desktop_file_from_dir(l->data, file_name, entry)) {
|
||||||
|
found = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (GList *l = subdirs; l; l = g_list_next(l)) {
|
||||||
|
g_free(l->data);
|
||||||
|
}
|
||||||
|
g_list_free(subdirs);
|
||||||
|
g_free(full_path);
|
||||||
|
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean read_desktop_file(const char *path, DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
entry->path = strdup(path);
|
||||||
|
entry->name = entry->generic_name = entry->icon = entry->exec = entry->cwd = NULL;
|
||||||
|
|
||||||
|
if (strchr(path, '/'))
|
||||||
|
return read_desktop_file_full_path(path, entry);
|
||||||
|
for (const GSList *location = get_apps_locations(); location; location = g_slist_next(location)) {
|
||||||
|
if (read_desktop_file_from_dir(location->data, path, entry))
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_desktop_entry(DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
free_and_null(entry->name);
|
||||||
|
free_and_null(entry->generic_name);
|
||||||
|
free_and_null(entry->icon);
|
||||||
|
free_and_null(entry->exec);
|
||||||
|
free_and_null(entry->path);
|
||||||
|
free_and_null(entry->cwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_read_desktop_file()
|
||||||
|
{
|
||||||
|
fprintf(stdout, "\033[1;33m");
|
||||||
|
DesktopEntry 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(stdout, "\033[0m");
|
||||||
|
}
|
||||||
|
|
||||||
|
GSList *apps_locations = NULL;
|
||||||
|
// Do not free the result.
|
||||||
|
const GSList *get_apps_locations()
|
||||||
|
{
|
||||||
|
if (apps_locations)
|
||||||
|
return apps_locations;
|
||||||
|
|
||||||
|
apps_locations = load_locations_from_env(apps_locations, "XDG_DATA_HOME", "applications", NULL);
|
||||||
|
|
||||||
|
apps_locations =
|
||||||
|
g_slist_append(apps_locations, g_build_filename(g_get_home_dir(), ".local/share/applications", NULL));
|
||||||
|
|
||||||
|
apps_locations = load_locations_from_env(apps_locations, "XDG_DATA_DIRS", "applications", NULL);
|
||||||
|
|
||||||
|
apps_locations = g_slist_append(apps_locations, g_strdup("/usr/local/share/applications"));
|
||||||
|
apps_locations = g_slist_append(apps_locations, g_strdup("/usr/share/applications"));
|
||||||
|
apps_locations = g_slist_append(apps_locations, g_strdup("/opt/share/applications"));
|
||||||
|
|
||||||
|
apps_locations = slist_remove_duplicates(apps_locations, g_str_equal, g_free);
|
||||||
|
|
||||||
|
return apps_locations;
|
||||||
|
}
|
||||||
39
src/launcher/apps-common.h
Normal file
39
src/launcher/apps-common.h
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
*
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef APPS_COMMON_H
|
||||||
|
#define APPS_COMMON_H
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
typedef struct DesktopEntry {
|
||||||
|
char *name;
|
||||||
|
char *generic_name;
|
||||||
|
char *exec;
|
||||||
|
char *icon;
|
||||||
|
char *path;
|
||||||
|
char *cwd;
|
||||||
|
gboolean hidden_from_menus;
|
||||||
|
} DesktopEntry;
|
||||||
|
|
||||||
|
// Parses a line of the form "key = value". Modifies the line.
|
||||||
|
// Returns 1 if successful, and parts are not empty.
|
||||||
|
// Key and value point to the parts.
|
||||||
|
int parse_dektop_line(char *line, char **key, char **value);
|
||||||
|
|
||||||
|
// Reads the .desktop file from the given path into the DesktopEntry entry.
|
||||||
|
// The DesktopEntry object must be initially empty.
|
||||||
|
// Returns 1 if successful.
|
||||||
|
gboolean read_desktop_file(const char *path, DesktopEntry *entry);
|
||||||
|
|
||||||
|
// Empties DesktopEntry: releases the memory of the *members* of entry.
|
||||||
|
void free_desktop_entry(DesktopEntry *entry);
|
||||||
|
|
||||||
|
// Returns a list of the directories used to store desktop files.
|
||||||
|
// Do not free the result, it is cached.
|
||||||
|
const GSList *get_apps_locations();
|
||||||
|
|
||||||
|
#endif
|
||||||
782
src/launcher/icon-theme-common.c
Normal file
782
src/launcher/icon-theme-common.c
Normal file
@@ -0,0 +1,782 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Tint2 : Icon theme handling
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
/* http://standards.freedesktop.org/icon-theme-spec/ */
|
||||||
|
|
||||||
|
#include "icon-theme-common.h"
|
||||||
|
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "apps-common.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "cache.h"
|
||||||
|
|
||||||
|
#define ICON_DIR_TYPE_SCALABLE 0
|
||||||
|
#define ICON_DIR_TYPE_FIXED 1
|
||||||
|
#define ICON_DIR_TYPE_THRESHOLD 2
|
||||||
|
typedef struct IconThemeDir {
|
||||||
|
char *name;
|
||||||
|
int size;
|
||||||
|
int type;
|
||||||
|
int max_size;
|
||||||
|
int min_size;
|
||||||
|
int threshold;
|
||||||
|
} IconThemeDir;
|
||||||
|
|
||||||
|
int parse_theme_line(char *line, char **key, char **value)
|
||||||
|
{
|
||||||
|
return parse_dektop_line(line, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
GSList *icon_locations = NULL;
|
||||||
|
// Do not free the result.
|
||||||
|
const GSList *get_icon_locations()
|
||||||
|
{
|
||||||
|
if (icon_locations)
|
||||||
|
return icon_locations;
|
||||||
|
|
||||||
|
icon_locations = load_locations_from_env(icon_locations, "XDG_DATA_HOME", ".icons", NULL);
|
||||||
|
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_build_filename(g_get_home_dir(), ".icons", NULL));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_build_filename(g_get_home_dir(), ".local/share/icons", NULL));
|
||||||
|
|
||||||
|
icon_locations = load_locations_from_env(icon_locations, "XDG_DATA_DIRS", ".icons", ".pixmaps", NULL);
|
||||||
|
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/usr/local/share/icons"));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/usr/local/share/pixmaps"));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/usr/share/icons"));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/usr/share/pixmaps"));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/opt/share/icons"));
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup("/opt/share/pixmaps"));
|
||||||
|
|
||||||
|
icon_locations = slist_remove_duplicates(icon_locations, g_str_equal, g_free);
|
||||||
|
|
||||||
|
return icon_locations;
|
||||||
|
}
|
||||||
|
|
||||||
|
IconTheme *make_theme(const char *name)
|
||||||
|
{
|
||||||
|
IconTheme *theme = calloc(1, sizeof(IconTheme));
|
||||||
|
theme->name = strdup(name);
|
||||||
|
theme->list_inherits = NULL;
|
||||||
|
theme->list_directories = NULL;
|
||||||
|
return theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Use UTF8 when parsing the file
|
||||||
|
IconTheme *load_theme_from_index(const char *file_name, const char *name)
|
||||||
|
{
|
||||||
|
IconTheme *theme;
|
||||||
|
FILE *f;
|
||||||
|
char *line = NULL;
|
||||||
|
size_t line_size;
|
||||||
|
|
||||||
|
if ((f = fopen(file_name, "rt")) == NULL) {
|
||||||
|
fprintf(stderr, "Could not open theme '%s'\n", file_name);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
theme = make_theme(name);
|
||||||
|
|
||||||
|
IconThemeDir *current_dir = NULL;
|
||||||
|
int inside_header = 1;
|
||||||
|
while (getline(&line, &line_size, f) >= 0) {
|
||||||
|
char *key, *value;
|
||||||
|
|
||||||
|
int line_len = strlen(line);
|
||||||
|
if (line_len >= 1) {
|
||||||
|
if (line[line_len - 1] == '\n') {
|
||||||
|
line[line_len - 1] = '\0';
|
||||||
|
line_len--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (line_len == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (inside_header) {
|
||||||
|
if (parse_theme_line(line, &key, &value)) {
|
||||||
|
if (strcmp(key, "Inherits") == 0) {
|
||||||
|
// value is like oxygen,wood,default
|
||||||
|
char *token;
|
||||||
|
token = strtok(value, ",\n");
|
||||||
|
while (token != NULL) {
|
||||||
|
theme->list_inherits = g_slist_append(theme->list_inherits, strdup(token));
|
||||||
|
token = strtok(NULL, ",\n");
|
||||||
|
}
|
||||||
|
} else if (strcmp(key, "Directories") == 0) {
|
||||||
|
// value is like 48x48/apps,48x48/mimetypes,32x32/apps,scalable/apps,scalable/mimetypes
|
||||||
|
char *token;
|
||||||
|
token = strtok(value, ",\n");
|
||||||
|
while (token != NULL) {
|
||||||
|
IconThemeDir *dir = calloc(1, sizeof(IconThemeDir));
|
||||||
|
dir->name = strdup(token);
|
||||||
|
dir->max_size = dir->min_size = dir->size = -1;
|
||||||
|
dir->type = ICON_DIR_TYPE_THRESHOLD;
|
||||||
|
dir->threshold = 2;
|
||||||
|
theme->list_directories = g_slist_append(theme->list_directories, dir);
|
||||||
|
token = strtok(NULL, ",\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (current_dir != NULL) {
|
||||||
|
if (parse_theme_line(line, &key, &value)) {
|
||||||
|
if (strcmp(key, "Size") == 0) {
|
||||||
|
// value is like 24
|
||||||
|
sscanf(value, "%d", ¤t_dir->size);
|
||||||
|
if (current_dir->max_size == -1)
|
||||||
|
current_dir->max_size = current_dir->size;
|
||||||
|
if (current_dir->min_size == -1)
|
||||||
|
current_dir->min_size = current_dir->size;
|
||||||
|
} else if (strcmp(key, "MaxSize") == 0) {
|
||||||
|
// value is like 24
|
||||||
|
sscanf(value, "%d", ¤t_dir->max_size);
|
||||||
|
} else if (strcmp(key, "MinSize") == 0) {
|
||||||
|
// value is like 24
|
||||||
|
sscanf(value, "%d", ¤t_dir->min_size);
|
||||||
|
} else if (strcmp(key, "Threshold") == 0) {
|
||||||
|
// value is like 2
|
||||||
|
sscanf(value, "%d", ¤t_dir->threshold);
|
||||||
|
} else if (strcmp(key, "Type") == 0) {
|
||||||
|
// value is Fixed, Scalable or Threshold : default to scalable for unknown Type.
|
||||||
|
if (strcmp(value, "Fixed") == 0) {
|
||||||
|
current_dir->type = ICON_DIR_TYPE_FIXED;
|
||||||
|
} else if (strcmp(value, "Threshold") == 0) {
|
||||||
|
current_dir->type = ICON_DIR_TYPE_THRESHOLD;
|
||||||
|
} else {
|
||||||
|
current_dir->type = ICON_DIR_TYPE_SCALABLE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (line[0] == '[' && line[line_len - 1] == ']' && strcmp(line, "[Icon Theme]") != 0) {
|
||||||
|
inside_header = 0;
|
||||||
|
current_dir = NULL;
|
||||||
|
line[line_len - 1] = '\0';
|
||||||
|
char *dir_name = line + 1;
|
||||||
|
GSList *dir_item = theme->list_directories;
|
||||||
|
while (dir_item != NULL) {
|
||||||
|
IconThemeDir *dir = dir_item->data;
|
||||||
|
if (strcmp(dir->name, dir_name) == 0) {
|
||||||
|
current_dir = dir;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
dir_item = g_slist_next(dir_item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose(f);
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
return theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_theme_from_fs_dir(IconTheme *theme, const char *dir_name)
|
||||||
|
{
|
||||||
|
gchar *file_name = g_build_filename(dir_name, "index.theme", NULL);
|
||||||
|
if (g_file_test(file_name, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_free(file_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
GDir *d = g_dir_open(dir_name, 0, NULL);
|
||||||
|
if (d) {
|
||||||
|
const gchar *size_name;
|
||||||
|
while ((size_name = g_dir_read_name(d))) {
|
||||||
|
gchar *full_size_name = g_build_filename(dir_name, size_name, NULL);
|
||||||
|
if (g_file_test(file_name, G_FILE_TEST_IS_DIR)) {
|
||||||
|
int size, size2;
|
||||||
|
if ((sscanf(size_name, "%dx%d", &size, &size2) == 2 && size == size2) ||
|
||||||
|
(sscanf(size_name, "%d", &size) == 1)) {
|
||||||
|
GDir *dSize = g_dir_open(full_size_name, 0, NULL);
|
||||||
|
if (dSize) {
|
||||||
|
const gchar *subdir_name;
|
||||||
|
while ((subdir_name = g_dir_read_name(dSize))) {
|
||||||
|
IconThemeDir *dir = calloc(1, sizeof(IconThemeDir));
|
||||||
|
// value is like 48x48/apps
|
||||||
|
gchar *value = g_build_filename(size_name, subdir_name, NULL);
|
||||||
|
dir->name = strdup(value);
|
||||||
|
g_free(value);
|
||||||
|
dir->max_size = dir->min_size = dir->size = size;
|
||||||
|
dir->type = ICON_DIR_TYPE_FIXED;
|
||||||
|
theme->list_directories = g_slist_append(theme->list_directories, dir);
|
||||||
|
}
|
||||||
|
g_dir_close(dSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
g_free(full_size_name);
|
||||||
|
}
|
||||||
|
g_dir_close(d);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
IconTheme *load_theme_from_fs(const char *name, IconTheme *theme)
|
||||||
|
{
|
||||||
|
gchar *dir_name = NULL;
|
||||||
|
for (const GSList *location = get_icon_locations(); location; location = g_slist_next(location)) {
|
||||||
|
gchar *path = (gchar *)location->data;
|
||||||
|
dir_name = g_build_filename(path, name, NULL);
|
||||||
|
if (g_file_test(dir_name, G_FILE_TEST_IS_DIR)) {
|
||||||
|
if (!theme) {
|
||||||
|
theme = make_theme(name);
|
||||||
|
}
|
||||||
|
load_theme_from_fs_dir(theme, dir_name);
|
||||||
|
}
|
||||||
|
g_free(dir_name);
|
||||||
|
dir_name = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
IconTheme *load_theme(const char *name)
|
||||||
|
{
|
||||||
|
// Look for name/index.theme in $HOME/.icons, /usr/share/icons, /usr/share/pixmaps (stop at the first found)
|
||||||
|
// Parse index.theme -> list of IconThemeDir with attributes
|
||||||
|
// Return IconTheme*
|
||||||
|
|
||||||
|
if (name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
gchar *file_name = NULL;
|
||||||
|
for (const GSList *location = get_icon_locations(); location; location = g_slist_next(location)) {
|
||||||
|
gchar *path = (gchar *)location->data;
|
||||||
|
file_name = g_build_filename(path, name, "index.theme", NULL);
|
||||||
|
if (!g_file_test(file_name, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_free(file_name);
|
||||||
|
file_name = NULL;
|
||||||
|
}
|
||||||
|
if (file_name)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
IconTheme *theme = NULL;
|
||||||
|
if (file_name) {
|
||||||
|
theme = load_theme_from_index(file_name, name);
|
||||||
|
g_free(file_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
return load_theme_from_fs(name, theme);
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_icon_theme(IconTheme *theme)
|
||||||
|
{
|
||||||
|
if (!theme)
|
||||||
|
return;
|
||||||
|
free(theme->name);
|
||||||
|
theme->name = NULL;
|
||||||
|
free(theme->description);
|
||||||
|
theme->description = NULL;
|
||||||
|
for (GSList *l_inherits = theme->list_inherits; l_inherits; l_inherits = l_inherits->next) {
|
||||||
|
free(l_inherits->data);
|
||||||
|
}
|
||||||
|
g_slist_free(theme->list_inherits);
|
||||||
|
theme->list_inherits = NULL;
|
||||||
|
for (GSList *l_dir = theme->list_directories; l_dir; l_dir = l_dir->next) {
|
||||||
|
IconThemeDir *dir = (IconThemeDir *)l_dir->data;
|
||||||
|
free(dir->name);
|
||||||
|
free(l_dir->data);
|
||||||
|
}
|
||||||
|
g_slist_free(theme->list_directories);
|
||||||
|
theme->list_directories = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_themes(IconThemeWrapper *wrapper)
|
||||||
|
{
|
||||||
|
if (!wrapper)
|
||||||
|
return;
|
||||||
|
free(wrapper->icon_theme_name);
|
||||||
|
for (GSList *l = wrapper->themes; l; l = l->next) {
|
||||||
|
IconTheme *theme = (IconTheme *)l->data;
|
||||||
|
free_icon_theme(theme);
|
||||||
|
free(theme);
|
||||||
|
}
|
||||||
|
g_slist_free(wrapper->themes);
|
||||||
|
for (GSList *l = wrapper->themes_fallback; l; l = l->next) {
|
||||||
|
IconTheme *theme = (IconTheme *)l->data;
|
||||||
|
free_icon_theme(theme);
|
||||||
|
free(theme);
|
||||||
|
}
|
||||||
|
g_slist_free(wrapper->themes_fallback);
|
||||||
|
g_slist_free_full(wrapper->_queued, free);
|
||||||
|
free_cache(&wrapper->_cache);
|
||||||
|
free(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_launcher_read_theme_file()
|
||||||
|
{
|
||||||
|
fprintf(stdout, "\033[1;33m");
|
||||||
|
IconTheme *theme = load_theme("oxygen");
|
||||||
|
if (!theme) {
|
||||||
|
printf("Could not load theme\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
printf("Loaded theme: %s\n", theme->name);
|
||||||
|
GSList *item = theme->list_inherits;
|
||||||
|
while (item != NULL) {
|
||||||
|
printf("Inherits:%s\n", (char *)item->data);
|
||||||
|
item = g_slist_next(item);
|
||||||
|
}
|
||||||
|
item = theme->list_directories;
|
||||||
|
while (item != NULL) {
|
||||||
|
IconThemeDir *dir = item->data;
|
||||||
|
printf("Dir:%s Size=%d MinSize=%d MaxSize=%d Threshold=%d Type=%s\n",
|
||||||
|
dir->name,
|
||||||
|
dir->size,
|
||||||
|
dir->min_size,
|
||||||
|
dir->max_size,
|
||||||
|
dir->threshold,
|
||||||
|
dir->type == ICON_DIR_TYPE_FIXED ? "Fixed" : dir->type == ICON_DIR_TYPE_SCALABLE
|
||||||
|
? "Scalable"
|
||||||
|
: dir->type == ICON_DIR_TYPE_THRESHOLD ? "Threshold"
|
||||||
|
: "?????");
|
||||||
|
item = g_slist_next(item);
|
||||||
|
}
|
||||||
|
fprintf(stdout, "\033[0m");
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean str_list_contains(const GSList *list, const char *value)
|
||||||
|
{
|
||||||
|
const GSList *item = list;
|
||||||
|
while (item != NULL) {
|
||||||
|
if (g_str_equal(item->data, value)) {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
item = g_slist_next(item);
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_themes_helper(const char *name, GSList **themes, GSList **queued)
|
||||||
|
{
|
||||||
|
if (str_list_contains(*queued, name))
|
||||||
|
return;
|
||||||
|
GSList *queue = g_slist_append(NULL, strdup(name));
|
||||||
|
*queued = g_slist_append(*queued, strdup(name));
|
||||||
|
|
||||||
|
// Load wrapper->themes
|
||||||
|
while (queue) {
|
||||||
|
char *queued_name = queue->data;
|
||||||
|
queue = g_slist_remove(queue, queued_name);
|
||||||
|
|
||||||
|
fprintf(stderr, " '%s',", queued_name);
|
||||||
|
IconTheme *theme = load_theme(queued_name);
|
||||||
|
if (theme != NULL) {
|
||||||
|
*themes = g_slist_append(*themes, theme);
|
||||||
|
|
||||||
|
GSList *item = theme->list_inherits;
|
||||||
|
int pos = 0;
|
||||||
|
while (item != NULL) {
|
||||||
|
char *parent = item->data;
|
||||||
|
if (!str_list_contains(*queued, parent)) {
|
||||||
|
queue = g_slist_insert(queue, strdup(parent), pos);
|
||||||
|
pos++;
|
||||||
|
*queued = g_slist_append(*queued, strdup(parent));
|
||||||
|
}
|
||||||
|
item = g_slist_next(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
free(queued_name);
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
|
||||||
|
// Free the queue
|
||||||
|
GSList *l;
|
||||||
|
for (l = queue; l; l = l->next)
|
||||||
|
free(l->data);
|
||||||
|
g_slist_free(queue);
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_default_theme(IconThemeWrapper *wrapper)
|
||||||
|
{
|
||||||
|
if (wrapper->_themes_loaded)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, GREEN "Loading icon theme %s:" RESET "\n", wrapper->icon_theme_name);
|
||||||
|
|
||||||
|
load_themes_helper(wrapper->icon_theme_name, &wrapper->themes, &wrapper->_queued);
|
||||||
|
load_themes_helper("hicolor", &wrapper->themes, &wrapper->_queued);
|
||||||
|
|
||||||
|
wrapper->_themes_loaded = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_fallbacks(IconThemeWrapper *wrapper)
|
||||||
|
{
|
||||||
|
if (wrapper->_fallback_loaded)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, RED "Loading additional icon themes (this means your icon theme is incomplete)..." RESET "\n");
|
||||||
|
|
||||||
|
// Load wrapper->themes_fallback
|
||||||
|
const GSList *location;
|
||||||
|
for (location = get_icon_locations(); location; location = g_slist_next(location)) {
|
||||||
|
gchar *path = (gchar *)location->data;
|
||||||
|
GDir *d = g_dir_open(path, 0, NULL);
|
||||||
|
if (d) {
|
||||||
|
const gchar *name;
|
||||||
|
while ((name = g_dir_read_name(d))) {
|
||||||
|
gchar *file_name = g_build_filename(path, name, "index.theme", NULL);
|
||||||
|
if (g_file_test(file_name, G_FILE_TEST_EXISTS) && !g_file_test(file_name, G_FILE_TEST_IS_DIR)) {
|
||||||
|
load_themes_helper(name, &wrapper->themes_fallback, &wrapper->_queued);
|
||||||
|
}
|
||||||
|
g_free(file_name);
|
||||||
|
}
|
||||||
|
g_dir_close(d);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper->_fallback_loaded = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gchar *get_icon_cache_path()
|
||||||
|
{
|
||||||
|
return g_build_filename(g_get_user_cache_dir(), "tint2", "icon.cache", NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_icon_cache(IconThemeWrapper *wrapper)
|
||||||
|
{
|
||||||
|
if (wrapper->_cache.loaded)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, GREEN "Loading icon theme cache..." RESET "\n");
|
||||||
|
|
||||||
|
gchar *cache_path = get_icon_cache_path();
|
||||||
|
load_cache(&wrapper->_cache, cache_path);
|
||||||
|
g_free(cache_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void save_icon_cache(IconThemeWrapper *wrapper)
|
||||||
|
{
|
||||||
|
if (!wrapper || !wrapper->_cache.dirty)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, GREEN "Saving icon theme cache..." RESET "\n");
|
||||||
|
gchar *cache_path = get_icon_cache_path();
|
||||||
|
save_cache(&wrapper->_cache, cache_path);
|
||||||
|
g_free(cache_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
IconThemeWrapper *load_themes(const char *icon_theme_name)
|
||||||
|
{
|
||||||
|
IconThemeWrapper *wrapper = calloc(1, sizeof(IconThemeWrapper));
|
||||||
|
|
||||||
|
if (!icon_theme_name) {
|
||||||
|
fprintf(stderr, "Missing icon_theme_name theme, default to 'hicolor'.\n");
|
||||||
|
icon_theme_name = "hicolor";
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper->icon_theme_name = strdup(icon_theme_name);
|
||||||
|
|
||||||
|
return wrapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
int directory_matches_size(IconThemeDir *dir, int size)
|
||||||
|
{
|
||||||
|
if (dir->type == ICON_DIR_TYPE_FIXED) {
|
||||||
|
return dir->size == size;
|
||||||
|
} else if (dir->type == ICON_DIR_TYPE_SCALABLE) {
|
||||||
|
return dir->min_size <= size && size <= dir->max_size;
|
||||||
|
} else /*if (dir->type == ICON_DIR_TYPE_THRESHOLD)*/ {
|
||||||
|
return dir->size - dir->threshold <= size && size <= dir->size + dir->threshold;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int directory_size_distance(IconThemeDir *dir, int size)
|
||||||
|
{
|
||||||
|
if (dir->type == ICON_DIR_TYPE_FIXED) {
|
||||||
|
return abs(dir->size - size);
|
||||||
|
} else if (dir->type == ICON_DIR_TYPE_SCALABLE) {
|
||||||
|
if (size < dir->min_size) {
|
||||||
|
return dir->min_size - size;
|
||||||
|
} else if (size > dir->max_size) {
|
||||||
|
return size - dir->max_size;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
} else /*if (dir->type == ICON_DIR_TYPE_THRESHOLD)*/ {
|
||||||
|
if (size < dir->size - dir->threshold) {
|
||||||
|
return dir->min_size - size;
|
||||||
|
} else if (size > dir->size + dir->threshold) {
|
||||||
|
return size - dir->max_size;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gint compare_theme_directories(gconstpointer a, gconstpointer b, gpointer size_query)
|
||||||
|
{
|
||||||
|
int size = GPOINTER_TO_INT(size_query);
|
||||||
|
const IconThemeDir *da = (const IconThemeDir *)a;
|
||||||
|
const IconThemeDir *db = (const IconThemeDir *)b;
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
if (icon_name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
// If the icon_name is already a path and the file exists, return it
|
||||||
|
if (strstr(icon_name, "/") == icon_name) {
|
||||||
|
if (g_file_test(icon_name, G_FILE_TEST_EXISTS))
|
||||||
|
return strdup(icon_name);
|
||||||
|
else
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
const GSList *basenames = get_icon_locations();
|
||||||
|
GSList *extensions = NULL;
|
||||||
|
extensions = g_slist_append(extensions, ".png");
|
||||||
|
extensions = g_slist_append(extensions, ".xpm");
|
||||||
|
#ifdef HAVE_RSVG
|
||||||
|
extensions = g_slist_append(extensions, ".svg");
|
||||||
|
#endif
|
||||||
|
// if the icon name already contains one of the extensions (e.g. vlc.png instead of vlc) add a special entry
|
||||||
|
for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) {
|
||||||
|
char *extension = (char *)ext->data;
|
||||||
|
if (strlen(icon_name) > strlen(extension) &&
|
||||||
|
strcmp(extension, icon_name + strlen(icon_name) - strlen(extension)) == 0) {
|
||||||
|
extensions = g_slist_append(extensions, "");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GSList *theme;
|
||||||
|
|
||||||
|
// Best size match
|
||||||
|
// Contrary to the freedesktop spec, we are not choosing the closest icon in size, but the next larger icon
|
||||||
|
// otherwise the quality is usually crap (for size 22, if you can choose 16 or 32, you're better with 32)
|
||||||
|
// We do fallback to the closest size if we cannot find a larger or equal icon
|
||||||
|
|
||||||
|
// These 3 variables are used for keeping the closest size match
|
||||||
|
int minimal_size = INT_MAX;
|
||||||
|
char *best_file_name = NULL;
|
||||||
|
GSList *best_file_theme = NULL;
|
||||||
|
|
||||||
|
// These 3 variables are used for keeping the next larger match
|
||||||
|
int next_larger_size = -1;
|
||||||
|
char *next_larger = NULL;
|
||||||
|
GSList *next_larger_theme = NULL;
|
||||||
|
|
||||||
|
int file_name_size = 4096;
|
||||||
|
char *file_name = calloc(file_name_size, 1);
|
||||||
|
|
||||||
|
for (theme = themes; theme; theme = g_slist_next(theme)) {
|
||||||
|
((IconTheme *)theme->data)->list_directories =
|
||||||
|
g_slist_sort_with_data(((IconTheme *)theme->data)->list_directories,
|
||||||
|
compare_theme_directories,
|
||||||
|
GINT_TO_POINTER(size));
|
||||||
|
GSList *dir;
|
||||||
|
for (dir = ((IconTheme *)theme->data)->list_directories; dir; dir = g_slist_next(dir)) {
|
||||||
|
// Closest match
|
||||||
|
gboolean possible = directory_size_distance((IconThemeDir *)dir->data, size) < minimal_size &&
|
||||||
|
(!best_file_theme ? TRUE : theme == best_file_theme);
|
||||||
|
// Next larger match
|
||||||
|
possible = possible || (((IconThemeDir *)dir->data)->size >= size &&
|
||||||
|
(next_larger_size == -1 || ((IconThemeDir *)dir->data)->size < next_larger_size) &&
|
||||||
|
(!next_larger_theme ? 1 : theme == next_larger_theme));
|
||||||
|
if (!possible)
|
||||||
|
continue;
|
||||||
|
const GSList *base;
|
||||||
|
for (base = basenames; base; base = g_slist_next(base)) {
|
||||||
|
for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) {
|
||||||
|
char *base_name = (char *)base->data;
|
||||||
|
char *theme_name = ((IconTheme *)theme->data)->name;
|
||||||
|
char *dir_name = ((IconThemeDir *)dir->data)->name;
|
||||||
|
char *extension = (char *)ext->data;
|
||||||
|
if (strlen(base_name) + strlen(theme_name) + strlen(dir_name) + strlen(icon_name) +
|
||||||
|
strlen(extension) + 100 >
|
||||||
|
file_name_size) {
|
||||||
|
file_name_size = strlen(base_name) + strlen(theme_name) + strlen(dir_name) + strlen(icon_name) +
|
||||||
|
strlen(extension) + 100;
|
||||||
|
file_name = realloc(file_name, file_name_size);
|
||||||
|
}
|
||||||
|
file_name[0] = 0;
|
||||||
|
// filename = directory/$(themename)/subdirectory/iconname.extension
|
||||||
|
sprintf(file_name, "%s/%s/%s/%s%s", base_name, theme_name, dir_name, icon_name, extension);
|
||||||
|
if (DEBUG_ICON_SEARCH)
|
||||||
|
printf("checking %s\n", file_name);
|
||||||
|
if (g_file_test(file_name, G_FILE_TEST_EXISTS)) {
|
||||||
|
if (DEBUG_ICON_SEARCH)
|
||||||
|
printf("found: %s\n", file_name);
|
||||||
|
// Closest match
|
||||||
|
if (directory_size_distance((IconThemeDir *)dir->data, size) < minimal_size &&
|
||||||
|
(!best_file_theme ? 1 : theme == best_file_theme)) {
|
||||||
|
if (best_file_name) {
|
||||||
|
free(best_file_name);
|
||||||
|
best_file_name = NULL;
|
||||||
|
}
|
||||||
|
best_file_name = strdup(file_name);
|
||||||
|
minimal_size = directory_size_distance((IconThemeDir *)dir->data, size);
|
||||||
|
best_file_theme = theme;
|
||||||
|
if (DEBUG_ICON_SEARCH)
|
||||||
|
printf("best_file_name = %s; minimal_size = %d\n", best_file_name, minimal_size);
|
||||||
|
}
|
||||||
|
// Next larger match
|
||||||
|
if (((IconThemeDir *)dir->data)->size >= size &&
|
||||||
|
(next_larger_size == -1 || ((IconThemeDir *)dir->data)->size < next_larger_size) &&
|
||||||
|
(!next_larger_theme ? 1 : theme == next_larger_theme)) {
|
||||||
|
if (next_larger) {
|
||||||
|
free(next_larger);
|
||||||
|
next_larger = NULL;
|
||||||
|
}
|
||||||
|
next_larger = strdup(file_name);
|
||||||
|
next_larger_size = ((IconThemeDir *)dir->data)->size;
|
||||||
|
next_larger_theme = theme;
|
||||||
|
if (DEBUG_ICON_SEARCH)
|
||||||
|
printf("next_larger = %s; next_larger_size = %d\n", next_larger, next_larger_size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(file_name);
|
||||||
|
file_name = NULL;
|
||||||
|
if (next_larger) {
|
||||||
|
g_slist_free(extensions);
|
||||||
|
free(best_file_name);
|
||||||
|
return next_larger;
|
||||||
|
}
|
||||||
|
if (best_file_name) {
|
||||||
|
g_slist_free(extensions);
|
||||||
|
return best_file_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look in unthemed icons
|
||||||
|
{
|
||||||
|
for (const GSList *base = basenames; base; base = g_slist_next(base)) {
|
||||||
|
for (GSList *ext = extensions; ext; ext = g_slist_next(ext)) {
|
||||||
|
char *base_name = (char *)base->data;
|
||||||
|
char *extension = (char *)ext->data;
|
||||||
|
file_name = calloc(strlen(base_name) + strlen(icon_name) + strlen(extension) + 100, 1);
|
||||||
|
// filename = directory/iconname.extension
|
||||||
|
sprintf(file_name, "%s/%s%s", base_name, icon_name, extension);
|
||||||
|
if (DEBUG_ICON_SEARCH)
|
||||||
|
printf("checking %s\n", file_name);
|
||||||
|
if (g_file_test(file_name, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_slist_free(extensions);
|
||||||
|
return file_name;
|
||||||
|
} else {
|
||||||
|
free(file_name);
|
||||||
|
file_name = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
g_slist_free(extensions);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *get_icon_path_from_cache(IconThemeWrapper *wrapper, const char *icon_name, int size)
|
||||||
|
{
|
||||||
|
if (!wrapper || !icon_name || strlen(icon_name) == 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
load_icon_cache(wrapper);
|
||||||
|
|
||||||
|
gchar *key = g_strdup_printf("%s\t%s\t%d", wrapper->icon_theme_name, icon_name, size);
|
||||||
|
const gchar *value = get_from_cache(&wrapper->_cache, key);
|
||||||
|
g_free(key);
|
||||||
|
|
||||||
|
if (!value) {
|
||||||
|
fprintf(stderr,
|
||||||
|
YELLOW "Icon path not found in cache: theme = %s, icon = %s, size = %d" RESET "\n",
|
||||||
|
wrapper->icon_theme_name,
|
||||||
|
icon_name,
|
||||||
|
size);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!g_file_test(value, G_FILE_TEST_EXISTS))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
// fprintf(stderr, "Icon path found in cache: theme = %s, icon = %s, size = %d, path = %s\n",
|
||||||
|
// wrapper->icon_theme_name, icon_name, size, value);
|
||||||
|
|
||||||
|
return strdup(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_icon_path_to_cache(IconThemeWrapper *wrapper, const char *icon_name, int size, const char *path)
|
||||||
|
{
|
||||||
|
if (!wrapper || !icon_name || strlen(icon_name) == 0 || !path || strlen(path) == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"Adding icon path to cache: theme = %s, icon = %s, size = %d, path = %s\n",
|
||||||
|
wrapper->icon_theme_name,
|
||||||
|
icon_name,
|
||||||
|
size,
|
||||||
|
path);
|
||||||
|
|
||||||
|
load_icon_cache(wrapper);
|
||||||
|
|
||||||
|
gchar *key = g_strdup_printf("%s\t%s\t%d", wrapper->icon_theme_name, icon_name, size);
|
||||||
|
add_to_cache(&wrapper->_cache, key, path);
|
||||||
|
g_free(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size, gboolean use_fallbacks)
|
||||||
|
{
|
||||||
|
if (!wrapper)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!icon_name || strlen(icon_name) == 0)
|
||||||
|
goto notfound;
|
||||||
|
|
||||||
|
char *path = get_icon_path_from_cache(wrapper, icon_name, size);
|
||||||
|
if (path)
|
||||||
|
return path;
|
||||||
|
|
||||||
|
load_default_theme(wrapper);
|
||||||
|
|
||||||
|
icon_name = icon_name ? icon_name : DEFAULT_ICON;
|
||||||
|
path = get_icon_path_helper(wrapper->themes, icon_name, size);
|
||||||
|
if (path) {
|
||||||
|
add_icon_path_to_cache(wrapper, icon_name, size, path);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!use_fallbacks)
|
||||||
|
goto notfound;
|
||||||
|
fprintf(stderr, YELLOW "Icon not found in default theme: %s" RESET "\n", icon_name);
|
||||||
|
load_fallbacks(wrapper);
|
||||||
|
|
||||||
|
path = get_icon_path_helper(wrapper->themes_fallback, icon_name, size);
|
||||||
|
if (path) {
|
||||||
|
add_icon_path_to_cache(wrapper, icon_name, size, path);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
notfound:
|
||||||
|
fprintf(stderr, RED "Could not find icon '%s', using default." RESET "\n", icon_name);
|
||||||
|
path = get_icon_path_helper(wrapper->themes, DEFAULT_ICON, size);
|
||||||
|
if (path)
|
||||||
|
return path;
|
||||||
|
path = get_icon_path_helper(wrapper->themes_fallback, DEFAULT_ICON, size);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
60
src/launcher/icon-theme-common.h
Normal file
60
src/launcher/icon-theme-common.h
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef ICON_THEME_COMMON_H
|
||||||
|
#define ICON_THEME_COMMON_H
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
#include "cache.h"
|
||||||
|
|
||||||
|
typedef struct IconThemeWrapper {
|
||||||
|
// The icon theme name for which this wrapper was created
|
||||||
|
char *icon_theme_name;
|
||||||
|
// List of IconTheme*
|
||||||
|
GSList *themes;
|
||||||
|
// Themes are loaded lazily when needed.
|
||||||
|
gboolean _themes_loaded;
|
||||||
|
// List of IconTheme*
|
||||||
|
GSList *themes_fallback;
|
||||||
|
// Fallback themes are loaded lazily when needed.
|
||||||
|
gboolean _fallback_loaded;
|
||||||
|
Cache _cache;
|
||||||
|
// List of icon theme names that have been queued for loading.
|
||||||
|
// Used to avoid loading the same theme twice, and to avoid cycles.
|
||||||
|
GSList *_queued;
|
||||||
|
} IconThemeWrapper;
|
||||||
|
|
||||||
|
typedef struct IconTheme {
|
||||||
|
char *name;
|
||||||
|
char *description;
|
||||||
|
GSList *list_inherits; // each item is a char* (theme name)
|
||||||
|
GSList *list_directories; // each item is an IconThemeDir*
|
||||||
|
} IconTheme;
|
||||||
|
|
||||||
|
// Parses a line of the form "key = value". Modifies the line.
|
||||||
|
// Returns 1 if successful, and parts are not empty.
|
||||||
|
// Key and value point to the parts.
|
||||||
|
int parse_theme_line(char *line, char **key, char **value);
|
||||||
|
|
||||||
|
// Returns an IconThemeWrapper* containing the icon theme identified by the name icon_theme_name, all the
|
||||||
|
// inherited themes, the hicolor theme and possibly fallback themes.
|
||||||
|
IconThemeWrapper *load_themes(const char *icon_theme_name);
|
||||||
|
|
||||||
|
void save_icon_cache(IconThemeWrapper *wrapper);
|
||||||
|
|
||||||
|
void free_themes(IconThemeWrapper *wrapper);
|
||||||
|
void free_icon_theme(IconTheme *theme);
|
||||||
|
|
||||||
|
#define DEFAULT_ICON "application-x-executable"
|
||||||
|
|
||||||
|
// Returns the full path to an icon file (or NULL) given the list of icon themes to search and the icon name
|
||||||
|
// Note: needs to be released with free().
|
||||||
|
char *get_icon_path(IconThemeWrapper *wrapper, const char *icon_name, int size, gboolean use_fallbacks);
|
||||||
|
|
||||||
|
// Returns a list of the directories used to store icons.
|
||||||
|
// Do not free the result, it is cached.
|
||||||
|
const GSList *get_icon_locations();
|
||||||
|
|
||||||
|
#endif
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -10,62 +10,47 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "area.h"
|
#include "area.h"
|
||||||
#include "xsettings-client.h"
|
#include "xsettings-client.h"
|
||||||
|
#include "icon-theme-common.h"
|
||||||
|
|
||||||
|
extern IconThemeWrapper *icon_theme_wrapper;
|
||||||
|
void load_icon_themes();
|
||||||
|
void free_icon_themes();
|
||||||
|
|
||||||
typedef struct Launcher {
|
typedef struct Launcher {
|
||||||
// always start with area
|
// always start with area
|
||||||
Area area;
|
Area area;
|
||||||
GSList *list_apps; // List of char*, each is a path to a app.desktop file
|
GSList *list_apps; // List of char*, each is a path to a app.desktop file
|
||||||
GSList *list_icons; // List of LauncherIcon*
|
GSList *list_icons; // List of LauncherIcon*
|
||||||
GSList *list_themes; // List of IconTheme*
|
int icon_size;
|
||||||
} Launcher;
|
} Launcher;
|
||||||
|
|
||||||
typedef struct LauncherIcon {
|
typedef struct LauncherIcon {
|
||||||
// always start with area
|
// always start with area
|
||||||
Area area;
|
Area area;
|
||||||
Imlib_Image icon_scaled;
|
char *config_path;
|
||||||
Imlib_Image icon_original;
|
Imlib_Image image;
|
||||||
char *cmd;
|
Imlib_Image image_hover;
|
||||||
char *icon_name;
|
Imlib_Image image_pressed;
|
||||||
char *icon_path;
|
char *cmd;
|
||||||
char *icon_tooltip;
|
char *cwd;
|
||||||
int icon_size;
|
char *icon_name;
|
||||||
int is_app_desktop;
|
char *icon_path;
|
||||||
int x, y;
|
char *icon_tooltip;
|
||||||
|
int icon_size;
|
||||||
|
int x, y;
|
||||||
} LauncherIcon;
|
} LauncherIcon;
|
||||||
|
|
||||||
typedef struct DesktopEntry {
|
extern gboolean launcher_enabled;
|
||||||
char *name;
|
|
||||||
char *exec;
|
|
||||||
char *icon;
|
|
||||||
} DesktopEntry;
|
|
||||||
|
|
||||||
#define ICON_DIR_TYPE_SCALABLE 0
|
|
||||||
#define ICON_DIR_TYPE_FIXED 1
|
|
||||||
#define ICON_DIR_TYPE_THRESHOLD 2
|
|
||||||
typedef struct IconThemeDir {
|
|
||||||
char *name;
|
|
||||||
int size;
|
|
||||||
int type;
|
|
||||||
int max_size;
|
|
||||||
int min_size;
|
|
||||||
int threshold;
|
|
||||||
char *context;
|
|
||||||
} IconThemeDir;
|
|
||||||
|
|
||||||
typedef struct IconTheme {
|
|
||||||
char *name;
|
|
||||||
GSList *list_inherits; // each item is a char* (theme name)
|
|
||||||
GSList *list_directories; // each item is an IconThemeDir*
|
|
||||||
} IconTheme;
|
|
||||||
|
|
||||||
extern int launcher_enabled;
|
|
||||||
extern int launcher_max_icon_size;
|
extern int launcher_max_icon_size;
|
||||||
extern int launcher_tooltip_enabled;
|
extern int launcher_tooltip_enabled;
|
||||||
extern int launcher_alpha;
|
extern int launcher_alpha;
|
||||||
extern int launcher_saturation;
|
extern int launcher_saturation;
|
||||||
extern int launcher_brightness;
|
extern int launcher_brightness;
|
||||||
extern char *icon_theme_name; // theme name
|
extern char *icon_theme_name_xsettings; // theme name
|
||||||
extern XSettingsClient *xsettings_client;
|
extern char *icon_theme_name_config;
|
||||||
|
extern int launcher_icon_theme_override;
|
||||||
|
extern Background *launcher_icon_bg;
|
||||||
|
extern GList *launcher_icon_gradients;
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_launcher();
|
void default_launcher();
|
||||||
@@ -76,14 +61,13 @@ void init_launcher_panel(void *panel);
|
|||||||
void cleanup_launcher();
|
void cleanup_launcher();
|
||||||
void cleanup_launcher_theme(Launcher *launcher);
|
void cleanup_launcher_theme(Launcher *launcher);
|
||||||
|
|
||||||
int resize_launcher(void *obj);
|
gboolean resize_launcher(void *obj);
|
||||||
void draw_launcher (void *obj, cairo_t *c);
|
void draw_launcher(void *obj, cairo_t *c);
|
||||||
|
void launcher_default_icon_theme_changed();
|
||||||
|
|
||||||
// Populates the list_themes list
|
|
||||||
void launcher_load_themes(Launcher *launcher);
|
|
||||||
// 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);
|
||||||
|
|
||||||
void test_launcher_read_desktop_file();
|
void test_launcher_read_desktop_file();
|
||||||
void test_launcher_read_theme_file();
|
void test_launcher_read_theme_file();
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
||||||
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
*
|
*
|
||||||
* Author: Owen Taylor, Red Hat, Inc.
|
* Author: Owen Taylor, Red Hat, Inc.
|
||||||
@@ -26,455 +26,452 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include <X11/Xlib.h>
|
#include <X11/Xlib.h>
|
||||||
#include <X11/Xmd.h> /* For CARD16 */
|
#include <X11/Xmd.h> /* For CARD16 */
|
||||||
|
|
||||||
#include "xsettings-client.h"
|
#include "xsettings-client.h"
|
||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "panel.h"
|
#include "panel.h"
|
||||||
#include "launcher.h"
|
#include "launcher.h"
|
||||||
|
|
||||||
struct _XSettingsClient
|
struct _XSettingsClient {
|
||||||
{
|
Display *display;
|
||||||
Display *display;
|
int screen;
|
||||||
int screen;
|
XSettingsNotifyFunc notify;
|
||||||
XSettingsNotifyFunc notify;
|
XSettingsWatchFunc watch;
|
||||||
XSettingsWatchFunc watch;
|
void *cb_data;
|
||||||
void *cb_data;
|
|
||||||
|
|
||||||
Window manager_window;
|
Window manager_window;
|
||||||
XSettingsList *settings;
|
XSettingsList *settings;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void xsettings_notify_cb(const char *name, XSettingsAction action, XSettingsSetting *setting, void *data)
|
||||||
void xsettings_notify_cb (const char *name, XSettingsAction action, XSettingsSetting *setting, void *data)
|
|
||||||
{
|
{
|
||||||
//printf("xsettings_notify_cb\n");
|
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") && setting->type == XSETTINGS_TYPE_STRING) {
|
fprintf(stderr, "xsettings: %s = %s\n", name, setting->data.v_string);
|
||||||
if (icon_theme_name) {
|
if (icon_theme_name_xsettings) {
|
||||||
if (strcmp(icon_theme_name, setting->data.v_string) == 0)
|
if (strcmp(icon_theme_name_xsettings, setting->data.v_string) == 0)
|
||||||
return;
|
return;
|
||||||
free(icon_theme_name);
|
free(icon_theme_name_xsettings);
|
||||||
}
|
}
|
||||||
icon_theme_name = strdup(setting->data.v_string);
|
icon_theme_name_xsettings = strdup(setting->data.v_string);
|
||||||
|
default_icon_theme_changed();
|
||||||
int i;
|
} else if (strcmp(name, "Gtk/FontName") == 0 && setting->type == XSETTINGS_TYPE_STRING) {
|
||||||
for (i = 0 ; i < nb_panel ; i++) {
|
fprintf(stderr, "xsettings: %s = %s\n", name, setting->data.v_string);
|
||||||
Launcher *launcher = &panel1[i].launcher;
|
if (default_font) {
|
||||||
cleanup_launcher_theme(launcher);
|
if (strcmp(default_font, setting->data.v_string) == 0)
|
||||||
launcher_load_themes(launcher);
|
return;
|
||||||
launcher_load_icons(launcher);
|
free(default_font);
|
||||||
launcher->area.resize = 1;
|
}
|
||||||
}
|
default_font = strdup(setting->data.v_string);
|
||||||
}
|
default_font_changed();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void notify_changes(XSettingsClient *client, XSettingsList *old_list)
|
||||||
static void notify_changes (XSettingsClient *client, XSettingsList *old_list)
|
|
||||||
{
|
{
|
||||||
XSettingsList *old_iter = old_list;
|
XSettingsList *old_iter = old_list;
|
||||||
XSettingsList *new_iter = client->settings;
|
XSettingsList *new_iter = client->settings;
|
||||||
|
|
||||||
if (!client->notify)
|
if (!client->notify)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
while (old_iter || new_iter) {
|
while (old_iter || new_iter) {
|
||||||
int cmp;
|
int cmp;
|
||||||
|
|
||||||
if (old_iter && new_iter)
|
if (old_iter && new_iter)
|
||||||
cmp = strcmp (old_iter->setting->name, new_iter->setting->name);
|
cmp = strcmp(old_iter->setting->name, new_iter->setting->name);
|
||||||
else if (old_iter)
|
else if (old_iter)
|
||||||
cmp = -1;
|
cmp = -1;
|
||||||
else
|
else
|
||||||
cmp = 1;
|
cmp = 1;
|
||||||
|
|
||||||
if (cmp < 0) {
|
if (cmp < 0) {
|
||||||
client->notify (old_iter->setting->name, XSETTINGS_ACTION_DELETED, NULL, client->cb_data);
|
client->notify(old_iter->setting->name, XSETTINGS_ACTION_DELETED, NULL, client->cb_data);
|
||||||
}
|
} else if (cmp == 0) {
|
||||||
else if (cmp == 0) {
|
if (!xsettings_setting_equal(old_iter->setting, new_iter->setting))
|
||||||
if (!xsettings_setting_equal (old_iter->setting, new_iter->setting))
|
client->notify(old_iter->setting->name, XSETTINGS_ACTION_CHANGED, new_iter->setting, client->cb_data);
|
||||||
client->notify (old_iter->setting->name, XSETTINGS_ACTION_CHANGED, new_iter->setting, client->cb_data);
|
} else {
|
||||||
}
|
client->notify(new_iter->setting->name, XSETTINGS_ACTION_NEW, new_iter->setting, client->cb_data);
|
||||||
else {
|
}
|
||||||
client->notify (new_iter->setting->name, XSETTINGS_ACTION_NEW, new_iter->setting, client->cb_data);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (old_iter)
|
if (old_iter)
|
||||||
old_iter = old_iter->next;
|
old_iter = old_iter->next;
|
||||||
if (new_iter)
|
if (new_iter)
|
||||||
new_iter = new_iter->next;
|
new_iter = new_iter->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ignore_errors(Display *display, XErrorEvent *event)
|
||||||
static int ignore_errors (Display *display, XErrorEvent *event)
|
|
||||||
{
|
{
|
||||||
return True;
|
return True;
|
||||||
}
|
}
|
||||||
|
|
||||||
static char local_byte_order = '\0';
|
static char local_byte_order = '\0';
|
||||||
|
|
||||||
#define BYTES_LEFT(buffer) ((buffer)->data + (buffer)->len - (buffer)->pos)
|
#define BYTES_LEFT(buffer) ((buffer)->data + (buffer)->len - (buffer)->pos)
|
||||||
|
|
||||||
static XSettingsResult fetch_card16 (XSettingsBuffer *buffer, CARD16 *result)
|
static XSettingsResult fetch_card16(XSettingsBuffer *buffer, CARD16 *result)
|
||||||
{
|
{
|
||||||
CARD16 x;
|
CARD16 x;
|
||||||
|
|
||||||
if (BYTES_LEFT (buffer) < 2)
|
if (BYTES_LEFT(buffer) < 2)
|
||||||
return XSETTINGS_ACCESS;
|
return XSETTINGS_ACCESS;
|
||||||
|
|
||||||
x = *(CARD16 *)buffer->pos;
|
x = *(CARD16 *)buffer->pos;
|
||||||
buffer->pos += 2;
|
buffer->pos += 2;
|
||||||
|
|
||||||
if (buffer->byte_order == local_byte_order)
|
if (buffer->byte_order == local_byte_order)
|
||||||
*result = x;
|
*result = x;
|
||||||
else
|
else
|
||||||
*result = (x << 8) | (x >> 8);
|
*result = (x << 8) | (x >> 8);
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
return XSETTINGS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static XSettingsResult fetch_ushort(XSettingsBuffer *buffer, unsigned short *result)
|
||||||
static XSettingsResult fetch_ushort (XSettingsBuffer *buffer, unsigned short *result)
|
|
||||||
{
|
{
|
||||||
CARD16 x;
|
CARD16 x;
|
||||||
XSettingsResult r;
|
XSettingsResult r;
|
||||||
|
|
||||||
r = fetch_card16 (buffer, &x);
|
r = fetch_card16(buffer, &x);
|
||||||
if (r == XSETTINGS_SUCCESS)
|
if (r == XSETTINGS_SUCCESS)
|
||||||
*result = x;
|
*result = x;
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static XSettingsResult fetch_card32(XSettingsBuffer *buffer, CARD32 *result)
|
||||||
static XSettingsResult fetch_card32 (XSettingsBuffer *buffer, CARD32 *result)
|
|
||||||
{
|
{
|
||||||
CARD32 x;
|
CARD32 x;
|
||||||
|
|
||||||
if (BYTES_LEFT (buffer) < 4)
|
if (BYTES_LEFT(buffer) < 4)
|
||||||
return XSETTINGS_ACCESS;
|
return XSETTINGS_ACCESS;
|
||||||
|
|
||||||
x = *(CARD32 *)buffer->pos;
|
x = *(CARD32 *)buffer->pos;
|
||||||
buffer->pos += 4;
|
buffer->pos += 4;
|
||||||
|
|
||||||
if (buffer->byte_order == local_byte_order)
|
if (buffer->byte_order == local_byte_order)
|
||||||
*result = x;
|
*result = x;
|
||||||
else
|
else
|
||||||
*result = (x << 24) | ((x & 0xff00) << 8) | ((x & 0xff0000) >> 8) | (x >> 24);
|
*result = (x << 24) | ((x & 0xff00) << 8) | ((x & 0xff0000) >> 8) | (x >> 24);
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
return XSETTINGS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
static XSettingsResult fetch_card8 (XSettingsBuffer *buffer, CARD8 *result)
|
static XSettingsResult fetch_card8(XSettingsBuffer *buffer, CARD8 *result)
|
||||||
{
|
{
|
||||||
if (BYTES_LEFT (buffer) < 1)
|
if (BYTES_LEFT(buffer) < 1)
|
||||||
return XSETTINGS_ACCESS;
|
return XSETTINGS_ACCESS;
|
||||||
|
|
||||||
*result = *(CARD8 *)buffer->pos;
|
*result = *(CARD8 *)buffer->pos;
|
||||||
buffer->pos += 1;
|
buffer->pos += 1;
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
return XSETTINGS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define XSETTINGS_PAD(n,m) ((n + m - 1) & (~(m-1)))
|
#define XSETTINGS_PAD(n, m) ((n + m - 1) & (~(m - 1)))
|
||||||
|
|
||||||
static XSettingsList *parse_settings (unsigned char *data, size_t len)
|
static XSettingsList *parse_settings(unsigned char *data, size_t len)
|
||||||
{
|
{
|
||||||
XSettingsBuffer buffer;
|
XSettingsBuffer buffer;
|
||||||
XSettingsResult result = XSETTINGS_SUCCESS;
|
XSettingsResult result = XSETTINGS_SUCCESS;
|
||||||
XSettingsList *settings = NULL;
|
XSettingsList *settings = NULL;
|
||||||
CARD32 serial;
|
CARD32 serial;
|
||||||
CARD32 n_entries;
|
CARD32 n_entries;
|
||||||
CARD32 i;
|
CARD32 i;
|
||||||
XSettingsSetting *setting = NULL;
|
XSettingsSetting *setting = NULL;
|
||||||
|
|
||||||
local_byte_order = xsettings_byte_order ();
|
local_byte_order = xsettings_byte_order();
|
||||||
|
|
||||||
buffer.pos = buffer.data = data;
|
buffer.byte_order = local_byte_order;
|
||||||
buffer.len = len;
|
buffer.pos = buffer.data = data;
|
||||||
|
buffer.len = 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, "Invalid byte order %x in XSETTINGS property\n", buffer.byte_order);
|
||||||
result = XSETTINGS_FAILED;
|
result = XSETTINGS_FAILED;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer.pos += 3;
|
buffer.pos += 3;
|
||||||
|
|
||||||
result = fetch_card32 (&buffer, &serial);
|
result = fetch_card32(&buffer, &serial);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
result = fetch_card32 (&buffer, &n_entries);
|
result = fetch_card32(&buffer, &n_entries);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
for (i = 0; i < n_entries; i++) {
|
for (i = 0; i < n_entries; i++) {
|
||||||
CARD8 type;
|
CARD8 type;
|
||||||
CARD16 name_len;
|
CARD16 name_len;
|
||||||
CARD32 v_int;
|
CARD32 v_int;
|
||||||
size_t pad_len;
|
size_t pad_len;
|
||||||
|
|
||||||
result = fetch_card8 (&buffer, &type);
|
result = fetch_card8(&buffer, &type);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
buffer.pos += 1;
|
buffer.pos += 1;
|
||||||
|
|
||||||
result = fetch_card16 (&buffer, &name_len);
|
result = fetch_card16(&buffer, &name_len);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
pad_len = XSETTINGS_PAD(name_len, 4);
|
pad_len = XSETTINGS_PAD(name_len, 4);
|
||||||
if (BYTES_LEFT (&buffer) < pad_len) {
|
if (BYTES_LEFT(&buffer) < pad_len) {
|
||||||
result = XSETTINGS_ACCESS;
|
result = XSETTINGS_ACCESS;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
setting = malloc (sizeof *setting);
|
setting = calloc(1, sizeof *setting);
|
||||||
if (!setting) {
|
if (!setting) {
|
||||||
result = XSETTINGS_NO_MEM;
|
result = XSETTINGS_NO_MEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
setting->type = XSETTINGS_TYPE_INT; /* No allocated memory */
|
setting->type = XSETTINGS_TYPE_INT; /* No allocated memory */
|
||||||
|
|
||||||
setting->name = malloc (name_len + 1);
|
setting->name = calloc(name_len + 1, 1);
|
||||||
if (!setting->name) {
|
if (!setting->name) {
|
||||||
result = XSETTINGS_NO_MEM;
|
result = XSETTINGS_NO_MEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy (setting->name, buffer.pos, name_len);
|
memcpy(setting->name, buffer.pos, name_len);
|
||||||
setting->name[name_len] = '\0';
|
setting->name[name_len] = '\0';
|
||||||
buffer.pos += pad_len;
|
buffer.pos += pad_len;
|
||||||
|
|
||||||
result = fetch_card32 (&buffer, &v_int);
|
result = fetch_card32(&buffer, &v_int);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
setting->last_change_serial = v_int;
|
setting->last_change_serial = v_int;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case XSETTINGS_TYPE_INT:
|
case XSETTINGS_TYPE_INT:
|
||||||
result = fetch_card32 (&buffer, &v_int);
|
result = fetch_card32(&buffer, &v_int);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
setting->data.v_int = (INT32)v_int;
|
setting->data.v_int = (INT32)v_int;
|
||||||
break;
|
break;
|
||||||
case XSETTINGS_TYPE_STRING:
|
case XSETTINGS_TYPE_STRING:
|
||||||
result = fetch_card32 (&buffer, &v_int);
|
result = fetch_card32(&buffer, &v_int);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
pad_len = XSETTINGS_PAD (v_int, 4);
|
pad_len = XSETTINGS_PAD(v_int, 4);
|
||||||
if (v_int + 1 == 0 || /* Guard against wrap-around */
|
if (v_int + 1 == 0 || /* Guard against wrap-around */
|
||||||
BYTES_LEFT (&buffer) < pad_len) {
|
BYTES_LEFT(&buffer) < pad_len) {
|
||||||
result = XSETTINGS_ACCESS;
|
result = XSETTINGS_ACCESS;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
setting->data.v_string = malloc (v_int + 1);
|
setting->data.v_string = calloc(v_int + 1, 1);
|
||||||
if (!setting->data.v_string) {
|
if (!setting->data.v_string) {
|
||||||
result = XSETTINGS_NO_MEM;
|
result = XSETTINGS_NO_MEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy (setting->data.v_string, buffer.pos, v_int);
|
memcpy(setting->data.v_string, buffer.pos, v_int);
|
||||||
setting->data.v_string[v_int] = '\0';
|
setting->data.v_string[v_int] = '\0';
|
||||||
buffer.pos += pad_len;
|
buffer.pos += pad_len;
|
||||||
break;
|
break;
|
||||||
case XSETTINGS_TYPE_COLOR:
|
case XSETTINGS_TYPE_COLOR:
|
||||||
result = fetch_ushort (&buffer, &setting->data.v_color.red);
|
result = fetch_ushort(&buffer, &setting->data.v_color.red);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
result = fetch_ushort (&buffer, &setting->data.v_color.green);
|
result = fetch_ushort(&buffer, &setting->data.v_color.green);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
result = fetch_ushort (&buffer, &setting->data.v_color.blue);
|
result = fetch_ushort(&buffer, &setting->data.v_color.blue);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
result = fetch_ushort (&buffer, &setting->data.v_color.alpha);
|
result = fetch_ushort(&buffer, &setting->data.v_color.alpha);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
/* Quietly ignore unknown types */
|
/* Quietly ignore unknown types */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
setting->type = type;
|
setting->type = type;
|
||||||
|
|
||||||
result = xsettings_list_insert (&settings, setting);
|
result = xsettings_list_insert(&settings, setting);
|
||||||
if (result != XSETTINGS_SUCCESS)
|
if (result != XSETTINGS_SUCCESS)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
setting = NULL;
|
setting = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
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, "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, "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, "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:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (setting)
|
if (setting)
|
||||||
xsettings_setting_free (setting);
|
xsettings_setting_free(setting);
|
||||||
|
|
||||||
xsettings_list_free (settings);
|
xsettings_list_free(settings);
|
||||||
settings = NULL;
|
settings = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
return settings;
|
return settings;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void read_settings(XSettingsClient *client)
|
||||||
static void read_settings (XSettingsClient *client)
|
|
||||||
{
|
{
|
||||||
Atom type;
|
Atom type;
|
||||||
int format;
|
int format;
|
||||||
unsigned long n_items;
|
unsigned long n_items;
|
||||||
unsigned long bytes_after;
|
unsigned long bytes_after;
|
||||||
unsigned char *data;
|
unsigned char *data;
|
||||||
int result;
|
|
||||||
|
|
||||||
int (*old_handler) (Display *, XErrorEvent *);
|
int (*old_handler)(Display *, XErrorEvent *);
|
||||||
|
|
||||||
XSettingsList *old_list = client->settings;
|
XSettingsList *old_list = client->settings;
|
||||||
client->settings = NULL;
|
client->settings = NULL;
|
||||||
|
|
||||||
old_handler = XSetErrorHandler (ignore_errors);
|
old_handler = XSetErrorHandler(ignore_errors);
|
||||||
result = XGetWindowProperty (client->display, client->manager_window, server.atom._XSETTINGS_SETTINGS, 0, LONG_MAX, False, server.atom._XSETTINGS_SETTINGS, &type, &format, &n_items, &bytes_after, &data);
|
int result = XGetWindowProperty(client->display,
|
||||||
XSetErrorHandler (old_handler);
|
client->manager_window,
|
||||||
|
server.atom._XSETTINGS_SETTINGS,
|
||||||
|
0,
|
||||||
|
LONG_MAX,
|
||||||
|
False,
|
||||||
|
server.atom._XSETTINGS_SETTINGS,
|
||||||
|
&type,
|
||||||
|
&format,
|
||||||
|
&n_items,
|
||||||
|
&bytes_after,
|
||||||
|
&data);
|
||||||
|
XSetErrorHandler(old_handler);
|
||||||
|
|
||||||
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, "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);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
notify_changes (client, old_list);
|
notify_changes(client, old_list);
|
||||||
xsettings_list_free (old_list);
|
xsettings_list_free(old_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void check_manager_window(XSettingsClient *client)
|
||||||
static void check_manager_window (XSettingsClient *client)
|
|
||||||
{
|
{
|
||||||
if (client->manager_window && client->watch)
|
if (client->manager_window && client->watch)
|
||||||
client->watch (client->manager_window, False, 0, client->cb_data);
|
client->watch(client->manager_window, False, 0, client->cb_data);
|
||||||
|
|
||||||
XGrabServer (client->display);
|
XGrabServer(client->display);
|
||||||
|
|
||||||
client->manager_window = XGetSelectionOwner (server.dsp, server.atom._XSETTINGS_SCREEN);
|
client->manager_window = XGetSelectionOwner(server.display, server.atom._XSETTINGS_SCREEN);
|
||||||
if (client->manager_window)
|
if (client->manager_window)
|
||||||
XSelectInput (server.dsp, client->manager_window, PropertyChangeMask | StructureNotifyMask);
|
XSelectInput(server.display, client->manager_window, PropertyChangeMask | StructureNotifyMask);
|
||||||
|
|
||||||
XUngrabServer (client->display);
|
XUngrabServer(client->display);
|
||||||
XFlush (client->display);
|
XFlush(client->display);
|
||||||
|
|
||||||
if (client->manager_window && client->watch)
|
if (client->manager_window && client->watch)
|
||||||
client->watch (client->manager_window, True, PropertyChangeMask | StructureNotifyMask, client->cb_data);
|
client->watch(client->manager_window, True, PropertyChangeMask | StructureNotifyMask, client->cb_data);
|
||||||
|
|
||||||
read_settings (client);
|
read_settings(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
XSettingsClient *xsettings_client_new(Display *display,
|
||||||
XSettingsClient *xsettings_client_new (Display *display, int screen, XSettingsNotifyFunc notify, XSettingsWatchFunc watch, void *cb_data)
|
int screen,
|
||||||
|
XSettingsNotifyFunc notify,
|
||||||
|
XSettingsWatchFunc watch,
|
||||||
|
void *cb_data)
|
||||||
{
|
{
|
||||||
XSettingsClient *client;
|
XSettingsClient *client = calloc(1, sizeof *client);
|
||||||
|
if (!client)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
client = malloc (sizeof *client);
|
client->display = display;
|
||||||
if (!client)
|
client->screen = screen;
|
||||||
return NULL;
|
client->notify = notify;
|
||||||
|
client->watch = watch;
|
||||||
|
client->cb_data = cb_data;
|
||||||
|
|
||||||
client->display = display;
|
client->manager_window = None;
|
||||||
client->screen = screen;
|
client->settings = NULL;
|
||||||
client->notify = notify;
|
|
||||||
client->watch = watch;
|
|
||||||
client->cb_data = cb_data;
|
|
||||||
|
|
||||||
client->manager_window = None;
|
if (client->watch)
|
||||||
client->settings = NULL;
|
client->watch(RootWindow(display, screen), True, StructureNotifyMask, client->cb_data);
|
||||||
|
|
||||||
if (client->watch)
|
check_manager_window(client);
|
||||||
client->watch (RootWindow (display, screen), True, StructureNotifyMask, client->cb_data);
|
|
||||||
|
|
||||||
check_manager_window (client);
|
if (client->manager_window == None) {
|
||||||
|
printf("No XSETTINGS manager, tint2 uses config option 'launcher_icon_theme'.\n");
|
||||||
if (client->manager_window == None) {
|
free(client);
|
||||||
printf("NO XSETTINGS manager, tint2 use config 'launcher_icon_theme'.\n");
|
return NULL;
|
||||||
free (client);
|
} else {
|
||||||
return NULL;
|
return client;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
return client;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void xsettings_client_destroy(XSettingsClient *client)
|
||||||
void xsettings_client_destroy (XSettingsClient *client)
|
|
||||||
{
|
{
|
||||||
if (client->watch)
|
if (!client)
|
||||||
client->watch (RootWindow (client->display, client->screen), False, 0, client->cb_data);
|
return;
|
||||||
if (client->manager_window && client->watch)
|
if (client->watch)
|
||||||
client->watch (client->manager_window, False, 0, client->cb_data);
|
client->watch(RootWindow(client->display, client->screen), False, 0, client->cb_data);
|
||||||
|
if (client->manager_window && client->watch)
|
||||||
|
client->watch(client->manager_window, False, 0, client->cb_data);
|
||||||
|
|
||||||
xsettings_list_free (client->settings);
|
xsettings_list_free(client->settings);
|
||||||
free (client);
|
free(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
XSettingsResult xsettings_client_get_setting(XSettingsClient *client, const char *name, XSettingsSetting **setting)
|
||||||
XSettingsResult xsettings_client_get_setting (XSettingsClient *client, const char *name, XSettingsSetting **setting)
|
|
||||||
{
|
{
|
||||||
XSettingsSetting *search = xsettings_list_lookup (client->settings, name);
|
XSettingsSetting *search = xsettings_list_lookup(client->settings, name);
|
||||||
if (search) {
|
if (search) {
|
||||||
*setting = xsettings_setting_copy (search);
|
*setting = xsettings_setting_copy(search);
|
||||||
return *setting ? XSETTINGS_SUCCESS : XSETTINGS_NO_MEM;
|
return *setting ? XSETTINGS_SUCCESS : XSETTINGS_NO_MEM;
|
||||||
}
|
} else
|
||||||
else
|
return XSETTINGS_NO_ENTRY;
|
||||||
return XSETTINGS_NO_ENTRY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Bool xsettings_client_process_event(XSettingsClient *client, XEvent *xev)
|
||||||
Bool xsettings_client_process_event (XSettingsClient *client, XEvent *xev)
|
|
||||||
{
|
{
|
||||||
/* The checks here will not unlikely cause us to reread
|
/* The checks here will not unlikely cause us to reread
|
||||||
* the properties from the manager window a number of
|
* the properties from the manager window a number of
|
||||||
* times when the manager changes from A->B. But manager changes
|
* times when the manager changes from A->B. But manager changes
|
||||||
* are going to be pretty rare.
|
* are going to be pretty rare.
|
||||||
*/
|
*/
|
||||||
if (xev->xany.window == RootWindow (server.dsp, server.screen)) {
|
if (xev->xany.window == RootWindow(server.display, server.screen)) {
|
||||||
if (xev->xany.type == ClientMessage && xev->xclient.message_type == server.atom.MANAGER) {
|
if (xev->xany.type == ClientMessage && xev->xclient.message_type == server.atom.MANAGER) {
|
||||||
check_manager_window (client);
|
check_manager_window(client);
|
||||||
return True;
|
return True;
|
||||||
}
|
}
|
||||||
}
|
} else if (xev->xany.window == client->manager_window) {
|
||||||
else if (xev->xany.window == client->manager_window) {
|
if (xev->xany.type == DestroyNotify) {
|
||||||
if (xev->xany.type == DestroyNotify) {
|
check_manager_window(client);
|
||||||
check_manager_window (client);
|
return True;
|
||||||
return True;
|
} else if (xev->xany.type == PropertyNotify) {
|
||||||
}
|
read_settings(client);
|
||||||
else if (xev->xany.type == PropertyNotify) {
|
return True;
|
||||||
read_settings (client);
|
}
|
||||||
return True;
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return False;
|
return False;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
||||||
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
*
|
*
|
||||||
* Author: Owen Taylor, Red Hat, Inc.
|
* Author: Owen Taylor, Red Hat, Inc.
|
||||||
@@ -32,23 +32,22 @@ extern "C" {
|
|||||||
|
|
||||||
typedef struct _XSettingsClient XSettingsClient;
|
typedef struct _XSettingsClient XSettingsClient;
|
||||||
|
|
||||||
typedef enum
|
typedef enum { XSETTINGS_ACTION_NEW, XSETTINGS_ACTION_CHANGED, XSETTINGS_ACTION_DELETED } XSettingsAction;
|
||||||
{
|
|
||||||
XSETTINGS_ACTION_NEW,
|
|
||||||
XSETTINGS_ACTION_CHANGED,
|
|
||||||
XSETTINGS_ACTION_DELETED
|
|
||||||
} XSettingsAction;
|
|
||||||
|
|
||||||
typedef void (*XSettingsNotifyFunc) (const char *name, XSettingsAction action, XSettingsSetting *setting, void *cb_data);
|
typedef void (*XSettingsNotifyFunc)(const char *name, XSettingsAction action, XSettingsSetting *setting, void *cb_data);
|
||||||
typedef void (*XSettingsWatchFunc) (Window window, Bool is_start, long mask, void *cb_data);
|
typedef void (*XSettingsWatchFunc)(Window window, Bool is_start, long mask, void *cb_data);
|
||||||
|
|
||||||
XSettingsClient *xsettings_client_new (Display *display, int screen, XSettingsNotifyFunc notify, XSettingsWatchFunc watch, void *cb_data);
|
XSettingsClient *xsettings_client_new(Display *display,
|
||||||
void xsettings_client_destroy (XSettingsClient *client);
|
int screen,
|
||||||
Bool xsettings_client_process_event (XSettingsClient *client, XEvent *xev);
|
XSettingsNotifyFunc notify,
|
||||||
|
XSettingsWatchFunc watch,
|
||||||
|
void *cb_data);
|
||||||
|
void xsettings_client_destroy(XSettingsClient *client);
|
||||||
|
Bool xsettings_client_process_event(XSettingsClient *client, XEvent *xev);
|
||||||
|
|
||||||
void xsettings_notify_cb (const char *name, XSettingsAction action, XSettingsSetting *setting, void *data);
|
void xsettings_notify_cb(const char *name, XSettingsAction action, XSettingsSetting *setting, void *data);
|
||||||
|
|
||||||
XSettingsResult xsettings_client_get_setting (XSettingsClient *client, const char *name, XSettingsSetting **setting);
|
XSettingsResult xsettings_client_get_setting(XSettingsClient *client, const char *name, XSettingsSetting **setting);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
||||||
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
*
|
*
|
||||||
* Author: Owen Taylor, Red Hat, Inc.
|
* Author: Owen Taylor, Red Hat, Inc.
|
||||||
@@ -24,245 +24,222 @@
|
|||||||
#include "stdlib.h"
|
#include "stdlib.h"
|
||||||
|
|
||||||
#include <X11/Xlib.h>
|
#include <X11/Xlib.h>
|
||||||
#include <X11/Xmd.h> /* For CARD32 */
|
#include <X11/Xmd.h> /* For CARD32 */
|
||||||
|
|
||||||
#include "xsettings-common.h"
|
#include "xsettings-common.h"
|
||||||
|
|
||||||
XSettingsSetting *
|
XSettingsSetting *xsettings_setting_copy(XSettingsSetting *setting)
|
||||||
xsettings_setting_copy (XSettingsSetting *setting)
|
|
||||||
{
|
{
|
||||||
XSettingsSetting *result;
|
XSettingsSetting *result;
|
||||||
size_t str_len;
|
size_t str_len;
|
||||||
|
|
||||||
result = malloc (sizeof *result);
|
|
||||||
if (!result)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
str_len = strlen (setting->name);
|
result = calloc(1, sizeof *result);
|
||||||
result->name = malloc (str_len + 1);
|
if (!result)
|
||||||
if (!result->name)
|
return NULL;
|
||||||
goto err;
|
|
||||||
|
|
||||||
memcpy (result->name, setting->name, str_len + 1);
|
str_len = strlen(setting->name);
|
||||||
|
result->name = calloc(str_len + 1, 1);
|
||||||
|
if (!result->name)
|
||||||
|
goto err;
|
||||||
|
|
||||||
result->type = setting->type;
|
memcpy(result->name, setting->name, str_len + 1);
|
||||||
|
|
||||||
switch (setting->type)
|
result->type = setting->type;
|
||||||
{
|
|
||||||
|
switch (setting->type) {
|
||||||
case XSETTINGS_TYPE_INT:
|
case XSETTINGS_TYPE_INT:
|
||||||
result->data.v_int = setting->data.v_int;
|
result->data.v_int = setting->data.v_int;
|
||||||
break;
|
break;
|
||||||
case XSETTINGS_TYPE_COLOR:
|
case XSETTINGS_TYPE_COLOR:
|
||||||
result->data.v_color = setting->data.v_color;
|
result->data.v_color = setting->data.v_color;
|
||||||
break;
|
break;
|
||||||
case XSETTINGS_TYPE_STRING:
|
case XSETTINGS_TYPE_STRING:
|
||||||
str_len = strlen (setting->data.v_string);
|
str_len = strlen(setting->data.v_string);
|
||||||
result->data.v_string = malloc (str_len + 1);
|
result->data.v_string = calloc(str_len + 1, 1);
|
||||||
if (!result->data.v_string)
|
if (!result->data.v_string)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
memcpy (result->data.v_string, setting->data.v_string, str_len + 1);
|
memcpy(result->data.v_string, setting->data.v_string, str_len + 1);
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
result->last_change_serial = setting->last_change_serial;
|
|
||||||
|
|
||||||
return result;
|
|
||||||
|
|
||||||
err:
|
|
||||||
if (result->name)
|
|
||||||
free (result->name);
|
|
||||||
free (result);
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
XSettingsList *
|
|
||||||
xsettings_list_copy (XSettingsList *list)
|
|
||||||
{
|
|
||||||
XSettingsList *new = NULL;
|
|
||||||
XSettingsList *old_iter = list;
|
|
||||||
XSettingsList *new_iter = NULL;
|
|
||||||
|
|
||||||
while (old_iter)
|
|
||||||
{
|
|
||||||
XSettingsList *new_node;
|
|
||||||
|
|
||||||
new_node = malloc (sizeof *new_node);
|
|
||||||
if (!new_node)
|
|
||||||
goto error;
|
|
||||||
|
|
||||||
new_node->setting = xsettings_setting_copy (old_iter->setting);
|
|
||||||
if (!new_node->setting)
|
|
||||||
{
|
|
||||||
free (new_node);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (new_iter)
|
|
||||||
new_iter->next = new_node;
|
|
||||||
else
|
|
||||||
new = new_node;
|
|
||||||
|
|
||||||
new_iter = new_node;
|
|
||||||
|
|
||||||
old_iter = old_iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return new;
|
|
||||||
|
|
||||||
error:
|
|
||||||
xsettings_list_free (new);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
xsettings_setting_equal (XSettingsSetting *setting_a,
|
|
||||||
XSettingsSetting *setting_b)
|
|
||||||
{
|
|
||||||
if (setting_a->type != setting_b->type)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (strcmp (setting_a->name, setting_b->name) != 0)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
switch (setting_a->type)
|
|
||||||
{
|
|
||||||
case XSETTINGS_TYPE_INT:
|
|
||||||
return setting_a->data.v_int == setting_b->data.v_int;
|
|
||||||
case XSETTINGS_TYPE_COLOR:
|
|
||||||
return (setting_a->data.v_color.red == setting_b->data.v_color.red &&
|
|
||||||
setting_a->data.v_color.green == setting_b->data.v_color.green &&
|
|
||||||
setting_a->data.v_color.blue == setting_b->data.v_color.blue &&
|
|
||||||
setting_a->data.v_color.alpha == setting_b->data.v_color.alpha);
|
|
||||||
case XSETTINGS_TYPE_STRING:
|
|
||||||
return strcmp (setting_a->data.v_string, setting_b->data.v_string) == 0;
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
result->last_change_serial = setting->last_change_serial;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
|
||||||
|
err:
|
||||||
|
if (result->name)
|
||||||
|
free(result->name);
|
||||||
|
free(result);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
XSettingsList *xsettings_list_copy(XSettingsList *list)
|
||||||
xsettings_setting_free (XSettingsSetting *setting)
|
|
||||||
{
|
{
|
||||||
if (setting->type == XSETTINGS_TYPE_STRING)
|
XSettingsList *new = NULL;
|
||||||
free (setting->data.v_string);
|
XSettingsList *old_iter = list;
|
||||||
|
XSettingsList *new_iter = NULL;
|
||||||
|
|
||||||
if (setting->name)
|
while (old_iter) {
|
||||||
free (setting->name);
|
XSettingsList *new_node;
|
||||||
|
|
||||||
free (setting);
|
new_node = calloc(1, sizeof *new_node);
|
||||||
|
if (!new_node)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
new_node->setting = xsettings_setting_copy(old_iter->setting);
|
||||||
|
if (!new_node->setting) {
|
||||||
|
free(new_node);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (new_iter)
|
||||||
|
new_iter->next = new_node;
|
||||||
|
else
|
||||||
|
new = new_node;
|
||||||
|
|
||||||
|
new_iter = new_node;
|
||||||
|
|
||||||
|
old_iter = old_iter->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new;
|
||||||
|
|
||||||
|
error:
|
||||||
|
xsettings_list_free(new);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
int xsettings_setting_equal(XSettingsSetting *setting_a, XSettingsSetting *setting_b)
|
||||||
xsettings_list_free (XSettingsList *list)
|
|
||||||
{
|
{
|
||||||
while (list)
|
if (setting_a->type != setting_b->type)
|
||||||
{
|
return 0;
|
||||||
XSettingsList *next = list->next;
|
|
||||||
|
|
||||||
xsettings_setting_free (list->setting);
|
if (strcmp(setting_a->name, setting_b->name) != 0)
|
||||||
free (list);
|
return 0;
|
||||||
|
|
||||||
list = next;
|
switch (setting_a->type) {
|
||||||
|
case XSETTINGS_TYPE_INT:
|
||||||
|
return setting_a->data.v_int == setting_b->data.v_int;
|
||||||
|
case XSETTINGS_TYPE_COLOR:
|
||||||
|
return (setting_a->data.v_color.red == setting_b->data.v_color.red &&
|
||||||
|
setting_a->data.v_color.green == setting_b->data.v_color.green &&
|
||||||
|
setting_a->data.v_color.blue == setting_b->data.v_color.blue &&
|
||||||
|
setting_a->data.v_color.alpha == setting_b->data.v_color.alpha);
|
||||||
|
case XSETTINGS_TYPE_STRING:
|
||||||
|
return strcmp(setting_a->data.v_string, setting_b->data.v_string) == 0;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void xsettings_setting_free(XSettingsSetting *setting)
|
||||||
|
{
|
||||||
|
if (setting->type == XSETTINGS_TYPE_STRING)
|
||||||
|
free(setting->data.v_string);
|
||||||
|
|
||||||
|
if (setting->name)
|
||||||
|
free(setting->name);
|
||||||
|
|
||||||
|
free(setting);
|
||||||
|
}
|
||||||
|
|
||||||
|
void xsettings_list_free(XSettingsList *list)
|
||||||
|
{
|
||||||
|
while (list) {
|
||||||
|
XSettingsList *next = list->next;
|
||||||
|
|
||||||
|
xsettings_setting_free(list->setting);
|
||||||
|
free(list);
|
||||||
|
|
||||||
|
list = next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
XSettingsResult
|
XSettingsResult xsettings_list_insert(XSettingsList **list, XSettingsSetting *setting)
|
||||||
xsettings_list_insert (XSettingsList **list,
|
|
||||||
XSettingsSetting *setting)
|
|
||||||
{
|
{
|
||||||
XSettingsList *node;
|
XSettingsList *node;
|
||||||
XSettingsList *iter;
|
XSettingsList *iter;
|
||||||
XSettingsList *last = NULL;
|
XSettingsList *last = NULL;
|
||||||
|
|
||||||
node = malloc (sizeof *node);
|
node = calloc(1, sizeof *node);
|
||||||
if (!node)
|
if (!node)
|
||||||
return XSETTINGS_NO_MEM;
|
return XSETTINGS_NO_MEM;
|
||||||
node->setting = setting;
|
node->setting = setting;
|
||||||
|
|
||||||
iter = *list;
|
iter = *list;
|
||||||
while (iter)
|
while (iter) {
|
||||||
{
|
int cmp = strcmp(setting->name, iter->setting->name);
|
||||||
int cmp = strcmp (setting->name, iter->setting->name);
|
|
||||||
|
|
||||||
if (cmp < 0)
|
if (cmp < 0)
|
||||||
break;
|
break;
|
||||||
else if (cmp == 0)
|
else if (cmp == 0) {
|
||||||
{
|
free(node);
|
||||||
free (node);
|
return XSETTINGS_DUPLICATE_ENTRY;
|
||||||
return XSETTINGS_DUPLICATE_ENTRY;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
last = iter;
|
last = iter;
|
||||||
iter = iter->next;
|
iter = iter->next;
|
||||||
}
|
|
||||||
|
|
||||||
if (last)
|
|
||||||
last->next = node;
|
|
||||||
else
|
|
||||||
*list = node;
|
|
||||||
|
|
||||||
node->next = iter;
|
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
XSettingsResult
|
|
||||||
xsettings_list_delete (XSettingsList **list,
|
|
||||||
const char *name)
|
|
||||||
{
|
|
||||||
XSettingsList *iter;
|
|
||||||
XSettingsList *last = NULL;
|
|
||||||
|
|
||||||
iter = *list;
|
|
||||||
while (iter)
|
|
||||||
{
|
|
||||||
if (strcmp (name, iter->setting->name) == 0)
|
|
||||||
{
|
|
||||||
if (last)
|
|
||||||
last->next = iter->next;
|
|
||||||
else
|
|
||||||
*list = iter->next;
|
|
||||||
|
|
||||||
xsettings_setting_free (iter->setting);
|
|
||||||
free (iter);
|
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
last = iter;
|
|
||||||
iter = iter->next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return XSETTINGS_FAILED;
|
if (last)
|
||||||
|
last->next = node;
|
||||||
|
else
|
||||||
|
*list = node;
|
||||||
|
|
||||||
|
node->next = iter;
|
||||||
|
|
||||||
|
return XSETTINGS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
XSettingsSetting *
|
XSettingsResult xsettings_list_delete(XSettingsList **list, const char *name)
|
||||||
xsettings_list_lookup (XSettingsList *list,
|
|
||||||
const char *name)
|
|
||||||
{
|
{
|
||||||
XSettingsList *iter;
|
XSettingsList *iter;
|
||||||
|
XSettingsList *last = NULL;
|
||||||
|
|
||||||
iter = list;
|
iter = *list;
|
||||||
while (iter)
|
while (iter) {
|
||||||
{
|
if (strcmp(name, iter->setting->name) == 0) {
|
||||||
if (strcmp (name, iter->setting->name) == 0)
|
if (last)
|
||||||
return iter->setting;
|
last->next = iter->next;
|
||||||
|
else
|
||||||
|
*list = iter->next;
|
||||||
|
|
||||||
iter = iter->next;
|
xsettings_setting_free(iter->setting);
|
||||||
|
free(iter);
|
||||||
|
|
||||||
|
return XSETTINGS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
last = iter;
|
||||||
|
iter = iter->next;
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL;
|
return XSETTINGS_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
char
|
XSettingsSetting *xsettings_list_lookup(XSettingsList *list, const char *name)
|
||||||
xsettings_byte_order (void)
|
|
||||||
{
|
{
|
||||||
CARD32 myint = 0x01020304;
|
XSettingsList *iter;
|
||||||
return (*(char *)&myint == 1) ? MSBFirst : LSBFirst;
|
|
||||||
|
iter = list;
|
||||||
|
while (iter) {
|
||||||
|
if (strcmp(name, iter->setting->name) == 0)
|
||||||
|
return iter->setting;
|
||||||
|
|
||||||
|
iter = iter->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char xsettings_byte_order(void)
|
||||||
|
{
|
||||||
|
CARD32 myint = 0x01020304;
|
||||||
|
return (*(char *)&myint == 1) ? MSBFirst : LSBFirst;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL RED HAT
|
||||||
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
* BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
*
|
*
|
||||||
* Author: Owen Taylor, Red Hat, Inc.
|
* Author: Owen Taylor, Red Hat, Inc.
|
||||||
@@ -27,82 +27,72 @@
|
|||||||
extern "C" {
|
extern "C" {
|
||||||
#endif /* __cplusplus */
|
#endif /* __cplusplus */
|
||||||
|
|
||||||
typedef struct _XSettingsBuffer XSettingsBuffer;
|
typedef struct _XSettingsBuffer XSettingsBuffer;
|
||||||
typedef struct _XSettingsColor XSettingsColor;
|
typedef struct _XSettingsColor XSettingsColor;
|
||||||
typedef struct _XSettingsList XSettingsList;
|
typedef struct _XSettingsList XSettingsList;
|
||||||
typedef struct _XSettingsSetting XSettingsSetting;
|
typedef struct _XSettingsSetting XSettingsSetting;
|
||||||
|
|
||||||
/* Types of settings possible. Enum values correspond to
|
/* Types of settings possible. Enum values correspond to
|
||||||
* protocol values.
|
* protocol values.
|
||||||
*/
|
*/
|
||||||
typedef enum
|
typedef enum {
|
||||||
{
|
XSETTINGS_TYPE_INT = 0,
|
||||||
XSETTINGS_TYPE_INT = 0,
|
XSETTINGS_TYPE_STRING = 1,
|
||||||
XSETTINGS_TYPE_STRING = 1,
|
XSETTINGS_TYPE_COLOR = 2,
|
||||||
XSETTINGS_TYPE_COLOR = 2,
|
XSETTINGS_TYPE_NONE = 0xff
|
||||||
XSETTINGS_TYPE_NONE = 0xff
|
|
||||||
} XSettingsType;
|
} XSettingsType;
|
||||||
|
|
||||||
typedef enum
|
typedef enum {
|
||||||
{
|
XSETTINGS_SUCCESS,
|
||||||
XSETTINGS_SUCCESS,
|
XSETTINGS_NO_MEM,
|
||||||
XSETTINGS_NO_MEM,
|
XSETTINGS_ACCESS,
|
||||||
XSETTINGS_ACCESS,
|
XSETTINGS_FAILED,
|
||||||
XSETTINGS_FAILED,
|
XSETTINGS_NO_ENTRY,
|
||||||
XSETTINGS_NO_ENTRY,
|
XSETTINGS_DUPLICATE_ENTRY
|
||||||
XSETTINGS_DUPLICATE_ENTRY
|
|
||||||
} XSettingsResult;
|
} XSettingsResult;
|
||||||
|
|
||||||
struct _XSettingsBuffer
|
struct _XSettingsBuffer {
|
||||||
{
|
char byte_order;
|
||||||
char byte_order;
|
size_t len;
|
||||||
size_t len;
|
unsigned char *data;
|
||||||
unsigned char *data;
|
unsigned char *pos;
|
||||||
unsigned char *pos;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _XSettingsColor
|
struct _XSettingsColor {
|
||||||
{
|
unsigned short red, green, blue, alpha;
|
||||||
unsigned short red, green, blue, alpha;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _XSettingsList
|
struct _XSettingsList {
|
||||||
{
|
XSettingsSetting *setting;
|
||||||
XSettingsSetting *setting;
|
XSettingsList *next;
|
||||||
XSettingsList *next;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _XSettingsSetting
|
struct _XSettingsSetting {
|
||||||
{
|
char *name;
|
||||||
char *name;
|
XSettingsType type;
|
||||||
XSettingsType type;
|
|
||||||
|
|
||||||
union {
|
|
||||||
int v_int;
|
|
||||||
char *v_string;
|
|
||||||
XSettingsColor v_color;
|
|
||||||
} data;
|
|
||||||
|
|
||||||
unsigned long last_change_serial;
|
union {
|
||||||
|
int v_int;
|
||||||
|
char *v_string;
|
||||||
|
XSettingsColor v_color;
|
||||||
|
} data;
|
||||||
|
|
||||||
|
unsigned long last_change_serial;
|
||||||
};
|
};
|
||||||
|
|
||||||
XSettingsSetting *xsettings_setting_copy (XSettingsSetting *setting);
|
XSettingsSetting *xsettings_setting_copy(XSettingsSetting *setting);
|
||||||
void xsettings_setting_free (XSettingsSetting *setting);
|
void xsettings_setting_free(XSettingsSetting *setting);
|
||||||
int xsettings_setting_equal (XSettingsSetting *setting_a,
|
int xsettings_setting_equal(XSettingsSetting *setting_a, XSettingsSetting *setting_b);
|
||||||
XSettingsSetting *setting_b);
|
|
||||||
|
|
||||||
void xsettings_list_free (XSettingsList *list);
|
void xsettings_list_free(XSettingsList *list);
|
||||||
XSettingsList *xsettings_list_copy (XSettingsList *list);
|
XSettingsList *xsettings_list_copy(XSettingsList *list);
|
||||||
XSettingsResult xsettings_list_insert (XSettingsList **list,
|
XSettingsResult xsettings_list_insert(XSettingsList **list, XSettingsSetting *setting);
|
||||||
XSettingsSetting *setting);
|
XSettingsSetting *xsettings_list_lookup(XSettingsList *list, const char *name);
|
||||||
XSettingsSetting *xsettings_list_lookup (XSettingsList *list,
|
XSettingsResult xsettings_list_delete(XSettingsList **list, const char *name);
|
||||||
const char *name);
|
|
||||||
XSettingsResult xsettings_list_delete (XSettingsList **list,
|
|
||||||
const char *name);
|
|
||||||
|
|
||||||
char xsettings_byte_order (void);
|
char xsettings_byte_order(void);
|
||||||
|
|
||||||
#define XSETTINGS_PAD(n,m) ((n + m - 1) & (~(m-1)))
|
#define XSETTINGS_PAD(n, m) ((n + m - 1) & (~(m - 1)))
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
|||||||
1618
src/panel.c
1618
src/panel.c
File diff suppressed because it is too large
Load Diff
218
src/panel.h
218
src/panel.h
@@ -20,109 +20,135 @@
|
|||||||
#include "taskbar.h"
|
#include "taskbar.h"
|
||||||
#include "systraybar.h"
|
#include "systraybar.h"
|
||||||
#include "launcher.h"
|
#include "launcher.h"
|
||||||
|
#include "freespace.h"
|
||||||
|
#include "execplugin.h"
|
||||||
|
#include "separator.h"
|
||||||
|
#include "button.h"
|
||||||
|
|
||||||
#ifdef ENABLE_BATTERY
|
#ifdef ENABLE_BATTERY
|
||||||
#include "battery.h"
|
#include "battery.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
extern int signal_pending;
|
extern int signal_pending;
|
||||||
// --------------------------------------------------
|
// --------------------------------------------------
|
||||||
// mouse events
|
// mouse events
|
||||||
extern int mouse_middle;
|
extern MouseAction mouse_left;
|
||||||
extern int mouse_right;
|
extern MouseAction mouse_middle;
|
||||||
extern int mouse_scroll_up;
|
extern MouseAction mouse_right;
|
||||||
extern int mouse_scroll_down;
|
extern MouseAction mouse_scroll_up;
|
||||||
extern int mouse_tilt_left;
|
extern MouseAction mouse_scroll_down;
|
||||||
extern int mouse_tilt_right;
|
extern MouseAction mouse_tilt_left;
|
||||||
|
extern MouseAction mouse_tilt_right;
|
||||||
|
|
||||||
//panel mode
|
// panel mode
|
||||||
enum { SINGLE_DESKTOP=0, MULTI_DESKTOP };
|
typedef enum TaskbarMode {
|
||||||
enum { BOTTOM_LAYER, NORMAL_LAYER, TOP_LAYER };
|
SINGLE_DESKTOP = 0,
|
||||||
extern int panel_mode;
|
MULTI_DESKTOP,
|
||||||
extern int wm_menu;
|
} TaskbarMode;
|
||||||
extern int panel_dock;
|
|
||||||
extern int panel_layer;
|
|
||||||
|
|
||||||
//panel position
|
typedef enum Layer {
|
||||||
enum { LEFT=0x01, RIGHT=0x02, CENTER=0X04, TOP=0X08, BOTTOM=0x10 };
|
BOTTOM_LAYER,
|
||||||
extern int panel_position;
|
NORMAL_LAYER,
|
||||||
extern int panel_horizontal;
|
TOP_LAYER,
|
||||||
|
} Layer;
|
||||||
|
|
||||||
extern int panel_refresh;
|
// panel position
|
||||||
extern int task_dragged;
|
typedef enum PanelPosition {
|
||||||
|
LEFT = 0x01,
|
||||||
|
RIGHT = 0x02,
|
||||||
|
CENTER = 0X04,
|
||||||
|
TOP = 0X08,
|
||||||
|
BOTTOM = 0x10,
|
||||||
|
} PanelPosition;
|
||||||
|
|
||||||
//panel autohide
|
typedef enum Strut {
|
||||||
enum { STRUT_MINIMUM, STRUT_FOLLOW_SIZE, STRUT_NONE };
|
STRUT_MINIMUM,
|
||||||
extern int panel_autohide;
|
STRUT_FOLLOW_SIZE,
|
||||||
|
STRUT_NONE,
|
||||||
|
} Strut;
|
||||||
|
|
||||||
|
extern TaskbarMode taskbar_mode;
|
||||||
|
extern gboolean wm_menu;
|
||||||
|
extern gboolean panel_dock;
|
||||||
|
extern Layer panel_layer;
|
||||||
|
extern char *panel_window_name;
|
||||||
|
extern PanelPosition panel_position;
|
||||||
|
extern gboolean panel_horizontal;
|
||||||
|
extern gboolean panel_refresh;
|
||||||
|
extern gboolean task_dragged;
|
||||||
|
extern gboolean panel_autohide;
|
||||||
extern int panel_autohide_show_timeout;
|
extern int panel_autohide_show_timeout;
|
||||||
extern int panel_autohide_hide_timeout;
|
extern int panel_autohide_hide_timeout;
|
||||||
extern int panel_autohide_height; // for vertical panels this is of course the width
|
extern int panel_autohide_height; // for vertical panels this is of course the width
|
||||||
extern int panel_strut_policy;
|
extern gboolean panel_shrink;
|
||||||
|
extern Strut panel_strut_policy;
|
||||||
extern char *panel_items_order;
|
extern char *panel_items_order;
|
||||||
|
extern int max_tick_urgent;
|
||||||
extern int max_tick_urgent;
|
extern GArray *backgrounds;
|
||||||
|
extern GArray *gradients;
|
||||||
extern GArray* backgrounds;
|
|
||||||
|
|
||||||
extern Imlib_Image default_icon;
|
extern Imlib_Image default_icon;
|
||||||
|
#define DEFAULT_FONT "sans 10"
|
||||||
|
extern char *default_font;
|
||||||
|
extern XSettingsClient *xsettings_client;
|
||||||
|
extern gboolean startup_notifications;
|
||||||
|
extern gboolean debug_geometry;
|
||||||
|
extern gboolean debug_fps;
|
||||||
|
extern gboolean debug_frames;
|
||||||
|
|
||||||
|
typedef struct Panel {
|
||||||
|
Area area;
|
||||||
|
|
||||||
// tint2 use one panel per monitor and one taskbar per desktop.
|
Window main_win;
|
||||||
typedef struct {
|
Pixmap temp_pmap;
|
||||||
// always start with area
|
|
||||||
// area.list own all objects of the panel according to config file
|
|
||||||
Area area;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
// position relative to root window
|
||||||
// panel
|
int posx, posy;
|
||||||
Window main_win;
|
int marginx, marginy;
|
||||||
Pixmap temp_pmap;
|
gboolean fractional_width, fractional_height;
|
||||||
|
int max_size;
|
||||||
|
int monitor;
|
||||||
|
int font_shadow;
|
||||||
|
gboolean mouse_effects;
|
||||||
|
// Mouse effects for icons
|
||||||
|
int mouse_over_alpha;
|
||||||
|
int mouse_over_saturation;
|
||||||
|
int mouse_over_brightness;
|
||||||
|
int mouse_pressed_alpha;
|
||||||
|
int mouse_pressed_saturation;
|
||||||
|
int mouse_pressed_brightness;
|
||||||
|
|
||||||
// position relative to root window
|
// Per-panel parameters and states for Taskbar and Task
|
||||||
int posx, posy;
|
GlobalTaskbar g_taskbar;
|
||||||
int marginx, marginy;
|
GlobalTask g_task;
|
||||||
int pourcentx, pourcenty;
|
|
||||||
// location of the panel (monitor number)
|
|
||||||
int monitor;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
// Array of Taskbar, with num_desktops items
|
||||||
// task and taskbar parameter per panel
|
Taskbar *taskbar;
|
||||||
Global_taskbar g_taskbar;
|
int num_desktops;
|
||||||
Global_task g_task;
|
gboolean taskbarname_has_font;
|
||||||
|
PangoFontDescription *taskbarname_font_desc;
|
||||||
|
|
||||||
// --------------------------------------------------
|
Clock clock;
|
||||||
// taskbar point to the first taskbar in panel.area.list.
|
|
||||||
// number of tasbar == nb_desktop. taskbar[i] is for desktop(i).
|
|
||||||
// taskbar[i] is used to loop over taskbar,
|
|
||||||
// while panel->area.list is used to loop over all panel's objects
|
|
||||||
Taskbar *taskbar;
|
|
||||||
int nb_desktop;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// clock
|
|
||||||
Clock clock;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// battery
|
|
||||||
#ifdef ENABLE_BATTERY
|
#ifdef ENABLE_BATTERY
|
||||||
Battery battery;
|
Battery battery;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Launcher launcher;
|
Launcher launcher;
|
||||||
|
GList *freespace_list;
|
||||||
|
GList *separator_list;
|
||||||
|
GList *execp_list;
|
||||||
|
GList *button_list;
|
||||||
|
|
||||||
// autohide
|
// Autohide
|
||||||
int is_hidden;
|
gboolean is_hidden;
|
||||||
int hidden_width, hidden_height;
|
int hidden_width, hidden_height;
|
||||||
Pixmap hidden_pixmap;
|
Pixmap hidden_pixmap;
|
||||||
timeout* autohide_timeout;
|
timeout *autohide_timeout;
|
||||||
} Panel;
|
} Panel;
|
||||||
|
|
||||||
|
|
||||||
extern Panel panel_config;
|
extern Panel panel_config;
|
||||||
extern Panel *panel1;
|
extern Panel *panels;
|
||||||
extern int nb_panel;
|
extern int num_panels;
|
||||||
|
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_panel();
|
void default_panel();
|
||||||
@@ -135,10 +161,18 @@ void cleanup_panel();
|
|||||||
void init_panel();
|
void init_panel();
|
||||||
|
|
||||||
void init_panel_size_and_position(Panel *panel);
|
void init_panel_size_and_position(Panel *panel);
|
||||||
int resize_panel(void *obj);
|
gboolean resize_panel(void *obj);
|
||||||
|
void render_panel(Panel *panel);
|
||||||
|
void shrink_panel(Panel *panel);
|
||||||
|
void _schedule_panel_redraw(const char *file, const char *function, const int line);
|
||||||
|
#define schedule_panel_redraw() _schedule_panel_redraw(__FILE__, __FUNCTION__, __LINE__)
|
||||||
|
|
||||||
void set_panel_items_order(Panel *p);
|
void set_panel_items_order(Panel *p);
|
||||||
|
void place_panel_all_desktops(Panel *p);
|
||||||
|
void replace_panel_all_desktops(Panel *p);
|
||||||
void set_panel_properties(Panel *p);
|
void set_panel_properties(Panel *p);
|
||||||
|
void set_panel_window_geometry(Panel *panel);
|
||||||
|
void set_panel_layer(Panel *p, Layer layer);
|
||||||
|
|
||||||
// draw background panel
|
// draw background panel
|
||||||
void set_panel_background(Panel *p);
|
void set_panel_background(Panel *p);
|
||||||
@@ -146,17 +180,31 @@ void set_panel_background(Panel *p);
|
|||||||
// detect witch panel
|
// detect witch panel
|
||||||
Panel *get_panel(Window win);
|
Panel *get_panel(Window win);
|
||||||
|
|
||||||
Taskbar *click_taskbar (Panel *panel, int x, int y);
|
Taskbar *click_taskbar(Panel *panel, int x, int y);
|
||||||
Task *click_task (Panel *panel, int x, int y);
|
Task *click_task(Panel *panel, int x, int y);
|
||||||
Launcher *click_launcher (Panel *panel, int x, int y);
|
Launcher *click_launcher(Panel *panel, int x, int y);
|
||||||
LauncherIcon *click_launcher_icon (Panel *panel, int x, int y);
|
LauncherIcon *click_launcher_icon(Panel *panel, int x, int y);
|
||||||
int click_padding(Panel *panel, int x, int y);
|
Clock *click_clock(Panel *panel, int x, int y);
|
||||||
int click_clock(Panel *panel, int x, int y);
|
|
||||||
Area* click_area(Panel *panel, int x, int y);
|
|
||||||
|
|
||||||
void autohide_show(void* p);
|
#ifdef ENABLE_BATTERY
|
||||||
void autohide_hide(void* p);
|
Battery *click_battery(Panel *panel, int x, int y);
|
||||||
void autohide_trigger_show(Panel* p);
|
#endif
|
||||||
void autohide_trigger_hide(Panel* p);
|
|
||||||
|
Area *click_area(Panel *panel, int x, int y);
|
||||||
|
Execp *click_execp(Panel *panel, int x, int y);
|
||||||
|
Button *click_button(Panel *panel, int x, int y);
|
||||||
|
|
||||||
|
void autohide_show(void *p);
|
||||||
|
void autohide_hide(void *p);
|
||||||
|
void autohide_trigger_show(Panel *p);
|
||||||
|
void autohide_trigger_hide(Panel *p);
|
||||||
|
|
||||||
|
const char *get_default_font();
|
||||||
|
|
||||||
|
void default_icon_theme_changed();
|
||||||
|
void default_font_changed();
|
||||||
|
|
||||||
|
void free_icon(Imlib_Image icon);
|
||||||
|
Imlib_Image scale_icon(Imlib_Image original, int icon_size);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
231
src/separator/separator.c
Normal file
231
src/separator/separator.c
Normal file
@@ -0,0 +1,231 @@
|
|||||||
|
// Tint2 : Separator plugin
|
||||||
|
// Author: Oskari Rauta
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <cairo.h>
|
||||||
|
#include <cairo-xlib.h>
|
||||||
|
#include <pango/pangocairo.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
#include "window.h"
|
||||||
|
#include "server.h"
|
||||||
|
#include "panel.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "separator.h"
|
||||||
|
|
||||||
|
int separator_compute_desired_size(void *obj);
|
||||||
|
|
||||||
|
Separator *create_separator()
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)calloc(1, sizeof(Separator));
|
||||||
|
separator->color.rgb[0] = 0.5;
|
||||||
|
separator->color.rgb[1] = 0.5;
|
||||||
|
separator->color.rgb[2] = 0.5;
|
||||||
|
separator->color.alpha = 0.9;
|
||||||
|
separator->style = SEPARATOR_DOTS;
|
||||||
|
separator->thickness = 3;
|
||||||
|
separator->area.paddingxlr = 1;
|
||||||
|
return separator;
|
||||||
|
}
|
||||||
|
|
||||||
|
void destroy_separator(void *obj)
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
remove_area(&separator->area);
|
||||||
|
free_area(&separator->area);
|
||||||
|
free_and_null(separator);
|
||||||
|
}
|
||||||
|
|
||||||
|
gpointer copy_separator(gconstpointer arg, gpointer data)
|
||||||
|
{
|
||||||
|
Separator *old = (Separator *)arg;
|
||||||
|
Separator *copy = (Separator *)calloc(1, sizeof(Separator));
|
||||||
|
memcpy(copy, old, sizeof(Separator));
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_separator()
|
||||||
|
{
|
||||||
|
GList *to_remove = panel_config.separator_list;
|
||||||
|
for (int k = 0; k < strlen(panel_items_order) && to_remove; k++) {
|
||||||
|
if (panel_items_order[k] == ':') {
|
||||||
|
to_remove = to_remove->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (to_remove) {
|
||||||
|
if (to_remove == panel_config.separator_list) {
|
||||||
|
g_list_free_full(to_remove, destroy_separator);
|
||||||
|
panel_config.separator_list = NULL;
|
||||||
|
} else {
|
||||||
|
// Cut panel_config.separator_list
|
||||||
|
if (to_remove->prev)
|
||||||
|
to_remove->prev->next = NULL;
|
||||||
|
to_remove->prev = NULL;
|
||||||
|
// Remove all elements of to_remove and to_remove itself
|
||||||
|
g_list_free_full(to_remove, destroy_separator);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_separator_panel(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
|
||||||
|
// Make sure this is only done once if there are multiple items
|
||||||
|
if (panel->separator_list)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// panel->separator_list is now a copy of the pointer panel_config.separator_list
|
||||||
|
// We make it a deep copy
|
||||||
|
panel->separator_list = g_list_copy_deep(panel_config.separator_list, copy_separator, NULL);
|
||||||
|
|
||||||
|
for (GList *l = panel->separator_list; l; l = l->next) {
|
||||||
|
Separator *separator = (Separator *)l->data;
|
||||||
|
if (!separator->area.bg)
|
||||||
|
separator->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
separator->area.parent = p;
|
||||||
|
separator->area.panel = p;
|
||||||
|
snprintf(separator->area.name, sizeof(separator->area.name), "separator");
|
||||||
|
separator->area.size_mode = LAYOUT_FIXED;
|
||||||
|
separator->area.resize_needed = 1;
|
||||||
|
separator->area.on_screen = TRUE;
|
||||||
|
separator->area._resize = resize_separator;
|
||||||
|
separator->area._compute_desired_size = separator_compute_desired_size;
|
||||||
|
separator->area._draw_foreground = draw_separator;
|
||||||
|
instantiate_area_gradients(&separator->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup_separator()
|
||||||
|
{
|
||||||
|
// Cleanup frontends
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
g_list_free_full(panels[i].separator_list, destroy_separator);
|
||||||
|
panels[i].separator_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup backends
|
||||||
|
g_list_free_full(panel_config.separator_list, destroy_separator);
|
||||||
|
panel_config.separator_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int separator_compute_desired_size(void *obj)
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
if (!separator->area.on_screen)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (panel_horizontal)
|
||||||
|
return separator->thickness + 2 * separator->area.paddingxlr + left_right_border_width(&separator->area);
|
||||||
|
else
|
||||||
|
return separator->thickness + 2 * separator->area.paddingxlr + top_bottom_border_width(&separator->area);
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean resize_separator(void *obj)
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
if (!separator->area.on_screen)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
separator->area.width =
|
||||||
|
separator->thickness + 2 * separator->area.paddingxlr + left_right_border_width(&separator->area);
|
||||||
|
separator->length =
|
||||||
|
separator->area.height - 2 * separator->area.paddingy - top_bottom_border_width(&separator->area);
|
||||||
|
} else {
|
||||||
|
separator->area.height =
|
||||||
|
separator->thickness + 2 * separator->area.paddingxlr + top_bottom_border_width(&separator->area);
|
||||||
|
separator->length =
|
||||||
|
separator->area.width - 2 * separator->area.paddingy - left_right_border_width(&separator->area);
|
||||||
|
}
|
||||||
|
|
||||||
|
schedule_redraw(&separator->area);
|
||||||
|
schedule_panel_redraw();
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_separator_line(void *obj, cairo_t *c);
|
||||||
|
void draw_separator_dots(void *obj, cairo_t *c);
|
||||||
|
|
||||||
|
void draw_separator(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
|
||||||
|
if (separator->style == SEPARATOR_EMPTY)
|
||||||
|
return;
|
||||||
|
else if (separator->style == SEPARATOR_LINE)
|
||||||
|
draw_separator_line(separator, c);
|
||||||
|
else if (separator->style == SEPARATOR_DOTS)
|
||||||
|
draw_separator_dots(separator, c);
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_separator_line(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
|
||||||
|
if (separator->thickness <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
cairo_set_source_rgba(c,
|
||||||
|
separator->color.rgb[0],
|
||||||
|
separator->color.rgb[1],
|
||||||
|
separator->color.rgb[2],
|
||||||
|
separator->color.alpha);
|
||||||
|
cairo_set_line_width(c, separator->thickness);
|
||||||
|
cairo_set_line_cap(c, CAIRO_LINE_CAP_ROUND);
|
||||||
|
if (panel_horizontal) {
|
||||||
|
cairo_move_to(c, separator->area.width / 2.0, separator->area.height / 2.0 - separator->length / 2.0);
|
||||||
|
cairo_line_to(c, separator->area.width / 2.0, separator->area.height / 2.0 + separator->length / 2.0);
|
||||||
|
} else {
|
||||||
|
cairo_move_to(c, separator->area.width / 2.0 - separator->length / 2.0, separator->area.height / 2.0);
|
||||||
|
cairo_line_to(c, separator->area.width / 2.0 + separator->length / 2.0, separator->area.height / 2.0);
|
||||||
|
}
|
||||||
|
cairo_stroke(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_separator_dots(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
const double PI = 3.14159265359;
|
||||||
|
Separator *separator = (Separator *)obj;
|
||||||
|
if (separator->thickness <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
cairo_set_source_rgba(c,
|
||||||
|
separator->color.rgb[0],
|
||||||
|
separator->color.rgb[1],
|
||||||
|
separator->color.rgb[2],
|
||||||
|
separator->color.alpha);
|
||||||
|
cairo_set_line_width(c, 0);
|
||||||
|
|
||||||
|
int num_circles = separator->length / (1.618 * separator->thickness - 1);
|
||||||
|
double spacing = (separator->length - num_circles * separator->thickness) / MAX(1.0, num_circles - 1.0);
|
||||||
|
if (spacing > separator->thickness)
|
||||||
|
num_circles++;
|
||||||
|
spacing = (separator->length - num_circles * separator->thickness) / MAX(1.0, num_circles - 1.0);
|
||||||
|
double offset = (panel_horizontal ? separator->area.height : separator->area.width) / 2.0 - separator->length / 2.0;
|
||||||
|
if (num_circles == 1)
|
||||||
|
offset += spacing / 2.0;
|
||||||
|
for (int i = 0; i < num_circles; i++) {
|
||||||
|
if (panel_horizontal) {
|
||||||
|
cairo_arc(c,
|
||||||
|
separator->area.width / 2.0,
|
||||||
|
offset + separator->thickness / 2.0,
|
||||||
|
separator->thickness / 2.0,
|
||||||
|
0,
|
||||||
|
2 * PI);
|
||||||
|
} else {
|
||||||
|
cairo_arc(c,
|
||||||
|
offset + separator->thickness / 2.0,
|
||||||
|
separator->area.height / 2.0,
|
||||||
|
separator->thickness / 2.0,
|
||||||
|
0,
|
||||||
|
2 * PI);
|
||||||
|
}
|
||||||
|
cairo_stroke_preserve(c);
|
||||||
|
cairo_fill(c);
|
||||||
|
offset += separator->thickness + spacing;
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/separator/separator.h
Normal file
28
src/separator/separator.h
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// Tint2 : Separator
|
||||||
|
// Author: Oskari Rauta <oskari.rauta@gmail.com>
|
||||||
|
|
||||||
|
#ifndef SEPARATOR_H
|
||||||
|
#define SEPARATOR_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "area.h"
|
||||||
|
|
||||||
|
typedef enum SeparatorStyle { SEPARATOR_EMPTY = 0, SEPARATOR_LINE, SEPARATOR_DOTS } SeparatorStyle;
|
||||||
|
|
||||||
|
typedef struct Separator {
|
||||||
|
Area area;
|
||||||
|
SeparatorStyle style;
|
||||||
|
Color color;
|
||||||
|
int thickness;
|
||||||
|
int length;
|
||||||
|
} Separator;
|
||||||
|
|
||||||
|
Separator *create_separator();
|
||||||
|
void destroy_separator(void *obj);
|
||||||
|
void init_separator();
|
||||||
|
void init_separator_panel(void *p);
|
||||||
|
void cleanup_separator();
|
||||||
|
gboolean resize_separator(void *obj);
|
||||||
|
void draw_separator(void *obj, cairo_t *c);
|
||||||
|
|
||||||
|
#endif
|
||||||
827
src/server.c
827
src/server.c
@@ -21,393 +21,588 @@
|
|||||||
#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 <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "task.h"
|
|
||||||
#include "window.h"
|
#include "window.h"
|
||||||
|
|
||||||
void server_catch_error (Display *d, XErrorEvent *ev){}
|
Server server;
|
||||||
|
|
||||||
void server_init_atoms ()
|
gboolean primary_monitor_first = FALSE;
|
||||||
|
|
||||||
|
void server_catch_error(Display *d, XErrorEvent *ev)
|
||||||
{
|
{
|
||||||
server.atom._XROOTPMAP_ID = XInternAtom (server.dsp, "_XROOTPMAP_ID", False);
|
|
||||||
server.atom._XROOTMAP_ID = XInternAtom (server.dsp, "_XROOTMAP_ID", False);
|
|
||||||
server.atom._NET_CURRENT_DESKTOP = XInternAtom (server.dsp, "_NET_CURRENT_DESKTOP", False);
|
|
||||||
server.atom._NET_NUMBER_OF_DESKTOPS = XInternAtom (server.dsp, "_NET_NUMBER_OF_DESKTOPS", False);
|
|
||||||
server.atom._NET_DESKTOP_NAMES = XInternAtom (server.dsp, "_NET_DESKTOP_NAMES", False);
|
|
||||||
server.atom._NET_DESKTOP_GEOMETRY = XInternAtom (server.dsp, "_NET_DESKTOP_GEOMETRY", False);
|
|
||||||
server.atom._NET_DESKTOP_VIEWPORT = XInternAtom (server.dsp, "_NET_DESKTOP_VIEWPORT", False);
|
|
||||||
server.atom._NET_ACTIVE_WINDOW = XInternAtom (server.dsp, "_NET_ACTIVE_WINDOW", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE", False);
|
|
||||||
server.atom._NET_WM_STATE_SKIP_PAGER = XInternAtom (server.dsp, "_NET_WM_STATE_SKIP_PAGER", False);
|
|
||||||
server.atom._NET_WM_STATE_SKIP_TASKBAR = XInternAtom (server.dsp, "_NET_WM_STATE_SKIP_TASKBAR", False);
|
|
||||||
server.atom._NET_WM_STATE_STICKY = XInternAtom (server.dsp, "_NET_WM_STATE_STICKY", False);
|
|
||||||
server.atom._NET_WM_STATE_DEMANDS_ATTENTION = XInternAtom (server.dsp, "_NET_WM_STATE_DEMANDS_ATTENTION", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_DOCK = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_DOCK", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_DESKTOP = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_DESKTOP", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_TOOLBAR = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_TOOLBAR", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_MENU = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_MENU", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_SPLASH = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_SPLASH", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_DIALOG = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_DIALOG", False);
|
|
||||||
server.atom._NET_WM_WINDOW_TYPE_NORMAL = XInternAtom (server.dsp, "_NET_WM_WINDOW_TYPE_NORMAL", False);
|
|
||||||
server.atom._NET_WM_DESKTOP = XInternAtom (server.dsp, "_NET_WM_DESKTOP", False);
|
|
||||||
server.atom.WM_STATE = XInternAtom (server.dsp, "WM_STATE", False);
|
|
||||||
server.atom._NET_WM_STATE = XInternAtom (server.dsp, "_NET_WM_STATE", False);
|
|
||||||
server.atom._NET_WM_STATE_MAXIMIZED_VERT = XInternAtom (server.dsp, "_NET_WM_STATE_MAXIMIZED_VERT", False);
|
|
||||||
server.atom._NET_WM_STATE_MAXIMIZED_HORZ = XInternAtom (server.dsp, "_NET_WM_STATE_MAXIMIZED_HORZ", False);
|
|
||||||
server.atom._NET_WM_STATE_SHADED = XInternAtom (server.dsp, "_NET_WM_STATE_SHADED", False);
|
|
||||||
server.atom._NET_WM_STATE_HIDDEN = XInternAtom (server.dsp, "_NET_WM_STATE_HIDDEN", False);
|
|
||||||
server.atom._NET_WM_STATE_BELOW = XInternAtom (server.dsp, "_NET_WM_STATE_BELOW", False);
|
|
||||||
server.atom._NET_WM_STATE_ABOVE = XInternAtom (server.dsp, "_NET_WM_STATE_ABOVE", False);
|
|
||||||
server.atom._NET_WM_STATE_MODAL = XInternAtom (server.dsp, "_NET_WM_STATE_MODAL", False);
|
|
||||||
server.atom._NET_CLIENT_LIST = XInternAtom (server.dsp, "_NET_CLIENT_LIST", False);
|
|
||||||
server.atom._NET_WM_VISIBLE_NAME = XInternAtom (server.dsp, "_NET_WM_VISIBLE_NAME", False);
|
|
||||||
server.atom._NET_WM_NAME = XInternAtom (server.dsp, "_NET_WM_NAME", False);
|
|
||||||
server.atom._NET_WM_STRUT = XInternAtom (server.dsp, "_NET_WM_STRUT", False);
|
|
||||||
server.atom._NET_WM_ICON = XInternAtom (server.dsp, "_NET_WM_ICON", False);
|
|
||||||
server.atom._NET_WM_ICON_GEOMETRY = XInternAtom(server.dsp, "_NET_WM_ICON_GEOMETRY", False );
|
|
||||||
server.atom._NET_CLOSE_WINDOW = XInternAtom (server.dsp, "_NET_CLOSE_WINDOW", False);
|
|
||||||
server.atom.UTF8_STRING = XInternAtom (server.dsp, "UTF8_STRING", False);
|
|
||||||
server.atom._NET_SUPPORTING_WM_CHECK = XInternAtom (server.dsp, "_NET_SUPPORTING_WM_CHECK", False);
|
|
||||||
server.atom._NET_WM_CM_S0 = XInternAtom (server.dsp, "_NET_WM_CM_S0", False);
|
|
||||||
server.atom._NET_SUPPORTING_WM_CHECK = XInternAtom (server.dsp, "_NET_WM_NAME", False);
|
|
||||||
server.atom._NET_WM_STRUT_PARTIAL = XInternAtom (server.dsp, "_NET_WM_STRUT_PARTIAL", False);
|
|
||||||
server.atom.WM_NAME = XInternAtom(server.dsp, "WM_NAME", False);
|
|
||||||
server.atom.__SWM_VROOT = XInternAtom(server.dsp, "__SWM_VROOT", False);
|
|
||||||
server.atom._MOTIF_WM_HINTS = XInternAtom(server.dsp, "_MOTIF_WM_HINTS", False);
|
|
||||||
server.atom.WM_HINTS = XInternAtom(server.dsp, "WM_HINTS", False);
|
|
||||||
char *name = g_strdup_printf("_XSETTINGS_S%d", DefaultScreen(server.dsp));
|
|
||||||
server.atom._XSETTINGS_SCREEN = XInternAtom(server.dsp, name, False);
|
|
||||||
g_free(name);
|
|
||||||
server.atom._XSETTINGS_SETTINGS = XInternAtom(server.dsp, "_XSETTINGS_SETTINGS", False);
|
|
||||||
|
|
||||||
// systray protocol
|
|
||||||
name = g_strdup_printf("_NET_SYSTEM_TRAY_S%d", DefaultScreen(server.dsp));
|
|
||||||
server.atom._NET_SYSTEM_TRAY_SCREEN = XInternAtom(server.dsp, name, False);
|
|
||||||
g_free(name);
|
|
||||||
server.atom._NET_SYSTEM_TRAY_OPCODE = XInternAtom(server.dsp, "_NET_SYSTEM_TRAY_OPCODE", False);
|
|
||||||
server.atom.MANAGER = XInternAtom(server.dsp, "MANAGER", False);
|
|
||||||
server.atom._NET_SYSTEM_TRAY_MESSAGE_DATA = XInternAtom(server.dsp, "_NET_SYSTEM_TRAY_MESSAGE_DATA", False);
|
|
||||||
server.atom._NET_SYSTEM_TRAY_ORIENTATION = XInternAtom(server.dsp, "_NET_SYSTEM_TRAY_ORIENTATION", False);
|
|
||||||
server.atom._XEMBED = XInternAtom(server.dsp, "_XEMBED", False);
|
|
||||||
server.atom._XEMBED_INFO = XInternAtom(server.dsp, "_XEMBED_INFO", False);
|
|
||||||
|
|
||||||
// drag 'n' drop
|
|
||||||
server.atom.XdndAware = XInternAtom(server.dsp, "XdndAware", False);
|
|
||||||
server.atom.XdndEnter = XInternAtom(server.dsp, "XdndEnter", False);
|
|
||||||
server.atom.XdndPosition = XInternAtom(server.dsp, "XdndPosition", False);
|
|
||||||
server.atom.XdndStatus = XInternAtom(server.dsp, "XdndStatus", False);
|
|
||||||
server.atom.XdndDrop = XInternAtom(server.dsp, "XdndDrop", False);
|
|
||||||
server.atom.XdndLeave = XInternAtom(server.dsp, "XdndLeave", False);
|
|
||||||
server.atom.XdndSelection = XInternAtom(server.dsp, "XdndSelection", False);
|
|
||||||
server.atom.XdndTypeList = XInternAtom(server.dsp, "XdndTypeList", False);
|
|
||||||
server.atom.XdndActionCopy = XInternAtom(server.dsp, "XdndActionCopy", False);
|
|
||||||
server.atom.XdndFinished = XInternAtom(server.dsp, "XdndFinished", False);
|
|
||||||
server.atom.TARGETS = XInternAtom(server.dsp, "TARGETS", False);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void server_init_atoms()
|
||||||
|
{
|
||||||
|
server.atom._XROOTPMAP_ID = XInternAtom(server.display, "_XROOTPMAP_ID", False);
|
||||||
|
server.atom._XROOTMAP_ID = XInternAtom(server.display, "_XROOTMAP_ID", False);
|
||||||
|
server.atom._NET_CURRENT_DESKTOP = XInternAtom(server.display, "_NET_CURRENT_DESKTOP", False);
|
||||||
|
server.atom._NET_NUMBER_OF_DESKTOPS = XInternAtom(server.display, "_NET_NUMBER_OF_DESKTOPS", False);
|
||||||
|
server.atom._NET_DESKTOP_NAMES = XInternAtom(server.display, "_NET_DESKTOP_NAMES", False);
|
||||||
|
server.atom._NET_DESKTOP_GEOMETRY = XInternAtom(server.display, "_NET_DESKTOP_GEOMETRY", False);
|
||||||
|
server.atom._NET_DESKTOP_VIEWPORT = XInternAtom(server.display, "_NET_DESKTOP_VIEWPORT", False);
|
||||||
|
server.atom._NET_WORKAREA = XInternAtom(server.display, "_NET_WORKAREA", False);
|
||||||
|
server.atom._NET_ACTIVE_WINDOW = XInternAtom(server.display, "_NET_ACTIVE_WINDOW", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE", False);
|
||||||
|
server.atom._NET_WM_STATE_SKIP_PAGER = XInternAtom(server.display, "_NET_WM_STATE_SKIP_PAGER", False);
|
||||||
|
server.atom._NET_WM_STATE_SKIP_TASKBAR = XInternAtom(server.display, "_NET_WM_STATE_SKIP_TASKBAR", False);
|
||||||
|
server.atom._NET_WM_STATE_STICKY = XInternAtom(server.display, "_NET_WM_STATE_STICKY", False);
|
||||||
|
server.atom._NET_WM_STATE_DEMANDS_ATTENTION = XInternAtom(server.display, "_NET_WM_STATE_DEMANDS_ATTENTION", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_DOCK = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_DOCK", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_DESKTOP = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_DESKTOP", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_TOOLBAR = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_TOOLBAR", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_MENU = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_MENU", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_SPLASH = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_SPLASH", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_DIALOG = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_DIALOG", False);
|
||||||
|
server.atom._NET_WM_WINDOW_TYPE_NORMAL = XInternAtom(server.display, "_NET_WM_WINDOW_TYPE_NORMAL", False);
|
||||||
|
server.atom._NET_WM_DESKTOP = XInternAtom(server.display, "_NET_WM_DESKTOP", False);
|
||||||
|
server.atom.WM_STATE = XInternAtom(server.display, "WM_STATE", False);
|
||||||
|
server.atom._NET_WM_STATE = XInternAtom(server.display, "_NET_WM_STATE", False);
|
||||||
|
server.atom._NET_WM_STATE_MAXIMIZED_VERT = XInternAtom(server.display, "_NET_WM_STATE_MAXIMIZED_VERT", False);
|
||||||
|
server.atom._NET_WM_STATE_MAXIMIZED_HORZ = XInternAtom(server.display, "_NET_WM_STATE_MAXIMIZED_HORZ", False);
|
||||||
|
server.atom._NET_WM_STATE_SHADED = XInternAtom(server.display, "_NET_WM_STATE_SHADED", False);
|
||||||
|
server.atom._NET_WM_STATE_HIDDEN = XInternAtom(server.display, "_NET_WM_STATE_HIDDEN", False);
|
||||||
|
server.atom._NET_WM_STATE_BELOW = XInternAtom(server.display, "_NET_WM_STATE_BELOW", False);
|
||||||
|
server.atom._NET_WM_STATE_ABOVE = XInternAtom(server.display, "_NET_WM_STATE_ABOVE", False);
|
||||||
|
server.atom._NET_WM_STATE_MODAL = XInternAtom(server.display, "_NET_WM_STATE_MODAL", False);
|
||||||
|
server.atom._NET_CLIENT_LIST = XInternAtom(server.display, "_NET_CLIENT_LIST", False);
|
||||||
|
server.atom._NET_WM_VISIBLE_NAME = XInternAtom(server.display, "_NET_WM_VISIBLE_NAME", False);
|
||||||
|
server.atom._NET_WM_NAME = XInternAtom(server.display, "_NET_WM_NAME", False);
|
||||||
|
server.atom._NET_WM_STRUT = XInternAtom(server.display, "_NET_WM_STRUT", False);
|
||||||
|
server.atom._NET_WM_ICON = XInternAtom(server.display, "_NET_WM_ICON", False);
|
||||||
|
server.atom._NET_WM_ICON_GEOMETRY = XInternAtom(server.display, "_NET_WM_ICON_GEOMETRY", False);
|
||||||
|
server.atom._NET_WM_ICON_NAME = XInternAtom(server.display, "_NET_WM_ICON_NAME", False);
|
||||||
|
server.atom._NET_CLOSE_WINDOW = XInternAtom(server.display, "_NET_CLOSE_WINDOW", False);
|
||||||
|
server.atom.UTF8_STRING = XInternAtom(server.display, "UTF8_STRING", False);
|
||||||
|
server.atom._NET_SUPPORTING_WM_CHECK = XInternAtom(server.display, "_NET_SUPPORTING_WM_CHECK", False);
|
||||||
|
server.atom._NET_WM_CM_S0 = XInternAtom(server.display, "_NET_WM_CM_S0", False);
|
||||||
|
server.atom._NET_SUPPORTING_WM_CHECK = XInternAtom(server.display, "_NET_WM_NAME", False);
|
||||||
|
server.atom._NET_WM_STRUT_PARTIAL = XInternAtom(server.display, "_NET_WM_STRUT_PARTIAL", False);
|
||||||
|
server.atom.WM_NAME = XInternAtom(server.display, "WM_NAME", False);
|
||||||
|
server.atom.__SWM_VROOT = XInternAtom(server.display, "__SWM_VROOT", False);
|
||||||
|
server.atom._MOTIF_WM_HINTS = XInternAtom(server.display, "_MOTIF_WM_HINTS", False);
|
||||||
|
server.atom.WM_HINTS = XInternAtom(server.display, "WM_HINTS", False);
|
||||||
|
gchar *name = g_strdup_printf("_XSETTINGS_S%d", DefaultScreen(server.display));
|
||||||
|
server.atom._XSETTINGS_SCREEN = XInternAtom(server.display, name, False);
|
||||||
|
g_free(name);
|
||||||
|
server.atom._XSETTINGS_SETTINGS = XInternAtom(server.display, "_XSETTINGS_SETTINGS", False);
|
||||||
|
|
||||||
|
// systray protocol
|
||||||
|
name = g_strdup_printf("_NET_SYSTEM_TRAY_S%d", DefaultScreen(server.display));
|
||||||
|
server.atom._NET_SYSTEM_TRAY_SCREEN = XInternAtom(server.display, name, False);
|
||||||
|
g_free(name);
|
||||||
|
server.atom._NET_SYSTEM_TRAY_OPCODE = XInternAtom(server.display, "_NET_SYSTEM_TRAY_OPCODE", False);
|
||||||
|
server.atom.MANAGER = XInternAtom(server.display, "MANAGER", False);
|
||||||
|
server.atom._NET_SYSTEM_TRAY_MESSAGE_DATA = XInternAtom(server.display, "_NET_SYSTEM_TRAY_MESSAGE_DATA", False);
|
||||||
|
server.atom._NET_SYSTEM_TRAY_ORIENTATION = XInternAtom(server.display, "_NET_SYSTEM_TRAY_ORIENTATION", False);
|
||||||
|
server.atom._NET_SYSTEM_TRAY_ICON_SIZE = XInternAtom(server.display, "_NET_SYSTEM_TRAY_ICON_SIZE", False);
|
||||||
|
server.atom._NET_SYSTEM_TRAY_PADDING = XInternAtom(server.display, "_NET_SYSTEM_TRAY_PADDING", False);
|
||||||
|
server.atom._XEMBED = XInternAtom(server.display, "_XEMBED", False);
|
||||||
|
server.atom._XEMBED_INFO = XInternAtom(server.display, "_XEMBED_INFO", False);
|
||||||
|
server.atom._NET_WM_PID = XInternAtom(server.display, "_NET_WM_PID", True);
|
||||||
|
|
||||||
|
// drag 'n' drop
|
||||||
|
server.atom.XdndAware = XInternAtom(server.display, "XdndAware", False);
|
||||||
|
server.atom.XdndEnter = XInternAtom(server.display, "XdndEnter", False);
|
||||||
|
server.atom.XdndPosition = XInternAtom(server.display, "XdndPosition", False);
|
||||||
|
server.atom.XdndStatus = XInternAtom(server.display, "XdndStatus", False);
|
||||||
|
server.atom.XdndDrop = XInternAtom(server.display, "XdndDrop", False);
|
||||||
|
server.atom.XdndLeave = XInternAtom(server.display, "XdndLeave", False);
|
||||||
|
server.atom.XdndSelection = XInternAtom(server.display, "XdndSelection", False);
|
||||||
|
server.atom.XdndTypeList = XInternAtom(server.display, "XdndTypeList", False);
|
||||||
|
server.atom.XdndActionCopy = XInternAtom(server.display, "XdndActionCopy", False);
|
||||||
|
server.atom.XdndFinished = XInternAtom(server.display, "XdndFinished", False);
|
||||||
|
server.atom.TARGETS = XInternAtom(server.display, "TARGETS", False);
|
||||||
|
}
|
||||||
|
|
||||||
void cleanup_server()
|
void cleanup_server()
|
||||||
{
|
{
|
||||||
if (server.colormap) XFreeColormap(server.dsp, server.colormap);
|
if (server.colormap)
|
||||||
if (server.colormap32) XFreeColormap(server.dsp, server.colormap32);
|
XFreeColormap(server.display, server.colormap);
|
||||||
if (server.monitor) {
|
server.colormap = 0;
|
||||||
int i;
|
if (server.colormap32)
|
||||||
for (i=0; i<server.nb_monitor; ++i)
|
XFreeColormap(server.display, server.colormap32);
|
||||||
if (server.monitor[i].names)
|
server.colormap32 = 0;
|
||||||
g_strfreev(server.monitor[i].names);
|
if (server.monitors) {
|
||||||
free(server.monitor);
|
for (int i = 0; i < server.num_monitors; ++i) {
|
||||||
}
|
g_strfreev(server.monitors[i].names);
|
||||||
if (server.gc) XFreeGC(server.dsp, server.gc);
|
server.monitors[i].names = NULL;
|
||||||
|
}
|
||||||
|
free(server.monitors);
|
||||||
|
server.monitors = NULL;
|
||||||
|
}
|
||||||
|
if (server.gc)
|
||||||
|
XFreeGC(server.display, server.gc);
|
||||||
|
server.gc = NULL;
|
||||||
|
server.disable_transparency = FALSE;
|
||||||
|
#ifdef HAVE_SN
|
||||||
|
if (server.pids)
|
||||||
|
g_tree_destroy(server.pids);
|
||||||
|
server.pids = NULL;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void send_event32(Window win, Atom at, long data1, long data2, long data3)
|
||||||
void send_event32 (Window win, Atom at, long data1, long data2, long data3)
|
|
||||||
{
|
{
|
||||||
XEvent event;
|
XEvent event;
|
||||||
|
|
||||||
event.xclient.type = ClientMessage;
|
event.xclient.type = ClientMessage;
|
||||||
event.xclient.serial = 0;
|
event.xclient.serial = 0;
|
||||||
event.xclient.send_event = True;
|
event.xclient.send_event = True;
|
||||||
event.xclient.display = server.dsp;
|
event.xclient.display = server.display;
|
||||||
event.xclient.window = win;
|
event.xclient.window = win;
|
||||||
event.xclient.message_type = at;
|
event.xclient.message_type = at;
|
||||||
|
|
||||||
event.xclient.format = 32;
|
event.xclient.format = 32;
|
||||||
event.xclient.data.l[0] = data1;
|
event.xclient.data.l[0] = data1;
|
||||||
event.xclient.data.l[1] = data2;
|
event.xclient.data.l[1] = data2;
|
||||||
event.xclient.data.l[2] = data3;
|
event.xclient.data.l[2] = data3;
|
||||||
event.xclient.data.l[3] = 0;
|
event.xclient.data.l[3] = 0;
|
||||||
event.xclient.data.l[4] = 0;
|
event.xclient.data.l[4] = 0;
|
||||||
|
|
||||||
XSendEvent(server.dsp, server.root_win, False, SubstructureRedirectMask|SubstructureNotifyMask, &event);
|
XSendEvent(server.display, server.root_win, False, SubstructureRedirectMask | SubstructureNotifyMask, &event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int get_property32(Window win, Atom at, Atom type)
|
||||||
int get_property32 (Window win, Atom at, Atom type)
|
|
||||||
{
|
{
|
||||||
Atom type_ret;
|
Atom type_ret;
|
||||||
int format_ret = 0, data = 0;
|
int format_ret = 0, data = 0;
|
||||||
unsigned long nitems_ret = 0;
|
unsigned long nitems_ret = 0;
|
||||||
unsigned long bafter_ret = 0;
|
unsigned long bafter_ret = 0;
|
||||||
unsigned char *prop_value = 0;
|
unsigned char *prop_value = 0;
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
if (!win) return 0;
|
if (!win)
|
||||||
|
return 0;
|
||||||
|
|
||||||
result = XGetWindowProperty(server.dsp, win, at, 0, 0x7fffffff, False, type, &type_ret, &format_ret, &nitems_ret, &bafter_ret, &prop_value);
|
result = XGetWindowProperty(server.display,
|
||||||
|
win,
|
||||||
|
at,
|
||||||
|
0,
|
||||||
|
0x7fffffff,
|
||||||
|
False,
|
||||||
|
type,
|
||||||
|
&type_ret,
|
||||||
|
&format_ret,
|
||||||
|
&nitems_ret,
|
||||||
|
&bafter_ret,
|
||||||
|
&prop_value);
|
||||||
|
|
||||||
if (result == Success && prop_value) {
|
if (result == Success && prop_value) {
|
||||||
data = ((gulong*)prop_value)[0];
|
data = ((gulong *)prop_value)[0];
|
||||||
XFree (prop_value);
|
XFree(prop_value);
|
||||||
}
|
}
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void *server_get_property(Window win, Atom at, Atom type, int *num_results)
|
||||||
void *server_get_property (Window win, Atom at, Atom type, int *num_results)
|
|
||||||
{
|
{
|
||||||
Atom type_ret;
|
Atom type_ret;
|
||||||
int format_ret = 0;
|
int format_ret = 0;
|
||||||
unsigned long nitems_ret = 0;
|
unsigned long nitems_ret = 0;
|
||||||
unsigned long bafter_ret = 0;
|
unsigned long bafter_ret = 0;
|
||||||
unsigned char *prop_value;
|
unsigned char *prop_value;
|
||||||
int result;
|
|
||||||
|
|
||||||
if (!win) return 0;
|
if (!win)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
result = XGetWindowProperty(server.dsp, win, at, 0, 0x7fffffff, False, type, &type_ret, &format_ret, &nitems_ret, &bafter_ret, &prop_value);
|
int result = XGetWindowProperty(server.display,
|
||||||
|
win,
|
||||||
|
at,
|
||||||
|
0,
|
||||||
|
0x7fffffff,
|
||||||
|
False,
|
||||||
|
type,
|
||||||
|
&type_ret,
|
||||||
|
&format_ret,
|
||||||
|
&nitems_ret,
|
||||||
|
&bafter_ret,
|
||||||
|
&prop_value);
|
||||||
|
|
||||||
// Send back resultcount
|
// Send fill_color resultcount
|
||||||
if (num_results) *num_results = (int)nitems_ret;
|
if (num_results)
|
||||||
|
*num_results = (int)nitems_ret;
|
||||||
|
|
||||||
if (result == Success && prop_value) return prop_value;
|
if (result == Success && prop_value)
|
||||||
else return 0;
|
return prop_value;
|
||||||
|
else
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void get_root_pixmap()
|
void get_root_pixmap()
|
||||||
{
|
{
|
||||||
Pixmap ret = None;
|
Pixmap ret = None;
|
||||||
|
|
||||||
unsigned long *res;
|
Atom pixmap_atoms[] = {server.atom._XROOTPMAP_ID, server.atom._XROOTMAP_ID};
|
||||||
Atom pixmap_atoms[] = { server.atom._XROOTPMAP_ID, server.atom._XROOTMAP_ID };
|
for (int i = 0; i < sizeof(pixmap_atoms) / sizeof(Atom); ++i) {
|
||||||
int i;
|
unsigned long *res = (unsigned long *)server_get_property(server.root_win, pixmap_atoms[i], XA_PIXMAP, NULL);
|
||||||
|
if (res) {
|
||||||
|
ret = *((Pixmap *)res);
|
||||||
|
XFree(res);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
server.root_pmap = ret;
|
||||||
|
|
||||||
for (i=0; i<sizeof(pixmap_atoms)/sizeof(Atom); ++i) {
|
if (server.root_pmap == None) {
|
||||||
res = server_get_property (server.root_win, pixmap_atoms[i], XA_PIXMAP, 0);
|
fprintf(stderr, "tint2 : pixmap background detection failed\n");
|
||||||
if (res) {
|
} else {
|
||||||
ret = *((Pixmap*)res);
|
XGCValues gcv;
|
||||||
XFree(res);
|
gcv.ts_x_origin = 0;
|
||||||
break;
|
gcv.ts_y_origin = 0;
|
||||||
}
|
gcv.fill_style = FillTiled;
|
||||||
}
|
unsigned mask = GCTileStipXOrigin | GCTileStipYOrigin | GCFillStyle | GCTile;
|
||||||
server.root_pmap = ret;
|
|
||||||
|
|
||||||
if (server.root_pmap == None)
|
gcv.tile = server.root_pmap;
|
||||||
fprintf(stderr, "tint2 : pixmap background detection failed\n");
|
XChangeGC(server.display, server.gc, mask, &gcv);
|
||||||
else {
|
}
|
||||||
XGCValues gcv;
|
|
||||||
gcv.ts_x_origin = 0;
|
|
||||||
gcv.ts_y_origin = 0;
|
|
||||||
gcv.fill_style = FillTiled;
|
|
||||||
uint mask = GCTileStipXOrigin | GCTileStipYOrigin | GCFillStyle | GCTile;
|
|
||||||
|
|
||||||
gcv.tile = server.root_pmap;
|
|
||||||
XChangeGC(server.dsp, server.gc, mask, &gcv);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int compare_monitor_pos(const void *monitor1, const void *monitor2)
|
||||||
int compareMonitorPos(const void *monitor1, const void *monitor2)
|
|
||||||
{
|
{
|
||||||
Monitor *m1 = (Monitor*)monitor1;
|
const Monitor *m1 = (const Monitor *)monitor1;
|
||||||
Monitor *m2 = (Monitor*)monitor2;
|
const Monitor *m2 = (const Monitor *)monitor2;
|
||||||
|
|
||||||
if (m1->x < m2->x) {
|
if (primary_monitor_first) {
|
||||||
return -1;
|
if (m1->primary && !m2->primary)
|
||||||
}
|
return -1;
|
||||||
else if (m1->x > m2->x) {
|
if (!m1->primary && m2->primary)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else if (m1->y < m2->y) {
|
|
||||||
return -1;
|
if (m1->x < m2->x) {
|
||||||
}
|
return -1;
|
||||||
else if (m1->y > m2->y) {
|
} else if (m1->x > m2->x) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
} else if (m1->y < m2->y) {
|
||||||
else {
|
return -1;
|
||||||
return 0;
|
} else if (m1->y > m2->y) {
|
||||||
}
|
return 1;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int monitor_includes_monitor(const void *monitor1, const void *monitor2)
|
||||||
int compareMonitorIncluded(const void *monitor1, const void *monitor2)
|
|
||||||
{
|
{
|
||||||
Monitor *m1 = (Monitor*)monitor1;
|
const Monitor *m1 = (const Monitor *)monitor1;
|
||||||
Monitor *m2 = (Monitor*)monitor2;
|
const Monitor *m2 = (const Monitor *)monitor2;
|
||||||
|
|
||||||
if (m1->x >= m2->x && m1->y >= m2->y && (m1->x+m1->width) <= (m2->x+m2->width) && (m1->y+m1->height) <= (m2->y+m2->height)) {
|
if (m1->x >= m2->x && m1->y >= m2->y && (m1->x + m1->width) <= (m2->x + m2->width) &&
|
||||||
// m1 included inside m2
|
(m1->y + m1->height) <= (m2->y + m2->height)) {
|
||||||
return 1;
|
// m1 included inside m2
|
||||||
}
|
return 1;
|
||||||
else {
|
} else {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void sort_monitors()
|
||||||
|
{
|
||||||
|
qsort(server.monitors, server.num_monitors, sizeof(Monitor), compare_monitor_pos);
|
||||||
|
}
|
||||||
|
|
||||||
void get_monitors()
|
void get_monitors()
|
||||||
{
|
{
|
||||||
int i, j, nbmonitor;
|
if (XineramaIsActive(server.display)) {
|
||||||
if (XineramaIsActive(server.dsp)) {
|
int num_monitors;
|
||||||
XineramaScreenInfo *info = XineramaQueryScreens(server.dsp, &nbmonitor);
|
XineramaScreenInfo *info = XineramaQueryScreens(server.display, &num_monitors);
|
||||||
XRRScreenResources *res = XRRGetScreenResourcesCurrent(server.dsp, server.root_win);
|
XRRScreenResources *res = XRRGetScreenResourcesCurrent(server.display, server.root_win);
|
||||||
|
RROutput primary_output = XRRGetOutputPrimary(server.display, server.root_win);
|
||||||
|
|
||||||
if (res && res->ncrtc >= nbmonitor) {
|
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 );
|
printf("xRandr: Found crtc's: %d\n", res->ncrtc);
|
||||||
server.monitor = malloc(res->ncrtc * sizeof(Monitor));
|
server.monitors = calloc(res->ncrtc, sizeof(Monitor));
|
||||||
for (i=0; i<res->ncrtc; ++i) {
|
num_monitors = 0;
|
||||||
XRRCrtcInfo* crtc_info = XRRGetCrtcInfo(server.dsp, res, res->crtcs[i]);
|
for (int i = 0; i < res->ncrtc; ++i) {
|
||||||
server.monitor[i].x = crtc_info->x;
|
XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(server.display, res, res->crtcs[i]);
|
||||||
server.monitor[i].y = crtc_info->y;
|
// Ignore empty crtc
|
||||||
server.monitor[i].width = crtc_info->width;
|
if (!crtc_info->width || !crtc_info->height) {
|
||||||
server.monitor[i].height = crtc_info->height;
|
printf("xRandr: crtc %d seems disabled\n", i);
|
||||||
server.monitor[i].names = malloc((crtc_info->noutput+1) * sizeof(char*));
|
XRRFreeCrtcInfo(crtc_info);
|
||||||
for (j=0; j<crtc_info->noutput; ++j) {
|
continue;
|
||||||
XRROutputInfo* output_info = XRRGetOutputInfo(server.dsp, res, crtc_info->outputs[j]);
|
}
|
||||||
printf("xRandr: Linking output %s with crtc %d\n", output_info->name, i);
|
int i_monitor = num_monitors;
|
||||||
server.monitor[i].names[j] = g_strdup(output_info->name);
|
num_monitors++;
|
||||||
XRRFreeOutputInfo(output_info);
|
server.monitors[i_monitor].x = crtc_info->x;
|
||||||
}
|
server.monitors[i_monitor].y = crtc_info->y;
|
||||||
server.monitor[i].names[j] = 0;
|
server.monitors[i_monitor].width = crtc_info->width;
|
||||||
XRRFreeCrtcInfo(crtc_info);
|
server.monitors[i_monitor].height = crtc_info->height;
|
||||||
}
|
server.monitors[i_monitor].names = calloc((crtc_info->noutput + 1), sizeof(gchar *));
|
||||||
nbmonitor = res->ncrtc;
|
for (int j = 0; j < crtc_info->noutput; ++j) {
|
||||||
}
|
XRROutputInfo *output_info = XRRGetOutputInfo(server.display, res, crtc_info->outputs[j]);
|
||||||
else if (info && nbmonitor > 0) {
|
printf("xRandr: Linking output %s with crtc %d\n", output_info->name, i);
|
||||||
server.monitor = malloc(nbmonitor * sizeof(Monitor));
|
server.monitors[i_monitor].names[j] = g_strdup(output_info->name);
|
||||||
for (i=0 ; i < nbmonitor ; i++) {
|
XRRFreeOutputInfo(output_info);
|
||||||
server.monitor[i].x = info[i].x_org;
|
server.monitors[i_monitor].primary = crtc_info->outputs[j] == primary_output;
|
||||||
server.monitor[i].y = info[i].y_org;
|
}
|
||||||
server.monitor[i].width = info[i].width;
|
server.monitors[i_monitor].names[crtc_info->noutput] = NULL;
|
||||||
server.monitor[i].height = info[i].height;
|
XRRFreeCrtcInfo(crtc_info);
|
||||||
server.monitor[i].names = 0;
|
}
|
||||||
}
|
} else if (info && num_monitors > 0) {
|
||||||
}
|
server.monitors = calloc(num_monitors, sizeof(Monitor));
|
||||||
|
for (int i = 0; i < num_monitors; i++) {
|
||||||
|
server.monitors[i].x = info[i].x_org;
|
||||||
|
server.monitors[i].y = info[i].y_org;
|
||||||
|
server.monitors[i].width = info[i].width;
|
||||||
|
server.monitors[i].height = info[i].height;
|
||||||
|
server.monitors[i].names = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ordered monitor
|
// Sort monitors by inclusion
|
||||||
qsort(server.monitor, nbmonitor, sizeof(Monitor), compareMonitorIncluded);
|
qsort(server.monitors, num_monitors, sizeof(Monitor), monitor_includes_monitor);
|
||||||
|
|
||||||
// remove monitor included into another one
|
// Remove monitors included in other ones
|
||||||
i = 0;
|
int i = 0;
|
||||||
while (i < nbmonitor) {
|
while (i < num_monitors) {
|
||||||
for (j=0; j < i ; j++) {
|
for (int j = 0; j < i; j++) {
|
||||||
if (compareMonitorIncluded(&server.monitor[i], &server.monitor[j]) > 0) {
|
if (monitor_includes_monitor(&server.monitors[i], &server.monitors[j]) > 0) {
|
||||||
goto next;
|
goto next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
next:
|
next:
|
||||||
for (j=i; j<nbmonitor; ++j)
|
for (int j = i; j < num_monitors; ++j)
|
||||||
if (server.monitor[j].names)
|
if (server.monitors[j].names)
|
||||||
g_strfreev(server.monitor[j].names);
|
g_strfreev(server.monitors[j].names);
|
||||||
server.nb_monitor = i;
|
server.num_monitors = i;
|
||||||
server.monitor = realloc(server.monitor, server.nb_monitor * sizeof(Monitor));
|
server.monitors = realloc(server.monitors, server.num_monitors * sizeof(Monitor));
|
||||||
qsort(server.monitor, server.nb_monitor, sizeof(Monitor), compareMonitorPos);
|
qsort(server.monitors, server.num_monitors, sizeof(Monitor), compare_monitor_pos);
|
||||||
|
|
||||||
if (res)
|
if (res)
|
||||||
XRRFreeScreenResources(res);
|
XRRFreeScreenResources(res);
|
||||||
XFree(info);
|
XFree(info);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!server.nb_monitor) {
|
if (!server.num_monitors) {
|
||||||
server.nb_monitor = 1;
|
server.num_monitors = 1;
|
||||||
server.monitor = malloc(sizeof(Monitor));
|
server.monitors = calloc(1, sizeof(Monitor));
|
||||||
server.monitor[0].x = server.monitor[0].y = 0;
|
server.monitors[0].x = server.monitors[0].y = 0;
|
||||||
server.monitor[0].width = DisplayWidth (server.dsp, server.screen);
|
server.monitors[0].width = DisplayWidth(server.display, server.screen);
|
||||||
server.monitor[0].height = DisplayHeight (server.dsp, server.screen);
|
server.monitors[0].height = DisplayHeight(server.display, server.screen);
|
||||||
server.monitor[0].names = 0;
|
server.monitors[0].names = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void print_monitors()
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Number of monitors: %d\n", server.num_monitors);
|
||||||
|
for (int i = 0; i < server.num_monitors; i++) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"Monitor %d: x = %d, y = %d, w = %d, h = %d\n",
|
||||||
|
i + 1,
|
||||||
|
server.monitors[i].x,
|
||||||
|
server.monitors[i].y,
|
||||||
|
server.monitors[i].width,
|
||||||
|
server.monitors[i].height);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void server_get_number_of_desktops()
|
||||||
|
{
|
||||||
|
if (server.viewports) {
|
||||||
|
free(server.viewports);
|
||||||
|
server.viewports = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
server.num_desktops = get_property32(server.root_win, server.atom._NET_NUMBER_OF_DESKTOPS, XA_CARDINAL);
|
||||||
|
if (server.num_desktops > 1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int num_results;
|
||||||
|
long *work_area_size = server_get_property(server.root_win, server.atom._NET_WORKAREA, XA_CARDINAL, &num_results);
|
||||||
|
if (!work_area_size)
|
||||||
|
return;
|
||||||
|
int work_area_width = work_area_size[0] + work_area_size[2];
|
||||||
|
int work_area_height = work_area_size[1] + work_area_size[3];
|
||||||
|
XFree(work_area_size);
|
||||||
|
|
||||||
|
long *x_screen_size =
|
||||||
|
server_get_property(server.root_win, server.atom._NET_DESKTOP_GEOMETRY, XA_CARDINAL, &num_results);
|
||||||
|
if (!x_screen_size)
|
||||||
|
return;
|
||||||
|
int x_screen_width = x_screen_size[0];
|
||||||
|
int x_screen_height = x_screen_size[1];
|
||||||
|
XFree(x_screen_size);
|
||||||
|
|
||||||
|
int num_viewports = MAX(x_screen_width / work_area_width, 1) * MAX(x_screen_height / work_area_height, 1);
|
||||||
|
if (num_viewports <= 1) {
|
||||||
|
server.num_desktops = 1;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
server.viewports = calloc(num_viewports, sizeof(Viewport));
|
||||||
|
int k = 0;
|
||||||
|
for (int i = 0; i < MAX(x_screen_height / work_area_height, 1); i++) {
|
||||||
|
for (int j = 0; j < MAX(x_screen_width / work_area_width, 1); j++) {
|
||||||
|
server.viewports[k].x = j * work_area_width;
|
||||||
|
server.viewports[k].y = i * work_area_height;
|
||||||
|
server.viewports[k].width = work_area_width;
|
||||||
|
server.viewports[k].height = work_area_height;
|
||||||
|
k++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
server.num_desktops = num_viewports;
|
||||||
|
}
|
||||||
|
|
||||||
|
GSList *get_desktop_names()
|
||||||
|
{
|
||||||
|
if (server.viewports) {
|
||||||
|
GSList *list = NULL;
|
||||||
|
for (int j = 0; j < server.num_desktops; j++) {
|
||||||
|
list = g_slist_append(list, g_strdup_printf("%d", j + 1));
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
int count;
|
||||||
|
GSList *list = NULL;
|
||||||
|
gchar *data_ptr =
|
||||||
|
server_get_property(server.root_win, server.atom._NET_DESKTOP_NAMES, server.atom.UTF8_STRING, &count);
|
||||||
|
if (data_ptr) {
|
||||||
|
list = g_slist_append(list, g_strdup(data_ptr));
|
||||||
|
for (int j = 0; j < count - 1; j++) {
|
||||||
|
if (*(data_ptr + j) == '\0') {
|
||||||
|
gchar *ptr = (gchar *)data_ptr + j + 1;
|
||||||
|
list = g_slist_append(list, g_strdup(ptr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
XFree(data_ptr);
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
int get_current_desktop()
|
||||||
|
{
|
||||||
|
if (!server.viewports) {
|
||||||
|
return MAX(0,
|
||||||
|
MIN(server.num_desktops - 1,
|
||||||
|
get_property32(server.root_win, server.atom._NET_CURRENT_DESKTOP, XA_CARDINAL)));
|
||||||
|
}
|
||||||
|
|
||||||
|
int num_results;
|
||||||
|
long *work_area_size = server_get_property(server.root_win, server.atom._NET_WORKAREA, XA_CARDINAL, &num_results);
|
||||||
|
if (!work_area_size)
|
||||||
|
return 0;
|
||||||
|
int work_area_width = work_area_size[0] + work_area_size[2];
|
||||||
|
int work_area_height = work_area_size[1] + work_area_size[3];
|
||||||
|
XFree(work_area_size);
|
||||||
|
|
||||||
|
if (work_area_width <= 0 || work_area_height <= 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
long *viewport = server_get_property(server.root_win, server.atom._NET_DESKTOP_VIEWPORT, XA_CARDINAL, &num_results);
|
||||||
|
if (!viewport)
|
||||||
|
return 0;
|
||||||
|
int viewport_x = viewport[0];
|
||||||
|
int viewport_y = viewport[1];
|
||||||
|
XFree(viewport);
|
||||||
|
|
||||||
|
long *x_screen_size =
|
||||||
|
server_get_property(server.root_win, server.atom._NET_DESKTOP_GEOMETRY, XA_CARDINAL, &num_results);
|
||||||
|
if (!x_screen_size)
|
||||||
|
return 0;
|
||||||
|
int x_screen_width = x_screen_size[0];
|
||||||
|
XFree(x_screen_size);
|
||||||
|
|
||||||
|
int ncols = x_screen_width / work_area_width;
|
||||||
|
|
||||||
|
// fprintf(stderr, "\n");
|
||||||
|
// fprintf(stderr, "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, "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;
|
||||||
|
return MAX(0, MIN(server.num_desktops - 1, result));
|
||||||
|
}
|
||||||
|
|
||||||
|
void change_desktop(int desktop)
|
||||||
|
{
|
||||||
|
if (!server.viewports) {
|
||||||
|
send_event32(server.root_win, server.atom._NET_CURRENT_DESKTOP, desktop, 0, 0);
|
||||||
|
} else {
|
||||||
|
send_event32(server.root_win,
|
||||||
|
server.atom._NET_DESKTOP_VIEWPORT,
|
||||||
|
server.viewports[desktop].x,
|
||||||
|
server.viewports[desktop].y,
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void get_desktops()
|
void get_desktops()
|
||||||
{
|
{
|
||||||
int i;
|
// detect number of desktops
|
||||||
|
// wait 15s to leave some time for window manager startup
|
||||||
// detect number of desktops
|
for (int i = 0; i < 15; i++) {
|
||||||
// wait 15s to leave some time for window manager startup
|
server_get_number_of_desktops();
|
||||||
for (i=0 ; i < 15 ; i++) {
|
if (server.num_desktops > 0)
|
||||||
server.nb_desktop = server_get_number_of_desktop ();
|
break;
|
||||||
if (server.nb_desktop > 0) break;
|
sleep(1);
|
||||||
sleep(1);
|
}
|
||||||
}
|
if (server.num_desktops == 0) {
|
||||||
if (server.nb_desktop == 0) {
|
server.num_desktops = 1;
|
||||||
server.nb_desktop = 1;
|
fprintf(stderr, "warning : WM doesn't respect NETWM specs. tint2 default to 1 desktop.\n");
|
||||||
fprintf(stderr, "warning : WM doesn't respect NETWM specs. tint2 default to 1 desktop.\n");
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void server_init_visual()
|
void server_init_visual()
|
||||||
{
|
{
|
||||||
// inspired by freedesktops fdclock ;)
|
// inspired by freedesktops fdclock ;)
|
||||||
XVisualInfo *xvi;
|
XVisualInfo templ = {.screen = server.screen, .depth = 32, .class = TrueColor};
|
||||||
XVisualInfo templ = { .screen=server.screen, .depth=32, .class=TrueColor };
|
int nvi;
|
||||||
int nvi;
|
XVisualInfo *xvi =
|
||||||
xvi = XGetVisualInfo(server.dsp, VisualScreenMask|VisualDepthMask|VisualClassMask, &templ, &nvi);
|
XGetVisualInfo(server.display, VisualScreenMask | VisualDepthMask | VisualClassMask, &templ, &nvi);
|
||||||
|
|
||||||
Visual *visual = 0;
|
Visual *visual = NULL;
|
||||||
if (xvi) {
|
if (xvi) {
|
||||||
int i;
|
XRenderPictFormat *format;
|
||||||
XRenderPictFormat *format;
|
for (int i = 0; i < nvi; i++) {
|
||||||
for (i = 0; i < nvi; i++) {
|
format = XRenderFindVisualFormat(server.display, xvi[i].visual);
|
||||||
format = XRenderFindVisualFormat(server.dsp, xvi[i].visual);
|
if (format->type == PictTypeDirect && format->direct.alphaMask) {
|
||||||
if (format->type == PictTypeDirect && format->direct.alphaMask) {
|
visual = xvi[i].visual;
|
||||||
visual = xvi[i].visual;
|
break;
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
XFree(xvi);
|
||||||
XFree (xvi);
|
|
||||||
|
|
||||||
// check composite manager
|
// check composite manager
|
||||||
server.composite_manager = XGetSelectionOwner(server.dsp, server.atom._NET_WM_CM_S0);
|
server.composite_manager = XGetSelectionOwner(server.display, server.atom._NET_WM_CM_S0);
|
||||||
if (server.colormap)
|
if (server.colormap)
|
||||||
XFreeColormap(server.dsp, server.colormap);
|
XFreeColormap(server.display, server.colormap);
|
||||||
if (server.colormap32)
|
if (server.colormap32)
|
||||||
XFreeColormap(server.dsp, server.colormap32);
|
XFreeColormap(server.display, server.colormap32);
|
||||||
|
|
||||||
if (visual) {
|
if (visual) {
|
||||||
server.visual32 = visual;
|
server.visual32 = visual;
|
||||||
server.colormap32 = XCreateColormap(server.dsp, server.root_win, visual, AllocNone);
|
server.colormap32 = XCreateColormap(server.display, server.root_win, visual, AllocNone);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (visual && server.composite_manager != None && snapshot_path == 0) {
|
if (!server.disable_transparency && visual && server.composite_manager != None && !snapshot_path) {
|
||||||
XSetWindowAttributes attrs;
|
XSetWindowAttributes attrs;
|
||||||
attrs.event_mask = StructureNotifyMask;
|
attrs.event_mask = StructureNotifyMask;
|
||||||
XChangeWindowAttributes (server.dsp, server.composite_manager, CWEventMask, &attrs);
|
XChangeWindowAttributes(server.display, server.composite_manager, CWEventMask, &attrs);
|
||||||
|
|
||||||
server.real_transparency = 1;
|
server.real_transparency = TRUE;
|
||||||
server.depth = 32;
|
server.depth = 32;
|
||||||
printf("real transparency on... depth: %d\n", server.depth);
|
printf("real transparency on... depth: %d\n", server.depth);
|
||||||
server.colormap = XCreateColormap(server.dsp, 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 = 0;
|
server.depth = DefaultDepth(server.display, server.screen);
|
||||||
server.depth = DefaultDepth(server.dsp, server.screen);
|
printf("real transparency off.... depth: %d\n", server.depth);
|
||||||
printf("real transparency off.... depth: %d\n", server.depth);
|
server.colormap = DefaultColormap(server.display, server.screen);
|
||||||
server.colormap = DefaultColormap(server.dsp, server.screen);
|
server.visual = DefaultVisual(server.display, server.screen);
|
||||||
server.visual = DefaultVisual(server.dsp, server.screen);
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
241
src/server.h
241
src/server.h
@@ -15,133 +15,143 @@
|
|||||||
|
|
||||||
#ifdef HAVE_SN
|
#ifdef HAVE_SN
|
||||||
#include <libsn/sn.h>
|
#include <libsn/sn.h>
|
||||||
#include <glib.h>
|
|
||||||
#endif
|
#endif
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
extern gboolean primary_monitor_first;
|
||||||
|
|
||||||
typedef struct Global_atom
|
typedef struct Global_atom {
|
||||||
{
|
Atom _XROOTPMAP_ID;
|
||||||
Atom _XROOTPMAP_ID;
|
Atom _XROOTMAP_ID;
|
||||||
Atom _XROOTMAP_ID;
|
Atom _NET_CURRENT_DESKTOP;
|
||||||
Atom _NET_CURRENT_DESKTOP;
|
Atom _NET_NUMBER_OF_DESKTOPS;
|
||||||
Atom _NET_NUMBER_OF_DESKTOPS;
|
Atom _NET_DESKTOP_NAMES;
|
||||||
Atom _NET_DESKTOP_NAMES;
|
Atom _NET_DESKTOP_GEOMETRY;
|
||||||
Atom _NET_DESKTOP_GEOMETRY;
|
Atom _NET_DESKTOP_VIEWPORT;
|
||||||
Atom _NET_DESKTOP_VIEWPORT;
|
Atom _NET_WORKAREA;
|
||||||
Atom _NET_ACTIVE_WINDOW;
|
Atom _NET_ACTIVE_WINDOW;
|
||||||
Atom _NET_WM_WINDOW_TYPE;
|
Atom _NET_WM_WINDOW_TYPE;
|
||||||
Atom _NET_WM_STATE_SKIP_PAGER;
|
Atom _NET_WM_STATE_SKIP_PAGER;
|
||||||
Atom _NET_WM_STATE_SKIP_TASKBAR;
|
Atom _NET_WM_STATE_SKIP_TASKBAR;
|
||||||
Atom _NET_WM_STATE_STICKY;
|
Atom _NET_WM_STATE_STICKY;
|
||||||
Atom _NET_WM_STATE_DEMANDS_ATTENTION;
|
Atom _NET_WM_STATE_DEMANDS_ATTENTION;
|
||||||
Atom _NET_WM_WINDOW_TYPE_DOCK;
|
Atom _NET_WM_WINDOW_TYPE_DOCK;
|
||||||
Atom _NET_WM_WINDOW_TYPE_DESKTOP;
|
Atom _NET_WM_WINDOW_TYPE_DESKTOP;
|
||||||
Atom _NET_WM_WINDOW_TYPE_TOOLBAR;
|
Atom _NET_WM_WINDOW_TYPE_TOOLBAR;
|
||||||
Atom _NET_WM_WINDOW_TYPE_MENU;
|
Atom _NET_WM_WINDOW_TYPE_MENU;
|
||||||
Atom _NET_WM_WINDOW_TYPE_SPLASH;
|
Atom _NET_WM_WINDOW_TYPE_SPLASH;
|
||||||
Atom _NET_WM_WINDOW_TYPE_DIALOG;
|
Atom _NET_WM_WINDOW_TYPE_DIALOG;
|
||||||
Atom _NET_WM_WINDOW_TYPE_NORMAL;
|
Atom _NET_WM_WINDOW_TYPE_NORMAL;
|
||||||
Atom _NET_WM_DESKTOP;
|
Atom _NET_WM_DESKTOP;
|
||||||
Atom WM_STATE;
|
Atom WM_STATE;
|
||||||
Atom _NET_WM_STATE;
|
Atom _NET_WM_STATE;
|
||||||
Atom _NET_WM_STATE_MAXIMIZED_VERT;
|
Atom _NET_WM_STATE_MAXIMIZED_VERT;
|
||||||
Atom _NET_WM_STATE_MAXIMIZED_HORZ;
|
Atom _NET_WM_STATE_MAXIMIZED_HORZ;
|
||||||
Atom _NET_WM_STATE_SHADED;
|
Atom _NET_WM_STATE_SHADED;
|
||||||
Atom _NET_WM_STATE_HIDDEN;
|
Atom _NET_WM_STATE_HIDDEN;
|
||||||
Atom _NET_WM_STATE_BELOW;
|
Atom _NET_WM_STATE_BELOW;
|
||||||
Atom _NET_WM_STATE_ABOVE;
|
Atom _NET_WM_STATE_ABOVE;
|
||||||
Atom _NET_WM_STATE_MODAL;
|
Atom _NET_WM_STATE_MODAL;
|
||||||
Atom _NET_CLIENT_LIST;
|
Atom _NET_CLIENT_LIST;
|
||||||
Atom _NET_WM_NAME;
|
Atom _NET_WM_NAME;
|
||||||
Atom _NET_WM_VISIBLE_NAME;
|
Atom _NET_WM_VISIBLE_NAME;
|
||||||
Atom _NET_WM_STRUT;
|
Atom _NET_WM_STRUT;
|
||||||
Atom _NET_WM_ICON;
|
Atom _NET_WM_ICON;
|
||||||
Atom _NET_WM_ICON_GEOMETRY;
|
Atom _NET_WM_ICON_GEOMETRY;
|
||||||
Atom _NET_CLOSE_WINDOW;
|
Atom _NET_WM_ICON_NAME;
|
||||||
Atom UTF8_STRING;
|
Atom _NET_CLOSE_WINDOW;
|
||||||
Atom _NET_SUPPORTING_WM_CHECK;
|
Atom UTF8_STRING;
|
||||||
Atom _NET_WM_CM_S0;
|
Atom _NET_SUPPORTING_WM_CHECK;
|
||||||
Atom _NET_WM_STRUT_PARTIAL;
|
Atom _NET_WM_CM_S0;
|
||||||
Atom WM_NAME;
|
Atom _NET_WM_STRUT_PARTIAL;
|
||||||
Atom __SWM_VROOT;
|
Atom WM_NAME;
|
||||||
Atom _MOTIF_WM_HINTS;
|
Atom __SWM_VROOT;
|
||||||
Atom WM_HINTS;
|
Atom _MOTIF_WM_HINTS;
|
||||||
Atom _NET_SYSTEM_TRAY_SCREEN;
|
Atom WM_HINTS;
|
||||||
Atom _NET_SYSTEM_TRAY_OPCODE;
|
Atom _NET_SYSTEM_TRAY_SCREEN;
|
||||||
Atom MANAGER;
|
Atom _NET_SYSTEM_TRAY_OPCODE;
|
||||||
Atom _NET_SYSTEM_TRAY_MESSAGE_DATA;
|
Atom MANAGER;
|
||||||
Atom _NET_SYSTEM_TRAY_ORIENTATION;
|
Atom _NET_SYSTEM_TRAY_MESSAGE_DATA;
|
||||||
Atom _XEMBED;
|
Atom _NET_SYSTEM_TRAY_ORIENTATION;
|
||||||
Atom _XEMBED_INFO;
|
Atom _NET_SYSTEM_TRAY_ICON_SIZE;
|
||||||
Atom _XSETTINGS_SCREEN;
|
Atom _NET_SYSTEM_TRAY_PADDING;
|
||||||
Atom _XSETTINGS_SETTINGS;
|
Atom _XEMBED;
|
||||||
Atom XdndAware;
|
Atom _XEMBED_INFO;
|
||||||
Atom XdndEnter;
|
Atom _NET_WM_PID;
|
||||||
Atom XdndPosition;
|
Atom _XSETTINGS_SCREEN;
|
||||||
Atom XdndStatus;
|
Atom _XSETTINGS_SETTINGS;
|
||||||
Atom XdndDrop;
|
Atom XdndAware;
|
||||||
Atom XdndLeave;
|
Atom XdndEnter;
|
||||||
Atom XdndSelection;
|
Atom XdndPosition;
|
||||||
Atom XdndTypeList;
|
Atom XdndStatus;
|
||||||
Atom XdndActionCopy;
|
Atom XdndDrop;
|
||||||
Atom XdndFinished;
|
Atom XdndLeave;
|
||||||
Atom TARGETS;
|
Atom XdndSelection;
|
||||||
|
Atom XdndTypeList;
|
||||||
|
Atom XdndActionCopy;
|
||||||
|
Atom XdndFinished;
|
||||||
|
Atom TARGETS;
|
||||||
} Global_atom;
|
} Global_atom;
|
||||||
|
|
||||||
|
typedef struct Monitor {
|
||||||
|
int x;
|
||||||
typedef struct Monitor
|
int y;
|
||||||
{
|
int width;
|
||||||
int x;
|
int height;
|
||||||
int y;
|
gboolean primary;
|
||||||
int width;
|
gchar **names;
|
||||||
int height;
|
|
||||||
char** names;
|
|
||||||
} Monitor;
|
} Monitor;
|
||||||
|
|
||||||
|
typedef struct Viewport {
|
||||||
|
int x;
|
||||||
|
int y;
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
} Viewport;
|
||||||
|
|
||||||
typedef struct
|
typedef struct Server {
|
||||||
{
|
Display *display;
|
||||||
Display *dsp;
|
Window root_win;
|
||||||
Window root_win;
|
Window composite_manager;
|
||||||
Window composite_manager;
|
gboolean real_transparency;
|
||||||
int real_transparency;
|
gboolean disable_transparency;
|
||||||
// current desktop
|
// current desktop
|
||||||
int desktop;
|
int desktop;
|
||||||
int screen;
|
int screen;
|
||||||
int depth;
|
int depth;
|
||||||
int nb_desktop;
|
int num_desktops;
|
||||||
// number of monitor (without monitor included into another one)
|
// number of monitor (without monitor included into another one)
|
||||||
int nb_monitor;
|
int num_monitors;
|
||||||
Monitor *monitor;
|
// Non-null only if WM uses viewports (compiz) and number of viewports > 1.
|
||||||
int got_root_win;
|
// In that case there are num_desktops viewports.
|
||||||
Visual *visual;
|
Viewport *viewports;
|
||||||
Visual *visual32;
|
Monitor *monitors;
|
||||||
// root background
|
gboolean got_root_win;
|
||||||
Pixmap root_pmap;
|
Visual *visual;
|
||||||
GC gc;
|
Visual *visual32;
|
||||||
Colormap colormap;
|
// root background
|
||||||
Colormap colormap32;
|
Pixmap root_pmap;
|
||||||
Global_atom atom;
|
GC gc;
|
||||||
|
Colormap colormap;
|
||||||
|
Colormap colormap32;
|
||||||
|
Global_atom atom;
|
||||||
#ifdef HAVE_SN
|
#ifdef HAVE_SN
|
||||||
SnDisplay *sn_dsp;
|
SnDisplay *sn_display;
|
||||||
GTree *pids;
|
GTree *pids;
|
||||||
#endif // HAVE_SN
|
#endif // HAVE_SN
|
||||||
} Server_global;
|
} Server;
|
||||||
|
|
||||||
|
|
||||||
Server_global server;
|
|
||||||
|
|
||||||
|
extern Server server;
|
||||||
|
|
||||||
// freed memory
|
// freed memory
|
||||||
void cleanup_server();
|
void cleanup_server();
|
||||||
|
|
||||||
void send_event32 (Window win, Atom at, long data1, long data2, long data3);
|
void send_event32(Window win, Atom at, long data1, long data2, long data3);
|
||||||
int get_property32 (Window win, Atom at, Atom type);
|
int get_property32(Window win, Atom at, Atom type);
|
||||||
void *server_get_property (Window win, Atom at, Atom type, int *num_results);
|
void *server_get_property(Window win, Atom at, Atom type, int *num_results);
|
||||||
Atom server_get_atom (char *atom_name);
|
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();
|
||||||
|
|
||||||
// detect root background
|
// detect root background
|
||||||
@@ -149,7 +159,12 @@ void get_root_pixmap();
|
|||||||
|
|
||||||
// detect monitors and desktops
|
// detect monitors and desktops
|
||||||
void get_monitors();
|
void get_monitors();
|
||||||
|
void sort_monitors();
|
||||||
|
void print_monitors();
|
||||||
void get_desktops();
|
void get_desktops();
|
||||||
|
void server_get_number_of_desktops();
|
||||||
|
GSList *get_desktop_names();
|
||||||
|
int get_current_desktop();
|
||||||
|
void change_desktop(int desktop);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -16,42 +16,66 @@
|
|||||||
#include <X11/extensions/Xdamage.h>
|
#include <X11/extensions/Xdamage.h>
|
||||||
|
|
||||||
// XEMBED messages
|
// XEMBED messages
|
||||||
#define XEMBED_EMBEDDED_NOTIFY 0
|
#define XEMBED_EMBEDDED_NOTIFY 0
|
||||||
// Flags for _XEMBED_INFO
|
// Flags for _XEMBED_INFO
|
||||||
#define XEMBED_MAPPED (1 << 0)
|
#define XEMBED_MAPPED (1 << 0)
|
||||||
|
|
||||||
|
typedef enum SystraySortMethod {
|
||||||
|
SYSTRAY_SORT_ASCENDING = 0,
|
||||||
|
SYSTRAY_SORT_DESCENDING,
|
||||||
|
SYSTRAY_SORT_LEFT2RIGHT,
|
||||||
|
SYSTRAY_SORT_RIGHT2LEFT,
|
||||||
|
} SystraySortMethod;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
// always start with area
|
// always start with area
|
||||||
Area area;
|
Area area;
|
||||||
|
|
||||||
GSList *list_icons;
|
GSList *list_icons;
|
||||||
int sort;
|
SystraySortMethod sort;
|
||||||
int alpha, saturation, brightness;
|
int alpha, saturation, brightness;
|
||||||
int icon_size, icons_per_column, icons_per_row, marging;
|
int icon_size, icons_per_column, icons_per_row, margin;
|
||||||
} Systraybar;
|
} Systray;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
typedef struct
|
// The actual tray icon window (created by the application)
|
||||||
{
|
Window win;
|
||||||
Window id;
|
// The parent window created by tint2 to embed the icon
|
||||||
Window tray_id;
|
Window parent;
|
||||||
int x, y;
|
int x, y;
|
||||||
int width, height;
|
int width, height;
|
||||||
// TODO: manage icon's show/hide
|
int depth;
|
||||||
int hide;
|
gboolean reparented;
|
||||||
int depth;
|
gboolean embedded;
|
||||||
Damage damage;
|
// Process PID or zero.
|
||||||
timeout* render_timeout;
|
int pid;
|
||||||
|
// A number that is incremented for each new icon, used to sort them by the order in which they were created.
|
||||||
|
int chrono;
|
||||||
|
// Name of the tray icon window.
|
||||||
|
char *name;
|
||||||
|
// Members used for rendering
|
||||||
|
struct timespec time_last_render;
|
||||||
|
int num_fast_renders;
|
||||||
|
timeout *render_timeout;
|
||||||
|
// Members used for resizing
|
||||||
|
int bad_size_counter;
|
||||||
|
struct timespec time_last_resize;
|
||||||
|
timeout *resize_timeout;
|
||||||
|
// Icon contents if we are compositing the icon, otherwise null
|
||||||
|
Imlib_Image image;
|
||||||
|
// XDamage
|
||||||
|
Damage damage;
|
||||||
} TrayWindow;
|
} TrayWindow;
|
||||||
|
|
||||||
|
|
||||||
// net_sel_win != None when protocol started
|
// net_sel_win != None when protocol started
|
||||||
extern Window net_sel_win;
|
extern Window net_sel_win;
|
||||||
extern Systraybar systray;
|
extern Systray systray;
|
||||||
extern int refresh_systray;
|
extern gboolean refresh_systray;
|
||||||
extern int systray_enabled;
|
extern gboolean systray_enabled;
|
||||||
extern int systray_max_icon_size;
|
extern int systray_max_icon_size;
|
||||||
|
extern int systray_monitor;
|
||||||
|
extern gboolean systray_profile;
|
||||||
|
extern char *systray_hide_name_filter;
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_systray();
|
void default_systray();
|
||||||
@@ -64,9 +88,9 @@ void init_systray();
|
|||||||
void init_systray_panel(void *p);
|
void init_systray_panel(void *p);
|
||||||
|
|
||||||
void draw_systray(void *obj, cairo_t *c);
|
void draw_systray(void *obj, cairo_t *c);
|
||||||
int resize_systray(void *obj);
|
gboolean resize_systray(void *obj);
|
||||||
void on_change_systray(void *obj);
|
void on_change_systray(void *obj);
|
||||||
|
gboolean systray_on_monitor(int i_monitor, int num_panels);
|
||||||
|
|
||||||
// systray protocol
|
// systray protocol
|
||||||
// many tray icon doesn't manage stop/restart of the systray manager
|
// many tray icon doesn't manage stop/restart of the systray manager
|
||||||
@@ -75,11 +99,20 @@ void stop_net();
|
|||||||
void net_message(XClientMessageEvent *e);
|
void net_message(XClientMessageEvent *e);
|
||||||
|
|
||||||
gboolean add_icon(Window id);
|
gboolean add_icon(Window id);
|
||||||
|
gboolean reparent_icon(TrayWindow *traywin);
|
||||||
|
gboolean embed_icon(TrayWindow *traywin);
|
||||||
void remove_icon(TrayWindow *traywin);
|
void remove_icon(TrayWindow *traywin);
|
||||||
|
|
||||||
void refresh_systray_icon();
|
void refresh_systray_icons();
|
||||||
void systray_render_icon(TrayWindow* traywin);
|
void systray_render_icon(void *t);
|
||||||
|
gboolean request_embed_icon(TrayWindow *traywin);
|
||||||
|
void systray_resize_request_event(TrayWindow *traywin, XEvent *e);
|
||||||
|
gboolean request_embed_icon(TrayWindow *traywin);
|
||||||
|
void systray_reconfigure_event(TrayWindow *traywin, XEvent *e);
|
||||||
|
void systray_property_notify(TrayWindow *traywin, XEvent *e);
|
||||||
|
void systray_destroy_event(TrayWindow *traywin);
|
||||||
void kde_update_icons();
|
void kde_update_icons();
|
||||||
|
|
||||||
#endif
|
TrayWindow *systray_find_icon(Window win);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|||||||
1069
src/taskbar/task.c
1069
src/taskbar/task.c
File diff suppressed because it is too large
Load Diff
@@ -13,77 +13,91 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
|
||||||
|
typedef enum TaskState {
|
||||||
|
TASK_NORMAL = 0,
|
||||||
|
TASK_ACTIVE,
|
||||||
|
TASK_ICONIFIED,
|
||||||
|
TASK_URGENT,
|
||||||
|
TASK_UNDEFINED,
|
||||||
|
TASK_STATE_COUNT,
|
||||||
|
} TaskState;
|
||||||
|
|
||||||
enum { TASK_NORMAL, TASK_ACTIVE, TASK_ICONIFIED, TASK_URGENT, TASK_STATE_COUNT };
|
typedef struct GlobalTask {
|
||||||
extern timeout* urgent_timeout;
|
Area area;
|
||||||
extern GSList* urgent_list;
|
gboolean has_text;
|
||||||
|
gboolean has_icon;
|
||||||
|
gboolean centered;
|
||||||
|
int icon_posy;
|
||||||
|
int icon_size1;
|
||||||
|
int maximum_width;
|
||||||
|
int maximum_height;
|
||||||
|
int alpha[TASK_STATE_COUNT];
|
||||||
|
int saturation[TASK_STATE_COUNT];
|
||||||
|
int brightness[TASK_STATE_COUNT];
|
||||||
|
int config_asb_mask;
|
||||||
|
Background *background[TASK_STATE_COUNT];
|
||||||
|
GList *gradient[TASK_STATE_COUNT];
|
||||||
|
int config_background_mask;
|
||||||
|
// starting position for text ~ task_padding + task_border + icon_size
|
||||||
|
double text_posx, text_height;
|
||||||
|
gboolean has_font;
|
||||||
|
PangoFontDescription *font_desc;
|
||||||
|
Color font[TASK_STATE_COUNT];
|
||||||
|
int config_font_mask;
|
||||||
|
gboolean tooltip_enabled;
|
||||||
|
} GlobalTask;
|
||||||
|
|
||||||
// --------------------------------------------------
|
// Stores information about a task.
|
||||||
// global task parameter
|
// Warning: any dynamically allocated members are shared between the Task instances created for the same window
|
||||||
typedef struct {
|
// (if the task appears on all desktops, there will be a different instance on each desktop's taskbar).
|
||||||
Area area;
|
typedef struct Task {
|
||||||
|
Area area;
|
||||||
int text;
|
Window win;
|
||||||
int icon;
|
int desktop;
|
||||||
int centered;
|
TaskState current_state;
|
||||||
|
Imlib_Image icon[TASK_STATE_COUNT];
|
||||||
int icon_posy;
|
Imlib_Image icon_hover[TASK_STATE_COUNT];
|
||||||
int icon_size1;
|
Imlib_Image icon_press[TASK_STATE_COUNT];
|
||||||
int maximum_width;
|
unsigned int icon_width;
|
||||||
int maximum_height;
|
unsigned int icon_height;
|
||||||
int alpha[TASK_STATE_COUNT];
|
char *title;
|
||||||
int saturation[TASK_STATE_COUNT];
|
int urgent_tick;
|
||||||
int brightness[TASK_STATE_COUNT];
|
// These may not be up-to-date
|
||||||
int config_asb_mask;
|
int win_x;
|
||||||
Background* background[TASK_STATE_COUNT];
|
int win_y;
|
||||||
int config_background_mask;
|
int win_w;
|
||||||
// starting position for text ~ task_padding + task_border + icon_size
|
int win_h;
|
||||||
double text_posx, text_height;
|
struct timespec last_activation_time;
|
||||||
|
int _text_width;
|
||||||
int font_shadow;
|
int _text_height;
|
||||||
PangoFontDescription *font_desc;
|
double _text_posy;
|
||||||
Color font[TASK_STATE_COUNT];
|
int _icon_x;
|
||||||
int config_font_mask;
|
int _icon_y;
|
||||||
int tooltip_enabled;
|
|
||||||
} Global_task;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
// always start with area
|
|
||||||
Area area;
|
|
||||||
|
|
||||||
// TODO: group task with list of windows here
|
|
||||||
Window win;
|
|
||||||
int desktop;
|
|
||||||
int current_state;
|
|
||||||
Imlib_Image icon[TASK_STATE_COUNT];
|
|
||||||
Pixmap state_pix[TASK_STATE_COUNT];
|
|
||||||
unsigned int icon_width;
|
|
||||||
unsigned int icon_height;
|
|
||||||
char *title;
|
|
||||||
int urgent_tick;
|
|
||||||
} Task;
|
} Task;
|
||||||
|
|
||||||
|
extern timeout *urgent_timeout;
|
||||||
|
extern GSList *urgent_list;
|
||||||
|
|
||||||
Task *add_task (Window win);
|
Task *add_task(Window win);
|
||||||
void remove_task (Task *tsk);
|
void remove_task(Task *task);
|
||||||
|
|
||||||
void draw_task (void *obj, cairo_t *c);
|
void draw_task(void *obj, cairo_t *c);
|
||||||
void on_change_task (void *obj);
|
void on_change_task(void *obj);
|
||||||
|
|
||||||
void get_icon (Task *tsk);
|
void task_update_icon(Task *task);
|
||||||
int get_title(Task *tsk);
|
gboolean task_update_title(Task *task);
|
||||||
void active_task();
|
void reset_active_task();
|
||||||
void set_task_state(Task* tsk, int state);
|
void set_task_state(Task *task, TaskState state);
|
||||||
void set_task_redraw(Task* tsk);
|
|
||||||
|
|
||||||
Task *find_active_task(Task *current_task, Task *active_task);
|
// Given a pointer to the task that is currently under the mouse (current_task),
|
||||||
Task *next_task (Task *tsk);
|
// returns a pointer to the Task for the active window on the same taskbar.
|
||||||
Task *prev_task (Task *tsk);
|
// If not found, returns the current task.
|
||||||
|
Task *find_active_task(Task *current_task);
|
||||||
|
|
||||||
void add_urgent(Task *tsk);
|
Task *next_task(Task *task);
|
||||||
void del_urgent(Task *tsk);
|
Task *prev_task(Task *task);
|
||||||
|
|
||||||
|
void add_urgent(Task *task);
|
||||||
|
void del_urgent(Task *task);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -31,361 +31,660 @@
|
|||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "window.h"
|
#include "window.h"
|
||||||
#include "panel.h"
|
#include "panel.h"
|
||||||
|
#include "strnatcmp.h"
|
||||||
|
|
||||||
|
GHashTable *win_to_task;
|
||||||
|
|
||||||
/* win_to_task_table holds for every Window an array of tasks. Usually the array contains only one
|
Task *active_task;
|
||||||
element. However for omnipresent windows (windows which are visible in every taskbar) the array
|
|
||||||
contains to every Task* on each panel a pointer (i.e. GPtrArray.len == server.nb_desktop)
|
|
||||||
*/
|
|
||||||
GHashTable* win_to_task_table;
|
|
||||||
|
|
||||||
Task *task_active;
|
|
||||||
Task *task_drag;
|
Task *task_drag;
|
||||||
int taskbar_enabled;
|
gboolean taskbar_enabled;
|
||||||
|
gboolean taskbar_distribute_size;
|
||||||
|
gboolean hide_inactive_tasks;
|
||||||
|
gboolean hide_task_diff_monitor;
|
||||||
|
gboolean hide_taskbar_if_empty;
|
||||||
|
gboolean always_show_all_desktop_tasks;
|
||||||
|
TaskbarSortMethod taskbar_sort_method;
|
||||||
|
Alignment taskbar_alignment;
|
||||||
|
|
||||||
guint win_hash(gconstpointer key) { return (guint)*((Window*)key); }
|
void taskbar_init_fonts();
|
||||||
gboolean win_compare(gconstpointer a, gconstpointer b) { return (*((Window*)a) == *((Window*)b)); }
|
int taskbar_compute_desired_size(void *obj);
|
||||||
void free_ptr_array(gpointer data) { g_ptr_array_free(data, 1); }
|
|
||||||
|
|
||||||
|
// Removes the task with &win = key. The other args are ignored.
|
||||||
|
void taskbar_remove_task(Window *win);
|
||||||
|
|
||||||
|
guint win_hash(gconstpointer key)
|
||||||
|
{
|
||||||
|
return *((const Window *)key);
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean win_compare(gconstpointer a, gconstpointer b)
|
||||||
|
{
|
||||||
|
return (*((const Window *)a) == *((const Window *)b));
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_ptr_array(gpointer data)
|
||||||
|
{
|
||||||
|
g_ptr_array_free(data, 1);
|
||||||
|
}
|
||||||
|
|
||||||
void default_taskbar()
|
void default_taskbar()
|
||||||
{
|
{
|
||||||
win_to_task_table = 0;
|
win_to_task = NULL;
|
||||||
urgent_timeout = 0;
|
urgent_timeout = NULL;
|
||||||
urgent_list = 0;
|
urgent_list = NULL;
|
||||||
taskbar_enabled = 0;
|
taskbar_enabled = FALSE;
|
||||||
default_taskbarname();
|
taskbar_distribute_size = FALSE;
|
||||||
|
hide_inactive_tasks = FALSE;
|
||||||
|
hide_task_diff_monitor = FALSE;
|
||||||
|
hide_taskbar_if_empty = FALSE;
|
||||||
|
always_show_all_desktop_tasks = FALSE;
|
||||||
|
taskbar_sort_method = TASKBAR_NOSORT;
|
||||||
|
taskbar_alignment = ALIGN_LEFT;
|
||||||
|
default_taskbarname();
|
||||||
}
|
}
|
||||||
|
|
||||||
void cleanup_taskbar()
|
void cleanup_taskbar()
|
||||||
{
|
{
|
||||||
Panel *panel;
|
if (win_to_task) {
|
||||||
Taskbar *tskbar;
|
while (g_hash_table_size(win_to_task)) {
|
||||||
int i, j, k;
|
GHashTableIter iter;
|
||||||
|
gpointer key, value;
|
||||||
|
|
||||||
cleanup_taskbarname();
|
g_hash_table_iter_init(&iter, win_to_task);
|
||||||
if (win_to_task_table) g_hash_table_foreach(win_to_task_table, taskbar_remove_task, 0);
|
if (g_hash_table_iter_next(&iter, &key, &value)) {
|
||||||
for (i=0 ; i < nb_panel ; i++) {
|
taskbar_remove_task(key);
|
||||||
panel = &panel1[i];
|
}
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
}
|
||||||
tskbar = &panel->taskbar[j];
|
g_hash_table_destroy(win_to_task);
|
||||||
for (k=0; k<TASKBAR_STATE_COUNT; ++k) {
|
win_to_task = NULL;
|
||||||
if (tskbar->state_pix[k]) XFreePixmap(server.dsp, tskbar->state_pix[k]);
|
}
|
||||||
}
|
cleanup_taskbarname();
|
||||||
free_area (&tskbar->area);
|
for (int i = 0; i < num_panels; i++) {
|
||||||
// remove taskbar from the panel
|
Panel *panel = &panels[i];
|
||||||
panel->area.list = g_slist_remove(panel->area.list, tskbar);
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
}
|
Taskbar *taskbar = &panel->taskbar[j];
|
||||||
if (panel->taskbar) {
|
free_area(&taskbar->area);
|
||||||
free(panel->taskbar);
|
// remove taskbar from the panel
|
||||||
panel->taskbar = 0;
|
remove_area((Area *)taskbar);
|
||||||
}
|
}
|
||||||
}
|
if (panel->taskbar) {
|
||||||
|
free(panel->taskbar);
|
||||||
|
panel->taskbar = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (win_to_task_table) {
|
g_slist_free(urgent_list);
|
||||||
g_hash_table_destroy(win_to_task_table);
|
urgent_list = NULL;
|
||||||
win_to_task_table = 0;
|
|
||||||
}
|
stop_timeout(urgent_timeout);
|
||||||
|
|
||||||
|
for (int state = 0; state < TASK_STATE_COUNT; state++) {
|
||||||
|
g_list_free(panel_config.g_task.gradient[state]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int state = 0; state < TASKBAR_STATE_COUNT; state++) {
|
||||||
|
g_list_free(panel_config.g_taskbar.gradient[state]);
|
||||||
|
g_list_free(panel_config.g_taskbar.gradient_name[state]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_taskbar()
|
void init_taskbar()
|
||||||
{
|
{
|
||||||
if (win_to_task_table == 0)
|
if (!panel_config.g_task.has_text && !panel_config.g_task.has_icon) {
|
||||||
win_to_task_table = g_hash_table_new_full(win_hash, win_compare, free, free_ptr_array);
|
panel_config.g_task.has_text = panel_config.g_task.has_icon = 1;
|
||||||
|
}
|
||||||
|
|
||||||
task_active = 0;
|
if (!win_to_task)
|
||||||
task_drag = 0;
|
win_to_task = g_hash_table_new_full(win_hash, win_compare, free, free_ptr_array);
|
||||||
|
|
||||||
|
active_task = 0;
|
||||||
|
task_drag = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_taskbar_panel(void *p)
|
void init_taskbar_panel(void *p)
|
||||||
{
|
{
|
||||||
Panel *panel =(Panel*)p;
|
Panel *panel = (Panel *)p;
|
||||||
int j;
|
|
||||||
|
|
||||||
if (panel->g_taskbar.background[TASKBAR_NORMAL] == 0) {
|
if (!panel->g_taskbar.background[TASKBAR_NORMAL]) {
|
||||||
panel->g_taskbar.background[TASKBAR_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
panel->g_taskbar.background[TASKBAR_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
||||||
panel->g_taskbar.background[TASKBAR_ACTIVE] = &g_array_index(backgrounds, Background, 0);
|
panel->g_taskbar.background[TASKBAR_ACTIVE] = &g_array_index(backgrounds, Background, 0);
|
||||||
}
|
}
|
||||||
if (panel->g_taskbar.background_name[TASKBAR_NORMAL] == 0) {
|
if (!panel->g_taskbar.background_name[TASKBAR_NORMAL]) {
|
||||||
panel->g_taskbar.background_name[TASKBAR_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
panel->g_taskbar.background_name[TASKBAR_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
||||||
panel->g_taskbar.background_name[TASKBAR_ACTIVE] = &g_array_index(backgrounds, Background, 0);
|
panel->g_taskbar.background_name[TASKBAR_ACTIVE] = &g_array_index(backgrounds, Background, 0);
|
||||||
}
|
}
|
||||||
if (panel->g_task.area.bg == 0)
|
if (!panel->g_task.area.bg)
|
||||||
panel->g_task.area.bg = &g_array_index(backgrounds, Background, 0);
|
panel->g_task.area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
taskbar_init_fonts();
|
||||||
|
|
||||||
// taskbar name
|
// taskbar name
|
||||||
panel->g_taskbar.area_name.panel = panel;
|
panel->g_taskbar.area_name.panel = panel;
|
||||||
panel->g_taskbar.area_name.size_mode = SIZE_BY_CONTENT;
|
snprintf(panel->g_taskbar.area_name.name, sizeof(panel->g_taskbar.area_name.name), "Taskbarname");
|
||||||
panel->g_taskbar.area_name._resize = resize_taskbarname;
|
panel->g_taskbar.area_name.size_mode = LAYOUT_FIXED;
|
||||||
panel->g_taskbar.area_name._draw_foreground = draw_taskbarname;
|
panel->g_taskbar.area_name._resize = resize_taskbarname;
|
||||||
panel->g_taskbar.area_name._on_change_layout = 0;
|
panel->g_taskbar.area_name._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
panel->g_taskbar.area_name.resize = 1;
|
panel->g_taskbar.area_name._draw_foreground = draw_taskbarname;
|
||||||
panel->g_taskbar.area_name.on_screen = 1;
|
panel->g_taskbar.area_name._on_change_layout = 0;
|
||||||
|
panel->g_taskbar.area_name.resize_needed = 1;
|
||||||
|
panel->g_taskbar.area_name.on_screen = TRUE;
|
||||||
|
|
||||||
// taskbar
|
// taskbar
|
||||||
panel->g_taskbar.area.parent = panel;
|
panel->g_taskbar.area.parent = panel;
|
||||||
panel->g_taskbar.area.panel = panel;
|
panel->g_taskbar.area.panel = panel;
|
||||||
panel->g_taskbar.area.size_mode = SIZE_BY_LAYOUT;
|
snprintf(panel->g_taskbar.area.name, sizeof(panel->g_taskbar.area.name), "Taskbar");
|
||||||
panel->g_taskbar.area._resize = resize_taskbar;
|
panel->g_taskbar.area.size_mode = LAYOUT_DYNAMIC;
|
||||||
panel->g_taskbar.area._draw_foreground = draw_taskbar;
|
panel->g_taskbar.area.alignment = taskbar_alignment;
|
||||||
panel->g_taskbar.area._on_change_layout = on_change_taskbar;
|
panel->g_taskbar.area._resize = resize_taskbar;
|
||||||
panel->g_taskbar.area.resize = 1;
|
panel->g_taskbar.area._compute_desired_size = taskbar_compute_desired_size;
|
||||||
panel->g_taskbar.area.on_screen = 1;
|
panel->g_taskbar.area._is_under_mouse = full_width_area_is_under_mouse;
|
||||||
if (panel_horizontal) {
|
panel->g_taskbar.area.resize_needed = 1;
|
||||||
panel->g_taskbar.area.posy = panel->area.bg->border.width + panel->area.paddingy;
|
panel->g_taskbar.area.on_screen = TRUE;
|
||||||
panel->g_taskbar.area.height = panel->area.height - (2 * panel->g_taskbar.area.posy);
|
if (panel_horizontal) {
|
||||||
panel->g_taskbar.area_name.posy = panel->g_taskbar.area.posy;
|
panel->g_taskbar.area.posy = top_border_width(&panel->area) + panel->area.paddingy;
|
||||||
panel->g_taskbar.area_name.height = panel->g_taskbar.area.height;
|
panel->g_taskbar.area.height =
|
||||||
}
|
panel->area.height - top_bottom_border_width(&panel->area) - 2 * panel->area.paddingy;
|
||||||
else {
|
panel->g_taskbar.area_name.posy = panel->g_taskbar.area.posy;
|
||||||
panel->g_taskbar.area.posx = panel->area.bg->border.width + panel->area.paddingy;
|
panel->g_taskbar.area_name.height = panel->g_taskbar.area.height;
|
||||||
panel->g_taskbar.area.width = panel->area.width - (2 * panel->g_taskbar.area.posx);
|
} else {
|
||||||
panel->g_taskbar.area_name.posx = panel->g_taskbar.area.posx;
|
panel->g_taskbar.area.posx = left_border_width(&panel->area) + panel->area.paddingy;
|
||||||
panel->g_taskbar.area_name.width = panel->g_taskbar.area.width;
|
panel->g_taskbar.area.width =
|
||||||
}
|
panel->area.width - left_right_border_width(&panel->area) - 2 * panel->area.paddingy;
|
||||||
|
panel->g_taskbar.area_name.posx = panel->g_taskbar.area.posx;
|
||||||
|
panel->g_taskbar.area_name.width = panel->g_taskbar.area.width;
|
||||||
|
}
|
||||||
|
|
||||||
// task
|
// task
|
||||||
panel->g_task.area.panel = panel;
|
panel->g_task.area.panel = panel;
|
||||||
panel->g_task.area.size_mode = SIZE_BY_LAYOUT;
|
snprintf(panel->g_task.area.name, sizeof(panel->g_task.area.name), "Task");
|
||||||
panel->g_task.area._draw_foreground = draw_task;
|
panel->g_task.area.size_mode = LAYOUT_DYNAMIC;
|
||||||
panel->g_task.area._on_change_layout = on_change_task;
|
panel->g_task.area._draw_foreground = draw_task;
|
||||||
panel->g_task.area.resize = 1;
|
panel->g_task.area._on_change_layout = on_change_task;
|
||||||
panel->g_task.area.on_screen = 1;
|
panel->g_task.area.resize_needed = 1;
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_NORMAL)) == 0) {
|
panel->g_task.area.on_screen = TRUE;
|
||||||
panel->g_task.alpha[TASK_NORMAL] = 100;
|
if ((panel->g_task.config_asb_mask & (1 << TASK_NORMAL)) == 0) {
|
||||||
panel->g_task.saturation[TASK_NORMAL] = 0;
|
panel->g_task.alpha[TASK_NORMAL] = 100;
|
||||||
panel->g_task.brightness[TASK_NORMAL] = 0;
|
panel->g_task.saturation[TASK_NORMAL] = 0;
|
||||||
}
|
panel->g_task.brightness[TASK_NORMAL] = 0;
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_ACTIVE)) == 0) {
|
}
|
||||||
panel->g_task.alpha[TASK_ACTIVE] = panel->g_task.alpha[TASK_NORMAL];
|
if ((panel->g_task.config_asb_mask & (1 << TASK_ACTIVE)) == 0) {
|
||||||
panel->g_task.saturation[TASK_ACTIVE] = panel->g_task.saturation[TASK_NORMAL];
|
panel->g_task.alpha[TASK_ACTIVE] = panel->g_task.alpha[TASK_NORMAL];
|
||||||
panel->g_task.brightness[TASK_ACTIVE] = panel->g_task.brightness[TASK_NORMAL];
|
panel->g_task.saturation[TASK_ACTIVE] = panel->g_task.saturation[TASK_NORMAL];
|
||||||
}
|
panel->g_task.brightness[TASK_ACTIVE] = panel->g_task.brightness[TASK_NORMAL];
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_ICONIFIED)) == 0) {
|
}
|
||||||
panel->g_task.alpha[TASK_ICONIFIED] = panel->g_task.alpha[TASK_NORMAL];
|
if ((panel->g_task.config_asb_mask & (1 << TASK_ICONIFIED)) == 0) {
|
||||||
panel->g_task.saturation[TASK_ICONIFIED] = panel->g_task.saturation[TASK_NORMAL];
|
panel->g_task.alpha[TASK_ICONIFIED] = panel->g_task.alpha[TASK_NORMAL];
|
||||||
panel->g_task.brightness[TASK_ICONIFIED] = panel->g_task.brightness[TASK_NORMAL];
|
panel->g_task.saturation[TASK_ICONIFIED] = panel->g_task.saturation[TASK_NORMAL];
|
||||||
}
|
panel->g_task.brightness[TASK_ICONIFIED] = panel->g_task.brightness[TASK_NORMAL];
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_URGENT)) == 0) {
|
}
|
||||||
panel->g_task.alpha[TASK_URGENT] = panel->g_task.alpha[TASK_ACTIVE];
|
if ((panel->g_task.config_asb_mask & (1 << TASK_URGENT)) == 0) {
|
||||||
panel->g_task.saturation[TASK_URGENT] = panel->g_task.saturation[TASK_ACTIVE];
|
panel->g_task.alpha[TASK_URGENT] = panel->g_task.alpha[TASK_ACTIVE];
|
||||||
panel->g_task.brightness[TASK_URGENT] = panel->g_task.brightness[TASK_ACTIVE];
|
panel->g_task.saturation[TASK_URGENT] = panel->g_task.saturation[TASK_ACTIVE];
|
||||||
}
|
panel->g_task.brightness[TASK_URGENT] = panel->g_task.brightness[TASK_ACTIVE];
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_NORMAL)) == 0) panel->g_task.font[TASK_NORMAL] = (Color){{0, 0, 0}, 0};
|
}
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_ACTIVE)) == 0) panel->g_task.font[TASK_ACTIVE] = panel->g_task.font[TASK_NORMAL];
|
if ((panel->g_task.config_font_mask & (1 << TASK_NORMAL)) == 0)
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_ICONIFIED)) == 0) panel->g_task.font[TASK_ICONIFIED] = panel->g_task.font[TASK_NORMAL];
|
panel->g_task.font[TASK_NORMAL] = (Color){{1, 1, 1}, 1};
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_URGENT)) == 0) panel->g_task.font[TASK_URGENT] = panel->g_task.font[TASK_ACTIVE];
|
if ((panel->g_task.config_font_mask & (1 << TASK_ACTIVE)) == 0)
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_NORMAL)) == 0) panel->g_task.background[TASK_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
panel->g_task.font[TASK_ACTIVE] = panel->g_task.font[TASK_NORMAL];
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_ACTIVE)) == 0) panel->g_task.background[TASK_ACTIVE] = panel->g_task.background[TASK_NORMAL];
|
if ((panel->g_task.config_font_mask & (1 << TASK_ICONIFIED)) == 0)
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_ICONIFIED)) == 0) panel->g_task.background[TASK_ICONIFIED] = panel->g_task.background[TASK_NORMAL];
|
panel->g_task.font[TASK_ICONIFIED] = panel->g_task.font[TASK_NORMAL];
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_URGENT)) == 0) panel->g_task.background[TASK_URGENT] = panel->g_task.background[TASK_ACTIVE];
|
if ((panel->g_task.config_font_mask & (1 << TASK_URGENT)) == 0)
|
||||||
|
panel->g_task.font[TASK_URGENT] = panel->g_task.font[TASK_ACTIVE];
|
||||||
|
if ((panel->g_task.config_background_mask & (1 << TASK_NORMAL)) == 0)
|
||||||
|
panel->g_task.background[TASK_NORMAL] = &g_array_index(backgrounds, Background, 0);
|
||||||
|
if ((panel->g_task.config_background_mask & (1 << TASK_ACTIVE)) == 0)
|
||||||
|
panel->g_task.background[TASK_ACTIVE] = panel->g_task.background[TASK_NORMAL];
|
||||||
|
if ((panel->g_task.config_background_mask & (1 << TASK_ICONIFIED)) == 0)
|
||||||
|
panel->g_task.background[TASK_ICONIFIED] = panel->g_task.background[TASK_NORMAL];
|
||||||
|
if ((panel->g_task.config_background_mask & (1 << TASK_URGENT)) == 0)
|
||||||
|
panel->g_task.background[TASK_URGENT] = panel->g_task.background[TASK_ACTIVE];
|
||||||
|
|
||||||
if (panel_horizontal) {
|
if (!panel->g_task.maximum_width)
|
||||||
panel->g_task.area.posy = panel->g_taskbar.area.posy + panel->g_taskbar.background[TASKBAR_NORMAL]->border.width + panel->g_taskbar.area.paddingy;
|
panel->g_task.maximum_width = server.monitors[panel->monitor].width;
|
||||||
panel->g_task.area.height = panel->area.height - (2 * panel->g_task.area.posy);
|
if (!panel->g_task.maximum_height)
|
||||||
}
|
panel->g_task.maximum_height = server.monitors[panel->monitor].height;
|
||||||
else {
|
|
||||||
panel->g_task.area.posx = panel->g_taskbar.area.posx + panel->g_taskbar.background[TASKBAR_NORMAL]->border.width + panel->g_taskbar.area.paddingy;
|
|
||||||
panel->g_task.area.width = panel->area.width - (2 * panel->g_task.area.posx);
|
|
||||||
panel->g_task.area.height = panel->g_task.maximum_height;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (j=0; j<TASK_STATE_COUNT; ++j) {
|
if (panel_horizontal) {
|
||||||
if (panel->g_task.background[j] == 0)
|
panel->g_task.area.posy = panel->g_taskbar.area.posy +
|
||||||
panel->g_task.background[j] = &g_array_index(backgrounds, Background, 0);
|
top_bg_border_width(panel->g_taskbar.background[TASKBAR_NORMAL]) +
|
||||||
if (panel->g_task.background[j]->border.rounded > panel->g_task.area.height/2) {
|
panel->g_taskbar.area.paddingy;
|
||||||
printf("task%sbackground_id has a too large rounded value. Please fix your tint2rc\n", j==0 ? "_" : j==1 ? "_active_" : j==2 ? "_iconified_" : "_urgent_");
|
panel->g_task.area.width = panel->g_task.maximum_width;
|
||||||
g_array_append_val(backgrounds, *panel->g_task.background[j]);
|
panel->g_task.area.height = panel->g_taskbar.area.height -
|
||||||
panel->g_task.background[j] = &g_array_index(backgrounds, Background, backgrounds->len-1);
|
top_bottom_bg_border_width(panel->g_taskbar.background[TASKBAR_NORMAL]) -
|
||||||
panel->g_task.background[j]->border.rounded = panel->g_task.area.height/2;
|
2 * panel->g_taskbar.area.paddingy;
|
||||||
}
|
} else {
|
||||||
}
|
panel->g_task.area.posx = panel->g_taskbar.area.posx +
|
||||||
|
left_bg_border_width(panel->g_taskbar.background[TASKBAR_NORMAL]) +
|
||||||
|
panel->g_taskbar.area.paddingy;
|
||||||
|
panel->g_task.area.width = panel->g_taskbar.area.width -
|
||||||
|
left_right_bg_border_width(panel->g_taskbar.background[TASKBAR_NORMAL]) -
|
||||||
|
2 * panel->g_taskbar.area.paddingy;
|
||||||
|
panel->g_task.area.height = panel->g_task.maximum_height;
|
||||||
|
}
|
||||||
|
|
||||||
// compute vertical position : text and icon
|
for (int j = 0; j < TASK_STATE_COUNT; ++j) {
|
||||||
int height_ink, height;
|
if (!panel->g_task.background[j])
|
||||||
get_text_size(panel->g_task.font_desc, &height_ink, &height, panel->area.height, "TAjpg", 5);
|
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) {
|
||||||
|
printf("task%sbackground_id has a too large rounded value. Please fix your tint2rc\n",
|
||||||
|
j == 0 ? "_" : j == 1 ? "_active_" : j == 2 ? "_iconified_" : "_urgent_");
|
||||||
|
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]->border.radius = panel->g_task.area.height / 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!panel->g_task.maximum_width && panel_horizontal)
|
// compute vertical position : text and icon
|
||||||
panel->g_task.maximum_width = server.monitor[panel->monitor].width;
|
int height_ink, height, width;
|
||||||
|
get_text_size2(panel->g_task.font_desc,
|
||||||
|
&height_ink,
|
||||||
|
&height,
|
||||||
|
&width,
|
||||||
|
panel->area.height,
|
||||||
|
panel->area.width,
|
||||||
|
"TAjpg",
|
||||||
|
5,
|
||||||
|
PANGO_WRAP_WORD_CHAR,
|
||||||
|
PANGO_ELLIPSIZE_END,
|
||||||
|
FALSE);
|
||||||
|
|
||||||
panel->g_task.text_posx = panel->g_task.background[0]->border.width + 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);
|
||||||
if (panel->g_task.icon) {
|
if (panel->g_task.has_icon) {
|
||||||
panel->g_task.icon_size1 = panel->g_task.area.height - (2 * panel->g_task.area.paddingy);
|
panel->g_task.icon_size1 = MIN(MIN(panel->g_task.maximum_width, panel->g_task.maximum_height),
|
||||||
panel->g_task.text_posx += panel->g_task.icon_size1;
|
MIN(panel->g_task.area.width, panel->g_task.area.height)) -
|
||||||
panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2;
|
2 * panel->g_task.area.paddingy - MAX(left_right_border_width(&panel->g_task.area),
|
||||||
}
|
top_bottom_border_width(&panel->g_task.area));
|
||||||
//printf("monitor %d, task_maximum_width %d\n", panel->monitor, panel->g_task.maximum_width);
|
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;
|
||||||
|
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 *tskbar;
|
Taskbar *taskbar;
|
||||||
panel->nb_desktop = server.nb_desktop;
|
panel->num_desktops = server.num_desktops;
|
||||||
panel->taskbar = calloc(server.nb_desktop, sizeof(Taskbar));
|
panel->taskbar = calloc(server.num_desktops, sizeof(Taskbar));
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
tskbar = &panel->taskbar[j];
|
taskbar = &panel->taskbar[j];
|
||||||
memcpy(&tskbar->area, &panel->g_taskbar.area, sizeof(Area));
|
memcpy(&taskbar->area, &panel->g_taskbar.area, sizeof(Area));
|
||||||
tskbar->desktop = j;
|
taskbar->desktop = j;
|
||||||
if (j == server.desktop)
|
if (j == server.desktop) {
|
||||||
tskbar->area.bg = panel->g_taskbar.background[TASKBAR_ACTIVE];
|
taskbar->area.bg = panel->g_taskbar.background[TASKBAR_ACTIVE];
|
||||||
else
|
free_area_gradient_instances(&taskbar->area);
|
||||||
tskbar->area.bg = panel->g_taskbar.background[TASKBAR_NORMAL];
|
instantiate_area_gradients(&taskbar->area);
|
||||||
}
|
} else {
|
||||||
init_taskbarname_panel(panel);
|
taskbar->area.bg = panel->g_taskbar.background[TASKBAR_NORMAL];
|
||||||
|
free_area_gradient_instances(&taskbar->area);
|
||||||
|
instantiate_area_gradients(&taskbar->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
init_taskbarname_panel(panel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void taskbar_init_fonts()
|
||||||
void taskbar_remove_task(gpointer key, gpointer value, gpointer user_data)
|
|
||||||
{
|
{
|
||||||
remove_task(task_get_task(*(Window*)key));
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
if (!panels[i].g_task.font_desc) {
|
||||||
|
panels[i].g_task.font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
pango_font_description_set_size(panels[i].g_task.font_desc,
|
||||||
|
pango_font_description_get_size(panels[i].g_task.font_desc) - PANGO_SCALE);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void taskbar_default_font_changed()
|
||||||
Task *task_get_task (Window win)
|
|
||||||
{
|
{
|
||||||
GPtrArray* task_group = task_get_tasks(win);
|
if (!taskbar_enabled)
|
||||||
if (task_group)
|
return;
|
||||||
return g_ptr_array_index(task_group, 0);
|
|
||||||
else
|
gboolean needs_update = FALSE;
|
||||||
return 0;
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
if (!panels[i].g_task.has_font) {
|
||||||
|
pango_font_description_free(panels[i].g_task.font_desc);
|
||||||
|
panels[i].g_task.font_desc = NULL;
|
||||||
|
needs_update = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!needs_update)
|
||||||
|
return;
|
||||||
|
taskbar_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
for (int j = 0; j < panels[i].num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panels[i].taskbar[j];
|
||||||
|
for (GList *c = taskbar->area.children; c; c = c->next) {
|
||||||
|
Task *t = c->data;
|
||||||
|
t->area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&t->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void taskbar_remove_task(Window *win)
|
||||||
GPtrArray* task_get_tasks(Window win)
|
|
||||||
{
|
{
|
||||||
if (win_to_task_table && taskbar_enabled)
|
remove_task(get_task(*win));
|
||||||
return g_hash_table_lookup(win_to_task_table, &win);
|
|
||||||
else
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Task *get_task(Window win)
|
||||||
void task_refresh_tasklist ()
|
|
||||||
{
|
{
|
||||||
Window *win;
|
GPtrArray *task_buttons = get_task_buttons(win);
|
||||||
int num_results, i;
|
if (task_buttons)
|
||||||
|
return g_ptr_array_index(task_buttons, 0);
|
||||||
if (!taskbar_enabled) return;
|
return NULL;
|
||||||
win = server_get_property (server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results);
|
|
||||||
if (!win) return;
|
|
||||||
|
|
||||||
GList* win_list = g_hash_table_get_keys(win_to_task_table);
|
|
||||||
GList* it;
|
|
||||||
for (it=win_list; it; it=it->next) {
|
|
||||||
for (i = 0; i < num_results; i++)
|
|
||||||
if (*((Window*)it->data) == win[i])
|
|
||||||
break;
|
|
||||||
if (i == num_results)
|
|
||||||
taskbar_remove_task(it->data, 0, 0);
|
|
||||||
}
|
|
||||||
g_list_free(win_list);
|
|
||||||
|
|
||||||
// Add any new
|
|
||||||
for (i = 0; i < num_results; i++)
|
|
||||||
if (!task_get_task (win[i]))
|
|
||||||
add_task (win[i]);
|
|
||||||
|
|
||||||
XFree (win);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GPtrArray *get_task_buttons(Window win)
|
||||||
void draw_taskbar (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Taskbar *taskbar = obj;
|
if (win_to_task && taskbar_enabled)
|
||||||
int state = (taskbar->desktop == server.desktop) ? TASKBAR_ACTIVE : TASKBAR_NORMAL;
|
return g_hash_table_lookup(win_to_task, &win);
|
||||||
|
return NULL;
|
||||||
taskbar->state_pix[state] = taskbar->area.pix;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void taskbar_refresh_tasklist()
|
||||||
int resize_taskbar(void *obj)
|
|
||||||
{
|
{
|
||||||
Taskbar *taskbar = (Taskbar*)obj;
|
if (!taskbar_enabled)
|
||||||
Panel *panel = (Panel*)taskbar->area.panel;
|
return;
|
||||||
int text_width;
|
// fprintf(stderr, "%s %d:\n", __FUNCTION__, __LINE__);
|
||||||
|
|
||||||
//printf("resize_taskbar %d %d\n", taskbar->area.posx, taskbar->area.posy);
|
int num_results;
|
||||||
if (panel_horizontal) {
|
Window *win = server_get_property(server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results);
|
||||||
resize_by_layout(obj, panel->g_task.maximum_width);
|
if (!win)
|
||||||
|
return;
|
||||||
text_width = panel->g_task.maximum_width;
|
|
||||||
GSList *l = taskbar->area.list;
|
GList *win_list = g_hash_table_get_keys(win_to_task);
|
||||||
if (taskbarname_enabled) l = l->next;
|
for (GList *it = win_list; it; it = it->next) {
|
||||||
if (l != NULL) {
|
int i;
|
||||||
text_width = ((Task *)l->data)->area.width;
|
for (i = 0; i < num_results; i++)
|
||||||
}
|
if (*((Window *)it->data) == win[i])
|
||||||
taskbar->text_width = text_width - panel->g_task.text_posx - panel->g_task.area.bg->border.width - panel->g_task.area.paddingx;
|
break;
|
||||||
}
|
if (i == num_results)
|
||||||
else {
|
taskbar_remove_task(it->data);
|
||||||
resize_by_layout(obj, panel->g_task.maximum_height);
|
}
|
||||||
|
g_list_free(win_list);
|
||||||
taskbar->text_width = taskbar->area.width - (2 * panel->g_taskbar.area.paddingy) - panel->g_task.text_posx - panel->g_task.area.bg->border.width - panel->g_task.area.paddingx;
|
|
||||||
}
|
// Add any new
|
||||||
return 0;
|
for (int i = 0; i < num_results; i++)
|
||||||
|
if (!get_task(win[i]))
|
||||||
|
add_task(win[i]);
|
||||||
|
|
||||||
|
XFree(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int taskbar_compute_desired_size(void *obj)
|
||||||
void on_change_taskbar (void *obj)
|
|
||||||
{
|
{
|
||||||
Taskbar *tskbar = obj;
|
Taskbar *taskbar = (Taskbar *)obj;
|
||||||
int k;
|
Panel *panel = (Panel *)taskbar->area.panel;
|
||||||
|
|
||||||
// reset Pixmap when position/size changed
|
if (taskbar_mode == MULTI_DESKTOP && !taskbar_distribute_size) {
|
||||||
for (k=0; k<TASKBAR_STATE_COUNT; ++k) {
|
int result = 0;
|
||||||
if (tskbar->state_pix[k]) XFreePixmap(server.dsp, tskbar->state_pix[k]);
|
for (int i = 0; i < panel->num_desktops; i++) {
|
||||||
tskbar->state_pix[k] = 0;
|
Taskbar *t = &panel->taskbar[i];
|
||||||
}
|
result = MAX(result, container_compute_desired_size(&t->area));
|
||||||
tskbar->area.pix = 0;
|
}
|
||||||
tskbar->area.redraw = 1;
|
return result;
|
||||||
|
}
|
||||||
|
return container_compute_desired_size(&taskbar->area);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean resize_taskbar(void *obj)
|
||||||
void set_taskbar_state(Taskbar *tskbar, int state)
|
|
||||||
{
|
{
|
||||||
tskbar->area.bg = panel1[0].g_taskbar.background[state];
|
Taskbar *taskbar = (Taskbar *)obj;
|
||||||
tskbar->area.pix = tskbar->state_pix[state];
|
Panel *panel = (Panel *)taskbar->area.panel;
|
||||||
if (taskbarname_enabled) {
|
|
||||||
tskbar->bar_name.area.bg = panel1[0].g_taskbar.background_name[state];
|
// printf("resize_taskbar %d %d\n", taskbar->area.posx, taskbar->area.posy);
|
||||||
tskbar->bar_name.area.pix = tskbar->bar_name.state_pix[state];
|
if (panel_horizontal) {
|
||||||
}
|
relayout_with_constraint(&taskbar->area, panel->g_task.maximum_width);
|
||||||
if (panel_mode != MULTI_DESKTOP) {
|
|
||||||
if (state == TASKBAR_NORMAL)
|
int text_width = panel->g_task.maximum_width;
|
||||||
tskbar->area.on_screen = 0;
|
GList *l = taskbar->area.children;
|
||||||
else
|
if (taskbarname_enabled)
|
||||||
tskbar->area.on_screen = 1;
|
l = l->next;
|
||||||
}
|
for (; l != NULL; l = l->next) {
|
||||||
if (tskbar->area.on_screen == 1) {
|
if (((Task *)l->data)->area.on_screen) {
|
||||||
if (tskbar->state_pix[state] == 0)
|
text_width = ((Task *)l->data)->area.width;
|
||||||
tskbar->area.redraw = 1;
|
break;
|
||||||
if (taskbarname_enabled && tskbar->bar_name.state_pix[state] == 0)
|
}
|
||||||
tskbar->bar_name.area.redraw = 1;
|
}
|
||||||
if (panel_mode == MULTI_DESKTOP && panel1[0].g_taskbar.background[TASKBAR_NORMAL] != panel1[0].g_taskbar.background[TASKBAR_ACTIVE]) {
|
taskbar->text_width = text_width - panel->g_task.text_posx - right_border_width(&panel->g_task.area) -
|
||||||
GSList *l = tskbar->area.list;
|
panel->g_task.area.paddingxlr;
|
||||||
if (taskbarname_enabled) l = l->next;
|
} else {
|
||||||
for ( ; l ; l = l->next)
|
relayout_with_constraint(&taskbar->area, panel->g_task.maximum_height);
|
||||||
set_task_redraw(l->data);
|
|
||||||
}
|
taskbar->text_width = taskbar->area.width - (2 * panel->g_taskbar.area.paddingy) - panel->g_task.text_posx -
|
||||||
}
|
right_border_width(&panel->g_task.area) - panel->g_task.area.paddingxlr;
|
||||||
panel_refresh = 1;
|
}
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean taskbar_is_empty(Taskbar *taskbar)
|
||||||
void visible_taskbar(void *p)
|
|
||||||
{
|
{
|
||||||
Panel *panel =(Panel*)p;
|
GList *l = taskbar->area.children;
|
||||||
int j;
|
if (taskbarname_enabled)
|
||||||
|
l = l->next;
|
||||||
Taskbar *taskbar;
|
for (; l != NULL; l = l->next) {
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
if (((Task *)l->data)->area.on_screen) {
|
||||||
taskbar = &panel->taskbar[j];
|
return FALSE;
|
||||||
if (panel_mode != MULTI_DESKTOP && taskbar->desktop != server.desktop) {
|
}
|
||||||
// SINGLE_DESKTOP and not current desktop
|
}
|
||||||
taskbar->area.on_screen = 0;
|
return TRUE;
|
||||||
}
|
|
||||||
else {
|
|
||||||
taskbar->area.on_screen = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void update_taskbar_visibility(Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
if (taskbar->desktop == server.desktop) {
|
||||||
|
// Taskbar for current desktop is always shown
|
||||||
|
show(&taskbar->area);
|
||||||
|
} else if (taskbar_mode == MULTI_DESKTOP) {
|
||||||
|
if (hide_taskbar_if_empty) {
|
||||||
|
if (taskbar_is_empty(taskbar)) {
|
||||||
|
hide(&taskbar->area);
|
||||||
|
} else {
|
||||||
|
show(&taskbar->area);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
show(&taskbar->area);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
hide(&taskbar->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void update_all_taskbars_visibility()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
Panel *panel = &panels[i];
|
||||||
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
|
update_taskbar_visibility(&panel->taskbar[j]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void set_taskbar_state(Taskbar *taskbar, TaskbarState state)
|
||||||
|
{
|
||||||
|
taskbar->area.bg = panels[0].g_taskbar.background[state];
|
||||||
|
free_area_gradient_instances(&taskbar->area);
|
||||||
|
instantiate_area_gradients(&taskbar->area);
|
||||||
|
|
||||||
|
if (taskbarname_enabled) {
|
||||||
|
taskbar->bar_name.area.bg = panels[0].g_taskbar.background_name[state];
|
||||||
|
free_area_gradient_instances(&taskbar->bar_name.area);
|
||||||
|
instantiate_area_gradients(&taskbar->bar_name.area);
|
||||||
|
}
|
||||||
|
|
||||||
|
update_taskbar_visibility(taskbar);
|
||||||
|
|
||||||
|
if (taskbar->area.on_screen) {
|
||||||
|
schedule_redraw(&taskbar->area);
|
||||||
|
if (taskbarname_enabled) {
|
||||||
|
schedule_redraw(&taskbar->bar_name.area);
|
||||||
|
}
|
||||||
|
if (taskbar_mode == MULTI_DESKTOP &&
|
||||||
|
panels[0].g_taskbar.background[TASKBAR_NORMAL] != panels[0].g_taskbar.background[TASKBAR_ACTIVE]) {
|
||||||
|
GList *l = taskbar->area.children;
|
||||||
|
if (taskbarname_enabled)
|
||||||
|
l = l->next;
|
||||||
|
for (; l; l = l->next)
|
||||||
|
schedule_redraw((Area *)l->data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
|
||||||
|
#define NONTRIVIAL 2
|
||||||
|
gint compare_tasks_trivial(Task *a, Task *b, Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
if (a == b)
|
||||||
|
return 0;
|
||||||
|
if (taskbarname_enabled) {
|
||||||
|
if (a == taskbar->area.children->data)
|
||||||
|
return -1;
|
||||||
|
if (b == taskbar->area.children->data)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return NONTRIVIAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean contained_within(Task *a, Task *b)
|
||||||
|
{
|
||||||
|
if ((a->win_x <= b->win_x) && (a->win_y <= b->win_y) && (a->win_x + a->win_w >= b->win_x + b->win_w) &&
|
||||||
|
(a->win_y + a->win_h >= b->win_y + b->win_h)) {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gint compare_task_centers(Task *a, Task *b, Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
int trivial = compare_tasks_trivial(a, b, taskbar);
|
||||||
|
if (trivial != NONTRIVIAL)
|
||||||
|
return trivial;
|
||||||
|
|
||||||
|
// If a window has the same coordinates and size as the other,
|
||||||
|
// they are considered to be equal in the comparison.
|
||||||
|
if ((a->win_x == b->win_x) && (a->win_y == b->win_y) && (a->win_w == b->win_w) && (a->win_h == b->win_h)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If a window is completely contained in another,
|
||||||
|
// then it is considered to come after (to the right/bottom) of the other.
|
||||||
|
if (contained_within(a, b))
|
||||||
|
return -1;
|
||||||
|
if (contained_within(b, a))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
// Compare centers
|
||||||
|
int a_horiz_c = a->win_x + a->win_w / 2;
|
||||||
|
int b_horiz_c = b->win_x + b->win_w / 2;
|
||||||
|
int a_vert_c = a->win_y + a->win_h / 2;
|
||||||
|
int b_vert_c = b->win_y + b->win_h / 2;
|
||||||
|
if (panel_horizontal) {
|
||||||
|
if (a_horiz_c != b_horiz_c) {
|
||||||
|
return a_horiz_c - b_horiz_c;
|
||||||
|
}
|
||||||
|
return a_vert_c - b_vert_c;
|
||||||
|
} else {
|
||||||
|
if (a_vert_c != b_vert_c) {
|
||||||
|
return a_vert_c - b_vert_c;
|
||||||
|
}
|
||||||
|
return a_horiz_c - b_horiz_c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gint compare_task_titles(Task *a, Task *b, Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
int trivial = compare_tasks_trivial(a, b, taskbar);
|
||||||
|
if (trivial != NONTRIVIAL)
|
||||||
|
return trivial;
|
||||||
|
return strnatcasecmp(a->title ? a->title : "", b->title ? b->title : "");
|
||||||
|
}
|
||||||
|
|
||||||
|
gint compare_tasks(Task *a, Task *b, Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
int trivial = compare_tasks_trivial(a, b, taskbar);
|
||||||
|
if (trivial != NONTRIVIAL)
|
||||||
|
return trivial;
|
||||||
|
if (taskbar_sort_method == TASKBAR_NOSORT) {
|
||||||
|
return 0;
|
||||||
|
} else if (taskbar_sort_method == TASKBAR_SORT_CENTER) {
|
||||||
|
return compare_task_centers(a, b, taskbar);
|
||||||
|
} else if (taskbar_sort_method == TASKBAR_SORT_TITLE) {
|
||||||
|
return compare_task_titles(a, b, taskbar);
|
||||||
|
} else if (taskbar_sort_method == TASKBAR_SORT_LRU) {
|
||||||
|
return compare_timespecs(&a->last_activation_time, &b->last_activation_time);
|
||||||
|
} else if (taskbar_sort_method == TASKBAR_SORT_MRU) {
|
||||||
|
return -compare_timespecs(&a->last_activation_time, &b->last_activation_time);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean taskbar_needs_sort(Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
if (taskbar_sort_method == TASKBAR_NOSORT)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
for (GList *i = taskbar->area.children, *j = i ? i->next : NULL; i && j; i = i->next, j = j->next) {
|
||||||
|
if (compare_tasks(i->data, j->data, taskbar) > 0) {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void sort_tasks(Taskbar *taskbar)
|
||||||
|
{
|
||||||
|
if (!taskbar)
|
||||||
|
return;
|
||||||
|
if (!taskbar_needs_sort(taskbar))
|
||||||
|
return;
|
||||||
|
|
||||||
|
taskbar->area.children = g_list_sort_with_data(taskbar->area.children, (GCompareDataFunc)compare_tasks, taskbar);
|
||||||
|
taskbar->area.resize_needed = TRUE;
|
||||||
|
schedule_panel_redraw();
|
||||||
|
((Panel *)taskbar->area.panel)->area.resize_needed = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void sort_taskbar_for_win(Window win)
|
||||||
|
{
|
||||||
|
if (taskbar_sort_method == TASKBAR_NOSORT)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GPtrArray *task_buttons = get_task_buttons(win);
|
||||||
|
if (task_buttons) {
|
||||||
|
Task *task0 = g_ptr_array_index(task_buttons, 0);
|
||||||
|
if (task0) {
|
||||||
|
get_window_coordinates(win, &task0->win_x, &task0->win_y, &task0->win_w, &task0->win_h);
|
||||||
|
}
|
||||||
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
|
Task *task = g_ptr_array_index(task_buttons, i);
|
||||||
|
task->win_x = task0->win_x;
|
||||||
|
task->win_y = task0->win_y;
|
||||||
|
task->win_w = task0->win_w;
|
||||||
|
task->win_h = task0->win_h;
|
||||||
|
sort_tasks(task->area.parent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -11,67 +11,90 @@
|
|||||||
#include "task.h"
|
#include "task.h"
|
||||||
#include "taskbarname.h"
|
#include "taskbarname.h"
|
||||||
|
|
||||||
enum { TASKBAR_NORMAL, TASKBAR_ACTIVE, TASKBAR_STATE_COUNT };
|
typedef enum TaskbarState {
|
||||||
extern GHashTable* win_to_task_table;
|
TASKBAR_NORMAL = 0,
|
||||||
extern Task *task_active;
|
TASKBAR_ACTIVE,
|
||||||
extern Task *task_drag;
|
TASKBAR_STATE_COUNT,
|
||||||
extern int taskbar_enabled;
|
} TaskbarState;
|
||||||
|
|
||||||
|
typedef enum TaskbarSortMethod {
|
||||||
|
TASKBAR_NOSORT = 0,
|
||||||
|
TASKBAR_SORT_CENTER,
|
||||||
|
TASKBAR_SORT_TITLE,
|
||||||
|
TASKBAR_SORT_LRU,
|
||||||
|
TASKBAR_SORT_MRU,
|
||||||
|
} TaskbarSortMethod;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
// always start with area
|
Area area;
|
||||||
Area area;
|
gchar *name;
|
||||||
Pixmap state_pix[TASKBAR_STATE_COUNT];
|
int posy;
|
||||||
|
} TaskbarName;
|
||||||
|
|
||||||
char *name;
|
|
||||||
int posy;
|
|
||||||
} Taskbarname;
|
|
||||||
|
|
||||||
// tint2 use one taskbar per desktop.
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
// always start with area
|
Area area;
|
||||||
Area area;
|
int desktop;
|
||||||
|
TaskbarName bar_name;
|
||||||
int desktop;
|
int text_width;
|
||||||
Pixmap state_pix[TASKBAR_STATE_COUNT];
|
|
||||||
|
|
||||||
Taskbarname bar_name;
|
|
||||||
|
|
||||||
// task parameters
|
|
||||||
int text_width;
|
|
||||||
} Taskbar;
|
} Taskbar;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct GlobalTaskbar {
|
||||||
//always start with area
|
Area area;
|
||||||
Area area;
|
Area area_name;
|
||||||
Area area_name;
|
Background *background[TASKBAR_STATE_COUNT];
|
||||||
Background* background[TASKBAR_STATE_COUNT];
|
Background *background_name[TASKBAR_STATE_COUNT];
|
||||||
Background* background_name[TASKBAR_STATE_COUNT];
|
GList *gradient[TASKBAR_STATE_COUNT];
|
||||||
} Global_taskbar;
|
GList *gradient_name[TASKBAR_STATE_COUNT];
|
||||||
|
} GlobalTaskbar;
|
||||||
|
|
||||||
|
extern gboolean taskbar_enabled;
|
||||||
|
extern gboolean taskbar_distribute_size;
|
||||||
|
extern gboolean hide_inactive_tasks;
|
||||||
|
extern gboolean hide_task_diff_monitor;
|
||||||
|
extern gboolean hide_taskbar_if_empty;
|
||||||
|
extern gboolean always_show_all_desktop_tasks;
|
||||||
|
extern TaskbarSortMethod taskbar_sort_method;
|
||||||
|
extern Alignment taskbar_alignment;
|
||||||
|
|
||||||
|
// win_to_task holds for every Window an array of tasks. Usually the array contains only one
|
||||||
|
// element. However for omnipresent windows (windows which are visible in every taskbar) the array
|
||||||
|
// contains to every Task* on each panel a pointer (i.e. GPtrArray.len == server.num_desktops)
|
||||||
|
extern GHashTable *win_to_task;
|
||||||
|
|
||||||
|
extern Task *active_task;
|
||||||
|
extern Task *task_drag;
|
||||||
|
|
||||||
// default global data
|
|
||||||
void default_taskbar();
|
void default_taskbar();
|
||||||
|
|
||||||
// freed memory
|
|
||||||
void cleanup_taskbar();
|
void cleanup_taskbar();
|
||||||
|
|
||||||
void init_taskbar();
|
void init_taskbar();
|
||||||
void init_taskbar_panel(void *p);
|
void init_taskbar_panel(void *p);
|
||||||
|
|
||||||
void draw_taskbar (void *obj, cairo_t *c);
|
gboolean resize_taskbar(void *obj);
|
||||||
void taskbar_remove_task(gpointer key, gpointer value, gpointer user_data);
|
void taskbar_default_font_changed();
|
||||||
Task *task_get_task (Window win);
|
|
||||||
GPtrArray* task_get_tasks(Window win);
|
|
||||||
void task_refresh_tasklist ();
|
|
||||||
|
|
||||||
int resize_taskbar(void *obj);
|
// Reloads the entire list of tasks from the window manager and recreates the task buttons.
|
||||||
void on_change_taskbar (void *obj);
|
void taskbar_refresh_tasklist();
|
||||||
void set_taskbar_state(Taskbar *tskbar, int state);
|
|
||||||
|
|
||||||
// show/hide taskbar according to current desktop
|
// Returns the task button for this window. If there are multiple buttons, returns the first one.
|
||||||
void visible_taskbar(void *p);
|
Task *get_task(Window win);
|
||||||
|
|
||||||
|
// Returns the task buttons for this window, usually having only one element.
|
||||||
|
// However for windows shown on all desktops, there are multiple buttons, one for each taskbar.
|
||||||
|
GPtrArray *get_task_buttons(Window win);
|
||||||
|
|
||||||
|
// Change state of a taskbar (ACTIVE or NORMAL)
|
||||||
|
void set_taskbar_state(Taskbar *taskbar, TaskbarState state);
|
||||||
|
|
||||||
|
// Updates the visibility of all taskbars
|
||||||
|
void update_all_taskbars_visibility();
|
||||||
|
|
||||||
|
void update_minimized_icon_positions(void *p);
|
||||||
|
|
||||||
|
// Sorts the taskbar(s) on which the window is present.
|
||||||
|
void sort_taskbar_for_win(Window win);
|
||||||
|
|
||||||
|
void sort_tasks(Taskbar *taskbar);
|
||||||
|
|
||||||
|
gboolean taskbar_is_under_mouse(void *obj, int x, int y);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -32,130 +32,185 @@
|
|||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "taskbarname.h"
|
#include "taskbarname.h"
|
||||||
|
|
||||||
int taskbarname_enabled;
|
gboolean taskbarname_enabled;
|
||||||
PangoFontDescription *taskbarname_font_desc;
|
|
||||||
Color taskbarname_font;
|
Color taskbarname_font;
|
||||||
Color taskbarname_active_font;
|
Color taskbarname_active_font;
|
||||||
|
|
||||||
|
void taskbarname_init_fonts();
|
||||||
|
int taskbarname_compute_desired_size(void *obj);
|
||||||
|
|
||||||
void default_taskbarname()
|
void default_taskbarname()
|
||||||
{
|
{
|
||||||
taskbarname_enabled = 0;
|
taskbarname_enabled = FALSE;
|
||||||
taskbarname_font_desc = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_taskbarname_panel(void *p)
|
void init_taskbarname_panel(void *p)
|
||||||
{
|
{
|
||||||
Panel *panel =(Panel*)p;
|
if (!taskbarname_enabled)
|
||||||
Taskbar *tskbar;
|
return;
|
||||||
int j;
|
|
||||||
|
|
||||||
if (!taskbarname_enabled) return;
|
|
||||||
|
|
||||||
GSList *l, *list = server_get_name_of_desktop();
|
Panel *panel = (Panel *)p;
|
||||||
for (j=0, l=list ; j < panel->nb_desktop ; j++) {
|
|
||||||
tskbar = &panel->taskbar[j];
|
|
||||||
memcpy(&tskbar->bar_name.area, &panel->g_taskbar.area_name, sizeof(Area));
|
|
||||||
tskbar->bar_name.area.parent = tskbar;
|
|
||||||
if (j == server.desktop)
|
|
||||||
tskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_ACTIVE];
|
|
||||||
else
|
|
||||||
tskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_NORMAL];
|
|
||||||
|
|
||||||
// use desktop number if name is missing
|
taskbarname_init_fonts();
|
||||||
if (l) {
|
|
||||||
tskbar->bar_name.name = g_strdup(l->data);
|
|
||||||
l = l->next;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
tskbar->bar_name.name = g_strdup_printf("%d", j+1);
|
|
||||||
|
|
||||||
// append the name at the beginning of taskbar
|
|
||||||
tskbar->area.list = g_slist_append(tskbar->area.list, &tskbar->bar_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (l=list ; l ; l = l->next)
|
GSList *list = get_desktop_names();
|
||||||
g_free(l->data);
|
GSList *l = list;
|
||||||
g_slist_free(list);
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panel->taskbar[j];
|
||||||
|
memcpy(&taskbar->bar_name.area, &panel->g_taskbar.area_name, sizeof(Area));
|
||||||
|
taskbar->bar_name.area.parent = taskbar;
|
||||||
|
taskbar->bar_name.area.has_mouse_over_effect = panel_config.mouse_effects;
|
||||||
|
taskbar->bar_name.area.has_mouse_press_effect = panel_config.mouse_effects;
|
||||||
|
taskbar->bar_name.area._compute_desired_size = taskbarname_compute_desired_size;
|
||||||
|
if (j == server.desktop) {
|
||||||
|
taskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_ACTIVE];
|
||||||
|
} else {
|
||||||
|
taskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_NORMAL];
|
||||||
|
}
|
||||||
|
|
||||||
|
// use desktop number if name is missing
|
||||||
|
if (l) {
|
||||||
|
taskbar->bar_name.name = g_strdup(l->data);
|
||||||
|
l = l->next;
|
||||||
|
} else {
|
||||||
|
taskbar->bar_name.name = g_strdup_printf("%d", j + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// append the name at the beginning of taskbar
|
||||||
|
taskbar->area.children = g_list_append(taskbar->area.children, &taskbar->bar_name);
|
||||||
|
instantiate_area_gradients(&taskbar->bar_name.area);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (l = list; l; l = l->next)
|
||||||
|
g_free(l->data);
|
||||||
|
g_slist_free(list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void taskbarname_init_fonts()
|
||||||
|
{
|
||||||
|
if (!panel_config.taskbarname_font_desc) {
|
||||||
|
panel_config.taskbarname_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
pango_font_description_set_weight(panel_config.taskbarname_font_desc, PANGO_WEIGHT_BOLD);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void taskbarname_default_font_changed()
|
||||||
|
{
|
||||||
|
if (!taskbar_enabled)
|
||||||
|
return;
|
||||||
|
if (!taskbarname_enabled)
|
||||||
|
return;
|
||||||
|
if (panel_config.taskbarname_has_font)
|
||||||
|
return;
|
||||||
|
|
||||||
|
pango_font_description_free(panel_config.taskbarname_font_desc);
|
||||||
|
panel_config.taskbarname_font_desc = NULL;
|
||||||
|
taskbarname_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
for (int j = 0; j < panels[i].num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panels[i].taskbar[j];
|
||||||
|
taskbar->bar_name.area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&taskbar->bar_name.area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
schedule_panel_redraw();
|
||||||
|
}
|
||||||
|
|
||||||
void cleanup_taskbarname()
|
void cleanup_taskbarname()
|
||||||
{
|
{
|
||||||
int i, j, k;
|
for (int i = 0; i < num_panels; i++) {
|
||||||
Panel *panel;
|
Panel *panel = &panels[i];
|
||||||
Taskbar *tskbar;
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panel->taskbar[j];
|
||||||
for (i=0 ; i < nb_panel ; i++) {
|
g_free(taskbar->bar_name.name);
|
||||||
panel = &panel1[i];
|
taskbar->bar_name.name = NULL;
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
free_area(&taskbar->bar_name.area);
|
||||||
tskbar = &panel->taskbar[j];
|
remove_area((Area *)&taskbar->bar_name);
|
||||||
if (tskbar->bar_name.name) g_free(tskbar->bar_name.name);
|
}
|
||||||
free_area (&tskbar->bar_name.area);
|
}
|
||||||
for (k=0; k<TASKBAR_STATE_COUNT; ++k) {
|
|
||||||
if (tskbar->bar_name.state_pix[k]) XFreePixmap(server.dsp, tskbar->bar_name.state_pix[k]);
|
|
||||||
}
|
|
||||||
tskbar->area.list = g_slist_remove(tskbar->area.list, &tskbar->bar_name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int taskbarname_compute_desired_size(void *obj)
|
||||||
void draw_taskbarname (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Taskbarname *taskbar_name = obj;
|
TaskbarName *taskbar_name = (TaskbarName *)obj;
|
||||||
Taskbar *taskbar = taskbar_name->area.parent;
|
Panel *panel = (Panel *)taskbar_name->area.panel;
|
||||||
PangoLayout *layout;
|
int name_height, name_width, name_height_ink;
|
||||||
Color *config_text = (taskbar->desktop == server.desktop) ? &taskbarname_active_font : &taskbarname_font;
|
get_text_size2(panel_config.taskbarname_font_desc,
|
||||||
|
&name_height_ink,
|
||||||
|
&name_height,
|
||||||
|
&name_width,
|
||||||
|
panel->area.height,
|
||||||
|
panel->area.width,
|
||||||
|
taskbar_name->name,
|
||||||
|
strlen(taskbar_name->name),
|
||||||
|
PANGO_WRAP_WORD_CHAR,
|
||||||
|
PANGO_ELLIPSIZE_NONE,
|
||||||
|
FALSE);
|
||||||
|
|
||||||
int state = (taskbar->desktop == server.desktop) ? TASKBAR_ACTIVE : TASKBAR_NORMAL;
|
if (panel_horizontal) {
|
||||||
taskbar_name->state_pix[state] = taskbar_name->area.pix;
|
return name_width + 2 * taskbar_name->area.paddingxlr + left_right_border_width(&taskbar_name->area);
|
||||||
|
} else {
|
||||||
// draw content
|
return name_height + 2 * taskbar_name->area.paddingxlr + top_bottom_border_width(&taskbar_name->area);
|
||||||
layout = pango_cairo_create_layout (c);
|
}
|
||||||
pango_layout_set_font_description (layout, taskbarname_font_desc);
|
|
||||||
pango_layout_set_width (layout, taskbar_name->area.width * PANGO_SCALE);
|
|
||||||
pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
|
||||||
pango_layout_set_text (layout, taskbar_name->name, strlen(taskbar_name->name));
|
|
||||||
|
|
||||||
cairo_set_source_rgba (c, config_text->color[0], config_text->color[1], config_text->color[2], config_text->alpha);
|
|
||||||
|
|
||||||
pango_cairo_update_layout (c, layout);
|
|
||||||
cairo_move_to (c, 0, taskbar_name->posy);
|
|
||||||
pango_cairo_show_layout (c, layout);
|
|
||||||
|
|
||||||
g_object_unref (layout);
|
|
||||||
//printf("draw_taskbarname %s ******************************\n", taskbar_name->name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean resize_taskbarname(void *obj)
|
||||||
int resize_taskbarname(void *obj)
|
|
||||||
{
|
{
|
||||||
Taskbarname *taskbar_name = obj;
|
TaskbarName *taskbar_name = (TaskbarName *)obj;
|
||||||
Panel *panel = taskbar_name->area.panel;
|
Panel *panel = (Panel *)taskbar_name->area.panel;
|
||||||
int name_height, name_width, name_height_ink;
|
|
||||||
int ret = 0;
|
|
||||||
|
|
||||||
taskbar_name->area.redraw = 1;
|
schedule_redraw(&taskbar_name->area);
|
||||||
get_text_size2(taskbarname_font_desc, &name_height_ink, &name_height, &name_width, panel->area.height, panel->area.width, taskbar_name->name, strlen(taskbar_name->name));
|
|
||||||
|
|
||||||
if (panel_horizontal) {
|
int name_height, name_width, name_height_ink;
|
||||||
int new_size = name_width + (2* (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
get_text_size2(panel_config.taskbarname_font_desc,
|
||||||
if (new_size != taskbar_name->area.width) {
|
&name_height_ink,
|
||||||
taskbar_name->area.width = new_size;
|
&name_height,
|
||||||
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
&name_width,
|
||||||
ret = 1;
|
panel->area.height,
|
||||||
}
|
panel->area.width,
|
||||||
}
|
taskbar_name->name,
|
||||||
else {
|
strlen(taskbar_name->name),
|
||||||
int new_size = name_height + (2 * (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
PANGO_WRAP_WORD_CHAR,
|
||||||
if (new_size != taskbar_name->area.height) {
|
PANGO_ELLIPSIZE_NONE,
|
||||||
taskbar_name->area.height = new_size;
|
FALSE);
|
||||||
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
|
||||||
ret = 1;
|
gboolean result = FALSE;
|
||||||
}
|
int new_size = taskbarname_compute_desired_size(obj);
|
||||||
}
|
if (panel_horizontal) {
|
||||||
return ret;
|
if (new_size != taskbar_name->area.width) {
|
||||||
|
taskbar_name->area.width = new_size;
|
||||||
|
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
||||||
|
result = TRUE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (new_size != taskbar_name->area.height) {
|
||||||
|
taskbar_name->area.height = new_size;
|
||||||
|
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
||||||
|
result = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void draw_taskbarname(void *obj, cairo_t *c)
|
||||||
|
{
|
||||||
|
TaskbarName *taskbar_name = obj;
|
||||||
|
Taskbar *taskbar = taskbar_name->area.parent;
|
||||||
|
Color *config_text = (taskbar->desktop == server.desktop) ? &taskbarname_active_font : &taskbarname_font;
|
||||||
|
|
||||||
|
// draw content
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
pango_layout_set_font_description(layout, panel_config.taskbarname_font_desc);
|
||||||
|
pango_layout_set_width(layout, taskbar_name->area.width * PANGO_SCALE);
|
||||||
|
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
|
||||||
|
pango_layout_set_text(layout, taskbar_name->name, strlen(taskbar_name->name));
|
||||||
|
|
||||||
|
cairo_set_source_rgba(c, config_text->rgb[0], config_text->rgb[1], config_text->rgb[2], config_text->alpha);
|
||||||
|
|
||||||
|
pango_cairo_update_layout(c, layout);
|
||||||
|
draw_text(layout, c, 0, taskbar_name->posy, config_text, ((Panel *)taskbar_name->area.panel)->font_shadow);
|
||||||
|
|
||||||
|
g_object_unref(layout);
|
||||||
|
}
|
||||||
|
|||||||
@@ -8,8 +8,7 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "area.h"
|
#include "area.h"
|
||||||
|
|
||||||
extern int taskbarname_enabled;
|
extern gboolean taskbarname_enabled;
|
||||||
extern PangoFontDescription *taskbarname_font_desc;
|
|
||||||
extern Color taskbarname_font;
|
extern Color taskbarname_font;
|
||||||
extern Color taskbarname_active_font;
|
extern Color taskbarname_active_font;
|
||||||
|
|
||||||
@@ -20,8 +19,8 @@ void init_taskbarname_panel(void *p);
|
|||||||
|
|
||||||
void draw_taskbarname(void *obj, cairo_t *c);
|
void draw_taskbarname(void *obj, cairo_t *c);
|
||||||
|
|
||||||
int resize_taskbarname(void *obj);
|
gboolean resize_taskbarname(void *obj);
|
||||||
|
|
||||||
|
void taskbarname_default_font_changed();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
3058
src/tint.c
3058
src/tint.c
File diff suppressed because it is too large
Load Diff
@@ -2,12 +2,13 @@ project(tint2conf)
|
|||||||
cmake_minimum_required(VERSION 2.6)
|
cmake_minimum_required(VERSION 2.6)
|
||||||
|
|
||||||
include( FindPkgConfig )
|
include( FindPkgConfig )
|
||||||
pkg_check_modules( X11_T2C REQUIRED x11 xrender )
|
pkg_check_modules( X11_T2C REQUIRED x11 xcomposite xdamage xinerama xrender xrandr>=1.3 )
|
||||||
pkg_check_modules( GLIB2 REQUIRED glib-2.0 )
|
pkg_check_modules( GLIB2 REQUIRED glib-2.0 )
|
||||||
pkg_check_modules( GOBJECT2 REQUIRED gobject-2.0 )
|
pkg_check_modules( GOBJECT2 REQUIRED gobject-2.0 )
|
||||||
pkg_check_modules( IMLIB2 REQUIRED imlib2 )
|
pkg_check_modules( IMLIB2 REQUIRED imlib2 )
|
||||||
pkg_check_modules( GTHREAD2 REQUIRED gthread-2.0 )
|
pkg_check_modules( GTHREAD2 REQUIRED gthread-2.0 )
|
||||||
pkg_check_modules( GTK2 REQUIRED gtk+-x11-2.0 )
|
pkg_check_modules( GTK2 REQUIRED gtk+-x11-2.0 )
|
||||||
|
pkg_check_modules( RSVG librsvg-2.0>=2.36.0 )
|
||||||
|
|
||||||
include_directories( ../util
|
include_directories( ../util
|
||||||
${X11_T2C_INCLUDE_DIRS}
|
${X11_T2C_INCLUDE_DIRS}
|
||||||
@@ -15,37 +16,59 @@ include_directories( ../util
|
|||||||
${GOBJECT2_INCLUDE_DIRS}
|
${GOBJECT2_INCLUDE_DIRS}
|
||||||
${IMLIB2_INCLUDE_DIRS}
|
${IMLIB2_INCLUDE_DIRS}
|
||||||
${GTHREAD2_INCLUDE_DIRS}
|
${GTHREAD2_INCLUDE_DIRS}
|
||||||
${GTK2_INCLUDE_DIRS} )
|
${GTK2_INCLUDE_DIRS}
|
||||||
|
${RSVG_INCLUDE_DIRS} )
|
||||||
|
|
||||||
set(SOURCES ../util/common.c
|
set(SOURCES ../util/common.c
|
||||||
|
../util/strnatcmp.c
|
||||||
|
../util/cache.c
|
||||||
|
../config.c
|
||||||
|
../server.c
|
||||||
|
../launcher/apps-common.c
|
||||||
|
../launcher/icon-theme-common.c
|
||||||
|
md4.c
|
||||||
main.c
|
main.c
|
||||||
properties.c
|
properties.c
|
||||||
properties_rw.c
|
properties_rw.c
|
||||||
theme_view.c )
|
theme_view.c
|
||||||
|
background_gui.c
|
||||||
|
gradient_gui.c )
|
||||||
|
|
||||||
|
add_definitions( -DTINT2CONF )
|
||||||
|
|
||||||
|
option( ENABLE_RSVG "Rsvg support (launcher only)" ON )
|
||||||
|
|
||||||
|
if( ENABLE_RSVG )
|
||||||
|
if( RSVG_FOUND )
|
||||||
|
add_definitions( -DHAVE_RSVG )
|
||||||
|
endif( RSVG_FOUND )
|
||||||
|
endif( ENABLE_RSVG )
|
||||||
|
|
||||||
link_directories( ${X11_T2C_LIBRARY_DIRS}
|
link_directories( ${X11_T2C_LIBRARY_DIRS}
|
||||||
${GLIB2_LIBRARY_DIRS}
|
${GLIB2_LIBRARY_DIRS}
|
||||||
${GOBJECT2_LIBRARY_DIRS}
|
${GOBJECT2_LIBRARY_DIRS}
|
||||||
${IMLIB2_LIBRARY_DIRS}
|
${IMLIB2_LIBRARY_DIRS}
|
||||||
${GTHREAD2_LIBRARY_DIRS}
|
${GTHREAD2_LIBRARY_DIRS}
|
||||||
${GTK2_LIBRARY_DIRS} )
|
${GTK2_LIBRARY_DIRS}
|
||||||
|
${RSVG_LIBRARY_DIRS} )
|
||||||
add_executable( tint2conf ${SOURCES} )
|
add_executable( tint2conf ${SOURCES} )
|
||||||
target_link_libraries( tint2conf ${X11_T2C_LIBRARIES}
|
target_link_libraries( tint2conf ${X11_T2C_LIBRARIES}
|
||||||
${GLIB2_LIBRARIES}
|
${GLIB2_LIBRARIES}
|
||||||
${GOBJECT2_LIBRARIES}
|
${GOBJECT2_LIBRARIES}
|
||||||
${IMLIB2_LIBRARIES}
|
${IMLIB2_LIBRARIES}
|
||||||
${GTHREAD2_LIBRARIES}
|
${GTHREAD2_LIBRARIES}
|
||||||
${GTK2_LIBRARIES} )
|
${GTK2_LIBRARIES}
|
||||||
|
${RSVG_LIBRARIES} )
|
||||||
if ( NOT DATADIR )
|
|
||||||
set( DATADIR share )
|
|
||||||
endif( NOT DATADIR )
|
|
||||||
|
|
||||||
add_definitions( -DINSTALL_PREFIX=\"${CMAKE_INSTALL_PREFIX}\" )
|
add_definitions( -DINSTALL_PREFIX=\"${CMAKE_INSTALL_PREFIX}\" )
|
||||||
set_target_properties( tint2conf PROPERTIES COMPILE_FLAGS "-Wall -pthread" )
|
add_definitions( -DLOCALEDIR=\"${CMAKE_INSTALL_FULL_LOCALEDIR}\" )
|
||||||
|
add_definitions( -DGETTEXT_PACKAGE=\"tint2conf\" )
|
||||||
|
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)
|
||||||
|
|
||||||
install( TARGETS tint2conf DESTINATION bin )
|
install( TARGETS tint2conf DESTINATION bin )
|
||||||
install( PROGRAMS tintwizard.py DESTINATION bin )
|
install( FILES tint2conf.svg DESTINATION ${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps )
|
||||||
install( FILES taskbar.svg DESTINATION ${DATADIR}/icons/hicolor/scalable/apps )
|
install( FILES tint2conf.desktop DESTINATION ${CMAKE_INSTALL_DATADIR}/applications )
|
||||||
install( FILES tint2conf.desktop DESTINATION ${DATADIR}/applications )
|
install( FILES tint2conf.xml DESTINATION ${CMAKE_INSTALL_DATADIR}/mime/packages )
|
||||||
install( CODE "execute_process(COMMAND gtk-update-icon-cache -f -t ${DATADIR}/icons/hicolor WORKING_DIRECTORY ${CMAKE_INSTALL_PREFIX})" )
|
|
||||||
|
|||||||
1038
src/tint2conf/background_gui.c
Normal file
1038
src/tint2conf/background_gui.c
Normal file
File diff suppressed because it is too large
Load Diff
15
src/tint2conf/background_gui.h
Normal file
15
src/tint2conf/background_gui.h
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#ifndef BACKGROUND_GUI_H
|
||||||
|
#define BACKGROUND_GUI_H
|
||||||
|
|
||||||
|
#include "gui.h"
|
||||||
|
|
||||||
|
void create_background(GtkWidget *parent);
|
||||||
|
void background_duplicate(GtkWidget *widget, gpointer data);
|
||||||
|
void background_delete(GtkWidget *widget, gpointer data);
|
||||||
|
void background_update_image(int index);
|
||||||
|
void background_update(GtkWidget *widget, gpointer data);
|
||||||
|
void current_background_changed(GtkWidget *widget, gpointer data);
|
||||||
|
void background_combo_changed(GtkWidget *widget, gpointer data);
|
||||||
|
GtkWidget *create_background_combo(const char *label);
|
||||||
|
|
||||||
|
#endif
|
||||||
611
src/tint2conf/gradient_gui.c
Normal file
611
src/tint2conf/gradient_gui.c
Normal file
@@ -0,0 +1,611 @@
|
|||||||
|
#include "gradient_gui.h"
|
||||||
|
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
GtkListStore *gradient_ids, *gradient_stop_ids;
|
||||||
|
GList *gradients;
|
||||||
|
GtkWidget *current_gradient, *gradient_combo_type, *gradient_start_color, *gradient_end_color, *current_gradient_stop,
|
||||||
|
*gradient_stop_color, *gradient_stop_offset;
|
||||||
|
|
||||||
|
int gradient_index_safe(int index)
|
||||||
|
{
|
||||||
|
if (index <= 0)
|
||||||
|
index = 0;
|
||||||
|
if (index >= get_model_length(GTK_TREE_MODEL(gradient_ids)))
|
||||||
|
index = 0;
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
GtkWidget *create_gradient_combo()
|
||||||
|
{
|
||||||
|
GtkWidget *combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(gradient_ids));
|
||||||
|
GtkCellRenderer *renderer = gtk_cell_renderer_pixbuf_new();
|
||||||
|
gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), renderer, FALSE);
|
||||||
|
gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(combo), renderer, "pixbuf", grColPixbuf, NULL);
|
||||||
|
renderer = gtk_cell_renderer_text_new();
|
||||||
|
g_object_set(renderer, "wrap-mode", PANGO_WRAP_WORD, NULL);
|
||||||
|
g_object_set(renderer, "wrap-width", 300, NULL);
|
||||||
|
gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), renderer, FALSE);
|
||||||
|
gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(combo), renderer, "text", grColText, NULL);
|
||||||
|
return combo;
|
||||||
|
}
|
||||||
|
|
||||||
|
void create_gradient(GtkWidget *parent)
|
||||||
|
{
|
||||||
|
gradient_ids = gtk_list_store_new(grNumCols, GDK_TYPE_PIXBUF, GTK_TYPE_INT, GTK_TYPE_STRING);
|
||||||
|
gradients = NULL;
|
||||||
|
|
||||||
|
gradient_stop_ids = gtk_list_store_new(grStopNumCols, GDK_TYPE_PIXBUF);
|
||||||
|
|
||||||
|
GtkWidget *table, *label, *button;
|
||||||
|
int row, col;
|
||||||
|
// GtkTooltips *tooltips = gtk_tooltips_new();
|
||||||
|
|
||||||
|
table = gtk_table_new(1, 4, FALSE);
|
||||||
|
gtk_widget_show(table);
|
||||||
|
gtk_box_pack_start(GTK_BOX(parent), table, FALSE, FALSE, 0);
|
||||||
|
gtk_table_set_row_spacings(GTK_TABLE(table), ROW_SPACING);
|
||||||
|
gtk_table_set_col_spacings(GTK_TABLE(table), COL_SPACING);
|
||||||
|
|
||||||
|
row = 0, col = 0;
|
||||||
|
label = gtk_label_new(_("<b>Gradient</b>"));
|
||||||
|
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
|
||||||
|
gtk_label_set_use_markup(GTK_LABEL(label), TRUE);
|
||||||
|
gtk_widget_show(label);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), label, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
current_gradient = create_gradient_combo();
|
||||||
|
gtk_widget_show(current_gradient);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), current_gradient, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
button = gtk_button_new_from_stock("gtk-add");
|
||||||
|
gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(gradient_duplicate), NULL);
|
||||||
|
gtk_widget_show(button);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), button, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
button = gtk_button_new_from_stock("gtk-remove");
|
||||||
|
gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(gradient_delete), NULL);
|
||||||
|
gtk_widget_show(button);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), button, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
table = gtk_table_new(3, 4, FALSE);
|
||||||
|
gtk_widget_show(table);
|
||||||
|
gtk_box_pack_start(GTK_BOX(parent), table, FALSE, FALSE, 0);
|
||||||
|
gtk_table_set_row_spacings(GTK_TABLE(table), ROW_SPACING);
|
||||||
|
gtk_table_set_col_spacings(GTK_TABLE(table), COL_SPACING);
|
||||||
|
|
||||||
|
row = 0, col = 2;
|
||||||
|
label = gtk_label_new(_("Type"));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
gradient_combo_type = gtk_combo_box_new_text();
|
||||||
|
gtk_widget_show(gradient_combo_type);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), gradient_combo_type, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
gtk_combo_box_append_text(GTK_COMBO_BOX(gradient_combo_type), _("Vertical"));
|
||||||
|
gtk_combo_box_append_text(GTK_COMBO_BOX(gradient_combo_type), _("Horizontal"));
|
||||||
|
gtk_combo_box_append_text(GTK_COMBO_BOX(gradient_combo_type), _("Radial"));
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(gradient_combo_type), 0);
|
||||||
|
|
||||||
|
row++, col = 2;
|
||||||
|
label = gtk_label_new(_("Start color"));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
gradient_start_color = gtk_color_button_new();
|
||||||
|
gtk_color_button_set_use_alpha(GTK_COLOR_BUTTON(gradient_start_color), TRUE);
|
||||||
|
gtk_widget_show(gradient_start_color);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), gradient_start_color, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
row++, col = 2;
|
||||||
|
label = gtk_label_new(_("End color"));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
gradient_end_color = gtk_color_button_new();
|
||||||
|
gtk_color_button_set_use_alpha(GTK_COLOR_BUTTON(gradient_end_color), TRUE);
|
||||||
|
gtk_widget_show(gradient_end_color);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), gradient_end_color, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
change_paragraph(parent);
|
||||||
|
|
||||||
|
table = gtk_table_new(1, 4, FALSE);
|
||||||
|
gtk_widget_show(table);
|
||||||
|
gtk_box_pack_start(GTK_BOX(parent), table, FALSE, FALSE, 0);
|
||||||
|
gtk_table_set_row_spacings(GTK_TABLE(table), ROW_SPACING);
|
||||||
|
gtk_table_set_col_spacings(GTK_TABLE(table), COL_SPACING);
|
||||||
|
|
||||||
|
row = 0, col = 0;
|
||||||
|
label = gtk_label_new(_("<b>Color stop</b>"));
|
||||||
|
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
|
||||||
|
gtk_label_set_use_markup(GTK_LABEL(label), TRUE);
|
||||||
|
gtk_widget_show(label);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), label, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
current_gradient_stop = create_gradient_stop_combo(NULL);
|
||||||
|
gtk_widget_show(current_gradient_stop);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), current_gradient_stop, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
button = gtk_button_new_from_stock("gtk-add");
|
||||||
|
gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(gradient_stop_duplicate), NULL);
|
||||||
|
gtk_widget_show(button);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), button, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
button = gtk_button_new_from_stock("gtk-remove");
|
||||||
|
gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(gradient_stop_delete), NULL);
|
||||||
|
gtk_widget_show(button);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), button, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
table = gtk_table_new(3, 4, FALSE);
|
||||||
|
gtk_widget_show(table);
|
||||||
|
gtk_box_pack_start(GTK_BOX(parent), table, FALSE, FALSE, 0);
|
||||||
|
gtk_table_set_row_spacings(GTK_TABLE(table), ROW_SPACING);
|
||||||
|
gtk_table_set_col_spacings(GTK_TABLE(table), COL_SPACING);
|
||||||
|
|
||||||
|
row = 0, col = 2;
|
||||||
|
label = gtk_label_new(_("Color"));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
gradient_stop_color = gtk_color_button_new();
|
||||||
|
gtk_color_button_set_use_alpha(GTK_COLOR_BUTTON(gradient_stop_color), TRUE);
|
||||||
|
gtk_widget_show(gradient_stop_color);
|
||||||
|
gtk_table_attach(GTK_TABLE(table), gradient_stop_color, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
row++, col = 2;
|
||||||
|
label = gtk_label_new(_("Position"));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
gradient_stop_offset = gtk_spin_button_new_with_range(0, 100, 1);
|
||||||
|
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);
|
||||||
|
col++;
|
||||||
|
|
||||||
|
g_signal_connect(G_OBJECT(current_gradient), "changed", G_CALLBACK(current_gradient_changed), NULL);
|
||||||
|
g_signal_connect(G_OBJECT(gradient_combo_type), "changed", G_CALLBACK(gradient_update), NULL);
|
||||||
|
g_signal_connect(G_OBJECT(gradient_start_color), "color-set", G_CALLBACK(gradient_update), NULL);
|
||||||
|
g_signal_connect(G_OBJECT(gradient_end_color), "color-set", G_CALLBACK(gradient_update), NULL);
|
||||||
|
|
||||||
|
g_signal_connect(G_OBJECT(current_gradient_stop), "changed", G_CALLBACK(current_gradient_stop_changed), NULL);
|
||||||
|
g_signal_connect(G_OBJECT(gradient_stop_color), "color-set", G_CALLBACK(gradient_stop_update), NULL);
|
||||||
|
g_signal_connect(G_OBJECT(gradient_stop_offset), "value-changed", G_CALLBACK(gradient_stop_update), NULL);
|
||||||
|
|
||||||
|
change_paragraph(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_create_new(GradientConfigType t)
|
||||||
|
{
|
||||||
|
int index = get_model_length(GTK_TREE_MODEL(gradient_ids));
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)calloc(1, sizeof(GradientConfig));
|
||||||
|
g->type = t;
|
||||||
|
gradients = g_list_append(gradients, g);
|
||||||
|
GtkTreeIter iter;
|
||||||
|
gtk_list_store_append(gradient_ids, &iter);
|
||||||
|
gtk_list_store_set(gradient_ids,
|
||||||
|
&iter,
|
||||||
|
grColPixbuf,
|
||||||
|
NULL,
|
||||||
|
grColId,
|
||||||
|
&index,
|
||||||
|
grColText,
|
||||||
|
index == 0 ? _("None") : "",
|
||||||
|
-1);
|
||||||
|
|
||||||
|
gradient_update_image(index);
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient), index);
|
||||||
|
current_gradient_changed(0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
gpointer copy_GradientConfigColorStop(gconstpointer arg, gpointer data)
|
||||||
|
{
|
||||||
|
GradientConfigColorStop *old = (GradientConfigColorStop *)arg;
|
||||||
|
|
||||||
|
GradientConfigColorStop *copy = (GradientConfigColorStop *)calloc(1, sizeof(GradientConfigColorStop));
|
||||||
|
memcpy(copy, old, sizeof(GradientConfigColorStop));
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_duplicate(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
int old_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (old_index < 0) {
|
||||||
|
gradient_create_new(GRADIENT_CONFIG_VERTICAL);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
GradientConfig *g_old = (GradientConfig *)g_list_nth(gradients, (guint)old_index)->data;
|
||||||
|
|
||||||
|
int index = get_model_length(GTK_TREE_MODEL(gradient_ids));
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)calloc(1, sizeof(GradientConfig));
|
||||||
|
g->type = g_old->type;
|
||||||
|
g->start_color = g_old->start_color;
|
||||||
|
g->end_color = g_old->end_color;
|
||||||
|
g->extra_color_stops = g_list_copy_deep(g->extra_color_stops, copy_GradientConfigColorStop, NULL);
|
||||||
|
gradients = g_list_append(gradients, g);
|
||||||
|
GtkTreeIter iter;
|
||||||
|
gtk_list_store_append(gradient_ids, &iter);
|
||||||
|
gtk_list_store_set(gradient_ids, &iter, grColPixbuf, NULL, grColId, &index, grColText, "", -1);
|
||||||
|
|
||||||
|
gradient_update_image(index);
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient), index);
|
||||||
|
current_gradient_changed(0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_delete(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (get_model_length(GTK_TREE_MODEL(gradient_ids)) <= 1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GtkTreePath *path;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
|
||||||
|
path = gtk_tree_path_new_from_indices(index, -1);
|
||||||
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_ids), &iter, path);
|
||||||
|
gtk_tree_path_free(path);
|
||||||
|
|
||||||
|
gradients = g_list_remove(gradients, g_list_nth(gradients, (guint)index)->data);
|
||||||
|
gtk_list_store_remove(gradient_ids, &iter);
|
||||||
|
|
||||||
|
background_update_for_gradient(index);
|
||||||
|
|
||||||
|
if (index >= get_model_length(GTK_TREE_MODEL(gradient_ids)))
|
||||||
|
index--;
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient), index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_draw(cairo_t *c, GradientConfig *g, int w, int h, gboolean preserve)
|
||||||
|
{
|
||||||
|
cairo_pattern_t *gpat;
|
||||||
|
if (g->type == GRADIENT_CONFIG_VERTICAL)
|
||||||
|
gpat = cairo_pattern_create_linear(0, 0, 0, h);
|
||||||
|
else if (g->type == GRADIENT_CONFIG_HORIZONTAL)
|
||||||
|
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);
|
||||||
|
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) {
|
||||||
|
GradientConfigColorStop *stop = (GradientConfigColorStop *)l->data;
|
||||||
|
cairo_pattern_add_color_stop_rgba(gpat,
|
||||||
|
stop->offset,
|
||||||
|
stop->color.rgb[0],
|
||||||
|
stop->color.rgb[1],
|
||||||
|
stop->color.rgb[2],
|
||||||
|
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_set_source(c, gpat);
|
||||||
|
cairo_rectangle(c, 0, 0, w, h);
|
||||||
|
if (preserve)
|
||||||
|
cairo_fill_preserve(c);
|
||||||
|
else
|
||||||
|
cairo_fill(c);
|
||||||
|
cairo_pattern_destroy(gpat);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_update_image(int index)
|
||||||
|
{
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)index)->data;
|
||||||
|
|
||||||
|
int w = 70;
|
||||||
|
int h = 30;
|
||||||
|
GdkPixmap *pixmap = gdk_pixmap_new(NULL, w, h, 24);
|
||||||
|
|
||||||
|
cairo_t *cr = gdk_cairo_create(pixmap);
|
||||||
|
cairo_set_source_rgb(cr, 0.5, 0.5, 0.5);
|
||||||
|
cairo_rectangle(cr, 0, 0, w, h);
|
||||||
|
cairo_fill(cr);
|
||||||
|
|
||||||
|
gradient_draw(cr, g, w, h, FALSE);
|
||||||
|
|
||||||
|
GdkPixbuf *pixbuf = gdk_pixbuf_get_from_drawable(NULL, pixmap, gdk_colormap_get_system(), 0, 0, 0, 0, w, h);
|
||||||
|
if (pixmap)
|
||||||
|
g_object_unref(pixmap);
|
||||||
|
|
||||||
|
GtkTreePath *path;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
|
||||||
|
path = gtk_tree_path_new_from_indices(index, -1);
|
||||||
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_ids), &iter, path);
|
||||||
|
gtk_tree_path_free(path);
|
||||||
|
|
||||||
|
gtk_list_store_set(gradient_ids, &iter, grColPixbuf, pixbuf, -1);
|
||||||
|
if (pixbuf)
|
||||||
|
g_object_unref(pixbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_force_update()
|
||||||
|
{
|
||||||
|
gradient_update(NULL, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean gradient_updates_disabled = FALSE;
|
||||||
|
void gradient_update(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
if (gradient_updates_disabled)
|
||||||
|
return;
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (index <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GtkTreePath *path;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
|
||||||
|
path = gtk_tree_path_new_from_indices(index, -1);
|
||||||
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_ids), &iter, path);
|
||||||
|
gtk_tree_path_free(path);
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)index)->data;
|
||||||
|
|
||||||
|
g->type = (GradientConfigType)MAX(0, gtk_combo_box_get_active(GTK_COMBO_BOX(gradient_combo_type)));
|
||||||
|
|
||||||
|
GdkColor color;
|
||||||
|
int opacity;
|
||||||
|
|
||||||
|
gtk_color_button_get_color(GTK_COLOR_BUTTON(gradient_start_color), &color);
|
||||||
|
opacity = MIN(100, 0.5 + gtk_color_button_get_alpha(GTK_COLOR_BUTTON(gradient_start_color)) * 100.0 / 0xffff);
|
||||||
|
gdkColor2CairoColor(color, &g->start_color.color.rgb[0], &g->start_color.color.rgb[1], &g->start_color.color.rgb[2]);
|
||||||
|
g->start_color.color.alpha = opacity / 100.0;
|
||||||
|
|
||||||
|
gtk_color_button_get_color(GTK_COLOR_BUTTON(gradient_end_color), &color);
|
||||||
|
opacity = MIN(100, 0.5 + gtk_color_button_get_alpha(GTK_COLOR_BUTTON(gradient_end_color)) * 100.0 / 0xffff);
|
||||||
|
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;
|
||||||
|
|
||||||
|
gtk_list_store_set(gradient_ids, &iter, grColPixbuf, NULL, grColId, &index, grColText, "", -1);
|
||||||
|
gradient_update_image(index);
|
||||||
|
background_update_for_gradient(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void current_gradient_changed(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
gtk_widget_set_sensitive(gradient_combo_type, index > 0);
|
||||||
|
gtk_widget_set_sensitive(gradient_start_color, index > 0);
|
||||||
|
gtk_widget_set_sensitive(gradient_end_color, index > 0);
|
||||||
|
gtk_widget_set_sensitive(current_gradient_stop, index > 0);
|
||||||
|
gtk_widget_set_sensitive(gradient_stop_color, index > 0);
|
||||||
|
gtk_widget_set_sensitive(gradient_stop_offset, index > 0);
|
||||||
|
|
||||||
|
gradient_updates_disabled = TRUE;
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)index)->data;
|
||||||
|
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(gradient_combo_type), g->type);
|
||||||
|
|
||||||
|
GdkColor color;
|
||||||
|
int opacity;
|
||||||
|
|
||||||
|
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 * 65535;
|
||||||
|
gtk_color_button_set_color(GTK_COLOR_BUTTON(gradient_start_color), &color);
|
||||||
|
gtk_color_button_set_alpha(GTK_COLOR_BUTTON(gradient_start_color), opacity);
|
||||||
|
|
||||||
|
cairoColor2GdkColor(g->end_color.color.rgb[0], g->end_color.color.rgb[1], g->end_color.color.rgb[2], &color);
|
||||||
|
opacity = g->end_color.color.alpha * 65535;
|
||||||
|
gtk_color_button_set_color(GTK_COLOR_BUTTON(gradient_end_color), &color);
|
||||||
|
gtk_color_button_set_alpha(GTK_COLOR_BUTTON(gradient_end_color), opacity);
|
||||||
|
|
||||||
|
int old_stop_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient_stop));
|
||||||
|
gtk_list_store_clear(gradient_stop_ids);
|
||||||
|
int stop_index = 0;
|
||||||
|
for (GList *l = g->extra_color_stops; l; l = l->next, stop_index++) {
|
||||||
|
GtkTreeIter iter;
|
||||||
|
gtk_list_store_append(gradient_stop_ids, &iter);
|
||||||
|
gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, NULL, -1);
|
||||||
|
gradient_stop_update_image(stop_index);
|
||||||
|
}
|
||||||
|
if (old_stop_index >= 0 && old_stop_index < stop_index)
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient_stop), old_stop_index);
|
||||||
|
else
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient_stop), stop_index - 1);
|
||||||
|
|
||||||
|
gradient_updates_disabled = FALSE;
|
||||||
|
gradient_update_image(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
//////// Gradient stops
|
||||||
|
|
||||||
|
GtkWidget *create_gradient_stop_combo()
|
||||||
|
{
|
||||||
|
GtkWidget *combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(gradient_stop_ids));
|
||||||
|
GtkCellRenderer *renderer = gtk_cell_renderer_pixbuf_new();
|
||||||
|
gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), renderer, FALSE);
|
||||||
|
gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(combo), renderer, "pixbuf", grStopColPixbuf, NULL);
|
||||||
|
return combo;
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_stop_create_new()
|
||||||
|
{
|
||||||
|
int g_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (g_index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)g_index)->data;
|
||||||
|
GradientConfigColorStop *stop = (GradientConfigColorStop *)calloc(1, sizeof(GradientConfigColorStop));
|
||||||
|
if (g->extra_color_stops)
|
||||||
|
memcpy(stop, g_list_last(g->extra_color_stops)->data, sizeof(GradientConfigColorStop));
|
||||||
|
else
|
||||||
|
memcpy(stop, &g->start_color, sizeof(GradientConfigColorStop));
|
||||||
|
g->extra_color_stops = g_list_append(g->extra_color_stops, stop);
|
||||||
|
GtkTreeIter iter;
|
||||||
|
gtk_list_store_append(gradient_stop_ids, &iter);
|
||||||
|
gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, NULL, -1);
|
||||||
|
|
||||||
|
int stop_index = g_list_length(g->extra_color_stops) - 1;
|
||||||
|
gradient_stop_update_image(stop_index);
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient_stop), stop_index);
|
||||||
|
current_gradient_stop_changed(0, 0);
|
||||||
|
gradient_update_image(g_index);
|
||||||
|
background_update_for_gradient(g_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_stop_duplicate(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
gradient_stop_create_new();
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_stop_delete(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
int g_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (g_index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)g_index)->data;
|
||||||
|
if (!g->extra_color_stops)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient_stop));
|
||||||
|
if (index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GtkTreePath *path;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
|
||||||
|
path = gtk_tree_path_new_from_indices(index, -1);
|
||||||
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_stop_ids), &iter, path);
|
||||||
|
gtk_tree_path_free(path);
|
||||||
|
|
||||||
|
g->extra_color_stops = g_list_remove(g->extra_color_stops, g_list_nth(g->extra_color_stops, (guint)index)->data);
|
||||||
|
gtk_list_store_remove(gradient_stop_ids, &iter);
|
||||||
|
|
||||||
|
if (index >= get_model_length(GTK_TREE_MODEL(gradient_stop_ids)))
|
||||||
|
index--;
|
||||||
|
gtk_combo_box_set_active(GTK_COMBO_BOX(current_gradient_stop), index);
|
||||||
|
|
||||||
|
gradient_update_image(g_index);
|
||||||
|
background_update_for_gradient(g_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_stop_update_image(int index)
|
||||||
|
{
|
||||||
|
int g_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (g_index < 0)
|
||||||
|
return;
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)g_index)->data;
|
||||||
|
GradientConfigColorStop *stop = (GradientConfigColorStop *)g_list_nth(g->extra_color_stops, (guint)index)->data;
|
||||||
|
|
||||||
|
int w = 70;
|
||||||
|
int h = 30;
|
||||||
|
GdkPixmap *pixmap = gdk_pixmap_new(NULL, w, h, 24);
|
||||||
|
|
||||||
|
cairo_t *cr = gdk_cairo_create(pixmap);
|
||||||
|
cairo_set_source_rgba(cr, stop->color.rgb[0], stop->color.rgb[1], stop->color.rgb[2], stop->color.alpha);
|
||||||
|
cairo_rectangle(cr, 0, 0, w, h);
|
||||||
|
cairo_fill(cr);
|
||||||
|
|
||||||
|
GdkPixbuf *pixbuf = gdk_pixbuf_get_from_drawable(NULL, pixmap, gdk_colormap_get_system(), 0, 0, 0, 0, w, h);
|
||||||
|
if (pixmap)
|
||||||
|
g_object_unref(pixmap);
|
||||||
|
|
||||||
|
GtkTreePath *path;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
|
||||||
|
path = gtk_tree_path_new_from_indices(index, -1);
|
||||||
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(gradient_stop_ids), &iter, path);
|
||||||
|
gtk_tree_path_free(path);
|
||||||
|
|
||||||
|
gtk_list_store_set(gradient_stop_ids, &iter, grStopColPixbuf, pixbuf, -1);
|
||||||
|
if (pixbuf)
|
||||||
|
g_object_unref(pixbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
void current_gradient_stop_changed(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
int g_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (g_index < 0)
|
||||||
|
return;
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)g_index)->data;
|
||||||
|
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient_stop));
|
||||||
|
if (index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GradientConfigColorStop *stop = (GradientConfigColorStop *)g_list_nth(g->extra_color_stops, (guint)index)->data;
|
||||||
|
|
||||||
|
gradient_updates_disabled = TRUE;
|
||||||
|
|
||||||
|
GdkColor color;
|
||||||
|
int opacity;
|
||||||
|
|
||||||
|
cairoColor2GdkColor(stop->color.rgb[0], stop->color.rgb[1], stop->color.rgb[2], &color);
|
||||||
|
opacity = stop->color.alpha * 65535;
|
||||||
|
gtk_color_button_set_color(GTK_COLOR_BUTTON(gradient_stop_color), &color);
|
||||||
|
gtk_color_button_set_alpha(GTK_COLOR_BUTTON(gradient_stop_color), opacity);
|
||||||
|
|
||||||
|
gtk_spin_button_set_value(GTK_SPIN_BUTTON(gradient_stop_offset), stop->offset * 100);
|
||||||
|
|
||||||
|
gradient_updates_disabled = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void gradient_stop_update(GtkWidget *widget, gpointer data)
|
||||||
|
{
|
||||||
|
if (gradient_updates_disabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int g_index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient));
|
||||||
|
if (g_index < 0)
|
||||||
|
return;
|
||||||
|
GradientConfig *g = (GradientConfig *)g_list_nth(gradients, (guint)g_index)->data;
|
||||||
|
|
||||||
|
int index = gtk_combo_box_get_active(GTK_COMBO_BOX(current_gradient_stop));
|
||||||
|
if (index < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
GradientConfigColorStop *stop = (GradientConfigColorStop *)g_list_nth(g->extra_color_stops, (guint)index)->data;
|
||||||
|
|
||||||
|
GdkColor color;
|
||||||
|
int opacity;
|
||||||
|
|
||||||
|
gtk_color_button_get_color(GTK_COLOR_BUTTON(gradient_stop_color), &color);
|
||||||
|
opacity = MIN(100, 0.5 + gtk_color_button_get_alpha(GTK_COLOR_BUTTON(gradient_stop_color)) * 100.0 / 0xffff);
|
||||||
|
gdkColor2CairoColor(color, &stop->color.rgb[0], &stop->color.rgb[1], &stop->color.rgb[2]);
|
||||||
|
stop->color.alpha = opacity / 100.0;
|
||||||
|
|
||||||
|
stop->offset = gtk_spin_button_get_value(GTK_SPIN_BUTTON(gradient_stop_offset)) / 100.;
|
||||||
|
|
||||||
|
gradient_stop_update_image(index);
|
||||||
|
gradient_update_image(g_index);
|
||||||
|
background_update_for_gradient(g_index);
|
||||||
|
}
|
||||||
47
src/tint2conf/gradient_gui.h
Normal file
47
src/tint2conf/gradient_gui.h
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
#ifndef GRADIENT_GUI_H
|
||||||
|
#define GRADIENT_GUI_H
|
||||||
|
|
||||||
|
#include "gui.h"
|
||||||
|
|
||||||
|
int gradient_index_safe(int index);
|
||||||
|
void create_gradient(GtkWidget *parent);
|
||||||
|
GtkWidget *create_gradient_combo();
|
||||||
|
void gradient_duplicate(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_delete(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_update_image(int index);
|
||||||
|
void gradient_update(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_force_update();
|
||||||
|
void current_gradient_changed(GtkWidget *widget, gpointer data);
|
||||||
|
void background_update_for_gradient(int gradient_id);
|
||||||
|
|
||||||
|
GtkWidget *create_gradient_stop_combo();
|
||||||
|
void gradient_stop_duplicate(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_stop_delete(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_stop_update(GtkWidget *widget, gpointer data);
|
||||||
|
void gradient_stop_update_image(int index);
|
||||||
|
void current_gradient_stop_changed(GtkWidget *widget, gpointer data);
|
||||||
|
|
||||||
|
typedef enum GradientConfigType {
|
||||||
|
GRADIENT_CONFIG_VERTICAL = 0,
|
||||||
|
GRADIENT_CONFIG_HORIZONTAL,
|
||||||
|
GRADIENT_CONFIG_RADIAL
|
||||||
|
} GradientConfigType;
|
||||||
|
|
||||||
|
typedef struct GradientConfigColorStop {
|
||||||
|
Color color;
|
||||||
|
// offset in 0-1
|
||||||
|
double offset;
|
||||||
|
} GradientConfigColorStop;
|
||||||
|
|
||||||
|
typedef struct GradientConfig {
|
||||||
|
GradientConfigType type;
|
||||||
|
GradientConfigColorStop start_color;
|
||||||
|
GradientConfigColorStop end_color;
|
||||||
|
// Each element is a GradientConfigColorStop
|
||||||
|
GList *extra_color_stops;
|
||||||
|
} GradientConfig;
|
||||||
|
|
||||||
|
void gradient_create_new(GradientConfigType t);
|
||||||
|
void gradient_draw(cairo_t *c, GradientConfig *g, int w, int h, gboolean preserve);
|
||||||
|
|
||||||
|
#endif
|
||||||
20
src/tint2conf/gui.h
Normal file
20
src/tint2conf/gui.h
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
#include <limits.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "main.h"
|
||||||
|
#include "properties.h"
|
||||||
|
#include "properties_rw.h"
|
||||||
|
#include "../launcher/apps-common.h"
|
||||||
|
#include "../launcher/icon-theme-common.h"
|
||||||
|
#include "../util/common.h"
|
||||||
|
#include "strnatcmp.h"
|
||||||
|
|
||||||
|
#define ROW_SPACING 10
|
||||||
|
#define COL_SPACING 8
|
||||||
|
#define DEFAULT_HOR_SPACING 5
|
||||||
|
|
||||||
|
gint compare_strings(gconstpointer a, gconstpointer b);
|
||||||
|
void change_paragraph(GtkWidget *widget);
|
||||||
|
int get_model_length(GtkTreeModel *model);
|
||||||
|
void gdkColor2CairoColor(GdkColor color, double *red, double *green, double *blue);
|
||||||
|
void cairoColor2GdkColor(double red, double green, double blue, GdkColor *color);
|
||||||
1373
src/tint2conf/main.c
1373
src/tint2conf/main.c
File diff suppressed because it is too large
Load Diff
@@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
@@ -12,5 +11,11 @@
|
|||||||
#include <glib/gi18n-lib.h>
|
#include <glib/gi18n-lib.h>
|
||||||
#else
|
#else
|
||||||
#define _(String) String
|
#define _(String) String
|
||||||
|
#define GETTEXT_PACKAGE "tint2conf"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define SNAPSHOT_TICK 190
|
||||||
|
gboolean update_snapshot(gpointer ignored);
|
||||||
|
void menuApply();
|
||||||
|
void refresh_current_theme();
|
||||||
|
extern GtkWidget *g_window;
|
||||||
|
|||||||
271
src/tint2conf/md4.c
Normal file
271
src/tint2conf/md4.c
Normal file
@@ -0,0 +1,271 @@
|
|||||||
|
/*
|
||||||
|
* Cryptographic API.
|
||||||
|
*
|
||||||
|
* MD4 Message Digest Algorithm (RFC1320).
|
||||||
|
*
|
||||||
|
* Implementation derived from Andrew Tridgell and Steve French's
|
||||||
|
* CIFS MD4 implementation, and the cryptoapi implementation
|
||||||
|
* originally based on the public domain implementation written
|
||||||
|
* by Colin Plumb in 1993.
|
||||||
|
*
|
||||||
|
* Copyright (c) Andrew Tridgell 1997-1998.
|
||||||
|
* Modified by Steve French (sfrench@us.ibm.com) 2002
|
||||||
|
* Copyright (c) Cryptoapi developers.
|
||||||
|
* Copyright (c) 2002 David S. Miller (davem@redhat.com)
|
||||||
|
* Copyright (c) 2002 James Morris <jmorris@intercode.com.au>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
typedef u_int8_t u8;
|
||||||
|
typedef u_int32_t u32;
|
||||||
|
typedef u_int64_t u64;
|
||||||
|
|
||||||
|
#ifndef ARRAY_SIZE
|
||||||
|
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define shash_desc md4_ctx
|
||||||
|
#define shash_desc_ctx(x) (x)
|
||||||
|
|
||||||
|
#define MD4_DIGEST_SIZE 16
|
||||||
|
#define MD4_HMAC_BLOCK_SIZE 64
|
||||||
|
#define MD4_BLOCK_WORDS 16
|
||||||
|
#define MD4_HASH_WORDS 4
|
||||||
|
|
||||||
|
struct md4_ctx {
|
||||||
|
u32 hash[MD4_HASH_WORDS];
|
||||||
|
u32 block[MD4_BLOCK_WORDS];
|
||||||
|
u64 byte_count;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline u32 lshift(u32 x, unsigned int s)
|
||||||
|
{
|
||||||
|
x &= 0xFFFFFFFF;
|
||||||
|
return ((x << s) & 0xFFFFFFFF) | (x >> (32 - s));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline u32 F(u32 x, u32 y, u32 z)
|
||||||
|
{
|
||||||
|
return (x & y) | ((~x) & z);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline u32 G(u32 x, u32 y, u32 z)
|
||||||
|
{
|
||||||
|
return (x & y) | (x & z) | (y & z);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline u32 H(u32 x, u32 y, u32 z)
|
||||||
|
{
|
||||||
|
return x ^ y ^ z;
|
||||||
|
}
|
||||||
|
|
||||||
|
#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 ROUND3(a, b, c, d, k, s) (a = lshift(a + H(b, c, d) + k + (u32)0x6ED9EBA1, s))
|
||||||
|
|
||||||
|
/* XXX: this stuff can be optimized */
|
||||||
|
static inline void le32_to_cpu_array(u32 *buf, unsigned int words)
|
||||||
|
{
|
||||||
|
while (words--) {
|
||||||
|
*buf = ntohl(*buf);
|
||||||
|
buf++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void cpu_to_le32_array(u32 *buf, unsigned int words)
|
||||||
|
{
|
||||||
|
while (words--) {
|
||||||
|
*buf = htonl(*buf);
|
||||||
|
buf++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void md4_transform(u32 *hash, u32 const *in)
|
||||||
|
{
|
||||||
|
u32 a, b, c, d;
|
||||||
|
|
||||||
|
a = hash[0];
|
||||||
|
b = hash[1];
|
||||||
|
c = hash[2];
|
||||||
|
d = hash[3];
|
||||||
|
|
||||||
|
ROUND1(a, b, c, d, in[0], 3);
|
||||||
|
ROUND1(d, a, b, c, in[1], 7);
|
||||||
|
ROUND1(c, d, a, b, in[2], 11);
|
||||||
|
ROUND1(b, c, d, a, in[3], 19);
|
||||||
|
ROUND1(a, b, c, d, in[4], 3);
|
||||||
|
ROUND1(d, a, b, c, in[5], 7);
|
||||||
|
ROUND1(c, d, a, b, in[6], 11);
|
||||||
|
ROUND1(b, c, d, a, in[7], 19);
|
||||||
|
ROUND1(a, b, c, d, in[8], 3);
|
||||||
|
ROUND1(d, a, b, c, in[9], 7);
|
||||||
|
ROUND1(c, d, a, b, in[10], 11);
|
||||||
|
ROUND1(b, c, d, a, in[11], 19);
|
||||||
|
ROUND1(a, b, c, d, in[12], 3);
|
||||||
|
ROUND1(d, a, b, c, in[13], 7);
|
||||||
|
ROUND1(c, d, a, b, in[14], 11);
|
||||||
|
ROUND1(b, c, d, a, in[15], 19);
|
||||||
|
|
||||||
|
ROUND2(a, b, c, d, in[0], 3);
|
||||||
|
ROUND2(d, a, b, c, in[4], 5);
|
||||||
|
ROUND2(c, d, a, b, in[8], 9);
|
||||||
|
ROUND2(b, c, d, a, in[12], 13);
|
||||||
|
ROUND2(a, b, c, d, in[1], 3);
|
||||||
|
ROUND2(d, a, b, c, in[5], 5);
|
||||||
|
ROUND2(c, d, a, b, in[9], 9);
|
||||||
|
ROUND2(b, c, d, a, in[13], 13);
|
||||||
|
ROUND2(a, b, c, d, in[2], 3);
|
||||||
|
ROUND2(d, a, b, c, in[6], 5);
|
||||||
|
ROUND2(c, d, a, b, in[10], 9);
|
||||||
|
ROUND2(b, c, d, a, in[14], 13);
|
||||||
|
ROUND2(a, b, c, d, in[3], 3);
|
||||||
|
ROUND2(d, a, b, c, in[7], 5);
|
||||||
|
ROUND2(c, d, a, b, in[11], 9);
|
||||||
|
ROUND2(b, c, d, a, in[15], 13);
|
||||||
|
|
||||||
|
ROUND3(a, b, c, d, in[0], 3);
|
||||||
|
ROUND3(d, a, b, c, in[8], 9);
|
||||||
|
ROUND3(c, d, a, b, in[4], 11);
|
||||||
|
ROUND3(b, c, d, a, in[12], 15);
|
||||||
|
ROUND3(a, b, c, d, in[2], 3);
|
||||||
|
ROUND3(d, a, b, c, in[10], 9);
|
||||||
|
ROUND3(c, d, a, b, in[6], 11);
|
||||||
|
ROUND3(b, c, d, a, in[14], 15);
|
||||||
|
ROUND3(a, b, c, d, in[1], 3);
|
||||||
|
ROUND3(d, a, b, c, in[9], 9);
|
||||||
|
ROUND3(c, d, a, b, in[5], 11);
|
||||||
|
ROUND3(b, c, d, a, in[13], 15);
|
||||||
|
ROUND3(a, b, c, d, in[3], 3);
|
||||||
|
ROUND3(d, a, b, c, in[11], 9);
|
||||||
|
ROUND3(c, d, a, b, in[7], 11);
|
||||||
|
ROUND3(b, c, d, a, in[15], 15);
|
||||||
|
|
||||||
|
hash[0] += a;
|
||||||
|
hash[1] += b;
|
||||||
|
hash[2] += c;
|
||||||
|
hash[3] += d;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void md4_transform_helper(struct md4_ctx *ctx)
|
||||||
|
{
|
||||||
|
le32_to_cpu_array(ctx->block, ARRAY_SIZE(ctx->block));
|
||||||
|
md4_transform(ctx->hash, ctx->block);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int md4_init(struct shash_desc *desc)
|
||||||
|
{
|
||||||
|
struct md4_ctx *mctx = shash_desc_ctx(desc);
|
||||||
|
|
||||||
|
mctx->hash[0] = 0x67452301;
|
||||||
|
mctx->hash[1] = 0xefcdab89;
|
||||||
|
mctx->hash[2] = 0x98badcfe;
|
||||||
|
mctx->hash[3] = 0x10325476;
|
||||||
|
mctx->byte_count = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int md4_update(struct shash_desc *desc, const u8 *data, unsigned int len)
|
||||||
|
{
|
||||||
|
struct md4_ctx *mctx = shash_desc_ctx(desc);
|
||||||
|
const u32 avail = sizeof(mctx->block) - (mctx->byte_count & 0x3f);
|
||||||
|
|
||||||
|
mctx->byte_count += len;
|
||||||
|
|
||||||
|
if (avail > len) {
|
||||||
|
memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), data, len);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy((char *)mctx->block + (sizeof(mctx->block) - avail), data, avail);
|
||||||
|
|
||||||
|
md4_transform_helper(mctx);
|
||||||
|
data += avail;
|
||||||
|
len -= avail;
|
||||||
|
|
||||||
|
while (len >= sizeof(mctx->block)) {
|
||||||
|
memcpy(mctx->block, data, sizeof(mctx->block));
|
||||||
|
md4_transform_helper(mctx);
|
||||||
|
data += sizeof(mctx->block);
|
||||||
|
len -= sizeof(mctx->block);
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(mctx->block, data, len);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int md4_final(struct shash_desc *desc, u8 *out)
|
||||||
|
{
|
||||||
|
struct md4_ctx *mctx = shash_desc_ctx(desc);
|
||||||
|
const unsigned int offset = mctx->byte_count & 0x3f;
|
||||||
|
char *p = (char *)mctx->block + offset;
|
||||||
|
int padding = 56 - (offset + 1);
|
||||||
|
|
||||||
|
*p++ = 0x80;
|
||||||
|
if (padding < 0) {
|
||||||
|
memset(p, 0x00, padding + sizeof(u64));
|
||||||
|
md4_transform_helper(mctx);
|
||||||
|
p = (char *)mctx->block;
|
||||||
|
padding = 56;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(p, 0, padding);
|
||||||
|
mctx->block[14] = mctx->byte_count << 3;
|
||||||
|
mctx->block[15] = mctx->byte_count >> 29;
|
||||||
|
le32_to_cpu_array(mctx->block, (sizeof(mctx->block) - sizeof(u64)) / sizeof(u32));
|
||||||
|
md4_transform(mctx->hash, mctx->block);
|
||||||
|
cpu_to_le32_array(mctx->hash, ARRAY_SIZE(mctx->hash));
|
||||||
|
memcpy(out, mctx->hash, sizeof(mctx->hash));
|
||||||
|
memset(mctx, 0, sizeof(*mctx));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char to_hex(u8 v)
|
||||||
|
{
|
||||||
|
v = v & 0xf;
|
||||||
|
if (v < 0xa)
|
||||||
|
return '0' + v;
|
||||||
|
return 'a' + v - 0xa;
|
||||||
|
}
|
||||||
|
|
||||||
|
void md4hexf(const char *path, char *hash)
|
||||||
|
{
|
||||||
|
struct md4_ctx mctx;
|
||||||
|
md4_init(&mctx);
|
||||||
|
|
||||||
|
int fd = open(path, O_RDONLY);
|
||||||
|
if (fd >= 0) {
|
||||||
|
u8 buffer[MD4_HMAC_BLOCK_SIZE];
|
||||||
|
while (1) {
|
||||||
|
ssize_t count = read(fd, buffer, sizeof(buffer));
|
||||||
|
if (count <= 0)
|
||||||
|
break;
|
||||||
|
md4_update(&mctx, buffer, (unsigned)count);
|
||||||
|
}
|
||||||
|
close(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
u8 out[MD4_DIGEST_SIZE];
|
||||||
|
md4_final(&mctx, out);
|
||||||
|
|
||||||
|
for (int i = 0; i < MD4_DIGEST_SIZE; i++) {
|
||||||
|
hash[2 * i + 0] = to_hex(out[i] >> 4);
|
||||||
|
hash[2 * i + 1] = to_hex(out[i] & 0xf);
|
||||||
|
}
|
||||||
|
hash[2 * MD4_DIGEST_SIZE] = 0;
|
||||||
|
}
|
||||||
7
src/tint2conf/md4.h
Normal file
7
src/tint2conf/md4.h
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#ifndef MD4_H
|
||||||
|
#define MD4_H
|
||||||
|
|
||||||
|
#define MD4_HEX_SIZE 33
|
||||||
|
void md4hexf(const char *path, char *hash);
|
||||||
|
|
||||||
|
#endif
|
||||||
20
src/tint2conf/po/CMakeLists.txt
Normal file
20
src/tint2conf/po/CMakeLists.txt
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
include(FindGettext)
|
||||||
|
if (GETTEXT_FOUND)
|
||||||
|
set(GETTEXT_PACKAGE tint2conf)
|
||||||
|
file(GLOB POTFILES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "*.po")
|
||||||
|
string(REPLACE ".po" " " LANGUAGES ${POTFILES})
|
||||||
|
message(STATUS "gettext found languages: ${LANGUAGES}")
|
||||||
|
string(REPLACE " " ";" LANGUAGES ${LANGUAGES})
|
||||||
|
if ("${CMAKE_MAJOR_VERSION}.${CMAKE_MINOR_VERSION}.${CMAKE_PATCH_VERSION}" VERSION_LESS "2.8.8")
|
||||||
|
GETTEXT_CREATE_TRANSLATIONS("${CMAKE_CURRENT_SOURCE_DIR}/tint2conf.pot" ALL ${POTFILES})
|
||||||
|
else()
|
||||||
|
foreach(LANG ${LANGUAGES})
|
||||||
|
GETTEXT_PROCESS_PO_FILES(${LANG} ALL PO_FILES ${LANG}.po)
|
||||||
|
install(FILES "${CMAKE_CURRENT_BINARY_DIR}/${LANG}.gmo"
|
||||||
|
DESTINATION "${CMAKE_INSTALL_LOCALEDIR}/${LANG}/LC_MESSAGES"
|
||||||
|
RENAME "${GETTEXT_PACKAGE}.mo")
|
||||||
|
endforeach ()
|
||||||
|
endif()
|
||||||
|
else ()
|
||||||
|
message(STATUS "gettext not found")
|
||||||
|
endif ()
|
||||||
2416
src/tint2conf/po/bs.po
Normal file
2416
src/tint2conf/po/bs.po
Normal file
File diff suppressed because it is too large
Load Diff
2434
src/tint2conf/po/fr.po
Normal file
2434
src/tint2conf/po/fr.po
Normal file
File diff suppressed because it is too large
Load Diff
2417
src/tint2conf/po/hr.po
Normal file
2417
src/tint2conf/po/hr.po
Normal file
File diff suppressed because it is too large
Load Diff
2424
src/tint2conf/po/pl.po
Normal file
2424
src/tint2conf/po/pl.po
Normal file
File diff suppressed because it is too large
Load Diff
11
src/tint2conf/po/readme.txt
Normal file
11
src/tint2conf/po/readme.txt
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
Updating pot file:
|
||||||
|
|
||||||
|
find .. -name '*.c' | sort -r | xargs xgettext --keyword=_ --language=C --output=updated.pot -
|
||||||
|
|
||||||
|
Followed by manual editing of updated.pot to make sure the header is OK. Then:
|
||||||
|
|
||||||
|
cat updated.pot > tint2conf.pot && rm -f updated.pot
|
||||||
|
|
||||||
|
Then update the po files:
|
||||||
|
|
||||||
|
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
|
||||||
2401
src/tint2conf/po/ru.po
Normal file
2401
src/tint2conf/po/ru.po
Normal file
File diff suppressed because it is too large
Load Diff
2415
src/tint2conf/po/sr.po
Normal file
2415
src/tint2conf/po/sr.po
Normal file
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user