Compare commits
474 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
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: ForIndentation
|
||||||
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
build
|
||||||
|
*.user
|
||||||
26
AUTHORS
26
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,22 @@ 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
|
||||||
|
|
||||||
|
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>
|
||||||
|
|||||||
179
CMakeLists.txt
179
CMakeLists.txt
@@ -1,8 +1,21 @@
|
|||||||
project( tint2 )
|
project( tint2 )
|
||||||
cmake_minimum_required( VERSION 2.6 )
|
cmake_minimum_required( VERSION 2.6 )
|
||||||
|
|
||||||
|
option( ENABLE_BATTERY "Enable battery status plugin" ON )
|
||||||
|
option( ENABLE_TINT2CONF "Enable tint2conf build, a GTK+2 theme configurator for tint2" ON )
|
||||||
|
option( ENABLE_EXAMPLES "Install additional tin2rc examples" 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( 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 +23,59 @@ 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 )
|
||||||
|
|
||||||
|
add_definitions( -D_GNU_SOURCE )
|
||||||
|
|
||||||
include_directories( ${PROJECT_BINARY_DIR}
|
include_directories( ${PROJECT_BINARY_DIR}
|
||||||
src
|
src
|
||||||
src/battery
|
src/battery
|
||||||
@@ -33,6 +85,8 @@ include_directories( ${PROJECT_BINARY_DIR}
|
|||||||
src/launcher
|
src/launcher
|
||||||
src/tooltip
|
src/tooltip
|
||||||
src/util
|
src/util
|
||||||
|
src/execplugin
|
||||||
|
src/freespace
|
||||||
${X11_INCLUDE_DIRS}
|
${X11_INCLUDE_DIRS}
|
||||||
${PANGOCAIRO_INCLUDE_DIRS}
|
${PANGOCAIRO_INCLUDE_DIRS}
|
||||||
${PANGO_INCLUDE_DIRS}
|
${PANGO_INCLUDE_DIRS}
|
||||||
@@ -40,7 +94,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 +104,82 @@ 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/freespace/freespace.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/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 +187,20 @@ 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 ")
|
||||||
|
else()
|
||||||
|
SET(ASAN_C_FLAGS "")
|
||||||
|
SET(ASAN_L_FLAGS "")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set( MANDIR share/man CACHE PATH "Directory for man pages" )
|
||||||
|
set( DATADIR share CACHE PATH "Directory for shared data" )
|
||||||
|
set( SYSCONFDIR /etc CACHE PATH "Directory for configuration files" )
|
||||||
|
set( DOCDIR share/doc/tint2 CACHE PATH "Directory for documentation files" )
|
||||||
|
|
||||||
|
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 +209,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 +218,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 tint2.svg DESTINATION ${DATADIR}/icons/hicolor/scalable/apps )
|
||||||
|
install( FILES tint2.desktop DESTINATION ${DATADIR}/applications )
|
||||||
|
install( CODE "execute_process(COMMAND gtk-update-icon-cache -f -t ${DATADIR}/icons/hicolor WORKING_DIRECTORY ${CMAKE_INSTALL_PREFIX})" )
|
||||||
install( FILES sample/tint2rc DESTINATION ${SYSCONFDIR}/xdg/tint2 )
|
install( FILES sample/tint2rc DESTINATION ${SYSCONFDIR}/xdg/tint2 )
|
||||||
install( FILES default_icon.png DESTINATION ${DATADIR}/tint2 )
|
install( FILES default_icon.png DESTINATION ${DATADIR}/tint2 )
|
||||||
install( FILES AUTHORS ChangeLog README DESTINATION ${DOCDIR} )
|
install( FILES AUTHORS ChangeLog README.md DESTINATION ${DOCDIR} )
|
||||||
install( FILES doc/tint2.1 DESTINATION ${MANDIR}/man1 )
|
install( FILES doc/tint2.1 DESTINATION ${MANDIR}/man1 )
|
||||||
if( ENABLE_EXAMPLES )
|
if( ENABLE_EXAMPLES )
|
||||||
file( GLOB SAMPLEFILES sample/*.tint2rc )
|
file( GLOB SAMPLEFILES sample/*.tint2rc )
|
||||||
install( FILES ${SAMPLEFILES} DESTINATION ${DOCDIR}/examples )
|
install( FILES ${SAMPLEFILES} DESTINATION ${DATADIR}/tint2 )
|
||||||
endif( ENABLE_EXAMPLES )
|
endif( ENABLE_EXAMPLES )
|
||||||
|
|||||||
168
ChangeLog
168
ChangeLog
@@ -1,3 +1,169 @@
|
|||||||
|
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 +749,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'.
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -7,5 +7,4 @@ cmake -DCMAKE_INSTALL_PREFIX=/usr ../
|
|||||||
make
|
make
|
||||||
sudo make install
|
sudo make install
|
||||||
|
|
||||||
To see additional options you can do after the cmake step a 'cmake -L ../'
|
To see additional options you can set at the 'cmake' step, run '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.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
81
README.md
Normal file
81
README.md
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
# New stable release: 0.12.4
|
||||||
|
Changes: https://gitlab.com/o9000/tint2/blob/0.12.4/ChangeLog
|
||||||
|
|
||||||
|
Documentation: https://gitlab.com/o9000/tint2/wikis/Configure
|
||||||
|
|
||||||
|
Try it out with (see also [dependencies](https://gitlab.com/o9000/tint2/wikis/Install#dependencies)):
|
||||||
|
```
|
||||||
|
git clone https://gitlab.com/o9000/tint2.git
|
||||||
|
cd tint2
|
||||||
|
git checkout 0.12.4
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake ..
|
||||||
|
make -j4
|
||||||
|
./tint2 &
|
||||||
|
./src/tint2conf/tint2conf &
|
||||||
|
```
|
||||||
|
|
||||||
|
To install from source, also run (as root):
|
||||||
|
```
|
||||||
|
make install
|
||||||
|
```
|
||||||
|
|
||||||
|
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 http://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/wikis/Configure)
|
||||||
|
* [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)
|
||||||
|
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
# 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
|
||||||
|
|
||||||
|
# Releases
|
||||||
|
* Latest stable release: tint2 0.12.4 (January 2016)
|
||||||
|
|
||||||
|
# Screenshots
|
||||||
|
|
||||||
|
## Default config of the latest release:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Various configurations:
|
||||||
|
|
||||||
|

|
||||||
@@ -1,4 +1,3 @@
|
|||||||
DEPENDENCIES:
|
DEPENDENCIES:
|
||||||
cairo (whit X support), pango, glib2, libX11, libXinerama, libXrandr, libXrender, libXcomposite, libXdamage imlib2 (with X support)
|
cairo (with X support), pango, glib2, libX11, libXinerama, libXrandr, libXrender, libXcomposite, libXdamage, imlib2 (with X support)
|
||||||
you might need -dev packages on debian
|
you might need -dev packages on Debian
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
.\" First parameter, NAME, should be all caps
|
.\" First parameter, NAME, should be all caps
|
||||||
.\" Second parameter, SECTION, should be 1-8, maybe w/ subsection
|
.\" Second parameter, SECTION, should be 1-8, maybe w/ subsection
|
||||||
.\" other parameters are allowed: see man(7), man(1)
|
.\" other parameters are allowed: see man(7), man(1)
|
||||||
.TH TINT2 1 "2009-01-17"
|
.TH TINT2 1 "2015-07-05"
|
||||||
.\" Please adjust this date whenever revising the manpage.
|
.\" Please adjust this date whenever revising the manpage.
|
||||||
.\"
|
.\"
|
||||||
.\" Some roff macros, for reference:
|
.\" Some roff macros, for reference:
|
||||||
@@ -37,7 +37,7 @@ compliance with freedesktop specifications.
|
|||||||
.PP
|
.PP
|
||||||
On the first startup tint2 creates a config file in $HOME/.config/tint2/tint2rc.
|
On the first startup tint2 creates a config file in $HOME/.config/tint2/tint2rc.
|
||||||
.PP
|
.PP
|
||||||
See wiki page on http://code.google.com/p/tint2/wiki/Welcome for more information.
|
See the wiki page at https://gitlab.com/o9000/tint2/wikis/home for more information.
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
.TP
|
.TP
|
||||||
.B \-c config-file
|
.B \-c config-file
|
||||||
|
|||||||
@@ -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
|
|
||||||
37
get_version.sh
Executable file
37
get_version.sh
Executable file
@@ -0,0 +1,37 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
MAJOR=0.12
|
||||||
|
DIRTY=""
|
||||||
|
|
||||||
|
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.%h | tr -d ':' | tr -d '-' | tr '.' '-' | sed 's/T[0-9\+]*//g')")$DIRTY
|
||||||
|
VERSION=$(echo "$VERSION" | sed 's/^v//')
|
||||||
|
|
||||||
|
echo '#define VERSION_STRING "'$VERSION'"' > version.h
|
||||||
|
echo $VERSION
|
||||||
@@ -1,23 +1,48 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
# usage: ./make_release.sh RELEASE_VERSION_NUMBER
|
# Usage: ./make_release.sh
|
||||||
|
# Creates a tar.gz archive of the current tree.
|
||||||
|
#
|
||||||
|
# To bump the version number for the current commit (make sure you are in HEAD!), run manually:
|
||||||
|
#
|
||||||
|
# git tag -a v0.12 -m 'Version 0.12'
|
||||||
|
#
|
||||||
|
# To generate a release for an older tagged commit, first list the tags with:
|
||||||
|
#
|
||||||
|
# git tag
|
||||||
|
#
|
||||||
|
# then checkout the tagged commit with:
|
||||||
|
#
|
||||||
|
# git checkout tags/v0.1
|
||||||
|
#
|
||||||
|
# Finally, to revert to HEAD:
|
||||||
|
#
|
||||||
|
# git checkout master
|
||||||
|
#
|
||||||
|
# See more at https://gitlab.com/o9000/tint2/wikis/Development
|
||||||
|
|
||||||
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}
|
DIR=tint2-$VERSION
|
||||||
echo "Making release ${DIR}"
|
ARCHIVE=$DIR.tar.gz
|
||||||
rm -Rf ${DIR}
|
echo "Making release $DIR"
|
||||||
svn export . ${DIR} > /dev/null
|
rm -rf $DIR $ARCHIVE
|
||||||
|
|
||||||
# delete unneeded files
|
git checkout-index --prefix=$DIR/ -a
|
||||||
rm -f ${DIR}/configure ${DIR}/make_release.sh
|
|
||||||
|
|
||||||
# replace get_svnrev.sh by a simple echo command
|
# Delete unneeded files
|
||||||
echo "echo \"#define VERSION_STRING \\\"${1}\\\"\" > version.h" > ${DIR}/get_svnrev.sh
|
rm -f $DIR/make_release.sh
|
||||||
|
|
||||||
# create tarball and remove the exported directory
|
echo "echo \"#define VERSION_STRING \\\"$VERSION\\\"\" > version.h" > $DIR/get_version.sh
|
||||||
tar -cjf ${DIR}.tar.bz2 ${DIR}
|
|
||||||
rm -Rf ${DIR}
|
# Create tarball and remove the exported directory
|
||||||
|
tar -czf $ARCHIVE $DIR
|
||||||
|
rm -rf $DIR
|
||||||
|
|
||||||
|
sha1sum -b $ARCHIVE
|
||||||
|
sha256sum -b $ARCHIVE
|
||||||
|
|||||||
213
sample/horizontal-dark-opaque.tint2rc
Normal file
213
sample/horizontal-dark-opaque.tint2rc
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
#---- Generated by tint2conf bd50 ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #333333 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #333333 100
|
||||||
|
background_color_pressed = #111111 100
|
||||||
|
border_color_pressed = #333333 100
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #222222 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #555555 100
|
||||||
|
background_color_pressed = #333333 100
|
||||||
|
border_color_pressed = #555555 100
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #222222 100
|
||||||
|
border_color = #777777 100
|
||||||
|
background_color_hover = #222222 100
|
||||||
|
border_color_hover = #777777 100
|
||||||
|
background_color_pressed = #333333 100
|
||||||
|
border_color_pressed = #777777 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #222222 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #555555 100
|
||||||
|
background_color_pressed = #333333 100
|
||||||
|
border_color_pressed = #555555 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #222222 100
|
||||||
|
border_color = #777777 100
|
||||||
|
background_color_hover = #222222 100
|
||||||
|
border_color_hover = #777777 100
|
||||||
|
background_color_pressed = #222222 100
|
||||||
|
border_color_pressed = #777777 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = LTSC
|
||||||
|
panel_size = 100% 32
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 4 2 4
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom center horizontal
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = single_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #dddddd 100
|
||||||
|
taskbar_name_active_font_color = #dddddd 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 1
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 140 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #eeeeee 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/chromium-browser.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 8
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 7
|
||||||
|
clock_font_color = #eeeeee 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #eeeeee 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
211
sample/horizontal-dark-transparent.tint2rc
Normal file
211
sample/horizontal-dark-transparent.tint2rc
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
#---- Generated by tint2conf ab72 ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #000000 80
|
||||||
|
border_color = #555555 80
|
||||||
|
background_color_hover = #000000 80
|
||||||
|
border_color_hover = #555555 80
|
||||||
|
background_color_pressed = #000000 80
|
||||||
|
border_color_pressed = #555555 80
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #777777 0
|
||||||
|
border_color = #777777 0
|
||||||
|
background_color_hover = #777777 4
|
||||||
|
border_color_hover = #cccccc 30
|
||||||
|
background_color_pressed = #333333 4
|
||||||
|
border_color_pressed = #777777 30
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #555555 10
|
||||||
|
border_color = #ffffff 60
|
||||||
|
background_color_hover = #cccccc 10
|
||||||
|
border_color_hover = #ffffff 60
|
||||||
|
background_color_pressed = #555555 10
|
||||||
|
border_color_pressed = #ffffff 60
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #777777 0
|
||||||
|
border_color = #777777 30
|
||||||
|
background_color_hover = #777777 4
|
||||||
|
border_color_hover = #cccccc 30
|
||||||
|
background_color_pressed = #777777 0
|
||||||
|
border_color_pressed = #777777 30
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
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
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = LTSC
|
||||||
|
panel_size = 100% 32
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 4 2 4
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom center horizontal
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = single_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #dddddd 100
|
||||||
|
taskbar_name_active_font_color = #dddddd 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 1
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 140 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #eeeeee 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 8
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 7
|
||||||
|
clock_font_color = #eeeeee 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #eeeeee 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
219
sample/horizontal-light-opaque.tint2rc
Normal file
219
sample/horizontal-light-opaque.tint2rc
Normal file
@@ -0,0 +1,219 @@
|
|||||||
|
#---- Generated by tint2conf 079a ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 0
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #bbbbbb 100
|
||||||
|
background_color_hover = #eeeeee 100
|
||||||
|
border_color_hover = #bbbbbb 100
|
||||||
|
background_color_pressed = #eeeeee 100
|
||||||
|
border_color_pressed = #bbbbbb 100
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #eeeeee 100
|
||||||
|
background_color_hover = #eeeeee 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #cccccc 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #eeeeee 100
|
||||||
|
border_color_hover = #aaaaaa 100
|
||||||
|
background_color_pressed = #cccccc 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #eeeeee 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #eeeeee 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #dddddd 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 8: Systray
|
||||||
|
rounded = 3
|
||||||
|
border_width = 0
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #dddddd 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #dddddd 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = LTSC
|
||||||
|
panel_size = 100% 32
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 4 2 4
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom center horizontal
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = single_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #222222 100
|
||||||
|
taskbar_name_active_font_color = #222222 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 1
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 140 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #222222 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 4 0 2
|
||||||
|
systray_background_id = 8
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 8
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 7
|
||||||
|
clock_font_color = #222222 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #222222 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
213
sample/horizontal-light-transparent.tint2rc
Normal file
213
sample/horizontal-light-transparent.tint2rc
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
#---- Generated by tint2conf 5b63 ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 0
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 0
|
||||||
|
border_color = #bbbbbb 10
|
||||||
|
background_color_hover = #eeeeee 0
|
||||||
|
border_color_hover = #bbbbbb 10
|
||||||
|
background_color_pressed = #eeeeee 0
|
||||||
|
border_color_pressed = #bbbbbb 10
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 4
|
||||||
|
border_color = #cccccc 44
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #eaeaea 44
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #eaeaea 44
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 4
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 4
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 3
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #dddddd 3
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 3
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = LTSC
|
||||||
|
panel_size = 100% 32
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 4 2 4
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom center horizontal
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = single_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #ffffff 100
|
||||||
|
taskbar_name_active_font_color = #ffffff 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 1
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 140 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #ffffff 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 4 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/chromium-browser.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 8
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 7
|
||||||
|
clock_font_color = #ffffff 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #ffffff 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
@@ -99,6 +99,7 @@ tooltip_font = sans 10
|
|||||||
tooltip_font_color = #000000 80
|
tooltip_font_color = #000000 80
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -115,4 +116,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -98,6 +98,7 @@ tooltip_font = Sans 10
|
|||||||
tooltip_font_color = #000000 80
|
tooltip_font_color = #000000 80
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = close
|
mouse_middle = close
|
||||||
mouse_right = none
|
mouse_right = none
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -114,4 +115,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -104,6 +104,7 @@ tooltip_font = Sans 9
|
|||||||
tooltip_font_color = #5E5E5E 100
|
tooltip_font_color = #5E5E5E 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -120,4 +121,4 @@ battery_font_color = #000000 100
|
|||||||
battery_padding = 1 1
|
battery_padding = 1 1
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -104,6 +104,7 @@ tooltip_font = Sans 12
|
|||||||
tooltip_font_color = #FFFFFF 100
|
tooltip_font_color = #FFFFFF 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -120,4 +121,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 1
|
battery_background_id = 1
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -118,6 +118,7 @@ tooltip_font = Sans 8
|
|||||||
tooltip_font_color = #FFFFFF 100
|
tooltip_font_color = #FFFFFF 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -134,4 +135,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 2 0
|
battery_padding = 2 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -106,6 +106,7 @@ tooltip_font = sans 10
|
|||||||
tooltip_font_color = #000000 80
|
tooltip_font_color = #000000 80
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -122,4 +123,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -95,6 +95,7 @@ tooltip_font = sans 10
|
|||||||
tooltip_font_color = #000000 80
|
tooltip_font_color = #000000 80
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = none
|
mouse_right = none
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -111,4 +112,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -95,6 +95,7 @@ tooltip_font = Sans 10
|
|||||||
tooltip_font_color = #FFFFFF 80
|
tooltip_font_color = #FFFFFF 80
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -111,4 +112,4 @@ battery_font_color = #FFFFFF 100
|
|||||||
battery_padding = 0 0
|
battery_padding = 0 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -106,6 +106,7 @@ tooltip_font = sans 8
|
|||||||
tooltip_font_color = #000000 89
|
tooltip_font_color = #000000 89
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -122,4 +123,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 4 2
|
battery_padding = 4 2
|
||||||
battery_background_id = 1
|
battery_background_id = 1
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -102,6 +102,7 @@ tooltip_font = Sans 7
|
|||||||
tooltip_font_color = #FFFFFF 100
|
tooltip_font_color = #FFFFFF 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -118,4 +119,4 @@ battery_font_color = #FFFFFF 100
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -112,6 +112,7 @@ tooltip_font = Aller 8
|
|||||||
tooltip_font_color = #D3CAAA 33
|
tooltip_font_color = #D3CAAA 33
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -128,4 +129,4 @@ battery_font_color = #D3CAAA 48
|
|||||||
battery_padding = 4 2
|
battery_padding = 4 2
|
||||||
battery_background_id = 4
|
battery_background_id = 4
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -112,6 +112,7 @@ tooltip_font = Aller 8
|
|||||||
tooltip_font_color = #D3CAAA 33
|
tooltip_font_color = #D3CAAA 33
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -128,4 +129,4 @@ battery_font_color = #D3CAAA 48
|
|||||||
battery_padding = 4 2
|
battery_padding = 4 2
|
||||||
battery_background_id = 4
|
battery_background_id = 4
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -90,6 +90,7 @@ tooltip_font = Sans 9
|
|||||||
tooltip_font_color = #FFFFFF 100
|
tooltip_font_color = #FFFFFF 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
|
|||||||
@@ -83,6 +83,7 @@ tooltip_font = Sans 12
|
|||||||
tooltip_font_color = #FFFFFF 100
|
tooltip_font_color = #FFFFFF 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
|
|||||||
@@ -96,6 +96,7 @@ tooltip_font = AvantGardeLTMedium 8
|
|||||||
tooltip_font_color = #434141 100
|
tooltip_font_color = #434141 100
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -112,4 +113,4 @@ battery_font_color = #151515 60
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
@@ -89,6 +89,7 @@ tooltip_font = Aller 8
|
|||||||
tooltip_font_color = #D3CAAA 33
|
tooltip_font_color = #D3CAAA 33
|
||||||
|
|
||||||
# Mouse
|
# Mouse
|
||||||
|
mouse_left = toggle_iconify
|
||||||
mouse_middle = none
|
mouse_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
@@ -105,4 +106,4 @@ battery_font_color = #FFFFFF 75
|
|||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
|
||||||
# End of config
|
# End of config
|
||||||
|
|||||||
247
sample/tint2rc
247
sample/tint2rc
@@ -1,118 +1,187 @@
|
|||||||
# Tint2 config file
|
#---- Generated by tint2conf 2641 ----
|
||||||
# Generated by tintwizard (http://code.google.com/p/tintwizard/)
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
# For information on manually configuring tint2 see http://code.google.com/p/tint2/wiki/Configure
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
# Background definitions
|
# Backgrounds
|
||||||
# ID 1
|
# Background 1: Panel
|
||||||
rounded = 7
|
rounded = 0
|
||||||
border_width = 2
|
border_width = 0
|
||||||
background_color = #000000 60
|
background_color = #000000 60
|
||||||
border_color = #FFFFFF 16
|
border_color = #000000 30
|
||||||
|
background_color_hover = #000000 60
|
||||||
|
border_color_hover = #000000 30
|
||||||
|
background_color_pressed = #000000 60
|
||||||
|
border_color_pressed = #000000 30
|
||||||
|
|
||||||
# ID 2
|
# Background 2: Default task, Iconified task
|
||||||
rounded = 5
|
rounded = 4
|
||||||
border_width = 0
|
border_width = 1
|
||||||
background_color = #FFFFFF 40
|
background_color = #777777 20
|
||||||
border_color = #FFFFFF 48
|
border_color = #777777 30
|
||||||
|
background_color_hover = #aaaaaa 22
|
||||||
|
border_color_hover = #eaeaea 44
|
||||||
|
background_color_pressed = #555555 4
|
||||||
|
border_color_pressed = #eaeaea 44
|
||||||
|
|
||||||
# ID 3
|
# Background 3: Active task
|
||||||
rounded = 5
|
rounded = 4
|
||||||
border_width = 0
|
border_width = 1
|
||||||
background_color = #FFFFFF 16
|
background_color = #777777 20
|
||||||
border_color = #FFFFFF 68
|
border_color = #ffffff 40
|
||||||
|
background_color_hover = #aaaaaa 22
|
||||||
|
border_color_hover = #eaeaea 44
|
||||||
|
background_color_pressed = #555555 4
|
||||||
|
border_color_pressed = #eaeaea 44
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 4
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #cc7700 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #555555 4
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 1
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #000000 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #000000 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #000000 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
# Panel
|
# Panel
|
||||||
panel_monitor = all
|
panel_items = LTSC
|
||||||
panel_position = bottom center horizontal
|
panel_size = 100% 30
|
||||||
panel_size = 94% 30
|
|
||||||
panel_margin = 0 0
|
panel_margin = 0 0
|
||||||
panel_padding = 7 0 7
|
panel_padding = 2 0 2
|
||||||
panel_dock = 0
|
|
||||||
wm_menu = 0
|
|
||||||
panel_layer = top
|
|
||||||
panel_background_id = 1
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
# Panel Autohide
|
panel_dock = 0
|
||||||
|
panel_position = bottom center horizontal
|
||||||
|
panel_layer = top
|
||||||
|
panel_monitor = all
|
||||||
|
primary_monitor_first = 0
|
||||||
autohide = 0
|
autohide = 0
|
||||||
autohide_show_timeout = 0.3
|
autohide_show_timeout = 0
|
||||||
autohide_hide_timeout = 2
|
autohide_hide_timeout = 0.5
|
||||||
autohide_height = 2
|
autohide_height = 2
|
||||||
strut_policy = follow_size
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 1
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
# Taskbar
|
# Taskbar
|
||||||
taskbar_mode = single_desktop
|
taskbar_mode = single_desktop
|
||||||
taskbar_padding = 2 3 2
|
taskbar_padding = 0 0 2
|
||||||
taskbar_background_id = 0
|
taskbar_background_id = 0
|
||||||
taskbar_active_background_id = 0
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_always_show_all_desktop_tasks = 0
|
||||||
|
taskbar_name_padding = 4 2
|
||||||
|
taskbar_name_background_id = 0
|
||||||
|
taskbar_name_active_background_id = 0
|
||||||
|
taskbar_name_font_color = #e3e3e3 100
|
||||||
|
taskbar_name_active_font_color = #ffffff 100
|
||||||
|
taskbar_distribute_size = 0
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
# Tasks
|
#-------------------------------------
|
||||||
urgent_nb_of_blink = 8
|
# Task
|
||||||
task_icon = 1
|
|
||||||
task_text = 1
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
task_centered = 1
|
task_centered = 1
|
||||||
task_maximum_size = 140 35
|
urgent_nb_of_blink = 100000
|
||||||
task_padding = 6 2
|
task_maximum_size = 150 35
|
||||||
task_background_id = 3
|
task_padding = 2 2 4
|
||||||
task_active_background_id = 2
|
task_tooltip = 1
|
||||||
task_urgent_background_id = 2
|
task_font_color = #ffffff 100
|
||||||
task_iconified_background_id = 3
|
task_background_id = 2
|
||||||
task_tooltip = 0
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
# Task Icons
|
task_iconified_background_id = 2
|
||||||
task_icon_asb = 70 0 0
|
mouse_left = toggle_iconify
|
||||||
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_middle = none
|
||||||
mouse_right = close
|
mouse_right = close
|
||||||
mouse_scroll_up = toggle
|
mouse_scroll_up = toggle
|
||||||
mouse_scroll_down = iconify
|
mouse_scroll_down = iconify
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 4 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 24
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 2 4 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 24
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/chromium-browser.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
clock_font_color = #ffffff 100
|
||||||
|
clock_padding = 2 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command =
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
# Battery
|
# Battery
|
||||||
battery = 0
|
battery_tooltip = 1
|
||||||
battery_low_status = 10
|
battery_low_status = 10
|
||||||
battery_low_cmd = notify-send "battery low"
|
battery_low_cmd = notify-send "battery low"
|
||||||
battery_hide = 98
|
battery_font_color = #ffffff 100
|
||||||
bat1_font = sans 8
|
|
||||||
bat2_font = sans 6
|
|
||||||
battery_font_color = #FFFFFF 74
|
|
||||||
battery_padding = 1 0
|
battery_padding = 1 0
|
||||||
battery_background_id = 0
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
|
||||||
# End of config
|
|
||||||
211
sample/vertical-dark-opaque.tint2rc
Normal file
211
sample/vertical-dark-opaque.tint2rc
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
#---- Generated by tint2conf 14e9 ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #333333 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #333333 100
|
||||||
|
background_color_pressed = #111111 100
|
||||||
|
border_color_pressed = #333333 100
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #222222 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #555555 100
|
||||||
|
background_color_pressed = #333333 100
|
||||||
|
border_color_pressed = #555555 100
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #222222 100
|
||||||
|
border_color = #777777 100
|
||||||
|
background_color_hover = #333333 100
|
||||||
|
border_color_hover = #777777 100
|
||||||
|
background_color_pressed = #555555 100
|
||||||
|
border_color_pressed = #777777 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #111111 100
|
||||||
|
border_color = #222222 100
|
||||||
|
background_color_hover = #111111 100
|
||||||
|
border_color_hover = #333333 100
|
||||||
|
background_color_pressed = #555555 100
|
||||||
|
border_color_pressed = #333333 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #222222 100
|
||||||
|
border_color = #777777 100
|
||||||
|
background_color_hover = #222222 100
|
||||||
|
border_color_hover = #777777 100
|
||||||
|
background_color_pressed = #222222 100
|
||||||
|
border_color_pressed = #777777 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = CSTL
|
||||||
|
panel_size = 100% 152
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 2 2 2
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom left vertical
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = multi_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #dddddd 100
|
||||||
|
taskbar_name_active_font_color = #dddddd 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 0
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 152 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #eeeeee 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = -1
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 9
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 9
|
||||||
|
clock_font_color = #eeeeee 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #eeeeee 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
211
sample/vertical-dark-transparent.tint2rc
Normal file
211
sample/vertical-dark-transparent.tint2rc
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
#---- Generated by tint2conf fc8e ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #000000 80
|
||||||
|
border_color = #555555 80
|
||||||
|
background_color_hover = #000000 80
|
||||||
|
border_color_hover = #555555 80
|
||||||
|
background_color_pressed = #000000 80
|
||||||
|
border_color_pressed = #555555 80
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #777777 0
|
||||||
|
border_color = #777777 10
|
||||||
|
background_color_hover = #777777 4
|
||||||
|
border_color_hover = #cccccc 30
|
||||||
|
background_color_pressed = #333333 4
|
||||||
|
border_color_pressed = #777777 30
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #555555 10
|
||||||
|
border_color = #ffffff 60
|
||||||
|
background_color_hover = #cccccc 10
|
||||||
|
border_color_hover = #ffffff 60
|
||||||
|
background_color_pressed = #555555 10
|
||||||
|
border_color_pressed = #ffffff 60
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #777777 0
|
||||||
|
border_color = #777777 30
|
||||||
|
background_color_hover = #777777 4
|
||||||
|
border_color_hover = #cccccc 30
|
||||||
|
background_color_pressed = #777777 0
|
||||||
|
border_color_pressed = #777777 30
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
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
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = CSTL
|
||||||
|
panel_size = 100% 152
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 2 2 2
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom left vertical
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = multi_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #dddddd 100
|
||||||
|
taskbar_name_active_font_color = #dddddd 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 0
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 152 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #eeeeee 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = -1
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 9
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 9
|
||||||
|
clock_font_color = #eeeeee 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #eeeeee 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
221
sample/vertical-light-opaque.tint2rc
Normal file
221
sample/vertical-light-opaque.tint2rc
Normal file
@@ -0,0 +1,221 @@
|
|||||||
|
#---- Generated by tint2conf 1f37 ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 0
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #bbbbbb 100
|
||||||
|
background_color_hover = #eeeeee 100
|
||||||
|
border_color_hover = #bbbbbb 100
|
||||||
|
background_color_pressed = #eeeeee 100
|
||||||
|
border_color_pressed = #bbbbbb 100
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #fafafa 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #cccccc 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #fafafa 100
|
||||||
|
border_color_hover = #aaaaaa 100
|
||||||
|
background_color_pressed = #cccccc 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 100
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #fafafa 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #eeeeee 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #dddddd 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 8: Systray
|
||||||
|
rounded = 3
|
||||||
|
border_width = 0
|
||||||
|
background_color = #dddddd 100
|
||||||
|
border_color = #cccccc 100
|
||||||
|
background_color_hover = #cccccc 100
|
||||||
|
border_color_hover = #cccccc 100
|
||||||
|
background_color_pressed = #cccccc 100
|
||||||
|
border_color_pressed = #cccccc 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = CSTL
|
||||||
|
panel_size = 100% 152
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 2 2 2
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom left vertical
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = multi_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #222222 100
|
||||||
|
taskbar_name_active_font_color = #222222 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 0
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 152 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #222222 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 8
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 9
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 9
|
||||||
|
clock_font_color = #222222 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #222222 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
213
sample/vertical-light-transparent.tint2rc
Normal file
213
sample/vertical-light-transparent.tint2rc
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
#---- Generated by tint2conf 812e ----
|
||||||
|
# See https://gitlab.com/o9000/tint2/wikis/Configure for
|
||||||
|
# full documentation of the configuration options.
|
||||||
|
#-------------------------------------
|
||||||
|
# Backgrounds
|
||||||
|
# Background 1: Panel
|
||||||
|
rounded = 0
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 0
|
||||||
|
border_color = #bbbbbb 10
|
||||||
|
background_color_hover = #eeeeee 0
|
||||||
|
border_color_hover = #bbbbbb 10
|
||||||
|
background_color_pressed = #eeeeee 0
|
||||||
|
border_color_pressed = #bbbbbb 10
|
||||||
|
|
||||||
|
# Background 2: Default task, Iconified task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 4
|
||||||
|
border_color = #eeeeee 0
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #eaeaea 44
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #eaeaea 44
|
||||||
|
|
||||||
|
# Background 3: Active task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 4
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 4: Urgent task
|
||||||
|
rounded = 5
|
||||||
|
border_width = 1
|
||||||
|
background_color = #aa4400 100
|
||||||
|
border_color = #aa7733 100
|
||||||
|
background_color_hover = #aa4400 100
|
||||||
|
border_color_hover = #aa7733 100
|
||||||
|
background_color_pressed = #aa4400 100
|
||||||
|
border_color_pressed = #aa7733 100
|
||||||
|
|
||||||
|
# Background 5: Tooltip
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #ffffaa 100
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #ffffaa 100
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #ffffaa 100
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 6: Inactive desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #eeeeee 4
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #eeeeee 22
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 4
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
# Background 7: Active desktop name
|
||||||
|
rounded = 2
|
||||||
|
border_width = 1
|
||||||
|
background_color = #dddddd 3
|
||||||
|
border_color = #999999 100
|
||||||
|
background_color_hover = #dddddd 3
|
||||||
|
border_color_hover = #999999 100
|
||||||
|
background_color_pressed = #dddddd 3
|
||||||
|
border_color_pressed = #999999 100
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Panel
|
||||||
|
panel_items = CSTL
|
||||||
|
panel_size = 100% 152
|
||||||
|
panel_margin = 0 0
|
||||||
|
panel_padding = 2 2 2
|
||||||
|
panel_background_id = 1
|
||||||
|
wm_menu = 1
|
||||||
|
panel_dock = 0
|
||||||
|
panel_position = bottom left vertical
|
||||||
|
panel_layer = normal
|
||||||
|
panel_monitor = all
|
||||||
|
autohide = 0
|
||||||
|
autohide_show_timeout = 0
|
||||||
|
autohide_hide_timeout = 0.5
|
||||||
|
autohide_height = 2
|
||||||
|
strut_policy = follow_size
|
||||||
|
panel_window_name = tint2
|
||||||
|
disable_transparency = 0
|
||||||
|
mouse_effects = 1
|
||||||
|
font_shadow = 0
|
||||||
|
mouse_hover_icon_asb = 100 0 10
|
||||||
|
mouse_pressed_icon_asb = 100 0 0
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Taskbar
|
||||||
|
taskbar_mode = multi_desktop
|
||||||
|
taskbar_padding = 0 0 2
|
||||||
|
taskbar_background_id = 0
|
||||||
|
taskbar_active_background_id = 0
|
||||||
|
taskbar_name = 1
|
||||||
|
taskbar_hide_inactive_tasks = 0
|
||||||
|
taskbar_hide_different_monitor = 0
|
||||||
|
taskbar_name_padding = 6 3
|
||||||
|
taskbar_name_background_id = 6
|
||||||
|
taskbar_name_active_background_id = 7
|
||||||
|
taskbar_name_font = sans bold 9
|
||||||
|
taskbar_name_font_color = #ffffff 100
|
||||||
|
taskbar_name_active_font_color = #ffffff 100
|
||||||
|
taskbar_distribute_size = 1
|
||||||
|
taskbar_sort_order = none
|
||||||
|
task_align = left
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Task
|
||||||
|
task_text = 1
|
||||||
|
task_icon = 1
|
||||||
|
task_centered = 0
|
||||||
|
urgent_nb_of_blink = 100000
|
||||||
|
task_maximum_size = 152 35
|
||||||
|
task_padding = 4 3 4
|
||||||
|
task_font = sans 8
|
||||||
|
task_tooltip = 1
|
||||||
|
task_font_color = #ffffff 100
|
||||||
|
task_icon_asb = 100 0 0
|
||||||
|
task_background_id = 2
|
||||||
|
task_active_background_id = 3
|
||||||
|
task_urgent_background_id = 4
|
||||||
|
task_iconified_background_id = 2
|
||||||
|
mouse_left = toggle_iconify
|
||||||
|
mouse_middle = none
|
||||||
|
mouse_right = close
|
||||||
|
mouse_scroll_up = prev_task
|
||||||
|
mouse_scroll_down = next_task
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# System tray (notification area)
|
||||||
|
systray_padding = 0 0 2
|
||||||
|
systray_background_id = 0
|
||||||
|
systray_sort = ascending
|
||||||
|
systray_icon_size = 22
|
||||||
|
systray_icon_asb = 100 0 0
|
||||||
|
systray_monitor = 1
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Launcher
|
||||||
|
launcher_padding = 0 0 2
|
||||||
|
launcher_background_id = 0
|
||||||
|
launcher_icon_background_id = 0
|
||||||
|
launcher_icon_size = 22
|
||||||
|
launcher_icon_asb = 100 0 0
|
||||||
|
launcher_icon_theme_override = 0
|
||||||
|
startup_notifications = 1
|
||||||
|
launcher_tooltip = 1
|
||||||
|
launcher_item_app = /usr/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/local/share/applications/tint2conf.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/firefox.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/iceweasel.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/chromium-browser.desktop
|
||||||
|
launcher_item_app = /usr/share/applications/google-chrome.desktop
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Clock
|
||||||
|
time1_format = %H:%M
|
||||||
|
time2_format = %A %d %B
|
||||||
|
time1_font = sans bold 9
|
||||||
|
time1_timezone =
|
||||||
|
time2_timezone =
|
||||||
|
time2_font = sans 9
|
||||||
|
clock_font_color = #ffffff 100
|
||||||
|
clock_padding = 1 0
|
||||||
|
clock_background_id = 0
|
||||||
|
clock_tooltip =
|
||||||
|
clock_tooltip_timezone =
|
||||||
|
clock_lclick_command = zenity --calendar --text ""
|
||||||
|
clock_rclick_command = orage
|
||||||
|
clock_mclick_command =
|
||||||
|
clock_uwheel_command =
|
||||||
|
clock_dwheel_command =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Battery
|
||||||
|
battery_tooltip = 1
|
||||||
|
battery_low_status = 10
|
||||||
|
battery_low_cmd = notify-send "battery low"
|
||||||
|
bat1_font = sans 8
|
||||||
|
bat2_font = sans 6
|
||||||
|
battery_font_color = #ffffff 100
|
||||||
|
battery_padding = 1 0
|
||||||
|
battery_background_id = 0
|
||||||
|
battery_hide = 101
|
||||||
|
battery_lclick_command =
|
||||||
|
battery_rclick_command =
|
||||||
|
battery_mclick_command =
|
||||||
|
battery_uwheel_command =
|
||||||
|
battery_dwheel_command =
|
||||||
|
ac_connected_cmd =
|
||||||
|
ac_disconnected_cmd =
|
||||||
|
|
||||||
|
#-------------------------------------
|
||||||
|
# Tooltip
|
||||||
|
tooltip_show_timeout = 0.5
|
||||||
|
tooltip_hide_timeout = 0.1
|
||||||
|
tooltip_padding = 2 2
|
||||||
|
tooltip_background_id = 5
|
||||||
|
tooltip_font_color = #222222 100
|
||||||
|
tooltip_font = sans 9
|
||||||
|
|
||||||
@@ -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,402 +31,261 @@
|
|||||||
#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__)
|
void battery_init_fonts();
|
||||||
int apm_fd;
|
char *battery_get_tooltip(void *obj);
|
||||||
#endif
|
|
||||||
|
|
||||||
void update_batterys(void* arg)
|
|
||||||
{
|
|
||||||
int old_percentage = battery_state.percentage;
|
|
||||||
int16_t old_hours = battery_state.time.hours;
|
|
||||||
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;
|
||||||
|
battery_tooltip_enabled = TRUE;
|
||||||
|
battery_found = FALSE;
|
||||||
percentage_hide = 101;
|
percentage_hide = 101;
|
||||||
battery_low_cmd_send = 0;
|
battery_low_cmd_sent = FALSE;
|
||||||
battery_timeout = 0;
|
battery_timeout = NULL;
|
||||||
bat1_font_desc = 0;
|
bat1_has_font = FALSE;
|
||||||
bat2_font_desc = 0;
|
bat1_font_desc = NULL;
|
||||||
battery_low_cmd = 0;
|
bat2_has_font = FALSE;
|
||||||
path_energy_now = 0;
|
bat2_font_desc = NULL;
|
||||||
path_energy_full = 0;
|
ac_connected_cmd = NULL;
|
||||||
path_current_now = 0;
|
ac_disconnected_cmd = NULL;
|
||||||
path_status = 0;
|
battery_low_cmd = NULL;
|
||||||
|
battery_lclick_command = NULL;
|
||||||
|
battery_mclick_command = NULL;
|
||||||
|
battery_rclick_command = NULL;
|
||||||
|
battery_uwheel_command = NULL;
|
||||||
|
battery_dwheel_command = NULL;
|
||||||
battery_state.percentage = 0;
|
battery_state.percentage = 0;
|
||||||
battery_state.time.hours = 0;
|
battery_state.time.hours = 0;
|
||||||
battery_state.time.minutes = 0;
|
battery_state.time.minutes = 0;
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
battery_state.time.seconds = 0;
|
||||||
apm_fd = -1;
|
battery_state.state = BATTERY_UNKNOWN;
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
|
||||||
apm_fd = open("/dev/apm", O_RDONLY);
|
|
||||||
if (apm_fd < 0) {
|
|
||||||
warn("init_battery: failed to open /dev/apm.");
|
|
||||||
battery_enabled = 0;
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
#elif !defined(__FreeBSD__)
|
battery_found = battery_os_init();
|
||||||
// check battery
|
|
||||||
GDir *directory = 0;
|
|
||||||
GError *error = NULL;
|
|
||||||
const char *entryname;
|
|
||||||
char *battery_dir = 0;
|
|
||||||
|
|
||||||
directory = g_dir_open("/sys/class/power_supply", 0, &error);
|
if (!battery_timeout)
|
||||||
if (error)
|
battery_timeout = add_timeout(10, 30000, update_battery_tick, 0, &battery_timeout);
|
||||||
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);
|
update_battery();
|
||||||
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();
|
||||||
|
|
||||||
|
if (!battery->area.bg)
|
||||||
battery->area.bg = &g_array_index(backgrounds, Background, 0);
|
battery->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
|
||||||
battery->area.parent = p;
|
battery->area.parent = p;
|
||||||
battery->area.panel = p;
|
battery->area.panel = p;
|
||||||
battery->area._draw_foreground = draw_battery;
|
battery->area._draw_foreground = draw_battery;
|
||||||
battery->area.size_mode = SIZE_BY_CONTENT;
|
battery->area.size_mode = LAYOUT_FIXED;
|
||||||
battery->area._resize = resize_battery;
|
battery->area._resize = resize_battery;
|
||||||
battery->area.on_screen = 1;
|
battery->area.on_screen = TRUE;
|
||||||
battery->area.resize = 1;
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void battery_init_fonts()
|
||||||
|
{
|
||||||
|
if (!bat1_font_desc) {
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
if (!bat2_font_desc) {
|
||||||
|
bat2_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
pango_font_description_set_size(bat2_font_desc,
|
||||||
|
pango_font_description_get_size(bat2_font_desc) - PANGO_SCALE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void update_battery() {
|
void battery_default_font_changed()
|
||||||
#if !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__FreeBSD__)
|
{
|
||||||
// unused on OpenBSD, silence compiler warnings
|
if (!battery_enabled)
|
||||||
FILE *fp;
|
return;
|
||||||
char tmp[25];
|
if (bat1_has_font && bat2_has_font)
|
||||||
int64_t current_now = 0;
|
return;
|
||||||
#endif
|
if (!bat1_has_font) {
|
||||||
#if defined(__FreeBSD__)
|
pango_font_description_free(bat1_font_desc);
|
||||||
int sysctl_out = 0;
|
bat1_font_desc = NULL;
|
||||||
size_t len = 0;
|
}
|
||||||
#endif
|
if (!bat2_has_font) {
|
||||||
int64_t energy_now = 0, energy_full = 0;
|
pango_font_description_free(bat2_font_desc);
|
||||||
int seconds = 0;
|
bat2_font_desc = NULL;
|
||||||
int8_t new_percentage = 0;
|
}
|
||||||
|
battery_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
panels[i].battery.area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&panels[i].battery.area);
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
#if defined(__OpenBSD__) || defined(__NetBSD__)
|
void update_battery_tick(void *arg)
|
||||||
struct apm_power_info info;
|
{
|
||||||
if (ioctl(apm_fd, APM_IOC_GETPOWER, &(info)) < 0)
|
if (!battery_enabled)
|
||||||
warn("power update: APM_IOC_GETPOWER");
|
return;
|
||||||
|
|
||||||
// best attempt at mapping to linux battery states
|
gboolean old_found = battery_found;
|
||||||
battery_state.state = BATTERY_UNKNOWN;
|
int old_percentage = battery_state.percentage;
|
||||||
switch (info.battery_state) {
|
gboolean old_ac_connected = battery_state.ac_connected;
|
||||||
case APM_BATT_CHARGING:
|
int16_t old_hours = battery_state.time.hours;
|
||||||
battery_state.state = BATTERY_CHARGING;
|
int8_t old_minutes = battery_state.time.minutes;
|
||||||
break;
|
|
||||||
default:
|
if (!battery_found) {
|
||||||
battery_state.state = BATTERY_DISCHARGING;
|
init_battery();
|
||||||
break;
|
old_ac_connected = battery_state.ac_connected;
|
||||||
|
}
|
||||||
|
if (update_battery() != 0) {
|
||||||
|
// Try to reconfigure on failed update
|
||||||
|
init_battery();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (info.battery_life == 100)
|
if (old_ac_connected != battery_state.ac_connected) {
|
||||||
battery_state.state = BATTERY_FULL;
|
if (battery_state.ac_connected)
|
||||||
|
tint_exec(ac_connected_cmd);
|
||||||
// no mapping for openbsd really
|
else
|
||||||
energy_full = 0;
|
tint_exec(ac_disconnected_cmd);
|
||||||
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
|
if (battery_state.percentage < battery_low_status && battery_state.state == BATTERY_DISCHARGING &&
|
||||||
energy_full = 0;
|
!battery_low_cmd_sent) {
|
||||||
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);
|
tint_exec(battery_low_cmd);
|
||||||
battery_low_cmd_send = 1;
|
battery_low_cmd_sent = TRUE;
|
||||||
}
|
}
|
||||||
if(battery_low_status < new_percentage && battery_state.state == BATTERY_CHARGING && battery_low_cmd_send) {
|
if (battery_state.percentage > battery_low_status && battery_state.state == BATTERY_CHARGING &&
|
||||||
battery_low_cmd_send = 0;
|
battery_low_cmd_sent) {
|
||||||
|
battery_low_cmd_sent = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
battery_state.percentage = new_percentage;
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
// Show/hide if needed
|
||||||
|
if (!battery_found) {
|
||||||
|
if (panels[i].battery.area.on_screen) {
|
||||||
|
hide(&panels[i].battery.area);
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (battery_state.percentage >= percentage_hide) {
|
||||||
|
if (panels[i].battery.area.on_screen) {
|
||||||
|
hide(&panels[i].battery.area);
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!panels[i].battery.area.on_screen) {
|
||||||
|
show(&panels[i].battery.area);
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 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;
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// clamp percentage to 100 in case battery is misreporting that its current charge is more than its max
|
int update_battery()
|
||||||
if(battery_state.percentage > 100) {
|
{
|
||||||
|
// 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;
|
battery_state.percentage = 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean resize_battery(void *obj)
|
||||||
void draw_battery (void *obj, cairo_t *c)
|
|
||||||
{
|
|
||||||
Battery *battery = obj;
|
|
||||||
PangoLayout *layout;
|
|
||||||
|
|
||||||
layout = pango_cairo_create_layout (c);
|
|
||||||
|
|
||||||
// draw layout
|
|
||||||
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_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);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int resize_battery(void *obj)
|
|
||||||
{
|
{
|
||||||
Battery *battery = obj;
|
Battery *battery = obj;
|
||||||
Panel *panel = battery->area.panel;
|
Panel *panel = battery->area.panel;
|
||||||
@@ -446,33 +293,53 @@ int resize_battery(void *obj)
|
|||||||
int bat_time_height, bat_time_width, bat_time_height_ink;
|
int bat_time_height, bat_time_width, bat_time_height_ink;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
battery->area.redraw = 1;
|
schedule_redraw(&battery->area);
|
||||||
|
|
||||||
snprintf(buf_bat_percentage, sizeof(buf_bat_percentage), "%d%%", battery_state.percentage);
|
snprintf(buf_bat_percentage, sizeof(buf_bat_percentage), "%d%%", battery_state.percentage);
|
||||||
if(battery_state.state == BATTERY_FULL) {
|
if (battery_state.state == BATTERY_FULL) {
|
||||||
strcpy(buf_bat_time, "Full");
|
strcpy(buf_bat_time, "Full");
|
||||||
} else {
|
} else {
|
||||||
snprintf(buf_bat_time, sizeof(buf_bat_time), "%02d:%02d", battery_state.time.hours, battery_state.time.minutes);
|
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));
|
get_text_size2(bat1_font_desc,
|
||||||
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));
|
&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) {
|
if (panel_horizontal) {
|
||||||
int new_size = (bat_percentage_width > bat_time_width) ? bat_percentage_width : bat_time_width;
|
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);
|
new_size += 2 * battery->area.paddingxlr + 2 * battery->area.bg->border.width;
|
||||||
if (new_size > battery->area.width || new_size < (battery->area.width-2)) {
|
if (new_size > battery->area.width || new_size < battery->area.width - 2) {
|
||||||
// we try to limit the number of resize
|
// we try to limit the number of resize
|
||||||
battery->area.width = new_size;
|
battery->area.width = new_size;
|
||||||
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height)/2;
|
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height) / 2;
|
||||||
battery->bat2_posy = battery->bat1_posy + bat_percentage_height;
|
battery->bat2_posy = battery->bat1_posy + bat_percentage_height;
|
||||||
ret = 1;
|
ret = 1;
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
int new_size =
|
||||||
int new_size = bat_percentage_height + bat_time_height + (2 * (battery->area.paddingxlr + battery->area.bg->border.width));
|
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)) {
|
if (new_size > battery->area.height || new_size < battery->area.height - 2) {
|
||||||
battery->area.height = new_size;
|
battery->area.height = new_size;
|
||||||
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height - 2)/2;
|
battery->bat1_posy = (battery->area.height - bat_percentage_height - bat_time_height - 2) / 2;
|
||||||
battery->bat2_posy = battery->bat1_posy + bat_percentage_height + 2;
|
battery->bat2_posy = battery->bat1_posy + bat_percentage_height + 2;
|
||||||
ret = 1;
|
ret = 1;
|
||||||
}
|
}
|
||||||
@@ -480,3 +347,65 @@ int resize_battery(void *obj)
|
|||||||
return ret;
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *battery_get_tooltip(void *obj)
|
||||||
|
{
|
||||||
|
return battery_os_tooltip();
|
||||||
|
}
|
||||||
|
|
||||||
|
void battery_action(int button)
|
||||||
|
{
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|||||||
@@ -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,77 @@
|
|||||||
#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;
|
||||||
Color font;
|
|
||||||
int bat1_posy;
|
int bat1_posy;
|
||||||
int bat2_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;
|
||||||
|
|
||||||
|
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 +94,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);
|
||||||
|
|
||||||
|
/* 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
|
||||||
533
src/battery/linux.c
Normal file
533
src/battery/linux.c
Normal file
@@ -0,0 +1,533 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* 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_energy_now;
|
||||||
|
gchar *path_energy_full;
|
||||||
|
gchar *path_power_now;
|
||||||
|
gchar *path_status;
|
||||||
|
/* sysfs hints */
|
||||||
|
gboolean energy_in_uamp;
|
||||||
|
gboolean power_in_uamp;
|
||||||
|
/* values */
|
||||||
|
gboolean present;
|
||||||
|
gint energy_now;
|
||||||
|
gint energy_full;
|
||||||
|
gint power_now;
|
||||||
|
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 (error) { \
|
||||||
|
g_error_free(err); \
|
||||||
|
return FALSE; \
|
||||||
|
}
|
||||||
|
|
||||||
|
static GList *batteries = NULL;
|
||||||
|
static GList *mains = NULL;
|
||||||
|
|
||||||
|
static guint8 energy_to_percent(gint energy_now, gint energy_full)
|
||||||
|
{
|
||||||
|
return 0.5 + ((energy_now <= energy_full ? energy_now : energy_full) * 100.0) / energy_full;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum psy_type power_supply_get_type(const gchar *entryname)
|
||||||
|
{
|
||||||
|
gchar *path_type = g_build_filename("/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) {
|
||||||
|
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->energy_in_uamp = FALSE;
|
||||||
|
bat->power_in_uamp = FALSE;
|
||||||
|
|
||||||
|
bat->path_present = g_build_filename("/sys/class/power_supply", entryname, "present", NULL);
|
||||||
|
if (!g_file_test(bat->path_present, G_FILE_TEST_EXISTS)) {
|
||||||
|
goto err0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bat->path_energy_now = g_build_filename("/sys/class/power_supply", entryname, "energy_now", NULL);
|
||||||
|
if (!g_file_test(bat->path_energy_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_free(bat->path_energy_now);
|
||||||
|
bat->path_energy_now = g_build_filename("/sys/class/power_supply", entryname, "charge_now", NULL);
|
||||||
|
bat->energy_in_uamp = TRUE;
|
||||||
|
}
|
||||||
|
if (!g_file_test(bat->path_energy_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
goto err1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!bat->energy_in_uamp) {
|
||||||
|
bat->path_energy_full = g_build_filename("/sys/class/power_supply", entryname, "energy_full", NULL);
|
||||||
|
if (!g_file_test(bat->path_energy_full, G_FILE_TEST_EXISTS))
|
||||||
|
goto err2;
|
||||||
|
} else {
|
||||||
|
bat->path_energy_full = g_build_filename("/sys/class/power_supply", entryname, "charge_full", NULL);
|
||||||
|
if (!g_file_test(bat->path_energy_full, G_FILE_TEST_EXISTS))
|
||||||
|
goto err2;
|
||||||
|
}
|
||||||
|
|
||||||
|
bat->path_power_now = g_build_filename("/sys/class/power_supply", entryname, "power_now", NULL);
|
||||||
|
if (!g_file_test(bat->path_power_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
g_free(bat->path_power_now);
|
||||||
|
bat->path_power_now = g_build_filename("/sys/class/power_supply", entryname, "current_now", NULL);
|
||||||
|
bat->power_in_uamp = TRUE;
|
||||||
|
}
|
||||||
|
if (!g_file_test(bat->path_power_now, G_FILE_TEST_EXISTS)) {
|
||||||
|
goto err3;
|
||||||
|
}
|
||||||
|
|
||||||
|
bat->path_status = g_build_filename("/sys/class/power_supply", entryname, "status", NULL);
|
||||||
|
if (!g_file_test(bat->path_status, G_FILE_TEST_EXISTS)) {
|
||||||
|
goto err4;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
err4:
|
||||||
|
g_free(bat->path_status);
|
||||||
|
err3:
|
||||||
|
g_free(bat->path_power_now);
|
||||||
|
err2:
|
||||||
|
g_free(bat->path_energy_full);
|
||||||
|
err1:
|
||||||
|
g_free(bat->path_energy_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("/sys/class/power_supply", entryname, "online", NULL);
|
||||||
|
if (!g_file_test(ac->path_online, G_FILE_TEST_EXISTS)) {
|
||||||
|
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_power_now);
|
||||||
|
g_free(bat->path_energy_full);
|
||||||
|
g_free(bat->path_energy_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, "found battery \"%s\"\n", bat->name);
|
||||||
|
} else {
|
||||||
|
g_free(bat);
|
||||||
|
fprintf(stderr, RED "failed to initialize battery \"%s\"\n" RESET, 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, "found mains \"%s\"\n", ac->name);
|
||||||
|
} else {
|
||||||
|
g_free(ac);
|
||||||
|
fprintf(stderr, RED "failed to initialize mains \"%s\"\n" RESET, entryname);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean battery_os_init()
|
||||||
|
{
|
||||||
|
GDir *directory = 0;
|
||||||
|
GError *error = NULL;
|
||||||
|
const char *entryname;
|
||||||
|
|
||||||
|
battery_os_free();
|
||||||
|
|
||||||
|
directory = g_dir_open("/sys/class/power_supply", 0, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
|
||||||
|
while ((entryname = g_dir_read_name(directory))) {
|
||||||
|
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_power_usage(struct psy_battery *bat, gint old_energy_now, gint64 old_timestamp)
|
||||||
|
{
|
||||||
|
gint64 diff_power = ABS(bat->energy_now - old_energy_now);
|
||||||
|
gint64 diff_time = bat->timestamp - old_timestamp;
|
||||||
|
|
||||||
|
/* µW = (µWh * 3600) / (µs / 1000000) */
|
||||||
|
gint power = diff_power * 3600 * 1000000 / MAX(1, diff_time);
|
||||||
|
|
||||||
|
return power;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean update_linux_battery(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
GError *error = NULL;
|
||||||
|
gchar *data;
|
||||||
|
gsize datalen;
|
||||||
|
|
||||||
|
gint64 old_timestamp = bat->timestamp;
|
||||||
|
int old_energy_now = bat->energy_now;
|
||||||
|
|
||||||
|
/* reset values */
|
||||||
|
bat->present = 0;
|
||||||
|
bat->status = BATTERY_UNKNOWN;
|
||||||
|
bat->energy_now = 0;
|
||||||
|
bat->energy_full = 0;
|
||||||
|
bat->power_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);
|
||||||
|
|
||||||
|
/* energy now */
|
||||||
|
g_file_get_contents(bat->path_energy_now, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
bat->energy_now = atoi(data);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* energy full */
|
||||||
|
g_file_get_contents(bat->path_energy_full, &data, &datalen, &error);
|
||||||
|
RETURN_ON_ERROR(error);
|
||||||
|
bat->energy_full = atoi(data);
|
||||||
|
g_free(data);
|
||||||
|
|
||||||
|
/* power now */
|
||||||
|
g_file_get_contents(bat->path_power_now, &data, &datalen, &error);
|
||||||
|
if (g_error_matches(error, G_FILE_ERROR, G_FILE_ERROR_NODEV)) {
|
||||||
|
/* some hardware does not support reading current power consumption */
|
||||||
|
g_error_free(error);
|
||||||
|
bat->power_now = estimate_power_usage(bat, old_energy_now, old_timestamp);
|
||||||
|
} else if (error) {
|
||||||
|
g_error_free(error);
|
||||||
|
return FALSE;
|
||||||
|
} else {
|
||||||
|
bat->power_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_energy_now = 0;
|
||||||
|
gint64 total_energy_full = 0;
|
||||||
|
gint64 total_power_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_energy_now += bat->energy_now;
|
||||||
|
total_energy_full += bat->energy_full;
|
||||||
|
total_power_now += bat->power_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_power_now > 0) {
|
||||||
|
if (state->state == BATTERY_CHARGING)
|
||||||
|
seconds = 3600 * (total_energy_full - total_energy_now) / total_power_now;
|
||||||
|
else if (state->state == BATTERY_DISCHARGING)
|
||||||
|
seconds = 3600 * total_energy_now / total_power_now;
|
||||||
|
}
|
||||||
|
battery_state_set_time(state, seconds);
|
||||||
|
|
||||||
|
/* calculate percentage */
|
||||||
|
state->percentage = energy_to_percent(total_energy_now, total_energy_full);
|
||||||
|
|
||||||
|
/* AC state */
|
||||||
|
state->ac_connected = ac_connected;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gchar *energy_human_readable(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
gint now = bat->energy_now;
|
||||||
|
gint full = bat->energy_full;
|
||||||
|
gchar unit = bat->energy_in_uamp ? 'A' : 'W';
|
||||||
|
|
||||||
|
if (full >= 1000000) {
|
||||||
|
return g_strdup_printf("%d.%d / %d.%d %ch",
|
||||||
|
now / 1000000,
|
||||||
|
(now % 1000000) / 100000,
|
||||||
|
full / 1000000,
|
||||||
|
(full % 1000000) / 100000,
|
||||||
|
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,
|
||||||
|
unit);
|
||||||
|
} else {
|
||||||
|
return g_strdup_printf("%d / %d µ%ch", now, full, unit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static gchar *power_human_readable(struct psy_battery *bat)
|
||||||
|
{
|
||||||
|
gint power = bat->power_now;
|
||||||
|
gchar unit = bat->power_in_uamp ? 'A' : 'W';
|
||||||
|
|
||||||
|
if (power >= 1000000) {
|
||||||
|
return g_strdup_printf("%d.%d %c", power / 1000000, (power % 1000000) / 100000, unit);
|
||||||
|
} else if (power >= 1000) {
|
||||||
|
return g_strdup_printf("%d.%d m%c", power / 1000, (power % 1000) / 100, unit);
|
||||||
|
} else if (power > 0) {
|
||||||
|
return g_strdup_printf("%d µ%c", power, 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 *power = power_human_readable(bat);
|
||||||
|
gchar *energy = energy_human_readable(bat);
|
||||||
|
gchar *state = (bat->status == BATTERY_UNKNOWN) ? "Level" : chargestate2str(bat->status);
|
||||||
|
|
||||||
|
guint8 percentage = energy_to_percent(bat->energy_now, bat->energy_full);
|
||||||
|
|
||||||
|
g_string_append_printf(tooltip, "\t%s: %s (%u %%)\n\tPower: %s", state, energy, percentage, power);
|
||||||
|
|
||||||
|
g_free(power);
|
||||||
|
g_free(energy);
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
||||||
@@ -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,124 +38,160 @@ 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);
|
||||||
|
|
||||||
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 (i=0 ; i < nb_panel ; i++)
|
for (int i = 0; i < num_panels; i++)
|
||||||
panel1[i].clock.area.resize = 1;
|
panels[i].clock.area.resize_needed = 1;
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
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 (i=0 ; i < nb_panel ; i++)
|
for (int i = 0; i < num_panels; i++)
|
||||||
panel1[i].clock.area.resize = 1;
|
panels[i].clock.area.resize_needed = 1;
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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 (!clock_timeout) {
|
||||||
if (strchr(time1_format, 'S') || strchr(time1_format, 'T') || strchr(time1_format, 'r'))
|
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 = add_timeout(10, 1000, update_clocks_sec, 0, &clock_timeout);
|
||||||
else
|
} else {
|
||||||
clock_timeout = add_timeout(10, 1000, update_clocks_min, 0);
|
clock_timeout = add_timeout(10, 1000, update_clocks_min, 0, &clock_timeout);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
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)
|
if (!clock->area.bg)
|
||||||
clock->area.bg = &g_array_index(backgrounds, Background, 0);
|
clock->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
clock_init_fonts();
|
||||||
clock->area.parent = p;
|
clock->area.parent = p;
|
||||||
clock->area.panel = p;
|
clock->area.panel = p;
|
||||||
|
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._draw_foreground = draw_clock;
|
||||||
clock->area.size_mode = SIZE_BY_CONTENT;
|
clock->area.size_mode = LAYOUT_FIXED;
|
||||||
clock->area._resize = resize_clock;
|
clock->area._resize = resize_clock;
|
||||||
// check consistency
|
// check consistency
|
||||||
if (time1_format == 0)
|
if (!time1_format)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
clock->area.resize = 1;
|
clock->area.resize_needed = 1;
|
||||||
clock->area.on_screen = 1;
|
clock->area.on_screen = TRUE;
|
||||||
|
|
||||||
if (time_tooltip_format) {
|
if (time_tooltip_format) {
|
||||||
clock->area._get_tooltip_text = clock_get_tooltip;
|
clock->area._get_tooltip_text = clock_get_tooltip;
|
||||||
@@ -164,100 +199,165 @@ void init_clock_panel(void *p)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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
|
}
|
||||||
pango_layout_set_font_description (layout, time1_font_desc);
|
if (!time2_font_desc) {
|
||||||
pango_layout_set_width (layout, clock->area.width * PANGO_SCALE);
|
time2_font_desc = pango_font_description_from_string(get_default_font());
|
||||||
pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
pango_font_description_set_size(time2_font_desc,
|
||||||
pango_layout_set_text (layout, buf_time, strlen(buf_time));
|
pango_font_description_get_size(time2_font_desc) - PANGO_SCALE);
|
||||||
|
|
||||||
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()
|
||||||
|
{
|
||||||
|
if (!clock_enabled)
|
||||||
|
return;
|
||||||
|
if (time1_has_font && time2_has_font)
|
||||||
|
return;
|
||||||
|
if (!time1_has_font) {
|
||||||
|
pango_font_description_free(time1_font_desc);
|
||||||
|
time1_font_desc = NULL;
|
||||||
|
}
|
||||||
|
if (!time2_has_font) {
|
||||||
|
pango_font_description_free(time2_font_desc);
|
||||||
|
time2_font_desc = NULL;
|
||||||
|
}
|
||||||
|
clock_init_fonts();
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
panels[i].clock.area.resize_needed = TRUE;
|
||||||
|
schedule_redraw(&panels[i].clock.area);
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
int resize_clock (void *obj)
|
gboolean resize_clock(void *obj)
|
||||||
{
|
{
|
||||||
Clock *clock = obj;
|
Clock *clock = obj;
|
||||||
Panel *panel = clock->area.panel;
|
Panel *panel = clock->area.panel;
|
||||||
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width, ret = 0;
|
int time_height_ink, time_height, time_width, date_height_ink, date_height, date_width;
|
||||||
|
gboolean result = FALSE;
|
||||||
|
|
||||||
|
schedule_redraw(&clock->area);
|
||||||
|
|
||||||
clock->area.redraw = 1;
|
|
||||||
|
|
||||||
date_height = date_width = 0;
|
date_height = date_width = 0;
|
||||||
strftime(buf_time, sizeof(buf_time), time1_format, clock_gettime_for_tz(time1_timezone));
|
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));
|
get_text_size2(time1_font_desc,
|
||||||
|
&time_height_ink,
|
||||||
|
&time_height,
|
||||||
|
&time_width,
|
||||||
|
panel->area.height,
|
||||||
|
panel->area.width,
|
||||||
|
buf_time,
|
||||||
|
strlen(buf_time),
|
||||||
|
PANGO_WRAP_WORD_CHAR,
|
||||||
|
PANGO_ELLIPSIZE_NONE,
|
||||||
|
FALSE);
|
||||||
if (time2_format) {
|
if (time2_format) {
|
||||||
strftime(buf_date, sizeof(buf_date), time2_format, clock_gettime_for_tz(time2_timezone));
|
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));
|
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);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
int new_size = (time_width > date_width) ? time_width : date_width;
|
int new_size = (time_width > date_width) ? time_width : date_width;
|
||||||
new_size += (2*clock->area.paddingxlr) + (2*clock->area.bg->border.width);
|
new_size += (2 * clock->area.paddingxlr) + (2 * clock->area.bg->border.width);
|
||||||
if (new_size > clock->area.width || new_size < (clock->area.width-6)) {
|
if (new_size > clock->area.width || new_size < (clock->area.width - 6)) {
|
||||||
// we try to limit the number of resize
|
// we try to limit the number of resizes
|
||||||
clock->area.width = new_size + 1;
|
clock->area.width = new_size + 1;
|
||||||
clock->time1_posy = (clock->area.height - time_height) / 2;
|
clock->time1_posy = (clock->area.height - time_height) / 2;
|
||||||
if (time2_format) {
|
if (time2_format) {
|
||||||
clock->time1_posy -= (date_height)/2;
|
clock->time1_posy -= (date_height) / 2;
|
||||||
clock->time2_posy = clock->time1_posy + time_height;
|
clock->time2_posy = clock->time1_posy + time_height;
|
||||||
}
|
}
|
||||||
ret = 1;
|
result = TRUE;
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
int new_size = time_height + date_height + (2 * (clock->area.paddingxlr + clock->area.bg->border.width));
|
int new_size = time_height + date_height + (2 * (clock->area.paddingxlr + clock->area.bg->border.width));
|
||||||
if (new_size != clock->area.height) {
|
if (new_size != clock->area.height) {
|
||||||
// we try to limit the number of resize
|
// we try to limit the number of resizes
|
||||||
clock->area.height = new_size;
|
clock->area.height = new_size;
|
||||||
clock->time1_posy = (clock->area.height - time_height) / 2;
|
clock->time1_posy = (clock->area.height - time_height) / 2;
|
||||||
if (time2_format) {
|
if (time2_format) {
|
||||||
clock->time1_posy -= (date_height)/2;
|
clock->time1_posy -= (date_height) / 2;
|
||||||
clock->time2_posy = clock->time1_posy + time_height;
|
clock->time2_posy = clock->time1_posy + time_height;
|
||||||
}
|
}
|
||||||
ret = 1;
|
result = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
void clock_action(int button)
|
||||||
{
|
{
|
||||||
char *command = 0;
|
char *command = NULL;
|
||||||
switch (button) {
|
switch (button) {
|
||||||
case 1:
|
case 1:
|
||||||
command = clock_lclick_command;
|
command = clock_lclick_command;
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 2:
|
||||||
|
command = clock_mclick_command;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
command = clock_rclick_command;
|
command = clock_rclick_command;
|
||||||
break;
|
break;
|
||||||
|
case 4:
|
||||||
|
command = clock_uwheel_command;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
command = clock_dwheel_command;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
tint_exec(command);
|
tint_exec(command);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,6 @@
|
|||||||
#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;
|
||||||
@@ -23,19 +22,22 @@ typedef struct Clock {
|
|||||||
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,10 +48,11 @@ 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);
|
||||||
|
|
||||||
|
|||||||
1229
src/config.c
1229
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
|
||||||
|
|
||||||
|
|||||||
725
src/execplugin/execplugin.c
Normal file
725
src/execplugin/execplugin.c
Normal file
@@ -0,0 +1,725 @@
|
|||||||
|
#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();
|
||||||
|
|
||||||
|
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->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);
|
||||||
|
} 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
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._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.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;
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean resize_execp(void *obj)
|
||||||
|
{
|
||||||
|
Execp *execp = obj;
|
||||||
|
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;
|
||||||
|
|
||||||
|
schedule_redraw(&execp->area);
|
||||||
|
|
||||||
|
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 + execp->area.bg->border.width)
|
||||||
|
: execp->area.width - 2 * (horiz_padding + execp->area.bg->border.width),
|
||||||
|
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 + execp->area.bg->border.width);
|
||||||
|
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 + execp->area.bg->border.width));
|
||||||
|
if (new_size < icon_h + (2 * (vert_padding + execp->area.bg->border.width))) {
|
||||||
|
new_size = icon_h + (2 * (vert_padding + execp->area.bg->border.width));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
new_size = icon_h + interior_padding + txt_height + (2 * (vert_padding + execp->area.bg->border.width));
|
||||||
|
}
|
||||||
|
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 = execp->area.bg->border.width + 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 = execp->area.bg->border.width + 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 = execp->area.bg->border.width + horiz_padding;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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_action(void *obj, int button)
|
||||||
|
{
|
||||||
|
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) {
|
||||||
|
tint_exec(command);
|
||||||
|
} else {
|
||||||
|
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_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) {
|
||||||
|
// 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);
|
||||||
|
}
|
||||||
142
src/execplugin/execplugin.h
Normal file
142
src/execplugin/execplugin.h
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
#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;
|
||||||
|
} 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);
|
||||||
|
|
||||||
|
// 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
|
||||||
93
src/freespace/freespace.c
Normal file
93
src/freespace/freespace.c
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
*
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
void init_freespace_panel(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
FreeSpace *freespace = &panel->freespace;
|
||||||
|
|
||||||
|
if (!freespace->area.bg)
|
||||||
|
freespace->area.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
freespace->area.parent = p;
|
||||||
|
freespace->area.panel = p;
|
||||||
|
freespace->area.size_mode = LAYOUT_FIXED;
|
||||||
|
freespace->area.resize_needed = 1;
|
||||||
|
freespace->area.on_screen = TRUE;
|
||||||
|
freespace->area._resize = resize_freespace;
|
||||||
|
}
|
||||||
|
|
||||||
|
int freespace_get_max_size(Panel *p)
|
||||||
|
{
|
||||||
|
// Get space used by every element except the freespace
|
||||||
|
int size = 0;
|
||||||
|
for (GList *walk = p->area.children; walk; walk = g_list_next(walk)) {
|
||||||
|
Area *a = (Area *)walk->data;
|
||||||
|
|
||||||
|
if (a->_resize == resize_freespace || !a->on_screen)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (panel_horizontal)
|
||||||
|
size += a->width + (a->bg->border.width * 2) + p->area.paddingx;
|
||||||
|
else
|
||||||
|
size += a->height + (a->bg->border.width * 2) + p->area.paddingy;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (panel_horizontal)
|
||||||
|
size = p->area.width - size - (p->area.bg->border.width * 2) - p->area.paddingxlr;
|
||||||
|
else
|
||||||
|
size = p->area.height - size - (p->area.bg->border.width * 2) - p->area.paddingxlr;
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
20
src/freespace/freespace.h
Normal file
20
src/freespace/freespace.h
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* 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;
|
||||||
|
|
||||||
|
void cleanup_freespace();
|
||||||
|
void init_freespace_panel(void *panel);
|
||||||
|
|
||||||
|
gboolean resize_freespace(void *obj);
|
||||||
|
|
||||||
|
#endif
|
||||||
196
src/launcher/apps-common.c
Normal file
196
src/launcher/apps-common.c
Normal file
@@ -0,0 +1,196 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* 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 <glib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int read_desktop_file(const char *path, DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
FILE *fp;
|
||||||
|
char *line = NULL;
|
||||||
|
size_t line_size;
|
||||||
|
char *key, *value;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
entry->path = strdup(path);
|
||||||
|
entry->name = entry->icon = entry->exec = NULL;
|
||||||
|
|
||||||
|
if ((fp = fopen(path, "rt")) == NULL) {
|
||||||
|
fprintf(stderr, "Could not open file %s\n", path);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
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, lang_index_default;
|
||||||
|
#define LANG_DBG 0
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf("Languages:");
|
||||||
|
for (i = 0; languages[i]; i++) {
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf(" %s", languages[i]);
|
||||||
|
}
|
||||||
|
if (LANG_DBG)
|
||||||
|
printf("\n");
|
||||||
|
lang_index_default = i;
|
||||||
|
// we currently do not know about any Name key at all, so use an invalid index
|
||||||
|
lang_index = lang_index_default + 1;
|
||||||
|
|
||||||
|
int inside_desktop_entry = 0;
|
||||||
|
while (getline(&line, &line_size, fp) >= 0) {
|
||||||
|
int len = strlen(line);
|
||||||
|
if (len == 0)
|
||||||
|
continue;
|
||||||
|
line[len - 1] = '\0';
|
||||||
|
if (line[0] == '[') {
|
||||||
|
inside_desktop_entry = (strcmp(line, "[Desktop Entry]") == 0);
|
||||||
|
}
|
||||||
|
if (inside_desktop_entry && parse_dektop_line(line, &key, &value)) {
|
||||||
|
if (strstr(key, "Name") == key) {
|
||||||
|
if (strcmp(key, "Name") == 0 && lang_index > lang_index_default) {
|
||||||
|
entry->name = strdup(value);
|
||||||
|
lang_index = lang_index_default;
|
||||||
|
} else {
|
||||||
|
for (i = 0; languages[i] && i < lang_index; 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 = i;
|
||||||
|
}
|
||||||
|
g_free(localized_key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (!entry->exec && strcmp(key, "Exec") == 0) {
|
||||||
|
entry->exec = strdup(value);
|
||||||
|
} else if (!entry->icon && strcmp(key, "Icon") == 0) {
|
||||||
|
entry->icon = strdup(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose(fp);
|
||||||
|
// From this point:
|
||||||
|
// entry->name, entry->icon, entry->exec will never be empty strings (can be NULL though)
|
||||||
|
|
||||||
|
expand_exec(entry, path);
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_desktop_entry(DesktopEntry *entry)
|
||||||
|
{
|
||||||
|
free(entry->name);
|
||||||
|
free(entry->icon);
|
||||||
|
free(entry->exec);
|
||||||
|
free(entry->path);
|
||||||
|
entry->name = entry->icon = entry->exec = entry->path = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_read_desktop_file()
|
||||||
|
{
|
||||||
|
fprintf(stdout, "\033[1;33m");
|
||||||
|
DesktopEntry entry;
|
||||||
|
read_desktop_file("/usr/share/applications/firefox.desktop", &entry);
|
||||||
|
printf("Name:%s Icon:%s Exec:%s\n", entry.name, entry.icon, entry.exec);
|
||||||
|
fprintf(stdout, "\033[0m");
|
||||||
|
}
|
||||||
30
src/launcher/apps-common.h
Normal file
30
src/launcher/apps-common.h
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
*
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef APPS_COMMON_H
|
||||||
|
#define APPS_COMMON_H
|
||||||
|
|
||||||
|
typedef struct DesktopEntry {
|
||||||
|
char *name;
|
||||||
|
char *exec;
|
||||||
|
char *icon;
|
||||||
|
char *path;
|
||||||
|
} 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.
|
||||||
|
int read_desktop_file(const char *path, DesktopEntry *entry);
|
||||||
|
|
||||||
|
// Empties DesktopEntry: releases the memory of the *members* of entry.
|
||||||
|
void free_desktop_entry(DesktopEntry *entry);
|
||||||
|
|
||||||
|
#endif
|
||||||
660
src/launcher/icon-theme-common.c
Normal file
660
src/launcher/icon-theme-common.c
Normal file
@@ -0,0 +1,660 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* 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 <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "apps-common.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;
|
||||||
|
|
||||||
|
gchar *path;
|
||||||
|
path = g_build_filename(g_get_home_dir(), ".icons", NULL);
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup(path));
|
||||||
|
g_free(path);
|
||||||
|
path = g_build_filename(g_get_home_dir(), ".local/share/icons", NULL);
|
||||||
|
icon_locations = g_slist_append(icon_locations, g_strdup(path));
|
||||||
|
g_free(path);
|
||||||
|
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"));
|
||||||
|
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;
|
||||||
|
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 *themes)
|
||||||
|
{
|
||||||
|
if (!themes)
|
||||||
|
return;
|
||||||
|
for (GSList *l = themes->themes; l; l = l->next) {
|
||||||
|
IconTheme *theme = (IconTheme *)l->data;
|
||||||
|
free_icon_theme(theme);
|
||||||
|
free(theme);
|
||||||
|
}
|
||||||
|
g_slist_free(themes->themes);
|
||||||
|
for (GSList *l = themes->themes_fallback; l; l = l->next) {
|
||||||
|
IconTheme *theme = (IconTheme *)l->data;
|
||||||
|
free_icon_theme(theme);
|
||||||
|
free(theme);
|
||||||
|
}
|
||||||
|
g_slist_free(themes->themes_fallback);
|
||||||
|
free(themes);
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
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";
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "Loading %s. Icon theme :", icon_theme_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
GSList *queued = NULL;
|
||||||
|
load_themes_helper(icon_theme_name, &wrapper->themes, &queued);
|
||||||
|
load_themes_helper("hicolor", &wrapper->themes, &queued);
|
||||||
|
|
||||||
|
// 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, &queued);
|
||||||
|
}
|
||||||
|
g_free(file_name);
|
||||||
|
}
|
||||||
|
g_dir_close(d);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Free the queued list
|
||||||
|
GSList *l;
|
||||||
|
for (l = queued; l; l = l->next)
|
||||||
|
free(l->data);
|
||||||
|
g_slist_free(queued);
|
||||||
|
|
||||||
|
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(IconThemeWrapper *theme, const char *icon_name, int size)
|
||||||
|
{
|
||||||
|
if (!theme)
|
||||||
|
return NULL;
|
||||||
|
icon_name = icon_name ? icon_name : DEFAULT_ICON;
|
||||||
|
char *path = get_icon_path_helper(theme->themes, icon_name, size);
|
||||||
|
if (!path) {
|
||||||
|
path = get_icon_path_helper(theme->themes_fallback, icon_name, size);
|
||||||
|
}
|
||||||
|
if (!path) {
|
||||||
|
fprintf(stderr, "Could not find icon %s\n", icon_name);
|
||||||
|
path = get_icon_path_helper(theme->themes, DEFAULT_ICON, size);
|
||||||
|
}
|
||||||
|
if (!path) {
|
||||||
|
path = get_icon_path_helper(theme->themes_fallback, DEFAULT_ICON, size);
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
}
|
||||||
45
src/launcher/icon-theme-common.h
Normal file
45
src/launcher/icon-theme-common.h
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
* Copyright (C) 2015 (mrovi9000@gmail.com)
|
||||||
|
*
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef ICON_THEME_COMMON_H
|
||||||
|
#define ICON_THEME_COMMON_H
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
typedef struct IconThemeWrapper {
|
||||||
|
// List of IconTheme*
|
||||||
|
GSList *themes;
|
||||||
|
// List of IconTheme*
|
||||||
|
GSList *themes_fallback;
|
||||||
|
} IconThemeWrapper;
|
||||||
|
|
||||||
|
typedef struct IconTheme {
|
||||||
|
char *name;
|
||||||
|
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 free_themes(IconThemeWrapper *themes);
|
||||||
|
|
||||||
|
#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 *theme, const char *icon_name, int size);
|
||||||
|
|
||||||
|
// 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,20 +10,22 @@
|
|||||||
#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"
|
||||||
|
|
||||||
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*
|
IconThemeWrapper *list_themes;
|
||||||
} Launcher;
|
} Launcher;
|
||||||
|
|
||||||
typedef struct LauncherIcon {
|
typedef struct LauncherIcon {
|
||||||
// always start with area
|
// always start with area
|
||||||
Area area;
|
Area area;
|
||||||
Imlib_Image icon_scaled;
|
Imlib_Image image;
|
||||||
Imlib_Image icon_original;
|
Imlib_Image image_hover;
|
||||||
|
Imlib_Image image_pressed;
|
||||||
char *cmd;
|
char *cmd;
|
||||||
char *icon_name;
|
char *icon_name;
|
||||||
char *icon_path;
|
char *icon_path;
|
||||||
@@ -33,39 +35,17 @@ typedef struct LauncherIcon {
|
|||||||
int x, y;
|
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 int startup_notifications;
|
||||||
|
extern Background *launcher_icon_bg;
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_launcher();
|
void default_launcher();
|
||||||
@@ -76,14 +56,15 @@ 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);
|
// Populates the list_themes list
|
||||||
|
void launcher_load_themes(Launcher *launcher);
|
||||||
|
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,15 +26,14 @@
|
|||||||
#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;
|
||||||
@@ -45,33 +44,32 @@ struct _XSettingsClient
|
|||||||
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") && setting->type == XSETTINGS_TYPE_STRING) {
|
if (strcmp(name, "Net/IconThemeName") == 0 && setting->type == XSETTINGS_TYPE_STRING) {
|
||||||
if (icon_theme_name) {
|
fprintf(stderr, "xsettings: %s = %s\n", name, setting->data.v_string);
|
||||||
if (strcmp(icon_theme_name, setting->data.v_string) == 0)
|
if (icon_theme_name_xsettings) {
|
||||||
|
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;
|
||||||
@@ -83,21 +81,19 @@ static void notify_changes (XSettingsClient *client, XSettingsList *old_list)
|
|||||||
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)
|
||||||
@@ -107,8 +103,7 @@ static void notify_changes (XSettingsClient *client, XSettingsList *old_list)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ignore_errors(Display *display, XErrorEvent *event)
|
||||||
static int ignore_errors (Display *display, XErrorEvent *event)
|
|
||||||
{
|
{
|
||||||
return True;
|
return True;
|
||||||
}
|
}
|
||||||
@@ -117,11 +112,11 @@ 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;
|
||||||
@@ -135,25 +130,23 @@ static XSettingsResult fetch_card16 (XSettingsBuffer *buffer, CARD16 *result)
|
|||||||
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;
|
||||||
@@ -167,20 +160,20 @@ static XSettingsResult fetch_card32 (XSettingsBuffer *buffer, CARD32 *result)
|
|||||||
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;
|
||||||
@@ -190,25 +183,26 @@ static XSettingsList *parse_settings (unsigned char *data, size_t len)
|
|||||||
CARD32 i;
|
CARD32 i;
|
||||||
XSettingsSetting *setting = NULL;
|
XSettingsSetting *setting = NULL;
|
||||||
|
|
||||||
local_byte_order = xsettings_byte_order ();
|
local_byte_order = xsettings_byte_order();
|
||||||
|
|
||||||
|
buffer.byte_order = local_byte_order;
|
||||||
buffer.pos = buffer.data = data;
|
buffer.pos = buffer.data = data;
|
||||||
buffer.len = len;
|
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;
|
||||||
|
|
||||||
@@ -218,188 +212,196 @@ static XSettingsList *parse_settings (unsigned char *data, size_t 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);
|
||||||
|
|
||||||
client = malloc (sizeof *client);
|
|
||||||
if (!client)
|
if (!client)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
@@ -413,68 +415,63 @@ XSettingsClient *xsettings_client_new (Display *display, int screen, XSettingsNo
|
|||||||
client->settings = NULL;
|
client->settings = NULL;
|
||||||
|
|
||||||
if (client->watch)
|
if (client->watch)
|
||||||
client->watch (RootWindow (display, screen), True, StructureNotifyMask, client->cb_data);
|
client->watch(RootWindow(display, screen), True, StructureNotifyMask, client->cb_data);
|
||||||
|
|
||||||
check_manager_window (client);
|
check_manager_window(client);
|
||||||
|
|
||||||
if (client->manager_window == None) {
|
if (client->manager_window == None) {
|
||||||
printf("NO XSETTINGS manager, tint2 use config 'launcher_icon_theme'.\n");
|
printf("No XSETTINGS manager, tint2 uses config option 'launcher_icon_theme'.\n");
|
||||||
free (client);
|
free(client);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
return client;
|
return client;
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void xsettings_client_destroy (XSettingsClient *client)
|
|
||||||
{
|
|
||||||
if (client->watch)
|
|
||||||
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);
|
|
||||||
free (client);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
XSettingsResult xsettings_client_get_setting (XSettingsClient *client, const char *name, XSettingsSetting **setting)
|
|
||||||
{
|
|
||||||
XSettingsSetting *search = xsettings_list_lookup (client->settings, name);
|
|
||||||
if (search) {
|
|
||||||
*setting = xsettings_setting_copy (search);
|
|
||||||
return *setting ? XSETTINGS_SUCCESS : XSETTINGS_NO_MEM;
|
|
||||||
}
|
}
|
||||||
else
|
}
|
||||||
|
|
||||||
|
void xsettings_client_destroy(XSettingsClient *client)
|
||||||
|
{
|
||||||
|
if (!client)
|
||||||
|
return;
|
||||||
|
if (client->watch)
|
||||||
|
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);
|
||||||
|
free(client);
|
||||||
|
}
|
||||||
|
|
||||||
|
XSettingsResult xsettings_client_get_setting(XSettingsClient *client, const char *name, XSettingsSetting **setting)
|
||||||
|
{
|
||||||
|
XSettingsSetting *search = xsettings_list_lookup(client->settings, name);
|
||||||
|
if (search) {
|
||||||
|
*setting = xsettings_setting_copy(search);
|
||||||
|
return *setting ? XSETTINGS_SUCCESS : XSETTINGS_NO_MEM;
|
||||||
|
} 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) {
|
||||||
else if (xev->xany.type == PropertyNotify) {
|
read_settings(client);
|
||||||
read_settings (client);
|
|
||||||
return True;
|
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;
|
||||||
{
|
|
||||||
case XSETTINGS_TYPE_INT:
|
|
||||||
result->data.v_int = setting->data.v_int;
|
|
||||||
break;
|
|
||||||
case XSETTINGS_TYPE_COLOR:
|
|
||||||
result->data.v_color = setting->data.v_color;
|
|
||||||
break;
|
|
||||||
case XSETTINGS_TYPE_STRING:
|
|
||||||
str_len = strlen (setting->data.v_string);
|
|
||||||
result->data.v_string = malloc (str_len + 1);
|
|
||||||
if (!result->data.v_string)
|
|
||||||
goto err;
|
|
||||||
|
|
||||||
memcpy (result->data.v_string, setting->data.v_string, str_len + 1);
|
switch (setting->type) {
|
||||||
break;
|
case XSETTINGS_TYPE_INT:
|
||||||
default:
|
result->data.v_int = setting->data.v_int;
|
||||||
break;
|
break;
|
||||||
}
|
case XSETTINGS_TYPE_COLOR:
|
||||||
|
result->data.v_color = setting->data.v_color;
|
||||||
|
break;
|
||||||
|
case XSETTINGS_TYPE_STRING:
|
||||||
|
str_len = strlen(setting->data.v_string);
|
||||||
|
result->data.v_string = calloc(str_len + 1, 1);
|
||||||
|
if (!result->data.v_string)
|
||||||
|
goto err;
|
||||||
|
|
||||||
result->last_change_serial = setting->last_change_serial;
|
memcpy(result->data.v_string, setting->data.v_string, str_len + 1);
|
||||||
|
break;
|
||||||
return result;
|
default:
|
||||||
|
break;
|
||||||
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)
|
result->last_change_serial = setting->last_change_serial;
|
||||||
new_iter->next = new_node;
|
|
||||||
else
|
|
||||||
new = new_node;
|
|
||||||
|
|
||||||
new_iter = new_node;
|
return result;
|
||||||
|
|
||||||
old_iter = old_iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return new;
|
err:
|
||||||
|
if (result->name)
|
||||||
|
free(result->name);
|
||||||
|
free(result);
|
||||||
|
|
||||||
error:
|
return NULL;
|
||||||
xsettings_list_free (new);
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
XSettingsList *xsettings_list_copy(XSettingsList *list)
|
||||||
xsettings_setting_equal (XSettingsSetting *setting_a,
|
|
||||||
XSettingsSetting *setting_b)
|
|
||||||
{
|
{
|
||||||
if (setting_a->type != setting_b->type)
|
XSettingsList *new = NULL;
|
||||||
return 0;
|
XSettingsList *old_iter = list;
|
||||||
|
XSettingsList *new_iter = NULL;
|
||||||
|
|
||||||
if (strcmp (setting_a->name, setting_b->name) != 0)
|
while (old_iter) {
|
||||||
return 0;
|
XSettingsList *new_node;
|
||||||
|
|
||||||
switch (setting_a->type)
|
new_node = calloc(1, sizeof *new_node);
|
||||||
{
|
if (!new_node)
|
||||||
case XSETTINGS_TYPE_INT:
|
goto error;
|
||||||
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;
|
new_node->setting = xsettings_setting_copy(old_iter->setting);
|
||||||
}
|
if (!new_node->setting) {
|
||||||
|
free(new_node);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
if (new_iter)
|
||||||
xsettings_setting_free (XSettingsSetting *setting)
|
new_iter->next = new_node;
|
||||||
{
|
else
|
||||||
if (setting->type == XSETTINGS_TYPE_STRING)
|
new = new_node;
|
||||||
free (setting->data.v_string);
|
|
||||||
|
|
||||||
if (setting->name)
|
new_iter = new_node;
|
||||||
free (setting->name);
|
|
||||||
|
|
||||||
free (setting);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
old_iter = old_iter->next;
|
||||||
xsettings_list_free (XSettingsList *list)
|
|
||||||
{
|
|
||||||
while (list)
|
|
||||||
{
|
|
||||||
XSettingsList *next = list->next;
|
|
||||||
|
|
||||||
xsettings_setting_free (list->setting);
|
|
||||||
free (list);
|
|
||||||
|
|
||||||
list = next;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
XSettingsResult
|
|
||||||
xsettings_list_insert (XSettingsList **list,
|
|
||||||
XSettingsSetting *setting)
|
|
||||||
{
|
|
||||||
XSettingsList *node;
|
|
||||||
XSettingsList *iter;
|
|
||||||
XSettingsList *last = NULL;
|
|
||||||
|
|
||||||
node = malloc (sizeof *node);
|
|
||||||
if (!node)
|
|
||||||
return XSETTINGS_NO_MEM;
|
|
||||||
node->setting = setting;
|
|
||||||
|
|
||||||
iter = *list;
|
|
||||||
while (iter)
|
|
||||||
{
|
|
||||||
int cmp = strcmp (setting->name, iter->setting->name);
|
|
||||||
|
|
||||||
if (cmp < 0)
|
|
||||||
break;
|
|
||||||
else if (cmp == 0)
|
|
||||||
{
|
|
||||||
free (node);
|
|
||||||
return XSETTINGS_DUPLICATE_ENTRY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
last = iter;
|
return new;
|
||||||
iter = iter->next;
|
|
||||||
}
|
error:
|
||||||
|
xsettings_list_free(new);
|
||||||
if (last)
|
return NULL;
|
||||||
last->next = node;
|
|
||||||
else
|
|
||||||
*list = node;
|
|
||||||
|
|
||||||
node->next = iter;
|
|
||||||
|
|
||||||
return XSETTINGS_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
XSettingsResult
|
int xsettings_setting_equal(XSettingsSetting *setting_a, XSettingsSetting *setting_b)
|
||||||
xsettings_list_delete (XSettingsList **list,
|
|
||||||
const char *name)
|
|
||||||
{
|
{
|
||||||
XSettingsList *iter;
|
if (setting_a->type != setting_b->type)
|
||||||
XSettingsList *last = NULL;
|
return 0;
|
||||||
|
|
||||||
iter = *list;
|
if (strcmp(setting_a->name, setting_b->name) != 0)
|
||||||
while (iter)
|
return 0;
|
||||||
{
|
|
||||||
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;
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
last = iter;
|
return 0;
|
||||||
iter = iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return XSETTINGS_FAILED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
XSettingsSetting *
|
void xsettings_setting_free(XSettingsSetting *setting)
|
||||||
xsettings_list_lookup (XSettingsList *list,
|
|
||||||
const char *name)
|
|
||||||
{
|
{
|
||||||
XSettingsList *iter;
|
if (setting->type == XSETTINGS_TYPE_STRING)
|
||||||
|
free(setting->data.v_string);
|
||||||
|
|
||||||
iter = list;
|
if (setting->name)
|
||||||
while (iter)
|
free(setting->name);
|
||||||
{
|
|
||||||
if (strcmp (name, iter->setting->name) == 0)
|
|
||||||
return iter->setting;
|
|
||||||
|
|
||||||
iter = iter->next;
|
free(setting);
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
char
|
void xsettings_list_free(XSettingsList *list)
|
||||||
xsettings_byte_order (void)
|
|
||||||
{
|
{
|
||||||
CARD32 myint = 0x01020304;
|
while (list) {
|
||||||
return (*(char *)&myint == 1) ? MSBFirst : LSBFirst;
|
XSettingsList *next = list->next;
|
||||||
|
|
||||||
|
xsettings_setting_free(list->setting);
|
||||||
|
free(list);
|
||||||
|
|
||||||
|
list = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
XSettingsResult xsettings_list_insert(XSettingsList **list, XSettingsSetting *setting)
|
||||||
|
{
|
||||||
|
XSettingsList *node;
|
||||||
|
XSettingsList *iter;
|
||||||
|
XSettingsList *last = NULL;
|
||||||
|
|
||||||
|
node = calloc(1, sizeof *node);
|
||||||
|
if (!node)
|
||||||
|
return XSETTINGS_NO_MEM;
|
||||||
|
node->setting = setting;
|
||||||
|
|
||||||
|
iter = *list;
|
||||||
|
while (iter) {
|
||||||
|
int cmp = strcmp(setting->name, iter->setting->name);
|
||||||
|
|
||||||
|
if (cmp < 0)
|
||||||
|
break;
|
||||||
|
else if (cmp == 0) {
|
||||||
|
free(node);
|
||||||
|
return XSETTINGS_DUPLICATE_ENTRY;
|
||||||
|
}
|
||||||
|
|
||||||
|
last = iter;
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
XSettingsSetting *xsettings_list_lookup(XSettingsList *list, const char *name)
|
||||||
|
{
|
||||||
|
XSettingsList *iter;
|
||||||
|
|
||||||
|
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
|
||||||
}
|
}
|
||||||
|
|||||||
1002
src/panel.c
1002
src/panel.c
File diff suppressed because it is too large
Load Diff
171
src/panel.h
171
src/panel.h
@@ -20,109 +20,124 @@
|
|||||||
#include "taskbar.h"
|
#include "taskbar.h"
|
||||||
#include "systraybar.h"
|
#include "systraybar.h"
|
||||||
#include "launcher.h"
|
#include "launcher.h"
|
||||||
|
#include "freespace.h"
|
||||||
|
#include "execplugin.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 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* backgrounds;
|
|
||||||
|
|
||||||
extern Imlib_Image default_icon;
|
extern Imlib_Image default_icon;
|
||||||
|
#define DEFAULT_FONT "sans 10"
|
||||||
|
extern char *default_font;
|
||||||
|
extern XSettingsClient *xsettings_client;
|
||||||
|
|
||||||
|
typedef struct Panel {
|
||||||
// tint2 use one panel per monitor and one taskbar per desktop.
|
|
||||||
typedef struct {
|
|
||||||
// always start with area
|
|
||||||
// area.list own all objects of the panel according to config file
|
|
||||||
Area area;
|
Area area;
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// panel
|
|
||||||
Window main_win;
|
Window main_win;
|
||||||
Pixmap temp_pmap;
|
Pixmap temp_pmap;
|
||||||
|
|
||||||
// position relative to root window
|
// position relative to root window
|
||||||
int posx, posy;
|
int posx, posy;
|
||||||
int marginx, marginy;
|
int marginx, marginy;
|
||||||
int pourcentx, pourcenty;
|
int fractional_width, fractional_height;
|
||||||
// location of the panel (monitor number)
|
|
||||||
int monitor;
|
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;
|
||||||
|
|
||||||
// --------------------------------------------------
|
// Per-panel parameters and states for Taskbar and Task
|
||||||
// task and taskbar parameter per panel
|
GlobalTaskbar g_taskbar;
|
||||||
Global_taskbar g_taskbar;
|
GlobalTask g_task;
|
||||||
Global_task g_task;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
// Array of Taskbar, with num_desktops items
|
||||||
// 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;
|
Taskbar *taskbar;
|
||||||
int nb_desktop;
|
int num_desktops;
|
||||||
|
gboolean taskbarname_has_font;
|
||||||
|
PangoFontDescription *taskbarname_font_desc;
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// clock
|
|
||||||
Clock clock;
|
Clock clock;
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// battery
|
|
||||||
#ifdef ENABLE_BATTERY
|
#ifdef ENABLE_BATTERY
|
||||||
Battery battery;
|
Battery battery;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Launcher launcher;
|
Launcher launcher;
|
||||||
|
FreeSpace freespace;
|
||||||
|
GList *execp_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,7 +150,8 @@ 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 set_panel_items_order(Panel *p);
|
void set_panel_items_order(Panel *p);
|
||||||
void set_panel_properties(Panel *p);
|
void set_panel_properties(Panel *p);
|
||||||
@@ -146,17 +162,28 @@ 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);
|
gboolean click_padding(Panel *panel, int x, int y);
|
||||||
int click_clock(Panel *panel, int x, int y);
|
gboolean 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);
|
gboolean 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);
|
||||||
|
|
||||||
|
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();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
617
src/server.c
617
src/server.c
@@ -21,120 +21,135 @@
|
|||||||
#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);
|
||||||
|
server.monitors[i].names = NULL;
|
||||||
|
}
|
||||||
|
free(server.monitors);
|
||||||
|
server.monitors = NULL;
|
||||||
}
|
}
|
||||||
if (server.gc) XFreeGC(server.dsp, server.gc);
|
if (server.gc)
|
||||||
|
XFreeGC(server.display, server.gc);
|
||||||
|
server.gc = NULL;
|
||||||
|
server.disable_transparency = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
@@ -145,11 +160,10 @@ void send_event32 (Window win, Atom at, long data1, long data2, long 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;
|
||||||
@@ -158,256 +172,421 @@ int get_property32 (Window win, Atom at, Atom type)
|
|||||||
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);
|
||||||
|
|
||||||
for (i=0; i<sizeof(pixmap_atoms)/sizeof(Atom); ++i) {
|
|
||||||
res = server_get_property (server.root_win, pixmap_atoms[i], XA_PIXMAP, 0);
|
|
||||||
if (res) {
|
if (res) {
|
||||||
ret = *((Pixmap*)res);
|
ret = *((Pixmap *)res);
|
||||||
XFree(res);
|
XFree(res);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
server.root_pmap = ret;
|
server.root_pmap = ret;
|
||||||
|
|
||||||
if (server.root_pmap == None)
|
if (server.root_pmap == None) {
|
||||||
fprintf(stderr, "tint2 : pixmap background detection failed\n");
|
fprintf(stderr, "tint2 : pixmap background detection failed\n");
|
||||||
else {
|
} else {
|
||||||
XGCValues gcv;
|
XGCValues gcv;
|
||||||
gcv.ts_x_origin = 0;
|
gcv.ts_x_origin = 0;
|
||||||
gcv.ts_y_origin = 0;
|
gcv.ts_y_origin = 0;
|
||||||
gcv.fill_style = FillTiled;
|
gcv.fill_style = FillTiled;
|
||||||
uint mask = GCTileStipXOrigin | GCTileStipYOrigin | GCFillStyle | GCTile;
|
uint mask = GCTileStipXOrigin | GCTileStipYOrigin | GCFillStyle | GCTile;
|
||||||
|
|
||||||
gcv.tile = server.root_pmap;
|
gcv.tile = server.root_pmap;
|
||||||
XChangeGC(server.dsp, server.gc, mask, &gcv);
|
XChangeGC(server.display, 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 (primary_monitor_first) {
|
||||||
|
if (m1->primary && !m2->primary)
|
||||||
|
return -1;
|
||||||
|
if (!m1->primary && m2->primary)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (m1->x < m2->x) {
|
if (m1->x < m2->x) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
} else if (m1->x > m2->x) {
|
||||||
else if (m1->x > m2->x) {
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
} else if (m1->y < m2->y) {
|
||||||
else if (m1->y < m2->y) {
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
} else if (m1->y > m2->y) {
|
||||||
else if (m1->y > m2->y) {
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
return 0;
|
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->y + m1->height) <= (m2->y + m2->height)) {
|
||||||
// m1 included inside m2
|
// m1 included inside m2
|
||||||
return 1;
|
return 1;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
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 );
|
|
||||||
server.monitor = malloc(res->ncrtc * sizeof(Monitor));
|
// Workaround for issue https://gitlab.com/o9000/tint2/issues/353
|
||||||
for (i=0; i<res->ncrtc; ++i) {
|
// on some recent configs, XRRGetScreenResourcesCurrent returns a fantom monitor at last position
|
||||||
XRRCrtcInfo* crtc_info = XRRGetCrtcInfo(server.dsp, res, res->crtcs[i]);
|
{
|
||||||
server.monitor[i].x = crtc_info->x;
|
int i = res->ncrtc - 1;
|
||||||
server.monitor[i].y = crtc_info->y;
|
XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(server.display, res, res->crtcs[i]);
|
||||||
server.monitor[i].width = crtc_info->width;
|
if (!(crtc_info->x || crtc_info->y || crtc_info->width || crtc_info->height)) {
|
||||||
server.monitor[i].height = crtc_info->height;
|
res->ncrtc -= 1;
|
||||||
server.monitor[i].names = malloc((crtc_info->noutput+1) * sizeof(char*));
|
|
||||||
for (j=0; j<crtc_info->noutput; ++j) {
|
|
||||||
XRROutputInfo* output_info = XRRGetOutputInfo(server.dsp, res, crtc_info->outputs[j]);
|
|
||||||
printf("xRandr: Linking output %s with crtc %d\n", output_info->name, i);
|
|
||||||
server.monitor[i].names[j] = g_strdup(output_info->name);
|
|
||||||
XRRFreeOutputInfo(output_info);
|
|
||||||
}
|
}
|
||||||
server.monitor[i].names[j] = 0;
|
|
||||||
XRRFreeCrtcInfo(crtc_info);
|
XRRFreeCrtcInfo(crtc_info);
|
||||||
}
|
}
|
||||||
nbmonitor = res->ncrtc;
|
|
||||||
}
|
printf("xRandr: Found crtc's: %d\n", res->ncrtc);
|
||||||
else if (info && nbmonitor > 0) {
|
server.monitors = calloc(res->ncrtc, sizeof(Monitor));
|
||||||
server.monitor = malloc(nbmonitor * sizeof(Monitor));
|
for (int i = 0; i < res->ncrtc; ++i) {
|
||||||
for (i=0 ; i < nbmonitor ; i++) {
|
XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(server.display, res, res->crtcs[i]);
|
||||||
server.monitor[i].x = info[i].x_org;
|
server.monitors[i].x = crtc_info->x;
|
||||||
server.monitor[i].y = info[i].y_org;
|
server.monitors[i].y = crtc_info->y;
|
||||||
server.monitor[i].width = info[i].width;
|
server.monitors[i].width = crtc_info->width;
|
||||||
server.monitor[i].height = info[i].height;
|
server.monitors[i].height = crtc_info->height;
|
||||||
server.monitor[i].names = 0;
|
server.monitors[i].names = calloc((crtc_info->noutput + 1), sizeof(gchar *));
|
||||||
|
for (int j = 0; j < crtc_info->noutput; ++j) {
|
||||||
|
XRROutputInfo *output_info = XRRGetOutputInfo(server.display, res, crtc_info->outputs[j]);
|
||||||
|
printf("xRandr: Linking output %s with crtc %d\n", output_info->name, i);
|
||||||
|
server.monitors[i].names[j] = g_strdup(output_info->name);
|
||||||
|
XRRFreeOutputInfo(output_info);
|
||||||
|
server.monitors[i].primary = crtc_info->outputs[j] == primary_output;
|
||||||
|
}
|
||||||
|
server.monitors[i].names[crtc_info->noutput] = NULL;
|
||||||
|
XRRFreeCrtcInfo(crtc_info);
|
||||||
|
}
|
||||||
|
num_monitors = res->ncrtc;
|
||||||
|
} 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 = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 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)
|
||||||
|
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 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);
|
||||||
|
|
||||||
|
return (viewport_y / work_area_height) * ncols + viewport_x / work_area_width;
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
// detect number of desktops
|
||||||
// wait 15s to leave some time for window manager startup
|
// wait 15s to leave some time for window manager startup
|
||||||
for (i=0 ; i < 15 ; i++) {
|
for (int i = 0; i < 15; i++) {
|
||||||
server.nb_desktop = server_get_number_of_desktop ();
|
server_get_number_of_desktops();
|
||||||
if (server.nb_desktop > 0) break;
|
if (server.num_desktops > 0)
|
||||||
|
break;
|
||||||
sleep(1);
|
sleep(1);
|
||||||
}
|
}
|
||||||
if (server.nb_desktop == 0) {
|
if (server.num_desktops == 0) {
|
||||||
server.nb_desktop = 1;
|
server.num_desktops = 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;
|
||||||
xvi = XGetVisualInfo(server.dsp, VisualScreenMask|VisualDepthMask|VisualClassMask, &templ, &nvi);
|
XVisualInfo *xvi = 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 (i = 0; i < nvi; i++) {
|
for (int i = 0; i < nvi; i++) {
|
||||||
format = XRenderFindVisualFormat(server.dsp, xvi[i].visual);
|
format = XRenderFindVisualFormat(server.display, 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 = 0;
|
server.real_transparency = FALSE;
|
||||||
server.depth = DefaultDepth(server.dsp, server.screen);
|
server.depth = DefaultDepth(server.display, server.screen);
|
||||||
printf("real transparency off.... depth: %d\n", server.depth);
|
printf("real transparency off.... depth: %d\n", server.depth);
|
||||||
server.colormap = DefaultColormap(server.dsp, server.screen);
|
server.colormap = DefaultColormap(server.display, server.screen);
|
||||||
server.visual = DefaultVisual(server.dsp, server.screen);
|
server.visual = DefaultVisual(server.display, server.screen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
69
src/server.h
69
src/server.h
@@ -15,12 +15,12 @@
|
|||||||
|
|
||||||
#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;
|
||||||
@@ -28,6 +28,7 @@ typedef struct Global_atom
|
|||||||
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;
|
||||||
@@ -57,6 +58,7 @@ typedef struct Global_atom
|
|||||||
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_WM_ICON_NAME;
|
||||||
Atom _NET_CLOSE_WINDOW;
|
Atom _NET_CLOSE_WINDOW;
|
||||||
Atom UTF8_STRING;
|
Atom UTF8_STRING;
|
||||||
Atom _NET_SUPPORTING_WM_CHECK;
|
Atom _NET_SUPPORTING_WM_CHECK;
|
||||||
@@ -71,8 +73,11 @@ typedef struct Global_atom
|
|||||||
Atom MANAGER;
|
Atom MANAGER;
|
||||||
Atom _NET_SYSTEM_TRAY_MESSAGE_DATA;
|
Atom _NET_SYSTEM_TRAY_MESSAGE_DATA;
|
||||||
Atom _NET_SYSTEM_TRAY_ORIENTATION;
|
Atom _NET_SYSTEM_TRAY_ORIENTATION;
|
||||||
|
Atom _NET_SYSTEM_TRAY_ICON_SIZE;
|
||||||
|
Atom _NET_SYSTEM_TRAY_PADDING;
|
||||||
Atom _XEMBED;
|
Atom _XEMBED;
|
||||||
Atom _XEMBED_INFO;
|
Atom _XEMBED_INFO;
|
||||||
|
Atom _NET_WM_PID;
|
||||||
Atom _XSETTINGS_SCREEN;
|
Atom _XSETTINGS_SCREEN;
|
||||||
Atom _XSETTINGS_SETTINGS;
|
Atom _XSETTINGS_SETTINGS;
|
||||||
Atom XdndAware;
|
Atom XdndAware;
|
||||||
@@ -88,33 +93,40 @@ typedef struct Global_atom
|
|||||||
Atom TARGETS;
|
Atom TARGETS;
|
||||||
} Global_atom;
|
} Global_atom;
|
||||||
|
|
||||||
|
typedef struct Monitor {
|
||||||
|
|
||||||
typedef struct Monitor
|
|
||||||
{
|
|
||||||
int x;
|
int x;
|
||||||
int y;
|
int y;
|
||||||
int width;
|
int width;
|
||||||
int height;
|
int height;
|
||||||
char** names;
|
gboolean primary;
|
||||||
|
gchar **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;
|
||||||
int real_transparency;
|
gboolean 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.
|
||||||
|
Viewport *viewports;
|
||||||
|
Monitor *monitors;
|
||||||
|
gboolean got_root_win;
|
||||||
Visual *visual;
|
Visual *visual;
|
||||||
Visual *visual32;
|
Visual *visual32;
|
||||||
// root background
|
// root background
|
||||||
@@ -124,24 +136,22 @@ typedef struct
|
|||||||
Colormap colormap32;
|
Colormap colormap32;
|
||||||
Global_atom atom;
|
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 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,59 @@
|
|||||||
#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;
|
} Systraybar;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
typedef struct
|
Window parent;
|
||||||
{
|
Window win;
|
||||||
Window id;
|
|
||||||
Window tray_id;
|
|
||||||
int x, y;
|
int x, y;
|
||||||
int width, height;
|
int width, height;
|
||||||
// TODO: manage icon's show/hide
|
// TODO: manage icon's show/hide
|
||||||
int hide;
|
gboolean hide;
|
||||||
int depth;
|
int depth;
|
||||||
Damage damage;
|
Damage damage;
|
||||||
timeout* render_timeout;
|
timeout *render_timeout;
|
||||||
|
gboolean empty;
|
||||||
|
int pid;
|
||||||
|
int chrono;
|
||||||
|
struct timespec time_last_render;
|
||||||
|
int num_fast_renders;
|
||||||
|
gboolean reparented;
|
||||||
|
gboolean embedded;
|
||||||
|
int bad_size_counter;
|
||||||
|
timeout *resize_timeout;
|
||||||
|
struct timespec time_last_resize;
|
||||||
|
char *name;
|
||||||
|
Imlib_Image image;
|
||||||
} 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 Systraybar 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;
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_systray();
|
void default_systray();
|
||||||
@@ -64,9 +81,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_panelss);
|
||||||
|
|
||||||
// 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 +92,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
|
||||||
|
|||||||
@@ -18,249 +18,277 @@
|
|||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
**************************************************************************/
|
**************************************************************************/
|
||||||
|
|
||||||
#include <X11/Xlib.h>
|
#include <glib.h>
|
||||||
#include <X11/Xutil.h>
|
|
||||||
#include <X11/Xatom.h>
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <glib.h>
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <X11/Xlib.h>
|
||||||
|
#include <X11/Xutil.h>
|
||||||
|
#include <X11/Xatom.h>
|
||||||
|
|
||||||
#include "window.h"
|
#include "panel.h"
|
||||||
|
#include "server.h"
|
||||||
#include "task.h"
|
#include "task.h"
|
||||||
#include "taskbar.h"
|
#include "taskbar.h"
|
||||||
#include "server.h"
|
|
||||||
#include "panel.h"
|
|
||||||
#include "tooltip.h"
|
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
#include "tooltip.h"
|
||||||
|
#include "window.h"
|
||||||
|
|
||||||
timeout* urgent_timeout;
|
timeout *urgent_timeout;
|
||||||
GSList* urgent_list;
|
GSList *urgent_list;
|
||||||
|
|
||||||
const char* task_get_tooltip(void* obj)
|
char *task_get_tooltip(void *obj)
|
||||||
{
|
{
|
||||||
Task* t = obj;
|
Task *t = (Task *)obj;
|
||||||
return t->title;
|
return strdup(t->title);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Task *add_task(Window win)
|
||||||
Task *add_task (Window win)
|
|
||||||
{
|
{
|
||||||
if (!win) return 0;
|
if (!win)
|
||||||
if (window_is_hidden(win)) return 0;
|
return NULL;
|
||||||
|
if (window_is_hidden(win))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
int monitor;
|
XSelectInput(server.display, win, PropertyChangeMask | StructureNotifyMask);
|
||||||
if (nb_panel > 1) {
|
XFlush(server.display);
|
||||||
monitor = window_get_monitor (win);
|
|
||||||
if (monitor >= nb_panel) monitor = 0;
|
int monitor = 0;
|
||||||
|
if (num_panels > 1) {
|
||||||
|
monitor = get_window_monitor(win);
|
||||||
|
if (monitor >= num_panels)
|
||||||
|
monitor = 0;
|
||||||
}
|
}
|
||||||
else monitor = 0;
|
|
||||||
|
|
||||||
Task new_tsk;
|
// TODO why do we add the task only to the panel for the current monitor, without checking hide_task_diff_monitor?
|
||||||
new_tsk.win = win;
|
|
||||||
new_tsk.desktop = window_get_desktop (win);
|
Task task_template;
|
||||||
new_tsk.area.panel = &panel1[monitor];
|
memset(&task_template, 0, sizeof(task_template));
|
||||||
new_tsk.current_state = window_is_iconified(win) ? TASK_ICONIFIED : TASK_NORMAL;
|
task_template.area.has_mouse_over_effect = panel_config.mouse_effects;
|
||||||
|
task_template.area.has_mouse_press_effect = panel_config.mouse_effects;
|
||||||
|
task_template.win = win;
|
||||||
|
task_template.desktop = get_window_desktop(win);
|
||||||
|
task_template.area.panel = &panels[monitor];
|
||||||
|
task_template.current_state = window_is_iconified(win) ? TASK_ICONIFIED : TASK_NORMAL;
|
||||||
|
get_window_coordinates(win, &task_template.win_x, &task_template.win_y, &task_template.win_w, &task_template.win_h);
|
||||||
|
|
||||||
// allocate only one title and one icon
|
// allocate only one title and one icon
|
||||||
// even with task_on_all_desktop and with task_on_all_panel
|
// even with task_on_all_desktop and with task_on_all_panel
|
||||||
new_tsk.title = 0;
|
task_template.title = NULL;
|
||||||
int k;
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
task_template.icon[k] = NULL;
|
||||||
new_tsk.icon[k] = 0;
|
|
||||||
new_tsk.state_pix[k] = 0;
|
|
||||||
}
|
}
|
||||||
get_title(&new_tsk);
|
task_update_title(&task_template);
|
||||||
get_icon(&new_tsk);
|
task_update_icon(&task_template);
|
||||||
|
|
||||||
//printf("task %s : desktop %d, monitor %d\n", new_tsk->title, desktop, monitor);
|
// fprintf(stderr, "%s %d: win = %ld, task = %s\n", __FUNCTION__, __LINE__, win, task_template.title ? task_template.title : "??");
|
||||||
XSelectInput (server.dsp, new_tsk.win, PropertyChangeMask|StructureNotifyMask);
|
// fprintf(stderr, "new task %s win %u: desktop %d, monitor %d\n", new_task.title, win, new_task.desktop, monitor);
|
||||||
|
|
||||||
GPtrArray* task_group = g_ptr_array_new();
|
GPtrArray *task_buttons = g_ptr_array_new();
|
||||||
Taskbar *tskbar;
|
for (int j = 0; j < panels[monitor].num_desktops; j++) {
|
||||||
Task *new_tsk2=0;
|
if (task_template.desktop != ALL_DESKTOPS && task_template.desktop != j)
|
||||||
int j;
|
continue;
|
||||||
for (j=0 ; j < panel1[monitor].nb_desktop ; j++) {
|
|
||||||
if (new_tsk.desktop != ALLDESKTOP && new_tsk.desktop != j) continue;
|
|
||||||
|
|
||||||
tskbar = &panel1[monitor].taskbar[j];
|
Taskbar *taskbar = &panels[monitor].taskbar[j];
|
||||||
new_tsk2 = malloc(sizeof(Task));
|
Task *task_instance = calloc(1, sizeof(Task));
|
||||||
memcpy(&new_tsk2->area, &panel1[monitor].g_task.area, sizeof(Area));
|
memcpy(&task_instance->area, &panels[monitor].g_task.area, sizeof(Area));
|
||||||
new_tsk2->area.parent = tskbar;
|
task_instance->area.has_mouse_over_effect = panel_config.mouse_effects;
|
||||||
new_tsk2->win = new_tsk.win;
|
task_instance->area.has_mouse_press_effect = panel_config.mouse_effects;
|
||||||
new_tsk2->desktop = new_tsk.desktop;
|
task_instance->win = task_template.win;
|
||||||
new_tsk2->current_state = -1; // to update the current state later in set_task_state...
|
task_instance->desktop = task_template.desktop;
|
||||||
if (new_tsk2->desktop == ALLDESKTOP && server.desktop != j) {
|
task_instance->win_x = task_template.win_x;
|
||||||
// hide ALLDESKTOP task on non-current desktop
|
task_instance->win_y = task_template.win_y;
|
||||||
new_tsk2->area.on_screen = 0;
|
task_instance->win_w = task_template.win_w;
|
||||||
|
task_instance->win_h = task_template.win_h;
|
||||||
|
task_instance->current_state = TASK_UNDEFINED; // to update the current state later in set_task_state...
|
||||||
|
if (task_instance->desktop == ALL_DESKTOPS && server.desktop != j) {
|
||||||
|
// fprintf(stderr, "%s %d: win = %ld hiding task: another desktop\n", __FUNCTION__, __LINE__, win);
|
||||||
|
task_instance->area.on_screen = always_show_all_desktop_tasks;
|
||||||
}
|
}
|
||||||
new_tsk2->title = new_tsk.title;
|
task_instance->title = task_template.title;
|
||||||
if (panel1[monitor].g_task.tooltip_enabled)
|
if (panels[monitor].g_task.tooltip_enabled)
|
||||||
new_tsk2->area._get_tooltip_text = task_get_tooltip;
|
task_instance->area._get_tooltip_text = task_get_tooltip;
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
new_tsk2->icon[k] = new_tsk.icon[k];
|
task_instance->icon[k] = task_template.icon[k];
|
||||||
new_tsk2->state_pix[k] = 0;
|
task_instance->icon_hover[k] = task_template.icon_hover[k];
|
||||||
|
task_instance->icon_press[k] = task_template.icon_press[k];
|
||||||
}
|
}
|
||||||
new_tsk2->icon_width = new_tsk.icon_width;
|
task_instance->icon_width = task_template.icon_width;
|
||||||
new_tsk2->icon_height = new_tsk.icon_height;
|
task_instance->icon_height = task_template.icon_height;
|
||||||
tskbar->area.list = g_slist_append(tskbar->area.list, new_tsk2);
|
|
||||||
tskbar->area.resize = 1;
|
add_area(&task_instance->area, &taskbar->area);
|
||||||
g_ptr_array_add(task_group, new_tsk2);
|
g_ptr_array_add(task_buttons, task_instance);
|
||||||
//printf("add_task panel %d, desktop %d, task %s\n", i, j, new_tsk2->title);
|
|
||||||
}
|
}
|
||||||
Window* key = malloc(sizeof(Window));
|
Window *key = calloc(1, sizeof(Window));
|
||||||
*key = new_tsk.win;
|
*key = task_template.win;
|
||||||
g_hash_table_insert(win_to_task_table, key, task_group);
|
g_hash_table_insert(win_to_task, key, task_buttons);
|
||||||
set_task_state(new_tsk2, new_tsk.current_state);
|
|
||||||
|
|
||||||
if (window_is_urgent(win))
|
set_task_state((Task*)g_ptr_array_index(task_buttons, 0), task_template.current_state);
|
||||||
add_urgent(new_tsk2);
|
|
||||||
|
|
||||||
return new_tsk2;
|
sort_taskbar_for_win(win);
|
||||||
|
|
||||||
|
if (taskbar_mode == MULTI_DESKTOP) {
|
||||||
|
Panel *panel = (Panel*)task_template.area.panel;
|
||||||
|
panel->area.resize_needed = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window_is_urgent(win)) {
|
||||||
|
add_urgent((Task*)g_ptr_array_index(task_buttons, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
return (Task*)g_ptr_array_index(task_buttons, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void remove_task(Task *task)
|
||||||
void remove_task (Task *tsk)
|
|
||||||
{
|
{
|
||||||
if (!tsk) return;
|
if (!task)
|
||||||
|
return;
|
||||||
|
|
||||||
Window win = tsk->win;
|
// fprintf(stderr, "%s %d: win = %ld, task = %s\n", __FUNCTION__, __LINE__, task->win, task->title ? task->title : "??");
|
||||||
|
|
||||||
|
if (taskbar_mode == MULTI_DESKTOP) {
|
||||||
|
Panel *panel = task->area.panel;
|
||||||
|
panel->area.resize_needed = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Window win = task->win;
|
||||||
|
|
||||||
// free title and icon just for the first task
|
// free title and icon just for the first task
|
||||||
// even with task_on_all_desktop and with task_on_all_panel
|
// even with task_on_all_desktop and with task_on_all_panel
|
||||||
//printf("remove_task %s %d\n", tsk->title, tsk->desktop);
|
// printf("remove_task %s %d\n", task->title, task->desktop);
|
||||||
if (tsk->title)
|
if (task->title)
|
||||||
free (tsk->title);
|
free(task->title);
|
||||||
int k;
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
if (task->icon[k]) {
|
||||||
if (tsk->icon[k]) {
|
imlib_context_set_image(task->icon[k]);
|
||||||
imlib_context_set_image(tsk->icon[k]);
|
|
||||||
imlib_free_image();
|
imlib_free_image();
|
||||||
tsk->icon[k] = 0;
|
task->icon[k] = 0;
|
||||||
if (tsk->state_pix[k]) XFreePixmap(server.dsp, tsk->state_pix[k]);
|
}
|
||||||
|
if (task->icon_hover[k]) {
|
||||||
|
imlib_context_set_image(task->icon_hover[k]);
|
||||||
|
imlib_free_image();
|
||||||
|
task->icon_hover[k] = 0;
|
||||||
|
}
|
||||||
|
if (task->icon_press[k]) {
|
||||||
|
imlib_context_set_image(task->icon_press[k]);
|
||||||
|
imlib_free_image();
|
||||||
|
task->icon_press[k] = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int i;
|
GPtrArray *task_buttons = g_hash_table_lookup(win_to_task, &win);
|
||||||
Task *tsk2;
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
Taskbar *tskbar;
|
Task *task2 = g_ptr_array_index(task_buttons, i);
|
||||||
GPtrArray* task_group = g_hash_table_lookup(win_to_task_table, &win);
|
if (task2 == active_task)
|
||||||
for (i=0; i<task_group->len; ++i) {
|
active_task = 0;
|
||||||
tsk2 = g_ptr_array_index(task_group, i);
|
if (task2 == task_drag)
|
||||||
tskbar = tsk2->area.parent;
|
task_drag = 0;
|
||||||
tskbar->area.list = g_slist_remove(tskbar->area.list, tsk2);
|
if (g_slist_find(urgent_list, task2))
|
||||||
tskbar->area.resize = 1;
|
del_urgent(task2);
|
||||||
if (tsk2 == task_active) task_active = 0;
|
remove_area((Area *)task2);
|
||||||
if (tsk2 == task_drag) task_drag = 0;
|
free(task2);
|
||||||
if (g_slist_find(urgent_list, tsk2)) del_urgent(tsk2);
|
|
||||||
free(tsk2);
|
|
||||||
}
|
}
|
||||||
g_hash_table_remove(win_to_task_table, &win);
|
g_hash_table_remove(win_to_task, &win);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean task_update_title(Task *task)
|
||||||
int get_title(Task *tsk)
|
|
||||||
{
|
{
|
||||||
Panel *panel = tsk->area.panel;
|
Panel *panel = task->area.panel;
|
||||||
char *title, *name;
|
|
||||||
|
|
||||||
if (!panel->g_task.text && !panel->g_task.tooltip_enabled) return 0;
|
if (!panel->g_task.has_text && !panel->g_task.tooltip_enabled && taskbar_sort_method != TASKBAR_SORT_TITLE)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
name = server_get_property (tsk->win, server.atom._NET_WM_VISIBLE_NAME, server.atom.UTF8_STRING, 0);
|
char *name = server_get_property(task->win, server.atom._NET_WM_VISIBLE_NAME, server.atom.UTF8_STRING, 0);
|
||||||
if (!name || !strlen(name)) {
|
if (!name || !strlen(name)) {
|
||||||
name = server_get_property (tsk->win, server.atom._NET_WM_NAME, server.atom.UTF8_STRING, 0);
|
name = server_get_property(task->win, server.atom._NET_WM_NAME, server.atom.UTF8_STRING, 0);
|
||||||
if (!name || !strlen(name)) {
|
if (!name || !strlen(name)) {
|
||||||
name = server_get_property (tsk->win, server.atom.WM_NAME, XA_STRING, 0);
|
name = server_get_property(task->win, server.atom.WM_NAME, XA_STRING, 0);
|
||||||
if (!name || !strlen(name)) {
|
|
||||||
name = malloc(10);
|
|
||||||
strcpy(name, "Untitled");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// add space before title
|
char *title;
|
||||||
title = malloc(strlen(name)+2);
|
if (name && strlen(name)) {
|
||||||
if (panel->g_task.icon) strcpy(title, " ");
|
title = strdup(name);
|
||||||
else title[0] = 0;
|
} else {
|
||||||
strcat(title, name);
|
title = strdup("Untitled");
|
||||||
if (name) XFree (name);
|
}
|
||||||
|
if (name)
|
||||||
if (tsk->title) {
|
XFree(name);
|
||||||
|
|
||||||
|
if (task->title) {
|
||||||
// check unecessary title change
|
// check unecessary title change
|
||||||
if (strcmp(tsk->title, title) == 0) {
|
if (strcmp(task->title, title) == 0) {
|
||||||
free(title);
|
free(title);
|
||||||
return 0;
|
return FALSE;
|
||||||
}
|
} else {
|
||||||
else
|
free(task->title);
|
||||||
free(tsk->title);
|
|
||||||
}
|
|
||||||
|
|
||||||
tsk->title = title;
|
|
||||||
GPtrArray* task_group = task_get_tasks(tsk->win);
|
|
||||||
if (task_group) {
|
|
||||||
int i;
|
|
||||||
for (i=0; i<task_group->len; ++i) {
|
|
||||||
Task* tsk2 = g_ptr_array_index(task_group, i);
|
|
||||||
tsk2->title = tsk->title;
|
|
||||||
set_task_redraw(tsk2);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 1;
|
|
||||||
|
task->title = title;
|
||||||
|
GPtrArray *task_buttons = get_task_buttons(task->win);
|
||||||
|
if (task_buttons) {
|
||||||
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
|
Task *task2 = g_ptr_array_index(task_buttons, i);
|
||||||
|
task2->title = task->title;
|
||||||
|
schedule_redraw(&task2->area);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void task_update_icon(Task *task)
|
||||||
void get_icon (Task *tsk)
|
|
||||||
{
|
{
|
||||||
Panel *panel = tsk->area.panel;
|
Panel *panel = task->area.panel;
|
||||||
if (!panel->g_task.icon) return;
|
if (!panel->g_task.has_icon)
|
||||||
int i;
|
return;
|
||||||
Imlib_Image img = NULL;
|
|
||||||
XWMHints *hints = 0;
|
|
||||||
gulong *data = 0;
|
|
||||||
|
|
||||||
int k;
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
if (task->icon[k]) {
|
||||||
if (tsk->icon[k]) {
|
imlib_context_set_image(task->icon[k]);
|
||||||
imlib_context_set_image(tsk->icon[k]);
|
|
||||||
imlib_free_image();
|
imlib_free_image();
|
||||||
tsk->icon[k] = 0;
|
task->icon[k] = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
data = server_get_property (tsk->win, server.atom._NET_WM_ICON, XA_CARDINAL, &i);
|
Imlib_Image img = NULL;
|
||||||
|
|
||||||
|
int i;
|
||||||
|
gulong *data = server_get_property(task->win, server.atom._NET_WM_ICON, XA_CARDINAL, &i);
|
||||||
if (data) {
|
if (data) {
|
||||||
// get ARGB icon
|
// get ARGB icon
|
||||||
int w, h;
|
int w, h;
|
||||||
gulong *tmp_data;
|
gulong *tmp_data;
|
||||||
|
|
||||||
tmp_data = get_best_icon (data, get_icon_count (data, i), i, &w, &h, panel->g_task.icon_size1);
|
tmp_data = get_best_icon(data, get_icon_count(data, i), i, &w, &h, panel->g_task.icon_size1);
|
||||||
#ifdef __x86_64__
|
#ifdef __x86_64__
|
||||||
DATA32 icon_data[w * h];
|
DATA32 icon_data[w * h];
|
||||||
int length = w * h;
|
int length = w * h;
|
||||||
for (i = 0; i < length; ++i)
|
for (int j = 0; j < length; ++j)
|
||||||
icon_data[i] = tmp_data[i];
|
icon_data[j] = tmp_data[j];
|
||||||
img = imlib_create_image_using_copied_data (w, h, icon_data);
|
img = imlib_create_image_using_copied_data(w, h, icon_data);
|
||||||
#else
|
#else
|
||||||
img = imlib_create_image_using_data (w, h, (DATA32*)tmp_data);
|
img = imlib_create_image_using_data(w, h, (DATA32 *)tmp_data);
|
||||||
#endif
|
#endif
|
||||||
}
|
XFree(data);
|
||||||
else {
|
} else {
|
||||||
// get Pixmap icon
|
// get Pixmap icon
|
||||||
hints = XGetWMHints(server.dsp, tsk->win);
|
XWMHints *hints = XGetWMHints(server.display, task->win);
|
||||||
if (hints) {
|
if (hints) {
|
||||||
if (hints->flags & IconPixmapHint && hints->icon_pixmap != 0) {
|
if (hints->flags & IconPixmapHint && hints->icon_pixmap != 0) {
|
||||||
// get width, height and depth for the pixmap
|
// get width, height and depth for the pixmap
|
||||||
Window root;
|
Window root;
|
||||||
int icon_x, icon_y;
|
int icon_x, icon_y;
|
||||||
uint border_width, bpp;
|
uint border_width, bpp;
|
||||||
uint w, h;
|
uint w, h;
|
||||||
|
|
||||||
//printf(" get pixmap\n");
|
// printf(" get pixmap\n");
|
||||||
XGetGeometry(server.dsp, hints->icon_pixmap, &root, &icon_x, &icon_y, &w, &h, &border_width, &bpp);
|
XGetGeometry(server.display, hints->icon_pixmap, &root, &icon_x, &icon_y, &w, &h, &border_width, &bpp);
|
||||||
imlib_context_set_drawable(hints->icon_pixmap);
|
imlib_context_set_drawable(hints->icon_pixmap);
|
||||||
img = imlib_create_image_from_drawable(hints->icon_mask, 0, 0, w, h, 0);
|
img = imlib_create_image_from_drawable(hints->icon_mask, 0, 0, w, h, 0);
|
||||||
}
|
}
|
||||||
|
XFree(hints);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (img == NULL) {
|
if (img == NULL) {
|
||||||
@@ -271,287 +299,301 @@ void get_icon (Task *tsk)
|
|||||||
// transform icons
|
// transform icons
|
||||||
imlib_context_set_image(img);
|
imlib_context_set_image(img);
|
||||||
imlib_image_set_has_alpha(1);
|
imlib_image_set_has_alpha(1);
|
||||||
int w, h;
|
int w = imlib_image_get_width();
|
||||||
w = imlib_image_get_width();
|
int h = imlib_image_get_height();
|
||||||
h = imlib_image_get_height();
|
Imlib_Image orig_image =
|
||||||
Imlib_Image orig_image = imlib_create_cropped_scaled_image(0, 0, w, h, panel->g_task.icon_size1, panel->g_task.icon_size1);
|
imlib_create_cropped_scaled_image(0, 0, w, h, panel->g_task.icon_size1, panel->g_task.icon_size1);
|
||||||
imlib_free_image();
|
imlib_free_image();
|
||||||
|
|
||||||
imlib_context_set_image(orig_image);
|
imlib_context_set_image(orig_image);
|
||||||
tsk->icon_width = imlib_image_get_width();
|
task->icon_width = imlib_image_get_width();
|
||||||
tsk->icon_height = imlib_image_get_height();
|
task->icon_height = imlib_image_get_height();
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
imlib_context_set_image(orig_image);
|
imlib_context_set_image(orig_image);
|
||||||
tsk->icon[k] = imlib_clone_image();
|
task->icon[k] = imlib_clone_image();
|
||||||
imlib_context_set_image(tsk->icon[k]);
|
imlib_context_set_image(task->icon[k]);
|
||||||
DATA32 *data32;
|
DATA32 *data32;
|
||||||
if (panel->g_task.alpha[k] != 100 || panel->g_task.saturation[k] != 0 || panel->g_task.brightness[k] != 0) {
|
if (panel->g_task.alpha[k] != 100 || panel->g_task.saturation[k] != 0 || panel->g_task.brightness[k] != 0) {
|
||||||
data32 = imlib_image_get_data();
|
data32 = imlib_image_get_data();
|
||||||
adjust_asb(data32, tsk->icon_width, tsk->icon_height, panel->g_task.alpha[k], (float)panel->g_task.saturation[k]/100, (float)panel->g_task.brightness[k]/100);
|
adjust_asb(data32,
|
||||||
|
task->icon_width,
|
||||||
|
task->icon_height,
|
||||||
|
panel->g_task.alpha[k],
|
||||||
|
(float)panel->g_task.saturation[k] / 100,
|
||||||
|
(float)panel->g_task.brightness[k] / 100);
|
||||||
imlib_image_put_back_data(data32);
|
imlib_image_put_back_data(data32);
|
||||||
}
|
}
|
||||||
|
if (panel_config.mouse_effects) {
|
||||||
|
task->icon_hover[k] = adjust_icon(task->icon[k],
|
||||||
|
panel_config.mouse_over_alpha,
|
||||||
|
panel_config.mouse_over_saturation,
|
||||||
|
panel_config.mouse_over_brightness);
|
||||||
|
task->icon_press[k] = adjust_icon(task->icon[k],
|
||||||
|
panel_config.mouse_pressed_alpha,
|
||||||
|
panel_config.mouse_pressed_saturation,
|
||||||
|
panel_config.mouse_pressed_brightness);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
imlib_context_set_image(orig_image);
|
imlib_context_set_image(orig_image);
|
||||||
imlib_free_image();
|
imlib_free_image();
|
||||||
|
|
||||||
if (hints)
|
GPtrArray *task_buttons = get_task_buttons(task->win);
|
||||||
XFree(hints);
|
if (task_buttons) {
|
||||||
if (data)
|
for (i = 0; i < task_buttons->len; ++i) {
|
||||||
XFree (data);
|
Task *task2 = g_ptr_array_index(task_buttons, i);
|
||||||
|
task2->icon_width = task->icon_width;
|
||||||
GPtrArray* task_group = task_get_tasks(tsk->win);
|
task2->icon_height = task->icon_height;
|
||||||
if (task_group) {
|
for (int k = 0; k < TASK_STATE_COUNT; ++k) {
|
||||||
for (i=0; i<task_group->len; ++i) {
|
task2->icon[k] = task->icon[k];
|
||||||
Task* tsk2 = g_ptr_array_index(task_group, i);
|
task2->icon_hover[k] = task->icon_hover[k];
|
||||||
tsk2->icon_width = tsk->icon_width;
|
task2->icon_press[k] = task->icon_press[k];
|
||||||
tsk2->icon_height = tsk->icon_height;
|
}
|
||||||
int k;
|
schedule_redraw(&task2->area);
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k)
|
|
||||||
tsk2->icon[k] = tsk->icon[k];
|
|
||||||
set_task_redraw(tsk2);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO icons look too large when the panel is large
|
||||||
void draw_task_icon (Task *tsk, int text_width)
|
void draw_task_icon(Task *task, int text_width)
|
||||||
{
|
{
|
||||||
if (tsk->icon[tsk->current_state] == 0) return;
|
if (!task->icon[task->current_state])
|
||||||
|
return;
|
||||||
|
|
||||||
// Find pos
|
// Find pos
|
||||||
int pos_x;
|
int pos_x;
|
||||||
Panel *panel = (Panel*)tsk->area.panel;
|
Panel *panel = (Panel *)task->area.panel;
|
||||||
if (panel->g_task.centered) {
|
if (panel->g_task.centered) {
|
||||||
if (panel->g_task.text)
|
if (panel->g_task.has_text)
|
||||||
pos_x = (tsk->area.width - text_width - panel->g_task.icon_size1) / 2;
|
pos_x = (task->area.width - text_width - panel->g_task.icon_size1) / 2;
|
||||||
else
|
else
|
||||||
pos_x = (tsk->area.width - panel->g_task.icon_size1) / 2;
|
pos_x = (task->area.width - panel->g_task.icon_size1) / 2;
|
||||||
|
} else {
|
||||||
|
pos_x = panel->g_task.area.paddingxlr + task->area.bg->border.width;
|
||||||
}
|
}
|
||||||
else pos_x = panel->g_task.area.paddingxlr + tsk->area.bg->border.width;
|
|
||||||
|
|
||||||
// Render
|
// Render
|
||||||
imlib_context_set_image (tsk->icon[tsk->current_state]);
|
|
||||||
if (server.real_transparency) {
|
Imlib_Image image;
|
||||||
render_image(tsk->area.pix, pos_x, panel->g_task.icon_posy, imlib_image_get_width(), imlib_image_get_height() );
|
// Render
|
||||||
}
|
if (panel_config.mouse_effects) {
|
||||||
else {
|
if (task->area.mouse_state == MOUSE_OVER)
|
||||||
imlib_context_set_drawable(tsk->area.pix);
|
image = task->icon_hover[task->current_state];
|
||||||
imlib_render_image_on_drawable (pos_x, panel->g_task.icon_posy);
|
else if (task->area.mouse_state == MOUSE_DOWN)
|
||||||
|
image = task->icon_press[task->current_state];
|
||||||
|
else
|
||||||
|
image = task->icon[task->current_state];
|
||||||
|
} else {
|
||||||
|
image = task->icon[task->current_state];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
imlib_context_set_image(image);
|
||||||
|
render_image(task->area.pix, pos_x, panel->g_task.icon_posy);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void draw_task(void *obj, cairo_t *c)
|
||||||
void draw_task (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Task *tsk = obj;
|
Task *task = (Task *)obj;
|
||||||
tsk->state_pix[tsk->current_state] = tsk->area.pix;
|
Panel *panel = (Panel *)task->area.panel;
|
||||||
PangoLayout *layout;
|
|
||||||
Color *config_text;
|
|
||||||
int width=0, height;
|
|
||||||
Panel *panel = (Panel*)tsk->area.panel;
|
|
||||||
//printf("draw_task %d %d\n", tsk->area.posx, tsk->area.posy);
|
|
||||||
|
|
||||||
if (panel->g_task.text) {
|
int text_width = 0;
|
||||||
/* Layout */
|
if (panel->g_task.has_text) {
|
||||||
layout = pango_cairo_create_layout (c);
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
pango_layout_set_font_description (layout, panel->g_task.font_desc);
|
pango_layout_set_font_description(layout, panel->g_task.font_desc);
|
||||||
pango_layout_set_text(layout, tsk->title, -1);
|
pango_layout_set_text(layout, task->title, -1);
|
||||||
|
|
||||||
/* Drawing width and Cut text */
|
pango_layout_set_width(layout, ((Taskbar *)task->area.parent)->text_width * PANGO_SCALE);
|
||||||
// pango use U+22EF or U+2026
|
|
||||||
pango_layout_set_width(layout, ((Taskbar*)tsk->area.parent)->text_width * PANGO_SCALE);
|
|
||||||
pango_layout_set_height(layout, panel->g_task.text_height * PANGO_SCALE);
|
pango_layout_set_height(layout, panel->g_task.text_height * PANGO_SCALE);
|
||||||
pango_layout_set_wrap(layout, PANGO_WRAP_CHAR);
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
|
||||||
pango_layout_set_ellipsize (layout, PANGO_ELLIPSIZE_END);
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
|
||||||
|
|
||||||
/* Center text */
|
if (panel->g_task.centered)
|
||||||
if (panel->g_task.centered) pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
|
||||||
else pango_layout_set_alignment (layout, PANGO_ALIGN_LEFT);
|
else
|
||||||
|
pango_layout_set_alignment(layout, PANGO_ALIGN_LEFT);
|
||||||
|
|
||||||
pango_layout_get_pixel_size (layout, &width, &height);
|
int text_height;
|
||||||
|
pango_layout_get_pixel_size(layout, &text_width, &text_height);
|
||||||
|
double text_posy = (panel->g_task.area.height - text_height) / 2.0;
|
||||||
|
|
||||||
config_text = &panel->g_task.font[tsk->current_state];
|
Color *config_text = &panel->g_task.font[task->current_state];
|
||||||
cairo_set_source_rgba (c, config_text->color[0], config_text->color[1], config_text->color[2], config_text->alpha);
|
draw_text(layout, c, panel->g_task.text_posx, text_posy, config_text, panel->font_shadow);
|
||||||
|
|
||||||
pango_cairo_update_layout (c, layout);
|
g_object_unref(layout);
|
||||||
double text_posy = (panel->g_task.area.height - height) / 2.0;
|
|
||||||
cairo_move_to (c, panel->g_task.text_posx, text_posy);
|
|
||||||
pango_cairo_show_layout (c, layout);
|
|
||||||
|
|
||||||
if (panel->g_task.font_shadow) {
|
|
||||||
cairo_set_source_rgba (c, 0.0, 0.0, 0.0, 0.5);
|
|
||||||
pango_cairo_update_layout (c, layout);
|
|
||||||
cairo_move_to (c, panel->g_task.text_posx + 1, text_posy + 1);
|
|
||||||
pango_cairo_show_layout (c, layout);
|
|
||||||
}
|
|
||||||
g_object_unref (layout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (panel->g_task.icon) {
|
if (panel->g_task.has_icon)
|
||||||
draw_task_icon (tsk, width);
|
draw_task_icon(task, text_width);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void on_change_task(void *obj)
|
||||||
void on_change_task (void *obj)
|
|
||||||
{
|
{
|
||||||
Task *tsk = obj;
|
Task *task = (Task *)obj;
|
||||||
Panel *panel = (Panel*)tsk->area.panel;
|
Panel *panel = (Panel *)task->area.panel;
|
||||||
|
|
||||||
long value[] = { panel->posx+tsk->area.posx, panel->posy+tsk->area.posy, tsk->area.width, tsk->area.height };
|
long value[] = {panel->posx + task->area.posx, panel->posy + task->area.posy, task->area.width, task->area.height};
|
||||||
XChangeProperty (server.dsp, tsk->win, server.atom._NET_WM_ICON_GEOMETRY, XA_CARDINAL, 32, PropModeReplace, (unsigned char*)value, 4);
|
XChangeProperty(server.display,
|
||||||
|
task->win,
|
||||||
// reset Pixmap when position/size changed
|
server.atom._NET_WM_ICON_GEOMETRY,
|
||||||
set_task_redraw(tsk);
|
XA_CARDINAL,
|
||||||
|
32,
|
||||||
|
PropModeReplace,
|
||||||
|
(unsigned char *)value,
|
||||||
|
4);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Given a pointer to the active task (active_task) and a pointer
|
Task *find_active_task(Task *current_task)
|
||||||
// to the task that is currently under the mouse (current_task),
|
|
||||||
// return a pointer to the active task that is on the same desktop
|
|
||||||
// as current_task. Normally this is simply active_task, except when
|
|
||||||
// it is set to appear on all desktops. In that case we search for
|
|
||||||
// another Task on current_task's taskbar, with the same window as
|
|
||||||
// active_task.
|
|
||||||
Task *find_active_task(Task *current_task, Task *active_task)
|
|
||||||
{
|
{
|
||||||
if (active_task == 0)
|
if (active_task == NULL)
|
||||||
return current_task;
|
return current_task;
|
||||||
if (active_task->desktop != ALLDESKTOP)
|
|
||||||
return active_task;
|
|
||||||
if (current_task == 0)
|
|
||||||
return active_task;
|
|
||||||
|
|
||||||
GSList *l0;
|
Taskbar *taskbar = (Taskbar *)current_task->area.parent;
|
||||||
Task *tsk;
|
|
||||||
Taskbar* tskbar = current_task->area.parent;
|
|
||||||
|
|
||||||
l0 = tskbar->area.list;
|
GList *l0 = taskbar->area.children;
|
||||||
if (taskbarname_enabled) l0 = l0->next;
|
if (taskbarname_enabled)
|
||||||
for (; l0 ; l0 = l0->next) {
|
l0 = l0->next;
|
||||||
tsk = l0->data;
|
for (; l0; l0 = l0->next) {
|
||||||
if (tsk->win == active_task->win)
|
Task *task = (Task *)l0->data;
|
||||||
return tsk;
|
if (task->win == active_task->win)
|
||||||
|
return task;
|
||||||
}
|
}
|
||||||
return active_task;
|
|
||||||
|
return current_task;
|
||||||
}
|
}
|
||||||
|
|
||||||
Task *next_task(Task *tsk)
|
Task *next_task(Task *task)
|
||||||
{
|
{
|
||||||
if (tsk == 0)
|
if (!task)
|
||||||
return 0;
|
return NULL;
|
||||||
|
|
||||||
GSList *l0, *lfirst_tsk;
|
Taskbar *taskbar = task->area.parent;
|
||||||
Task *tsk1;
|
|
||||||
Taskbar* tskbar = tsk->area.parent;
|
|
||||||
|
|
||||||
l0 = tskbar->area.list;
|
GList *l0 = taskbar->area.children;
|
||||||
if (taskbarname_enabled) l0 = l0->next;
|
if (taskbarname_enabled)
|
||||||
lfirst_tsk = l0;
|
l0 = l0->next;
|
||||||
for (; l0 ; l0 = l0->next) {
|
GList *lfirst_task = l0;
|
||||||
tsk1 = l0->data;
|
for (; l0; l0 = l0->next) {
|
||||||
if (tsk1 == tsk) {
|
Task *task1 = l0->data;
|
||||||
if (l0->next == 0) l0 = lfirst_tsk;
|
if (task1 == task) {
|
||||||
else l0 = l0->next;
|
l0 = l0->next ? l0->next : lfirst_task;
|
||||||
return l0->data;
|
return l0->data;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
Task *prev_task(Task *tsk)
|
Task *prev_task(Task *task)
|
||||||
{
|
{
|
||||||
if (tsk == 0)
|
if (!task)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
GSList *l0, *lfirst_tsk;
|
Taskbar *taskbar = task->area.parent;
|
||||||
Task *tsk1, *tsk2;
|
|
||||||
Taskbar* tskbar = tsk->area.parent;
|
|
||||||
|
|
||||||
tsk2 = 0;
|
Task *task2 = NULL;
|
||||||
l0 = tskbar->area.list;
|
GList *l0 = taskbar->area.children;
|
||||||
if (taskbarname_enabled) l0 = l0->next;
|
if (taskbarname_enabled)
|
||||||
lfirst_tsk = l0;
|
l0 = l0->next;
|
||||||
for (; l0 ; l0 = l0->next) {
|
GList *lfirst_task = l0;
|
||||||
tsk1 = l0->data;
|
for (; l0; l0 = l0->next) {
|
||||||
if (tsk1 == tsk) {
|
Task *task1 = l0->data;
|
||||||
if (l0 == lfirst_tsk) {
|
if (task1 == task) {
|
||||||
l0 = g_slist_last ( l0 );
|
if (l0 == lfirst_task) {
|
||||||
tsk2 = l0->data;
|
l0 = g_list_last(l0);
|
||||||
|
task2 = l0->data;
|
||||||
}
|
}
|
||||||
return tsk2;
|
return task2;
|
||||||
}
|
}
|
||||||
tsk2 = tsk1;
|
task2 = task1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void reset_active_task()
|
||||||
void active_task()
|
|
||||||
{
|
{
|
||||||
if (task_active) {
|
if (active_task) {
|
||||||
set_task_state(task_active, window_is_iconified(task_active->win) ? TASK_ICONIFIED : TASK_NORMAL);
|
set_task_state(active_task, window_is_iconified(active_task->win) ? TASK_ICONIFIED : TASK_NORMAL);
|
||||||
task_active = 0;
|
active_task = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
Window w1 = window_get_active();
|
Window w1 = get_active_window();
|
||||||
//printf("Change active task %ld\n", w1);
|
// printf("Change active task %ld\n", w1);
|
||||||
|
|
||||||
if (w1) {
|
if (w1) {
|
||||||
if (!task_get_tasks(w1)) {
|
if (!get_task_buttons(w1)) {
|
||||||
Window w2;
|
Window w2;
|
||||||
while (XGetTransientForHint(server.dsp, w1, &w2))
|
while (XGetTransientForHint(server.display, w1, &w2))
|
||||||
w1 = w2;
|
w1 = w2;
|
||||||
}
|
}
|
||||||
set_task_state((task_active = task_get_task(w1)), TASK_ACTIVE);
|
set_task_state((active_task = get_task(w1)), TASK_ACTIVE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void set_task_state(Task *task, TaskState state)
|
||||||
void set_task_state(Task *tsk, int state)
|
|
||||||
{
|
{
|
||||||
if (tsk == 0 || state < 0 || state >= TASK_STATE_COUNT)
|
if (!task || state == TASK_UNDEFINED || state >= TASK_STATE_COUNT)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (tsk->current_state != state) {
|
if (state == TASK_ACTIVE && task->current_state != state) {
|
||||||
GPtrArray* task_group = task_get_tasks(tsk->win);
|
clock_gettime(CLOCK_MONOTONIC, &task->last_activation_time);
|
||||||
if (task_group) {
|
if (taskbar_sort_method == TASKBAR_SORT_LRU || taskbar_sort_method == TASKBAR_SORT_MRU) {
|
||||||
int i;
|
GPtrArray *task_buttons = get_task_buttons(task->win);
|
||||||
for (i=0; i<task_group->len; ++i) {
|
if (task_buttons) {
|
||||||
Task* tsk1 = g_ptr_array_index(task_group, i);
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
tsk1->current_state = state;
|
Task *task1 = g_ptr_array_index(task_buttons, i);
|
||||||
tsk1->area.bg = panel1[0].g_task.background[state];
|
Taskbar *taskbar = (Taskbar *)task1->area.parent;
|
||||||
tsk1->area.pix = tsk1->state_pix[state];
|
sort_tasks(taskbar);
|
||||||
if (tsk1->state_pix[state] == 0)
|
}
|
||||||
tsk1->area.redraw = 1;
|
|
||||||
if (state == TASK_ACTIVE && g_slist_find(urgent_list, tsk1))
|
|
||||||
del_urgent(tsk1);
|
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (task->current_state != state || hide_task_diff_monitor) {
|
||||||
|
GPtrArray *task_buttons = get_task_buttons(task->win);
|
||||||
|
if (task_buttons) {
|
||||||
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
|
Task *task1 = g_ptr_array_index(task_buttons, i);
|
||||||
|
task1->current_state = state;
|
||||||
|
task1->area.bg = panels[0].g_task.background[state];
|
||||||
|
schedule_redraw(&task1->area);
|
||||||
|
if (state == TASK_ACTIVE && g_slist_find(urgent_list, task1))
|
||||||
|
del_urgent(task1);
|
||||||
|
gboolean hide = FALSE;
|
||||||
|
Taskbar *taskbar = (Taskbar *)task1->area.parent;
|
||||||
|
if (task->desktop == ALL_DESKTOPS && server.desktop != taskbar->desktop) {
|
||||||
|
// Hide ALL_DESKTOPS task on non-current desktop
|
||||||
|
hide = !always_show_all_desktop_tasks;
|
||||||
|
}
|
||||||
|
if (hide_inactive_tasks) {
|
||||||
|
// Show only the active task
|
||||||
|
if (state != TASK_ACTIVE) {
|
||||||
|
hide = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (get_window_monitor(task->win) != ((Panel *)task->area.panel)->monitor &&
|
||||||
|
(hide_task_diff_monitor || num_panels > 1)) {
|
||||||
|
hide = TRUE;
|
||||||
|
}
|
||||||
|
if ((!hide) != task1->area.on_screen) {
|
||||||
|
task1->area.on_screen = !hide;
|
||||||
|
schedule_redraw(&task1->area);
|
||||||
|
Panel *p = (Panel *)task->area.panel;
|
||||||
|
task->area.resize_needed = TRUE;
|
||||||
|
p->taskbar->area.resize_needed = TRUE;
|
||||||
|
p->area.resize_needed = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void blink_urgent(void *arg)
|
||||||
void set_task_redraw(Task* tsk)
|
|
||||||
{
|
{
|
||||||
int k;
|
GSList *urgent_task = urgent_list;
|
||||||
for (k=0; k<TASK_STATE_COUNT; ++k) {
|
|
||||||
if (tsk->state_pix[k]) XFreePixmap(server.dsp, tsk->state_pix[k]);
|
|
||||||
tsk->state_pix[k] = 0;
|
|
||||||
}
|
|
||||||
tsk->area.pix = 0;
|
|
||||||
tsk->area.redraw = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void blink_urgent(void* arg)
|
|
||||||
{
|
|
||||||
GSList* urgent_task = urgent_list;
|
|
||||||
while (urgent_task) {
|
while (urgent_task) {
|
||||||
Task* t = urgent_task->data;
|
Task *t = urgent_task->data;
|
||||||
if ( t->urgent_tick < max_tick_urgent) {
|
if (t->urgent_tick < max_tick_urgent) {
|
||||||
if (t->urgent_tick++ % 2)
|
if (t->urgent_tick++ % 2)
|
||||||
set_task_state(t, TASK_URGENT);
|
set_task_state(t, TASK_URGENT);
|
||||||
else
|
else
|
||||||
@@ -559,37 +601,39 @@ void blink_urgent(void* arg)
|
|||||||
}
|
}
|
||||||
urgent_task = urgent_task->next;
|
urgent_task = urgent_task->next;
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void add_urgent(Task *task)
|
||||||
void add_urgent(Task *tsk)
|
|
||||||
{
|
{
|
||||||
if (!tsk)
|
if (!task)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// some programs set urgency hint although they are active
|
// some programs set urgency hint although they are active
|
||||||
if ( task_active && task_active->win == tsk->win )
|
if (active_task && active_task->win == task->win)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
tsk = task_get_task(tsk->win); // always add the first tsk for a task group (omnipresent windows)
|
task = get_task(task->win); // always add the first task for the task buttons (omnipresent windows)
|
||||||
tsk->urgent_tick = 0;
|
task->urgent_tick = 0;
|
||||||
if (g_slist_find(urgent_list, tsk))
|
if (g_slist_find(urgent_list, task))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// not yet in the list, so we have to add it
|
// not yet in the list, so we have to add it
|
||||||
urgent_list = g_slist_prepend(urgent_list, tsk);
|
urgent_list = g_slist_prepend(urgent_list, task);
|
||||||
|
|
||||||
if (urgent_timeout == 0)
|
if (!urgent_timeout)
|
||||||
urgent_timeout = add_timeout(10, 1000, blink_urgent, 0);
|
urgent_timeout = add_timeout(10, 1000, blink_urgent, 0, &urgent_timeout);
|
||||||
|
|
||||||
|
Panel *panel = task->area.panel;
|
||||||
|
if (panel->is_hidden)
|
||||||
|
autohide_show(panel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void del_urgent(Task *task)
|
||||||
void del_urgent(Task *tsk)
|
|
||||||
{
|
{
|
||||||
urgent_list = g_slist_remove(urgent_list, tsk);
|
urgent_list = g_slist_remove(urgent_list, task);
|
||||||
if (urgent_list == 0) {
|
if (!urgent_list) {
|
||||||
stop_timeout(urgent_timeout);
|
stop_timeout(urgent_timeout);
|
||||||
urgent_timeout = 0;
|
urgent_timeout = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,20 +13,20 @@
|
|||||||
#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;
|
|
||||||
extern GSList* urgent_list;
|
|
||||||
|
|
||||||
// --------------------------------------------------
|
|
||||||
// global task parameter
|
|
||||||
typedef struct {
|
|
||||||
Area area;
|
Area area;
|
||||||
|
gboolean has_text;
|
||||||
int text;
|
gboolean has_icon;
|
||||||
int icon;
|
gboolean centered;
|
||||||
int centered;
|
|
||||||
|
|
||||||
int icon_posy;
|
int icon_posy;
|
||||||
int icon_size1;
|
int icon_size1;
|
||||||
int maximum_width;
|
int maximum_width;
|
||||||
@@ -35,55 +35,63 @@ typedef struct {
|
|||||||
int saturation[TASK_STATE_COUNT];
|
int saturation[TASK_STATE_COUNT];
|
||||||
int brightness[TASK_STATE_COUNT];
|
int brightness[TASK_STATE_COUNT];
|
||||||
int config_asb_mask;
|
int config_asb_mask;
|
||||||
Background* background[TASK_STATE_COUNT];
|
Background *background[TASK_STATE_COUNT];
|
||||||
int config_background_mask;
|
int config_background_mask;
|
||||||
// starting position for text ~ task_padding + task_border + icon_size
|
// starting position for text ~ task_padding + task_border + icon_size
|
||||||
double text_posx, text_height;
|
double text_posx, text_height;
|
||||||
|
gboolean has_font;
|
||||||
int font_shadow;
|
|
||||||
PangoFontDescription *font_desc;
|
PangoFontDescription *font_desc;
|
||||||
Color font[TASK_STATE_COUNT];
|
Color font[TASK_STATE_COUNT];
|
||||||
int config_font_mask;
|
int config_font_mask;
|
||||||
int tooltip_enabled;
|
gboolean tooltip_enabled;
|
||||||
} Global_task;
|
} GlobalTask;
|
||||||
|
|
||||||
|
// Stores information about a task.
|
||||||
|
// 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).
|
||||||
// always start with area
|
typedef struct Task {
|
||||||
Area area;
|
Area area;
|
||||||
|
|
||||||
// TODO: group task with list of windows here
|
|
||||||
Window win;
|
Window win;
|
||||||
int desktop;
|
int desktop;
|
||||||
int current_state;
|
TaskState current_state;
|
||||||
Imlib_Image icon[TASK_STATE_COUNT];
|
Imlib_Image icon[TASK_STATE_COUNT];
|
||||||
Pixmap state_pix[TASK_STATE_COUNT];
|
Imlib_Image icon_hover[TASK_STATE_COUNT];
|
||||||
|
Imlib_Image icon_press[TASK_STATE_COUNT];
|
||||||
unsigned int icon_width;
|
unsigned int icon_width;
|
||||||
unsigned int icon_height;
|
unsigned int icon_height;
|
||||||
char *title;
|
char *title;
|
||||||
int urgent_tick;
|
int urgent_tick;
|
||||||
|
// These may not be up-to-date
|
||||||
|
int win_x;
|
||||||
|
int win_y;
|
||||||
|
int win_w;
|
||||||
|
int win_h;
|
||||||
|
struct timespec last_activation_time;
|
||||||
} 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,115 +31,143 @@
|
|||||||
#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 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)); }
|
|
||||||
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;
|
||||||
|
taskbar_distribute_size = FALSE;
|
||||||
|
hide_inactive_tasks = FALSE;
|
||||||
|
hide_task_diff_monitor = FALSE;
|
||||||
|
always_show_all_desktop_tasks = FALSE;
|
||||||
|
taskbar_sort_method = TASKBAR_NOSORT;
|
||||||
|
taskbar_alignment = ALIGN_LEFT;
|
||||||
default_taskbarname();
|
default_taskbarname();
|
||||||
}
|
}
|
||||||
|
|
||||||
void cleanup_taskbar()
|
void cleanup_taskbar()
|
||||||
{
|
{
|
||||||
Panel *panel;
|
|
||||||
Taskbar *tskbar;
|
|
||||||
int i, j, k;
|
|
||||||
|
|
||||||
cleanup_taskbarname();
|
cleanup_taskbarname();
|
||||||
if (win_to_task_table) g_hash_table_foreach(win_to_task_table, taskbar_remove_task, 0);
|
if (win_to_task) {
|
||||||
for (i=0 ; i < nb_panel ; i++) {
|
while (g_hash_table_size(win_to_task)) {
|
||||||
panel = &panel1[i];
|
GHashTableIter iter;
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
gpointer key, value;
|
||||||
tskbar = &panel->taskbar[j];
|
|
||||||
for (k=0; k<TASKBAR_STATE_COUNT; ++k) {
|
g_hash_table_iter_init(&iter, win_to_task);
|
||||||
if (tskbar->state_pix[k]) XFreePixmap(server.dsp, tskbar->state_pix[k]);
|
if (g_hash_table_iter_next(&iter, &key, &value)) {
|
||||||
|
taskbar_remove_task(key);
|
||||||
}
|
}
|
||||||
free_area (&tskbar->area);
|
}
|
||||||
|
g_hash_table_destroy(win_to_task);
|
||||||
|
win_to_task = NULL;
|
||||||
|
}
|
||||||
|
for (int i = 0; i < num_panels; i++) {
|
||||||
|
Panel *panel = &panels[i];
|
||||||
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panel->taskbar[j];
|
||||||
|
free_area(&taskbar->area);
|
||||||
// remove taskbar from the panel
|
// remove taskbar from the panel
|
||||||
panel->area.list = g_slist_remove(panel->area.list, tskbar);
|
remove_area((Area *)taskbar);
|
||||||
}
|
}
|
||||||
if (panel->taskbar) {
|
if (panel->taskbar) {
|
||||||
free(panel->taskbar);
|
free(panel->taskbar);
|
||||||
panel->taskbar = 0;
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
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)
|
||||||
|
win_to_task = g_hash_table_new_full(win_hash, win_compare, free, free_ptr_array);
|
||||||
|
|
||||||
|
active_task = 0;
|
||||||
task_drag = 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;
|
panel->g_taskbar.area_name.size_mode = LAYOUT_FIXED;
|
||||||
panel->g_taskbar.area_name._resize = resize_taskbarname;
|
panel->g_taskbar.area_name._resize = resize_taskbarname;
|
||||||
panel->g_taskbar.area_name._draw_foreground = draw_taskbarname;
|
panel->g_taskbar.area_name._draw_foreground = draw_taskbarname;
|
||||||
panel->g_taskbar.area_name._on_change_layout = 0;
|
panel->g_taskbar.area_name._on_change_layout = 0;
|
||||||
panel->g_taskbar.area_name.resize = 1;
|
panel->g_taskbar.area_name.resize_needed = 1;
|
||||||
panel->g_taskbar.area_name.on_screen = 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;
|
panel->g_taskbar.area.size_mode = LAYOUT_DYNAMIC;
|
||||||
|
panel->g_taskbar.area.alignment = taskbar_alignment;
|
||||||
panel->g_taskbar.area._resize = resize_taskbar;
|
panel->g_taskbar.area._resize = resize_taskbar;
|
||||||
panel->g_taskbar.area._draw_foreground = draw_taskbar;
|
panel->g_taskbar.area.resize_needed = 1;
|
||||||
panel->g_taskbar.area._on_change_layout = on_change_taskbar;
|
panel->g_taskbar.area.on_screen = TRUE;
|
||||||
panel->g_taskbar.area.resize = 1;
|
|
||||||
panel->g_taskbar.area.on_screen = 1;
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
panel->g_taskbar.area.posy = panel->area.bg->border.width + panel->area.paddingy;
|
panel->g_taskbar.area.posy = panel->area.bg->border.width + panel->area.paddingy;
|
||||||
panel->g_taskbar.area.height = panel->area.height - (2 * panel->g_taskbar.area.posy);
|
panel->g_taskbar.area.height = panel->area.height - (2 * panel->g_taskbar.area.posy);
|
||||||
panel->g_taskbar.area_name.posy = panel->g_taskbar.area.posy;
|
panel->g_taskbar.area_name.posy = panel->g_taskbar.area.posy;
|
||||||
panel->g_taskbar.area_name.height = panel->g_taskbar.area.height;
|
panel->g_taskbar.area_name.height = panel->g_taskbar.area.height;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
panel->g_taskbar.area.posx = panel->area.bg->border.width + panel->area.paddingy;
|
panel->g_taskbar.area.posx = panel->area.bg->border.width + panel->area.paddingy;
|
||||||
panel->g_taskbar.area.width = panel->area.width - (2 * panel->g_taskbar.area.posx);
|
panel->g_taskbar.area.width = panel->area.width - (2 * panel->g_taskbar.area.posx);
|
||||||
panel->g_taskbar.area_name.posx = panel->g_taskbar.area.posx;
|
panel->g_taskbar.area_name.posx = panel->g_taskbar.area.posx;
|
||||||
@@ -148,244 +176,411 @@ void init_taskbar_panel(void *p)
|
|||||||
|
|
||||||
// task
|
// task
|
||||||
panel->g_task.area.panel = panel;
|
panel->g_task.area.panel = panel;
|
||||||
panel->g_task.area.size_mode = SIZE_BY_LAYOUT;
|
panel->g_task.area.size_mode = LAYOUT_DYNAMIC;
|
||||||
panel->g_task.area._draw_foreground = draw_task;
|
panel->g_task.area._draw_foreground = draw_task;
|
||||||
panel->g_task.area._on_change_layout = on_change_task;
|
panel->g_task.area._on_change_layout = on_change_task;
|
||||||
panel->g_task.area.resize = 1;
|
panel->g_task.area.resize_needed = 1;
|
||||||
panel->g_task.area.on_screen = 1;
|
panel->g_task.area.on_screen = TRUE;
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_NORMAL)) == 0) {
|
if ((panel->g_task.config_asb_mask & (1 << TASK_NORMAL)) == 0) {
|
||||||
panel->g_task.alpha[TASK_NORMAL] = 100;
|
panel->g_task.alpha[TASK_NORMAL] = 100;
|
||||||
panel->g_task.saturation[TASK_NORMAL] = 0;
|
panel->g_task.saturation[TASK_NORMAL] = 0;
|
||||||
panel->g_task.brightness[TASK_NORMAL] = 0;
|
panel->g_task.brightness[TASK_NORMAL] = 0;
|
||||||
}
|
}
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_ACTIVE)) == 0) {
|
if ((panel->g_task.config_asb_mask & (1 << TASK_ACTIVE)) == 0) {
|
||||||
panel->g_task.alpha[TASK_ACTIVE] = panel->g_task.alpha[TASK_NORMAL];
|
panel->g_task.alpha[TASK_ACTIVE] = panel->g_task.alpha[TASK_NORMAL];
|
||||||
panel->g_task.saturation[TASK_ACTIVE] = panel->g_task.saturation[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];
|
panel->g_task.brightness[TASK_ACTIVE] = panel->g_task.brightness[TASK_NORMAL];
|
||||||
}
|
}
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_ICONIFIED)) == 0) {
|
if ((panel->g_task.config_asb_mask & (1 << TASK_ICONIFIED)) == 0) {
|
||||||
panel->g_task.alpha[TASK_ICONIFIED] = panel->g_task.alpha[TASK_NORMAL];
|
panel->g_task.alpha[TASK_ICONIFIED] = panel->g_task.alpha[TASK_NORMAL];
|
||||||
panel->g_task.saturation[TASK_ICONIFIED] = panel->g_task.saturation[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];
|
panel->g_task.brightness[TASK_ICONIFIED] = panel->g_task.brightness[TASK_NORMAL];
|
||||||
}
|
}
|
||||||
if ((panel->g_task.config_asb_mask & (1<<TASK_URGENT)) == 0) {
|
if ((panel->g_task.config_asb_mask & (1 << TASK_URGENT)) == 0) {
|
||||||
panel->g_task.alpha[TASK_URGENT] = panel->g_task.alpha[TASK_ACTIVE];
|
panel->g_task.alpha[TASK_URGENT] = panel->g_task.alpha[TASK_ACTIVE];
|
||||||
panel->g_task.saturation[TASK_URGENT] = panel->g_task.saturation[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];
|
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_NORMAL)) == 0)
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_ACTIVE)) == 0) panel->g_task.font[TASK_ACTIVE] = 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_ICONIFIED)) == 0) panel->g_task.font[TASK_ICONIFIED] = panel->g_task.font[TASK_NORMAL];
|
if ((panel->g_task.config_font_mask & (1 << TASK_ACTIVE)) == 0)
|
||||||
if ((panel->g_task.config_font_mask & (1<<TASK_URGENT)) == 0) panel->g_task.font[TASK_URGENT] = panel->g_task.font[TASK_ACTIVE];
|
panel->g_task.font[TASK_ACTIVE] = panel->g_task.font[TASK_NORMAL];
|
||||||
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_font_mask & (1 << TASK_ICONIFIED)) == 0)
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_ACTIVE)) == 0) panel->g_task.background[TASK_ACTIVE] = 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_ICONIFIED)) == 0) panel->g_task.background[TASK_ICONIFIED] = panel->g_task.background[TASK_NORMAL];
|
if ((panel->g_task.config_font_mask & (1 << TASK_URGENT)) == 0)
|
||||||
if ((panel->g_task.config_background_mask & (1<<TASK_URGENT)) == 0) panel->g_task.background[TASK_URGENT] = panel->g_task.background[TASK_ACTIVE];
|
panel->g_task.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_horizontal) {
|
||||||
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.area.posy = panel->g_taskbar.area.posy +
|
||||||
|
panel->g_taskbar.background[TASKBAR_NORMAL]->border.width +
|
||||||
|
panel->g_taskbar.area.paddingy;
|
||||||
panel->g_task.area.height = panel->area.height - (2 * panel->g_task.area.posy);
|
panel->g_task.area.height = panel->area.height - (2 * panel->g_task.area.posy);
|
||||||
}
|
} else {
|
||||||
else {
|
panel->g_task.area.posx = panel->g_taskbar.area.posx +
|
||||||
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_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.width = panel->area.width - (2 * panel->g_task.area.posx);
|
||||||
panel->g_task.area.height = panel->g_task.maximum_height;
|
panel->g_task.area.height = panel->g_task.maximum_height;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (j=0; j<TASK_STATE_COUNT; ++j) {
|
for (int j = 0; j < TASK_STATE_COUNT; ++j) {
|
||||||
if (panel->g_task.background[j] == 0)
|
if (!panel->g_task.background[j])
|
||||||
panel->g_task.background[j] = &g_array_index(backgrounds, Background, 0);
|
panel->g_task.background[j] = &g_array_index(backgrounds, Background, 0);
|
||||||
if (panel->g_task.background[j]->border.rounded > panel->g_task.area.height/2) {
|
if (panel->g_task.background[j]->border.radius > panel->g_task.area.height / 2) {
|
||||||
printf("task%sbackground_id has a too large rounded value. Please fix your tint2rc\n", j==0 ? "_" : j==1 ? "_active_" : j==2 ? "_iconified_" : "_urgent_");
|
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]);
|
g_array_append_val(backgrounds, *panel->g_task.background[j]);
|
||||||
panel->g_task.background[j] = &g_array_index(backgrounds, Background, backgrounds->len-1);
|
panel->g_task.background[j] = &g_array_index(backgrounds, Background, backgrounds->len - 1);
|
||||||
panel->g_task.background[j]->border.rounded = panel->g_task.area.height/2;
|
panel->g_task.background[j]->border.radius = panel->g_task.area.height / 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// compute vertical position : text and icon
|
// compute vertical position : text and icon
|
||||||
int height_ink, height;
|
int height_ink, height, width;
|
||||||
get_text_size(panel->g_task.font_desc, &height_ink, &height, panel->area.height, "TAjpg", 5);
|
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);
|
||||||
|
|
||||||
if (!panel->g_task.maximum_width && panel_horizontal)
|
if (!panel->g_task.maximum_width && panel_horizontal)
|
||||||
panel->g_task.maximum_width = server.monitor[panel->monitor].width;
|
panel->g_task.maximum_width = server.monitors[panel->monitor].width;
|
||||||
|
|
||||||
panel->g_task.text_posx = panel->g_task.background[0]->border.width + panel->g_task.area.paddingxlr;
|
panel->g_task.text_posx = panel->g_task.background[0]->border.width + 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 = panel->g_task.area.height - (2 * panel->g_task.area.paddingy);
|
||||||
panel->g_task.text_posx += panel->g_task.icon_size1;
|
panel->g_task.text_posx += panel->g_task.icon_size1 + panel->g_task.area.paddingx;
|
||||||
panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2;
|
panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2;
|
||||||
}
|
}
|
||||||
//printf("monitor %d, task_maximum_width %d\n", panel->monitor, panel->g_task.maximum_width);
|
// printf("monitor %d, task_maximum_width %d\n", panel->monitor, panel->g_task.maximum_width);
|
||||||
|
|
||||||
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
|
else
|
||||||
tskbar->area.bg = panel->g_taskbar.background[TASKBAR_NORMAL];
|
taskbar->area.bg = panel->g_taskbar.background[TASKBAR_NORMAL];
|
||||||
}
|
}
|
||||||
init_taskbarname_panel(panel);
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (!taskbar_enabled) return;
|
GPtrArray *get_task_buttons(Window win)
|
||||||
win = server_get_property (server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results);
|
{
|
||||||
if (!win) return;
|
if (win_to_task && taskbar_enabled)
|
||||||
|
return g_hash_table_lookup(win_to_task, &win);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
GList* win_list = g_hash_table_get_keys(win_to_task_table);
|
void taskbar_refresh_tasklist()
|
||||||
GList* it;
|
{
|
||||||
for (it=win_list; it; it=it->next) {
|
if (!taskbar_enabled)
|
||||||
|
return;
|
||||||
|
// fprintf(stderr, "%s %d:\n", __FUNCTION__, __LINE__);
|
||||||
|
|
||||||
|
int num_results;
|
||||||
|
Window *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);
|
||||||
|
for (GList *it = win_list; it; it = it->next) {
|
||||||
|
int i;
|
||||||
for (i = 0; i < num_results; i++)
|
for (i = 0; i < num_results; i++)
|
||||||
if (*((Window*)it->data) == win[i])
|
if (*((Window *)it->data) == win[i])
|
||||||
break;
|
break;
|
||||||
if (i == num_results)
|
if (i == num_results)
|
||||||
taskbar_remove_task(it->data, 0, 0);
|
taskbar_remove_task(it->data);
|
||||||
}
|
}
|
||||||
g_list_free(win_list);
|
g_list_free(win_list);
|
||||||
|
|
||||||
// Add any new
|
// Add any new
|
||||||
for (i = 0; i < num_results; i++)
|
for (int i = 0; i < num_results; i++)
|
||||||
if (!task_get_task (win[i]))
|
if (!get_task(win[i]))
|
||||||
add_task (win[i]);
|
add_task(win[i]);
|
||||||
|
|
||||||
XFree (win);
|
XFree(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean resize_taskbar(void *obj)
|
||||||
void draw_taskbar (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Taskbar *taskbar = obj;
|
Taskbar *taskbar = (Taskbar *)obj;
|
||||||
int state = (taskbar->desktop == server.desktop) ? TASKBAR_ACTIVE : TASKBAR_NORMAL;
|
Panel *panel = (Panel *)taskbar->area.panel;
|
||||||
|
|
||||||
taskbar->state_pix[state] = taskbar->area.pix;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// printf("resize_taskbar %d %d\n", taskbar->area.posx, taskbar->area.posy);
|
||||||
int resize_taskbar(void *obj)
|
|
||||||
{
|
|
||||||
Taskbar *taskbar = (Taskbar*)obj;
|
|
||||||
Panel *panel = (Panel*)taskbar->area.panel;
|
|
||||||
int text_width;
|
|
||||||
|
|
||||||
//printf("resize_taskbar %d %d\n", taskbar->area.posx, taskbar->area.posy);
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
resize_by_layout(obj, panel->g_task.maximum_width);
|
relayout_with_constraint(&taskbar->area, panel->g_task.maximum_width);
|
||||||
|
|
||||||
text_width = panel->g_task.maximum_width;
|
int text_width = panel->g_task.maximum_width;
|
||||||
GSList *l = taskbar->area.list;
|
GList *l = taskbar->area.children;
|
||||||
if (taskbarname_enabled) l = l->next;
|
if (taskbarname_enabled)
|
||||||
if (l != NULL) {
|
l = l->next;
|
||||||
text_width = ((Task *)l->data)->area.width;
|
for (; l != NULL; l = l->next) {
|
||||||
|
if (((Task *)l->data)->area.on_screen) {
|
||||||
|
text_width = ((Task *)l->data)->area.width;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
taskbar->text_width = text_width - panel->g_task.text_posx - panel->g_task.area.bg->border.width - panel->g_task.area.paddingx;
|
taskbar->text_width =
|
||||||
|
text_width - panel->g_task.text_posx - panel->g_task.area.bg->border.width - panel->g_task.area.paddingxlr;
|
||||||
|
} else {
|
||||||
|
relayout_with_constraint(&taskbar->area, panel->g_task.maximum_height);
|
||||||
|
|
||||||
|
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.paddingxlr;
|
||||||
}
|
}
|
||||||
else {
|
return FALSE;
|
||||||
resize_by_layout(obj, panel->g_task.maximum_height);
|
}
|
||||||
|
|
||||||
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;
|
void set_taskbar_state(Taskbar *taskbar, TaskbarState state)
|
||||||
|
{
|
||||||
|
taskbar->area.bg = panels[0].g_taskbar.background[state];
|
||||||
|
if (taskbarname_enabled) {
|
||||||
|
taskbar->bar_name.area.bg = panels[0].g_taskbar.background_name[state];
|
||||||
|
}
|
||||||
|
if (taskbar_mode != MULTI_DESKTOP) {
|
||||||
|
if (state == TASKBAR_NORMAL)
|
||||||
|
taskbar->area.on_screen = FALSE;
|
||||||
|
else
|
||||||
|
taskbar->area.on_screen = TRUE;
|
||||||
|
}
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void update_taskbar_visibility(void *p)
|
||||||
|
{
|
||||||
|
Panel *panel = (Panel *)p;
|
||||||
|
|
||||||
|
for (int j = 0; j < panel->num_desktops; j++) {
|
||||||
|
Taskbar *taskbar = &panel->taskbar[j];
|
||||||
|
if (taskbar_mode != MULTI_DESKTOP && taskbar->desktop != server.desktop) {
|
||||||
|
// SINGLE_DESKTOP and not current desktop
|
||||||
|
taskbar->area.on_screen = FALSE;
|
||||||
|
} else {
|
||||||
|
taskbar->area.on_screen = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#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;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean taskbar_needs_sort(Taskbar *taskbar)
|
||||||
void on_change_taskbar (void *obj)
|
|
||||||
{
|
{
|
||||||
Taskbar *tskbar = obj;
|
if (taskbar_sort_method == TASKBAR_NOSORT)
|
||||||
int k;
|
return FALSE;
|
||||||
|
|
||||||
// reset Pixmap when position/size changed
|
for (GList *i = taskbar->area.children, *j = i ? i->next : NULL; i && j; i = i->next, j = j->next) {
|
||||||
for (k=0; k<TASKBAR_STATE_COUNT; ++k) {
|
if (compare_tasks(i->data, j->data, taskbar) > 0) {
|
||||||
if (tskbar->state_pix[k]) XFreePixmap(server.dsp, tskbar->state_pix[k]);
|
return TRUE;
|
||||||
tskbar->state_pix[k] = 0;
|
|
||||||
}
|
|
||||||
tskbar->area.pix = 0;
|
|
||||||
tskbar->area.redraw = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void set_taskbar_state(Taskbar *tskbar, int state)
|
|
||||||
{
|
|
||||||
tskbar->area.bg = panel1[0].g_taskbar.background[state];
|
|
||||||
tskbar->area.pix = tskbar->state_pix[state];
|
|
||||||
if (taskbarname_enabled) {
|
|
||||||
tskbar->bar_name.area.bg = panel1[0].g_taskbar.background_name[state];
|
|
||||||
tskbar->bar_name.area.pix = tskbar->bar_name.state_pix[state];
|
|
||||||
}
|
|
||||||
if (panel_mode != MULTI_DESKTOP) {
|
|
||||||
if (state == TASKBAR_NORMAL)
|
|
||||||
tskbar->area.on_screen = 0;
|
|
||||||
else
|
|
||||||
tskbar->area.on_screen = 1;
|
|
||||||
}
|
|
||||||
if (tskbar->area.on_screen == 1) {
|
|
||||||
if (tskbar->state_pix[state] == 0)
|
|
||||||
tskbar->area.redraw = 1;
|
|
||||||
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]) {
|
|
||||||
GSList *l = tskbar->area.list;
|
|
||||||
if (taskbarname_enabled) l = l->next;
|
|
||||||
for ( ; l ; l = l->next)
|
|
||||||
set_task_redraw(l->data);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void sort_tasks(Taskbar *taskbar)
|
||||||
void visible_taskbar(void *p)
|
|
||||||
{
|
{
|
||||||
Panel *panel =(Panel*)p;
|
if (!taskbar)
|
||||||
int j;
|
return;
|
||||||
|
if (!taskbar_needs_sort(taskbar))
|
||||||
|
return;
|
||||||
|
|
||||||
Taskbar *taskbar;
|
taskbar->area.children = g_list_sort_with_data(taskbar->area.children, (GCompareDataFunc)compare_tasks, taskbar);
|
||||||
for (j=0 ; j < panel->nb_desktop ; j++) {
|
taskbar->area.resize_needed = TRUE;
|
||||||
taskbar = &panel->taskbar[j];
|
panel_refresh = TRUE;
|
||||||
if (panel_mode != MULTI_DESKTOP && taskbar->desktop != server.desktop) {
|
((Panel *)taskbar->area.panel)->area.resize_needed = TRUE;
|
||||||
// SINGLE_DESKTOP and not current desktop
|
}
|
||||||
taskbar->area.on_screen = 0;
|
|
||||||
|
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);
|
||||||
}
|
}
|
||||||
else {
|
for (int i = 0; i < task_buttons->len; ++i) {
|
||||||
taskbar->area.on_screen = 1;
|
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
panel_refresh = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -11,67 +11,82 @@
|
|||||||
#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;
|
||||||
Pixmap state_pix[TASKBAR_STATE_COUNT];
|
gchar *name;
|
||||||
|
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;
|
int desktop;
|
||||||
Pixmap state_pix[TASKBAR_STATE_COUNT];
|
TaskbarName bar_name;
|
||||||
|
|
||||||
Taskbarname bar_name;
|
|
||||||
|
|
||||||
// task parameters
|
|
||||||
int text_width;
|
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];
|
||||||
} Global_taskbar;
|
} GlobalTaskbar;
|
||||||
|
|
||||||
|
extern gboolean taskbar_enabled;
|
||||||
|
extern gboolean taskbar_distribute_size;
|
||||||
|
extern gboolean hide_inactive_tasks;
|
||||||
|
extern gboolean hide_task_diff_monitor;
|
||||||
|
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);
|
||||||
|
|
||||||
|
void set_taskbar_state(Taskbar *taskbar, TaskbarState state);
|
||||||
|
|
||||||
|
// Updates the visibility of each taskbar when the current desktop changes.
|
||||||
|
void update_taskbar_visibility(void *p);
|
||||||
|
|
||||||
|
// Sorts the taskbar(s) on which the window is present.
|
||||||
|
void sort_taskbar_for_win(Window win);
|
||||||
|
|
||||||
|
void sort_tasks(Taskbar *taskbar);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -32,130 +32,157 @@
|
|||||||
#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();
|
||||||
|
|
||||||
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];
|
taskbarname_init_fonts();
|
||||||
memcpy(&tskbar->bar_name.area, &panel->g_taskbar.area_name, sizeof(Area));
|
|
||||||
tskbar->bar_name.area.parent = tskbar;
|
GSList *list = get_desktop_names();
|
||||||
|
GSList *l = 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;
|
||||||
if (j == server.desktop)
|
if (j == server.desktop)
|
||||||
tskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_ACTIVE];
|
taskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_ACTIVE];
|
||||||
else
|
else
|
||||||
tskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_NORMAL];
|
taskbar->bar_name.area.bg = panel->g_taskbar.background_name[TASKBAR_NORMAL];
|
||||||
|
|
||||||
// use desktop number if name is missing
|
// use desktop number if name is missing
|
||||||
if (l) {
|
if (l) {
|
||||||
tskbar->bar_name.name = g_strdup(l->data);
|
taskbar->bar_name.name = g_strdup(l->data);
|
||||||
l = l->next;
|
l = l->next;
|
||||||
|
} else {
|
||||||
|
taskbar->bar_name.name = g_strdup_printf("%d", j + 1);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
tskbar->bar_name.name = g_strdup_printf("%d", j+1);
|
|
||||||
|
|
||||||
// append the name at the beginning of taskbar
|
// append the name at the beginning of taskbar
|
||||||
tskbar->area.list = g_slist_append(tskbar->area.list, &tskbar->bar_name);
|
taskbar->area.children = g_list_append(taskbar->area.children, &taskbar->bar_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (l=list ; l ; l = l->next)
|
for (l = list; l; l = l->next)
|
||||||
g_free(l->data);
|
g_free(l->data);
|
||||||
g_slist_free(list);
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean resize_taskbarname(void *obj)
|
||||||
void draw_taskbarname (void *obj, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
Taskbarname *taskbar_name = obj;
|
TaskbarName *taskbar_name = obj;
|
||||||
Taskbar *taskbar = taskbar_name->area.parent;
|
|
||||||
PangoLayout *layout;
|
|
||||||
Color *config_text = (taskbar->desktop == server.desktop) ? &taskbarname_active_font : &taskbarname_font;
|
|
||||||
|
|
||||||
int state = (taskbar->desktop == server.desktop) ? TASKBAR_ACTIVE : TASKBAR_NORMAL;
|
|
||||||
taskbar_name->state_pix[state] = taskbar_name->area.pix;
|
|
||||||
|
|
||||||
// draw content
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int resize_taskbarname(void *obj)
|
|
||||||
{
|
|
||||||
Taskbarname *taskbar_name = obj;
|
|
||||||
Panel *panel = taskbar_name->area.panel;
|
Panel *panel = taskbar_name->area.panel;
|
||||||
int name_height, name_width, name_height_ink;
|
int name_height, name_width, name_height_ink;
|
||||||
int ret = 0;
|
gboolean result = FALSE;
|
||||||
|
|
||||||
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));
|
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);
|
||||||
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
int new_size = name_width + (2* (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
int new_size = name_width + (2 * (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
||||||
if (new_size != taskbar_name->area.width) {
|
if (new_size != taskbar_name->area.width) {
|
||||||
taskbar_name->area.width = new_size;
|
taskbar_name->area.width = new_size;
|
||||||
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
||||||
ret = 1;
|
result = TRUE;
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
int new_size = name_height + (2 * (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
int new_size = name_height + (2 * (taskbar_name->area.paddingxlr + taskbar_name->area.bg->border.width));
|
||||||
if (new_size != taskbar_name->area.height) {
|
if (new_size != taskbar_name->area.height) {
|
||||||
taskbar_name->area.height = new_size;
|
taskbar_name->area.height = new_size;
|
||||||
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
taskbar_name->posy = (taskbar_name->area.height - name_height) / 2;
|
||||||
ret = 1;
|
result = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ret;
|
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
|
||||||
|
|
||||||
|
|||||||
1989
src/tint.c
1989
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
|
||||||
|
../config.c
|
||||||
|
../server.c
|
||||||
|
../launcher/apps-common.c
|
||||||
|
../launcher/icon-theme-common.c
|
||||||
main.c
|
main.c
|
||||||
properties.c
|
properties.c
|
||||||
properties_rw.c
|
properties_rw.c
|
||||||
theme_view.c )
|
theme_view.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 )
|
if ( NOT DATADIR )
|
||||||
set( DATADIR share )
|
set(DATADIR share)
|
||||||
endif( NOT DATADIR )
|
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_PREFIX}/${DATADIR}/locale\" )
|
||||||
|
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 ${DATADIR}/icons/hicolor/scalable/apps )
|
||||||
install( FILES taskbar.svg DESTINATION ${DATADIR}/icons/hicolor/scalable/apps )
|
|
||||||
install( FILES tint2conf.desktop DESTINATION ${DATADIR}/applications )
|
install( FILES tint2conf.desktop DESTINATION ${DATADIR}/applications )
|
||||||
install( CODE "execute_process(COMMAND gtk-update-icon-cache -f -t ${DATADIR}/icons/hicolor WORKING_DIRECTORY ${CMAKE_INSTALL_PREFIX})" )
|
install( CODE "execute_process(COMMAND gtk-update-icon-cache -f -t ${DATADIR}/icons/hicolor WORKING_DIRECTORY ${CMAKE_INSTALL_PREFIX})" )
|
||||||
|
|||||||
@@ -17,68 +17,87 @@
|
|||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
**************************************************************************/
|
**************************************************************************/
|
||||||
|
|
||||||
|
#include <time.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
#ifdef HAVE_VERSION_H
|
#ifdef HAVE_VERSION_H
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "main.h"
|
#include "main.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "theme_view.h"
|
#include "theme_view.h"
|
||||||
#include "properties.h"
|
#include "properties.h"
|
||||||
#include "properties_rw.h"
|
#include "properties_rw.h"
|
||||||
|
|
||||||
#define SNAPSHOT_TICK 190
|
|
||||||
|
|
||||||
|
// ====== Utilities ======
|
||||||
|
|
||||||
// default config file and directory
|
gchar *get_default_config_path()
|
||||||
char *g_path_config = NULL;
|
{
|
||||||
char *g_path_dir = NULL;
|
gchar *path = NULL;
|
||||||
char *g_default_theme = NULL;
|
const gchar * const * system_dirs = g_get_system_config_dirs();
|
||||||
char *g_cmd_property = NULL;
|
int i;
|
||||||
int g_width, g_height;
|
for (i = 0; system_dirs[i]; i++) {
|
||||||
|
path = g_build_filename(system_dirs[i], "tint2", "tint2rc", NULL);
|
||||||
|
if (g_file_test(path, G_FILE_TEST_EXISTS))
|
||||||
|
return path;
|
||||||
|
g_free(path);
|
||||||
|
path = NULL;
|
||||||
|
}
|
||||||
|
return g_strdup("/dev/null");
|
||||||
|
}
|
||||||
|
|
||||||
GtkWidget *g_window;
|
int endswith(const char *str, const char *suffix)
|
||||||
|
{
|
||||||
|
return strlen(str) >= strlen(suffix) &&
|
||||||
|
strcmp(str + strlen(str) - strlen(suffix), suffix) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
static GtkUIManager *globalUIManager = NULL;
|
static void menuAddWidget(GtkUIManager *ui_manager, GtkWidget *p_widget, GtkContainer *p_box)
|
||||||
|
{
|
||||||
|
gtk_box_pack_start(GTK_BOX(p_box), p_widget, FALSE, FALSE, 0);
|
||||||
|
gtk_widget_show(p_widget);
|
||||||
|
}
|
||||||
|
|
||||||
static void menuAddWidget (GtkUIManager *, GtkWidget *, GtkContainer *);
|
static void menuAddWidget(GtkUIManager *, GtkWidget *, GtkContainer *);
|
||||||
|
static void menuImport();
|
||||||
// action on menus
|
static void menuImportDefault();
|
||||||
static void menuAdd();
|
|
||||||
static void menuSaveAs();
|
static void menuSaveAs();
|
||||||
static void menuDelete();
|
static void menuDelete();
|
||||||
static void menuProperties();
|
static void edit_current_theme();
|
||||||
static void menuQuit();
|
static void set_current_theme();
|
||||||
static void menuRefresh();
|
static void refresh_current_theme();
|
||||||
static void menuRefreshAll();
|
|
||||||
static void menuApply();
|
|
||||||
static void menuAbout();
|
static void menuAbout();
|
||||||
|
static gboolean view_onPopupMenu(GtkWidget *treeview, gpointer userdata);
|
||||||
static gboolean view_onPopupMenu (GtkWidget *treeview, gpointer userdata);
|
static gboolean view_onButtonPressed(GtkWidget *treeview, GdkEventButton *event, gpointer userdata);
|
||||||
static gboolean view_onButtonPressed (GtkWidget *treeview, GdkEventButton *event, gpointer userdata);
|
|
||||||
static void windowSizeAllocated();
|
|
||||||
static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data);
|
static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data);
|
||||||
|
static gboolean theme_selected(GtkTreeSelection *selection,
|
||||||
|
GtkTreeModel *model,
|
||||||
|
GtkTreePath *path,
|
||||||
|
gboolean path_currently_selected,
|
||||||
|
gpointer userdata);
|
||||||
|
|
||||||
|
static void select_first_theme();
|
||||||
|
static void load_all_themes();
|
||||||
|
|
||||||
|
|
||||||
// theme files
|
// ====== Globals ======
|
||||||
static void selectTheme(const gchar *name);
|
|
||||||
static gboolean searchTheme(const gchar *name_theme, GtkTreeModel *model, GtkTreeIter *iter);
|
|
||||||
static void load_theme();
|
|
||||||
static void initTheme();
|
|
||||||
static void read_config();
|
|
||||||
static void write_config();
|
|
||||||
|
|
||||||
|
GtkWidget *g_window;
|
||||||
|
static GtkUIManager *globalUIManager = NULL;
|
||||||
|
GtkWidget *tint_cmd;
|
||||||
|
|
||||||
// define menubar, toolbar and popup
|
|
||||||
static const char *global_ui =
|
static const char *global_ui =
|
||||||
"<ui>"
|
"<ui>"
|
||||||
" <menubar name='MenuBar'>"
|
" <menubar name='MenuBar'>"
|
||||||
" <menu action='ThemeMenu'>"
|
" <menu action='ThemeMenu'>"
|
||||||
" <menuitem action='ThemeAdd'/>"
|
" <menuitem action='ThemeAdd'/>"
|
||||||
" <menuitem action='ThemeSaveAs'/>"
|
" <menuitem action='ThemeDefault'/>"
|
||||||
" <separator/>"
|
|
||||||
" <menuitem action='ThemeDelete'/>"
|
|
||||||
" <separator/>"
|
" <separator/>"
|
||||||
" <menuitem action='ThemeProperties'/>"
|
" <menuitem action='ThemeProperties'/>"
|
||||||
|
" <menuitem action='ThemeSaveAs'/>"
|
||||||
|
" <menuitem action='ThemeDelete'/>"
|
||||||
" <separator/>"
|
" <separator/>"
|
||||||
" <menuitem action='ThemeQuit'/>"
|
" <menuitem action='ThemeQuit'/>"
|
||||||
" </menu>"
|
" </menu>"
|
||||||
@@ -86,7 +105,6 @@ static const char *global_ui =
|
|||||||
" <menuitem action='EditRefresh'/>"
|
" <menuitem action='EditRefresh'/>"
|
||||||
" <menuitem action='EditRefreshAll'/>"
|
" <menuitem action='EditRefreshAll'/>"
|
||||||
" <separator/>"
|
" <separator/>"
|
||||||
" <menuitem action='EditPreferences'/>"
|
|
||||||
" </menu>"
|
" </menu>"
|
||||||
" <menu action='HelpMenu'>"
|
" <menu action='HelpMenu'>"
|
||||||
" <menuitem action='HelpAbout'/>"
|
" <menuitem action='HelpAbout'/>"
|
||||||
@@ -94,67 +112,100 @@ static const char *global_ui =
|
|||||||
" </menubar>"
|
" </menubar>"
|
||||||
" <toolbar name='ToolBar'>"
|
" <toolbar name='ToolBar'>"
|
||||||
" <toolitem action='ThemeProperties'/>"
|
" <toolitem action='ThemeProperties'/>"
|
||||||
" <toolitem action='ViewApply'/>"
|
" <toolitem action='ThemeSelect'/>"
|
||||||
" </toolbar>"
|
" </toolbar>"
|
||||||
" <popup name='ThemePopup'>"
|
" <popup name='ThemePopup'>"
|
||||||
" <menuitem action='ThemeProperties'/>"
|
" <menuitem action='ThemeProperties'/>"
|
||||||
" <menuitem action='EditRefresh'/>"
|
" <menuitem action='EditRefresh'/>"
|
||||||
" <menuitem action='ViewApply'/>"
|
|
||||||
" <separator/>"
|
" <separator/>"
|
||||||
" <menuitem action='ThemeDelete'/>"
|
" <menuitem action='ThemeDelete'/>"
|
||||||
" </popup>"
|
" </popup>"
|
||||||
"</ui>";
|
"</ui>";
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
// define menubar and toolbar action
|
|
||||||
static GtkActionEntry entries[] = {
|
|
||||||
{"ThemeMenu", NULL, _("Theme"), NULL, NULL, NULL},
|
|
||||||
{"ThemeAdd", GTK_STOCK_ADD, _("_Add..."), "<Control>N", _("Add theme"), G_CALLBACK (menuAdd)},
|
|
||||||
{"ThemeSaveAs", GTK_STOCK_SAVE_AS, _("_Save as..."), NULL, _("Save theme as"), G_CALLBACK (menuSaveAs)},
|
|
||||||
{"ThemeDelete", GTK_STOCK_DELETE, _("_Delete"), NULL, _("Delete theme"), G_CALLBACK (menuDelete)},
|
|
||||||
{"ThemeProperties", GTK_STOCK_PROPERTIES, _("_Properties..."), NULL, _("Show properties"), G_CALLBACK (menuProperties)},
|
|
||||||
{"ThemeQuit", GTK_STOCK_QUIT, _("_Quit"), "<control>Q", _("Quit"), G_CALLBACK (menuQuit)},
|
|
||||||
{"EditMenu", NULL, "Edit", NULL, NULL, NULL},
|
|
||||||
{"EditRefresh", GTK_STOCK_REFRESH, _("Refresh"), NULL, _("Refresh"), G_CALLBACK (menuRefresh)},
|
|
||||||
{"EditRefreshAll", GTK_STOCK_REFRESH, _("Refresh all"), NULL, _("Refresh all"), G_CALLBACK (menuRefreshAll)},
|
|
||||||
// {"EditPreferences", GTK_STOCK_PREFERENCES, "Preferences", NULL, "Preferences", G_CALLBACK (menuPreferences)},
|
|
||||||
{"ViewApply", GTK_STOCK_APPLY, _("Apply"), NULL, _("Apply theme"), G_CALLBACK (menuApply)},
|
|
||||||
{"HelpMenu", NULL, _("Help"), NULL, NULL, NULL},
|
|
||||||
{"HelpAbout", GTK_STOCK_ABOUT, _("_About"), "<Control>A", _("About"), G_CALLBACK (menuAbout)}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
int main (int argc, char ** argv)
|
|
||||||
{
|
{
|
||||||
|
setlocale(LC_ALL, "");
|
||||||
|
bindtextdomain(GETTEXT_PACKAGE, LOCALEDIR);
|
||||||
|
bind_textdomain_codeset(GETTEXT_PACKAGE, "UTF-8");
|
||||||
|
textdomain(GETTEXT_PACKAGE);
|
||||||
|
|
||||||
GtkWidget *vBox = NULL, *scrollbar = NULL;
|
GtkWidget *vBox = NULL, *scrollbar = NULL;
|
||||||
GtkActionGroup *actionGroup;
|
GtkActionGroup *actionGroup;
|
||||||
|
|
||||||
gtk_init (&argc, &argv);
|
gtk_init(&argc, &argv);
|
||||||
g_thread_init( NULL );
|
|
||||||
read_config();
|
#if !GLIB_CHECK_VERSION(2, 31, 0)
|
||||||
initTheme();
|
g_thread_init(NULL);
|
||||||
g_set_application_name (_("tint2conf"));
|
#endif
|
||||||
|
|
||||||
|
{
|
||||||
|
gchar *tint2_config_dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL);
|
||||||
|
if (!g_file_test(tint2_config_dir, G_FILE_TEST_IS_DIR))
|
||||||
|
g_mkdir(tint2_config_dir, 0777);
|
||||||
|
g_free(tint2_config_dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_set_application_name(_("tint2conf"));
|
||||||
gtk_window_set_default_icon_name("taskbar");
|
gtk_window_set_default_icon_name("taskbar");
|
||||||
|
|
||||||
// config file use '.' as decimal separator
|
// config file uses '.' as decimal separator
|
||||||
setlocale(LC_NUMERIC, "POSIX");
|
setlocale(LC_NUMERIC, "POSIX");
|
||||||
|
|
||||||
// define main layout : container, menubar, toolbar
|
// define main layout : container, menubar, toolbar
|
||||||
g_window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
g_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
||||||
gtk_window_set_title(GTK_WINDOW(g_window), _("Panel theming"));
|
gtk_window_set_title(GTK_WINDOW(g_window), _("Panel theming"));
|
||||||
gtk_window_resize(GTK_WINDOW(g_window), g_width, g_height);
|
gtk_window_resize(GTK_WINDOW(g_window), 800, 600);
|
||||||
g_signal_connect(G_OBJECT(g_window), "destroy", G_CALLBACK (menuQuit), NULL);
|
g_signal_connect(G_OBJECT(g_window), "destroy", G_CALLBACK(gtk_main_quit), NULL);
|
||||||
g_signal_connect(g_window, "size-allocate", G_CALLBACK(windowSizeAllocated), NULL);
|
vBox = gtk_vbox_new(FALSE, 0);
|
||||||
vBox = gtk_vbox_new (FALSE, 0);
|
gtk_container_add(GTK_CONTAINER(g_window), vBox);
|
||||||
gtk_container_add (GTK_CONTAINER(g_window), vBox);
|
|
||||||
|
|
||||||
actionGroup = gtk_action_group_new ("menuActionGroup");
|
actionGroup = gtk_action_group_new("menuActionGroup");
|
||||||
gtk_action_group_add_actions (actionGroup, entries, G_N_ELEMENTS (entries), NULL);
|
|
||||||
|
// Menubar and toolbar entries
|
||||||
|
GtkActionEntry entries[] = {
|
||||||
|
{"ThemeMenu", NULL, _("Theme"), NULL, NULL, NULL},
|
||||||
|
{"ThemeAdd", GTK_STOCK_ADD, _("_Import theme..."), "<Control>N", _("Import theme"), G_CALLBACK(menuImport)},
|
||||||
|
{"ThemeDefault", GTK_STOCK_NEW, _("_Import default theme..."), NULL, _("Import default theme"), G_CALLBACK(menuImportDefault)},
|
||||||
|
{"ThemeSaveAs", GTK_STOCK_SAVE_AS, _("_Save as..."), NULL, _("Save theme as"), G_CALLBACK(menuSaveAs)},
|
||||||
|
{"ThemeDelete", GTK_STOCK_DELETE, _("_Delete"), NULL, _("Delete theme"), G_CALLBACK(menuDelete)},
|
||||||
|
{"ThemeProperties", GTK_STOCK_PROPERTIES, _("_Edit theme..."), NULL, _("Edit selected theme"), G_CALLBACK(edit_current_theme)},
|
||||||
|
{"ThemeSelect", GTK_STOCK_APPLY, _("_Make default"), NULL, _("Replace the default theme with the selected one"), G_CALLBACK(set_current_theme)},
|
||||||
|
{"ThemeQuit", GTK_STOCK_QUIT, _("_Quit"), "<control>Q", _("Quit"), G_CALLBACK(gtk_main_quit)},
|
||||||
|
{"EditMenu", NULL, _("Edit"), NULL, NULL, NULL},
|
||||||
|
{"EditRefresh", GTK_STOCK_REFRESH, _("Refresh"), NULL, _("Refresh"), G_CALLBACK(refresh_current_theme)},
|
||||||
|
{"EditRefreshAll", GTK_STOCK_REFRESH, _("Refresh all"), NULL, _("Refresh all"), G_CALLBACK(load_all_themes)},
|
||||||
|
{"HelpMenu", NULL, _("Help"), NULL, NULL, NULL},
|
||||||
|
{"HelpAbout", GTK_STOCK_ABOUT, _("_About"), "<Control>A", _("About"), G_CALLBACK(menuAbout)}
|
||||||
|
};
|
||||||
|
|
||||||
|
gtk_action_group_add_actions(actionGroup, entries, G_N_ELEMENTS(entries), NULL);
|
||||||
globalUIManager = gtk_ui_manager_new();
|
globalUIManager = gtk_ui_manager_new();
|
||||||
gtk_ui_manager_insert_action_group (globalUIManager, actionGroup, 0);
|
gtk_ui_manager_insert_action_group(globalUIManager, actionGroup, 0);
|
||||||
gtk_ui_manager_add_ui_from_string (globalUIManager, global_ui, -1, NULL );
|
gtk_ui_manager_add_ui_from_string(globalUIManager, global_ui, -1, (NULL));
|
||||||
g_signal_connect(globalUIManager, "add_widget", G_CALLBACK (menuAddWidget), vBox);
|
g_signal_connect(globalUIManager, "add_widget", G_CALLBACK(menuAddWidget), vBox);
|
||||||
gtk_ui_manager_ensure_update(globalUIManager);
|
gtk_ui_manager_ensure_update(globalUIManager);
|
||||||
|
|
||||||
|
GtkWidget *table, *label;
|
||||||
|
int row, col;
|
||||||
|
|
||||||
|
row = col = 0;
|
||||||
|
table = gtk_table_new(1, 2, FALSE);
|
||||||
|
gtk_widget_show(table);
|
||||||
|
gtk_box_pack_start(GTK_BOX(vBox), table, FALSE, TRUE, 0);
|
||||||
|
gtk_table_set_row_spacings(GTK_TABLE(table), 8);
|
||||||
|
gtk_table_set_col_spacings(GTK_TABLE(table), 8);
|
||||||
|
|
||||||
|
label = gtk_label_new(_("Command to run tint2: "));
|
||||||
|
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++;
|
||||||
|
|
||||||
|
tint_cmd = gtk_entry_new();
|
||||||
|
gtk_widget_show(tint_cmd);
|
||||||
|
gtk_entry_set_text(GTK_ENTRY(tint_cmd), "");
|
||||||
|
gtk_table_attach(GTK_TABLE(table), tint_cmd, col, col+1, row, row+1, GTK_FILL | GTK_EXPAND, 0, 0, 0);
|
||||||
|
|
||||||
scrollbar = gtk_scrolled_window_new(NULL, NULL);
|
scrollbar = gtk_scrolled_window_new(NULL, NULL);
|
||||||
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollbar), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollbar), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
||||||
gtk_box_pack_start(GTK_BOX(vBox), scrollbar, TRUE, TRUE, 0);
|
gtk_box_pack_start(GTK_BOX(vBox), scrollbar, TRUE, TRUE, 0);
|
||||||
@@ -166,107 +217,101 @@ int main (int argc, char ** argv)
|
|||||||
g_signal_connect(g_theme_view, "button-press-event", (GCallback)view_onButtonPressed, NULL);
|
g_signal_connect(g_theme_view, "button-press-event", (GCallback)view_onButtonPressed, NULL);
|
||||||
g_signal_connect(g_theme_view, "popup-menu", (GCallback)view_onPopupMenu, NULL);
|
g_signal_connect(g_theme_view, "popup-menu", (GCallback)view_onPopupMenu, NULL);
|
||||||
g_signal_connect(g_theme_view, "row-activated", G_CALLBACK(viewRowActivated), NULL);
|
g_signal_connect(g_theme_view, "row-activated", G_CALLBACK(viewRowActivated), NULL);
|
||||||
|
gtk_tree_selection_set_select_function(gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view)), theme_selected, NULL, NULL);
|
||||||
|
|
||||||
// load themes
|
// load themes
|
||||||
load_theme(g_theme_view);
|
load_all_themes();
|
||||||
|
|
||||||
gtk_widget_show_all(g_window);
|
gtk_widget_show_all(g_window);
|
||||||
gtk_main ();
|
gtk_main();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void menuAddWidget (GtkUIManager * p_uiManager, GtkWidget * p_widget, GtkContainer * p_box)
|
|
||||||
{
|
|
||||||
gtk_box_pack_start(GTK_BOX(p_box), p_widget, FALSE, FALSE, 0);
|
|
||||||
gtk_widget_show(p_widget);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void menuAbout()
|
static void menuAbout()
|
||||||
{
|
{
|
||||||
const char *authors[] = { "Thierry Lorthiois <lorthiois@bbsoft.fr>", "Andreas Fink <andreas.fink85@googlemail.com>", "Christian Ruppert <Spooky85@gmail.com> (Build system)", "Euan Freeman <euan04@gmail.com> (tintwizard)\n See http://code.google.com/p/tintwizard/", NULL };
|
const char *authors[] = {
|
||||||
|
"Thierry Lorthiois <lorthiois@bbsoft.fr>",
|
||||||
|
"Andreas Fink <andreas.fink85@googlemail.com>",
|
||||||
|
"Christian Ruppert <Spooky85@gmail.com> (Build system)",
|
||||||
|
"Euan Freeman <euan04@gmail.com> (tintwizard http://code.google.com/p/tintwizard)",
|
||||||
|
(NULL)
|
||||||
|
};
|
||||||
|
|
||||||
gtk_show_about_dialog(GTK_WINDOW(g_window), "name", g_get_application_name( ),
|
gtk_show_about_dialog(GTK_WINDOW(g_window),
|
||||||
"comments", _("Theming tool for tint2 panel"),
|
"name", g_get_application_name( ),
|
||||||
"version", VERSION_STRING,
|
"comments", _("Theming tool for tint2 panel"),
|
||||||
"copyright", _("Copyright 2009 tint2 team\nTint2 License GNU GPL version 2\nTintwizard License GNU GPL version 3"),
|
"version", VERSION_STRING,
|
||||||
"logo-icon-name", "taskbar", "authors", authors,
|
"copyright", _("Copyright 2009-2015 tint2 team\nTint2 License GNU GPL version 2\nTintwizard License GNU GPL version 3"),
|
||||||
/* Translators: translate "translator-credits" as
|
"logo-icon-name", "taskbar", "authors", authors,
|
||||||
your name to have it appear in the credits in the "About"
|
/* Translators: translate "translator-credits" as
|
||||||
dialog */
|
your name to have it appear in the credits in the "About"
|
||||||
"translator-credits", _("translator-credits"),
|
dialog */
|
||||||
NULL);
|
"translator-credits", _("translator-credits"),
|
||||||
|
NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void menuAdd()
|
// ====== Theme import/copy/delete ======
|
||||||
|
|
||||||
|
static void menuImport()
|
||||||
{
|
{
|
||||||
GtkWidget *dialog;
|
GtkWidget *dialog = gtk_file_chooser_dialog_new(_("Import theme(s)"), GTK_WINDOW(g_window), GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT, NULL);
|
||||||
GtkFileChooser *chooser;
|
GtkFileChooser *chooser = GTK_FILE_CHOOSER(dialog);
|
||||||
GtkFileFilter *filter;
|
|
||||||
|
|
||||||
dialog = gtk_file_chooser_dialog_new(_("Add a theme"), GTK_WINDOW(g_window), GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT, NULL);
|
|
||||||
chooser = GTK_FILE_CHOOSER(dialog);
|
|
||||||
|
|
||||||
gtk_file_chooser_set_current_folder(chooser, g_get_home_dir());
|
|
||||||
gtk_file_chooser_set_select_multiple(chooser, TRUE);
|
gtk_file_chooser_set_select_multiple(chooser, TRUE);
|
||||||
|
|
||||||
filter = gtk_file_filter_new();
|
if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
|
||||||
gtk_file_filter_set_name(filter, _("Tint2 theme files"));
|
|
||||||
gtk_file_filter_add_pattern(filter, "*.tint2rc");
|
|
||||||
gtk_file_chooser_add_filter(chooser, filter);
|
|
||||||
|
|
||||||
if (gtk_dialog_run (GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
|
|
||||||
gtk_widget_destroy(dialog);
|
gtk_widget_destroy(dialog);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
GtkTreeIter iter;
|
GSList *list = gtk_file_chooser_get_filenames(chooser);
|
||||||
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
GSList *l;
|
||||||
GSList *l, *list = gtk_file_chooser_get_filenames(chooser);
|
|
||||||
gchar *file, *pt1, *name, *path, *name_first=NULL;
|
|
||||||
for (l = list; l ; l = l->next) {
|
for (l = list; l ; l = l->next) {
|
||||||
file = (char *)l->data;
|
gchar *file = (char *)l->data;
|
||||||
pt1 = strrchr(file, '/');
|
gchar *pt1 = strrchr(file, '/');
|
||||||
if (pt1 == NULL) continue;
|
if (!pt1)
|
||||||
pt1++;
|
continue;
|
||||||
if (*pt1 == 0) continue;
|
pt1++;
|
||||||
|
if (!*pt1)
|
||||||
name = g_strdup(pt1);
|
|
||||||
path = g_build_filename (g_get_user_config_dir(), "tint2", name, NULL);
|
|
||||||
|
|
||||||
// check existing
|
|
||||||
if (searchTheme(path, model, &iter)) {
|
|
||||||
gchar *message;
|
|
||||||
message = g_strdup_printf(_("Couldn't add duplicate theme\n\'%s\'."), pt1);
|
|
||||||
|
|
||||||
GtkWidget *w = gtk_message_dialog_new(GTK_WINDOW(g_window), 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, message, NULL);
|
|
||||||
g_signal_connect_swapped(w, "response", G_CALLBACK(gtk_widget_destroy), w);
|
|
||||||
gtk_widget_show(w);
|
|
||||||
g_free(message);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
|
||||||
|
|
||||||
// append theme
|
gchar *name = pt1;
|
||||||
|
gchar *path = g_build_filename(g_get_user_config_dir(), "tint2", name, NULL);
|
||||||
|
if (g_file_test(path, G_FILE_TEST_EXISTS))
|
||||||
|
continue;
|
||||||
copy_file(file, path);
|
copy_file(file, path);
|
||||||
custom_list_append(path);
|
|
||||||
if (name_first == NULL)
|
|
||||||
name_first = g_strdup(path);
|
|
||||||
g_free(path);
|
g_free(path);
|
||||||
g_free(name);
|
|
||||||
}
|
}
|
||||||
g_slist_foreach(list, (GFunc)g_free, NULL);
|
g_slist_foreach(list, (GFunc)g_free, NULL);
|
||||||
g_slist_free(list);
|
g_slist_free(list);
|
||||||
gtk_widget_destroy(dialog);
|
gtk_widget_destroy(dialog);
|
||||||
|
load_all_themes();
|
||||||
selectTheme(name_first);
|
|
||||||
g_free(name_first);
|
|
||||||
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void menuImportDefault()
|
||||||
|
{
|
||||||
|
GtkWidget *dialog = gtk_file_chooser_dialog_new(_("Save default theme as"), GTK_WINDOW(g_window), GTK_FILE_CHOOSER_ACTION_SAVE, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT, NULL);
|
||||||
|
GtkFileChooser *chooser = GTK_FILE_CHOOSER(dialog);
|
||||||
|
|
||||||
static void menuSaveAs ()
|
gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE);
|
||||||
|
gchar *config_dir;
|
||||||
|
config_dir = g_build_filename(g_get_home_dir(), ".config", "tint2", NULL);
|
||||||
|
gtk_file_chooser_set_current_folder(chooser, config_dir);
|
||||||
|
g_free(config_dir);
|
||||||
|
|
||||||
|
if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
|
||||||
|
gchar *save_name = gtk_file_chooser_get_filename(chooser);
|
||||||
|
gchar *path_default = get_default_config_path();
|
||||||
|
copy_file(path_default, save_name);
|
||||||
|
g_free(path_default);
|
||||||
|
g_free(save_name);
|
||||||
|
}
|
||||||
|
gtk_widget_destroy(dialog);
|
||||||
|
|
||||||
|
load_all_themes();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void menuSaveAs()
|
||||||
{
|
{
|
||||||
GtkWidget *dialog;
|
GtkWidget *dialog;
|
||||||
GtkFileChooser *chooser;
|
GtkFileChooser *chooser;
|
||||||
@@ -284,25 +329,29 @@ static void menuSaveAs ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
||||||
pt1 = strrchr (file, '/');
|
pt1 = strrchr(file, '/');
|
||||||
if (pt1) pt1++;
|
if (pt1) pt1++;
|
||||||
|
|
||||||
dialog = gtk_file_chooser_dialog_new(_("Save theme as"), GTK_WINDOW(g_window), GTK_FILE_CHOOSER_ACTION_SAVE, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT, NULL);
|
dialog = gtk_file_chooser_dialog_new(_("Save theme as"), GTK_WINDOW(g_window), GTK_FILE_CHOOSER_ACTION_SAVE, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT, NULL);
|
||||||
chooser = GTK_FILE_CHOOSER(dialog);
|
chooser = GTK_FILE_CHOOSER(dialog);
|
||||||
|
|
||||||
gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE);
|
gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE);
|
||||||
gtk_file_chooser_set_current_folder(chooser, g_get_home_dir());
|
gchar *config_dir;
|
||||||
|
config_dir = g_build_filename(g_get_home_dir(), ".config", "tint2", NULL);
|
||||||
|
gtk_file_chooser_set_current_folder(chooser, config_dir);
|
||||||
|
g_free(config_dir);
|
||||||
gtk_file_chooser_set_current_name(chooser, pt1);
|
gtk_file_chooser_set_current_name(chooser, pt1);
|
||||||
|
|
||||||
if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
|
if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
|
||||||
char *filename = gtk_file_chooser_get_filename(chooser);
|
char *filename = gtk_file_chooser_get_filename(chooser);
|
||||||
copy_file(file, filename);
|
copy_file(file, filename);
|
||||||
g_free (filename);
|
g_free(filename);
|
||||||
}
|
}
|
||||||
g_free(file);
|
g_free(file);
|
||||||
gtk_widget_destroy (dialog);
|
gtk_widget_destroy(dialog);
|
||||||
}
|
|
||||||
|
|
||||||
|
load_all_themes();
|
||||||
|
}
|
||||||
|
|
||||||
static void menuDelete()
|
static void menuDelete()
|
||||||
{
|
{
|
||||||
@@ -326,105 +375,7 @@ static void menuDelete()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void menuProperties()
|
// ====== Theme popup menu ======
|
||||||
{
|
|
||||||
GtkTreeSelection *sel;
|
|
||||||
GtkTreeIter iter;
|
|
||||||
GtkTreeModel *model;
|
|
||||||
char *file;
|
|
||||||
|
|
||||||
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
|
||||||
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
|
||||||
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
|
||||||
//*
|
|
||||||
GtkWidget *prop;
|
|
||||||
prop = create_properties();
|
|
||||||
config_read_file(file);
|
|
||||||
gtk_window_present(GTK_WINDOW(prop));
|
|
||||||
//printf("menuProperties : fin\n");
|
|
||||||
//*/
|
|
||||||
/*
|
|
||||||
char *cmd = g_strdup_printf("%s \'%s\' &", g_cmd_property, file);
|
|
||||||
printf("cmd %s\n", cmd);
|
|
||||||
system(cmd);
|
|
||||||
g_free(cmd);
|
|
||||||
//*/
|
|
||||||
g_free(file);
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void menuQuit()
|
|
||||||
{
|
|
||||||
write_config();
|
|
||||||
|
|
||||||
if (g_path_config)
|
|
||||||
g_free(g_path_config);
|
|
||||||
if (g_path_dir)
|
|
||||||
g_free(g_path_dir);
|
|
||||||
if (g_default_theme)
|
|
||||||
g_free(g_default_theme);
|
|
||||||
if (g_cmd_property)
|
|
||||||
g_free(g_cmd_property);
|
|
||||||
|
|
||||||
gtk_main_quit ();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void menuRefresh()
|
|
||||||
{
|
|
||||||
GtkTreeSelection *sel;
|
|
||||||
GtkTreeIter iter;
|
|
||||||
GtkTreeModel *model;
|
|
||||||
|
|
||||||
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
|
||||||
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
|
||||||
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, NULL, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void menuRefreshAll()
|
|
||||||
{
|
|
||||||
GtkTreeIter iter;
|
|
||||||
GtkTreeModel *model;
|
|
||||||
gboolean have_iter;
|
|
||||||
|
|
||||||
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
|
||||||
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
|
||||||
while (have_iter) {
|
|
||||||
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, NULL, -1);
|
|
||||||
have_iter = gtk_tree_model_iter_next(model, &iter);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void menuApply()
|
|
||||||
{
|
|
||||||
GtkTreeSelection *sel;
|
|
||||||
GtkTreeIter iter;
|
|
||||||
GtkTreeModel *model;
|
|
||||||
|
|
||||||
if (g_default_theme) {
|
|
||||||
g_free(g_default_theme);
|
|
||||||
g_default_theme = NULL;
|
|
||||||
}
|
|
||||||
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
|
||||||
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
|
||||||
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &g_default_theme, -1);
|
|
||||||
// overwrite tint2rc
|
|
||||||
copy_file(g_default_theme, g_path_config);
|
|
||||||
|
|
||||||
// restart panel
|
|
||||||
system("killall -SIGUSR1 tint2");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void view_popup_menu(GtkWidget *treeview, GdkEventButton *event, gpointer userdata)
|
static void view_popup_menu(GtkWidget *treeview, GdkEventButton *event, gpointer userdata)
|
||||||
{
|
{
|
||||||
@@ -433,8 +384,7 @@ static void view_popup_menu(GtkWidget *treeview, GdkEventButton *event, gpointer
|
|||||||
gtk_menu_popup(GTK_MENU(w), NULL, NULL, NULL, NULL, (event != NULL) ? event->button : 0, gdk_event_get_time((GdkEvent*)event));
|
gtk_menu_popup(GTK_MENU(w), NULL, NULL, NULL, NULL, (event != NULL) ? event->button : 0, gdk_event_get_time((GdkEvent*)event));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean view_onButtonPressed(GtkWidget *treeview, GdkEventButton *event, gpointer userdata)
|
||||||
static gboolean view_onButtonPressed (GtkWidget *treeview, GdkEventButton *event, gpointer userdata)
|
|
||||||
{
|
{
|
||||||
// single click with the right mouse button?
|
// single click with the right mouse button?
|
||||||
if (event->type == GDK_BUTTON_PRESS && event->button == 3) {
|
if (event->type == GDK_BUTTON_PRESS && event->button == 3) {
|
||||||
@@ -459,186 +409,232 @@ static gboolean view_onButtonPressed (GtkWidget *treeview, GdkEventButton *event
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean view_onPopupMenu(GtkWidget *treeview, gpointer userdata)
|
||||||
static gboolean view_onPopupMenu (GtkWidget *treeview, gpointer userdata)
|
|
||||||
{
|
{
|
||||||
view_popup_menu(treeview, NULL, userdata);
|
view_popup_menu(treeview, NULL, userdata);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data)
|
// ====== Theme selection ======
|
||||||
|
|
||||||
|
gboolean theme_selected(GtkTreeSelection *selection,
|
||||||
|
GtkTreeModel *model,
|
||||||
|
GtkTreePath *path,
|
||||||
|
gboolean path_currently_selected,
|
||||||
|
gpointer userdata)
|
||||||
{
|
{
|
||||||
menuApply();
|
GtkTreeIter iter;
|
||||||
}
|
if (gtk_tree_model_get_iter(model, &iter, path)) {
|
||||||
|
gchar *current_theme = NULL;
|
||||||
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, ¤t_theme, -1);
|
||||||
static void windowSizeAllocated()
|
if (!path_currently_selected) {
|
||||||
{
|
gchar *text = g_strdup_printf("tint2 -c %s", current_theme);
|
||||||
const gboolean isMaximized = g_window->window && (gdk_window_get_state(g_window->window) & GDK_WINDOW_STATE_MAXIMIZED);
|
gtk_entry_set_text(GTK_ENTRY(tint_cmd), text);
|
||||||
|
g_free(text);
|
||||||
if(!isMaximized)
|
} else {
|
||||||
gtk_window_get_size(GTK_WINDOW(g_window), &g_width, &g_height);
|
gtk_entry_set_text(GTK_ENTRY(tint_cmd), "");
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void load_theme(GtkWidget *list)
|
|
||||||
{
|
|
||||||
GDir *dir;
|
|
||||||
gchar *pt1, *name;
|
|
||||||
const gchar *file;
|
|
||||||
gboolean found_theme = FALSE;
|
|
||||||
|
|
||||||
dir = g_dir_open(g_path_dir, 0, NULL);
|
|
||||||
if (dir == NULL) return;
|
|
||||||
while ((file = g_dir_read_name(dir))) {
|
|
||||||
pt1 = strstr(file, ".tint2rc");
|
|
||||||
if (pt1) {
|
|
||||||
found_theme = TRUE;
|
|
||||||
name = g_build_filename (g_path_dir, file, NULL);
|
|
||||||
custom_list_append(name);
|
|
||||||
g_free(name);
|
|
||||||
}
|
}
|
||||||
|
g_free(current_theme);
|
||||||
}
|
}
|
||||||
g_dir_close(dir);
|
return TRUE;
|
||||||
|
|
||||||
if (!found_theme) {
|
|
||||||
// create default theme file
|
|
||||||
name = g_build_filename(g_get_user_config_dir(), "tint2", "default.tint2rc", NULL);
|
|
||||||
copy_file(g_path_config, name);
|
|
||||||
custom_list_append(name);
|
|
||||||
if (g_default_theme) g_free(g_default_theme);
|
|
||||||
g_default_theme = strdup(name);
|
|
||||||
g_free(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
selectTheme(g_default_theme);
|
|
||||||
|
|
||||||
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void select_first_theme()
|
||||||
void selectTheme(const gchar *name_theme)
|
|
||||||
{
|
{
|
||||||
gboolean have_iter, found_theme;
|
gboolean have_iter;
|
||||||
GtkTreeIter iter;
|
GtkTreeIter iter;
|
||||||
GtkTreeModel *model;
|
GtkTreeModel *model;
|
||||||
|
|
||||||
if (!name_theme) return;
|
|
||||||
|
|
||||||
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
||||||
found_theme = searchTheme(name_theme, model, &iter);
|
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
||||||
|
if (have_iter) {
|
||||||
GtkTreePath *path = NULL;
|
GtkTreePath *path = gtk_tree_model_get_path(model, &iter);
|
||||||
if (found_theme)
|
|
||||||
path = gtk_tree_model_get_path(model, &iter);
|
|
||||||
else {
|
|
||||||
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
|
||||||
if (have_iter)
|
|
||||||
path = gtk_tree_model_get_path(model, &iter);
|
|
||||||
}
|
|
||||||
if (path) {
|
|
||||||
gtk_tree_selection_select_iter(gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view)), &iter);
|
gtk_tree_selection_select_iter(gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view)), &iter);
|
||||||
gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(g_theme_view), path, NULL, FALSE, 0, 0);
|
gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(g_theme_view), path, NULL, FALSE, 0, 0);
|
||||||
gtk_tree_path_free(path);
|
gtk_tree_path_free(path);
|
||||||
}
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *get_current_theme_file_name()
|
||||||
gboolean searchTheme(const gchar *name_theme, GtkTreeModel *model, GtkTreeIter *iter)
|
|
||||||
{
|
{
|
||||||
gchar *name;
|
GtkTreeSelection *sel;
|
||||||
gboolean have_iter, found = FALSE;
|
GtkTreeIter iter;
|
||||||
|
GtkTreeModel *model;
|
||||||
|
char *file;
|
||||||
|
|
||||||
have_iter = gtk_tree_model_get_iter_first(model, iter);
|
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
||||||
while (have_iter) {
|
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
||||||
gtk_tree_model_get(model, iter, COL_THEME_FILE, &name, -1);
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
||||||
found = (strcmp(name, name_theme) == 0);
|
return file;
|
||||||
g_free(name);
|
|
||||||
if (found)
|
|
||||||
break;
|
|
||||||
have_iter = gtk_tree_model_iter_next(model, iter);
|
|
||||||
}
|
}
|
||||||
return found;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void edit_current_theme()
|
||||||
void initTheme()
|
|
||||||
{
|
{
|
||||||
g_path_dir = g_build_filename (g_get_user_config_dir(), "tint2", NULL);
|
GtkTreeSelection *sel;
|
||||||
if (!g_file_test (g_path_dir, G_FILE_TEST_IS_DIR))
|
GtkTreeIter iter;
|
||||||
g_mkdir(g_path_dir, 0777);
|
GtkTreeModel *model;
|
||||||
|
char *file;
|
||||||
|
|
||||||
g_path_config = g_build_filename (g_get_user_config_dir(), "tint2", "tint2rc", NULL);
|
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
||||||
}
|
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
||||||
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
||||||
|
GtkWidget *prop;
|
||||||
void read_config()
|
prop = create_properties();
|
||||||
{
|
config_read_file(file);
|
||||||
char *path;
|
gtk_window_present(GTK_WINDOW(prop));
|
||||||
|
g_free(file);
|
||||||
// default values
|
|
||||||
if (g_default_theme != NULL) {
|
|
||||||
g_free(g_default_theme);
|
|
||||||
g_default_theme = NULL;
|
|
||||||
}
|
}
|
||||||
g_width = 500;
|
}
|
||||||
g_height = 350;
|
|
||||||
g_cmd_property = g_strconcat( "/usr/bin/env python ", INSTALL_PREFIX, "/bin/tintwizard.py", (void*)0 );
|
|
||||||
|
|
||||||
// load config
|
static void set_current_theme()
|
||||||
path = g_build_filename (g_get_user_config_dir(), "tint2", "tint2confrc", NULL);
|
{
|
||||||
if (g_file_test (path, G_FILE_TEST_EXISTS)) {
|
GtkTreeSelection *sel;
|
||||||
FILE *fp;
|
GtkTreeIter iter;
|
||||||
char line[80];
|
GtkTreeModel *model;
|
||||||
char *key, *value;
|
gchar *file;
|
||||||
if ((fp = fopen(path, "r")) != NULL) {
|
|
||||||
while (fgets(line, sizeof(line), fp) != NULL) {
|
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
||||||
if (parse_line(line, &key, &value)) {
|
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
||||||
if (strcmp (key, "default_theme") == 0)
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &file, -1);
|
||||||
g_default_theme = strdup(value);
|
// config_read_file(file);
|
||||||
else if (strcmp (key, "cmd_property") == 0) {
|
|
||||||
g_free(g_cmd_property);
|
gchar *main_file = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL);
|
||||||
g_cmd_property = strdup(value);
|
{
|
||||||
|
gchar *backup_path = g_strdup_printf("%s.backup.%ld", main_file, time(NULL));
|
||||||
|
copy_file(main_file, backup_path);
|
||||||
|
g_free(backup_path);
|
||||||
|
}
|
||||||
|
copy_file(file, main_file);
|
||||||
|
int unused = system("killall -SIGUSR1 tint2 || pkill -SIGUSR1 -x tint2");
|
||||||
|
(void)unused;
|
||||||
|
g_free(file);
|
||||||
|
select_first_theme();
|
||||||
|
refresh_current_theme();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void viewRowActivated(GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column, gpointer user_data)
|
||||||
|
{
|
||||||
|
edit_current_theme();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ====== Theme load/reload ======
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
static void copy_default_themes()
|
||||||
|
{
|
||||||
|
gchar *path_home = g_build_filename(g_get_user_config_dir(), "tint2", "tint2rc", NULL);
|
||||||
|
if (!g_file_test(path_home, G_FILE_TEST_EXISTS)) {
|
||||||
|
const gchar * const * system_dirs = g_get_system_config_dirs();
|
||||||
|
int i;
|
||||||
|
for (i = 0; system_dirs[i]; i++) {
|
||||||
|
gchar *path = g_build_filename(system_dirs[i], "tint2", "tint2rc", NULL);
|
||||||
|
if (g_file_test(path, G_FILE_TEST_EXISTS)) {
|
||||||
|
copy_file(path, path_home);
|
||||||
|
}
|
||||||
|
g_free(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
g_free(path_home);
|
||||||
|
|
||||||
|
const gchar * const * data_dirs = g_get_system_data_dirs();
|
||||||
|
int i;
|
||||||
|
for (i = 0; data_dirs[i]; i++) {
|
||||||
|
gchar *path_tint2 = g_build_filename(data_dirs[i], "tint2", NULL);
|
||||||
|
fprintf(stderr, "%s\n", path_tint2);
|
||||||
|
GDir *dir = g_dir_open(path_tint2, 0, NULL);
|
||||||
|
if (dir) {
|
||||||
|
const gchar *file_name;
|
||||||
|
while ((file_name = g_dir_read_name(dir))) {
|
||||||
|
if (!g_file_test(file_name, G_FILE_TEST_IS_DIR) &&
|
||||||
|
!strstr(file_name, "backup") &&
|
||||||
|
!strstr(file_name, "copy") &&
|
||||||
|
!strstr(file_name, "~") &&
|
||||||
|
(endswith(file_name, "tint2rc") ||
|
||||||
|
endswith(file_name, ".conf"))) {
|
||||||
|
gchar *path_home = g_build_filename(g_get_user_config_dir(), "tint2", file_name, NULL);
|
||||||
|
if (!g_file_test(path_home, G_FILE_TEST_EXISTS)) {
|
||||||
|
gchar *path_usr = g_build_filename(path_tint2, file_name, NULL);
|
||||||
|
copy_file(path_usr, path_home);
|
||||||
|
g_free(path_usr);
|
||||||
}
|
}
|
||||||
else if (strcmp (key, "width") == 0)
|
g_free(path_home);
|
||||||
g_width = atoi(value);
|
|
||||||
else if (strcmp (key, "height") == 0)
|
|
||||||
g_height = atoi(value);
|
|
||||||
free (key);
|
|
||||||
free (value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fclose (fp);
|
g_dir_close(dir);
|
||||||
}
|
}
|
||||||
|
g_free(path_tint2);
|
||||||
}
|
}
|
||||||
g_free(path);
|
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void load_all_themes()
|
||||||
void write_config()
|
|
||||||
{
|
{
|
||||||
char *path;
|
// We don't do this anymore since it has proven unpopular...
|
||||||
FILE *fp;
|
#if 0
|
||||||
|
copy_default_themes();
|
||||||
|
#endif
|
||||||
|
|
||||||
path = g_build_filename (g_get_user_config_dir(), "tint2", "tint2confrc", NULL);
|
gtk_list_store_clear(GTK_LIST_STORE(g_store));
|
||||||
fp = fopen(path, "w");
|
|
||||||
if (fp != NULL) {
|
gchar *tint2_config_dir = g_build_filename(g_get_user_config_dir(), "tint2", NULL);
|
||||||
fputs("#---------------------------------------------\n", fp);
|
GDir *dir = g_dir_open(tint2_config_dir, 0, NULL);
|
||||||
fputs("# TINT2CONF CONFIG FILE\n", fp);
|
if (dir == NULL) {
|
||||||
if (g_default_theme != NULL) {
|
g_free(tint2_config_dir);
|
||||||
fprintf(fp, "default_theme = %s\n", g_default_theme);
|
return;
|
||||||
}
|
|
||||||
if (g_cmd_property != NULL) {
|
|
||||||
fprintf(fp, "cmd_property = %s\n", g_cmd_property);
|
|
||||||
}
|
|
||||||
fprintf(fp, "width = %d\n", g_width);
|
|
||||||
fprintf(fp, "height = %d\n", g_height);
|
|
||||||
fputs("\n", fp);
|
|
||||||
fclose (fp);
|
|
||||||
}
|
}
|
||||||
g_free(path);
|
gboolean found_theme = FALSE;
|
||||||
|
|
||||||
|
const gchar *file_name;
|
||||||
|
while ((file_name = g_dir_read_name(dir))) {
|
||||||
|
if (!g_file_test(file_name, G_FILE_TEST_IS_DIR) &&
|
||||||
|
!strstr(file_name, "backup") &&
|
||||||
|
!strstr(file_name, "copy") &&
|
||||||
|
!strstr(file_name, "~") &&
|
||||||
|
(endswith(file_name, "tint2rc") ||
|
||||||
|
endswith(file_name, ".conf"))) {
|
||||||
|
found_theme = TRUE;
|
||||||
|
gchar *name = g_build_filename(tint2_config_dir, file_name, NULL);
|
||||||
|
custom_list_append(name);
|
||||||
|
g_free(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found_theme) {
|
||||||
|
select_first_theme();
|
||||||
|
|
||||||
|
GtkTreeIter iter;
|
||||||
|
GtkTreeModel *model;
|
||||||
|
gboolean have_iter;
|
||||||
|
|
||||||
|
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
||||||
|
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
||||||
|
while (have_iter) {
|
||||||
|
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, NULL, -1);
|
||||||
|
have_iter = gtk_tree_model_iter_next(model, &iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_dir_close(dir);
|
||||||
|
g_free(tint2_config_dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void refresh_current_theme()
|
||||||
|
{
|
||||||
|
GtkTreeSelection *sel;
|
||||||
|
GtkTreeIter iter;
|
||||||
|
GtkTreeModel *model;
|
||||||
|
|
||||||
|
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(g_theme_view));
|
||||||
|
if (gtk_tree_selection_get_selected(GTK_TREE_SELECTION(sel), &model, &iter)) {
|
||||||
|
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, NULL, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_timeout_add(SNAPSHOT_TICK, (GSourceFunc)update_snapshot, NULL);
|
||||||
|
}
|
||||||
|
|||||||
@@ -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,9 @@
|
|||||||
#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();
|
||||||
|
void menuApply();
|
||||||
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_PREFIX}/${DATADIR}/locale/${LANG}/LC_MESSAGES"
|
||||||
|
RENAME "${GETTEXT_PACKAGE}.mo")
|
||||||
|
endforeach ()
|
||||||
|
endif()
|
||||||
|
else ()
|
||||||
|
message(STATUS "gettext not found")
|
||||||
|
endif ()
|
||||||
2170
src/tint2conf/po/bs.po
Normal file
2170
src/tint2conf/po/bs.po
Normal file
File diff suppressed because it is too large
Load Diff
2226
src/tint2conf/po/fr.po
Normal file
2226
src/tint2conf/po/fr.po
Normal file
File diff suppressed because it is too large
Load Diff
2170
src/tint2conf/po/hr.po
Normal file
2170
src/tint2conf/po/hr.po
Normal file
File diff suppressed because it is too large
Load Diff
2204
src/tint2conf/po/pl.po
Normal file
2204
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
|
||||||
2169
src/tint2conf/po/sr.po
Normal file
2169
src/tint2conf/po/sr.po
Normal file
File diff suppressed because it is too large
Load Diff
2025
src/tint2conf/po/tint2conf.pot
Normal file
2025
src/tint2conf/po/tint2conf.pot
Normal file
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -3,49 +3,205 @@
|
|||||||
#define PROPERTIES
|
#define PROPERTIES
|
||||||
|
|
||||||
#include <gtk/gtk.h>
|
#include <gtk/gtk.h>
|
||||||
|
#include <gdk/gdk.h>
|
||||||
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
||||||
|
|
||||||
|
#include "../launcher/icon-theme-common.h"
|
||||||
|
|
||||||
|
|
||||||
// panel
|
// panel
|
||||||
GtkWidget *panel_width, *panel_height, *panel_margin_x, *panel_margin_y, *panel_padding_x, *panel_padding_y, *panel_spacing;
|
extern GtkWidget *panel_width, *panel_height, *panel_margin_x, *panel_margin_y, *panel_padding_x, *panel_padding_y, *panel_spacing;
|
||||||
GtkWidget *panel_wm_menu, *panel_dock, *panel_autohide, *panel_autohide_show_time, *panel_autohide_hide_time, *panel_autohide_size;
|
extern GtkWidget *panel_wm_menu, *panel_dock, *panel_autohide, *panel_autohide_show_time, *panel_autohide_hide_time, *panel_autohide_size;
|
||||||
GtkWidget *panel_combo_strut_policy, *panel_combo_layer, *panel_combo_width_type, *panel_combo_height_type, *panel_combo_monitor;
|
extern GtkWidget *panel_combo_strut_policy, *panel_combo_layer, *panel_combo_width_type, *panel_combo_height_type, *panel_combo_monitor;
|
||||||
GtkWidget *items_order;
|
extern GtkWidget *panel_window_name, *disable_transparency;
|
||||||
|
extern GtkWidget *panel_mouse_effects;
|
||||||
|
extern GtkWidget *mouse_hover_icon_opacity, *mouse_hover_icon_saturation, *mouse_hover_icon_brightness;
|
||||||
|
extern GtkWidget *mouse_pressed_icon_opacity, *mouse_pressed_icon_saturation, *mouse_pressed_icon_brightness;
|
||||||
|
extern GtkWidget *panel_primary_monitor_first;
|
||||||
|
|
||||||
|
enum {
|
||||||
|
itemsColName = 0,
|
||||||
|
itemsColValue,
|
||||||
|
itemsNumCols
|
||||||
|
};
|
||||||
|
extern GtkListStore *panel_items, *all_items;
|
||||||
|
extern GtkWidget *panel_items_view, *all_items_view;
|
||||||
|
char *get_panel_items();
|
||||||
|
void set_panel_items(const char *items);
|
||||||
|
|
||||||
|
extern GtkWidget *screen_position[12];
|
||||||
|
extern GSList *screen_position_group;
|
||||||
|
extern GtkWidget *panel_background;
|
||||||
|
|
||||||
|
#define POS_TLH 0
|
||||||
|
#define POS_TCH 1
|
||||||
|
#define POS_TRH 2
|
||||||
|
|
||||||
|
#define POS_TLV 3
|
||||||
|
#define POS_CLV 4
|
||||||
|
#define POS_BLV 5
|
||||||
|
|
||||||
|
#define POS_TRV 6
|
||||||
|
#define POS_CRV 7
|
||||||
|
#define POS_BRV 8
|
||||||
|
|
||||||
|
#define POS_BLH 9
|
||||||
|
#define POS_BCH 10
|
||||||
|
#define POS_BRH 11
|
||||||
|
|
||||||
// taskbar
|
// taskbar
|
||||||
GtkWidget *taskbar_show_desktop, *taskbar_show_name, *taskbar_padding_x, *taskbar_padding_y, *taskbar_spacing;
|
extern GtkWidget *taskbar_show_desktop, *taskbar_show_name, *taskbar_padding_x, *taskbar_padding_y, *taskbar_spacing;
|
||||||
GtkWidget *taskbar_name_padding_x, *taskbar_name_inactive_color, *taskbar_name_active_color, *taskbar_name_font;
|
extern GtkWidget *taskbar_hide_inactive_tasks, *taskbar_hide_diff_monitor;
|
||||||
|
extern GtkWidget *taskbar_name_padding_x, *taskbar_name_padding_y, *taskbar_name_inactive_color, *taskbar_name_active_color;
|
||||||
|
extern GtkWidget *taskbar_name_font, *taskbar_name_font_set;
|
||||||
|
extern GtkWidget *taskbar_active_background, *taskbar_inactive_background;
|
||||||
|
extern GtkWidget *taskbar_name_active_background, *taskbar_name_inactive_background;
|
||||||
|
extern GtkWidget *taskbar_distribute_size, *taskbar_sort_order, *taskbar_alignment, *taskbar_always_show_all_desktop_tasks;
|
||||||
|
|
||||||
// task
|
// task
|
||||||
GtkWidget *task_mouse_middle, *task_mouse_right, *task_mouse_scroll_up, *task_mouse_scroll_down;
|
extern GtkWidget *task_mouse_left, *task_mouse_middle, *task_mouse_right, *task_mouse_scroll_up, *task_mouse_scroll_down;
|
||||||
GtkWidget *task_show_icon, *task_show_text, *task_align_center, *task_font_shadow;
|
extern GtkWidget *task_show_icon, *task_show_text, *task_align_center, *font_shadow;
|
||||||
GtkWidget *task_maximum_width, *task_maximum_height, *task_padding_x, *task_padding_y, *task_font;
|
extern GtkWidget *task_maximum_width, *task_maximum_height, *task_padding_x, *task_padding_y, *task_spacing;
|
||||||
|
extern GtkWidget *task_font, *task_font_set;
|
||||||
|
extern GtkWidget *task_default_color, *task_default_color_set,
|
||||||
|
*task_default_icon_opacity, *task_default_icon_osb_set,
|
||||||
|
*task_default_icon_saturation,
|
||||||
|
*task_default_icon_brightness,
|
||||||
|
*task_default_background, *task_default_background_set;
|
||||||
|
extern GtkWidget *task_normal_color, *task_normal_color_set,
|
||||||
|
*task_normal_icon_opacity, *task_normal_icon_osb_set,
|
||||||
|
*task_normal_icon_saturation,
|
||||||
|
*task_normal_icon_brightness,
|
||||||
|
*task_normal_background, *task_normal_background_set;
|
||||||
|
extern GtkWidget *task_active_color, *task_active_color_set,
|
||||||
|
*task_active_icon_opacity, *task_active_icon_osb_set,
|
||||||
|
*task_active_icon_saturation,
|
||||||
|
*task_active_icon_brightness,
|
||||||
|
*task_active_background, *task_active_background_set;
|
||||||
|
extern GtkWidget *task_urgent_color, *task_urgent_color_set,
|
||||||
|
*task_urgent_icon_opacity, *task_urgent_icon_osb_set,
|
||||||
|
*task_urgent_icon_saturation,
|
||||||
|
*task_urgent_icon_brightness,
|
||||||
|
*task_urgent_background, *task_urgent_background_set;
|
||||||
|
extern GtkWidget *task_urgent_blinks;
|
||||||
|
extern GtkWidget *task_iconified_color, *task_iconified_color_set,
|
||||||
|
*task_iconified_icon_opacity, *task_iconified_icon_osb_set,
|
||||||
|
*task_iconified_icon_saturation,
|
||||||
|
*task_iconified_icon_brightness,
|
||||||
|
*task_iconified_background, *task_iconified_background_set;
|
||||||
|
|
||||||
// clock
|
// clock
|
||||||
GtkWidget *clock_format_line1, *clock_format_line2, *clock_tmz_line1, *clock_tmz_line2;
|
extern GtkWidget *clock_format_line1, *clock_format_line2, *clock_tmz_line1, *clock_tmz_line2;
|
||||||
GtkWidget *clock_left_command, *clock_right_command;
|
extern GtkWidget *clock_left_command, *clock_right_command;
|
||||||
GtkWidget *clock_padding_x, *clock_padding_y, *clock_font_line1, *clock_font_line2, *clock_font_color;
|
extern GtkWidget *clock_mclick_command, *clock_rclick_command, *clock_uwheel_command, *clock_dwheel_command;
|
||||||
|
extern GtkWidget *clock_padding_x, *clock_padding_y;
|
||||||
|
extern GtkWidget *clock_font_line1, *clock_font_line1_set, *clock_font_line2, *clock_font_line2_set, *clock_font_color;
|
||||||
|
extern GtkWidget *clock_background;
|
||||||
|
|
||||||
// battery
|
// battery
|
||||||
GtkWidget *battery_hide_if_higher, *battery_alert_if_lower, *battery_alert_cmd;
|
extern GtkWidget *battery_hide_if_higher, *battery_alert_if_lower, *battery_alert_cmd;
|
||||||
GtkWidget *battery_padding_x, *battery_padding_y, *battery_font_line1, *battery_font_line2, *battery_font_color;
|
extern GtkWidget *battery_padding_x, *battery_padding_y;
|
||||||
|
extern GtkWidget *battery_font_line1, *battery_font_line1_set, *battery_font_line2, *battery_font_line2_set, *battery_font_color;
|
||||||
|
extern GtkWidget *battery_background;
|
||||||
|
extern GtkWidget *battery_tooltip;
|
||||||
|
extern GtkWidget *battery_left_command, *battery_mclick_command, *battery_right_command, *battery_uwheel_command, *battery_dwheel_command;
|
||||||
|
extern GtkWidget *ac_connected_cmd, *ac_disconnected_cmd;
|
||||||
|
|
||||||
// systray
|
// systray
|
||||||
GtkWidget *systray_icon_order, *systray_padding_x, *systray_padding_y, *systray_spacing;
|
extern GtkWidget *systray_icon_order, *systray_padding_x, *systray_padding_y, *systray_spacing;
|
||||||
GtkWidget *systray_icon_size, *systray_icon_opacity, *systray_icon_saturation, *systray_icon_brightness;
|
extern GtkWidget *systray_icon_size, *systray_icon_opacity, *systray_icon_saturation, *systray_icon_brightness;
|
||||||
|
extern GtkWidget *systray_background, *systray_monitor;
|
||||||
|
|
||||||
// tooltip
|
// tooltip
|
||||||
GtkWidget *tooltip_padding_x, *tooltip_padding_y, *tooltip_font, *tooltip_font_color;
|
extern GtkWidget *tooltip_padding_x, *tooltip_padding_y, *tooltip_font, *tooltip_font_set, *tooltip_font_color;
|
||||||
GtkWidget *tooltip_task_show, *tooltip_show_after, *tooltip_hide_after;
|
extern GtkWidget *tooltip_task_show, *tooltip_show_after, *tooltip_hide_after;
|
||||||
GtkWidget *clock_format_tooltip, *clock_tmz_tooltip;
|
extern GtkWidget *clock_format_tooltip, *clock_tmz_tooltip;
|
||||||
|
extern GtkWidget *tooltip_background;
|
||||||
|
|
||||||
|
// Executor
|
||||||
|
typedef struct Executor {
|
||||||
|
char name[256];
|
||||||
|
GtkWidget *container;
|
||||||
|
GtkWidget *page_execp;
|
||||||
|
GtkWidget *page_label;
|
||||||
|
GtkWidget *execp_command, *execp_interval, *execp_has_icon, *execp_cache_icon, *execp_show_tooltip;
|
||||||
|
GtkWidget *execp_continuous, *execp_markup, *execp_tooltip;
|
||||||
|
GtkWidget *execp_left_command, *execp_right_command;
|
||||||
|
GtkWidget *execp_mclick_command, *execp_rclick_command, *execp_uwheel_command, *execp_dwheel_command;
|
||||||
|
GtkWidget *execp_font, *execp_font_set, *execp_font_color, *execp_padding_x, *execp_padding_y, *execp_centered;
|
||||||
|
GtkWidget *execp_background, *execp_icon_w, *execp_icon_h;
|
||||||
|
} Executor;
|
||||||
|
|
||||||
|
extern GArray *executors;
|
||||||
|
|
||||||
// launcher
|
// launcher
|
||||||
GtkWidget *launcher_icon_size, *launcher_icon_theme, *launcher_padding_x, *launcher_padding_y, *launcher_spacing;
|
|
||||||
|
enum {
|
||||||
|
appsColIcon = 0,
|
||||||
|
appsColIconName,
|
||||||
|
appsColText,
|
||||||
|
appsColPath,
|
||||||
|
appsNumCols
|
||||||
|
};
|
||||||
|
|
||||||
|
extern GtkListStore *launcher_apps, *all_apps;
|
||||||
|
extern GtkWidget *launcher_apps_view, *all_apps_view;
|
||||||
|
extern GtkWidget *launcher_apps_dirs;
|
||||||
|
|
||||||
|
extern GtkWidget *launcher_icon_size, *launcher_icon_theme, *launcher_padding_x, *launcher_padding_y, *launcher_spacing;
|
||||||
|
extern GtkWidget *launcher_icon_opacity, *launcher_icon_saturation, *launcher_icon_brightness;
|
||||||
|
extern GtkWidget *margin_x, *margin_y;
|
||||||
|
extern GtkWidget *launcher_background, *launcher_icon_background;
|
||||||
|
extern GtkWidget *startup_notifications;
|
||||||
|
extern IconThemeWrapper *icon_theme;
|
||||||
|
extern GtkWidget *launcher_tooltip;
|
||||||
|
extern GtkWidget *launcher_icon_theme_override;
|
||||||
|
|
||||||
|
void load_desktop_file(const char *file, gboolean selected);
|
||||||
|
void set_current_icon_theme(const char *theme);
|
||||||
|
gchar *get_current_icon_theme();
|
||||||
|
|
||||||
// background
|
// background
|
||||||
GtkWidget *combo_background;
|
enum {
|
||||||
GtkWidget *margin_x, *margin_y;
|
bgColPixbuf = 0,
|
||||||
|
bgColFillColor,
|
||||||
|
bgColFillOpacity,
|
||||||
|
bgColBorderColor,
|
||||||
|
bgColBorderOpacity,
|
||||||
|
bgColBorderWidth,
|
||||||
|
bgColCornerRadius,
|
||||||
|
bgColText,
|
||||||
|
bgColFillColorOver,
|
||||||
|
bgColFillOpacityOver,
|
||||||
|
bgColBorderColorOver,
|
||||||
|
bgColBorderOpacityOver,
|
||||||
|
bgColFillColorPress,
|
||||||
|
bgColFillOpacityPress,
|
||||||
|
bgColBorderColorPress,
|
||||||
|
bgColBorderOpacityPress,
|
||||||
|
bgNumCols
|
||||||
|
};
|
||||||
|
|
||||||
|
extern GtkListStore *backgrounds;
|
||||||
|
extern GtkWidget *current_background,
|
||||||
|
*background_fill_color,
|
||||||
|
*background_border_color,
|
||||||
|
*background_fill_color_over,
|
||||||
|
*background_border_color_over,
|
||||||
|
*background_fill_color_press,
|
||||||
|
*background_border_color_press,
|
||||||
|
*background_border_width,
|
||||||
|
*background_corner_radius;
|
||||||
|
|
||||||
|
void background_create_new();
|
||||||
|
void background_force_update();
|
||||||
|
int background_index_safe(int index);
|
||||||
|
|
||||||
GtkWidget *create_properties();
|
GtkWidget *create_properties();
|
||||||
|
|
||||||
|
void execp_create_new();
|
||||||
|
Executor *execp_get_last();
|
||||||
|
void execp_remove(int i);
|
||||||
|
void execp_update_indices();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,11 +1,11 @@
|
|||||||
|
|
||||||
#ifndef PROPERTIES_RW
|
#ifndef PROPERTIES_RW
|
||||||
#define PROPERTIES_RW
|
#define PROPERTIES_RW
|
||||||
|
|
||||||
|
#include <gtk/gtk.h>
|
||||||
|
|
||||||
|
char *get_current_theme_file_name();
|
||||||
|
gboolean config_is_manual(const char *path);
|
||||||
void config_read_file (const char *path);
|
void config_read_file (const char *path);
|
||||||
void config_save_file(const char *path);
|
void config_save_file(const char *path);
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -19,6 +19,7 @@
|
|||||||
|
|
||||||
|
|
||||||
#include "main.h"
|
#include "main.h"
|
||||||
|
#include "strnatcmp.h"
|
||||||
#include "theme_view.h"
|
#include "theme_view.h"
|
||||||
|
|
||||||
// The data columns that we export via the tree model interface
|
// The data columns that we export via the tree model interface
|
||||||
@@ -27,7 +28,10 @@ GtkListStore *g_store;
|
|||||||
int g_width_list, g_height_list;
|
int g_width_list, g_height_list;
|
||||||
GtkCellRenderer *g_renderer;
|
GtkCellRenderer *g_renderer;
|
||||||
|
|
||||||
|
gint theme_name_compare(GtkTreeModel *model,
|
||||||
|
GtkTreeIter *a,
|
||||||
|
GtkTreeIter *b,
|
||||||
|
gpointer user_data);
|
||||||
|
|
||||||
GtkWidget *create_view()
|
GtkWidget *create_view()
|
||||||
{
|
{
|
||||||
@@ -71,23 +75,50 @@ GtkWidget *create_view()
|
|||||||
GtkTreeSortable *sortable;
|
GtkTreeSortable *sortable;
|
||||||
sortable = GTK_TREE_SORTABLE(g_store);
|
sortable = GTK_TREE_SORTABLE(g_store);
|
||||||
gtk_tree_sortable_set_sort_column_id(sortable, COL_THEME_FILE, GTK_SORT_ASCENDING);
|
gtk_tree_sortable_set_sort_column_id(sortable, COL_THEME_FILE, GTK_SORT_ASCENDING);
|
||||||
|
gtk_tree_sortable_set_sort_func(sortable, COL_THEME_FILE, theme_name_compare, NULL, NULL);
|
||||||
return view;
|
return view;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gint theme_name_compare(GtkTreeModel *model,
|
||||||
|
GtkTreeIter *a,
|
||||||
|
GtkTreeIter *b,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
gchar *path_a, *path_b;
|
||||||
|
gtk_tree_model_get(model, a, COL_THEME_FILE, &path_a, -1);
|
||||||
|
gtk_tree_model_get(model, b, COL_THEME_FILE, &path_b, -1);
|
||||||
|
gchar *name_a = path_a;
|
||||||
|
gchar *p;
|
||||||
|
for (p = name_a; *p; p++) {
|
||||||
|
if (*p == '/')
|
||||||
|
name_a = p + 1;
|
||||||
|
}
|
||||||
|
gchar *name_b = path_b;
|
||||||
|
for (p = name_b; *p; p++) {
|
||||||
|
if (*p == '/')
|
||||||
|
name_b = p + 1;
|
||||||
|
}
|
||||||
|
if (g_str_equal(name_a, name_b))
|
||||||
|
return 0;
|
||||||
|
if (g_str_equal(name_a, "tint2rc"))
|
||||||
|
return -1;
|
||||||
|
if (g_str_equal(name_b, "tint2rc"))
|
||||||
|
return 1;
|
||||||
|
gint result = strnatcasecmp(name_a, name_b);
|
||||||
|
g_free(path_a);
|
||||||
|
g_free(path_b);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
void custom_list_append(const gchar *name)
|
void custom_list_append(const gchar *path)
|
||||||
{
|
{
|
||||||
GtkTreeIter iter;
|
GtkTreeIter iter;
|
||||||
|
|
||||||
gtk_list_store_append(g_store, &iter);
|
gtk_list_store_append(g_store, &iter);
|
||||||
gtk_list_store_set(g_store, &iter, COL_THEME_FILE, name, -1);
|
gtk_list_store_set(g_store, &iter, COL_THEME_FILE, path, -1);
|
||||||
|
|
||||||
gchar *b, *n;
|
gchar *name = g_strdup(strrchr(path, '/') + 1);
|
||||||
b = strrchr(name, '/');
|
gtk_list_store_set(g_store, &iter, COL_THEME_NAME, name, -1);
|
||||||
n = g_strdup(b+1);
|
|
||||||
b = strrchr(n, '.');
|
|
||||||
*b = '\0';
|
|
||||||
gtk_list_store_set(g_store, &iter, COL_THEME_NAME, n, -1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -96,7 +127,9 @@ gboolean update_snapshot()
|
|||||||
GtkTreeModel *model;
|
GtkTreeModel *model;
|
||||||
GtkTreeIter iter;
|
GtkTreeIter iter;
|
||||||
GdkPixbuf *icon;
|
GdkPixbuf *icon;
|
||||||
gboolean have_iter, found = FALSE;
|
gboolean have_iter;
|
||||||
|
|
||||||
|
gint pixWidth = 200, pixHeight = 30;
|
||||||
|
|
||||||
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
model = gtk_tree_view_get_model(GTK_TREE_VIEW(g_theme_view));
|
||||||
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
have_iter = gtk_tree_model_get_iter_first(model, &iter);
|
||||||
@@ -104,54 +137,42 @@ gboolean update_snapshot()
|
|||||||
gtk_tree_model_get(model, &iter, COL_SNAPSHOT, &icon, -1);
|
gtk_tree_model_get(model, &iter, COL_SNAPSHOT, &icon, -1);
|
||||||
if (icon != NULL) {
|
if (icon != NULL) {
|
||||||
g_object_unref(icon);
|
g_object_unref(icon);
|
||||||
have_iter = gtk_tree_model_iter_next(model, &iter);
|
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
found = TRUE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (found) {
|
|
||||||
// build panel's snapshot
|
// build panel's snapshot
|
||||||
GdkPixbuf *pixbuf;
|
GdkPixbuf *pixbuf = NULL;
|
||||||
gchar *name, *snap, *cmd;
|
gchar *name, *snap, *cmd;
|
||||||
gint pixWidth, pixHeight;
|
|
||||||
gboolean changeSize = FALSE;
|
|
||||||
|
|
||||||
snap = g_build_filename (g_get_user_config_dir(), "tint2", "snap.jpg", NULL);
|
snap = g_build_filename(g_get_user_config_dir(), "tint2", "snap.jpg", NULL);
|
||||||
g_remove(snap);
|
g_remove(snap);
|
||||||
|
|
||||||
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &name, -1);
|
gtk_tree_model_get(model, &iter, COL_THEME_FILE, &name, -1);
|
||||||
cmd = g_strdup_printf("tint2 -c \'%s\' -s \'%s\'", name, snap);
|
cmd = g_strdup_printf("tint2 -c \'%s\' -s \'%s\'", name, snap);
|
||||||
system(cmd);
|
if (system(cmd) == 0) {
|
||||||
|
// load
|
||||||
// load
|
pixbuf = gdk_pixbuf_new_from_file(snap, NULL);
|
||||||
pixbuf = gdk_pixbuf_new_from_file(snap, NULL);
|
if (pixbuf == NULL) {
|
||||||
if (pixbuf == NULL) {
|
printf("snapshot NULL : %s\n", cmd);
|
||||||
printf("snapshot NULL : %s\n", cmd);
|
}
|
||||||
found = FALSE;
|
|
||||||
}
|
}
|
||||||
g_free(snap);
|
g_free(snap);
|
||||||
g_free(cmd);
|
g_free(cmd);
|
||||||
g_free(name);
|
g_free(name);
|
||||||
|
|
||||||
pixWidth = gdk_pixbuf_get_width(pixbuf);
|
gint w, h;
|
||||||
pixHeight = gdk_pixbuf_get_height(pixbuf);
|
w = gdk_pixbuf_get_width(pixbuf);
|
||||||
if (g_width_list != pixWidth) {
|
h = gdk_pixbuf_get_height(pixbuf);
|
||||||
g_width_list = pixWidth;
|
pixWidth = w > pixWidth ? w : pixWidth;
|
||||||
changeSize = TRUE;
|
pixHeight = h > pixHeight ? h : pixHeight;
|
||||||
}
|
|
||||||
if (g_height_list != (pixHeight+30)) {
|
|
||||||
g_height_list = pixHeight+30;
|
|
||||||
changeSize = TRUE;
|
|
||||||
}
|
|
||||||
if (changeSize)
|
|
||||||
gtk_cell_renderer_set_fixed_size(g_renderer, g_width_list, g_height_list);
|
|
||||||
|
|
||||||
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, pixbuf, -1);
|
gtk_list_store_set(g_store, &iter, COL_SNAPSHOT, pixbuf, -1);
|
||||||
|
|
||||||
|
have_iter = gtk_tree_model_iter_next(model, &iter);
|
||||||
}
|
}
|
||||||
return found;
|
|
||||||
|
gtk_cell_renderer_set_fixed_size(g_renderer, pixWidth + 30, pixHeight + 30);
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,9 +10,7 @@ enum { COL_THEME_FILE = 0, COL_THEME_NAME, COL_SNAPSHOT, NB_COL, };
|
|||||||
|
|
||||||
GtkWidget *create_view();
|
GtkWidget *create_view();
|
||||||
|
|
||||||
void custom_list_append(const gchar *name);
|
void custom_list_append(const gchar *path);
|
||||||
|
|
||||||
gboolean update_snapshot();
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -1,149 +1,58 @@
|
|||||||
[Desktop Entry]
|
[Desktop Entry]
|
||||||
Type=Application
|
Type=Application
|
||||||
Encoding=UTF-8
|
Name=Tint2 panel settings
|
||||||
Name=Panel tint2
|
Name[am]=ፓነል አስተዳዳሪ
|
||||||
Name[am]=ፓነል tint2
|
Name[ar]=مدير الائحة :
|
||||||
Name[ar]=الشريط tint2
|
Name[ast]=Alministrador de panel
|
||||||
Name[ast]=Panel tint2
|
Name[be]=Кіраўнік Панэляў
|
||||||
Name[be]=Панэль tint2
|
Name[ca]=Gestor de quadres
|
||||||
Name[ca]=Quadre tint2
|
Name[cs]=Správce panelu
|
||||||
Name[cs]=Panel tint2
|
Name[da]=Panelhåndtering
|
||||||
Name[da]=Panel tint2
|
Name[de]=Leistenverwaltung
|
||||||
Name[de]=Leiste tint2
|
Name[dz]=པེ་ནཱལ་འཛིན་སྐྱོང་པ།
|
||||||
Name[dz]=པེ་ནཱལ། tint2
|
Name[el]=Διαχειριστής ταμπλό
|
||||||
Name[el]=Ταμπλό tint2
|
Name[en_GB]=Panel Manager
|
||||||
Name[en_GB]=Panel tint2
|
Name[eo]=Administrilo de Panelo
|
||||||
Name[eo]=Panelo tint2
|
Name[es]=Administrador de panel
|
||||||
Name[es]=Panel tint2
|
Name[et]=Paneelihaldur
|
||||||
Name[et]=Ääreriba tint2
|
Name[eu]=Panel Kudeatzailea:
|
||||||
Name[eu]=Panela tint2
|
Name[fi]=Paneelin hallinta
|
||||||
Name[fi]=Paneeli tint2
|
Name[fr]=Gestionnaire de panneau
|
||||||
Name[fr]=Panneau tint2
|
Name[gl]=Xestor de paneis
|
||||||
Name[gl]=Panel tint2
|
Name[he]=מנהל הלוח
|
||||||
Name[he]=לוח tint2
|
Name[hu]=Panelkezelő
|
||||||
Name[hu]=Panel tint2
|
Name[id]=Manajer Panel
|
||||||
Name[id]=Panel tint2
|
Name[it]=Gestore dei pannelli
|
||||||
Name[it]=Pannello tint2
|
Name[ja]=パネルマネージャ
|
||||||
Name[ja]=パネル tint2
|
Name[kk]=Панель менеджері
|
||||||
Name[kk]=Панель tint2
|
Name[ko]=패널 관리자
|
||||||
Name[ko]=패널 tint2
|
Name[ku]=Gerinendeyê panelan
|
||||||
Name[ku]=Panel tint2
|
Name[lv]=Paneļu pārvaldnieks
|
||||||
Name[lv]=Panelis tint2
|
Name[mk]=Менаџер за панели
|
||||||
Name[mk]=Панел tint2
|
Name[nb]=Panelbehandler
|
||||||
Name[nb]=Panel tint2
|
Name[nl]=Paneel Manager
|
||||||
Name[nl]=Paneel tint2
|
Name[nn]=Panelhandsamar
|
||||||
Name[nn]=Panel tint2
|
Name[pa]=ਪੈਨਲ ਮੈਨੇਜਰ
|
||||||
Name[pa]=ਪੈਨਲ tint2
|
Name[pl]=Ustawienia panelu
|
||||||
Name[pl]=Panel tint2
|
Name[pt]=Gestor do Painel
|
||||||
Name[pt]=Painel tint2
|
Name[pt_BR]=Gerenciador do painel
|
||||||
Name[pt_BR]=Painel tint2
|
Name[ro]=Manager de panouri
|
||||||
Name[ro]=Panou tint2
|
Name[ru]=Диспетчер панелей
|
||||||
Name[ru]=Панель tint2
|
Name[si]=පුවරු කළමණාකරු
|
||||||
Name[si]=පුවරුව tint2
|
Name[sk]=Nastavenie panelu
|
||||||
Name[sk]=Panel tint2
|
Name[sq]=Përgjegjës Panelesh
|
||||||
Name[sq]=Panel tint2
|
Name[sv]=Panelhanterare
|
||||||
Name[sv]=Panel tint2
|
Name[ta]=பலகை மேளாலர்:
|
||||||
Name[ta]=பலகை tint2
|
Name[tr]=Panel Yöneticisi
|
||||||
Name[tr]=Panel tint2
|
Name[ug]=Panel باشقۇرغۇ
|
||||||
Name[ug]=panel tint2
|
Name[uk]=Менеджер панелей
|
||||||
Name[uk]=Панель tint2
|
Name[ur]=پینل منیجر
|
||||||
Name[ur]=پینل tint2
|
Name[ur_PK]=پینل منیجر
|
||||||
Name[ur_PK]=پینل tint2
|
Name[vi]=Quản lý panel
|
||||||
Name[vi]=Panel tint2
|
Name[zh_CN]=面板管理器
|
||||||
Name[zh_CN]=面板 tint2
|
Name[zh_TW]=面板管理程式
|
||||||
Name[zh_TW]=面板 tint2
|
|
||||||
GenericName=Panel Manager
|
|
||||||
GenericName[am]=ፓነል አስተዳዳሪ
|
|
||||||
GenericName[ar]=مدير الائحة :
|
|
||||||
GenericName[ast]=Alministrador de panel
|
|
||||||
GenericName[be]=Кіраўнік Панэляў
|
|
||||||
GenericName[ca]=Gestor de quadres
|
|
||||||
GenericName[cs]=Správce panelu
|
|
||||||
GenericName[da]=Panelhåndtering
|
|
||||||
GenericName[de]=Leistenverwaltung
|
|
||||||
GenericName[dz]=པེ་ནཱལ་འཛིན་སྐྱོང་པ།
|
|
||||||
GenericName[el]=Διαχειριστής ταμπλό
|
|
||||||
GenericName[en_GB]=Panel Manager
|
|
||||||
GenericName[eo]=Administrilo de Panelo
|
|
||||||
GenericName[es]=Administrador de panel
|
|
||||||
GenericName[et]=Paneelihaldur
|
|
||||||
GenericName[eu]=Panel Kudeatzailea:
|
|
||||||
GenericName[fi]=Paneelin hallinta
|
|
||||||
GenericName[fr]=Gestionnaire de panneau
|
|
||||||
GenericName[gl]=Xestor de paneis
|
|
||||||
GenericName[he]=מנהל הלוח
|
|
||||||
GenericName[hu]=Panelkezelő
|
|
||||||
GenericName[id]=Manajer Panel
|
|
||||||
GenericName[it]=Gestore dei pannelli
|
|
||||||
GenericName[ja]=パネルマネージャ
|
|
||||||
GenericName[kk]=Панель менеджері
|
|
||||||
GenericName[ko]=패널 관리자
|
|
||||||
GenericName[ku]=Gerinendeyê panelan
|
|
||||||
GenericName[lv]=Paneļu pārvaldnieks
|
|
||||||
GenericName[mk]=Менаџер за панели
|
|
||||||
GenericName[nb]=Panelbehandler
|
|
||||||
GenericName[nl]=Paneel Manager
|
|
||||||
GenericName[nn]=Panelhandsamar
|
|
||||||
GenericName[pa]=ਪੈਨਲ ਮੈਨੇਜਰ
|
|
||||||
GenericName[pl]=Panel
|
|
||||||
GenericName[pt]=Gestor do Painel
|
|
||||||
GenericName[pt_BR]=Gerenciador do painel
|
|
||||||
GenericName[ro]=Manager de panouri
|
|
||||||
GenericName[ru]=Диспетчер панелей
|
|
||||||
GenericName[si]=පුවරු කළමණාකරු
|
|
||||||
GenericName[sk]=Nastavenie panelu
|
|
||||||
GenericName[sq]=Përgjegjës Panelesh
|
|
||||||
GenericName[sv]=Panelhanterare
|
|
||||||
GenericName[ta]=பலகை மேளாலர்:
|
|
||||||
GenericName[tr]=Panel Yöneticisi
|
|
||||||
GenericName[ug]=Panel باشقۇرغۇ
|
|
||||||
GenericName[uk]=Менеджер панелей
|
|
||||||
GenericName[ur]=پینل منیجر
|
|
||||||
GenericName[ur_PK]=پینل منیجر
|
|
||||||
GenericName[vi]=Quản lý panel
|
|
||||||
GenericName[zh_CN]=面板管理器
|
|
||||||
GenericName[zh_TW]=面板管理程式
|
|
||||||
Comment=Customize the panel settings
|
|
||||||
Comment[ast]=Personalice la configuración del panel
|
|
||||||
Comment[ca]=Personalitza els ajustaments del quadre
|
|
||||||
Comment[cs]=Přizpůsobit nastavení panelu
|
|
||||||
Comment[da]=Tilpas panelets indstillinger
|
|
||||||
Comment[de]=Leiste anpassen
|
|
||||||
Comment[el]=Προσαρμογή ρυθμίσεων ταμπλό
|
|
||||||
Comment[en_GB]=Customise the panel settings
|
|
||||||
Comment[es]=Personalice la configuración del panel
|
|
||||||
Comment[et]=Ääreriba seadistamine käepäraseks
|
|
||||||
Comment[eu]=Pertsonalizatu panel ezapenak
|
|
||||||
Comment[fi]=Mukauta paneelin asetuksia
|
|
||||||
Comment[fr]=Configurer le panneau
|
|
||||||
Comment[gl]=Personalizar a configuración do panel
|
|
||||||
Comment[hu]=A panel beállításainak személyre szabása
|
|
||||||
Comment[id]=Kustomisasi pengaturan panel
|
|
||||||
Comment[it]=Personalizzazione delle impostazioni del pannello
|
|
||||||
Comment[ja]=パネルの設定をカスタマイズします
|
|
||||||
Comment[kk]=Панельді баптау
|
|
||||||
Comment[lv]=Pielāgot paneļa uzstādījumus
|
|
||||||
Comment[nb]=Tilpass panelet
|
|
||||||
Comment[nl]=Paneel configureren
|
|
||||||
Comment[nn]=Tilpass innstillingane for panelet
|
|
||||||
Comment[pl]=Konfiguruje ustawienia paneli
|
|
||||||
Comment[pt]=Personalizar as definições do painel
|
|
||||||
Comment[pt_BR]=Personalizar as configurações do painel
|
|
||||||
Comment[ro]=Schimbați opțiunile panoului
|
|
||||||
Comment[ru]=Настроить панель
|
|
||||||
Comment[si]=පුවරු සිටුවම් සකසන්න
|
|
||||||
Comment[sk]=Prispôsobiť nastavenia panelu
|
|
||||||
Comment[sq]=Përshtasni rregullimet për panelin
|
|
||||||
Comment[sv]=Anpassa panelinställningarna
|
|
||||||
Comment[ta]=நிரல்பலகை அமைப்பை தன்விருப்பமாற்றம் செய்க
|
|
||||||
Comment[tr]=Panel ayarlarını özelleştir
|
|
||||||
Comment[ug]=panel تەڭشىكىنى Customize قىلىش
|
|
||||||
Comment[uk]=Налаштувати властивості панелі
|
|
||||||
Comment[ur]=پینل کی ترتیبات
|
|
||||||
Comment[ur_PK]=پینل کی ترتیبات
|
|
||||||
Comment[zh_CN]=自定义面板设置
|
|
||||||
Exec=tint2conf
|
Exec=tint2conf
|
||||||
Icon=taskbar
|
Icon=tint2conf
|
||||||
Terminal=false
|
Terminal=false
|
||||||
Categories=Settings;DesktopSettings;
|
Categories=Settings;DesktopSettings;
|
||||||
|
|
||||||
|
|||||||
509
src/tint2conf/tint2conf.svg
Normal file
509
src/tint2conf/tint2conf.svg
Normal file
@@ -0,0 +1,509 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:cc="http://creativecommons.org/ns#"
|
||||||
|
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
version="1.1"
|
||||||
|
width="48"
|
||||||
|
height="48"
|
||||||
|
id="svg1325"
|
||||||
|
inkscape:version="0.48.3.1 r9886"
|
||||||
|
sodipodi:docname="tint2conf.svg"
|
||||||
|
inkscape:export-filename="/home/omega_dist/4_devel_open_source/9_tint/tint2/src/tint2conf/tint2conf.png"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-ydpi="90">
|
||||||
|
<metadata
|
||||||
|
id="metadata50">
|
||||||
|
<rdf:RDF>
|
||||||
|
<cc:Work
|
||||||
|
rdf:about="">
|
||||||
|
<dc:format>image/svg+xml</dc:format>
|
||||||
|
<dc:type
|
||||||
|
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
|
||||||
|
<dc:title />
|
||||||
|
</cc:Work>
|
||||||
|
</rdf:RDF>
|
||||||
|
</metadata>
|
||||||
|
<sodipodi:namedview
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#666666"
|
||||||
|
borderopacity="1"
|
||||||
|
objecttolerance="10"
|
||||||
|
gridtolerance="10"
|
||||||
|
guidetolerance="10"
|
||||||
|
inkscape:pageopacity="0"
|
||||||
|
inkscape:pageshadow="2"
|
||||||
|
inkscape:window-width="1768"
|
||||||
|
inkscape:window-height="1056"
|
||||||
|
id="namedview48"
|
||||||
|
showgrid="false"
|
||||||
|
inkscape:zoom="16"
|
||||||
|
inkscape:cx="30.824904"
|
||||||
|
inkscape:cy="2.1959811"
|
||||||
|
inkscape:window-x="152"
|
||||||
|
inkscape:window-y="0"
|
||||||
|
inkscape:window-maximized="1"
|
||||||
|
inkscape:current-layer="g3841" />
|
||||||
|
<defs
|
||||||
|
id="defs1327">
|
||||||
|
<inkscape:perspective
|
||||||
|
sodipodi:type="inkscape:persp3d"
|
||||||
|
inkscape:vp_x="0 : 24 : 1"
|
||||||
|
inkscape:vp_y="0 : 1000 : 0"
|
||||||
|
inkscape:vp_z="48 : 24 : 1"
|
||||||
|
inkscape:persp3d-origin="24 : 16 : 1"
|
||||||
|
id="perspective52" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient4708">
|
||||||
|
<stop
|
||||||
|
id="stop4710"
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop4712"
|
||||||
|
style="stop-color:black;stop-opacity:1"
|
||||||
|
offset="0.57954973" />
|
||||||
|
<stop
|
||||||
|
id="stop4714"
|
||||||
|
style="stop-color:black;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient2270">
|
||||||
|
<stop
|
||||||
|
id="stop2272"
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop2274"
|
||||||
|
style="stop-color:white;stop-opacity:0"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
x1="20"
|
||||||
|
y1="18"
|
||||||
|
x2="20"
|
||||||
|
y2="46"
|
||||||
|
id="linearGradient2851"
|
||||||
|
xlink:href="#linearGradient2270"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,-9.9999992)" />
|
||||||
|
<linearGradient
|
||||||
|
x1="30"
|
||||||
|
y1="17"
|
||||||
|
x2="30"
|
||||||
|
y2="47"
|
||||||
|
id="linearGradient2854"
|
||||||
|
xlink:href="#linearGradient2181"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,-9.9999992)" />
|
||||||
|
<linearGradient
|
||||||
|
x1="63.397362"
|
||||||
|
y1="-12.489107"
|
||||||
|
x2="63.397362"
|
||||||
|
y2="5.4675598"
|
||||||
|
id="linearGradient2685"
|
||||||
|
xlink:href="#linearGradient4873"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.5436509,0,0,1.5436158,-80.015712,21.419381)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient4873">
|
||||||
|
<stop
|
||||||
|
id="stop4875"
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop4877"
|
||||||
|
style="stop-color:white;stop-opacity:0"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<radialGradient
|
||||||
|
cx="23.895569"
|
||||||
|
cy="3.9900031"
|
||||||
|
r="20.397499"
|
||||||
|
fx="23.895569"
|
||||||
|
fy="3.9900031"
|
||||||
|
id="radialGradient2688"
|
||||||
|
xlink:href="#linearGradient3242-187-536"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0,1.7008781,-2.245129,0,24.958071,-40.236051)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient3242-187-536">
|
||||||
|
<stop
|
||||||
|
id="stop2778"
|
||||||
|
style="stop-color:#8badea;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop2780"
|
||||||
|
style="stop-color:#6396cd;stop-opacity:1"
|
||||||
|
offset="0.26238" />
|
||||||
|
<stop
|
||||||
|
id="stop2782"
|
||||||
|
style="stop-color:#3b7caf;stop-opacity:1"
|
||||||
|
offset="0.66093999" />
|
||||||
|
<stop
|
||||||
|
id="stop2784"
|
||||||
|
style="stop-color:#194c70;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
x1="18.379412"
|
||||||
|
y1="44.980297"
|
||||||
|
x2="18.379412"
|
||||||
|
y2="3.0816143"
|
||||||
|
id="linearGradient2690"
|
||||||
|
xlink:href="#linearGradient2490-182-124"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0.7126612,0,0,0.7126613,-1.1038706,-1.1038704)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient2490-182-124">
|
||||||
|
<stop
|
||||||
|
id="stop2788"
|
||||||
|
style="stop-color:#1f4b6a;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop2790"
|
||||||
|
style="stop-color:#4083c2;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<radialGradient
|
||||||
|
cx="62.625"
|
||||||
|
cy="4.625"
|
||||||
|
r="10.625"
|
||||||
|
fx="62.625"
|
||||||
|
fy="4.625"
|
||||||
|
id="radialGradient2693"
|
||||||
|
xlink:href="#linearGradient8838"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.5058824,0,0,0.37647,-78.305888,26.258824)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient8838">
|
||||||
|
<stop
|
||||||
|
id="stop8840"
|
||||||
|
style="stop-color:black;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop8842"
|
||||||
|
style="stop-color:black;stop-opacity:0"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient2181">
|
||||||
|
<stop
|
||||||
|
id="stop2183"
|
||||||
|
style="stop-color:#f0f0f0;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop2185"
|
||||||
|
style="stop-color:#d3d3d3;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
x1="50.23077"
|
||||||
|
y1="22"
|
||||||
|
x2="53"
|
||||||
|
y2="22"
|
||||||
|
id="linearGradient4720"
|
||||||
|
xlink:href="#linearGradient4708"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(4.3333333,0,0,1,-176.66667,0)" />
|
||||||
|
<mask
|
||||||
|
id="mask4716">
|
||||||
|
<rect
|
||||||
|
width="52"
|
||||||
|
height="32"
|
||||||
|
x="1"
|
||||||
|
y="6"
|
||||||
|
id="rect4718"
|
||||||
|
style="fill:url(#linearGradient4720);fill-opacity:1;stroke:none" />
|
||||||
|
</mask>
|
||||||
|
<linearGradient
|
||||||
|
x1="50.23077"
|
||||||
|
y1="22"
|
||||||
|
x2="53"
|
||||||
|
y2="22"
|
||||||
|
id="linearGradient4726"
|
||||||
|
xlink:href="#linearGradient4708"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(4.3333333,0,0,1,-176.66667,0)" />
|
||||||
|
<mask
|
||||||
|
id="mask4722">
|
||||||
|
<rect
|
||||||
|
width="52"
|
||||||
|
height="32"
|
||||||
|
x="1"
|
||||||
|
y="6"
|
||||||
|
id="rect4724"
|
||||||
|
style="fill:url(#linearGradient4726);fill-opacity:1;stroke:none" />
|
||||||
|
</mask>
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4873"
|
||||||
|
id="linearGradient2861"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.8085571,0,0,0.74493792,-86.459085,24.622683)"
|
||||||
|
x1="63.397362"
|
||||||
|
y1="-12.489107"
|
||||||
|
x2="63.397362"
|
||||||
|
y2="5.4675598" />
|
||||||
|
<radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient3242-187-536"
|
||||||
|
id="radialGradient2864"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0,0.82083158,-2.630416,0,36.52927,-5.1317867)"
|
||||||
|
cx="23.895569"
|
||||||
|
cy="3.9900031"
|
||||||
|
fx="23.895569"
|
||||||
|
fy="3.9900031"
|
||||||
|
r="20.397499" />
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2490-182-124"
|
||||||
|
id="linearGradient2866"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0.83496113,0,0,0.34392523,5.994833,13.753123)"
|
||||||
|
x1="18.379412"
|
||||||
|
y1="44.980297"
|
||||||
|
x2="18.379412"
|
||||||
|
y2="3.0816143" />
|
||||||
|
<radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8838"
|
||||||
|
id="radialGradient2869"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.7643072,0,0,0.18168172,-84.455838,26.958163)"
|
||||||
|
cx="62.625"
|
||||||
|
cy="4.625"
|
||||||
|
fx="62.625"
|
||||||
|
fy="4.625"
|
||||||
|
r="10.625" />
|
||||||
|
<radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8838"
|
||||||
|
id="radialGradient2876"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.7643072,0,0,0.18168172,-84.455838,26.958163)"
|
||||||
|
cx="62.625"
|
||||||
|
cy="4.625"
|
||||||
|
fx="62.625"
|
||||||
|
fy="4.625"
|
||||||
|
r="10.625" />
|
||||||
|
<radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient3242-187-536"
|
||||||
|
id="radialGradient2878"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0,0.82083158,-2.630416,0,36.52927,-5.1317867)"
|
||||||
|
cx="23.895569"
|
||||||
|
cy="3.9900031"
|
||||||
|
fx="23.895569"
|
||||||
|
fy="3.9900031"
|
||||||
|
r="20.397499" />
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2490-182-124"
|
||||||
|
id="linearGradient2880"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0.83496113,0,0,0.34392523,5.994833,13.753123)"
|
||||||
|
x1="18.379412"
|
||||||
|
y1="44.980297"
|
||||||
|
x2="18.379412"
|
||||||
|
y2="3.0816143" />
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4873"
|
||||||
|
id="linearGradient2882"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.8085571,0,0,0.74493792,-86.459085,24.622683)"
|
||||||
|
x1="63.397362"
|
||||||
|
y1="-12.489107"
|
||||||
|
x2="63.397362"
|
||||||
|
y2="5.4675598" />
|
||||||
|
<inkscape:perspective
|
||||||
|
id="perspective2924"
|
||||||
|
inkscape:persp3d-origin="0.5 : 0.33333333 : 1"
|
||||||
|
inkscape:vp_z="1 : 0.5 : 1"
|
||||||
|
inkscape:vp_y="0 : 1000 : 0"
|
||||||
|
inkscape:vp_x="0 : 0.5 : 1"
|
||||||
|
sodipodi:type="inkscape:persp3d" />
|
||||||
|
<linearGradient
|
||||||
|
x1="30"
|
||||||
|
y1="17"
|
||||||
|
x2="30"
|
||||||
|
y2="47"
|
||||||
|
id="linearGradient2854-0"
|
||||||
|
xlink:href="#linearGradient2181-7"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,-9.9999992)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient2181-7">
|
||||||
|
<stop
|
||||||
|
id="stop2183-8"
|
||||||
|
style="stop-color:#f0f0f0;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop2185-6"
|
||||||
|
style="stop-color:#d3d3d3;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<mask
|
||||||
|
id="mask4722-8">
|
||||||
|
<rect
|
||||||
|
width="52"
|
||||||
|
height="32"
|
||||||
|
x="1"
|
||||||
|
y="6"
|
||||||
|
id="rect4724-8"
|
||||||
|
style="fill:url(#linearGradient4726-4);fill-opacity:1;stroke:none" />
|
||||||
|
</mask>
|
||||||
|
<linearGradient
|
||||||
|
x1="50.23077"
|
||||||
|
y1="22"
|
||||||
|
x2="53"
|
||||||
|
y2="22"
|
||||||
|
id="linearGradient4726-4"
|
||||||
|
xlink:href="#linearGradient4708-3"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(4.3333333,0,0,1,-176.66667,0)" />
|
||||||
|
<linearGradient
|
||||||
|
id="linearGradient4708-3">
|
||||||
|
<stop
|
||||||
|
id="stop4710-1"
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0" />
|
||||||
|
<stop
|
||||||
|
id="stop4712-4"
|
||||||
|
style="stop-color:black;stop-opacity:1"
|
||||||
|
offset="0.57954973" />
|
||||||
|
<stop
|
||||||
|
id="stop4714-9"
|
||||||
|
style="stop-color:black;stop-opacity:1"
|
||||||
|
offset="1" />
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient
|
||||||
|
y2="47"
|
||||||
|
x2="30"
|
||||||
|
y1="17"
|
||||||
|
x1="30"
|
||||||
|
gradientTransform="translate(0,-9.9999992)"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
id="linearGradient2940"
|
||||||
|
xlink:href="#linearGradient2181-7"
|
||||||
|
inkscape:collect="always" />
|
||||||
|
<inkscape:perspective
|
||||||
|
id="perspective2894"
|
||||||
|
inkscape:persp3d-origin="0.5 : 0.33333333 : 1"
|
||||||
|
inkscape:vp_z="1 : 0.5 : 1"
|
||||||
|
inkscape:vp_y="0 : 1000 : 0"
|
||||||
|
inkscape:vp_x="0 : 0.5 : 1"
|
||||||
|
sodipodi:type="inkscape:persp3d" />
|
||||||
|
</defs>
|
||||||
|
<rect
|
||||||
|
width="46"
|
||||||
|
height="29"
|
||||||
|
x="2.5"
|
||||||
|
y="7.5"
|
||||||
|
mask="url(#mask4722)"
|
||||||
|
id="rect1333"
|
||||||
|
style="fill:url(#linearGradient2854);fill-opacity:1;stroke:#2c2c2c;stroke-width:2;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
|
||||||
|
transform="translate(-0.20338983,3.4237288)" />
|
||||||
|
<rect
|
||||||
|
width="45"
|
||||||
|
height="27"
|
||||||
|
x="3.5"
|
||||||
|
y="8.5"
|
||||||
|
mask="url(#mask4716)"
|
||||||
|
id="rect2210"
|
||||||
|
style="opacity:0.7;fill:none;stroke:url(#linearGradient2851);stroke-width:0.99999976;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
|
||||||
|
transform="translate(-0.20338983,3.4237288)" />
|
||||||
|
<rect
|
||||||
|
style="fill:#ffffff;fill-opacity:1;stroke:#333424;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0.69999999999999996"
|
||||||
|
id="rect6306"
|
||||||
|
width="29.45332"
|
||||||
|
height="9.5719652"
|
||||||
|
x="10.697065"
|
||||||
|
y="21.044527"
|
||||||
|
rx="1.4456253"
|
||||||
|
ry="1.6443363" />
|
||||||
|
<g
|
||||||
|
id="g3841"
|
||||||
|
transform="translate(130.875,44.5)">
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="csscccsscc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3833"
|
||||||
|
d="M -71.433946,14.735703 C -80.071947,4.6612321 -89.248707,-5.737975 -92.147263,-9.1977463 c -3.224562,-3.8488997 -3.263797,-3.3549747 -1.493188,-4.5254517 l 1.278694,-0.845292 14.113986,16.2862263 14.113987,16.2862257 -1.004295,0.889458 c -0.552362,0.489201 -1.149443,0.973153 -1.326848,1.075448 -0.259255,0.149489 -1.234399,-0.877491 -4.969019,-5.233165 z"
|
||||||
|
style="fill:#87aade" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="cscsssc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3825"
|
||||||
|
d="m -98.105823,-13.116897 c -0.09278,-0.307786 -1.541317,-8.963836 -1.697567,-8.726554 -0.0244,0.03706 7.876058,3.167414 7.876435,3.661796 0.27526,-0.22635 0.628731,0.02647 0.174381,0.954172 -0.406614,0.830238 -0.712942,1.441488 -2.270947,2.615226 -0.993109,0.748168 -2.553662,1.682052 -2.971367,1.846131 -0.688741,0.270545 -1.081414,0.266981 -1.110935,-0.350771 z"
|
||||||
|
style="fill:#e9c6af" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="scscscsss"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3831"
|
||||||
|
d="m -77.291221,0.7427987 -14.60455,-16.3513767 0.637035,-0.730536 c 0.266185,-0.305255 0.324002,-1.389769 0.39974,-1.43161 0.07574,-0.04184 6.588518,8.0823714 14.511041,17.25171965 l 14.404588,16.67154235 -0.456991,0.486444 c -0.251345,0.267543 -0.509994,0.486443 -0.574776,0.486443 -0.06478,0 -6.507022,-7.3721814 -14.316087,-16.3826263 z"
|
||||||
|
style="fill:#afc6e9" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="sssssscs"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3835"
|
||||||
|
d="m -72.642373,17.731344 c -15.785516,-18.3468257 -22.67882,-27.078419 -24.317731,-29.175546 -0.237129,-0.303427 -0.02287,-0.396673 0.25262,-0.547032 0.446221,-0.24354 0.48891,-0.24519 1.064375,0.556916 0.626398,0.873099 10.342493,12.0396466 21.114327,24.608034 6.242539,7.283685 7.651572,8.214809 7.295404,8.484526 -0.209506,0.158654 -1.332932,1.444171 -1.357614,0.66226 -0.39968,0.56189 -2.205614,-2.443908 -4.051381,-4.589158 z"
|
||||||
|
style="fill:#3771c8" />
|
||||||
|
<path
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3837"
|
||||||
|
d="m -99.864407,-18.915254 3.050848,-2.237288"
|
||||||
|
style="fill:#0b2822;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="cccc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3049"
|
||||||
|
d="m -98.144068,-11.76536 29.460805,34.842161 7.728814,-6.711865 -29.898305,-34.576271"
|
||||||
|
style="fill:none;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="cccccc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3823"
|
||||||
|
d="m -98.112288,-12.03125 2.174788,-0.688559 3.864407,-2.847458 1.079449,-2.691208 -9.621826,-4.630826 z"
|
||||||
|
style="fill:none;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="ccscccsscc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3827"
|
||||||
|
d="m -80.955348,4.2773755 c -7.750968,-9.0443333 -14.97509,-16.8628515 -14.945966,-16.9543465 0.02912,-0.09149 1.512719,-0.624766 2.1979,-1.135322 l 1.426197,-1.062718 14.171177,16.60582 14.202426,16.387071 -0.88474,0.775212 c -0.486606,0.426368 -1.11724,0.948927 -1.401408,1.161244 l -0.516667,0.386032 z"
|
||||||
|
style="fill:#5f8dd3" />
|
||||||
|
<path
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3829"
|
||||||
|
d="m -83.124992,5.0906125 c -7.869016,-9.1612493 -14.285686,-16.7328745 -14.259267,-16.8258325 0.02642,-0.09296 0.288522,-0.270134 0.582452,-0.393722 0.525603,-0.220999 0.770084,0.05025 14.820977,16.4434263 l 14.286559,16.6681317 -0.492096,0.387084 c -0.270652,0.212894 -0.523421,0.384976 -0.561709,0.382405 -0.03829,-0.0026 -6.507899,-7.500244 -14.376916,-16.6614925 l 0,0 z"
|
||||||
|
style="fill:#3771c8" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="cc"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3051"
|
||||||
|
d="m -96.265889,-12.891949 29.350635,34.044492"
|
||||||
|
style="fill:none;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" />
|
||||||
|
<path
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3053"
|
||||||
|
d="m -92.338983,-15.661017 29.084746,33.559322"
|
||||||
|
style="fill:none;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" />
|
||||||
|
<path
|
||||||
|
sodipodi:nodetypes="scscccccs"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
id="path3839"
|
||||||
|
d="m -100.69352,-20.893751 c -0.0917,-0.34644 -0.71612,-3.00338 -0.8271,-3.157561 0.0437,-0.0437 2.636115,1.39898 2.993406,1.577103 l 1.143051,0.524237 -0.435163,0.963572 -0.709103,0.627398 -0.787319,0.476477 -1.110472,0.181053 c 0.17185,0.510627 -0.0972,-0.549821 -0.2673,-1.192279 z"
|
||||||
|
style="fill:#5f8dd3;fill-opacity:1" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 18 KiB |
File diff suppressed because it is too large
Load Diff
@@ -28,42 +28,46 @@
|
|||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
|
||||||
static int x, y, width, height;
|
static int x, y, width, height;
|
||||||
|
static gboolean just_shown;
|
||||||
|
|
||||||
// the next functions are helper functions for tooltip handling
|
// the next functions are helper functions for tooltip handling
|
||||||
void start_show_timeout();
|
void start_show_timeout();
|
||||||
void start_hide_timeout();
|
void start_hide_timeout();
|
||||||
void stop_tooltip_timeout();
|
void stop_tooltip_timeout();
|
||||||
|
|
||||||
Tooltip g_tooltip;
|
void tooltip_init_fonts();
|
||||||
|
|
||||||
|
Tooltip g_tooltip;
|
||||||
|
|
||||||
void default_tooltip()
|
void default_tooltip()
|
||||||
{
|
{
|
||||||
// give the tooltip some reasonable default values
|
// give the tooltip some reasonable default values
|
||||||
memset(&g_tooltip, 0, sizeof(Tooltip));
|
memset(&g_tooltip, 0, sizeof(Tooltip));
|
||||||
|
|
||||||
g_tooltip.font_color.color[0] = 1;
|
g_tooltip.font_color.rgb[0] = 1;
|
||||||
g_tooltip.font_color.color[1] = 1;
|
g_tooltip.font_color.rgb[1] = 1;
|
||||||
g_tooltip.font_color.color[2] = 1;
|
g_tooltip.font_color.rgb[2] = 1;
|
||||||
g_tooltip.font_color.alpha = 1;
|
g_tooltip.font_color.alpha = 1;
|
||||||
|
just_shown = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cleanup_tooltip()
|
void cleanup_tooltip()
|
||||||
{
|
{
|
||||||
stop_tooltip_timeout();
|
stop_tooltip_timeout();
|
||||||
tooltip_hide(0);
|
tooltip_hide(NULL);
|
||||||
tooltip_copy_text(0);
|
tooltip_copy_text(NULL);
|
||||||
if (g_tooltip.window) XDestroyWindow(server.dsp, g_tooltip.window);
|
if (g_tooltip.window)
|
||||||
if (g_tooltip.font_desc) pango_font_description_free(g_tooltip.font_desc);
|
XDestroyWindow(server.display, g_tooltip.window);
|
||||||
|
g_tooltip.window = 0;
|
||||||
|
pango_font_description_free(g_tooltip.font_desc);
|
||||||
|
g_tooltip.font_desc = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void init_tooltip()
|
void init_tooltip()
|
||||||
{
|
{
|
||||||
if (!g_tooltip.font_desc)
|
if (!g_tooltip.bg)
|
||||||
g_tooltip.font_desc = pango_font_description_from_string("sans 10");
|
|
||||||
if (g_tooltip.bg == 0)
|
|
||||||
g_tooltip.bg = &g_array_index(backgrounds, Background, 0);
|
g_tooltip.bg = &g_array_index(backgrounds, Background, 0);
|
||||||
|
tooltip_init_fonts();
|
||||||
|
|
||||||
XSetWindowAttributes attr;
|
XSetWindowAttributes attr;
|
||||||
attr.override_redirect = True;
|
attr.override_redirect = True;
|
||||||
@@ -71,69 +75,97 @@ void init_tooltip()
|
|||||||
attr.colormap = server.colormap;
|
attr.colormap = server.colormap;
|
||||||
attr.background_pixel = 0;
|
attr.background_pixel = 0;
|
||||||
attr.border_pixel = 0;
|
attr.border_pixel = 0;
|
||||||
unsigned long mask = CWEventMask|CWColormap|CWBorderPixel|CWBackPixel|CWOverrideRedirect;
|
unsigned long mask = CWEventMask | CWColormap | CWBorderPixel | CWBackPixel | CWOverrideRedirect;
|
||||||
if (g_tooltip.window) XDestroyWindow(server.dsp, g_tooltip.window);
|
if (g_tooltip.window)
|
||||||
g_tooltip.window = XCreateWindow(server.dsp, server.root_win, 0, 0, 100, 20, 0, server.depth, InputOutput, server.visual, mask, &attr);
|
XDestroyWindow(server.display, g_tooltip.window);
|
||||||
|
g_tooltip.window = XCreateWindow(server.display,
|
||||||
|
server.root_win,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
100,
|
||||||
|
20,
|
||||||
|
0,
|
||||||
|
server.depth,
|
||||||
|
InputOutput,
|
||||||
|
server.visual,
|
||||||
|
mask,
|
||||||
|
&attr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tooltip_init_fonts()
|
||||||
|
{
|
||||||
|
if (!g_tooltip.font_desc)
|
||||||
|
g_tooltip.font_desc = pango_font_description_from_string(get_default_font());
|
||||||
|
}
|
||||||
|
|
||||||
|
void tooltip_default_font_changed()
|
||||||
|
{
|
||||||
|
if (g_tooltip.has_font)
|
||||||
|
return;
|
||||||
|
if (!g_tooltip.has_font) {
|
||||||
|
pango_font_description_free(g_tooltip.font_desc);
|
||||||
|
g_tooltip.font_desc = NULL;
|
||||||
|
}
|
||||||
|
tooltip_init_fonts();
|
||||||
|
tooltip_update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void tooltip_trigger_show(Area* area, Panel* p, XEvent *e)
|
void tooltip_trigger_show(Area *area, Panel *p, XEvent *e)
|
||||||
{
|
{
|
||||||
// Position the tooltip in the center of the area
|
// Position the tooltip in the center of the area
|
||||||
x = area->posx + area->width / 2 + e->xmotion.x_root - e->xmotion.x;
|
x = area->posx + MIN(area->width / 3, 22) + e->xmotion.x_root - e->xmotion.x;
|
||||||
y = area->posy + area->height / 2 + e->xmotion.y_root - e->xmotion.y;
|
y = area->posy + area->height / 2 + e->xmotion.y_root - e->xmotion.y;
|
||||||
if (!panel_horizontal)
|
just_shown = TRUE;
|
||||||
y -= height/2;
|
|
||||||
g_tooltip.panel = p;
|
g_tooltip.panel = p;
|
||||||
if (g_tooltip.mapped && g_tooltip.area != area) {
|
if (g_tooltip.mapped && g_tooltip.area != area) {
|
||||||
tooltip_copy_text(area);
|
tooltip_copy_text(area);
|
||||||
tooltip_update();
|
tooltip_update();
|
||||||
stop_tooltip_timeout();
|
stop_tooltip_timeout();
|
||||||
}
|
} else if (!g_tooltip.mapped) {
|
||||||
else if (!g_tooltip.mapped) {
|
|
||||||
start_show_timeout();
|
start_show_timeout();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void tooltip_show(void *arg)
|
||||||
void tooltip_show(void* arg)
|
|
||||||
{
|
{
|
||||||
int mx, my;
|
int mx, my;
|
||||||
Window w;
|
Window w;
|
||||||
XTranslateCoordinates( server.dsp, server.root_win, g_tooltip.panel->main_win, x, y, &mx, &my, &w);
|
XTranslateCoordinates(server.display, server.root_win, g_tooltip.panel->main_win, x, y, &mx, &my, &w);
|
||||||
Area* area;
|
Area *area = click_area(g_tooltip.panel, mx, my);
|
||||||
if (!panel_horizontal)
|
|
||||||
my += height/2; /* we adjusted y in tooltip_trigger_show, revert or we won't find the correct area anymore */
|
|
||||||
area = click_area(g_tooltip.panel, mx, my);
|
|
||||||
stop_tooltip_timeout();
|
|
||||||
if (!g_tooltip.mapped && area->_get_tooltip_text) {
|
if (!g_tooltip.mapped && area->_get_tooltip_text) {
|
||||||
tooltip_copy_text(area);
|
tooltip_copy_text(area);
|
||||||
g_tooltip.mapped = True;
|
g_tooltip.mapped = True;
|
||||||
XMapWindow(server.dsp, g_tooltip.window);
|
XMapWindow(server.display, g_tooltip.window);
|
||||||
tooltip_update();
|
tooltip_update();
|
||||||
XFlush(server.dsp);
|
XFlush(server.display);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void tooltip_update_geometry()
|
void tooltip_update_geometry()
|
||||||
{
|
{
|
||||||
cairo_surface_t *cs;
|
Panel *panel = g_tooltip.panel;
|
||||||
cairo_t *c;
|
int screen_width = server.monitors[panel->monitor].x + server.monitors[panel->monitor].width;
|
||||||
PangoLayout* layout;
|
|
||||||
cs = cairo_xlib_surface_create(server.dsp, g_tooltip.window, server.visual, width, height);
|
cairo_surface_t *cs = cairo_xlib_surface_create(server.display, g_tooltip.window, server.visual, width, height);
|
||||||
c = cairo_create(cs);
|
cairo_t *c = cairo_create(cs);
|
||||||
layout = pango_cairo_create_layout(c);
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
pango_layout_set_font_description(layout, g_tooltip.font_desc);
|
|
||||||
pango_layout_set_text(layout, g_tooltip.tooltip_text, -1);
|
pango_layout_set_font_description(layout, g_tooltip.font_desc);
|
||||||
PangoRectangle r1, r2;
|
PangoRectangle r1, r2;
|
||||||
pango_layout_get_pixel_extents(layout, &r1, &r2);
|
pango_layout_set_text(layout, "1234567890", -1);
|
||||||
width = 2*g_tooltip.bg->border.width + 2*g_tooltip.paddingx + r2.width;
|
pango_layout_get_pixel_extents(layout, &r1, &r2);
|
||||||
height = 2*g_tooltip.bg->border.width + 2*g_tooltip.paddingy + r2.height;
|
int max_width = MIN(r2.width * 7, screen_width * 2 / 3);
|
||||||
|
pango_layout_set_width(layout, max_width * PANGO_SCALE);
|
||||||
|
|
||||||
|
pango_layout_set_text(layout, g_tooltip.tooltip_text, -1);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD);
|
||||||
|
pango_layout_get_pixel_extents(layout, &r1, &r2);
|
||||||
|
width = 2 * g_tooltip.bg->border.width + 2 * g_tooltip.paddingx + r2.width;
|
||||||
|
height = 2 * g_tooltip.bg->border.width + 2 * g_tooltip.paddingy + r2.height;
|
||||||
|
|
||||||
Panel* panel = g_tooltip.panel;
|
|
||||||
if (panel_horizontal && panel_position & BOTTOM)
|
if (panel_horizontal && panel_position & BOTTOM)
|
||||||
y = panel->posy-height;
|
y = panel->posy - height;
|
||||||
else if (panel_horizontal && panel_position & TOP)
|
else if (panel_horizontal && panel_position & TOP)
|
||||||
y = panel->posy + panel->area.height;
|
y = panel->posy + panel->area.height;
|
||||||
else if (panel_position & LEFT)
|
else if (panel_position & LEFT)
|
||||||
@@ -146,51 +178,50 @@ void tooltip_update_geometry()
|
|||||||
cairo_surface_destroy(cs);
|
cairo_surface_destroy(cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void tooltip_adjust_geometry()
|
void tooltip_adjust_geometry()
|
||||||
{
|
{
|
||||||
// adjust coordinates and size to not go offscreen
|
// adjust coordinates and size to not go offscreen
|
||||||
// it seems quite impossible that the height needs to be adjusted, but we do it anyway.
|
// it seems quite impossible that the height needs to be adjusted, but we do it anyway.
|
||||||
|
|
||||||
|
Panel *panel = g_tooltip.panel;
|
||||||
|
int screen_width = server.monitors[panel->monitor].x + server.monitors[panel->monitor].width;
|
||||||
|
int screen_height = server.monitors[panel->monitor].y + server.monitors[panel->monitor].height;
|
||||||
|
if (x + width <= screen_width && y + height <= screen_height && x >= server.monitors[panel->monitor].x &&
|
||||||
|
y >= server.monitors[panel->monitor].y)
|
||||||
|
return; // no adjustment needed
|
||||||
|
|
||||||
int min_x, min_y, max_width, max_height;
|
int min_x, min_y, max_width, max_height;
|
||||||
Panel* panel = g_tooltip.panel;
|
|
||||||
int screen_width = server.monitor[panel->monitor].x + server.monitor[panel->monitor].width;
|
|
||||||
int screen_height = server.monitor[panel->monitor].y + server.monitor[panel->monitor].height;
|
|
||||||
if ( x+width <= screen_width && y+height <= screen_height && x>=server.monitor[panel->monitor].x && y>=server.monitor[panel->monitor].y )
|
|
||||||
return; // no adjustment needed
|
|
||||||
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
min_x=0;
|
min_x = 0;
|
||||||
max_width=server.monitor[panel->monitor].width;
|
max_width = server.monitors[panel->monitor].width;
|
||||||
max_height=server.monitor[panel->monitor].height-panel->area.height;
|
max_height = server.monitors[panel->monitor].height - panel->area.height;
|
||||||
if (panel_position & BOTTOM)
|
if (panel_position & BOTTOM)
|
||||||
min_y=0;
|
min_y = 0;
|
||||||
else
|
else
|
||||||
min_y=panel->area.height;
|
min_y = panel->area.height;
|
||||||
}
|
} else {
|
||||||
else {
|
max_width = server.monitors[panel->monitor].width - panel->area.width;
|
||||||
max_width=server.monitor[panel->monitor].width-panel->area.width;
|
min_y = 0;
|
||||||
min_y=0;
|
max_height = server.monitors[panel->monitor].height;
|
||||||
max_height=server.monitor[panel->monitor].height;
|
|
||||||
if (panel_position & LEFT)
|
if (panel_position & LEFT)
|
||||||
min_x=panel->area.width;
|
min_x = panel->area.width;
|
||||||
else
|
else
|
||||||
min_x=0;
|
min_x = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (x+width > server.monitor[panel->monitor].x + server.monitor[panel->monitor].width)
|
if (x + width > server.monitors[panel->monitor].x + server.monitors[panel->monitor].width)
|
||||||
x = server.monitor[panel->monitor].x + server.monitor[panel->monitor].width - width;
|
x = server.monitors[panel->monitor].x + server.monitors[panel->monitor].width - width;
|
||||||
if ( y+height > server.monitor[panel->monitor].y + server.monitor[panel->monitor].height)
|
if (y + height > server.monitors[panel->monitor].y + server.monitors[panel->monitor].height)
|
||||||
y = server.monitor[panel->monitor].y + server.monitor[panel->monitor].height - height;
|
y = server.monitors[panel->monitor].y + server.monitors[panel->monitor].height - height;
|
||||||
|
|
||||||
if (x<min_x)
|
if (x < min_x)
|
||||||
x=min_x;
|
x = min_x;
|
||||||
if (width>max_width)
|
if (width > max_width)
|
||||||
width = max_width;
|
width = max_width;
|
||||||
if (y<min_y)
|
if (y < min_y)
|
||||||
y=min_y;
|
y = min_y;
|
||||||
if (height>max_height)
|
if (height > max_height)
|
||||||
height=max_height;
|
height = max_height;
|
||||||
}
|
}
|
||||||
|
|
||||||
void tooltip_update()
|
void tooltip_update()
|
||||||
@@ -201,112 +232,100 @@ void tooltip_update()
|
|||||||
}
|
}
|
||||||
|
|
||||||
tooltip_update_geometry();
|
tooltip_update_geometry();
|
||||||
|
if (just_shown) {
|
||||||
|
if (!panel_horizontal)
|
||||||
|
y -= height / 2; // center vertically
|
||||||
|
just_shown = FALSE;
|
||||||
|
}
|
||||||
tooltip_adjust_geometry();
|
tooltip_adjust_geometry();
|
||||||
XMoveResizeWindow(server.dsp, g_tooltip.window, x, y, width, height);
|
XMoveResizeWindow(server.display, g_tooltip.window, x, y, width, height);
|
||||||
|
|
||||||
// Stuff for drawing the tooltip
|
// Stuff for drawing the tooltip
|
||||||
cairo_surface_t *cs;
|
cairo_surface_t *cs = cairo_xlib_surface_create(server.display, g_tooltip.window, server.visual, width, height);
|
||||||
cairo_t *c;
|
cairo_t *c = cairo_create(cs);
|
||||||
PangoLayout* layout;
|
Color bc = g_tooltip.bg->fill_color;
|
||||||
cs = cairo_xlib_surface_create(server.dsp, g_tooltip.window, server.visual, width, height);
|
|
||||||
c = cairo_create(cs);
|
|
||||||
Color bc = g_tooltip.bg->back;
|
|
||||||
Border b = g_tooltip.bg->border;
|
Border b = g_tooltip.bg->border;
|
||||||
if (server.real_transparency) {
|
if (server.real_transparency) {
|
||||||
clear_pixmap(g_tooltip.window, 0, 0, width, height);
|
clear_pixmap(g_tooltip.window, 0, 0, width, height);
|
||||||
draw_rect(c, b.width, b.width, width-2*b.width, height-2*b.width, b.rounded-b.width/1.571);
|
draw_rect(c, b.width, b.width, width - 2 * b.width, height - 2 * b.width, b.radius - b.width / 1.571);
|
||||||
cairo_set_source_rgba(c, bc.color[0], bc.color[1], bc.color[2], bc.alpha);
|
cairo_set_source_rgba(c, bc.rgb[0], bc.rgb[1], bc.rgb[2], bc.alpha);
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
cairo_rectangle(c, 0., 0, width, height);
|
cairo_rectangle(c, 0., 0, width, height);
|
||||||
cairo_set_source_rgb(c, bc.color[0], bc.color[1], bc.color[2]);
|
cairo_set_source_rgb(c, bc.rgb[0], bc.rgb[1], bc.rgb[2]);
|
||||||
}
|
}
|
||||||
cairo_fill(c);
|
cairo_fill(c);
|
||||||
cairo_set_line_width(c, b.width);
|
cairo_set_line_width(c, b.width);
|
||||||
if (server.real_transparency)
|
if (server.real_transparency)
|
||||||
draw_rect(c, b.width/2.0, b.width/2.0, width - b.width, height - b.width, b.rounded);
|
draw_rect(c, b.width / 2.0, b.width / 2.0, width - b.width, height - b.width, b.radius);
|
||||||
else
|
else
|
||||||
cairo_rectangle(c, b.width/2.0, b.width/2.0, width-b.width, height-b.width);
|
cairo_rectangle(c, b.width / 2.0, b.width / 2.0, width - b.width, height - b.width);
|
||||||
cairo_set_source_rgba(c, b.color[0], b.color[1], b.color[2], b.alpha);
|
cairo_set_source_rgba(c, b.color.rgb[0], b.color.rgb[1], b.color.rgb[2], b.color.alpha);
|
||||||
cairo_stroke(c);
|
cairo_stroke(c);
|
||||||
|
|
||||||
Color fc = g_tooltip.font_color;
|
Color fc = g_tooltip.font_color;
|
||||||
cairo_set_source_rgba(c, fc.color[0], fc.color[1], fc.color[2], fc.alpha);
|
cairo_set_source_rgba(c, fc.rgb[0], fc.rgb[1], fc.rgb[2], fc.alpha);
|
||||||
layout = pango_cairo_create_layout(c);
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
pango_layout_set_font_description(layout, g_tooltip.font_desc);
|
pango_layout_set_font_description(layout, g_tooltip.font_desc);
|
||||||
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD);
|
||||||
pango_layout_set_text(layout, g_tooltip.tooltip_text, -1);
|
pango_layout_set_text(layout, g_tooltip.tooltip_text, -1);
|
||||||
PangoRectangle r1, r2;
|
PangoRectangle r1, r2;
|
||||||
pango_layout_get_pixel_extents(layout, &r1, &r2);
|
pango_layout_get_pixel_extents(layout, &r1, &r2);
|
||||||
pango_layout_set_width(layout, width*PANGO_SCALE);
|
pango_layout_set_width(layout, width * PANGO_SCALE);
|
||||||
pango_layout_set_height(layout, height*PANGO_SCALE);
|
pango_layout_set_height(layout, height * PANGO_SCALE);
|
||||||
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
|
||||||
// I do not know why this is the right way, but with the below cairo_move_to it seems to be centered (horiz. and vert.)
|
// I do not know why this is the right way, but with the below cairo_move_to it seems to be centered (horiz. and
|
||||||
cairo_move_to(c, -r1.x/2+g_tooltip.bg->border.width+g_tooltip.paddingx, -r1.y/2+g_tooltip.bg->border.width+g_tooltip.paddingy);
|
// vert.)
|
||||||
pango_cairo_show_layout (c, layout);
|
cairo_move_to(c,
|
||||||
|
-r1.x / 2 + g_tooltip.bg->border.width + g_tooltip.paddingx,
|
||||||
|
-r1.y / 2 + 1 + g_tooltip.bg->border.width + g_tooltip.paddingy);
|
||||||
|
pango_cairo_show_layout(c, layout);
|
||||||
|
|
||||||
g_object_unref (layout);
|
g_object_unref(layout);
|
||||||
cairo_destroy (c);
|
cairo_destroy(c);
|
||||||
cairo_surface_destroy (cs);
|
cairo_surface_destroy(cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void tooltip_trigger_hide()
|
||||||
void tooltip_trigger_hide(Tooltip* tooltip)
|
|
||||||
{
|
{
|
||||||
if (g_tooltip.mapped) {
|
if (g_tooltip.mapped) {
|
||||||
tooltip_copy_text(0);
|
tooltip_copy_text(0);
|
||||||
start_hide_timeout();
|
start_hide_timeout();
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// tooltip not visible yet, but maybe a timeout is still pending
|
// tooltip not visible yet, but maybe a timeout is still pending
|
||||||
stop_tooltip_timeout();
|
stop_tooltip_timeout();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void tooltip_hide(void *arg)
|
||||||
void tooltip_hide(void* arg)
|
|
||||||
{
|
{
|
||||||
stop_tooltip_timeout();
|
|
||||||
if (g_tooltip.mapped) {
|
if (g_tooltip.mapped) {
|
||||||
g_tooltip.mapped = False;
|
g_tooltip.mapped = False;
|
||||||
XUnmapWindow(server.dsp, g_tooltip.window);
|
XUnmapWindow(server.display, g_tooltip.window);
|
||||||
XFlush(server.dsp);
|
XFlush(server.display);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void start_show_timeout()
|
void start_show_timeout()
|
||||||
{
|
{
|
||||||
if (g_tooltip.timeout)
|
change_timeout(&g_tooltip.timeout, g_tooltip.show_timeout_msec, 0, tooltip_show, 0);
|
||||||
change_timeout(g_tooltip.timeout, g_tooltip.show_timeout_msec, 0, tooltip_show, 0);
|
|
||||||
else
|
|
||||||
g_tooltip.timeout = add_timeout(g_tooltip.show_timeout_msec, 0, tooltip_show, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void start_hide_timeout()
|
void start_hide_timeout()
|
||||||
{
|
{
|
||||||
if (g_tooltip.timeout)
|
change_timeout(&g_tooltip.timeout, g_tooltip.hide_timeout_msec, 0, tooltip_hide, 0);
|
||||||
change_timeout(g_tooltip.timeout, g_tooltip.hide_timeout_msec, 0, tooltip_hide, 0);
|
|
||||||
else
|
|
||||||
g_tooltip.timeout = add_timeout(g_tooltip.hide_timeout_msec, 0, tooltip_hide, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void stop_tooltip_timeout()
|
void stop_tooltip_timeout()
|
||||||
{
|
{
|
||||||
if (g_tooltip.timeout) {
|
stop_timeout(g_tooltip.timeout);
|
||||||
stop_timeout(g_tooltip.timeout);
|
|
||||||
g_tooltip.timeout = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void tooltip_copy_text(Area *area)
|
||||||
void tooltip_copy_text(Area* area)
|
|
||||||
{
|
{
|
||||||
free(g_tooltip.tooltip_text);
|
free(g_tooltip.tooltip_text);
|
||||||
if (area && area->_get_tooltip_text)
|
if (area && area->_get_tooltip_text)
|
||||||
g_tooltip.tooltip_text = strdup(area->_get_tooltip_text(area));
|
g_tooltip.tooltip_text = area->_get_tooltip_text(area);
|
||||||
else
|
else
|
||||||
g_tooltip.tooltip_text = 0;
|
g_tooltip.tooltip_text = NULL;
|
||||||
g_tooltip.area = area;
|
g_tooltip.area = area;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,26 +22,25 @@
|
|||||||
#include "panel.h"
|
#include "panel.h"
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
Area* area; // never ever use the area attribut if you are not 100% sure that this area was not freed
|
Area *area; // never ever use the area attribut if you are not 100% sure that this area was not freed
|
||||||
char* tooltip_text;
|
char *tooltip_text;
|
||||||
Panel* panel;
|
Panel *panel;
|
||||||
Window window;
|
Window window;
|
||||||
int show_timeout_msec;
|
int show_timeout_msec;
|
||||||
int hide_timeout_msec;
|
int hide_timeout_msec;
|
||||||
Bool mapped;
|
Bool mapped;
|
||||||
int paddingx;
|
int paddingx;
|
||||||
int paddingy;
|
int paddingy;
|
||||||
PangoFontDescription* font_desc;
|
gboolean has_font;
|
||||||
|
PangoFontDescription *font_desc;
|
||||||
Color font_color;
|
Color font_color;
|
||||||
Background* bg;
|
Background *bg;
|
||||||
timeout* timeout;
|
timeout *timeout;
|
||||||
} Tooltip;
|
} Tooltip;
|
||||||
|
|
||||||
extern Tooltip g_tooltip;
|
extern Tooltip g_tooltip;
|
||||||
|
|
||||||
|
|
||||||
// default global data
|
// default global data
|
||||||
void default_tooltip();
|
void default_tooltip();
|
||||||
|
|
||||||
@@ -49,11 +48,12 @@ void default_tooltip();
|
|||||||
void cleanup_tooltip();
|
void cleanup_tooltip();
|
||||||
|
|
||||||
void init_tooltip();
|
void init_tooltip();
|
||||||
void tooltip_trigger_show(Area* area, Panel* p, XEvent *e);
|
void tooltip_trigger_show(Area *area, Panel *p, XEvent *e);
|
||||||
void tooltip_show(void* /*arg*/);
|
void tooltip_show(void * /*arg*/);
|
||||||
void tooltip_update();
|
void tooltip_update();
|
||||||
void tooltip_trigger_hide();
|
void tooltip_trigger_hide();
|
||||||
void tooltip_hide(void* /*arg*/);
|
void tooltip_hide(void * /*arg*/);
|
||||||
void tooltip_copy_text(Area* area);
|
void tooltip_copy_text(Area *area);
|
||||||
|
void tooltip_default_font_changed();
|
||||||
|
|
||||||
#endif // TOOLTIP_H
|
#endif // TOOLTIP_H
|
||||||
|
|||||||
700
src/util/area.c
700
src/util/area.c
@@ -29,308 +29,329 @@
|
|||||||
#include "area.h"
|
#include "area.h"
|
||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "panel.h"
|
#include "panel.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
Area *mouse_over_area = NULL;
|
||||||
|
|
||||||
/************************************************************
|
void init_background(Background *bg)
|
||||||
* !!! This design is experimental and not yet fully implemented !!!!!!!!!!!!!
|
|
||||||
*
|
|
||||||
* DATA ORGANISATION :
|
|
||||||
* Areas in tint2 are similar to widgets in a GUI.
|
|
||||||
* All graphical objects (panel, taskbar, task, systray, clock, ...) 'inherit' an abstract class 'Area'.
|
|
||||||
* This class 'Area' manage the background, border, size, position and padding.
|
|
||||||
* Area is at the begining of each object (&object == &area).
|
|
||||||
*
|
|
||||||
* tint2 define one panel per monitor. And each panel have a tree of Area.
|
|
||||||
* The root of the tree is Panel.Area. And task, clock, systray, taskbar,... are nodes.
|
|
||||||
*
|
|
||||||
* The tree give the localisation of each object :
|
|
||||||
* - tree's root is in the background while tree's leafe are foreground objects
|
|
||||||
* - position of a node/Area depend on the layout : parent's position (posx, posy), size of previous brothers and parent's padding
|
|
||||||
* - size of a node/Area depend on the content (SIZE_BY_CONTENT objects) or on the layout (SIZE_BY_LAYOUT objects)
|
|
||||||
*
|
|
||||||
* DRAWING AND LAYERING ENGINE :
|
|
||||||
* Redrawing an object (like the clock) could come from an 'external event' (date change)
|
|
||||||
* or from a 'layering event' (position change).
|
|
||||||
* The following 'drawing engine' take care of :
|
|
||||||
* - posx/posy of all Area
|
|
||||||
* - 'layering event' propagation between object
|
|
||||||
* 1) browse tree SIZE_BY_CONTENT
|
|
||||||
* - resize SIZE_BY_CONTENT node : children are resized before parent
|
|
||||||
* - if 'size' changed then 'resize = 1' on the parent
|
|
||||||
* 2) browse tree SIZE_BY_LAYOUT and POSITION
|
|
||||||
* - resize SIZE_BY_LAYOUT node : parent is resized before children
|
|
||||||
* - calculate position (posx,posy) : parent is calculated before children
|
|
||||||
* - if 'position' changed then 'redraw = 1'
|
|
||||||
* 3) browse tree REDRAW
|
|
||||||
* - redraw needed objects : parent is drawn before children
|
|
||||||
*
|
|
||||||
* CONFIGURE PANEL'S LAYOUT :
|
|
||||||
* 'panel_items' parameter (in config) define the list and the order of nodes in tree's panel.
|
|
||||||
* 'panel_items = SC' define a panel with just Systray and Clock.
|
|
||||||
* So the tree 'Panel.Area' will have 2 childs (Systray and Clock).
|
|
||||||
*
|
|
||||||
************************************************************/
|
|
||||||
|
|
||||||
void init_rendering(void *obj, int pos)
|
|
||||||
{
|
{
|
||||||
Area *a = (Area*)obj;
|
memset(bg, 0, sizeof(Background));
|
||||||
|
}
|
||||||
// initialize fixed position/size
|
|
||||||
GSList *l;
|
void initialize_positions(void *obj, int offset)
|
||||||
for (l = a->list; l ; l = l->next) {
|
{
|
||||||
Area *child = ((Area*)l->data);
|
Area *a = (Area *)obj;
|
||||||
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
|
Area *child = ((Area *)l->data);
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
child->posy = pos + a->bg->border.width + a->paddingy;
|
child->posy = offset + a->bg->border.width + a->paddingy;
|
||||||
child->height = a->height - (2 * (a->bg->border.width + a->paddingy));
|
child->height = a->height - (2 * (a->bg->border.width + a->paddingy));
|
||||||
init_rendering(child, child->posy);
|
if (child->_on_change_layout)
|
||||||
}
|
child->_on_change_layout(child);
|
||||||
else {
|
initialize_positions(child, child->posy);
|
||||||
child->posx = pos + a->bg->border.width + a->paddingy;
|
} else {
|
||||||
|
child->posx = offset + a->bg->border.width + a->paddingy;
|
||||||
child->width = a->width - (2 * (a->bg->border.width + a->paddingy));
|
child->width = a->width - (2 * (a->bg->border.width + a->paddingy));
|
||||||
init_rendering(child, child->posx);
|
if (child->_on_change_layout)
|
||||||
|
child->_on_change_layout(child);
|
||||||
|
initialize_positions(child, child->posx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void relayout_fixed(Area *a)
|
||||||
void rendering(void *obj)
|
|
||||||
{
|
{
|
||||||
Panel *panel = (Panel*)obj;
|
if (!a->on_screen)
|
||||||
|
return;
|
||||||
|
|
||||||
size_by_content(&panel->area);
|
// Children are resized before the parent
|
||||||
size_by_layout(&panel->area, 0, 1);
|
GList *l;
|
||||||
|
for (l = a->children; l; l = l->next)
|
||||||
refresh(&panel->area);
|
relayout_fixed(l->data);
|
||||||
}
|
|
||||||
|
|
||||||
|
// Recalculate size
|
||||||
void size_by_content (Area *a)
|
a->_changed = 0;
|
||||||
{
|
if (a->resize_needed && a->size_mode == LAYOUT_FIXED) {
|
||||||
// don't resize hiden objects
|
a->resize_needed = 0;
|
||||||
if (!a->on_screen) return;
|
|
||||||
|
|
||||||
// children node are resized before its parent
|
|
||||||
GSList *l;
|
|
||||||
for (l = a->list; l ; l = l->next)
|
|
||||||
size_by_content(l->data);
|
|
||||||
|
|
||||||
// calculate area's size
|
|
||||||
a->on_changed = 0;
|
|
||||||
if (a->resize && a->size_mode == SIZE_BY_CONTENT) {
|
|
||||||
a->resize = 0;
|
|
||||||
|
|
||||||
if (a->_resize) {
|
if (a->_resize) {
|
||||||
if (a->_resize(a)) {
|
if (a->_resize(a)) {
|
||||||
// 'size' changed => 'resize = 1' on the parent
|
// The size hash changed => resize needed for the parent
|
||||||
((Area*)a->parent)->resize = 1;
|
if (a->parent)
|
||||||
a->on_changed = 1;
|
((Area *)a->parent)->resize_needed = 1;
|
||||||
|
a->_changed = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void relayout_dynamic(Area *a, int level)
|
||||||
void size_by_layout (Area *a, int pos, int level)
|
|
||||||
{
|
{
|
||||||
// don't resize hiden objects
|
if (!a->on_screen)
|
||||||
if (!a->on_screen) return;
|
return;
|
||||||
|
|
||||||
// parent node is resized before its children
|
// Area is resized before its children
|
||||||
// calculate area's size
|
if (a->resize_needed && a->size_mode == LAYOUT_DYNAMIC) {
|
||||||
GSList *l;
|
a->resize_needed = 0;
|
||||||
if (a->resize && a->size_mode == SIZE_BY_LAYOUT) {
|
|
||||||
a->resize = 0;
|
|
||||||
|
|
||||||
if (a->_resize) {
|
if (a->_resize) {
|
||||||
a->_resize(a);
|
a->_resize(a);
|
||||||
// resize childs with SIZE_BY_LAYOUT
|
// resize children with LAYOUT_DYNAMIC
|
||||||
for (l = a->list; l ; l = l->next) {
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
Area *child = ((Area*)l->data);
|
Area *child = ((Area *)l->data);
|
||||||
if (child->size_mode == SIZE_BY_LAYOUT && child->list)
|
if (child->size_mode == LAYOUT_DYNAMIC && child->children)
|
||||||
child->resize = 1;
|
child->resize_needed = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// update position of childs
|
// Layout children
|
||||||
pos += a->paddingxlr + a->bg->border.width;
|
if (a->children) {
|
||||||
int i=0;
|
if (a->alignment == ALIGN_LEFT) {
|
||||||
for (l = a->list; l ; l = l->next) {
|
int pos = (panel_horizontal ? a->posx : a->posy) + a->bg->border.width + a->paddingxlr;
|
||||||
Area *child = ((Area*)l->data);
|
|
||||||
if (!child->on_screen) continue;
|
|
||||||
i++;
|
|
||||||
|
|
||||||
if (panel_horizontal) {
|
|
||||||
if (pos != child->posx) {
|
|
||||||
// pos changed => redraw
|
|
||||||
child->posx = pos;
|
|
||||||
child->on_changed = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (pos != child->posy) {
|
|
||||||
// pos changed => redraw
|
|
||||||
child->posy = pos;
|
|
||||||
child->on_changed = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*// position of each visible object
|
|
||||||
int k;
|
|
||||||
for (k=0 ; k < level ; k++) printf(" ");
|
|
||||||
printf("tree level %d, object %d, pos %d, %s\n", level, i, pos, (child->size_mode == SIZE_BY_LAYOUT) ? "SIZE_BY_LAYOUT" : "SIZE_BY_CONTENT");*/
|
|
||||||
size_by_layout(child, pos, level+1);
|
|
||||||
|
|
||||||
if (panel_horizontal)
|
|
||||||
pos += child->width + a->paddingx;
|
|
||||||
else
|
|
||||||
pos += child->height + a->paddingx;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (a->on_changed) {
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
|
Area *child = ((Area *)l->data);
|
||||||
|
if (!child->on_screen)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
if (pos != child->posx) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posx = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (pos != child->posy) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posy = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
relayout_dynamic(child, level + 1);
|
||||||
|
|
||||||
|
pos += panel_horizontal ? child->width + a->paddingx : child->height + a->paddingx;
|
||||||
|
}
|
||||||
|
} else if (a->alignment == ALIGN_RIGHT) {
|
||||||
|
int pos = (panel_horizontal ? a->posx + a->width : a->posy + a->height) - a->bg->border.width - a->paddingxlr;
|
||||||
|
|
||||||
|
for (GList *l = g_list_last(a->children); l; l = l->prev) {
|
||||||
|
Area *child = ((Area *)l->data);
|
||||||
|
if (!child->on_screen)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
pos -= panel_horizontal ? child->width : child->height;
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
if (pos != child->posx) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posx = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (pos != child->posy) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posy = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
relayout_dynamic(child, level + 1);
|
||||||
|
|
||||||
|
pos -= a->paddingx;
|
||||||
|
}
|
||||||
|
} else if (a->alignment == ALIGN_CENTER) {
|
||||||
|
|
||||||
|
int children_size = 0;
|
||||||
|
|
||||||
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
|
Area *child = ((Area *)l->data);
|
||||||
|
if (!child->on_screen)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
children_size += panel_horizontal ? child->width : child->height;
|
||||||
|
children_size += (l == a->children) ? 0 : a->paddingx;
|
||||||
|
}
|
||||||
|
|
||||||
|
int pos = (panel_horizontal ? a->posx : a->posy) + a->bg->border.width + a->paddingxlr;
|
||||||
|
pos += ((panel_horizontal ? a->width : a->height) - children_size) / 2;
|
||||||
|
|
||||||
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
|
Area *child = ((Area *)l->data);
|
||||||
|
if (!child->on_screen)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (panel_horizontal) {
|
||||||
|
if (pos != child->posx) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posx = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (pos != child->posy) {
|
||||||
|
// pos changed => redraw
|
||||||
|
child->posy = pos;
|
||||||
|
child->_changed = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
relayout_dynamic(child, level + 1);
|
||||||
|
|
||||||
|
pos += panel_horizontal ? child->width + a->paddingx : child->height + a->paddingx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a->_changed) {
|
||||||
// pos/size changed
|
// pos/size changed
|
||||||
a->redraw = 1;
|
a->_redraw_needed = TRUE;
|
||||||
if (a->_on_change_layout)
|
if (a->_on_change_layout)
|
||||||
a->_on_change_layout (a);
|
a->_on_change_layout(a);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void relayout(Area *a)
|
||||||
void refresh (Area *a)
|
|
||||||
{
|
{
|
||||||
// don't draw and resize hide objects
|
relayout_fixed(a);
|
||||||
if (!a->on_screen) return;
|
relayout_dynamic(a, 1);
|
||||||
|
}
|
||||||
|
|
||||||
// don't draw transparent objects (without foreground and without background)
|
void draw_tree(Area *a)
|
||||||
if (a->redraw) {
|
{
|
||||||
a->redraw = 0;
|
if (!a->on_screen)
|
||||||
// force redraw of child
|
return;
|
||||||
//GSList *l;
|
|
||||||
//for (l = a->list ; l ; l = l->next)
|
|
||||||
//((Area*)l->data)->redraw = 1;
|
|
||||||
|
|
||||||
//printf("draw area posx %d, width %d\n", a->posx, a->width);
|
if (a->_redraw_needed) {
|
||||||
|
a->_redraw_needed = 0;
|
||||||
draw(a);
|
draw(a);
|
||||||
}
|
}
|
||||||
|
|
||||||
// draw current Area
|
if (a->pix)
|
||||||
if (a->pix == 0) printf("empty area posx %d, width %d\n", a->posx, a->width);
|
XCopyArea(server.display, a->pix, ((Panel *)a->panel)->temp_pmap, server.gc, 0, 0, a->width, a->height, a->posx, a->posy);
|
||||||
XCopyArea (server.dsp, a->pix, ((Panel *)a->panel)->temp_pmap, server.gc, 0, 0, a->width, a->height, a->posx, a->posy);
|
|
||||||
|
|
||||||
// and then refresh child object
|
for (GList *l = a->children; l; l = l->next)
|
||||||
GSList *l;
|
draw_tree((Area *)l->data);
|
||||||
for (l = a->list; l ; l = l->next)
|
|
||||||
refresh(l->data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int relayout_with_constraint(Area *a, int maximum_size)
|
||||||
int resize_by_layout(void *obj, int maximum_size)
|
|
||||||
{
|
{
|
||||||
Area *child, *a = (Area*)obj;
|
int fixed_children_count = 0;
|
||||||
int size, nb_by_content=0, nb_by_layout=0;
|
int dynamic_children_count = 0;
|
||||||
|
|
||||||
if (panel_horizontal) {
|
if (panel_horizontal) {
|
||||||
// detect free size for SIZE_BY_LAYOUT's Area
|
// detect free size for LAYOUT_DYNAMIC Areas
|
||||||
size = a->width - (2 * (a->paddingxlr + a->bg->border.width));
|
int size = a->width - (2 * (a->paddingxlr + a->bg->border.width));
|
||||||
GSList *l;
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
for (l = a->list ; l ; l = l->next) {
|
Area *child = (Area *)l->data;
|
||||||
child = (Area*)l->data;
|
if (child->on_screen && child->size_mode == LAYOUT_FIXED) {
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_CONTENT) {
|
|
||||||
size -= child->width;
|
size -= child->width;
|
||||||
nb_by_content++;
|
fixed_children_count++;
|
||||||
}
|
}
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_LAYOUT)
|
if (child->on_screen && child->size_mode == LAYOUT_DYNAMIC)
|
||||||
nb_by_layout++;
|
dynamic_children_count++;
|
||||||
}
|
}
|
||||||
//printf(" resize_by_layout Deb %d, %d\n", nb_by_content, nb_by_layout);
|
if (fixed_children_count + dynamic_children_count > 0)
|
||||||
if (nb_by_content+nb_by_layout)
|
size -= (fixed_children_count + dynamic_children_count - 1) * a->paddingx;
|
||||||
size -= ((nb_by_content+nb_by_layout-1) * a->paddingx);
|
|
||||||
|
|
||||||
int width=0, modulo=0, old_width;
|
int width = 0;
|
||||||
if (nb_by_layout) {
|
int modulo = 0;
|
||||||
width = size / nb_by_layout;
|
if (dynamic_children_count > 0) {
|
||||||
modulo = size % nb_by_layout;
|
width = size / dynamic_children_count;
|
||||||
if (width > maximum_size && maximum_size != 0) {
|
modulo = size % dynamic_children_count;
|
||||||
|
if (width > maximum_size && maximum_size > 0) {
|
||||||
width = maximum_size;
|
width = maximum_size;
|
||||||
modulo = 0;
|
modulo = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// resize SIZE_BY_LAYOUT objects
|
// Resize LAYOUT_DYNAMIC objects
|
||||||
for (l = a->list ; l ; l = l->next) {
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
child = (Area*)l->data;
|
Area *child = (Area *)l->data;
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_LAYOUT) {
|
if (child->on_screen && child->size_mode == LAYOUT_DYNAMIC) {
|
||||||
old_width = child->width;
|
int old_width = child->width;
|
||||||
child->width = width;
|
child->width = width;
|
||||||
if (modulo) {
|
if (modulo) {
|
||||||
child->width++;
|
child->width++;
|
||||||
modulo--;
|
modulo--;
|
||||||
}
|
}
|
||||||
if (child->width != old_width)
|
if (child->width != old_width)
|
||||||
child->on_changed = 1;
|
child->_changed = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
// detect free size for LAYOUT_DYNAMIC's Area
|
||||||
// detect free size for SIZE_BY_LAYOUT's Area
|
int size = a->height - (2 * (a->paddingxlr + a->bg->border.width));
|
||||||
size = a->height - (2 * (a->paddingxlr + a->bg->border.width));
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
GSList *l;
|
Area *child = (Area *)l->data;
|
||||||
for (l = a->list ; l ; l = l->next) {
|
if (child->on_screen && child->size_mode == LAYOUT_FIXED) {
|
||||||
child = (Area*)l->data;
|
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_CONTENT) {
|
|
||||||
size -= child->height;
|
size -= child->height;
|
||||||
nb_by_content++;
|
fixed_children_count++;
|
||||||
}
|
}
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_LAYOUT)
|
if (child->on_screen && child->size_mode == LAYOUT_DYNAMIC)
|
||||||
nb_by_layout++;
|
dynamic_children_count++;
|
||||||
}
|
}
|
||||||
if (nb_by_content+nb_by_layout)
|
if (fixed_children_count + dynamic_children_count > 0)
|
||||||
size -= ((nb_by_content+nb_by_layout-1) * a->paddingx);
|
size -= (fixed_children_count + dynamic_children_count - 1) * a->paddingx;
|
||||||
|
|
||||||
int height=0, modulo=0, old_height;
|
int height = 0;
|
||||||
if (nb_by_layout) {
|
int modulo = 0;
|
||||||
height = size / nb_by_layout;
|
if (dynamic_children_count) {
|
||||||
modulo = size % nb_by_layout;
|
height = size / dynamic_children_count;
|
||||||
|
modulo = size % dynamic_children_count;
|
||||||
if (height > maximum_size && maximum_size != 0) {
|
if (height > maximum_size && maximum_size != 0) {
|
||||||
height = maximum_size;
|
height = maximum_size;
|
||||||
modulo = 0;
|
modulo = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// resize SIZE_BY_LAYOUT objects
|
// Resize LAYOUT_DYNAMIC objects
|
||||||
for (l = a->list ; l ; l = l->next) {
|
for (GList *l = a->children; l; l = l->next) {
|
||||||
child = (Area*)l->data;
|
Area *child = (Area *)l->data;
|
||||||
if (child->on_screen && child->size_mode == SIZE_BY_LAYOUT) {
|
if (child->on_screen && child->size_mode == LAYOUT_DYNAMIC) {
|
||||||
old_height = child->height;
|
int old_height = child->height;
|
||||||
child->height = height;
|
child->height = height;
|
||||||
if (modulo) {
|
if (modulo) {
|
||||||
child->height++;
|
child->height++;
|
||||||
modulo--;
|
modulo--;
|
||||||
}
|
}
|
||||||
if (child->height != old_height)
|
if (child->height != old_height)
|
||||||
child->on_changed = 1;
|
child->_changed = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void schedule_redraw(Area *a)
|
||||||
void set_redraw (Area *a)
|
|
||||||
{
|
{
|
||||||
a->redraw = 1;
|
a->_redraw_needed = TRUE;
|
||||||
|
|
||||||
GSList *l;
|
if (a->has_mouse_over_effect) {
|
||||||
for (l = a->list ; l ; l = l->next)
|
for (int i = 0; i < MOUSE_STATE_COUNT; i++) {
|
||||||
set_redraw(l->data);
|
XFreePixmap(server.display, a->pix_by_state[i]);
|
||||||
|
if (a->pix == a->pix_by_state[i])
|
||||||
|
a->pix = None;
|
||||||
|
a->pix_by_state[i] = None;
|
||||||
|
}
|
||||||
|
if (a->pix) {
|
||||||
|
XFreePixmap(server.display, a->pix);
|
||||||
|
a->pix = None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (GList *l = a->children; l; l = l->next)
|
||||||
|
schedule_redraw((Area *)l->data);
|
||||||
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void hide(Area *a)
|
void hide(Area *a)
|
||||||
{
|
{
|
||||||
Area *parent = (Area*)a->parent;
|
Area *parent = (Area *)a->parent;
|
||||||
|
|
||||||
a->on_screen = 0;
|
a->on_screen = FALSE;
|
||||||
parent->resize = 1;
|
if (parent)
|
||||||
|
parent->resize_needed = 1;
|
||||||
if (panel_horizontal)
|
if (panel_horizontal)
|
||||||
a->width = 0;
|
a->width = 0;
|
||||||
else
|
else
|
||||||
@@ -339,158 +360,203 @@ void hide(Area *a)
|
|||||||
|
|
||||||
void show(Area *a)
|
void show(Area *a)
|
||||||
{
|
{
|
||||||
Area *parent = (Area*)a->parent;
|
Area *parent = (Area *)a->parent;
|
||||||
|
|
||||||
a->on_screen = 1;
|
a->on_screen = TRUE;
|
||||||
parent->resize = 1;
|
if (parent)
|
||||||
a->resize = 1;
|
parent->resize_needed = 1;
|
||||||
|
a->resize_needed = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void draw (Area *a)
|
void draw(Area *a)
|
||||||
{
|
{
|
||||||
if (a->pix) XFreePixmap (server.dsp, a->pix);
|
if (a->pix) {
|
||||||
a->pix = XCreatePixmap (server.dsp, server.root_win, a->width, a->height, server.depth);
|
XFreePixmap(server.display, a->pix);
|
||||||
|
if (a->pix_by_state[a->has_mouse_over_effect ? a->mouse_state : 0] != a->pix)
|
||||||
|
XFreePixmap(server.display, a->pix_by_state[a->has_mouse_over_effect ? a->mouse_state : 0]);
|
||||||
|
}
|
||||||
|
a->pix = XCreatePixmap(server.display, server.root_win, a->width, a->height, server.depth);
|
||||||
|
a->pix_by_state[a->has_mouse_over_effect ? a->mouse_state : 0] = a->pix;
|
||||||
|
|
||||||
// add layer of root pixmap (or clear pixmap if real_transparency==true)
|
// Add layer of root pixmap (or clear pixmap if real_transparency==true)
|
||||||
if (server.real_transparency)
|
if (server.real_transparency)
|
||||||
clear_pixmap(a->pix, 0 ,0, a->width, a->height);
|
clear_pixmap(a->pix, 0, 0, a->width, a->height);
|
||||||
XCopyArea (server.dsp, ((Panel *)a->panel)->temp_pmap, a->pix, server.gc, a->posx, a->posy, a->width, a->height, 0, 0);
|
XCopyArea(server.display, ((Panel *)a->panel)->temp_pmap, a->pix, server.gc, a->posx, a->posy, a->width, a->height, 0, 0);
|
||||||
|
|
||||||
cairo_surface_t *cs;
|
cairo_surface_t *cs = cairo_xlib_surface_create(server.display, a->pix, server.visual, a->width, a->height);
|
||||||
cairo_t *c;
|
cairo_t *c = cairo_create(cs);
|
||||||
|
|
||||||
cs = cairo_xlib_surface_create (server.dsp, a->pix, server.visual, a->width, a->height);
|
draw_background(a, c);
|
||||||
c = cairo_create (cs);
|
|
||||||
|
|
||||||
draw_background (a, c);
|
|
||||||
|
|
||||||
if (a->_draw_foreground)
|
if (a->_draw_foreground)
|
||||||
a->_draw_foreground(a, c);
|
a->_draw_foreground(a, c);
|
||||||
|
|
||||||
cairo_destroy (c);
|
cairo_destroy(c);
|
||||||
cairo_surface_destroy (cs);
|
cairo_surface_destroy(cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void draw_background(Area *a, cairo_t *c)
|
||||||
void draw_background (Area *a, cairo_t *c)
|
|
||||||
{
|
{
|
||||||
if (a->bg->back.alpha > 0.0) {
|
if (a->bg->fill_color.alpha > 0.0 ||
|
||||||
//printf(" draw_background (%d %d) RGBA (%lf, %lf, %lf, %lf)\n", a->posx, a->posy, pix->back.color[0], pix->back.color[1], pix->back.color[2], pix->back.alpha);
|
(panel_config.mouse_effects && (a->has_mouse_over_effect || a->has_mouse_press_effect))) {
|
||||||
draw_rect(c, a->bg->border.width, a->bg->border.width, a->width-(2.0 * a->bg->border.width), a->height-(2.0*a->bg->border.width), a->bg->border.rounded - a->bg->border.width/1.571);
|
if (a->mouse_state == MOUSE_OVER)
|
||||||
cairo_set_source_rgba(c, a->bg->back.color[0], a->bg->back.color[1], a->bg->back.color[2], a->bg->back.alpha);
|
cairo_set_source_rgba(c,
|
||||||
|
a->bg->fill_color_hover.rgb[0],
|
||||||
|
a->bg->fill_color_hover.rgb[1],
|
||||||
|
a->bg->fill_color_hover.rgb[2],
|
||||||
|
a->bg->fill_color_hover.alpha);
|
||||||
|
else if (a->mouse_state == MOUSE_DOWN)
|
||||||
|
cairo_set_source_rgba(c,
|
||||||
|
a->bg->fill_color_pressed.rgb[0],
|
||||||
|
a->bg->fill_color_pressed.rgb[1],
|
||||||
|
a->bg->fill_color_pressed.rgb[2],
|
||||||
|
a->bg->fill_color_pressed.alpha);
|
||||||
|
else
|
||||||
|
cairo_set_source_rgba(c,
|
||||||
|
a->bg->fill_color.rgb[0],
|
||||||
|
a->bg->fill_color.rgb[1],
|
||||||
|
a->bg->fill_color.rgb[2],
|
||||||
|
a->bg->fill_color.alpha);
|
||||||
|
draw_rect(c,
|
||||||
|
a->bg->border.width,
|
||||||
|
a->bg->border.width,
|
||||||
|
a->width - (2.0 * a->bg->border.width),
|
||||||
|
a->height - (2.0 * a->bg->border.width),
|
||||||
|
a->bg->border.radius - a->bg->border.width / 1.571);
|
||||||
cairo_fill(c);
|
cairo_fill(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (a->bg->border.width > 0 && a->bg->border.alpha > 0.0) {
|
if (a->bg->border.width > 0) {
|
||||||
cairo_set_line_width (c, a->bg->border.width);
|
cairo_set_line_width(c, a->bg->border.width);
|
||||||
|
|
||||||
// draw border inside (x, y, width, height)
|
// draw border inside (x, y, width, height)
|
||||||
draw_rect(c, a->bg->border.width/2.0, a->bg->border.width/2.0, a->width - a->bg->border.width, a->height - a->bg->border.width, a->bg->border.rounded);
|
if (a->mouse_state == MOUSE_OVER)
|
||||||
/*
|
cairo_set_source_rgba(c,
|
||||||
// convert : radian = degre * M_PI/180
|
a->bg->border_color_hover.rgb[0],
|
||||||
// definir le degrade dans un carre de (0,0) (100,100)
|
a->bg->border_color_hover.rgb[1],
|
||||||
// ensuite ce degrade est extrapoler selon le ratio width/height
|
a->bg->border_color_hover.rgb[2],
|
||||||
// dans repere (0, 0) (100, 100)
|
a->bg->border_color_hover.alpha);
|
||||||
double X0, Y0, X1, Y1, degre;
|
else if (a->mouse_state == MOUSE_DOWN)
|
||||||
// x = X * (a->width / 100), y = Y * (a->height / 100)
|
cairo_set_source_rgba(c,
|
||||||
double x0, y0, x1, y1;
|
a->bg->border_color_pressed.rgb[0],
|
||||||
X0 = 0;
|
a->bg->border_color_pressed.rgb[1],
|
||||||
Y0 = 100;
|
a->bg->border_color_pressed.rgb[2],
|
||||||
X1 = 100;
|
a->bg->border_color_pressed.alpha);
|
||||||
Y1 = 0;
|
else
|
||||||
degre = 45;
|
cairo_set_source_rgba(c,
|
||||||
// et ensuite faire la changement d'unite du repere
|
a->bg->border.color.rgb[0],
|
||||||
// car ce qui doit reste inchangee est les traits et pas la direction
|
a->bg->border.color.rgb[1],
|
||||||
|
a->bg->border.color.rgb[2],
|
||||||
|
a->bg->border.color.alpha);
|
||||||
|
draw_rect(c,
|
||||||
|
a->bg->border.width / 2.0,
|
||||||
|
a->bg->border.width / 2.0,
|
||||||
|
a->width - a->bg->border.width,
|
||||||
|
a->height - a->bg->border.width,
|
||||||
|
a->bg->border.radius);
|
||||||
|
|
||||||
// il faut d'abord appliquer une rotation de 90 (et -180 si l'angle est superieur a 180)
|
cairo_stroke(c);
|
||||||
// ceci peut etre applique une fois pour toute au depart
|
|
||||||
// ensuite calculer l'angle dans le nouveau repare
|
|
||||||
// puis faire une rotation de 90
|
|
||||||
x0 = X0 * ((double)a->width / 100);
|
|
||||||
x1 = X1 * ((double)a->width / 100);
|
|
||||||
y0 = Y0 * ((double)a->height / 100);
|
|
||||||
y1 = Y1 * ((double)a->height / 100);
|
|
||||||
|
|
||||||
x0 = X0 * ((double)a->height / 100);
|
|
||||||
x1 = X1 * ((double)a->height / 100);
|
|
||||||
y0 = Y0 * ((double)a->width / 100);
|
|
||||||
y1 = Y1 * ((double)a->width / 100);
|
|
||||||
|
|
||||||
cairo_pattern_t *linpat;
|
|
||||||
linpat = cairo_pattern_create_linear (x0, y0, x1, y1);
|
|
||||||
cairo_pattern_add_color_stop_rgba (linpat, 0, a->border.color[0], a->border.color[1], a->border.color[2], a->border.alpha);
|
|
||||||
cairo_pattern_add_color_stop_rgba (linpat, 1, a->border.color[0], a->border.color[1], a->border.color[2], 0);
|
|
||||||
cairo_set_source (c, linpat);
|
|
||||||
*/
|
|
||||||
cairo_set_source_rgba (c, a->bg->border.color[0], a->bg->border.color[1], a->bg->border.color[2], a->bg->border.alpha);
|
|
||||||
|
|
||||||
cairo_stroke (c);
|
|
||||||
//cairo_pattern_destroy (linpat);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void remove_area(Area *a)
|
||||||
void remove_area (Area *a)
|
|
||||||
{
|
{
|
||||||
Area *parent = (Area*)a->parent;
|
Area *area = (Area *)a;
|
||||||
|
Area *parent = (Area *)area->parent;
|
||||||
|
|
||||||
parent->list = g_slist_remove(parent->list, a);
|
if (parent) {
|
||||||
set_redraw (parent);
|
parent->children = g_list_remove(parent->children, area);
|
||||||
|
parent->resize_needed = TRUE;
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
schedule_redraw(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mouse_over_area == a) {
|
||||||
|
mouse_out();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void add_area(Area *a, Area *parent)
|
||||||
void add_area (Area *a)
|
|
||||||
{
|
{
|
||||||
Area *parent = (Area*)a->parent;
|
g_assert_null(a->parent);
|
||||||
|
|
||||||
parent->list = g_slist_append(parent->list, a);
|
|
||||||
set_redraw (parent);
|
|
||||||
|
|
||||||
|
a->parent = parent;
|
||||||
|
if (parent) {
|
||||||
|
parent->children = g_list_append(parent->children, a);
|
||||||
|
parent->resize_needed = TRUE;
|
||||||
|
schedule_redraw(parent);
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void free_area(Area *a)
|
||||||
void free_area (Area *a)
|
|
||||||
{
|
{
|
||||||
GSList *l0;
|
if (!a)
|
||||||
for (l0 = a->list; l0 ; l0 = l0->next)
|
return;
|
||||||
free_area (l0->data);
|
|
||||||
|
|
||||||
if (a->list) {
|
for (GList *l = a->children; l; l = l->next)
|
||||||
g_slist_free(a->list);
|
free_area(l->data);
|
||||||
a->list = 0;
|
|
||||||
|
if (a->children) {
|
||||||
|
g_list_free(a->children);
|
||||||
|
a->children = NULL;
|
||||||
|
}
|
||||||
|
for (int i = 0; i < MOUSE_STATE_COUNT; i++) {
|
||||||
|
XFreePixmap(server.display, a->pix_by_state[i]);
|
||||||
|
if (a->pix == a->pix_by_state[i]) {
|
||||||
|
a->pix = None;
|
||||||
|
}
|
||||||
|
a->pix_by_state[i] = None;
|
||||||
}
|
}
|
||||||
if (a->pix) {
|
if (a->pix) {
|
||||||
XFreePixmap (server.dsp, a->pix);
|
XFreePixmap(server.display, a->pix);
|
||||||
a->pix = 0;
|
a->pix = None;
|
||||||
|
}
|
||||||
|
if (mouse_over_area == a) {
|
||||||
|
mouse_over_area = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void mouse_over(Area *area, int pressed)
|
||||||
void draw_rect(cairo_t *c, double x, double y, double w, double h, double r)
|
|
||||||
{
|
{
|
||||||
if (r > 0.0) {
|
if (mouse_over_area == area && !area)
|
||||||
double c1 = 0.55228475 * r;
|
return;
|
||||||
|
|
||||||
cairo_move_to(c, x+r, y);
|
MouseState new_state = MOUSE_NORMAL;
|
||||||
cairo_rel_line_to(c, w-2*r, 0);
|
if (area) {
|
||||||
cairo_rel_curve_to(c, c1, 0.0, r, c1, r, r);
|
if (!pressed) {
|
||||||
cairo_rel_line_to(c, 0, h-2*r);
|
new_state = area->has_mouse_over_effect ? MOUSE_OVER : MOUSE_NORMAL;
|
||||||
cairo_rel_curve_to(c, 0.0, c1, c1-r, r, -r, r);
|
} else {
|
||||||
cairo_rel_line_to (c, -w +2*r, 0);
|
new_state = area->has_mouse_press_effect
|
||||||
cairo_rel_curve_to (c, -c1, 0, -r, -c1, -r, -r);
|
? MOUSE_DOWN
|
||||||
cairo_rel_line_to (c, 0, -h + 2 * r);
|
: area->has_mouse_over_effect
|
||||||
cairo_rel_curve_to (c, 0, -c1, r - c1, -r, r, -r);
|
? MOUSE_OVER
|
||||||
|
: MOUSE_NORMAL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
|
||||||
cairo_rectangle(c, x, y, w, h);
|
if (mouse_over_area == area && mouse_over_area->mouse_state == new_state)
|
||||||
|
return;
|
||||||
|
mouse_out();
|
||||||
|
if (new_state == MOUSE_NORMAL)
|
||||||
|
return;
|
||||||
|
mouse_over_area = area;
|
||||||
|
|
||||||
|
mouse_over_area->mouse_state = new_state;
|
||||||
|
mouse_over_area->pix = mouse_over_area->pix_by_state[mouse_over_area->has_mouse_over_effect ? mouse_over_area->mouse_state : 0];
|
||||||
|
if (!mouse_over_area->pix)
|
||||||
|
mouse_over_area->_redraw_needed = TRUE;
|
||||||
|
panel_refresh = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void mouse_out()
|
||||||
void clear_pixmap(Pixmap p, int x, int y, int w, int h)
|
|
||||||
{
|
{
|
||||||
Picture pict = XRenderCreatePicture(server.dsp, p, XRenderFindVisualFormat(server.dsp, server.visual), 0, 0);
|
if (!mouse_over_area)
|
||||||
XRenderColor col = { .red=0, .green=0, .blue=0, .alpha=0 };
|
return;
|
||||||
XRenderFillRectangle(server.dsp, PictOpSrc, pict, &col, x, y, w, h);
|
mouse_over_area->mouse_state = MOUSE_NORMAL;
|
||||||
XRenderFreePicture(server.dsp, pict);
|
mouse_over_area->pix = mouse_over_area->pix_by_state[mouse_over_area->has_mouse_over_effect ? mouse_over_area->mouse_state : 0];
|
||||||
|
if (!mouse_over_area->pix)
|
||||||
|
mouse_over_area->_redraw_needed = TRUE;
|
||||||
|
panel_refresh = TRUE;
|
||||||
|
mouse_over_area = NULL;
|
||||||
}
|
}
|
||||||
|
|||||||
314
src/util/area.h
314
src/util/area.h
@@ -1,19 +1,6 @@
|
|||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
* Copyright (C) 2008 thierry lorthiois (lorthiois@bbsoft.fr)
|
* Copyright (C) 2008 thierry lorthiois (lorthiois@bbsoft.fr)
|
||||||
*
|
*
|
||||||
* base class for all graphical objects (panel, taskbar, task, systray, clock, ...).
|
|
||||||
* Area is at the begining of each object (&object == &area).
|
|
||||||
*
|
|
||||||
* Area manage the background and border drawing, size and padding.
|
|
||||||
* Each Area has one Pixmap (pix).
|
|
||||||
*
|
|
||||||
* Area manage the tree of all objects. Parent object drawn before child object.
|
|
||||||
* panel -> taskbars -> tasks
|
|
||||||
* -> systray -> icons
|
|
||||||
* -> clock
|
|
||||||
*
|
|
||||||
* draw_foreground(obj) and resize(obj) are virtual function.
|
|
||||||
*
|
|
||||||
**************************************************************************/
|
**************************************************************************/
|
||||||
|
|
||||||
#ifndef AREA_H
|
#ifndef AREA_H
|
||||||
@@ -24,104 +11,261 @@
|
|||||||
#include <cairo.h>
|
#include <cairo.h>
|
||||||
#include <cairo-xlib.h>
|
#include <cairo-xlib.h>
|
||||||
|
|
||||||
|
// DATA ORGANISATION
|
||||||
|
//
|
||||||
|
// Areas in tint2 are similar to widgets in a GUI.
|
||||||
|
// All graphical objects (panel, taskbar, task, systray, clock, ...) inherit the abstract class Area.
|
||||||
|
// This class 'Area' stores data about the background, border, size, position, padding and the child areas.
|
||||||
|
// Inheritance is simulated by having an Area member as the first member of each object (thus &object == &area).
|
||||||
|
//
|
||||||
|
// tint2 uses multiple panels, one per monitor. Each panel has an area containing the children objects in a tree of
|
||||||
|
// areas. The level in the tree gives the z-order: child areas are always displayed on top of their parents.
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// LAYOUT
|
||||||
|
//
|
||||||
|
// Sibling areas never overlap.
|
||||||
|
//
|
||||||
|
// The position of an Area (posx, posy) is relative to the window (i.e. absolute) and
|
||||||
|
// is computed based on a simple box model:
|
||||||
|
// * parent position + parent padding + sum of the sizes of the previous siblings and spacing
|
||||||
|
//
|
||||||
|
// The size of an Area is:
|
||||||
|
// * SIZE_BY_CONTENT objects:
|
||||||
|
// * fixed and set by the Area
|
||||||
|
// * childred are resized before the parent
|
||||||
|
// * if a child size has changed then the parent is resized
|
||||||
|
// * SIZE_BY_LAYOUT objects:
|
||||||
|
// * expandable and computed as the total size of the parent - padding -
|
||||||
|
// the size of the fixed sized siblings - spacing and divided by the number of expandable siblings
|
||||||
|
// * the parent is resized before the children
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// RENDERING
|
||||||
|
//
|
||||||
|
// Redrawing an object (like the clock) could come from an 'external event' (date change)
|
||||||
|
// or from a 'layout event' (position change).
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// WIDGET LIFECYCLE
|
||||||
|
//
|
||||||
|
// Each widget that occurs once per panel is defined as a struct (e.g. Clock) which is stored as a member of Panel.
|
||||||
|
// Widgets that can occur more than once should be stored as an array, still as a member of Panel.
|
||||||
|
//
|
||||||
|
// There is a special Panel instance called 'panel_config' which stores the config options and the state variables
|
||||||
|
// of the widgets (however some config options are stored as global variables by the widgets).
|
||||||
|
//
|
||||||
|
// Tint2 maintains an array of Panel instances, one for each monitor. These contain the actual Areas that are used to
|
||||||
|
// render the panels on screen, interact with user input etc.
|
||||||
|
// Each Panel is initialized as a raw copy (memcpy, see init_panel()) of panel_config.
|
||||||
|
//
|
||||||
|
// Normally, widgets should implement the following functions:
|
||||||
|
//
|
||||||
|
// * void default_widget();
|
||||||
|
//
|
||||||
|
// Called before the config is read and panel_config/panels are created.
|
||||||
|
// Afterwards, the config parsing code creates the widget/widget array in panel_config and
|
||||||
|
// populates the configuration fields.
|
||||||
|
// If the widget uses global variables to store config options or other state variables, they should be initialized
|
||||||
|
// here (e.g. with zero, NULL etc).
|
||||||
|
//
|
||||||
|
// * void init_widget();
|
||||||
|
//
|
||||||
|
// Called after the config is read and panel_config is populated, but before panels are created.
|
||||||
|
// Initializes the state of the widget in panel_config.
|
||||||
|
// If the widget uses global variables to store config options or other state variables which depend on the config
|
||||||
|
// options but not on the panel instance, they should be initialized here.
|
||||||
|
// panel_config.panel_items can be used to determine which backend items are enabled.
|
||||||
|
//
|
||||||
|
// * void init_widget_panel(void *panel);
|
||||||
|
//
|
||||||
|
// Called after each on-screen panel is created, with a pointer to the panel.
|
||||||
|
// Completes the initialization of the widget.
|
||||||
|
// At this point the widget Area has not been added yet to the GUI tree, but it will be added right afterwards.
|
||||||
|
//
|
||||||
|
// * void cleanup_widget();
|
||||||
|
//
|
||||||
|
// Called just before the panels are destroyed. Afterwards, tint2 exits or restarts and reads the config again.
|
||||||
|
// Must releases all resources.
|
||||||
|
// The widget itself should not be freed by this function, only its members or global variables that were set.
|
||||||
|
// The widget is freed by the Area tree cleanup function (remove_area).
|
||||||
|
//
|
||||||
|
// * void draw_widget(void *obj, cairo_t *c);
|
||||||
|
//
|
||||||
|
// Called on draw, obj = pointer to the widget instance from the panel that is redrawn.
|
||||||
|
// The Area's _draw_foreground member must point to this function.
|
||||||
|
//
|
||||||
|
// * int resize_widget(void *obj);
|
||||||
|
//
|
||||||
|
// Called on resize, obj = pointer to the front-end Execp item.
|
||||||
|
// Returns 1 if the new size is different than the previous size.
|
||||||
|
// The Area's _resize member must point to this function.
|
||||||
|
//
|
||||||
|
// * void widget_action(void *obj, int button);
|
||||||
|
//
|
||||||
|
// Called on mouse click event.
|
||||||
|
//
|
||||||
|
// * void widget_on_change_layout(void *obj);
|
||||||
|
//
|
||||||
|
// Implemented only to override the default layout algorithm for this widget.
|
||||||
|
// For example, if this widget is a cell in a table, its position and size should be computed here.
|
||||||
|
// The Area's _on_change_layout member must point to this function.
|
||||||
|
//
|
||||||
|
// * char* widget_get_tooltip_text(void *obj);
|
||||||
|
//
|
||||||
|
// Returns a copy of the tooltip to be displayed for this widget.
|
||||||
|
// The caller takes ownership of the pointer.
|
||||||
|
// The Area's _get_tooltip_text member must point to this function.
|
||||||
|
|
||||||
typedef struct
|
typedef struct Color {
|
||||||
{
|
// Values are in [0, 1], with 0 meaning no intensity.
|
||||||
double color[3];
|
double rgb[3];
|
||||||
double alpha;
|
// Values are in [0, 1], with 0 meaning fully transparent, 1 meaning fully opaque.
|
||||||
int width;
|
|
||||||
int rounded;
|
|
||||||
} Border;
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct
|
|
||||||
{
|
|
||||||
double color[3];
|
|
||||||
double alpha;
|
double alpha;
|
||||||
} Color;
|
} Color;
|
||||||
|
|
||||||
typedef struct
|
typedef struct Border {
|
||||||
{
|
// It's essential that the first member is color
|
||||||
Color back;
|
Color color;
|
||||||
|
// Width in pixels
|
||||||
|
int width;
|
||||||
|
// Corner radius
|
||||||
|
int radius;
|
||||||
|
} Border;
|
||||||
|
|
||||||
|
typedef struct Background {
|
||||||
|
// Normal state
|
||||||
|
Color fill_color;
|
||||||
Border border;
|
Border border;
|
||||||
|
// On mouse hover
|
||||||
|
Color fill_color_hover;
|
||||||
|
Color border_color_hover;
|
||||||
|
// On mouse press
|
||||||
|
Color fill_color_pressed;
|
||||||
|
Color border_color_pressed;
|
||||||
} Background;
|
} Background;
|
||||||
|
|
||||||
|
typedef enum Layout {
|
||||||
|
LAYOUT_DYNAMIC,
|
||||||
|
LAYOUT_FIXED,
|
||||||
|
} Layout;
|
||||||
|
|
||||||
// way to calculate the size
|
typedef enum Alignment {
|
||||||
// SIZE_BY_LAYOUT objects : taskbar and task
|
ALIGN_LEFT = 0,
|
||||||
// SIZE_BY_CONTENT objects : clock, battery, launcher, systray
|
ALIGN_CENTER = 1,
|
||||||
enum { SIZE_BY_LAYOUT, SIZE_BY_CONTENT };
|
ALIGN_RIGHT = 2,
|
||||||
|
} Alignment;
|
||||||
|
|
||||||
typedef struct {
|
typedef enum MouseState {
|
||||||
// coordinate relative to panel window
|
MOUSE_NORMAL = 0,
|
||||||
|
MOUSE_OVER = 1,
|
||||||
|
MOUSE_DOWN = 2,
|
||||||
|
MOUSE_STATE_COUNT
|
||||||
|
} MouseState;
|
||||||
|
|
||||||
|
struct Panel;
|
||||||
|
|
||||||
|
typedef struct Area {
|
||||||
|
// Position relative to the panel window
|
||||||
int posx, posy;
|
int posx, posy;
|
||||||
// width and height including border
|
// Size, including borders
|
||||||
int width, height;
|
int width, height;
|
||||||
Pixmap pix;
|
|
||||||
Background *bg;
|
Background *bg;
|
||||||
|
// List of children, each one a pointer to Area
|
||||||
// list of child : Area object
|
GList *children;
|
||||||
GSList *list;
|
// Pointer to the parent Area or NULL
|
||||||
|
|
||||||
// object visible on screen.
|
|
||||||
// An object (like systray) could be enabled but hidden (because no tray icon).
|
|
||||||
int on_screen;
|
|
||||||
// way to calculate the size (SIZE_BY_CONTENT or SIZE_BY_LAYOUT)
|
|
||||||
int size_mode;
|
|
||||||
// need to calculate position and width
|
|
||||||
int resize;
|
|
||||||
// need redraw Pixmap
|
|
||||||
int redraw;
|
|
||||||
// paddingxlr = horizontal padding left/right
|
|
||||||
// paddingx = horizontal padding between childs
|
|
||||||
int paddingxlr, paddingx, paddingy;
|
|
||||||
// parent Area
|
|
||||||
void *parent;
|
void *parent;
|
||||||
// panel
|
// Pointer to the Panel that contains this Area
|
||||||
void *panel;
|
void *panel;
|
||||||
|
Layout size_mode;
|
||||||
|
Alignment alignment;
|
||||||
|
gboolean has_mouse_over_effect;
|
||||||
|
gboolean has_mouse_press_effect;
|
||||||
|
// TODO padding/spacing is a clusterfuck
|
||||||
|
// paddingxlr = padding
|
||||||
|
// paddingy = vertical padding, sometimes
|
||||||
|
// paddingx = spacing
|
||||||
|
int paddingxlr, paddingx, paddingy;
|
||||||
|
MouseState mouse_state;
|
||||||
|
// Set to non-zero if the Area is visible. An object may exist but stay hidden.
|
||||||
|
gboolean on_screen;
|
||||||
|
// Set to non-zero if the size of the Area has to be recalculated.
|
||||||
|
gboolean resize_needed;
|
||||||
|
// Set to non-zero if the Area has to be redrawn.
|
||||||
|
// Do not set this directly; use schedule_redraw() instead.
|
||||||
|
gboolean _redraw_needed;
|
||||||
|
// Set to non-zero if the position/size has changed, thus _on_change_layout needs to be called
|
||||||
|
gboolean _changed;
|
||||||
|
// This is the pixmap on which the Area is rendered. Render to it directly if needed.
|
||||||
|
Pixmap pix;
|
||||||
|
Pixmap pix_by_state[MOUSE_STATE_COUNT];
|
||||||
|
|
||||||
// each object can overwrite following function
|
// Callbacks
|
||||||
|
|
||||||
|
// Called on draw, obj = pointer to the Area
|
||||||
void (*_draw_foreground)(void *obj, cairo_t *c);
|
void (*_draw_foreground)(void *obj, cairo_t *c);
|
||||||
// update area's content and update size (width/heith).
|
|
||||||
// return '1' if size changed, '0' otherwise.
|
// Called on resize, obj = pointer to the Area
|
||||||
int (*_resize)(void *obj);
|
// Returns 1 if the new size is different than the previous size.
|
||||||
// after pos/size changed, the rendering engine will call _on_change_layout(Area*)
|
gboolean (*_resize)(void *obj);
|
||||||
int on_changed;
|
|
||||||
|
// Implemented only to override the default layout algorithm for this widget.
|
||||||
|
// For example, if this widget is a cell in a table, its position and size should be computed here.
|
||||||
void (*_on_change_layout)(void *obj);
|
void (*_on_change_layout)(void *obj);
|
||||||
const char* (*_get_tooltip_text)(void *obj);
|
|
||||||
|
// Returns a copy of the tooltip to be displayed for this widget.
|
||||||
|
// The caller takes ownership of the pointer.
|
||||||
|
char *(*_get_tooltip_text)(void *obj);
|
||||||
} Area;
|
} Area;
|
||||||
|
|
||||||
// on startup, initialize fixed pos/size
|
// Initializes the Background member to default values.
|
||||||
void init_rendering(void *obj, int pos);
|
void init_background(Background *bg);
|
||||||
|
|
||||||
void rendering(void *obj);
|
// Layout
|
||||||
void size_by_content (Area *a);
|
|
||||||
void size_by_layout (Area *a, int pos, int level);
|
|
||||||
// draw background and foreground
|
|
||||||
void refresh (Area *a);
|
|
||||||
|
|
||||||
// generic resize for SIZE_BY_LAYOUT objects
|
|
||||||
int resize_by_layout(void *obj, int maximum_size);
|
|
||||||
|
|
||||||
// set 'redraw' on an area and childs
|
// Called on startup to initialize the positions of all Areas in the Area tree.
|
||||||
void set_redraw (Area *a);
|
// Parameters:
|
||||||
|
// * obj: pointer to Area
|
||||||
|
// * offset: offset in pixels from left/top, relative to the window
|
||||||
|
void initialize_positions(void *obj, int offset);
|
||||||
|
|
||||||
// hide/unhide area
|
// Relayouts the Area and its children. Normally called on the root of the tree (i.e. the Panel).
|
||||||
|
void relayout(Area *a);
|
||||||
|
|
||||||
|
// Distributes the Area's size to its children, repositioning them as needed.
|
||||||
|
// If maximum_size > 0, it is an upper limit for the child size.
|
||||||
|
int relayout_with_constraint(Area *a, int maximum_size);
|
||||||
|
|
||||||
|
// Rendering
|
||||||
|
|
||||||
|
// Sets the redraw_needed flag on the area and its descendants
|
||||||
|
void schedule_redraw(Area *a);
|
||||||
|
|
||||||
|
// Recreates the Area pixmap and draws the background and the foreground
|
||||||
|
void draw(Area *a);
|
||||||
|
|
||||||
|
// Draws the background of the Area
|
||||||
|
void draw_background(Area *a, cairo_t *c);
|
||||||
|
|
||||||
|
// Explores the entire Area subtree (only if the on_screen flag set)
|
||||||
|
// and draws the areas with the redraw_needed flag set
|
||||||
|
void draw_tree(Area *a);
|
||||||
|
|
||||||
|
// Clears the on_screen flag, sets the size to zero and triggers a parent resize
|
||||||
void hide(Area *a);
|
void hide(Area *a);
|
||||||
|
|
||||||
|
// Sets the on_screen flag and triggers a parent and area resize
|
||||||
void show(Area *a);
|
void show(Area *a);
|
||||||
|
|
||||||
// draw pixmap
|
// Area tree
|
||||||
void draw (Area *a);
|
|
||||||
void draw_background (Area *a, cairo_t *c);
|
|
||||||
|
|
||||||
void remove_area (Area *a);
|
void add_area(Area *a, Area *parent);
|
||||||
void add_area (Area *a);
|
void remove_area(Area *a);
|
||||||
void free_area (Area *a);
|
void free_area(Area *a);
|
||||||
|
|
||||||
// draw rounded rectangle
|
// Mouse move events
|
||||||
void draw_rect(cairo_t *c, double x, double y, double w, double h, double r);
|
|
||||||
|
void mouse_over(Area *area, int pressed);
|
||||||
|
void mouse_out();
|
||||||
|
|
||||||
// clear pixmap with transparent color
|
|
||||||
void clear_pixmap(Pixmap p, int x, int y, int w, int h);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -25,166 +25,219 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <math.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <glib.h>
|
||||||
|
#include <glib/gstdio.h>
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "../server.h"
|
#include "../server.h"
|
||||||
|
#include <sys/wait.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/resource.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <sys/sysctl.h>
|
||||||
|
|
||||||
|
#ifdef HAVE_RSVG
|
||||||
|
#include <librsvg/rsvg.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void copy_file(const char *path_src, const char *path_dest)
|
||||||
void copy_file(const char *pathSrc, const char *pathDest)
|
|
||||||
{
|
{
|
||||||
FILE *fileSrc, *fileDest;
|
if (g_str_equal(path_src, path_dest))
|
||||||
char line[100];
|
return;
|
||||||
int nb;
|
|
||||||
|
|
||||||
fileSrc = fopen(pathSrc, "rb");
|
FILE *file_src, *file_dest;
|
||||||
if (fileSrc == NULL) return;
|
char buffer[4096];
|
||||||
|
int nb;
|
||||||
|
|
||||||
fileDest = fopen(pathDest, "wb");
|
file_src = fopen(path_src, "rb");
|
||||||
if (fileDest == NULL) return;
|
if (file_src == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
while ((nb = fread(line, 1, 100, fileSrc)) > 0)
|
file_dest = fopen(path_dest, "wb");
|
||||||
if ( nb != fwrite(line, 1, nb, fileDest))
|
if (file_dest == NULL) {
|
||||||
printf("Error while copying file %s to %s\n", pathSrc, pathDest);
|
fclose(file_src);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
fclose (fileDest);
|
while ((nb = fread(buffer, 1, sizeof(buffer), file_src)) > 0) {
|
||||||
fclose (fileSrc);
|
if (nb != fwrite(buffer, 1, nb, file_dest)) {
|
||||||
|
printf("Error while copying file %s to %s\n", path_src, path_dest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(file_dest);
|
||||||
|
fclose(file_src);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int parse_line(const char *line, char **key, char **value)
|
||||||
int parse_line (const char *line, char **key, char **value)
|
|
||||||
{
|
{
|
||||||
char *a, *b;
|
char *a, *b;
|
||||||
|
|
||||||
/* Skip useless lines */
|
/* Skip useless lines */
|
||||||
if ((line[0] == '#') || (line[0] == '\n')) return 0;
|
if ((line[0] == '#') || (line[0] == '\n'))
|
||||||
if (!(a = strchr (line, '='))) return 0;
|
return 0;
|
||||||
|
if (!(a = strchr(line, '=')))
|
||||||
|
return 0;
|
||||||
|
|
||||||
/* overwrite '=' with '\0' */
|
/* overwrite '=' with '\0' */
|
||||||
a[0] = '\0';
|
a[0] = '\0';
|
||||||
*key = strdup (line);
|
*key = strdup(line);
|
||||||
a++;
|
a++;
|
||||||
|
|
||||||
/* overwrite '\n' with '\0' if '\n' present */
|
/* overwrite '\n' with '\0' if '\n' present */
|
||||||
if ((b = strchr (a, '\n'))) b[0] = '\0';
|
if ((b = strchr(a, '\n')))
|
||||||
|
b[0] = '\0';
|
||||||
|
|
||||||
*value = strdup (a);
|
*value = strdup(a);
|
||||||
|
|
||||||
g_strstrip(*key);
|
g_strstrip(*key);
|
||||||
g_strstrip(*value);
|
g_strstrip(*value);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void tint_exec(const char *command)
|
void tint_exec(const char *command)
|
||||||
{
|
{
|
||||||
if (command) {
|
if (command) {
|
||||||
pid_t pid;
|
if (fork() == 0) {
|
||||||
pid = fork();
|
|
||||||
if (pid == 0) {
|
|
||||||
// change for the fork the signal mask
|
// change for the fork the signal mask
|
||||||
// sigset_t sigset;
|
// sigset_t sigset;
|
||||||
// sigprocmask(SIG_SETMASK, &sigset, 0);
|
// sigprocmask(SIG_SETMASK, &sigset, 0);
|
||||||
// sigprocmask(SIG_UNBLOCK, &sigset, 0);
|
// sigprocmask(SIG_UNBLOCK, &sigset, 0);
|
||||||
execl("/bin/sh", "/bin/sh", "-c", command, NULL);
|
execl("/bin/sh", "/bin/sh", "-c", command, NULL);
|
||||||
_exit(0);
|
_exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *expand_tilde(char *s)
|
||||||
|
{
|
||||||
|
const gchar *home = g_get_home_dir();
|
||||||
|
if (home && (strcmp(s, "~") == 0 || strstr(s, "~/") == s)) {
|
||||||
|
char *result = calloc(strlen(home) + strlen(s), 1);
|
||||||
|
strcat(result, home);
|
||||||
|
strcat(result, s + 1);
|
||||||
|
return result;
|
||||||
|
} else {
|
||||||
|
return strdup(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int hex_char_to_int (char c)
|
char *contract_tilde(char *s)
|
||||||
|
{
|
||||||
|
const gchar *home = g_get_home_dir();
|
||||||
|
if (!home)
|
||||||
|
return strdup(s);
|
||||||
|
|
||||||
|
char *home_slash = calloc(strlen(home) + 2, 1);
|
||||||
|
strcat(home_slash, home);
|
||||||
|
strcat(home_slash, "/");
|
||||||
|
|
||||||
|
if ((strcmp(s, home) == 0 || strstr(s, home_slash) == s)) {
|
||||||
|
char *result = calloc(strlen(s) - strlen(home) + 2, 1);
|
||||||
|
strcat(result, "~");
|
||||||
|
strcat(result, s + strlen(home));
|
||||||
|
free(home_slash);
|
||||||
|
return result;
|
||||||
|
} else {
|
||||||
|
free(home_slash);
|
||||||
|
return strdup(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int hex_char_to_int(char c)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
if (c >= '0' && c <= '9') r = c - '0';
|
if (c >= '0' && c <= '9')
|
||||||
else if (c >= 'a' && c <= 'f') r = c - 'a' + 10;
|
r = c - '0';
|
||||||
else if (c >= 'A' && c <= 'F') r = c - 'A' + 10;
|
else if (c >= 'a' && c <= 'f')
|
||||||
else r = 0;
|
r = c - 'a' + 10;
|
||||||
|
else if (c >= 'A' && c <= 'F')
|
||||||
|
r = c - 'A' + 10;
|
||||||
|
else
|
||||||
|
r = 0;
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int hex_to_rgb(char *hex, int *r, int *g, int *b)
|
||||||
int hex_to_rgb (char *hex, int *r, int *g, int *b)
|
|
||||||
{
|
{
|
||||||
int len;
|
if (hex == NULL || hex[0] != '#')
|
||||||
|
return (0);
|
||||||
|
|
||||||
if (hex == NULL || hex[0] != '#') return (0);
|
int len = strlen(hex);
|
||||||
|
|
||||||
len = strlen (hex);
|
|
||||||
if (len == 3 + 1) {
|
if (len == 3 + 1) {
|
||||||
*r = hex_char_to_int (hex[1]);
|
*r = hex_char_to_int(hex[1]);
|
||||||
*g = hex_char_to_int (hex[2]);
|
*g = hex_char_to_int(hex[2]);
|
||||||
*b = hex_char_to_int (hex[3]);
|
*b = hex_char_to_int(hex[3]);
|
||||||
}
|
} else if (len == 6 + 1) {
|
||||||
else if (len == 6 + 1) {
|
*r = hex_char_to_int(hex[1]) * 16 + hex_char_to_int(hex[2]);
|
||||||
*r = hex_char_to_int (hex[1]) * 16 + hex_char_to_int (hex[2]);
|
*g = hex_char_to_int(hex[3]) * 16 + hex_char_to_int(hex[4]);
|
||||||
*g = hex_char_to_int (hex[3]) * 16 + hex_char_to_int (hex[4]);
|
*b = hex_char_to_int(hex[5]) * 16 + hex_char_to_int(hex[6]);
|
||||||
*b = hex_char_to_int (hex[5]) * 16 + hex_char_to_int (hex[6]);
|
} else if (len == 12 + 1) {
|
||||||
}
|
*r = hex_char_to_int(hex[1]) * 16 + hex_char_to_int(hex[2]);
|
||||||
else if (len == 12 + 1) {
|
*g = hex_char_to_int(hex[5]) * 16 + hex_char_to_int(hex[6]);
|
||||||
*r = hex_char_to_int (hex[1]) * 16 + hex_char_to_int (hex[2]);
|
*b = hex_char_to_int(hex[9]) * 16 + hex_char_to_int(hex[10]);
|
||||||
*g = hex_char_to_int (hex[5]) * 16 + hex_char_to_int (hex[6]);
|
} else
|
||||||
*b = hex_char_to_int (hex[9]) * 16 + hex_char_to_int (hex[10]);
|
return 0;
|
||||||
}
|
|
||||||
else return 0;
|
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void get_color(char *hex, double *rgb)
|
||||||
void get_color (char *hex, double *rgb)
|
|
||||||
{
|
{
|
||||||
int r, g, b;
|
int r, g, b;
|
||||||
hex_to_rgb (hex, &r, &g, &b);
|
r = g = b = 0;
|
||||||
|
hex_to_rgb(hex, &r, &g, &b);
|
||||||
|
|
||||||
rgb[0] = (r / 255.0);
|
rgb[0] = (r / 255.0);
|
||||||
rgb[1] = (g / 255.0);
|
rgb[1] = (g / 255.0);
|
||||||
rgb[2] = (b / 255.0);
|
rgb[2] = (b / 255.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void extract_values(const char *value, char **value1, char **value2, char **value3)
|
||||||
void extract_values (const char *value, char **value1, char **value2, char **value3)
|
|
||||||
{
|
{
|
||||||
char *b=0, *c=0;
|
char *b = 0, *c = 0;
|
||||||
|
|
||||||
if (*value1) free (*value1);
|
if (*value1)
|
||||||
if (*value2) free (*value2);
|
free(*value1);
|
||||||
if (*value3) free (*value3);
|
if (*value2)
|
||||||
|
free(*value2);
|
||||||
|
if (*value3)
|
||||||
|
free(*value3);
|
||||||
|
|
||||||
if ((b = strchr (value, ' '))) {
|
if ((b = strchr(value, ' '))) {
|
||||||
b[0] = '\0';
|
b[0] = '\0';
|
||||||
b++;
|
b++;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
*value2 = 0;
|
*value2 = 0;
|
||||||
*value3 = 0;
|
*value3 = 0;
|
||||||
}
|
}
|
||||||
*value1 = strdup (value);
|
*value1 = strdup(value);
|
||||||
g_strstrip(*value1);
|
g_strstrip(*value1);
|
||||||
|
|
||||||
if (b) {
|
if (b) {
|
||||||
if ((c = strchr (b, ' '))) {
|
if ((c = strchr(b, ' '))) {
|
||||||
c[0] = '\0';
|
c[0] = '\0';
|
||||||
c++;
|
c++;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
c = 0;
|
c = 0;
|
||||||
*value3 = 0;
|
*value3 = 0;
|
||||||
}
|
}
|
||||||
*value2 = strdup (b);
|
*value2 = strdup(b);
|
||||||
g_strstrip(*value2);
|
g_strstrip(*value2);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (c) {
|
if (c) {
|
||||||
*value3 = strdup (c);
|
*value3 = strdup(c);
|
||||||
g_strstrip(*value3);
|
g_strstrip(*value3);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright)
|
void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright)
|
||||||
{
|
{
|
||||||
unsigned int x, y;
|
unsigned int x, y;
|
||||||
@@ -195,21 +248,24 @@ void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright
|
|||||||
float hue, saturation, brightness;
|
float hue, saturation, brightness;
|
||||||
float redc, greenc, bluec;
|
float redc, greenc, bluec;
|
||||||
|
|
||||||
for(y = 0; y < h; y++) {
|
for (y = 0; y < h; y++) {
|
||||||
for(id = y * w, x = 0; x < w; x++, id++) {
|
for (id = y * w, x = 0; x < w; x++, id++) {
|
||||||
argb = data[id];
|
argb = data[id];
|
||||||
a = (argb >> 24) & 0xff;
|
a = (argb >> 24) & 0xff;
|
||||||
// transparent => nothing to do.
|
// transparent => nothing to do.
|
||||||
if (a == 0) continue;
|
if (a == 0)
|
||||||
|
continue;
|
||||||
r = (argb >> 16) & 0xff;
|
r = (argb >> 16) & 0xff;
|
||||||
g = (argb >> 8) & 0xff;
|
g = (argb >> 8) & 0xff;
|
||||||
b = (argb) & 0xff;
|
b = (argb)&0xff;
|
||||||
|
|
||||||
// convert RGB to HSB
|
// convert RGB to HSB
|
||||||
cmax = (r > g) ? r : g;
|
cmax = (r > g) ? r : g;
|
||||||
if (b > cmax) cmax = b;
|
if (b > cmax)
|
||||||
|
cmax = b;
|
||||||
cmin = (r < g) ? r : g;
|
cmin = (r < g) ? r : g;
|
||||||
if (b < cmin) cmin = b;
|
if (b < cmin)
|
||||||
|
cmin = b;
|
||||||
brightness = ((float)cmax) / 255.0f;
|
brightness = ((float)cmax) / 255.0f;
|
||||||
if (cmax != 0)
|
if (cmax != 0)
|
||||||
saturation = ((float)(cmax - cmin)) / ((float)cmax);
|
saturation = ((float)(cmax - cmin)) / ((float)cmax);
|
||||||
@@ -234,13 +290,17 @@ void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright
|
|||||||
|
|
||||||
// adjust
|
// adjust
|
||||||
saturation += satur;
|
saturation += satur;
|
||||||
if (saturation < 0.0) saturation = 0.0;
|
if (saturation < 0.0)
|
||||||
if (saturation > 1.0) saturation = 1.0;
|
saturation = 0.0;
|
||||||
|
if (saturation > 1.0)
|
||||||
|
saturation = 1.0;
|
||||||
brightness += bright;
|
brightness += bright;
|
||||||
if (brightness < 0.0) brightness = 0.0;
|
if (brightness < 0.0)
|
||||||
if (brightness > 1.0) brightness = 1.0;
|
brightness = 0.0;
|
||||||
|
if (brightness > 1.0)
|
||||||
|
brightness = 1.0;
|
||||||
if (alpha != 100)
|
if (alpha != 100)
|
||||||
a = (a * alpha)/100;
|
a = (a * alpha) / 100;
|
||||||
|
|
||||||
// convert HSB to RGB
|
// convert HSB to RGB
|
||||||
if (saturation == 0) {
|
if (saturation == 0) {
|
||||||
@@ -251,7 +311,7 @@ void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright
|
|||||||
p = brightness * (1.0f - saturation);
|
p = brightness * (1.0f - saturation);
|
||||||
q = brightness * (1.0f - saturation * f);
|
q = brightness * (1.0f - saturation * f);
|
||||||
t = brightness * (1.0f - (saturation * (1.0f - f)));
|
t = brightness * (1.0f - (saturation * (1.0f - f)));
|
||||||
switch ((int) h2) {
|
switch ((int)h2) {
|
||||||
case 0:
|
case 0:
|
||||||
r = (int)(brightness * 255.0f + 0.5f);
|
r = (int)(brightness * 255.0f + 0.5f);
|
||||||
g = (int)(t * 255.0f + 0.5f);
|
g = (int)(t * 255.0f + 0.5f);
|
||||||
@@ -294,51 +354,385 @@ void adjust_asb(DATA32 *data, int w, int h, int alpha, float satur, float bright
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void create_heuristic_mask(DATA32 *data, int w, int h)
|
||||||
void createHeuristicMask(DATA32* data, int w, int h)
|
|
||||||
{
|
{
|
||||||
// first we need to find the mask color, therefore we check all 4 edge pixel and take the color which
|
// first we need to find the mask color, therefore we check all 4 edge pixel and take the color which
|
||||||
// appears most often (we only need to check three edges, the 4th is implicitly clear)
|
// appears most often (we only need to check three edges, the 4th is implicitly clear)
|
||||||
unsigned int topLeft = data[0], topRight = data[w-1], bottomLeft = data[w*h-w], bottomRight = data[w*h-1];
|
unsigned int topLeft = data[0], topRight = data[w - 1], bottomLeft = data[w * h - w], bottomRight = data[w * h - 1];
|
||||||
int max = (topLeft == topRight) + (topLeft == bottomLeft) + (topLeft == bottomRight);
|
int max = (topLeft == topRight) + (topLeft == bottomLeft) + (topLeft == bottomRight);
|
||||||
int maskPos = 0;
|
int maskPos = 0;
|
||||||
if ( max < (topRight == topLeft) + (topRight == bottomLeft) + (topRight == bottomRight) ) {
|
if (max < (topRight == topLeft) + (topRight == bottomLeft) + (topRight == bottomRight)) {
|
||||||
max = (topRight == topLeft) + (topRight == bottomLeft) + (topRight == bottomRight);
|
max = (topRight == topLeft) + (topRight == bottomLeft) + (topRight == bottomRight);
|
||||||
maskPos = w-1;
|
maskPos = w - 1;
|
||||||
}
|
}
|
||||||
if ( max < (bottomLeft == topRight) + (bottomLeft == topLeft) + (bottomLeft == bottomRight) )
|
if (max < (bottomLeft == topRight) + (bottomLeft == topLeft) + (bottomLeft == bottomRight))
|
||||||
maskPos = w*h-w;
|
maskPos = w * h - w;
|
||||||
|
|
||||||
// now mask out every pixel which has the same color as the edge pixels
|
// now mask out every pixel which has the same color as the edge pixels
|
||||||
unsigned char* udata = (unsigned char*)data;
|
unsigned char *udata = (unsigned char *)data;
|
||||||
unsigned char b = udata[4*maskPos];
|
unsigned char b = udata[4 * maskPos];
|
||||||
unsigned char g = udata[4*maskPos+1];
|
unsigned char g = udata[4 * maskPos + 1];
|
||||||
unsigned char r = udata[4*maskPos+1];
|
unsigned char r = udata[4 * maskPos + 1];
|
||||||
int i;
|
for (int i = 0; i < h * w; ++i) {
|
||||||
for (i=0; i<h*w; ++i) {
|
if (b - udata[0] == 0 && g - udata[1] == 0 && r - udata[2] == 0)
|
||||||
if ( b-udata[0] == 0 && g-udata[1] == 0 && r-udata[2] == 0 )
|
|
||||||
udata[3] = 0;
|
udata[3] = 0;
|
||||||
udata += 4;
|
udata += 4;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int pixel_empty(DATA32 argb)
|
||||||
void render_image(Drawable d, int x, int y, int w, int h)
|
|
||||||
{
|
{
|
||||||
// in real_transparency mode imlib_render_image_on_drawable does not the right thing, because
|
|
||||||
// the operation is IMLIB_OP_COPY, but we would need IMLIB_OP_OVER (which does not exist)
|
DATA32 a = (argb >> 24) & 0xff;
|
||||||
// Therefore we have to do it with the XRender extension (i.e. copy what imlib is doing internally)
|
if (a == 0)
|
||||||
// But first we need to render the image onto itself with PictOpIn to adjust the colors to the alpha channel
|
return 1;
|
||||||
Pixmap pmap_tmp = XCreatePixmap(server.dsp, server.root_win, w, h, 32);
|
|
||||||
imlib_context_set_drawable(pmap_tmp);
|
DATA32 rgb = argb & 0xffFFff;
|
||||||
|
return rgb == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int image_empty(DATA32 *data, int w, int h)
|
||||||
|
{
|
||||||
|
if (w > 0 && h > 0) {
|
||||||
|
int x = w / 2;
|
||||||
|
int y = h / 2;
|
||||||
|
if (!pixel_empty(data[y * w + x])) {
|
||||||
|
// fprintf(stderr, "Non-empty pixel: [%u, %u] = %x\n", x, y, data[y * w + x]);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int y = 0; y < h; y++) {
|
||||||
|
for (int x = 0; x < w; x++) {
|
||||||
|
if (!pixel_empty(data[y * w + x])) {
|
||||||
|
// fprintf(stderr, "Non-empty pixel: [%u, %u] = %x\n", x, y, data[y * w + x]);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fprintf(stderr, "All pixels are empty\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void render_image(Drawable d, int x, int y)
|
||||||
|
{
|
||||||
|
if (!server.real_transparency) {
|
||||||
|
imlib_context_set_blend(1);
|
||||||
|
imlib_context_set_drawable(d);
|
||||||
|
imlib_render_image_on_drawable(x, y);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int w = imlib_image_get_width(), h = imlib_image_get_height();
|
||||||
|
|
||||||
|
Pixmap pixmap = XCreatePixmap(server.display, server.root_win, w, h, 32);
|
||||||
|
imlib_context_set_drawable(pixmap);
|
||||||
imlib_context_set_blend(0);
|
imlib_context_set_blend(0);
|
||||||
imlib_render_image_on_drawable(0, 0);
|
imlib_render_image_on_drawable(0, 0);
|
||||||
Picture pict_image = XRenderCreatePicture(server.dsp, pmap_tmp, XRenderFindStandardFormat(server.dsp, PictStandardARGB32), 0, 0);
|
|
||||||
Picture pict_drawable = XRenderCreatePicture(server.dsp, d, XRenderFindVisualFormat(server.dsp, server.visual), 0, 0);
|
Pixmap mask = XCreatePixmap(server.display, server.root_win, w, h, 32);
|
||||||
XRenderComposite(server.dsp, PictOpIn, pict_image, None, pict_image, 0, 0, 0, 0, 0, 0, w, h);
|
imlib_context_set_drawable(mask);
|
||||||
XRenderComposite(server.dsp, PictOpOver, pict_image, None, pict_drawable, 0, 0, 0, 0, x, y, w, h);
|
imlib_context_set_blend(0);
|
||||||
imlib_context_set_blend(1);
|
imlib_render_image_on_drawable(0, 0);
|
||||||
XFreePixmap(server.dsp, pmap_tmp);
|
|
||||||
XRenderFreePicture(server.dsp, pict_image);
|
Picture pict = XRenderCreatePicture(server.display, pixmap, XRenderFindStandardFormat(server.display, PictStandardARGB32), 0, 0);
|
||||||
XRenderFreePicture(server.dsp, pict_drawable);
|
Picture pict_drawable = XRenderCreatePicture(server.display, d, XRenderFindVisualFormat(server.display, server.visual), 0, 0);
|
||||||
|
Picture pict_mask = XRenderCreatePicture(server.display, mask, XRenderFindStandardFormat(server.display, PictStandardARGB32), 0, 0);
|
||||||
|
XRenderComposite(server.display, PictOpOver, pict, pict_mask, pict_drawable, 0, 0, 0, 0, x, y, w, h);
|
||||||
|
|
||||||
|
XRenderFreePicture(server.display, pict_mask);
|
||||||
|
XRenderFreePicture(server.display, pict_drawable);
|
||||||
|
XRenderFreePicture(server.display, pict);
|
||||||
|
XFreePixmap(server.display, mask);
|
||||||
|
XFreePixmap(server.display, pixmap);
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_text(PangoLayout *layout, cairo_t *c, int posx, int posy, Color *color, int font_shadow)
|
||||||
|
{
|
||||||
|
if (font_shadow) {
|
||||||
|
const int shadow_size = 3;
|
||||||
|
const double shadow_edge_alpha = 0.0;
|
||||||
|
int i, j;
|
||||||
|
for (i = -shadow_size; i <= shadow_size; i++) {
|
||||||
|
for (j = -shadow_size; j <= shadow_size; j++) {
|
||||||
|
cairo_set_source_rgba(c,
|
||||||
|
0.0,
|
||||||
|
0.0,
|
||||||
|
0.0,
|
||||||
|
1.0 -
|
||||||
|
(1.0 - shadow_edge_alpha) *
|
||||||
|
sqrt((i * i + j * j) / (double)(shadow_size * shadow_size)));
|
||||||
|
pango_cairo_update_layout(c, layout);
|
||||||
|
cairo_move_to(c, posx + i, posy + j);
|
||||||
|
pango_cairo_show_layout(c, layout);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cairo_set_source_rgba(c, color->rgb[0], color->rgb[1], color->rgb[2], color->alpha);
|
||||||
|
pango_cairo_update_layout(c, layout);
|
||||||
|
cairo_move_to(c, posx, posy);
|
||||||
|
pango_cairo_show_layout(c, layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
Imlib_Image load_image(const char *path, int cached)
|
||||||
|
{
|
||||||
|
Imlib_Image image;
|
||||||
|
#ifdef HAVE_RSVG
|
||||||
|
if (cached) {
|
||||||
|
image = imlib_load_image_immediately(path);
|
||||||
|
} else {
|
||||||
|
image = imlib_load_image_immediately_without_cache(path);
|
||||||
|
}
|
||||||
|
if (!image && g_str_has_suffix(path, ".svg")) {
|
||||||
|
char suffix[128];
|
||||||
|
sprintf(suffix, "tmpimage-%d.png", getpid());
|
||||||
|
// We fork here because librsvg allocates memory like crazy
|
||||||
|
pid_t pid = fork();
|
||||||
|
if (pid == 0) {
|
||||||
|
// Child
|
||||||
|
GError *err = NULL;
|
||||||
|
RsvgHandle *svg = rsvg_handle_new_from_file(path, &err);
|
||||||
|
|
||||||
|
if (err != NULL) {
|
||||||
|
fprintf(stderr, "Could not load svg image!: %s", err->message);
|
||||||
|
g_error_free(err);
|
||||||
|
} else {
|
||||||
|
gchar *name = g_build_filename(g_get_user_config_dir(), "tint2", suffix, NULL);
|
||||||
|
GdkPixbuf *pixbuf = rsvg_handle_get_pixbuf(svg);
|
||||||
|
gdk_pixbuf_save(pixbuf, name, "png", NULL, NULL);
|
||||||
|
}
|
||||||
|
exit(0);
|
||||||
|
} else {
|
||||||
|
// Parent
|
||||||
|
waitpid(pid, 0, 0);
|
||||||
|
gchar *name = g_build_filename(g_get_user_config_dir(), "tint2", suffix, NULL);
|
||||||
|
image = imlib_load_image_immediately_without_cache(name);
|
||||||
|
g_remove(name);
|
||||||
|
g_free(name);
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
if (cached) {
|
||||||
|
image = imlib_load_image_immediately(path);
|
||||||
|
} else {
|
||||||
|
image = imlib_load_image_immediately_without_cache(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
Imlib_Image adjust_icon(Imlib_Image original, int alpha, int saturation, int brightness)
|
||||||
|
{
|
||||||
|
if (!original)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
imlib_context_set_image(original);
|
||||||
|
Imlib_Image copy = imlib_clone_image();
|
||||||
|
|
||||||
|
imlib_context_set_image(copy);
|
||||||
|
imlib_image_set_has_alpha(1);
|
||||||
|
DATA32 *data = imlib_image_get_data();
|
||||||
|
adjust_asb(data,
|
||||||
|
imlib_image_get_width(),
|
||||||
|
imlib_image_get_height(),
|
||||||
|
alpha,
|
||||||
|
(float)saturation / 100,
|
||||||
|
(float)brightness / 100);
|
||||||
|
imlib_image_put_back_data(data);
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_rect(cairo_t *c, double x, double y, double w, double h, double r)
|
||||||
|
{
|
||||||
|
if (r > 0.0) {
|
||||||
|
double c1 = 0.55228475 * r;
|
||||||
|
|
||||||
|
cairo_move_to(c, x + r, y);
|
||||||
|
cairo_rel_line_to(c, w - 2 * r, 0);
|
||||||
|
cairo_rel_curve_to(c, c1, 0.0, r, c1, r, r);
|
||||||
|
cairo_rel_line_to(c, 0, h - 2 * r);
|
||||||
|
cairo_rel_curve_to(c, 0.0, c1, c1 - r, r, -r, r);
|
||||||
|
cairo_rel_line_to(c, -w + 2 * r, 0);
|
||||||
|
cairo_rel_curve_to(c, -c1, 0, -r, -c1, -r, -r);
|
||||||
|
cairo_rel_line_to(c, 0, -h + 2 * r);
|
||||||
|
cairo_rel_curve_to(c, 0, -c1, r - c1, -r, r, -r);
|
||||||
|
} else
|
||||||
|
cairo_rectangle(c, x, y, w, h);
|
||||||
|
}
|
||||||
|
|
||||||
|
void clear_pixmap(Pixmap p, int x, int y, int w, int h)
|
||||||
|
{
|
||||||
|
Picture pict = XRenderCreatePicture(server.display, p, XRenderFindVisualFormat(server.display, server.visual), 0, 0);
|
||||||
|
XRenderColor col;
|
||||||
|
col.red = col.green = col.blue = col.alpha = 0;
|
||||||
|
XRenderFillRectangle(server.display, PictOpSrc, pict, &col, x, y, w, h);
|
||||||
|
XRenderFreePicture(server.display, pict);
|
||||||
|
}
|
||||||
|
|
||||||
|
void get_text_size2(PangoFontDescription *font,
|
||||||
|
int *height_ink,
|
||||||
|
int *height,
|
||||||
|
int *width,
|
||||||
|
int panel_height,
|
||||||
|
int panel_width,
|
||||||
|
char *text,
|
||||||
|
int len,
|
||||||
|
PangoWrapMode wrap,
|
||||||
|
PangoEllipsizeMode ellipsis,
|
||||||
|
gboolean markup)
|
||||||
|
{
|
||||||
|
PangoRectangle rect_ink, rect;
|
||||||
|
|
||||||
|
Pixmap pmap = XCreatePixmap(server.display, server.root_win, panel_height, panel_width, server.depth);
|
||||||
|
|
||||||
|
cairo_surface_t *cs = cairo_xlib_surface_create(server.display, pmap, server.visual, panel_height, panel_width);
|
||||||
|
cairo_t *c = cairo_create(cs);
|
||||||
|
|
||||||
|
PangoLayout *layout = pango_cairo_create_layout(c);
|
||||||
|
pango_layout_set_width(layout, panel_width * PANGO_SCALE);
|
||||||
|
pango_layout_set_height(layout, panel_height * PANGO_SCALE);
|
||||||
|
pango_layout_set_wrap(layout, wrap);
|
||||||
|
pango_layout_set_ellipsize(layout, ellipsis);
|
||||||
|
pango_layout_set_font_description(layout, font);
|
||||||
|
if (!markup)
|
||||||
|
pango_layout_set_text(layout, text, len);
|
||||||
|
else
|
||||||
|
pango_layout_set_markup(layout, text, len);
|
||||||
|
|
||||||
|
pango_layout_get_pixel_extents(layout, &rect_ink, &rect);
|
||||||
|
*height_ink = rect_ink.height;
|
||||||
|
*height = rect.height;
|
||||||
|
*width = rect.width;
|
||||||
|
// printf("dimension : %d - %d\n", rect_ink.height, rect.height);
|
||||||
|
|
||||||
|
g_object_unref(layout);
|
||||||
|
cairo_destroy(c);
|
||||||
|
cairo_surface_destroy(cs);
|
||||||
|
XFreePixmap(server.display, pmap);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if !GLIB_CHECK_VERSION (2, 33, 4)
|
||||||
|
GList *g_list_copy_deep(GList *list, GCopyFunc func, gpointer user_data)
|
||||||
|
{
|
||||||
|
list = g_list_copy(list);
|
||||||
|
|
||||||
|
if (func) {
|
||||||
|
for (GList *l = list; l; l = l->next) {
|
||||||
|
l->data = func(l->data, user_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Based loosely on close_allv from
|
||||||
|
// https://git.gnome.org/browse/glib/tree/gio/libasyncns/asyncns.c?h=2.21.0#n205
|
||||||
|
// license: LGPL version 2.1
|
||||||
|
// but with all the junk removed
|
||||||
|
// and
|
||||||
|
// https://opensource.apple.com/source/sudo/sudo-46/src/closefrom.c
|
||||||
|
// license: BSD
|
||||||
|
void close_all_fds()
|
||||||
|
{
|
||||||
|
const int from_fd = 3;
|
||||||
|
|
||||||
|
#ifdef __linux__
|
||||||
|
DIR *d = opendir("/proc/self/fd");
|
||||||
|
if (d) {
|
||||||
|
for (struct dirent *de = readdir(d); de; de = readdir(d)) {
|
||||||
|
if (de->d_name[0] == '.')
|
||||||
|
continue;
|
||||||
|
int fd = atoi(de->d_name);
|
||||||
|
if (fd < from_fd)
|
||||||
|
continue;
|
||||||
|
if (fd == dirfd(d))
|
||||||
|
continue;
|
||||||
|
close(fd);
|
||||||
|
}
|
||||||
|
closedir(d);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__) || defined(__DragonFly__) || defined(__OpenBSD__)
|
||||||
|
closefrom(from_fd);
|
||||||
|
return;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__NetBSD__)
|
||||||
|
fcntl(from_fd, F_CLOSEM, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Worst case scenario: iterate over all possible fds
|
||||||
|
int max_fd = sysconf(_SC_OPEN_MAX);
|
||||||
|
for (int fd = from_fd; fd < max_fd; fd++) {
|
||||||
|
close(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
char* get_own_path()
|
||||||
|
{
|
||||||
|
const int buf_size = 4096;
|
||||||
|
char *buf = calloc(buf_size, 1);
|
||||||
|
|
||||||
|
#ifdef __linux__
|
||||||
|
if (readlink("/proc/self/exe", buf, buf_size) > 0)
|
||||||
|
return buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__)
|
||||||
|
int mib[4] = {
|
||||||
|
CTL_KERN,
|
||||||
|
KERN_PROC,
|
||||||
|
KERN_PROC_PATHNAME,
|
||||||
|
getpid()
|
||||||
|
};
|
||||||
|
|
||||||
|
size_t max_len = buf_size;
|
||||||
|
if (sysctl(mib, 4, buf, &max_len, NULL, 0) == 0)
|
||||||
|
return buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__DragonFly__)
|
||||||
|
if (readlink("/proc/curproc/file", buf, buf_size) > 0)
|
||||||
|
return buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__NetBSD__)
|
||||||
|
if (readlink("/proc/curproc/exe", buf, buf_size) > 0)
|
||||||
|
return buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__OpenBSD__)
|
||||||
|
int mib[4] = {
|
||||||
|
CTL_KERN,
|
||||||
|
KERN_PROC_ARGS,
|
||||||
|
getpid(),
|
||||||
|
KERN_PROC_ARGV
|
||||||
|
};
|
||||||
|
|
||||||
|
char *path = NULL;
|
||||||
|
size_t len;
|
||||||
|
if (sysctl(mib, 4, NULL, &len, NULL, 0) == 0 && len > 0) {
|
||||||
|
char **argv = malloc(len);
|
||||||
|
if (argv) {
|
||||||
|
if (sysctl(mib, 4, argv, &len, NULL, 0) == 0) {
|
||||||
|
path = realpath(argv[0], NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(argv);
|
||||||
|
}
|
||||||
|
if (path) {
|
||||||
|
free(buf);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
sprintf(buf, "tint2");
|
||||||
|
return buf;
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user