Compare commits
116 Commits
citadel
...
wip/cherge
Author | SHA1 | Date | |
---|---|---|---|
|
7f7965d622 | ||
|
92cfb9ab1b | ||
|
f3082a3683 | ||
|
24f400d28a | ||
|
4cac5b4b28 | ||
|
b48e95cba2 | ||
|
5ccf92e804 | ||
|
2040c380bd | ||
|
4e22989f07 | ||
|
cfb92ad392 | ||
|
f87b9f374a | ||
|
2c549bfbbe | ||
|
f597a0a11c | ||
|
3f35ad0cbf | ||
|
bd3227e23f | ||
|
04f847a3c2 | ||
|
41fe2d2c01 | ||
|
161beb71eb | ||
|
13ec3169a6 | ||
|
8702d6647b | ||
|
5703a25e2b | ||
|
09fbb4a127 | ||
|
4c7b20e584 | ||
|
b3045cb964 | ||
|
c41175af45 | ||
|
7b1544a7a2 | ||
|
a0f1ac87e9 | ||
|
1045b35c8b | ||
|
24cdcc56da | ||
|
1b28cdfbf4 | ||
|
a73294312f | ||
|
5e7a7e31a1 | ||
|
8111286463 | ||
|
06c6ecc15b | ||
|
36f9dcd2bd | ||
|
219118e633 | ||
|
1e929357e0 | ||
|
e7463e38ca | ||
|
420c1cbfd1 | ||
|
d53bd98532 | ||
|
128a501b9a | ||
|
b29e243fec | ||
|
fa2ddcc52f | ||
|
fbf194f6a1 | ||
|
5610a2435d | ||
|
a5b36fc7f4 | ||
|
923d80bba8 | ||
|
9e196d2765 | ||
|
94897492be | ||
|
6baf82eb83 | ||
|
65d27aaa43 | ||
|
176aaa4a97 | ||
|
5ae5811155 | ||
|
e027af9548 | ||
|
73649a0d6a | ||
|
7dbcd26619 | ||
|
f30ec4a4c8 | ||
|
c04289853f | ||
|
b72b773d87 | ||
|
da7cd2807f | ||
|
20373ba64e | ||
|
025f6eb68e | ||
|
dc5df5c4c9 | ||
|
81db908339 | ||
|
706a2259b8 | ||
|
e94af71430 | ||
|
a662e7bb87 | ||
|
0b82388c49 | ||
|
b359b937e9 | ||
|
98fb7c33f2 | ||
|
4dfc2e0fd1 | ||
|
9152d3a286 | ||
|
ba33a05dd2 | ||
|
ddb309815c | ||
|
6a796675bd | ||
|
d08497414f | ||
|
82886b7ee8 | ||
|
dca43c7b24 | ||
|
af50a7829f | ||
|
67cb02d46a | ||
|
721ce54037 | ||
|
445eed40a7 | ||
|
fddd122956 | ||
|
df1b46eee2 | ||
|
bf318df7f3 | ||
|
2a2f3c981e | ||
|
8e5eab0498 | ||
|
6ed21e1ce0 | ||
|
9c51c87d8c | ||
|
db2245d60b | ||
|
f26cc3ac23 | ||
|
02c5b4b947 | ||
|
df57829ea1 | ||
|
da96408098 | ||
|
4b2e0247af | ||
|
2c617e5a3a | ||
|
4ff7e84c51 | ||
|
9f76b6e4a2 | ||
|
0ac0f7e85b | ||
|
73b00ff1a7 | ||
|
a52597ac5b | ||
|
9a2597f80b | ||
|
e1ed4b25e1 | ||
|
c70b18764b | ||
|
4398516520 | ||
|
220514d10e | ||
|
18312d9ccd | ||
|
234b1441e4 | ||
|
e909db5848 | ||
|
702338bc7d | ||
|
7c9dbc66d9 | ||
|
0d031dc20f | ||
|
b476e851b7 | ||
|
a27be6a540 | ||
|
4b6a57fabe | ||
|
92758890bb |
2
.gitmodules
vendored
2
.gitmodules
vendored
@ -1,3 +1,3 @@
|
|||||||
[submodule "subprojects/gvc"]
|
[submodule "subprojects/gvc"]
|
||||||
path = subprojects/gvc
|
path = subprojects/gvc
|
||||||
url = https://git.gnome.org/browse/libgnome-volume-control
|
url = https://gitlab.gnome.org/GNOME/libgnome-volume-control.git
|
||||||
|
55
NEWS
55
NEWS
@ -1,3 +1,58 @@
|
|||||||
|
3.28.4
|
||||||
|
======
|
||||||
|
* Fix wrong window positions in overview on wayland [Marco; #776588]
|
||||||
|
* overview: Fix handling of confirmation dialogs on wayland [verdre; !180]
|
||||||
|
* Avoid some full relayout/redraws [Carlos; !197]
|
||||||
|
* Keep workspace switcher slid out when workspaces are in use [Florian; !161]
|
||||||
|
* Cancel search on overview hiding [Marco; !205]
|
||||||
|
* Fix disappearing network icon [Iain; #140]
|
||||||
|
* Improve performance of app icon animations [Daniel; !253]
|
||||||
|
* notifications: Support icon theme names in 'image-path' hint [Marco; !285]
|
||||||
|
* Avoid focus changes when updating keyboard options [Takao; #391]
|
||||||
|
* Fix unresponsive-app dialog blocking input in other windows [Florian; #273]
|
||||||
|
* Fix ellipsization in dialog subtitles/bodies [Marco; !531]
|
||||||
|
* Misc. bug fixes [Marco, Andrea, Florian, Jasper, Sam, verdre, Jonas,
|
||||||
|
Cosimo, Carlos; #792681, #372, !112, !162, #414, #663461, #788882, #787260,
|
||||||
|
!188, #791233, #602, #632, !305, !286, !314, #781, #693, #618, #430, #799,
|
||||||
|
#783, !293, #298, #1015, #539, #1270]
|
||||||
|
|
||||||
|
Contributors:
|
||||||
|
Jonas Ådahl, Andrea Azzarone, Cosimo Cecchi, Takao Fujiwara, Carlos Garnacho,
|
||||||
|
Iain Lane, Florian Müllner, Georges Basile Stavracas Neto, Jasper St. Pierre,
|
||||||
|
Sam Spilsbury, Ray Strode, Will Thompson, Marco Trevisan (Treviño), verdre,
|
||||||
|
Daniel van Vugt
|
||||||
|
|
||||||
|
Translators:
|
||||||
|
Marek Černocký [cs]
|
||||||
|
|
||||||
|
3.28.3
|
||||||
|
======
|
||||||
|
* Fix lagging pointer when zoomed [Daniel; #682013]
|
||||||
|
* Fix "Clear All" for calendar events [Florian; #325]
|
||||||
|
* Misc. bug fixes [Florian, Mario, Marco; #136, #214, #788931, #791233]
|
||||||
|
|
||||||
|
Contributors:
|
||||||
|
Carlos Garnacho, Florian Müllner, Mario Sanchez Prada, Joe Rabinoff,
|
||||||
|
Didier Roche, Marco Trevisan (Treviño), Daniel van Vugt
|
||||||
|
|
||||||
|
Translators:
|
||||||
|
Pieter Schalk Schoeman [af], Gun Chleoc [gd]
|
||||||
|
|
||||||
|
3.28.2
|
||||||
|
======
|
||||||
|
* Fix lock-up on cancelling polkit dialog [Florian; #221]
|
||||||
|
* Guard against untimely keyboard map changes [Carlos; #240]
|
||||||
|
* Fix blurriness of OSD under some resolutions [Silvère; #782011]
|
||||||
|
* Fix icons in search provider results [Florian; #249]
|
||||||
|
* Misc. bug fixes [Marco, Florian; #792687, #781471]
|
||||||
|
|
||||||
|
Contributors:
|
||||||
|
Carlos Garnacho, Silvère Latchurié, Florian Müllner, Mario Sanchez Prada,
|
||||||
|
Ray Strode, Marco Trevisan (Treviño)
|
||||||
|
|
||||||
|
Translators:
|
||||||
|
Stas Solovey [ru], Rafael Fontenelle [pt_BR]
|
||||||
|
|
||||||
3.28.1
|
3.28.1
|
||||||
======
|
======
|
||||||
* Fix compose characters in shell entries [Carlos; #115]
|
* Fix compose characters in shell entries [Carlos; #115]
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
<file>be.json</file>
|
<file>be.json</file>
|
||||||
<file>bg.json</file>
|
<file>bg.json</file>
|
||||||
<file>by.json</file>
|
<file>by.json</file>
|
||||||
|
<file>ca.json</file>
|
||||||
<file>cz.json</file>
|
<file>cz.json</file>
|
||||||
<file>de.json</file>
|
<file>de.json</file>
|
||||||
<file>dk.json</file>
|
<file>dk.json</file>
|
||||||
|
599
data/osk-layouts/ca.json
Normal file
599
data/osk-layouts/ca.json
Normal file
@ -0,0 +1,599 @@
|
|||||||
|
{
|
||||||
|
"levels": [
|
||||||
|
{
|
||||||
|
"level": "",
|
||||||
|
"mode": "default",
|
||||||
|
"rows": [
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"q"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"w"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"e",
|
||||||
|
"é",
|
||||||
|
"è",
|
||||||
|
"ê",
|
||||||
|
"ë",
|
||||||
|
"%",
|
||||||
|
"ę",
|
||||||
|
"ė",
|
||||||
|
"ē"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"r"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"t"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"y",
|
||||||
|
"%",
|
||||||
|
"ÿ"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"u",
|
||||||
|
"ù",
|
||||||
|
"û",
|
||||||
|
"%",
|
||||||
|
"ü",
|
||||||
|
"ú",
|
||||||
|
"ū"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"i",
|
||||||
|
"î",
|
||||||
|
"%",
|
||||||
|
"ï",
|
||||||
|
"ì",
|
||||||
|
"í",
|
||||||
|
"į",
|
||||||
|
"ī"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"o",
|
||||||
|
"ô",
|
||||||
|
"œ",
|
||||||
|
"%",
|
||||||
|
"ö",
|
||||||
|
"ò",
|
||||||
|
"ó",
|
||||||
|
"õ",
|
||||||
|
"ø",
|
||||||
|
"ō",
|
||||||
|
"º"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"p"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"a",
|
||||||
|
"à",
|
||||||
|
"â",
|
||||||
|
"%",
|
||||||
|
"æ",
|
||||||
|
"á",
|
||||||
|
"ä",
|
||||||
|
"ã",
|
||||||
|
"å",
|
||||||
|
"ā",
|
||||||
|
"ª"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"s"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"d"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"f"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"g"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"h"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"j"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"k"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"l"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"z"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"x"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"c",
|
||||||
|
"ç",
|
||||||
|
"ć",
|
||||||
|
"č"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"v"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"b"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"n"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"m"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
","
|
||||||
|
],
|
||||||
|
[
|
||||||
|
" "
|
||||||
|
],
|
||||||
|
[
|
||||||
|
".",
|
||||||
|
"#",
|
||||||
|
"!",
|
||||||
|
",",
|
||||||
|
"?",
|
||||||
|
"-",
|
||||||
|
":",
|
||||||
|
"'",
|
||||||
|
"@"
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"level": "shift",
|
||||||
|
"mode": "latched",
|
||||||
|
"rows": [
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"Q"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"W"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"E",
|
||||||
|
"É",
|
||||||
|
"È",
|
||||||
|
"Ê",
|
||||||
|
"Ë",
|
||||||
|
"%",
|
||||||
|
"Ę",
|
||||||
|
"Ė",
|
||||||
|
"Ē"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"R"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"T"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"Y",
|
||||||
|
"%",
|
||||||
|
"Ÿ"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"U",
|
||||||
|
"Ù",
|
||||||
|
"Û",
|
||||||
|
"%",
|
||||||
|
"Ü",
|
||||||
|
"Ú",
|
||||||
|
"Ū"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"I",
|
||||||
|
"Î",
|
||||||
|
"%",
|
||||||
|
"Ï",
|
||||||
|
"Ì",
|
||||||
|
"Í",
|
||||||
|
"Į",
|
||||||
|
"Ī"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"O",
|
||||||
|
"Ô",
|
||||||
|
"Œ",
|
||||||
|
"%",
|
||||||
|
"Ö",
|
||||||
|
"Ò",
|
||||||
|
"Ó",
|
||||||
|
"Õ",
|
||||||
|
"Ø",
|
||||||
|
"Ō",
|
||||||
|
"º"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"P"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"A",
|
||||||
|
"À",
|
||||||
|
"Â",
|
||||||
|
"%",
|
||||||
|
"Æ",
|
||||||
|
"Á",
|
||||||
|
"Ä",
|
||||||
|
"Ã",
|
||||||
|
"Å",
|
||||||
|
"Ā",
|
||||||
|
"ª"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"S"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"D"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"F"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"G"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"H"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"J"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"K"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"L"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"Z"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"X"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"C",
|
||||||
|
"Ç",
|
||||||
|
"Ć",
|
||||||
|
"Č"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"V"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"B"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"N"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"M"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
","
|
||||||
|
],
|
||||||
|
[
|
||||||
|
" "
|
||||||
|
],
|
||||||
|
[
|
||||||
|
".",
|
||||||
|
"#",
|
||||||
|
"!",
|
||||||
|
",",
|
||||||
|
"?",
|
||||||
|
"-",
|
||||||
|
":",
|
||||||
|
"'",
|
||||||
|
"@"
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"level": "opt",
|
||||||
|
"mode": "locked",
|
||||||
|
"rows": [
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"1",
|
||||||
|
"¹",
|
||||||
|
"½",
|
||||||
|
"⅓",
|
||||||
|
"¼",
|
||||||
|
"⅛"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"2",
|
||||||
|
"²",
|
||||||
|
"⅔"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"3",
|
||||||
|
"³",
|
||||||
|
"¾",
|
||||||
|
"⅜"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"4",
|
||||||
|
"⁴"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"5",
|
||||||
|
"⅝"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"6"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"7",
|
||||||
|
"⅞"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"8"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"9"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"0",
|
||||||
|
"ⁿ",
|
||||||
|
"∅"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"@"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"#"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"$",
|
||||||
|
"¢",
|
||||||
|
"£",
|
||||||
|
"€",
|
||||||
|
"¥",
|
||||||
|
"₱"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"%",
|
||||||
|
"‰"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"&"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"-",
|
||||||
|
"_",
|
||||||
|
"–",
|
||||||
|
"—",
|
||||||
|
"·"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"+",
|
||||||
|
"±"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"(",
|
||||||
|
"<",
|
||||||
|
"{",
|
||||||
|
"["
|
||||||
|
],
|
||||||
|
[
|
||||||
|
")",
|
||||||
|
">",
|
||||||
|
"}",
|
||||||
|
"]"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"*",
|
||||||
|
"†",
|
||||||
|
"‡",
|
||||||
|
"★"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"\"",
|
||||||
|
"“",
|
||||||
|
"”",
|
||||||
|
"«",
|
||||||
|
"»"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"'",
|
||||||
|
"‘",
|
||||||
|
"’",
|
||||||
|
"‹",
|
||||||
|
"›"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
":"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
";"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"!",
|
||||||
|
"¡"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"?",
|
||||||
|
"¿"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"_"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"/"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
" "
|
||||||
|
],
|
||||||
|
[
|
||||||
|
","
|
||||||
|
],
|
||||||
|
[
|
||||||
|
".",
|
||||||
|
"…"
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"level": "opt+shift",
|
||||||
|
"mode": "locked",
|
||||||
|
"rows": [
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"~"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"`"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"|"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"•",
|
||||||
|
"♪",
|
||||||
|
"♥",
|
||||||
|
"♠",
|
||||||
|
"♦",
|
||||||
|
"♣"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"√"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"Π",
|
||||||
|
"π"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"÷"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"×"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"¶",
|
||||||
|
"§"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"∆"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"£"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"¢"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"€"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"¥"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"^",
|
||||||
|
"↑",
|
||||||
|
"↓",
|
||||||
|
"←",
|
||||||
|
"→"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"°",
|
||||||
|
"′",
|
||||||
|
"″"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"=",
|
||||||
|
"≠",
|
||||||
|
"≈",
|
||||||
|
"∞"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"{"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"}"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"\\"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"©"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"®"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"™"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"℅"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"["
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"]"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"<",
|
||||||
|
"‹",
|
||||||
|
"≤",
|
||||||
|
"«"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
">",
|
||||||
|
"›",
|
||||||
|
"≥",
|
||||||
|
"»"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
" "
|
||||||
|
],
|
||||||
|
[
|
||||||
|
","
|
||||||
|
],
|
||||||
|
[
|
||||||
|
".",
|
||||||
|
"…"
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"locale": "fr-CA",
|
||||||
|
"name": "French Canada"
|
||||||
|
}
|
@ -6,10 +6,20 @@
|
|||||||
"rows": [
|
"rows": [
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"q"
|
"a",
|
||||||
|
"à",
|
||||||
|
"â",
|
||||||
|
"%",
|
||||||
|
"æ",
|
||||||
|
"á",
|
||||||
|
"ä",
|
||||||
|
"ã",
|
||||||
|
"å",
|
||||||
|
"ā",
|
||||||
|
"ª"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"w"
|
"z"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"e",
|
"e",
|
||||||
@ -71,17 +81,7 @@
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"a",
|
"q"
|
||||||
"à",
|
|
||||||
"â",
|
|
||||||
"%",
|
|
||||||
"æ",
|
|
||||||
"á",
|
|
||||||
"ä",
|
|
||||||
"ã",
|
|
||||||
"å",
|
|
||||||
"ā",
|
|
||||||
"ª"
|
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"s"
|
"s"
|
||||||
@ -106,11 +106,14 @@
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
"l"
|
"l"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"m"
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"z"
|
"w"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"x"
|
"x"
|
||||||
@ -131,7 +134,11 @@
|
|||||||
"n"
|
"n"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"m"
|
"'",
|
||||||
|
"‘",
|
||||||
|
"’",
|
||||||
|
"‹",
|
||||||
|
"›"
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
@ -161,10 +168,20 @@
|
|||||||
"rows": [
|
"rows": [
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"Q"
|
"A",
|
||||||
|
"À",
|
||||||
|
"Â",
|
||||||
|
"%",
|
||||||
|
"Æ",
|
||||||
|
"Á",
|
||||||
|
"Ä",
|
||||||
|
"Ã",
|
||||||
|
"Å",
|
||||||
|
"Ā",
|
||||||
|
"ª"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"W"
|
"Z"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"E",
|
"E",
|
||||||
@ -226,17 +243,7 @@
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"A",
|
"Q"
|
||||||
"À",
|
|
||||||
"Â",
|
|
||||||
"%",
|
|
||||||
"Æ",
|
|
||||||
"Á",
|
|
||||||
"Ä",
|
|
||||||
"Ã",
|
|
||||||
"Å",
|
|
||||||
"Ā",
|
|
||||||
"ª"
|
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"S"
|
"S"
|
||||||
@ -261,11 +268,14 @@
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
"L"
|
"L"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"M"
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
"Z"
|
"W"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"X"
|
"X"
|
||||||
@ -286,7 +296,11 @@
|
|||||||
"N"
|
"N"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"M"
|
"'",
|
||||||
|
"‘",
|
||||||
|
"’",
|
||||||
|
"‹",
|
||||||
|
"›"
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
@ -369,10 +383,10 @@
|
|||||||
"#"
|
"#"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"$",
|
"€",
|
||||||
"¢",
|
"¢",
|
||||||
"£",
|
"£",
|
||||||
"€",
|
"$",
|
||||||
"¥",
|
"¥",
|
||||||
"₱"
|
"₱"
|
||||||
],
|
],
|
||||||
@ -511,13 +525,14 @@
|
|||||||
"£"
|
"£"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
|
"¥"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"$",
|
||||||
"¢"
|
"¢"
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"€"
|
"¢"
|
||||||
],
|
|
||||||
[
|
|
||||||
"¥"
|
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"^",
|
"^",
|
||||||
@ -594,6 +609,6 @@
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"locale": "fr-CA",
|
"locale": "fr",
|
||||||
"name": "French Canada"
|
"name": "French"
|
||||||
}
|
}
|
@ -242,11 +242,11 @@ var AuthPrompt = new Lang.Class({
|
|||||||
this.emit('prompted');
|
this.emit('prompted');
|
||||||
},
|
},
|
||||||
|
|
||||||
_onVerificationFailed() {
|
_onVerificationFailed(userVerifier, canRetry) {
|
||||||
this._queryingService = null;
|
this._queryingService = null;
|
||||||
this.clear();
|
this.clear();
|
||||||
|
|
||||||
this.updateSensitivity(true);
|
this.updateSensitivity(canRetry);
|
||||||
this.setActorInDefaultButtonWell(null);
|
this.setActorInDefaultButtonWell(null);
|
||||||
this.verificationStatus = AuthPromptStatus.VERIFICATION_FAILED;
|
this.verificationStatus = AuthPromptStatus.VERIFICATION_FAILED;
|
||||||
},
|
},
|
||||||
@ -439,6 +439,7 @@ var AuthPrompt = new Lang.Class({
|
|||||||
this.verificationStatus = AuthPromptStatus.NOT_VERIFYING;
|
this.verificationStatus = AuthPromptStatus.NOT_VERIFYING;
|
||||||
this.cancelButton.reactive = true;
|
this.cancelButton.reactive = true;
|
||||||
this.nextButton.label = _("Next");
|
this.nextButton.label = _("Next");
|
||||||
|
this._preemptiveAnswer = null;
|
||||||
|
|
||||||
if (this._userVerifier)
|
if (this._userVerifier)
|
||||||
this._userVerifier.cancel();
|
this._userVerifier.cancel();
|
||||||
|
@ -534,12 +534,13 @@ var ShellUserVerifier = new Lang.Class({
|
|||||||
_verificationFailed(retry) {
|
_verificationFailed(retry) {
|
||||||
// For Not Listed / enterprise logins, immediately reset
|
// For Not Listed / enterprise logins, immediately reset
|
||||||
// the dialog
|
// the dialog
|
||||||
// Otherwise, we allow ALLOWED_FAILURES attempts. After that, we
|
// Otherwise, when in login mode we allow ALLOWED_FAILURES attempts.
|
||||||
// go back to the welcome screen.
|
// After that, we go back to the welcome screen.
|
||||||
|
|
||||||
this._failCounter++;
|
this._failCounter++;
|
||||||
let canRetry = retry && this._userName &&
|
let canRetry = retry && this._userName &&
|
||||||
this._failCounter < this._settings.get_int(ALLOWED_FAILURES_KEY);
|
(this._reauthOnly ||
|
||||||
|
this._failCounter < this._settings.get_int(ALLOWED_FAILURES_KEY));
|
||||||
|
|
||||||
if (canRetry) {
|
if (canRetry) {
|
||||||
if (!this.hasPendingMessages) {
|
if (!this.hasPendingMessages) {
|
||||||
@ -562,7 +563,7 @@ var ShellUserVerifier = new Lang.Class({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.emit('verification-failed');
|
this.emit('verification-failed', canRetry);
|
||||||
},
|
},
|
||||||
|
|
||||||
_onConversationStopped(client, serviceName) {
|
_onConversationStopped(client, serviceName) {
|
||||||
|
@ -115,6 +115,11 @@ var IBusManager = new Lang.Class({
|
|||||||
object_path: IBus.PATH_PANEL });
|
object_path: IBus.PATH_PANEL });
|
||||||
this._candidatePopup.setPanelService(this._panelService);
|
this._candidatePopup.setPanelService(this._panelService);
|
||||||
this._panelService.connect('update-property', this._updateProperty.bind(this));
|
this._panelService.connect('update-property', this._updateProperty.bind(this));
|
||||||
|
this._panelService.connect('set-cursor-location', (ps, x, y, w, h) => {
|
||||||
|
let cursorLocation = { x, y, width: w, height: h };
|
||||||
|
this.emit('set-cursor-location', cursorLocation);
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// IBus versions older than 1.5.10 have a bug which
|
// IBus versions older than 1.5.10 have a bug which
|
||||||
// causes spurious set-content-type emissions when
|
// causes spurious set-content-type emissions when
|
||||||
@ -200,7 +205,7 @@ var IBusManager = new Lang.Class({
|
|||||||
}
|
}
|
||||||
|
|
||||||
this._ibus.set_global_engine_async(id, this._MAX_INPUT_SOURCE_ACTIVATION_TIME,
|
this._ibus.set_global_engine_async(id, this._MAX_INPUT_SOURCE_ACTIVATION_TIME,
|
||||||
null, callback);
|
null, callback || null);
|
||||||
},
|
},
|
||||||
|
|
||||||
preloadEngines(ids) {
|
preloadEngines(ids) {
|
||||||
|
@ -89,6 +89,8 @@ var KeyboardManager = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
setUserLayouts(ids) {
|
setUserLayouts(ids) {
|
||||||
|
let currentId = this._current ? this._current.id : null;
|
||||||
|
let currentGroupIndex = this._current ? this._current.groupIndex : null;
|
||||||
this._current = null;
|
this._current = null;
|
||||||
this._layoutInfos = {};
|
this._layoutInfos = {};
|
||||||
|
|
||||||
@ -115,6 +117,9 @@ var KeyboardManager = new Lang.Class({
|
|||||||
info.group = group;
|
info.group = group;
|
||||||
info.groupIndex = groupIndex;
|
info.groupIndex = groupIndex;
|
||||||
|
|
||||||
|
if (currentId == id && currentGroupIndex == groupIndex)
|
||||||
|
this._current = info;
|
||||||
|
|
||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -236,11 +236,12 @@ var ObjectManager = new Lang.Class({
|
|||||||
_onNameVanished() {
|
_onNameVanished() {
|
||||||
let objectPaths = Object.keys(this._objects);
|
let objectPaths = Object.keys(this._objects);
|
||||||
for (let i = 0; i < objectPaths.length; i++) {
|
for (let i = 0; i < objectPaths.length; i++) {
|
||||||
let object = this._objects[objectPaths];
|
let objectPath = objectPaths[i];
|
||||||
|
let object = this._objects[objectPath];
|
||||||
|
|
||||||
let interfaceNames = Object.keys(object);
|
let interfaceNames = Object.keys(object);
|
||||||
for (let j = 0; i < interfaceNames.length; i++) {
|
for (let j = 0; j < interfaceNames.length; j++) {
|
||||||
let interfaceName = interfaceNames[i];
|
let interfaceName = interfaceNames[j];
|
||||||
|
|
||||||
if (object[interfaceName])
|
if (object[interfaceName])
|
||||||
this._removeInterface(objectPath, interfaceName);
|
this._removeInterface(objectPath, interfaceName);
|
||||||
|
@ -92,17 +92,8 @@ function _listsIntersect(a, b) {
|
|||||||
function _getFolderName(folder) {
|
function _getFolderName(folder) {
|
||||||
let name = folder.get_string('name');
|
let name = folder.get_string('name');
|
||||||
|
|
||||||
if (folder.get_boolean('translate')) {
|
if (folder.get_boolean('translate'))
|
||||||
let keyfile = new GLib.KeyFile();
|
return Shell.AppCache.get_default().translate_folder(name);
|
||||||
let path = 'desktop-directories/' + name;
|
|
||||||
|
|
||||||
try {
|
|
||||||
keyfile.load_from_data_dirs(path, GLib.KeyFileFlags.NONE);
|
|
||||||
name = keyfile.get_locale_string('Desktop Entry', 'Name', null);
|
|
||||||
} catch(e) {
|
|
||||||
return name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
@ -499,7 +490,7 @@ var AllView = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_loadApps() {
|
_loadApps() {
|
||||||
let apps = Gio.AppInfo.get_all().filter(appInfo => {
|
let apps = Shell.AppCache.get_default().get_all().filter(appInfo => {
|
||||||
try {
|
try {
|
||||||
let id = appInfo.get_id(); // catch invalid file encodings
|
let id = appInfo.get_id(); // catch invalid file encodings
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
@ -1329,7 +1320,7 @@ var FolderIcon = new Lang.Class({
|
|||||||
folderApps.forEach(addAppId);
|
folderApps.forEach(addAppId);
|
||||||
|
|
||||||
let folderCategories = this._folder.get_strv('categories');
|
let folderCategories = this._folder.get_strv('categories');
|
||||||
Gio.AppInfo.get_all().forEach(appInfo => {
|
Shell.AppCache.get_default().get_all().forEach(appInfo => {
|
||||||
let appCategories = _getCategories(appInfo);
|
let appCategories = _getCategories(appInfo);
|
||||||
if (!_listsIntersect(folderCategories, appCategories))
|
if (!_listsIntersect(folderCategories, appCategories))
|
||||||
return;
|
return;
|
||||||
|
@ -802,6 +802,8 @@ var NotificationMessage = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_onDestroy() {
|
_onDestroy() {
|
||||||
|
this.parent();
|
||||||
|
|
||||||
if (this._updatedId)
|
if (this._updatedId)
|
||||||
this.notification.disconnect(this._updatedId);
|
this.notification.disconnect(this._updatedId);
|
||||||
this._updatedId = 0;
|
this._updatedId = 0;
|
||||||
@ -821,6 +823,8 @@ var EventsSection = new Lang.Class({
|
|||||||
this._desktopSettings.connect('changed', this._reloadEvents.bind(this));
|
this._desktopSettings.connect('changed', this._reloadEvents.bind(this));
|
||||||
this._eventSource = new EmptyEventSource();
|
this._eventSource = new EmptyEventSource();
|
||||||
|
|
||||||
|
this._messageById = new Map();
|
||||||
|
|
||||||
this.parent();
|
this.parent();
|
||||||
|
|
||||||
this._title = new St.Button({ style_class: 'events-section-title',
|
this._title = new St.Button({ style_class: 'events-section-title',
|
||||||
@ -875,20 +879,32 @@ var EventsSection = new Lang.Class({
|
|||||||
|
|
||||||
this._reloading = true;
|
this._reloading = true;
|
||||||
|
|
||||||
this._list.destroy_all_children();
|
|
||||||
|
|
||||||
let periodBegin = _getBeginningOfDay(this._date);
|
let periodBegin = _getBeginningOfDay(this._date);
|
||||||
let periodEnd = _getEndOfDay(this._date);
|
let periodEnd = _getEndOfDay(this._date);
|
||||||
let events = this._eventSource.getEvents(periodBegin, periodEnd);
|
let events = this._eventSource.getEvents(periodBegin, periodEnd);
|
||||||
|
|
||||||
|
let ids = events.map(e => e.id);
|
||||||
|
this._messageById.forEach((message, id) => {
|
||||||
|
if (ids.includes(id))
|
||||||
|
return;
|
||||||
|
this._messageById.delete(id);
|
||||||
|
this.removeMessage(message);
|
||||||
|
});
|
||||||
|
|
||||||
for (let i = 0; i < events.length; i++) {
|
for (let i = 0; i < events.length; i++) {
|
||||||
let event = events[i];
|
let event = events[i];
|
||||||
|
|
||||||
let message = new EventMessage(event, this._date);
|
let message = this._messageById.get(event.id);
|
||||||
|
if (!message) {
|
||||||
|
message = new EventMessage(event, this._date);
|
||||||
message.connect('close', () => {
|
message.connect('close', () => {
|
||||||
this._ignoreEvent(event);
|
this._ignoreEvent(event);
|
||||||
});
|
});
|
||||||
|
this._messageById.set(event.id, message);
|
||||||
this.addMessage(message, false);
|
this.addMessage(message, false);
|
||||||
|
} else {
|
||||||
|
this.moveMessage(message, i, false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this._reloading = false;
|
this._reloading = false;
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
const Clutter = imports.gi.Clutter;
|
const Clutter = imports.gi.Clutter;
|
||||||
const Gio = imports.gi.Gio;
|
const Gio = imports.gi.Gio;
|
||||||
|
const GLib = imports.gi.GLib;
|
||||||
const GObject = imports.gi.GObject;
|
const GObject = imports.gi.GObject;
|
||||||
const Lang = imports.lang;
|
const Lang = imports.lang;
|
||||||
const Meta = imports.gi.Meta;
|
const Meta = imports.gi.Meta;
|
||||||
@ -13,6 +14,7 @@ const Tweener = imports.ui.tweener;
|
|||||||
|
|
||||||
var FROZEN_WINDOW_BRIGHTNESS = -0.3
|
var FROZEN_WINDOW_BRIGHTNESS = -0.3
|
||||||
var DIALOG_TRANSITION_TIME = 0.15
|
var DIALOG_TRANSITION_TIME = 0.15
|
||||||
|
var ALIVE_TIMEOUT = 5000;
|
||||||
|
|
||||||
var CloseDialog = new Lang.Class({
|
var CloseDialog = new Lang.Class({
|
||||||
Name: 'CloseDialog',
|
Name: 'CloseDialog',
|
||||||
@ -26,6 +28,10 @@ var CloseDialog = new Lang.Class({
|
|||||||
this.parent();
|
this.parent();
|
||||||
this._window = window;
|
this._window = window;
|
||||||
this._dialog = null;
|
this._dialog = null;
|
||||||
|
this._tracked = undefined;
|
||||||
|
this._timeoutId = 0;
|
||||||
|
this._windowFocusChangedId = 0;
|
||||||
|
this._keyFocusChangedId = 0;
|
||||||
},
|
},
|
||||||
|
|
||||||
get window() {
|
get window() {
|
||||||
@ -93,10 +99,57 @@ var CloseDialog = new Lang.Class({
|
|||||||
this.response(Meta.CloseDialogResponse.FORCE_CLOSE);
|
this.response(Meta.CloseDialogResponse.FORCE_CLOSE);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_onFocusChanged() {
|
||||||
|
if (Meta.is_wayland_compositor())
|
||||||
|
return;
|
||||||
|
|
||||||
|
let focusWindow = global.display.focus_window;
|
||||||
|
let keyFocus = global.stage.key_focus;
|
||||||
|
|
||||||
|
let shouldTrack;
|
||||||
|
if (focusWindow != null)
|
||||||
|
shouldTrack = focusWindow == this._window;
|
||||||
|
else
|
||||||
|
shouldTrack = keyFocus && this._dialog.contains(keyFocus);
|
||||||
|
|
||||||
|
if (this._tracked === shouldTrack)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (shouldTrack)
|
||||||
|
Main.layoutManager.trackChrome(this._dialog,
|
||||||
|
{ affectsInputRegion: true });
|
||||||
|
else
|
||||||
|
Main.layoutManager.untrackChrome(this._dialog);
|
||||||
|
|
||||||
|
// The buttons are broken when they aren't added to the input region,
|
||||||
|
// so disable them properly in that case
|
||||||
|
this._dialog.buttonLayout.get_children().forEach(b => {
|
||||||
|
b.reactive = shouldTrack;
|
||||||
|
});
|
||||||
|
|
||||||
|
this._tracked = shouldTrack;
|
||||||
|
},
|
||||||
|
|
||||||
vfunc_show() {
|
vfunc_show() {
|
||||||
if (this._dialog != null)
|
if (this._dialog != null)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
Meta.disable_unredirect_for_screen(global.screen);
|
||||||
|
|
||||||
|
this._timeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, ALIVE_TIMEOUT,
|
||||||
|
() => {
|
||||||
|
this._window.check_alive(global.display.get_current_time_roundtrip());
|
||||||
|
return GLib.SOURCE_CONTINUE;
|
||||||
|
});
|
||||||
|
|
||||||
|
this._windowFocusChangedId =
|
||||||
|
global.display.connect('notify::focus-window',
|
||||||
|
this._onFocusChanged.bind(this));
|
||||||
|
|
||||||
|
this._keyFocusChangedId =
|
||||||
|
global.stage.connect('notify::key-focus',
|
||||||
|
this._onFocusChanged.bind(this));
|
||||||
|
|
||||||
this._addWindowEffect();
|
this._addWindowEffect();
|
||||||
this._initDialog();
|
this._initDialog();
|
||||||
|
|
||||||
@ -107,9 +160,7 @@ var CloseDialog = new Lang.Class({
|
|||||||
{ scale_y: 1,
|
{ scale_y: 1,
|
||||||
transition: 'linear',
|
transition: 'linear',
|
||||||
time: DIALOG_TRANSITION_TIME,
|
time: DIALOG_TRANSITION_TIME,
|
||||||
onComplete: () => {
|
onComplete: this._onFocusChanged.bind(this)
|
||||||
Main.layoutManager.trackChrome(this._dialog, { affectsInputRegion: true });
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -117,6 +168,17 @@ var CloseDialog = new Lang.Class({
|
|||||||
if (this._dialog == null)
|
if (this._dialog == null)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
Meta.enable_unredirect_for_screen(global.screen);
|
||||||
|
|
||||||
|
GLib.source_remove(this._timeoutId);
|
||||||
|
this._timeoutId = 0;
|
||||||
|
|
||||||
|
global.display.disconnect(this._windowFocusChangedId)
|
||||||
|
this._windowFocusChangedId = 0;
|
||||||
|
|
||||||
|
global.stage.disconnect(this._keyFocusChangedId);
|
||||||
|
this._keyFocusChangedId = 0;
|
||||||
|
|
||||||
let dialog = this._dialog;
|
let dialog = this._dialog;
|
||||||
this._dialog = null;
|
this._dialog = null;
|
||||||
this._removeWindowEffect();
|
this._removeWindowEffect();
|
||||||
|
@ -210,6 +210,10 @@ var AutomountManager = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_onVolumeRemoved(monitor, volume) {
|
_onVolumeRemoved(monitor, volume) {
|
||||||
|
if (volume._allowAutorunExpireId && volume._allowAutorunExpireId > 0) {
|
||||||
|
Mainloop.source_remove(volume._allowAutorunExpireId);
|
||||||
|
delete volume._allowAutorunExpireId;
|
||||||
|
}
|
||||||
this._volumeQueue =
|
this._volumeQueue =
|
||||||
this._volumeQueue.filter(element => (element != volume));
|
this._volumeQueue.filter(element => (element != volume));
|
||||||
},
|
},
|
||||||
@ -234,8 +238,10 @@ var AutomountManager = new Lang.Class({
|
|||||||
_allowAutorunExpire(volume) {
|
_allowAutorunExpire(volume) {
|
||||||
let id = Mainloop.timeout_add_seconds(AUTORUN_EXPIRE_TIMEOUT_SECS, () => {
|
let id = Mainloop.timeout_add_seconds(AUTORUN_EXPIRE_TIMEOUT_SECS, () => {
|
||||||
volume.allowAutorun = false;
|
volume.allowAutorun = false;
|
||||||
|
delete volume._allowAutorunExpireId;
|
||||||
return GLib.SOURCE_REMOVE;
|
return GLib.SOURCE_REMOVE;
|
||||||
});
|
});
|
||||||
|
volume._allowAutorunExpireId = id;
|
||||||
GLib.Source.set_name_by_id(id, '[gnome-shell] volume.allowAutorun');
|
GLib.Source.set_name_by_id(id, '[gnome-shell] volume.allowAutorun');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -655,7 +655,7 @@ var NetworkAgent = new Lang.Class({
|
|||||||
switch (connectionType) {
|
switch (connectionType) {
|
||||||
case '802-11-wireless':
|
case '802-11-wireless':
|
||||||
let wirelessSetting = connection.get_setting_wireless();
|
let wirelessSetting = connection.get_setting_wireless();
|
||||||
let ssid = NM.utils_ssid_to_utf8(wirelessSetting.get_ssid());
|
let ssid = NM.utils_ssid_to_utf8(wirelessSetting.get_ssid().get_data());
|
||||||
title = _("Authentication required by wireless network");
|
title = _("Authentication required by wireless network");
|
||||||
body = _("Passwords or encryption keys are required to access the wireless network “%s”.").format(ssid);
|
body = _("Passwords or encryption keys are required to access the wireless network “%s”.").format(ssid);
|
||||||
break;
|
break;
|
||||||
|
@ -201,7 +201,9 @@ var AuthenticationDialog = new Lang.Class({
|
|||||||
close(timestamp) {
|
close(timestamp) {
|
||||||
this.parent(timestamp);
|
this.parent(timestamp);
|
||||||
|
|
||||||
|
if (this._sessionUpdatedId)
|
||||||
Main.sessionMode.disconnect(this._sessionUpdatedId);
|
Main.sessionMode.disconnect(this._sessionUpdatedId);
|
||||||
|
this._sessionUpdatedId = 0;
|
||||||
},
|
},
|
||||||
|
|
||||||
_ensureOpen() {
|
_ensureOpen() {
|
||||||
|
@ -52,6 +52,8 @@ var DashItemContainer = new Lang.Class({
|
|||||||
this.animatingOut = false;
|
this.animatingOut = false;
|
||||||
|
|
||||||
this.connect('destroy', () => {
|
this.connect('destroy', () => {
|
||||||
|
if (this.child != null)
|
||||||
|
this.child.destroy();
|
||||||
this.label.destroy();
|
this.label.destroy();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -183,7 +183,7 @@ var MessageDialogContent = new Lang.Class({
|
|||||||
this[`_${prop}`].add_style_class_name(`message-dialog-${prop}`);
|
this[`_${prop}`].add_style_class_name(`message-dialog-${prop}`);
|
||||||
});
|
});
|
||||||
|
|
||||||
let textProps = { ellipsize_mode: Pango.EllipsizeMode.NONE,
|
let textProps = { ellipsize: Pango.EllipsizeMode.NONE,
|
||||||
line_wrap: true };
|
line_wrap: true };
|
||||||
Object.assign(this._subtitle.clutter_text, textProps);
|
Object.assign(this._subtitle.clutter_text, textProps);
|
||||||
Object.assign(this._body.clutter_text, textProps);
|
Object.assign(this._body.clutter_text, textProps);
|
||||||
|
32
js/ui/dnd.js
32
js/ui/dnd.js
@ -125,6 +125,16 @@ var _Draggable = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_onTouchEvent(actor, event) {
|
_onTouchEvent(actor, event) {
|
||||||
|
// Here we only handle touch events on wayland. On X11
|
||||||
|
// we do get emulated pointer events, which already works
|
||||||
|
// for single-touch cases. Besides, the X11 passive touch grab
|
||||||
|
// set up by Mutter will make us see first the touch events
|
||||||
|
// and later the pointer events, so it will look like two
|
||||||
|
// unrelated series of events, we want to avoid double handling
|
||||||
|
// in these cases.
|
||||||
|
if (!Meta.is_wayland_compositor())
|
||||||
|
return Clutter.EVENT_PROPAGATE;
|
||||||
|
|
||||||
if (event.type() != Clutter.EventType.TOUCH_BEGIN ||
|
if (event.type() != Clutter.EventType.TOUCH_BEGIN ||
|
||||||
!global.display.is_pointer_emulating_sequence(event.get_event_sequence()))
|
!global.display.is_pointer_emulating_sequence(event.get_event_sequence()))
|
||||||
return Clutter.EVENT_PROPAGATE;
|
return Clutter.EVENT_PROPAGATE;
|
||||||
@ -396,10 +406,15 @@ var _Draggable = new Lang.Class({
|
|||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_pickTargetActor() {
|
||||||
|
return this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL,
|
||||||
|
this._dragX, this._dragY);
|
||||||
|
},
|
||||||
|
|
||||||
_updateDragHover() {
|
_updateDragHover() {
|
||||||
this._updateHoverId = 0;
|
this._updateHoverId = 0;
|
||||||
let target = this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL,
|
let target = this._pickTargetActor();
|
||||||
this._dragX, this._dragY);
|
|
||||||
let dragEvent = {
|
let dragEvent = {
|
||||||
x: this._dragX,
|
x: this._dragX,
|
||||||
y: this._dragY,
|
y: this._dragY,
|
||||||
@ -407,6 +422,18 @@ var _Draggable = new Lang.Class({
|
|||||||
source: this.actor._delegate,
|
source: this.actor._delegate,
|
||||||
targetActor: target
|
targetActor: target
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let targetActorDestroyHandlerId;
|
||||||
|
let handleTargetActorDestroyClosure;
|
||||||
|
handleTargetActorDestroyClosure = () => {
|
||||||
|
target = this._pickTargetActor();
|
||||||
|
dragEvent.targetActor = target;
|
||||||
|
targetActorDestroyHandlerId =
|
||||||
|
target.connect('destroy', handleTargetActorDestroyClosure);
|
||||||
|
};
|
||||||
|
targetActorDestroyHandlerId =
|
||||||
|
target.connect('destroy', handleTargetActorDestroyClosure);
|
||||||
|
|
||||||
for (let i = 0; i < dragMonitors.length; i++) {
|
for (let i = 0; i < dragMonitors.length; i++) {
|
||||||
let motionFunc = dragMonitors[i].dragMotion;
|
let motionFunc = dragMonitors[i].dragMotion;
|
||||||
if (motionFunc) {
|
if (motionFunc) {
|
||||||
@ -417,6 +444,7 @@ var _Draggable = new Lang.Class({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
dragEvent.targetActor.disconnect(targetActorDestroyHandlerId);
|
||||||
|
|
||||||
while (target) {
|
while (target) {
|
||||||
if (target._delegate && target._delegate.handleDragOver) {
|
if (target._delegate && target._delegate.handleDragOver) {
|
||||||
|
@ -760,7 +760,7 @@ var EndSessionDialog = new Lang.Class({
|
|||||||
let updatePrepared = this._pkOfflineProxy.UpdatePrepared;
|
let updatePrepared = this._pkOfflineProxy.UpdatePrepared;
|
||||||
let updatesAllowed = this._updatesPermission && this._updatesPermission.allowed;
|
let updatesAllowed = this._updatesPermission && this._updatesPermission.allowed;
|
||||||
|
|
||||||
_setCheckBoxLabel(this._checkBox, dialogContent.checkBoxText);
|
_setCheckBoxLabel(this._checkBox, dialogContent.checkBoxText || '');
|
||||||
this._checkBox.actor.visible = (dialogContent.checkBoxText && updatePrepared && updatesAllowed);
|
this._checkBox.actor.visible = (dialogContent.checkBoxText && updatePrepared && updatesAllowed);
|
||||||
this._checkBox.actor.checked = (updatePrepared && updateTriggered);
|
this._checkBox.actor.checked = (updatePrepared && updateTriggered);
|
||||||
|
|
||||||
|
@ -17,6 +17,9 @@ const Gtk = imports.gi.Gtk;
|
|||||||
const Lang = imports.lang;
|
const Lang = imports.lang;
|
||||||
const Shell = imports.gi.Shell;
|
const Shell = imports.gi.Shell;
|
||||||
const St = imports.gi.St;
|
const St = imports.gi.St;
|
||||||
|
const System = imports.system;
|
||||||
|
|
||||||
|
let _localTimeZone = null;
|
||||||
|
|
||||||
// We can't import shell JS modules yet, because they may have
|
// We can't import shell JS modules yet, because they may have
|
||||||
// variable initializations, etc, that depend on init() already having
|
// variable initializations, etc, that depend on init() already having
|
||||||
@ -116,9 +119,26 @@ function init() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Override to clear our own timezone cache as well
|
||||||
|
const origClearDateCaches = System.clearDateCaches;
|
||||||
|
System.clearDateCaches = function () {
|
||||||
|
_localTimeZone = null;
|
||||||
|
origClearDateCaches();
|
||||||
|
};
|
||||||
|
|
||||||
// Work around https://bugzilla.mozilla.org/show_bug.cgi?id=508783
|
// Work around https://bugzilla.mozilla.org/show_bug.cgi?id=508783
|
||||||
Date.prototype.toLocaleFormat = function(format) {
|
Date.prototype.toLocaleFormat = function(format) {
|
||||||
return Shell.util_format_date(format, this.getTime());
|
if (_localTimeZone === null)
|
||||||
|
_localTimeZone = GLib.TimeZone.new_local();
|
||||||
|
|
||||||
|
let dt = GLib.DateTime.new(_localTimeZone,
|
||||||
|
this.getYear(),
|
||||||
|
this.getMonth() + 1,
|
||||||
|
this.getDate(),
|
||||||
|
this.getHours(),
|
||||||
|
this.getMinutes(),
|
||||||
|
this.getSeconds());
|
||||||
|
return dt ? dt.format(format) : '';
|
||||||
};
|
};
|
||||||
|
|
||||||
let slowdownEnv = GLib.getenv('GNOME_SHELL_SLOWDOWN_FACTOR');
|
let slowdownEnv = GLib.getenv('GNOME_SHELL_SLOWDOWN_FACTOR');
|
||||||
|
@ -76,6 +76,7 @@ function disableExtension(uuid) {
|
|||||||
if (extension.stylesheet) {
|
if (extension.stylesheet) {
|
||||||
let theme = St.ThemeContext.get_for_stage(global.stage).get_theme();
|
let theme = St.ThemeContext.get_for_stage(global.stage).get_theme();
|
||||||
theme.unload_stylesheet(extension.stylesheet);
|
theme.unload_stylesheet(extension.stylesheet);
|
||||||
|
delete extension.stylesheet;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -115,13 +116,18 @@ function enableExtension(uuid) {
|
|||||||
extensionOrder.push(uuid);
|
extensionOrder.push(uuid);
|
||||||
|
|
||||||
let stylesheetNames = [global.session_mode + '.css', 'stylesheet.css'];
|
let stylesheetNames = [global.session_mode + '.css', 'stylesheet.css'];
|
||||||
for (let i = 0; i < stylesheetNames.length; i++) {
|
|
||||||
let stylesheetFile = extension.dir.get_child(stylesheetNames[i]);
|
|
||||||
if (stylesheetFile.query_exists(null)) {
|
|
||||||
let theme = St.ThemeContext.get_for_stage(global.stage).get_theme();
|
let theme = St.ThemeContext.get_for_stage(global.stage).get_theme();
|
||||||
|
for (let i = 0; i < stylesheetNames.length; i++) {
|
||||||
|
try {
|
||||||
|
let stylesheetFile = extension.dir.get_child(stylesheetNames[i]);
|
||||||
theme.load_stylesheet(stylesheetFile);
|
theme.load_stylesheet(stylesheetFile);
|
||||||
extension.stylesheet = stylesheetFile;
|
extension.stylesheet = stylesheetFile;
|
||||||
break;
|
break;
|
||||||
|
} catch (e) {
|
||||||
|
if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.NOT_FOUND))
|
||||||
|
continue; // not an error
|
||||||
|
log(`Failed to load stylesheet for extension ${uuid}: ${e.message}`);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -131,6 +137,10 @@ function enableExtension(uuid) {
|
|||||||
_signals.emit('extension-state-changed', extension);
|
_signals.emit('extension-state-changed', extension);
|
||||||
return;
|
return;
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
|
if (extension.stylesheet) {
|
||||||
|
theme.unload_stylesheet(extension.stylesheet);
|
||||||
|
delete extension.stylesheet;
|
||||||
|
}
|
||||||
logExtensionError(uuid, e);
|
logExtensionError(uuid, e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -418,6 +418,11 @@ var IconGrid = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_animationDone() {
|
_animationDone() {
|
||||||
|
this._clonesAnimating.forEach(clone => {
|
||||||
|
clone.source.reactive = true;
|
||||||
|
clone.source.opacity = 255;
|
||||||
|
clone.destroy();
|
||||||
|
});
|
||||||
this._clonesAnimating = [];
|
this._clonesAnimating = [];
|
||||||
this.emit('animation-done');
|
this.emit('animation-done');
|
||||||
},
|
},
|
||||||
@ -538,10 +543,6 @@ var IconGrid = new Lang.Class({
|
|||||||
onComplete: () => {
|
onComplete: () => {
|
||||||
if (isLastItem)
|
if (isLastItem)
|
||||||
this._animationDone();
|
this._animationDone();
|
||||||
|
|
||||||
actor.opacity = 255;
|
|
||||||
actor.reactive = true;
|
|
||||||
actorClone.destroy();
|
|
||||||
}};
|
}};
|
||||||
fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM,
|
fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM,
|
||||||
transition: 'easeInOutQuad',
|
transition: 'easeInOutQuad',
|
||||||
@ -562,12 +563,8 @@ var IconGrid = new Lang.Class({
|
|||||||
scale_x: scaleX,
|
scale_x: scaleX,
|
||||||
scale_y: scaleY,
|
scale_y: scaleY,
|
||||||
onComplete: () => {
|
onComplete: () => {
|
||||||
if (isLastItem) {
|
if (isLastItem)
|
||||||
this._animationDone();
|
this._animationDone();
|
||||||
this._restoreItemsOpacity();
|
|
||||||
}
|
|
||||||
actor.reactive = true;
|
|
||||||
actorClone.destroy();
|
|
||||||
}};
|
}};
|
||||||
fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM,
|
fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM,
|
||||||
transition: 'easeInOutQuad',
|
transition: 'easeInOutQuad',
|
||||||
@ -581,12 +578,6 @@ var IconGrid = new Lang.Class({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
_restoreItemsOpacity() {
|
|
||||||
for (let index = 0; index < this._items.length; index++) {
|
|
||||||
this._items[index].actor.opacity = 255;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_getAllocatedChildSizeAndSpacing(child) {
|
_getAllocatedChildSizeAndSpacing(child) {
|
||||||
let [,, natWidth, natHeight] = child.get_preferred_size();
|
let [,, natWidth, natHeight] = child.get_preferred_size();
|
||||||
let width = Math.min(this._getHItemSize(), natWidth);
|
let width = Math.min(this._getHItemSize(), natWidth);
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
||||||
|
|
||||||
const FocusCaretTracker = imports.ui.focusCaretTracker;
|
|
||||||
const Atspi = imports.gi.Atspi;
|
const Atspi = imports.gi.Atspi;
|
||||||
const Clutter = imports.gi.Clutter;
|
const Clutter = imports.gi.Clutter;
|
||||||
const Gdk = imports.gi.Gdk;
|
const Gdk = imports.gi.Gdk;
|
||||||
@ -13,6 +12,7 @@ const Signals = imports.signals;
|
|||||||
const St = imports.gi.St;
|
const St = imports.gi.St;
|
||||||
const InputSourceManager = imports.ui.status.keyboard;
|
const InputSourceManager = imports.ui.status.keyboard;
|
||||||
|
|
||||||
|
const IBusManager = imports.misc.ibusManager;
|
||||||
const BoxPointer = imports.ui.boxpointer;
|
const BoxPointer = imports.ui.boxpointer;
|
||||||
const Layout = imports.ui.layout;
|
const Layout = imports.ui.layout;
|
||||||
const Main = imports.ui.main;
|
const Main = imports.ui.main;
|
||||||
@ -261,6 +261,7 @@ var Key = new Lang.Class({
|
|||||||
this._extended_keyboard = null;
|
this._extended_keyboard = null;
|
||||||
this._pressTimeoutId = 0;
|
this._pressTimeoutId = 0;
|
||||||
this._capturedPress = false;
|
this._capturedPress = false;
|
||||||
|
|
||||||
this._capturedEventId = 0;
|
this._capturedEventId = 0;
|
||||||
this._unmapId = 0;
|
this._unmapId = 0;
|
||||||
this._longPress = false;
|
this._longPress = false;
|
||||||
@ -484,6 +485,79 @@ var KeyboardModel = new Lang.Class({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
var FocusTracker = new Lang.Class({
|
||||||
|
Name: 'FocusTracker',
|
||||||
|
|
||||||
|
_init() {
|
||||||
|
this._currentWindow = null;
|
||||||
|
this._currentWindowPositionId = 0;
|
||||||
|
|
||||||
|
global.screen.get_display().connect('notify::focus-window', () => {
|
||||||
|
this._setCurrentWindow(global.screen.get_display().focus_window);
|
||||||
|
this.emit('window-changed', this._currentWindow);
|
||||||
|
});
|
||||||
|
|
||||||
|
/* Valid for wayland clients */
|
||||||
|
Main.inputMethod.connect('cursor-location-changed', (o, rect) => {
|
||||||
|
let newRect = { x: rect.get_x(), y: rect.get_y(), width: rect.get_width(), height: rect.get_height() };
|
||||||
|
this._setCurrentRect(newRect);
|
||||||
|
});
|
||||||
|
|
||||||
|
this._ibusManager = IBusManager.getIBusManager();
|
||||||
|
this._ibusManager.connect('set-cursor-location', (manager, rect) => {
|
||||||
|
/* Valid for X11 clients only */
|
||||||
|
if (Main.inputMethod.currentFocus)
|
||||||
|
return;
|
||||||
|
|
||||||
|
this._setCurrentRect(rect);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
get currentWindow() {
|
||||||
|
return this._currentWindow;
|
||||||
|
},
|
||||||
|
|
||||||
|
_setCurrentWindow(window) {
|
||||||
|
if (this._currentWindow)
|
||||||
|
this._currentWindow.disconnect(this._currentWindowPositionId);
|
||||||
|
|
||||||
|
this._currentWindow = window;
|
||||||
|
if (window) {
|
||||||
|
this._currentWindowPositionId = this._currentWindow.connect('position-changed', () => {
|
||||||
|
if (global.display.get_grab_op() == Meta.GrabOp.NONE)
|
||||||
|
this.emit('position-changed');
|
||||||
|
else
|
||||||
|
this.emit('reset');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_setCurrentRect(rect) {
|
||||||
|
if (this._currentWindow) {
|
||||||
|
let frameRect = this._currentWindow.get_frame_rect();
|
||||||
|
rect.x -= frameRect.x;
|
||||||
|
rect.y -= frameRect.y;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._rect = rect;
|
||||||
|
this.emit('position-changed');
|
||||||
|
},
|
||||||
|
|
||||||
|
getCurrentRect() {
|
||||||
|
let rect = { x: this._rect.x, y: this._rect.y,
|
||||||
|
width: this._rect.width, height: this._rect.height };
|
||||||
|
|
||||||
|
if (this._currentWindow) {
|
||||||
|
let frameRect = this._currentWindow.get_frame_rect();
|
||||||
|
rect.x += frameRect.x;
|
||||||
|
rect.y += frameRect.y;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rect;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Signals.addSignalMethods(FocusTracker.prototype);
|
||||||
|
|
||||||
var Keyboard = new Lang.Class({
|
var Keyboard = new Lang.Class({
|
||||||
Name: 'Keyboard',
|
Name: 'Keyboard',
|
||||||
|
|
||||||
@ -491,15 +565,10 @@ var Keyboard = new Lang.Class({
|
|||||||
this.actor = null;
|
this.actor = null;
|
||||||
this._focusInExtendedKeys = false;
|
this._focusInExtendedKeys = false;
|
||||||
|
|
||||||
this._focusCaretTracker = new FocusCaretTracker.FocusCaretTracker();
|
|
||||||
this._focusCaretTracker.connect('focus-changed', this._onFocusChanged.bind(this));
|
|
||||||
this._focusCaretTracker.connect('caret-moved', this._onCaretMoved.bind(this));
|
|
||||||
this._languagePopup = null;
|
this._languagePopup = null;
|
||||||
this._currentAccessible = null;
|
|
||||||
this._caretTrackingEnabled = false;
|
|
||||||
this._updateCaretPositionId = 0;
|
|
||||||
this._currentFocusWindow = null;
|
this._currentFocusWindow = null;
|
||||||
this._originalWindowY = null;
|
this._animFocusedWindow = null;
|
||||||
|
this._delayedAnimFocusWindow = null;
|
||||||
|
|
||||||
this._enableKeyboard = false; // a11y settings value
|
this._enableKeyboard = false; // a11y settings value
|
||||||
this._enabled = false; // enabled state (by setting or device type)
|
this._enabled = false; // enabled state (by setting or device type)
|
||||||
@ -510,6 +579,14 @@ var Keyboard = new Lang.Class({
|
|||||||
this._lastDeviceId = null;
|
this._lastDeviceId = null;
|
||||||
this._suggestions = null;
|
this._suggestions = null;
|
||||||
|
|
||||||
|
this._focusTracker = new FocusTracker();
|
||||||
|
this._focusTracker.connect('position-changed', this._onFocusPositionChanged.bind(this));
|
||||||
|
this._focusTracker.connect('reset', () => {
|
||||||
|
this._delayedAnimFocusWindow = null;
|
||||||
|
this._animFocusedWindow = null;
|
||||||
|
this._oskFocusWindow = null;
|
||||||
|
});
|
||||||
|
|
||||||
Meta.get_backend().connect('last-device-changed',
|
Meta.get_backend().connect('last-device-changed',
|
||||||
(backend, deviceId) => {
|
(backend, deviceId) => {
|
||||||
let manager = Clutter.DeviceManager.get_default();
|
let manager = Clutter.DeviceManager.get_default();
|
||||||
@ -532,102 +609,15 @@ var Keyboard = new Lang.Class({
|
|||||||
this._keyboardRestingId = 0;
|
this._keyboardRestingId = 0;
|
||||||
|
|
||||||
Main.layoutManager.connect('monitors-changed', this._relayout.bind(this));
|
Main.layoutManager.connect('monitors-changed', this._relayout.bind(this));
|
||||||
//Main.inputMethod.connect('cursor-location-changed', (o, rect) => {
|
|
||||||
// if (this._keyboardVisible) {
|
|
||||||
// let currentWindow = global.screen.get_display().focus_window;
|
|
||||||
// this.setCursorLocation(currentWindow, rect.get_x(), rect.get_y(),
|
|
||||||
// rect.get_width(), rect.get_height());
|
|
||||||
// }
|
|
||||||
//});
|
|
||||||
},
|
},
|
||||||
|
|
||||||
get visible() {
|
get visible() {
|
||||||
return this._keyboardVisible;
|
return this._keyboardVisible;
|
||||||
},
|
},
|
||||||
|
|
||||||
_setCaretTrackerEnabled(enabled) {
|
_onFocusPositionChanged(focusTracker) {
|
||||||
if (this._caretTrackingEnabled == enabled)
|
let rect = focusTracker.getCurrentRect();
|
||||||
return;
|
this.setCursorLocation(focusTracker.currentWindow, rect.x, rect.y, rect.width, rect.height);
|
||||||
|
|
||||||
this._caretTrackingEnabled = enabled;
|
|
||||||
|
|
||||||
if (enabled) {
|
|
||||||
this._focusCaretTracker.registerFocusListener();
|
|
||||||
this._focusCaretTracker.registerCaretListener();
|
|
||||||
} else {
|
|
||||||
this._focusCaretTracker.deregisterFocusListener();
|
|
||||||
this._focusCaretTracker.deregisterCaretListener();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_updateCaretPosition(accessible) {
|
|
||||||
if (this._updateCaretPositionId)
|
|
||||||
GLib.source_remove(this._updateCaretPositionId);
|
|
||||||
if (!this._keyboardRequested)
|
|
||||||
return;
|
|
||||||
this._updateCaretPositionId = GLib.idle_add(GLib.PRIORITY_DEFAULT_IDLE, () => {
|
|
||||||
this._updateCaretPositionId = 0;
|
|
||||||
|
|
||||||
let currentWindow = global.screen.get_display().focus_window;
|
|
||||||
if (!currentWindow) {
|
|
||||||
this.setCursorLocation(null);
|
|
||||||
return GLib.SOURCE_REMOVE;
|
|
||||||
}
|
|
||||||
|
|
||||||
let windowRect = currentWindow.get_frame_rect();
|
|
||||||
let text = accessible.get_text_iface();
|
|
||||||
let component = accessible.get_component_iface();
|
|
||||||
|
|
||||||
try {
|
|
||||||
let caretOffset = text.get_caret_offset();
|
|
||||||
let caretRect = text.get_character_extents(caretOffset, Atspi.CoordType.WINDOW);
|
|
||||||
let focusRect = component.get_extents(Atspi.CoordType.WINDOW);
|
|
||||||
|
|
||||||
if (caretRect.width == 0 && caretRect.height == 0)
|
|
||||||
caretRect = focusRect;
|
|
||||||
|
|
||||||
this.setCursorLocation(currentWindow, caretRect.x, caretRect.y, caretRect.width, caretRect.height);
|
|
||||||
} catch (e) {
|
|
||||||
log('Error updating caret position for OSK: ' + e.message);
|
|
||||||
}
|
|
||||||
|
|
||||||
return GLib.SOURCE_REMOVE;
|
|
||||||
});
|
|
||||||
|
|
||||||
GLib.Source.set_name_by_id(this._updateCaretPositionId, '[gnome-shell] this._updateCaretPosition');
|
|
||||||
},
|
|
||||||
|
|
||||||
_focusIsTextEntry(accessible) {
|
|
||||||
try {
|
|
||||||
let role = accessible.get_role();
|
|
||||||
let stateSet = accessible.get_state_set();
|
|
||||||
return stateSet.contains(Atspi.StateType.EDITABLE) || role == Atspi.Role.TERMINAL;
|
|
||||||
} catch (e) {
|
|
||||||
log('Error determining accessible role: ' + e.message);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_onFocusChanged(caretTracker, event) {
|
|
||||||
let accessible = event.source;
|
|
||||||
if (!this._focusIsTextEntry(accessible))
|
|
||||||
return;
|
|
||||||
|
|
||||||
let focused = event.detail1 != 0;
|
|
||||||
if (focused) {
|
|
||||||
this._currentAccessible = accessible;
|
|
||||||
this._updateCaretPosition(accessible);
|
|
||||||
this.show(Main.layoutManager.focusIndex);
|
|
||||||
} else if (this._currentAccessible == accessible) {
|
|
||||||
this._currentAccessible = null;
|
|
||||||
this.hide();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_onCaretMoved(caretTracker, event) {
|
|
||||||
let accessible = event.source;
|
|
||||||
if (this._currentAccessible == accessible)
|
|
||||||
this._updateCaretPosition(accessible);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
_lastDeviceIsTouchscreen() {
|
_lastDeviceIsTouchscreen() {
|
||||||
@ -650,8 +640,6 @@ var Keyboard = new Lang.Class({
|
|||||||
if (!this._enabled && !this._keyboardController)
|
if (!this._enabled && !this._keyboardController)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
this._setCaretTrackerEnabled(this._enabled);
|
|
||||||
|
|
||||||
if (this._enabled && !this._keyboardController)
|
if (this._enabled && !this._keyboardController)
|
||||||
this._setupKeyboard();
|
this._setupKeyboard();
|
||||||
else if (!this._enabled)
|
else if (!this._enabled)
|
||||||
@ -936,9 +924,11 @@ var Keyboard = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_relayout() {
|
_relayout() {
|
||||||
if (this.actor == null)
|
|
||||||
return;
|
|
||||||
let monitor = Main.layoutManager.keyboardMonitor;
|
let monitor = Main.layoutManager.keyboardMonitor;
|
||||||
|
|
||||||
|
if (this.actor == null || monitor == null)
|
||||||
|
return;
|
||||||
|
|
||||||
let maxHeight = monitor.height / 3;
|
let maxHeight = monitor.height / 3;
|
||||||
this.actor.width = monitor.width;
|
this.actor.width = monitor.width;
|
||||||
this.actor.height = maxHeight;
|
this.actor.height = maxHeight;
|
||||||
@ -1027,11 +1017,14 @@ var Keyboard = new Lang.Class({
|
|||||||
if (!this._keyboardRequested)
|
if (!this._keyboardRequested)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (this._currentAccessible)
|
|
||||||
this._updateCaretPosition(this._currentAccessible);
|
|
||||||
Main.layoutManager.keyboardIndex = monitor;
|
Main.layoutManager.keyboardIndex = monitor;
|
||||||
this._relayout();
|
this._relayout();
|
||||||
Main.layoutManager.showKeyboard();
|
Main.layoutManager.showKeyboard();
|
||||||
|
|
||||||
|
if (this._delayedAnimFocusWindow) {
|
||||||
|
this._setAnimationWindow(this._delayedAnimFocusWindow);
|
||||||
|
this._delayedAnimFocusWindow = null;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
hide() {
|
hide() {
|
||||||
@ -1102,8 +1095,9 @@ var Keyboard = new Lang.Class({
|
|||||||
window.move_frame(true, frameRect.x, frameRect.y);
|
window.move_frame(true, frameRect.x, frameRect.y);
|
||||||
},
|
},
|
||||||
|
|
||||||
_animateWindow(window, show, deltaY) {
|
_animateWindow(window, show) {
|
||||||
let windowActor = window.get_compositor_private();
|
let windowActor = window.get_compositor_private();
|
||||||
|
let deltaY = Main.layoutManager.keyboardBox.height;
|
||||||
if (!windowActor)
|
if (!windowActor)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -1124,35 +1118,39 @@ var Keyboard = new Lang.Class({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
setCursorLocation(window, x, y , w, h) {
|
_setAnimationWindow(window) {
|
||||||
if (window == this._oskFocusWindow)
|
if (this._animFocusedWindow == window)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (this._oskFocusWindow) {
|
if (this._animFocusedWindow)
|
||||||
let display = global.screen.get_display();
|
this._animateWindow(this._animFocusedWindow, false);
|
||||||
|
if (window)
|
||||||
|
this._animateWindow(window, true);
|
||||||
|
|
||||||
if (display.get_grab_op() == Meta.GrabOp.NONE ||
|
this._animFocusedWindow = window;
|
||||||
display.get_focus_window() != this._oskFocusWindow)
|
},
|
||||||
this._animateWindow(this._oskFocusWindow, false, this._oskFocusWindowDelta);
|
|
||||||
|
|
||||||
this._oskFocusWindow = null;
|
setCursorLocation(window, x, y , w, h) {
|
||||||
this._oskFocusWindowDelta = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (window) {
|
|
||||||
let monitor = Main.layoutManager.keyboardMonitor;
|
let monitor = Main.layoutManager.keyboardMonitor;
|
||||||
|
|
||||||
|
if (window && monitor) {
|
||||||
let keyboardHeight = Main.layoutManager.keyboardBox.height;
|
let keyboardHeight = Main.layoutManager.keyboardBox.height;
|
||||||
let frameRect = window.get_frame_rect();
|
let focusObscured = false;
|
||||||
let windowActor = window.get_compositor_private();
|
|
||||||
let delta = 0;
|
|
||||||
|
|
||||||
if (frameRect.y + y + h >= monitor.height - keyboardHeight)
|
if (y + h >= monitor.y + monitor.height - keyboardHeight) {
|
||||||
delta = keyboardHeight;
|
if (this._keyboardVisible)
|
||||||
|
this._setAnimationWindow(window);
|
||||||
this._animateWindow(window, true, delta);
|
else
|
||||||
this._oskFocusWindow = window;
|
this._delayedAnimFocusWindow = window;
|
||||||
this._oskFocusWindowDelta = delta;
|
} else if (y < keyboardHeight) {
|
||||||
|
this._delayedAnimFocusWindow = null;
|
||||||
|
this._setAnimationWindow(null);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
this._setAnimationWindow(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
this._oskFocusWindow = window;
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -203,6 +203,7 @@ var LayoutManager = new Lang.Class({
|
|||||||
|
|
||||||
// Set up stage hierarchy to group all UI actors under one container.
|
// Set up stage hierarchy to group all UI actors under one container.
|
||||||
this.uiGroup = new Shell.GenericContainer({ name: 'uiGroup' });
|
this.uiGroup = new Shell.GenericContainer({ name: 'uiGroup' });
|
||||||
|
this.uiGroup.set_flags(Clutter.ActorFlags.NO_LAYOUT);
|
||||||
this.uiGroup.connect('allocate', (actor, box, flags) => {
|
this.uiGroup.connect('allocate', (actor, box, flags) => {
|
||||||
let children = actor.get_children();
|
let children = actor.get_children();
|
||||||
for (let i = 0; i < children.length; i++)
|
for (let i = 0; i < children.length; i++)
|
||||||
@ -557,6 +558,8 @@ var LayoutManager = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
get focusMonitor() {
|
get focusMonitor() {
|
||||||
|
if (this.focusIndex < 0)
|
||||||
|
return null;
|
||||||
return this.monitors[this.focusIndex];
|
return this.monitors[this.focusIndex];
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -19,7 +19,6 @@ const MagnifierDBus = imports.ui.magnifierDBus;
|
|||||||
const Params = imports.misc.params;
|
const Params = imports.misc.params;
|
||||||
const PointerWatcher = imports.ui.pointerWatcher;
|
const PointerWatcher = imports.ui.pointerWatcher;
|
||||||
|
|
||||||
var MOUSE_POLL_FREQUENCY = 50;
|
|
||||||
var CROSSHAIRS_CLIP_SIZE = [100, 100];
|
var CROSSHAIRS_CLIP_SIZE = [100, 100];
|
||||||
var NO_CHANGE = 0.0;
|
var NO_CHANGE = 0.0;
|
||||||
|
|
||||||
@ -152,8 +151,10 @@ var Magnifier = new Lang.Class({
|
|||||||
* Turn on mouse tracking, if not already doing so.
|
* Turn on mouse tracking, if not already doing so.
|
||||||
*/
|
*/
|
||||||
startTrackingMouse() {
|
startTrackingMouse() {
|
||||||
if (!this._pointerWatch)
|
if (!this._pointerWatch) {
|
||||||
this._pointerWatch = PointerWatcher.getPointerWatcher().addWatch(MOUSE_POLL_FREQUENCY, this.scrollToMousePos.bind(this));
|
let interval = 1000 / Clutter.get_default_frame_rate();
|
||||||
|
this._pointerWatch = PointerWatcher.getPointerWatcher().addWatch(interval, this.scrollToMousePos.bind(this));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -256,6 +256,14 @@ function _getStylesheet(name) {
|
|||||||
if (stylesheet.query_exists(null))
|
if (stylesheet.query_exists(null))
|
||||||
return stylesheet;
|
return stylesheet;
|
||||||
|
|
||||||
|
let dataDirs = GLib.get_system_data_dirs();
|
||||||
|
for (let i = 0; i < dataDirs.length; i++) {
|
||||||
|
let path = GLib.build_filenamev([dataDirs[i], 'gnome-shell', 'theme', name]);
|
||||||
|
let stylesheet = Gio.file_new_for_path(path);
|
||||||
|
if (stylesheet.query_exists(null))
|
||||||
|
return stylesheet;
|
||||||
|
}
|
||||||
|
|
||||||
stylesheet = Gio.File.new_for_path(global.datadir + '/theme/' + name);
|
stylesheet = Gio.File.new_for_path(global.datadir + '/theme/' + name);
|
||||||
if (stylesheet.query_exists(null))
|
if (stylesheet.query_exists(null))
|
||||||
return stylesheet;
|
return stylesheet;
|
||||||
@ -335,6 +343,9 @@ function loadTheme() {
|
|||||||
let theme = new St.Theme ({ application_stylesheet: _cssStylesheet,
|
let theme = new St.Theme ({ application_stylesheet: _cssStylesheet,
|
||||||
default_stylesheet: _defaultCssStylesheet });
|
default_stylesheet: _defaultCssStylesheet });
|
||||||
|
|
||||||
|
if (theme.default_stylesheet == null)
|
||||||
|
throw new Error("No valid stylesheet found for '%s'".format(sessionMode.stylesheetName));
|
||||||
|
|
||||||
if (previousTheme) {
|
if (previousTheme) {
|
||||||
let customStylesheets = previousTheme.get_custom_stylesheets();
|
let customStylesheets = previousTheme.get_custom_stylesheets();
|
||||||
|
|
||||||
|
@ -362,7 +362,8 @@ var Message = new Lang.Class({
|
|||||||
this.setBody(body);
|
this.setBody(body);
|
||||||
|
|
||||||
this._closeButton.connect('clicked', this.close.bind(this));
|
this._closeButton.connect('clicked', this.close.bind(this));
|
||||||
this.actor.connect('notify::hover', this._sync.bind(this));
|
let actorHoverId = this.actor.connect('notify::hover', this._sync.bind(this));
|
||||||
|
this._closeButton.connect('destroy', this.actor.disconnect.bind(this.actor, actorHoverId));
|
||||||
this.actor.connect('clicked', this._onClicked.bind(this));
|
this.actor.connect('clicked', this._onClicked.bind(this));
|
||||||
this.actor.connect('destroy', this._onDestroy.bind(this));
|
this.actor.connect('destroy', this._onDestroy.bind(this));
|
||||||
this._sync();
|
this._sync();
|
||||||
|
@ -1320,6 +1320,7 @@ var MessageTray = new Lang.Class({
|
|||||||
this._bannerBin.y = -this._banner.actor.height;
|
this._bannerBin.y = -this._banner.actor.height;
|
||||||
this.actor.show();
|
this.actor.show();
|
||||||
|
|
||||||
|
Meta.disable_unredirect_for_screen(global.screen);
|
||||||
this._updateShowingNotification();
|
this._updateShowingNotification();
|
||||||
|
|
||||||
let [x, y, mods] = global.get_pointer();
|
let [x, y, mods] = global.get_pointer();
|
||||||
@ -1457,6 +1458,7 @@ var MessageTray = new Lang.Class({
|
|||||||
|
|
||||||
this._pointerInNotification = false;
|
this._pointerInNotification = false;
|
||||||
this._notificationRemoved = false;
|
this._notificationRemoved = false;
|
||||||
|
Meta.enable_unredirect_for_screen(global.screen);
|
||||||
|
|
||||||
this._banner.actor.destroy();
|
this._banner.actor.destroy();
|
||||||
this._banner = null;
|
this._banner = null;
|
||||||
|
@ -117,10 +117,8 @@ var FdoNotificationDaemon = new Lang.Class({
|
|||||||
bitsPerSample, nChannels, data] = hints['image-data'];
|
bitsPerSample, nChannels, data] = hints['image-data'];
|
||||||
return Shell.util_create_pixbuf_from_data(data, GdkPixbuf.Colorspace.RGB, hasAlpha,
|
return Shell.util_create_pixbuf_from_data(data, GdkPixbuf.Colorspace.RGB, hasAlpha,
|
||||||
bitsPerSample, width, height, rowStride);
|
bitsPerSample, width, height, rowStride);
|
||||||
} else if (hints['image-path']) {
|
|
||||||
return new Gio.FileIcon({ file: Gio.File.new_for_path(hints['image-path']) });
|
|
||||||
}
|
}
|
||||||
return null;
|
return this._iconForNotificationData(hints['image-path']);
|
||||||
},
|
},
|
||||||
|
|
||||||
_fallbackIconForNotificationData(hints) {
|
_fallbackIconForNotificationData(hints) {
|
||||||
|
@ -108,15 +108,30 @@ var OsdWindow = new Lang.Class({
|
|||||||
this._hideTimeoutId = 0;
|
this._hideTimeoutId = 0;
|
||||||
this._reset();
|
this._reset();
|
||||||
|
|
||||||
|
this.actor.connect('destroy', this._onDestroy.bind(this));
|
||||||
|
|
||||||
|
this._monitorsChangedId =
|
||||||
Main.layoutManager.connect('monitors-changed',
|
Main.layoutManager.connect('monitors-changed',
|
||||||
this._relayout.bind(this));
|
this._relayout.bind(this));
|
||||||
let themeContext = St.ThemeContext.get_for_stage(global.stage);
|
let themeContext = St.ThemeContext.get_for_stage(global.stage);
|
||||||
|
this._scaleChangedId =
|
||||||
themeContext.connect('notify::scale-factor',
|
themeContext.connect('notify::scale-factor',
|
||||||
this._relayout.bind(this));
|
this._relayout.bind(this));
|
||||||
this._relayout();
|
this._relayout();
|
||||||
Main.uiGroup.add_child(this.actor);
|
Main.uiGroup.add_child(this.actor);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_onDestroy() {
|
||||||
|
if (this._monitorsChangedId)
|
||||||
|
Main.layoutManager.disconnect(this._monitorsChangedId);
|
||||||
|
this._monitorsChangedId = 0;
|
||||||
|
|
||||||
|
let themeContext = St.ThemeContext.get_for_stage(global.stage);
|
||||||
|
if (this._scaleChangedId)
|
||||||
|
themeContext.disconnect(this._scaleChangedId);
|
||||||
|
this._scaleChangedId = 0;
|
||||||
|
},
|
||||||
|
|
||||||
setIcon(icon) {
|
setIcon(icon) {
|
||||||
this._icon.gicon = icon;
|
this._icon.gicon = icon;
|
||||||
},
|
},
|
||||||
@ -204,7 +219,7 @@ var OsdWindow = new Lang.Class({
|
|||||||
|
|
||||||
let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor;
|
let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor;
|
||||||
this._icon.icon_size = popupSize / (2 * scaleFactor);
|
this._icon.icon_size = popupSize / (2 * scaleFactor);
|
||||||
this._box.translation_y = monitor.height / 4;
|
this._box.translation_y = Math.round(monitor.height / 4);
|
||||||
this._boxConstraint.minSize = popupSize;
|
this._boxConstraint.minSize = popupSize;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -393,10 +393,8 @@ var Overview = new Lang.Class({
|
|||||||
if (!Main.layoutManager.primaryMonitor)
|
if (!Main.layoutManager.primaryMonitor)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex);
|
this._coverPane.set_position(0, 0);
|
||||||
|
this._coverPane.set_size(global.screen_width, global.screen_height);
|
||||||
this._coverPane.set_position(0, workArea.y);
|
|
||||||
this._coverPane.set_size(workArea.width, workArea.height);
|
|
||||||
|
|
||||||
this._updateBackgrounds();
|
this._updateBackgrounds();
|
||||||
},
|
},
|
||||||
|
@ -284,6 +284,11 @@ var ThumbnailsSlider = new Lang.Class({
|
|||||||
return child.get_theme_node().get_length('visible-width');
|
return child.get_theme_node().get_length('visible-width');
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_onDragEnd() {
|
||||||
|
this.actor.sync_hover();
|
||||||
|
this.parent();
|
||||||
|
},
|
||||||
|
|
||||||
_getSlide() {
|
_getSlide() {
|
||||||
if (!this._visible)
|
if (!this._visible)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -796,6 +796,7 @@ var Panel = new Lang.Class({
|
|||||||
this.actor.connect('get-preferred-height', this._getPreferredHeight.bind(this));
|
this.actor.connect('get-preferred-height', this._getPreferredHeight.bind(this));
|
||||||
this.actor.connect('allocate', this._allocate.bind(this));
|
this.actor.connect('allocate', this._allocate.bind(this));
|
||||||
this.actor.connect('button-press-event', this._onButtonPress.bind(this));
|
this.actor.connect('button-press-event', this._onButtonPress.bind(this));
|
||||||
|
this.actor.connect('touch-event', this._onButtonPress.bind(this));
|
||||||
this.actor.connect('key-press-event', this._onKeyPress.bind(this));
|
this.actor.connect('key-press-event', this._onKeyPress.bind(this));
|
||||||
|
|
||||||
Main.overview.connect('showing', () => {
|
Main.overview.connect('showing', () => {
|
||||||
@ -939,8 +940,13 @@ var Panel = new Lang.Class({
|
|||||||
if (event.get_source() != actor)
|
if (event.get_source() != actor)
|
||||||
return Clutter.EVENT_PROPAGATE;
|
return Clutter.EVENT_PROPAGATE;
|
||||||
|
|
||||||
let button = event.get_button();
|
let type = event.type();
|
||||||
if (button != 1)
|
let isPress = type == Clutter.EventType.BUTTON_PRESS;
|
||||||
|
if (!isPress && type != Clutter.EventType.TOUCH_BEGIN)
|
||||||
|
return Clutter.EVENT_PROPAGATE;
|
||||||
|
|
||||||
|
let button = isPress ? event.get_button() : -1;
|
||||||
|
if (isPress && button != 1)
|
||||||
return Clutter.EVENT_PROPAGATE;
|
return Clutter.EVENT_PROPAGATE;
|
||||||
|
|
||||||
let focusWindow = global.display.focus_window;
|
let focusWindow = global.display.focus_window;
|
||||||
@ -1079,6 +1085,7 @@ var Panel = new Lang.Class({
|
|||||||
let windows = activeWorkspace.list_windows().filter(metaWindow => {
|
let windows = activeWorkspace.list_windows().filter(metaWindow => {
|
||||||
return metaWindow.is_on_primary_monitor() &&
|
return metaWindow.is_on_primary_monitor() &&
|
||||||
metaWindow.showing_on_its_workspace() &&
|
metaWindow.showing_on_its_workspace() &&
|
||||||
|
!metaWindow.is_hidden() &&
|
||||||
metaWindow.get_window_type() != Meta.WindowType.DESKTOP;
|
metaWindow.get_window_type() != Meta.WindowType.DESKTOP;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -141,8 +141,17 @@ var PopupBaseMenuItem = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_onKeyPressEvent(actor, event) {
|
_onKeyPressEvent(actor, event) {
|
||||||
let symbol = event.get_key_symbol();
|
let state = event.get_state();
|
||||||
|
|
||||||
|
// if user has a modifier down (except capslock)
|
||||||
|
// then don't handle the key press here
|
||||||
|
state &= ~Clutter.ModifierType.LOCK_MASK;
|
||||||
|
state &= Clutter.ModifierType.MODIFIER_MASK;
|
||||||
|
|
||||||
|
if (state)
|
||||||
|
return Clutter.EVENT_PROPAGATE;
|
||||||
|
|
||||||
|
let symbol = event.get_key_symbol();
|
||||||
if (symbol == Clutter.KEY_space || symbol == Clutter.KEY_Return) {
|
if (symbol == Clutter.KEY_space || symbol == Clutter.KEY_Return) {
|
||||||
this.activate(event);
|
this.activate(event);
|
||||||
return Clutter.EVENT_STOP;
|
return Clutter.EVENT_STOP;
|
||||||
@ -394,8 +403,9 @@ var PopupImageMenuItem = new Lang.Class({
|
|||||||
_init(text, icon, params) {
|
_init(text, icon, params) {
|
||||||
this.parent(params);
|
this.parent(params);
|
||||||
|
|
||||||
this._icon = new St.Icon({ style_class: 'popup-menu-icon' });
|
this._icon = new St.Icon({ style_class: 'popup-menu-icon',
|
||||||
this.actor.add_child(this._icon, { align: St.Align.END });
|
x_align: Clutter.ActorAlign.END });
|
||||||
|
this.actor.add_child(this._icon);
|
||||||
this.label = new St.Label({ text: text });
|
this.label = new St.Label({ text: text });
|
||||||
this.actor.add_child(this.label);
|
this.actor.add_child(this.label);
|
||||||
this.actor.label_actor = this.label;
|
this.actor.label_actor = this.label;
|
||||||
|
@ -295,7 +295,7 @@ var RemoteSearchProvider = new Lang.Class({
|
|||||||
name: metas[i]['name'],
|
name: metas[i]['name'],
|
||||||
description: metas[i]['description'],
|
description: metas[i]['description'],
|
||||||
createIcon: size => {
|
createIcon: size => {
|
||||||
this.createIcon(size, metas[i]);
|
return this.createIcon(size, metas[i]);
|
||||||
},
|
},
|
||||||
clipboardText: metas[i]['clipboardText'] });
|
clipboardText: metas[i]['clipboardText'] });
|
||||||
}
|
}
|
||||||
|
@ -114,18 +114,16 @@ var RunDialog = new Lang.Class({
|
|||||||
|
|
||||||
this._history = new History.HistoryManager({ gsettingsKey: HISTORY_KEY,
|
this._history = new History.HistoryManager({ gsettingsKey: HISTORY_KEY,
|
||||||
entry: this._entryText });
|
entry: this._entryText });
|
||||||
this._entryText.connect('key-press-event', (o, e) => {
|
this._entryText.connect('activate', (o) => {
|
||||||
let symbol = e.get_key_symbol();
|
|
||||||
if (symbol == Clutter.Return || symbol == Clutter.KP_Enter) {
|
|
||||||
this.popModal();
|
this.popModal();
|
||||||
this._run(o.get_text(),
|
this._run(o.get_text(),
|
||||||
e.get_state() & Clutter.ModifierType.CONTROL_MASK);
|
Clutter.get_current_event().get_state() & Clutter.ModifierType.CONTROL_MASK);
|
||||||
if (!this._commandError ||
|
if (!this._commandError ||
|
||||||
!this.pushModal())
|
!this.pushModal())
|
||||||
this.close();
|
this.close();
|
||||||
|
});
|
||||||
return Clutter.EVENT_STOP;
|
this._entryText.connect('key-press-event', (o, e) => {
|
||||||
}
|
let symbol = e.get_key_symbol();
|
||||||
if (symbol == Clutter.Tab) {
|
if (symbol == Clutter.Tab) {
|
||||||
let text = o.get_text();
|
let text = o.get_text();
|
||||||
let prefix;
|
let prefix;
|
||||||
|
@ -192,6 +192,7 @@ var SearchResultsBase = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
clear() {
|
clear() {
|
||||||
|
this._cancellable.cancel();
|
||||||
for (let resultId in this._resultDisplays)
|
for (let resultId in this._resultDisplays)
|
||||||
this._resultDisplays[resultId].actor.destroy();
|
this._resultDisplays[resultId].actor.destroy();
|
||||||
this._resultDisplays = {};
|
this._resultDisplays = {};
|
||||||
@ -225,6 +226,12 @@ var SearchResultsBase = new Lang.Class({
|
|||||||
this._cancellable.reset();
|
this._cancellable.reset();
|
||||||
|
|
||||||
this.provider.getResultMetas(metasNeeded, metas => {
|
this.provider.getResultMetas(metasNeeded, metas => {
|
||||||
|
if (this._cancellable.is_cancelled()) {
|
||||||
|
if (metas.length > 0)
|
||||||
|
log(`Search provider ${this.provider.id} returned results after the request was canceled`);
|
||||||
|
callback(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (metas.length != metasNeeded.length) {
|
if (metas.length != metasNeeded.length) {
|
||||||
log('Wrong number of result metas returned by search provider ' + this.provider.id +
|
log('Wrong number of result metas returned by search provider ' + this.provider.id +
|
||||||
': expected ' + metasNeeded.length + ' but got ' + metas.length);
|
': expected ' + metasNeeded.length + ' but got ' + metas.length);
|
||||||
|
@ -360,11 +360,14 @@ var InputSourceManager = new Lang.Class({
|
|||||||
this._settings.connect('per-window-changed', this._sourcesPerWindowChanged.bind(this));
|
this._settings.connect('per-window-changed', this._sourcesPerWindowChanged.bind(this));
|
||||||
this._sourcesPerWindowChanged();
|
this._sourcesPerWindowChanged();
|
||||||
this._disableIBus = false;
|
this._disableIBus = false;
|
||||||
|
this._reloading = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
reload() {
|
reload() {
|
||||||
|
this._reloading = true;
|
||||||
this._keyboardManager.setKeyboardOptions(this._settings.keyboardOptions);
|
this._keyboardManager.setKeyboardOptions(this._settings.keyboardOptions);
|
||||||
this._inputSourcesChanged();
|
this._inputSourcesChanged();
|
||||||
|
this._reloading = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
_ibusReadyCallback(im, ready) {
|
_ibusReadyCallback(im, ready) {
|
||||||
@ -458,6 +461,14 @@ var InputSourceManager = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
activateInputSource(is, interactive) {
|
activateInputSource(is, interactive) {
|
||||||
|
// The focus changes during holdKeyboard/releaseKeyboard may trick
|
||||||
|
// the client into hiding UI containing the currently focused entry.
|
||||||
|
// So holdKeyboard/releaseKeyboard are not called when
|
||||||
|
// 'set-content-type' signal is received.
|
||||||
|
// E.g. Focusing on a password entry in a popup in Xorg Firefox
|
||||||
|
// will emit 'set-content-type' signal.
|
||||||
|
// https://gitlab.gnome.org/GNOME/gnome-shell/issues/391
|
||||||
|
if (!this._reloading)
|
||||||
KeyboardManager.holdKeyboard();
|
KeyboardManager.holdKeyboard();
|
||||||
this._keyboardManager.apply(is.xkbId);
|
this._keyboardManager.apply(is.xkbId);
|
||||||
|
|
||||||
@ -473,7 +484,10 @@ var InputSourceManager = new Lang.Class({
|
|||||||
else
|
else
|
||||||
engine = 'xkb:us::eng';
|
engine = 'xkb:us::eng';
|
||||||
|
|
||||||
|
if (!this._reloading)
|
||||||
this._ibusManager.setEngine(engine, KeyboardManager.releaseKeyboard);
|
this._ibusManager.setEngine(engine, KeyboardManager.releaseKeyboard);
|
||||||
|
else
|
||||||
|
this._ibusManager.setEngine(engine);
|
||||||
this._currentInputSourceChanged(is);
|
this._currentInputSourceChanged(is);
|
||||||
|
|
||||||
if (interactive)
|
if (interactive)
|
||||||
|
@ -419,14 +419,16 @@ var NMConnectionDevice = new Lang.Class({
|
|||||||
this._deactivateItem.actor.visible = this._device.state > NM.DeviceState.DISCONNECTED;
|
this._deactivateItem.actor.visible = this._device.state > NM.DeviceState.DISCONNECTED;
|
||||||
|
|
||||||
if (this._activeConnection == null) {
|
if (this._activeConnection == null) {
|
||||||
this._activeConnection = this._device.active_connection;
|
let activeConnection = this._device.active_connection;
|
||||||
|
if (activeConnection && activeConnection.connection) {
|
||||||
if (this._activeConnection) {
|
let item = this._connectionItems.get(activeConnection.connection.get_uuid());
|
||||||
|
if (item) {
|
||||||
|
this._activeConnection = activeConnection;
|
||||||
ensureActiveConnectionProps(this._activeConnection, this._client);
|
ensureActiveConnectionProps(this._activeConnection, this._client);
|
||||||
let item = this._connectionItems.get(this._activeConnection.connection.get_uuid());
|
|
||||||
item.setActiveConnection(this._activeConnection);
|
item.setActiveConnection(this._activeConnection);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this.parent();
|
this.parent();
|
||||||
},
|
},
|
||||||
@ -1944,6 +1946,7 @@ var NMApplet = new Lang.Class({
|
|||||||
this.indicators.visible = this._client.nm_running;
|
this.indicators.visible = this._client.nm_running;
|
||||||
this.menu.actor.visible = this._client.networking_enabled;
|
this.menu.actor.visible = this._client.networking_enabled;
|
||||||
|
|
||||||
|
this._updateIcon();
|
||||||
this._syncConnectivity();
|
this._syncConnectivity();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -58,6 +58,9 @@ var AltSwitcher = new Lang.Class({
|
|||||||
childToShow = this._standard;
|
childToShow = this._standard;
|
||||||
} else if (this._alternate.visible) {
|
} else if (this._alternate.visible) {
|
||||||
childToShow = this._alternate;
|
childToShow = this._alternate;
|
||||||
|
} else {
|
||||||
|
this.actor.hide();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let childShown = this.actor.get_child();
|
let childShown = this.actor.get_child();
|
||||||
@ -79,7 +82,7 @@ var AltSwitcher = new Lang.Class({
|
|||||||
global.sync_pointer();
|
global.sync_pointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
this.actor.visible = (childToShow != null);
|
this.actor.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
_onDestroy() {
|
_onDestroy() {
|
||||||
|
@ -311,6 +311,7 @@ var ViewSelector = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
hide() {
|
hide() {
|
||||||
|
this.reset();
|
||||||
this._workspacesDisplay.hide();
|
this._workspacesDisplay.hide();
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -459,6 +460,10 @@ var ViewSelector = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
reset() {
|
reset() {
|
||||||
|
// Don't drop the key focus on Clutter's side if anything but the
|
||||||
|
// overview has pushed a modal (e.g. system modals when activated using
|
||||||
|
// the overview).
|
||||||
|
if (Main.modalCount <= 1)
|
||||||
global.stage.set_key_focus(null);
|
global.stage.set_key_focus(null);
|
||||||
|
|
||||||
this._entry.text = '';
|
this._entry.text = '';
|
||||||
|
@ -24,7 +24,7 @@ const EdgeDragAction = imports.ui.edgeDragAction;
|
|||||||
const CloseDialog = imports.ui.closeDialog;
|
const CloseDialog = imports.ui.closeDialog;
|
||||||
const SwitchMonitor = imports.ui.switchMonitor;
|
const SwitchMonitor = imports.ui.switchMonitor;
|
||||||
|
|
||||||
const SHELL_KEYBINDINGS_SCHEMA = 'org.gnome.shell.keybindings';
|
var SHELL_KEYBINDINGS_SCHEMA = 'org.gnome.shell.keybindings';
|
||||||
var MINIMIZE_WINDOW_ANIMATION_TIME = 0.2;
|
var MINIMIZE_WINDOW_ANIMATION_TIME = 0.2;
|
||||||
var SHOW_WINDOW_ANIMATION_TIME = 0.15;
|
var SHOW_WINDOW_ANIMATION_TIME = 0.15;
|
||||||
var DIALOG_SHOW_WINDOW_ANIMATION_TIME = 0.1;
|
var DIALOG_SHOW_WINDOW_ANIMATION_TIME = 0.1;
|
||||||
@ -627,8 +627,8 @@ var AppSwitchAction = new Lang.Class({
|
|||||||
|
|
||||||
if (this.get_n_current_points() == 3) {
|
if (this.get_n_current_points() == 3) {
|
||||||
for (let i = 0; i < this.get_n_current_points(); i++) {
|
for (let i = 0; i < this.get_n_current_points(); i++) {
|
||||||
[startX, startY] = this.get_press_coords(i);
|
let [startX, startY] = this.get_press_coords(i);
|
||||||
[x, y] = this.get_motion_coords(i);
|
let [x, y] = this.get_motion_coords(i);
|
||||||
|
|
||||||
if (Math.abs(x - startX) > MOTION_THRESHOLD ||
|
if (Math.abs(x - startX) > MOTION_THRESHOLD ||
|
||||||
Math.abs(y - startY) > MOTION_THRESHOLD)
|
Math.abs(y - startY) > MOTION_THRESHOLD)
|
||||||
@ -1173,6 +1173,10 @@ var WindowManager = new Lang.Class({
|
|||||||
yScale = geom.height / actor.height;
|
yScale = geom.height / actor.height;
|
||||||
} else {
|
} else {
|
||||||
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
||||||
|
if (!monitor) {
|
||||||
|
this._minimizeWindowDone();
|
||||||
|
return;
|
||||||
|
}
|
||||||
xDest = monitor.x;
|
xDest = monitor.x;
|
||||||
yDest = monitor.y;
|
yDest = monitor.y;
|
||||||
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
||||||
@ -1248,6 +1252,11 @@ var WindowManager = new Lang.Class({
|
|||||||
geom.height / actor.height);
|
geom.height / actor.height);
|
||||||
} else {
|
} else {
|
||||||
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
||||||
|
if (!monitor) {
|
||||||
|
actor.show();
|
||||||
|
this._unminimizeWindowDone();
|
||||||
|
return;
|
||||||
|
}
|
||||||
actor.set_position(monitor.x, monitor.y);
|
actor.set_position(monitor.x, monitor.y);
|
||||||
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
||||||
actor.x += monitor.width;
|
actor.x += monitor.width;
|
||||||
@ -1754,6 +1763,14 @@ var WindowManager = new Lang.Class({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < switchData.windows.length; i++) {
|
||||||
|
let w = switchData.windows[i];
|
||||||
|
|
||||||
|
w.windowDestroyId = w.window.connect('destroy', () => {
|
||||||
|
switchData.windows.splice(switchData.windows.indexOf(w), 1);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
switchData.inGroup.set_position(-xDest, -yDest);
|
switchData.inGroup.set_position(-xDest, -yDest);
|
||||||
switchData.inGroup.raise_top();
|
switchData.inGroup.raise_top();
|
||||||
|
|
||||||
@ -1784,8 +1801,8 @@ var WindowManager = new Lang.Class({
|
|||||||
|
|
||||||
for (let i = 0; i < switchData.windows.length; i++) {
|
for (let i = 0; i < switchData.windows.length; i++) {
|
||||||
let w = switchData.windows[i];
|
let w = switchData.windows[i];
|
||||||
if (w.window.is_destroyed()) // Window gone
|
w.window.disconnect(w.windowDestroyId);
|
||||||
continue;
|
|
||||||
if (w.window.get_parent() == switchData.outGroup) {
|
if (w.window.get_parent() == switchData.outGroup) {
|
||||||
w.window.reparent(w.parent);
|
w.window.reparent(w.parent);
|
||||||
w.window.hide();
|
w.window.hide();
|
||||||
|
@ -128,10 +128,9 @@ var WindowMenu = new Lang.Class({
|
|||||||
|
|
||||||
let screen = global.screen;
|
let screen = global.screen;
|
||||||
let nMonitors = screen.get_n_monitors();
|
let nMonitors = screen.get_n_monitors();
|
||||||
if (nMonitors > 1) {
|
|
||||||
this.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
|
||||||
|
|
||||||
let monitorIndex = window.get_monitor();
|
let monitorIndex = window.get_monitor();
|
||||||
|
if (nMonitors > 1 && monitorIndex >= 0) {
|
||||||
|
this.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
||||||
|
|
||||||
let dir = Meta.ScreenDirection.UP;
|
let dir = Meta.ScreenDirection.UP;
|
||||||
let upMonitorIndex =
|
let upMonitorIndex =
|
||||||
|
@ -137,8 +137,10 @@ var WindowClone = new Lang.Class({
|
|||||||
this._dragSlot = [0, 0, 0, 0];
|
this._dragSlot = [0, 0, 0, 0];
|
||||||
this._stackAbove = null;
|
this._stackAbove = null;
|
||||||
|
|
||||||
this._windowClone._updateId = this.metaWindow.connect('size-changed',
|
this._windowClone._sizeChangedId = this.metaWindow.connect('size-changed',
|
||||||
this._onRealWindowSizeChanged.bind(this));
|
this._onMetaWindowSizeChanged.bind(this));
|
||||||
|
this._windowClone._posChangedId = this.metaWindow.connect('position-changed',
|
||||||
|
this._computeBoundingBox.bind(this));
|
||||||
this._windowClone._destroyId =
|
this._windowClone._destroyId =
|
||||||
this.realWindow.connect('destroy', () => {
|
this.realWindow.connect('destroy', () => {
|
||||||
// First destroy the clone and then destroy everything
|
// First destroy the clone and then destroy everything
|
||||||
@ -206,8 +208,7 @@ var WindowClone = new Lang.Class({
|
|||||||
|
|
||||||
addAttachedDialog(win) {
|
addAttachedDialog(win) {
|
||||||
this._doAddAttachedDialog(win, win.get_compositor_private());
|
this._doAddAttachedDialog(win, win.get_compositor_private());
|
||||||
this._computeBoundingBox();
|
this._onMetaWindowSizeChanged();
|
||||||
this.emit('size-changed');
|
|
||||||
},
|
},
|
||||||
|
|
||||||
hasAttachedDialogs() {
|
hasAttachedDialogs() {
|
||||||
@ -216,15 +217,14 @@ var WindowClone = new Lang.Class({
|
|||||||
|
|
||||||
_doAddAttachedDialog(metaWin, realWin) {
|
_doAddAttachedDialog(metaWin, realWin) {
|
||||||
let clone = new Clutter.Clone({ source: realWin });
|
let clone = new Clutter.Clone({ source: realWin });
|
||||||
clone._updateId = metaWin.connect('size-changed', () => {
|
clone._sizeChangedId = metaWin.connect('size-changed',
|
||||||
this._computeBoundingBox();
|
this._onMetaWindowSizeChanged.bind(this));
|
||||||
this.emit('size-changed');
|
clone._posChangedId = metaWin.connect('position-changed',
|
||||||
});
|
this._onMetaWindowSizeChanged.bind(this));
|
||||||
clone._destroyId = realWin.connect('destroy', () => {
|
clone._destroyId = realWin.connect('destroy', () => {
|
||||||
clone.destroy();
|
clone.destroy();
|
||||||
|
|
||||||
this._computeBoundingBox();
|
this._onMetaWindowSizeChanged();
|
||||||
this.emit('size-changed');
|
|
||||||
});
|
});
|
||||||
this.actor.add_child(clone);
|
this.actor.add_child(clone);
|
||||||
},
|
},
|
||||||
@ -321,12 +321,13 @@ var WindowClone = new Lang.Class({
|
|||||||
else
|
else
|
||||||
realWindow = child.source;
|
realWindow = child.source;
|
||||||
|
|
||||||
realWindow.meta_window.disconnect(child._updateId);
|
realWindow.meta_window.disconnect(child._sizeChangedId);
|
||||||
|
realWindow.meta_window.disconnect(child._posChangedId);
|
||||||
realWindow.disconnect(child._destroyId);
|
realWindow.disconnect(child._destroyId);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
_onRealWindowSizeChanged() {
|
_onMetaWindowSizeChanged() {
|
||||||
this._computeBoundingBox();
|
this._computeBoundingBox();
|
||||||
this.emit('size-changed');
|
this.emit('size-changed');
|
||||||
},
|
},
|
||||||
@ -469,7 +470,6 @@ var WindowOverlay = new Lang.Class({
|
|||||||
this._windowAddedId = 0;
|
this._windowAddedId = 0;
|
||||||
|
|
||||||
button.hide();
|
button.hide();
|
||||||
title.hide();
|
|
||||||
|
|
||||||
this.title = title;
|
this.title = title;
|
||||||
this.closeButton = button;
|
this.closeButton = button;
|
||||||
@ -544,12 +544,10 @@ var WindowOverlay = new Lang.Class({
|
|||||||
let titleX = cloneX + (cloneWidth - title.width) / 2;
|
let titleX = cloneX + (cloneWidth - title.width) / 2;
|
||||||
let titleY = cloneY + cloneHeight - (title.height - this.borderSize) / 2;
|
let titleY = cloneY + cloneHeight - (title.height - this.borderSize) / 2;
|
||||||
|
|
||||||
if (animate) {
|
if (animate)
|
||||||
this._animateOverlayActor(title, Math.floor(titleX), Math.floor(titleY), title.width);
|
this._animateOverlayActor(title, Math.floor(titleX), Math.floor(titleY));
|
||||||
} else {
|
else
|
||||||
title.width = title.width;
|
|
||||||
title.set_position(Math.floor(titleX), Math.floor(titleY));
|
title.set_position(Math.floor(titleX), Math.floor(titleY));
|
||||||
}
|
|
||||||
|
|
||||||
let borderX = cloneX - this.borderSize;
|
let borderX = cloneX - this.borderSize;
|
||||||
let borderY = cloneY - this.borderSize;
|
let borderY = cloneY - this.borderSize;
|
||||||
@ -568,10 +566,12 @@ var WindowOverlay = new Lang.Class({
|
|||||||
_animateOverlayActor(actor, x, y, width, height) {
|
_animateOverlayActor(actor, x, y, width, height) {
|
||||||
let params = { x: x,
|
let params = { x: x,
|
||||||
y: y,
|
y: y,
|
||||||
width: width,
|
|
||||||
time: Overview.ANIMATION_TIME,
|
time: Overview.ANIMATION_TIME,
|
||||||
transition: 'easeOutQuad' };
|
transition: 'easeOutQuad' };
|
||||||
|
|
||||||
|
if (width !== undefined)
|
||||||
|
params.width = width;
|
||||||
|
|
||||||
if (height !== undefined)
|
if (height !== undefined)
|
||||||
params.height = height;
|
params.height = height;
|
||||||
|
|
||||||
@ -1431,17 +1431,9 @@ var Workspace = new Lang.Class({
|
|||||||
_doRemoveWindow(metaWin) {
|
_doRemoveWindow(metaWin) {
|
||||||
let win = metaWin.get_compositor_private();
|
let win = metaWin.get_compositor_private();
|
||||||
|
|
||||||
// find the position of the window in our list
|
let clone = this._removeWindowClone(metaWin);
|
||||||
let index = this._lookupIndex (metaWin);
|
|
||||||
|
|
||||||
if (index == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
let clone = this._windows[index];
|
|
||||||
|
|
||||||
this._windows.splice(index, 1);
|
|
||||||
this._windowOverlays.splice(index, 1);
|
|
||||||
|
|
||||||
|
if (clone) {
|
||||||
// If metaWin.get_compositor_private() returned non-NULL, that
|
// If metaWin.get_compositor_private() returned non-NULL, that
|
||||||
// means the window still exists (and is just being moved to
|
// means the window still exists (and is just being moved to
|
||||||
// another workspace or something), so set its overviewHint
|
// another workspace or something), so set its overviewHint
|
||||||
@ -1458,7 +1450,7 @@ var Workspace = new Lang.Class({
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
clone.destroy();
|
clone.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
// We need to reposition the windows; to avoid shuffling windows
|
// We need to reposition the windows; to avoid shuffling windows
|
||||||
// around while the user is interacting with the workspace, we delay
|
// around while the user is interacting with the workspace, we delay
|
||||||
@ -1514,7 +1506,7 @@ var Workspace = new Lang.Class({
|
|||||||
if (metaWin.is_attached_dialog()) {
|
if (metaWin.is_attached_dialog()) {
|
||||||
let parent = metaWin.get_transient_for();
|
let parent = metaWin.get_transient_for();
|
||||||
while (parent.is_attached_dialog())
|
while (parent.is_attached_dialog())
|
||||||
parent = metaWin.get_transient_for();
|
parent = parent.get_transient_for();
|
||||||
|
|
||||||
let idx = this._lookupIndex (parent);
|
let idx = this._lookupIndex (parent);
|
||||||
if (idx < 0) {
|
if (idx < 0) {
|
||||||
@ -1848,6 +1840,9 @@ var Workspace = new Lang.Class({
|
|||||||
clone.connect('size-changed', () => {
|
clone.connect('size-changed', () => {
|
||||||
this._recalculateWindowPositions(WindowPositionFlags.NONE);
|
this._recalculateWindowPositions(WindowPositionFlags.NONE);
|
||||||
});
|
});
|
||||||
|
clone.actor.connect('destroy', () => {
|
||||||
|
this._removeWindowClone(clone.metaWindow);
|
||||||
|
});
|
||||||
|
|
||||||
this.actor.add_actor(clone.actor);
|
this.actor.add_actor(clone.actor);
|
||||||
|
|
||||||
@ -1869,6 +1864,17 @@ var Workspace = new Lang.Class({
|
|||||||
return [clone, overlay];
|
return [clone, overlay];
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_removeWindowClone(metaWin) {
|
||||||
|
// find the position of the window in our list
|
||||||
|
let index = this._lookupIndex (metaWin);
|
||||||
|
|
||||||
|
if (index == -1)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
this._windowOverlays.splice(index, 1);
|
||||||
|
return this._windows.splice(index, 1).pop();
|
||||||
|
},
|
||||||
|
|
||||||
_onShowOverlayClose(windowOverlay) {
|
_onShowOverlayClose(windowOverlay) {
|
||||||
for (let i = 0; i < this._windowOverlays.length; i++) {
|
for (let i = 0; i < this._windowOverlays.length; i++) {
|
||||||
let overlay = this._windowOverlays[i];
|
let overlay = this._windowOverlays[i];
|
||||||
|
@ -31,7 +31,7 @@ var WORKSPACE_CUT_SIZE = 10;
|
|||||||
|
|
||||||
var WORKSPACE_KEEP_ALIVE_TIME = 100;
|
var WORKSPACE_KEEP_ALIVE_TIME = 100;
|
||||||
|
|
||||||
const OVERRIDE_SCHEMA = 'org.gnome.shell.overrides';
|
var OVERRIDE_SCHEMA = 'org.gnome.shell.overrides';
|
||||||
|
|
||||||
/* A layout manager that requests size only for primary_actor, but then allocates
|
/* A layout manager that requests size only for primary_actor, but then allocates
|
||||||
all using a fixed layout */
|
all using a fixed layout */
|
||||||
@ -68,7 +68,7 @@ var WindowClone = new Lang.Class({
|
|||||||
this.realWindow = realWindow;
|
this.realWindow = realWindow;
|
||||||
this.metaWindow = realWindow.meta_window;
|
this.metaWindow = realWindow.meta_window;
|
||||||
|
|
||||||
this.clone._updateId = this.metaWindow.connect('position-changed',
|
this.clone._updateId = this.realWindow.connect('notify::position',
|
||||||
this._onPositionChanged.bind(this));
|
this._onPositionChanged.bind(this));
|
||||||
this.clone._destroyId = this.realWindow.connect('destroy', () => {
|
this.clone._destroyId = this.realWindow.connect('destroy', () => {
|
||||||
// First destroy the clone and then destroy everything
|
// First destroy the clone and then destroy everything
|
||||||
@ -153,7 +153,7 @@ var WindowClone = new Lang.Class({
|
|||||||
let clone = new Clutter.Clone({ source: realDialog });
|
let clone = new Clutter.Clone({ source: realDialog });
|
||||||
this._updateDialogPosition(realDialog, clone);
|
this._updateDialogPosition(realDialog, clone);
|
||||||
|
|
||||||
clone._updateId = metaDialog.connect('position-changed', dialog => {
|
clone._updateId = realDialog.connect('notify::position', dialog => {
|
||||||
this._updateDialogPosition(dialog, clone);
|
this._updateDialogPosition(dialog, clone);
|
||||||
});
|
});
|
||||||
clone._destroyId = realDialog.connect('destroy', () => {
|
clone._destroyId = realDialog.connect('destroy', () => {
|
||||||
@ -171,7 +171,6 @@ var WindowClone = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_onPositionChanged() {
|
_onPositionChanged() {
|
||||||
let rect = this.metaWindow.get_frame_rect();
|
|
||||||
this.actor.set_position(this.realWindow.x, this.realWindow.y);
|
this.actor.set_position(this.realWindow.x, this.realWindow.y);
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -179,7 +178,7 @@ var WindowClone = new Lang.Class({
|
|||||||
this.actor.get_children().forEach(child => {
|
this.actor.get_children().forEach(child => {
|
||||||
let realWindow = child.source;
|
let realWindow = child.source;
|
||||||
|
|
||||||
realWindow.meta_window.disconnect(child._updateId);
|
realWindow.disconnect(child._updateId);
|
||||||
realWindow.disconnect(child._destroyId);
|
realWindow.disconnect(child._destroyId);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
@ -241,7 +240,7 @@ var WindowClone = new Lang.Class({
|
|||||||
Signals.addSignalMethods(WindowClone.prototype);
|
Signals.addSignalMethods(WindowClone.prototype);
|
||||||
|
|
||||||
|
|
||||||
const ThumbnailState = {
|
var ThumbnailState = {
|
||||||
NEW : 0,
|
NEW : 0,
|
||||||
ANIMATING_IN : 1,
|
ANIMATING_IN : 1,
|
||||||
NORMAL: 2,
|
NORMAL: 2,
|
||||||
@ -275,8 +274,8 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
|
|
||||||
this._createBackground();
|
this._createBackground();
|
||||||
|
|
||||||
let monitor = Main.layoutManager.primaryMonitor;
|
let workArea = Main.layoutManager.getWorkAreaForMonitor(this.monitorIndex);
|
||||||
this.setPorthole(monitor.x, monitor.y, monitor.width, monitor.height);
|
this.setPorthole(workArea.x, workArea.y, workArea.width, workArea.height);
|
||||||
|
|
||||||
let windows = global.get_window_actors().filter(actor => {
|
let windows = global.get_window_actors().filter(actor => {
|
||||||
let win = actor.meta_window;
|
let win = actor.meta_window;
|
||||||
@ -321,8 +320,6 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
setPorthole(x, y, width, height) {
|
setPorthole(x, y, width, height) {
|
||||||
this._portholeX = x;
|
|
||||||
this._portholeY = y;
|
|
||||||
this.actor.set_size(width, height);
|
this.actor.set_size(width, height);
|
||||||
this._contents.set_position(-x, -y);
|
this._contents.set_position(-x, -y);
|
||||||
},
|
},
|
||||||
@ -374,17 +371,8 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_doRemoveWindow(metaWin) {
|
_doRemoveWindow(metaWin) {
|
||||||
let win = metaWin.get_compositor_private();
|
let clone = this._removeWindowClone(metaWin);
|
||||||
|
if (clone)
|
||||||
// find the position of the window in our list
|
|
||||||
let index = this._lookupIndex (metaWin);
|
|
||||||
|
|
||||||
if (index == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
let clone = this._windows[index];
|
|
||||||
this._windows.splice(index, 1);
|
|
||||||
|
|
||||||
clone.destroy();
|
clone.destroy();
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -428,7 +416,7 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
} else if (metaWin.is_attached_dialog()) {
|
} else if (metaWin.is_attached_dialog()) {
|
||||||
let parent = metaWin.get_transient_for();
|
let parent = metaWin.get_transient_for();
|
||||||
while (parent.is_attached_dialog())
|
while (parent.is_attached_dialog())
|
||||||
parent = metaWin.get_transient_for();
|
parent = parent.get_transient_for();
|
||||||
|
|
||||||
let idx = this._lookupIndex (parent);
|
let idx = this._lookupIndex (parent);
|
||||||
if (idx < 0) {
|
if (idx < 0) {
|
||||||
@ -537,6 +525,9 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
clone.connect('drag-end', () => {
|
clone.connect('drag-end', () => {
|
||||||
Main.overview.endWindowDrag(clone.metaWindow);
|
Main.overview.endWindowDrag(clone.metaWindow);
|
||||||
});
|
});
|
||||||
|
clone.actor.connect('destroy', () => {
|
||||||
|
this._removeWindowClone(clone.metaWindow);
|
||||||
|
});
|
||||||
this._contents.add_actor(clone.actor);
|
this._contents.add_actor(clone.actor);
|
||||||
|
|
||||||
if (this._windows.length == 0)
|
if (this._windows.length == 0)
|
||||||
@ -549,6 +540,16 @@ var WorkspaceThumbnail = new Lang.Class({
|
|||||||
return clone;
|
return clone;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_removeWindowClone(metaWin) {
|
||||||
|
// find the position of the window in our list
|
||||||
|
let index = this._lookupIndex (metaWin);
|
||||||
|
|
||||||
|
if (index == -1)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return this._windows.splice(index, 1).pop();
|
||||||
|
},
|
||||||
|
|
||||||
activate(time) {
|
activate(time) {
|
||||||
if (this.state > ThumbnailState.NORMAL)
|
if (this.state > ThumbnailState.NORMAL)
|
||||||
return;
|
return;
|
||||||
@ -1159,7 +1160,7 @@ var ThumbnailsBox = new Lang.Class({
|
|||||||
// The "porthole" is the portion of the screen that we show in the
|
// The "porthole" is the portion of the screen that we show in the
|
||||||
// workspaces
|
// workspaces
|
||||||
_ensurePorthole() {
|
_ensurePorthole() {
|
||||||
if (!Main.layoutManager.primaryMonitor)
|
if (!Main.layoutManager.primaryMonitor || !Main.overview.visible)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!this._porthole)
|
if (!this._porthole)
|
||||||
|
@ -470,6 +470,7 @@ var WorkspacesDisplay = new Lang.Class({
|
|||||||
this._switchWorkspaceNotifyId = 0;
|
this._switchWorkspaceNotifyId = 0;
|
||||||
|
|
||||||
this._notifyOpacityId = 0;
|
this._notifyOpacityId = 0;
|
||||||
|
this._restackedNotifyId = 0;
|
||||||
this._scrollEventId = 0;
|
this._scrollEventId = 0;
|
||||||
this._keyPressEventId = 0;
|
this._keyPressEventId = 0;
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
project('gnome-shell', 'c',
|
project('gnome-shell', 'c',
|
||||||
version: '3.28.1',
|
version: '3.28.4',
|
||||||
meson_version: '>= 0.42.0',
|
meson_version: '>= 0.42.0',
|
||||||
license: 'GPLv2+'
|
license: 'GPLv2+'
|
||||||
)
|
)
|
||||||
|
2
po/cs.po
2
po/cs.po
@ -1064,7 +1064,7 @@ msgstr "Načítá se…"
|
|||||||
#: js/ui/dateMenu.js:321
|
#: js/ui/dateMenu.js:321
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "Feels like %s."
|
msgid "Feels like %s."
|
||||||
msgstr "Pocitově jako %s."
|
msgstr "Pocitová teplota %s."
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:324
|
#: js/ui/dateMenu.js:324
|
||||||
msgid "Go online for weather information"
|
msgid "Go online for weather information"
|
||||||
|
46
po/pt_BR.po
46
po/pt_BR.po
@ -15,22 +15,22 @@
|
|||||||
# Georges Basile Stavracas Neto <georges.stavracas@gmail.com>, 2014.
|
# Georges Basile Stavracas Neto <georges.stavracas@gmail.com>, 2014.
|
||||||
# Felipe Braga <fbobraga@gmail.com>, 2015.
|
# Felipe Braga <fbobraga@gmail.com>, 2015.
|
||||||
# Artur de Aquino Morais <artur.morais93@outlook.com>, 2016.
|
# Artur de Aquino Morais <artur.morais93@outlook.com>, 2016.
|
||||||
# Rafael Fontenelle <rafaelff@gnome.org>, 2013-2017.
|
# Rafael Fontenelle <rafaelff@gnome.org>, 2013-2018.
|
||||||
# Enrico Nicoletto <liverig@gmail.com>, 2013-2018.
|
# Enrico Nicoletto <liverig@gmail.com>, 2013-2018.
|
||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: gnome-shell\n"
|
"Project-Id-Version: gnome-shell\n"
|
||||||
"Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n"
|
"Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n"
|
||||||
"POT-Creation-Date: 2018-03-16 21:34+0000\n"
|
"POT-Creation-Date: 2018-04-13 18:31+0000\n"
|
||||||
"PO-Revision-Date: 2018-02-09 21:52-0200\n"
|
"PO-Revision-Date: 2018-05-02 15:45-0200\n"
|
||||||
"Last-Translator: Enrico Nicoletto <liverig@gmail.com>\n"
|
"Last-Translator: Rafael Fontenelle <rafaelff@gnome.org>\n"
|
||||||
"Language-Team: Brazilian Portuguese <gnome-pt_br-list@gnome.org>\n"
|
"Language-Team: Brazilian Portuguese <gnome-pt_br-list@gnome.org>\n"
|
||||||
"Language: pt_BR\n"
|
"Language: pt_BR\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
|
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
|
||||||
"X-Generator: Poedit 2.0.6\n"
|
"X-Generator: Virtaal 1.0.0-beta1\n"
|
||||||
"X-Project-Style: gnome\n"
|
"X-Project-Style: gnome\n"
|
||||||
|
|
||||||
#: data/50-gnome-shell-system.xml:6
|
#: data/50-gnome-shell-system.xml:6
|
||||||
@ -356,7 +356,7 @@ msgid "There was an error loading the preferences dialog for %s:"
|
|||||||
msgstr "Ocorreu um erro ao carregar o dialogo de preferências para %s:"
|
msgstr "Ocorreu um erro ao carregar o dialogo de preferências para %s:"
|
||||||
|
|
||||||
#: js/gdm/authPrompt.js:147 js/ui/audioDeviceSelection.js:71
|
#: js/gdm/authPrompt.js:147 js/ui/audioDeviceSelection.js:71
|
||||||
#: js/ui/components/networkAgent.js:117 js/ui/components/polkitAgent.js:148
|
#: js/ui/components/networkAgent.js:117 js/ui/components/polkitAgent.js:153
|
||||||
#: js/ui/endSessionDialog.js:482 js/ui/extensionDownloader.js:197
|
#: js/ui/endSessionDialog.js:482 js/ui/extensionDownloader.js:197
|
||||||
#: js/ui/shellMountOperation.js:343 js/ui/status/network.js:919
|
#: js/ui/shellMountOperation.js:343 js/ui/status/network.js:919
|
||||||
msgid "Cancel"
|
msgid "Cancel"
|
||||||
@ -642,7 +642,7 @@ msgstr "Negar acesso"
|
|||||||
|
|
||||||
#: js/ui/accessDialog.js:64 js/ui/status/location.js:396
|
#: js/ui/accessDialog.js:64 js/ui/status/location.js:396
|
||||||
msgid "Grant Access"
|
msgid "Grant Access"
|
||||||
msgstr "Garantir acesso"
|
msgstr "Conceder acesso"
|
||||||
|
|
||||||
#: js/ui/appDisplay.js:793
|
#: js/ui/appDisplay.js:793
|
||||||
msgid "Frequently used applications will appear here"
|
msgid "Frequently used applications will appear here"
|
||||||
@ -676,12 +676,12 @@ msgstr "Adicionar aos favoritos"
|
|||||||
msgid "Show Details"
|
msgid "Show Details"
|
||||||
msgstr "Mostrar detalhes"
|
msgstr "Mostrar detalhes"
|
||||||
|
|
||||||
#: js/ui/appFavorites.js:138
|
#: js/ui/appFavorites.js:140
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s has been added to your favorites."
|
msgid "%s has been added to your favorites."
|
||||||
msgstr "%s foi adicionado aos seus favoritos."
|
msgstr "%s foi adicionado aos seus favoritos."
|
||||||
|
|
||||||
#: js/ui/appFavorites.js:172
|
#: js/ui/appFavorites.js:174
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s has been removed from your favorites."
|
msgid "%s has been removed from your favorites."
|
||||||
msgstr "%s foi removido dos seus favoritos."
|
msgstr "%s foi removido dos seus favoritos."
|
||||||
@ -876,7 +876,7 @@ msgstr "Unidade externa desconectada"
|
|||||||
msgid "Open with %s"
|
msgid "Open with %s"
|
||||||
msgstr "Abrir com %s"
|
msgstr "Abrir com %s"
|
||||||
|
|
||||||
#: js/ui/components/keyring.js:107 js/ui/components/polkitAgent.js:284
|
#: js/ui/components/keyring.js:107 js/ui/components/polkitAgent.js:295
|
||||||
msgid "Password:"
|
msgid "Password:"
|
||||||
msgstr "Senha:"
|
msgstr "Senha:"
|
||||||
|
|
||||||
@ -964,15 +964,15 @@ msgstr "Uma senha é necessária para se conectar a “%s”."
|
|||||||
msgid "Network Manager"
|
msgid "Network Manager"
|
||||||
msgstr "Gerenciador de rede"
|
msgstr "Gerenciador de rede"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:43
|
#: js/ui/components/polkitAgent.js:48
|
||||||
msgid "Authentication Required"
|
msgid "Authentication Required"
|
||||||
msgstr "Autenticação necessária"
|
msgstr "Autenticação necessária"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:71
|
#: js/ui/components/polkitAgent.js:76
|
||||||
msgid "Administrator"
|
msgid "Administrator"
|
||||||
msgstr "Administrador"
|
msgstr "Administrador"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:151
|
#: js/ui/components/polkitAgent.js:156
|
||||||
msgid "Authenticate"
|
msgid "Authenticate"
|
||||||
msgstr "Autenticação"
|
msgstr "Autenticação"
|
||||||
|
|
||||||
@ -980,7 +980,7 @@ msgstr "Autenticação"
|
|||||||
#. * requested authentication was not gained; this can happen
|
#. * requested authentication was not gained; this can happen
|
||||||
#. * because of an authentication error (like invalid password),
|
#. * because of an authentication error (like invalid password),
|
||||||
#. * for instance.
|
#. * for instance.
|
||||||
#: js/ui/components/polkitAgent.js:270 js/ui/shellMountOperation.js:327
|
#: js/ui/components/polkitAgent.js:281 js/ui/shellMountOperation.js:327
|
||||||
msgid "Sorry, that didn’t work. Please try again."
|
msgid "Sorry, that didn’t work. Please try again."
|
||||||
msgstr "Desculpe, isto não funcionou. Por favor, tente novamente."
|
msgstr "Desculpe, isto não funcionou. Por favor, tente novamente."
|
||||||
|
|
||||||
@ -1028,7 +1028,7 @@ msgstr "Adicionar relógios mundiais…"
|
|||||||
msgid "World Clocks"
|
msgid "World Clocks"
|
||||||
msgstr "Relógios mundiais"
|
msgstr "Relógios mundiais"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:225
|
#: js/ui/dateMenu.js:227
|
||||||
msgid "Weather"
|
msgid "Weather"
|
||||||
msgstr "Meteorologia"
|
msgstr "Meteorologia"
|
||||||
|
|
||||||
@ -1036,7 +1036,7 @@ msgstr "Meteorologia"
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:289
|
#: js/ui/dateMenu.js:291
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s all day."
|
msgid "%s all day."
|
||||||
msgstr "%s por todo o dia."
|
msgstr "%s por todo o dia."
|
||||||
@ -1045,7 +1045,7 @@ msgstr "%s por todo o dia."
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:295
|
#: js/ui/dateMenu.js:297
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s, then %s later."
|
msgid "%s, then %s later."
|
||||||
msgstr "%s, depois %s mais tarde."
|
msgstr "%s, depois %s mais tarde."
|
||||||
@ -1054,30 +1054,30 @@ msgstr "%s, depois %s mais tarde."
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:301
|
#: js/ui/dateMenu.js:303
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s, then %s, followed by %s later."
|
msgid "%s, then %s, followed by %s later."
|
||||||
msgstr "%s, depois %s, seguido de %s mais tarde."
|
msgstr "%s, depois %s, seguido de %s mais tarde."
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:312
|
#: js/ui/dateMenu.js:314
|
||||||
msgid "Select a location…"
|
msgid "Select a location…"
|
||||||
msgstr "Selecione uma localização…"
|
msgstr "Selecione uma localização…"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:315
|
#: js/ui/dateMenu.js:317
|
||||||
msgid "Loading…"
|
msgid "Loading…"
|
||||||
msgstr "Carregando…"
|
msgstr "Carregando…"
|
||||||
|
|
||||||
#. Translators: %s is a temperature with unit, e.g. "23℃"
|
#. Translators: %s is a temperature with unit, e.g. "23℃"
|
||||||
#: js/ui/dateMenu.js:321
|
#: js/ui/dateMenu.js:323
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "Feels like %s."
|
msgid "Feels like %s."
|
||||||
msgstr "Sensação térmica de %s."
|
msgstr "Sensação térmica de %s."
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:324
|
#: js/ui/dateMenu.js:326
|
||||||
msgid "Go online for weather information"
|
msgid "Go online for weather information"
|
||||||
msgstr "Conecte-se à internet para obter as informações meteorológicas"
|
msgstr "Conecte-se à internet para obter as informações meteorológicas"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:326
|
#: js/ui/dateMenu.js:328
|
||||||
msgid "Weather information is currently unavailable"
|
msgid "Weather information is currently unavailable"
|
||||||
msgstr "No momento as informações meteorológicas não estão disponíveis"
|
msgstr "No momento as informações meteorológicas não estão disponíveis"
|
||||||
|
|
||||||
|
40
po/ru.po
40
po/ru.po
@ -18,8 +18,8 @@ msgid ""
|
|||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: gnome-shell\n"
|
"Project-Id-Version: gnome-shell\n"
|
||||||
"Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n"
|
"Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n"
|
||||||
"POT-Creation-Date: 2018-04-12 11:47+0000\n"
|
"POT-Creation-Date: 2018-04-13 18:31+0000\n"
|
||||||
"PO-Revision-Date: 2018-04-12 16:48+0300\n"
|
"PO-Revision-Date: 2018-04-19 23:31+0300\n"
|
||||||
"Last-Translator: Stas Solovey <whats_up@tut.by>\n"
|
"Last-Translator: Stas Solovey <whats_up@tut.by>\n"
|
||||||
"Language-Team: Русский <gnome-cyr@gnome.org>\n"
|
"Language-Team: Русский <gnome-cyr@gnome.org>\n"
|
||||||
"Language: ru\n"
|
"Language: ru\n"
|
||||||
@ -344,7 +344,7 @@ msgid "There was an error loading the preferences dialog for %s:"
|
|||||||
msgstr "Возникла ошибка загрузки диалогового окна параметров для %s:"
|
msgstr "Возникла ошибка загрузки диалогового окна параметров для %s:"
|
||||||
|
|
||||||
#: js/gdm/authPrompt.js:147 js/ui/audioDeviceSelection.js:71
|
#: js/gdm/authPrompt.js:147 js/ui/audioDeviceSelection.js:71
|
||||||
#: js/ui/components/networkAgent.js:117 js/ui/components/polkitAgent.js:149
|
#: js/ui/components/networkAgent.js:117 js/ui/components/polkitAgent.js:153
|
||||||
#: js/ui/endSessionDialog.js:482 js/ui/extensionDownloader.js:197
|
#: js/ui/endSessionDialog.js:482 js/ui/extensionDownloader.js:197
|
||||||
#: js/ui/shellMountOperation.js:343 js/ui/status/network.js:919
|
#: js/ui/shellMountOperation.js:343 js/ui/status/network.js:919
|
||||||
msgid "Cancel"
|
msgid "Cancel"
|
||||||
@ -563,7 +563,6 @@ msgstr "Вчера, %H∶%M"
|
|||||||
msgid "%A, %H∶%M"
|
msgid "%A, %H∶%M"
|
||||||
msgstr "%A, %H∶%M"
|
msgstr "%A, %H∶%M"
|
||||||
|
|
||||||
# fix даты "11 мар., 20:35"
|
|
||||||
#. Translators: this is the month name and day number
|
#. Translators: this is the month name and day number
|
||||||
#. followed by a time string in 24h format.
|
#. followed by a time string in 24h format.
|
||||||
#. i.e. "May 25, 14:30"
|
#. i.e. "May 25, 14:30"
|
||||||
@ -572,7 +571,6 @@ msgstr "%A, %H∶%M"
|
|||||||
msgid "%B %d, %H∶%M"
|
msgid "%B %d, %H∶%M"
|
||||||
msgstr "%-d %B, %H∶%M"
|
msgstr "%-d %B, %H∶%M"
|
||||||
|
|
||||||
# fix даты
|
|
||||||
#. Translators: this is the month name, day number, year
|
#. Translators: this is the month name, day number, year
|
||||||
#. number followed by a time string in 24h format.
|
#. number followed by a time string in 24h format.
|
||||||
#. i.e. "May 25 2012, 14:30"
|
#. i.e. "May 25 2012, 14:30"
|
||||||
@ -581,7 +579,6 @@ msgstr "%-d %B, %H∶%M"
|
|||||||
msgid "%B %d %Y, %H∶%M"
|
msgid "%B %d %Y, %H∶%M"
|
||||||
msgstr "%-d %B %Y, %H∶%M"
|
msgstr "%-d %B %Y, %H∶%M"
|
||||||
|
|
||||||
# по всей видимости разрабы коммент перепутали c "Translators: Time in 12h format"
|
|
||||||
#. Translators: Time in 12h format
|
#. Translators: Time in 12h format
|
||||||
#: js/misc/util.js:257
|
#: js/misc/util.js:257
|
||||||
msgid "%l∶%M %p"
|
msgid "%l∶%M %p"
|
||||||
@ -823,7 +820,6 @@ msgctxt "calendar heading"
|
|||||||
msgid "%A, %B %d"
|
msgid "%A, %B %d"
|
||||||
msgstr "%A, %-d %B"
|
msgstr "%A, %-d %B"
|
||||||
|
|
||||||
# fix для даты в календаре и на экране блокировки
|
|
||||||
#: js/ui/calendar.js:868
|
#: js/ui/calendar.js:868
|
||||||
msgctxt "calendar heading"
|
msgctxt "calendar heading"
|
||||||
msgid "%A, %B %d, %Y"
|
msgid "%A, %B %d, %Y"
|
||||||
@ -876,7 +872,7 @@ msgstr "Внешний диск отключён"
|
|||||||
msgid "Open with %s"
|
msgid "Open with %s"
|
||||||
msgstr "Открыть с помощью %s"
|
msgstr "Открыть с помощью %s"
|
||||||
|
|
||||||
#: js/ui/components/keyring.js:107 js/ui/components/polkitAgent.js:285
|
#: js/ui/components/keyring.js:107 js/ui/components/polkitAgent.js:295
|
||||||
msgid "Password:"
|
msgid "Password:"
|
||||||
msgstr "Пароль:"
|
msgstr "Пароль:"
|
||||||
|
|
||||||
@ -963,15 +959,15 @@ msgstr "Для подключения к «%s» требуется пароль.
|
|||||||
msgid "Network Manager"
|
msgid "Network Manager"
|
||||||
msgstr "Диспетчер сети"
|
msgstr "Диспетчер сети"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:44
|
#: js/ui/components/polkitAgent.js:48
|
||||||
msgid "Authentication Required"
|
msgid "Authentication Required"
|
||||||
msgstr "Требуется подтверждение подлинности"
|
msgstr "Требуется подтверждение подлинности"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:72
|
#: js/ui/components/polkitAgent.js:76
|
||||||
msgid "Administrator"
|
msgid "Administrator"
|
||||||
msgstr "Администратор"
|
msgstr "Администратор"
|
||||||
|
|
||||||
#: js/ui/components/polkitAgent.js:152
|
#: js/ui/components/polkitAgent.js:156
|
||||||
msgid "Authenticate"
|
msgid "Authenticate"
|
||||||
msgstr "Подтвердить"
|
msgstr "Подтвердить"
|
||||||
|
|
||||||
@ -979,7 +975,7 @@ msgstr "Подтвердить"
|
|||||||
#. * requested authentication was not gained; this can happen
|
#. * requested authentication was not gained; this can happen
|
||||||
#. * because of an authentication error (like invalid password),
|
#. * because of an authentication error (like invalid password),
|
||||||
#. * for instance.
|
#. * for instance.
|
||||||
#: js/ui/components/polkitAgent.js:271 js/ui/shellMountOperation.js:327
|
#: js/ui/components/polkitAgent.js:281 js/ui/shellMountOperation.js:327
|
||||||
msgid "Sorry, that didn’t work. Please try again."
|
msgid "Sorry, that didn’t work. Please try again."
|
||||||
msgstr "Не удалось подтвердить подлинность. Попробуйте снова."
|
msgstr "Не удалось подтвердить подлинность. Попробуйте снова."
|
||||||
|
|
||||||
@ -1004,7 +1000,6 @@ msgstr "Показать приложения"
|
|||||||
msgid "Dash"
|
msgid "Dash"
|
||||||
msgstr "Панель приложений"
|
msgstr "Панель приложений"
|
||||||
|
|
||||||
# fix для даты в календаре и на экране блокировки
|
|
||||||
#. Translators: This is the date format to use when the calendar popup is
|
#. Translators: This is the date format to use when the calendar popup is
|
||||||
#. * shown - it is shown just below the time in the shell (e.g. "Tue 9:29 AM").
|
#. * shown - it is shown just below the time in the shell (e.g. "Tue 9:29 AM").
|
||||||
#.
|
#.
|
||||||
@ -1012,7 +1007,6 @@ msgstr "Панель приложений"
|
|||||||
msgid "%B %e %Y"
|
msgid "%B %e %Y"
|
||||||
msgstr "%-d %B %Y"
|
msgstr "%-d %B %Y"
|
||||||
|
|
||||||
# fix для даты в календаре и на экране блокировки
|
|
||||||
#. Translators: This is the accessible name of the date button shown
|
#. Translators: This is the accessible name of the date button shown
|
||||||
#. * below the time in the shell; it should combine the weekday and the
|
#. * below the time in the shell; it should combine the weekday and the
|
||||||
#. * date, e.g. "Tuesday February 17 2015".
|
#. * date, e.g. "Tuesday February 17 2015".
|
||||||
@ -1029,7 +1023,7 @@ msgstr "Добавить мировые часы…"
|
|||||||
msgid "World Clocks"
|
msgid "World Clocks"
|
||||||
msgstr "Мировые часы"
|
msgstr "Мировые часы"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:225
|
#: js/ui/dateMenu.js:227
|
||||||
msgid "Weather"
|
msgid "Weather"
|
||||||
msgstr "Погода"
|
msgstr "Погода"
|
||||||
|
|
||||||
@ -1037,7 +1031,7 @@ msgstr "Погода"
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:289
|
#: js/ui/dateMenu.js:291
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s all day."
|
msgid "%s all day."
|
||||||
msgstr "%s весь день."
|
msgstr "%s весь день."
|
||||||
@ -1046,7 +1040,7 @@ msgstr "%s весь день."
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:295
|
#: js/ui/dateMenu.js:297
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s, then %s later."
|
msgid "%s, then %s later."
|
||||||
msgstr "%s, затем позднее %s."
|
msgstr "%s, затем позднее %s."
|
||||||
@ -1055,30 +1049,30 @@ msgstr "%s, затем позднее %s."
|
|||||||
#. libgweather for the possible condition strings. If at all
|
#. libgweather for the possible condition strings. If at all
|
||||||
#. possible, the sentence should match the grammatical case etc. of
|
#. possible, the sentence should match the grammatical case etc. of
|
||||||
#. the inserted conditions.
|
#. the inserted conditions.
|
||||||
#: js/ui/dateMenu.js:301
|
#: js/ui/dateMenu.js:303
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "%s, then %s, followed by %s later."
|
msgid "%s, then %s, followed by %s later."
|
||||||
msgstr "%s, затем %s, позже %s."
|
msgstr "%s, затем %s, позже %s."
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:312
|
#: js/ui/dateMenu.js:314
|
||||||
msgid "Select a location…"
|
msgid "Select a location…"
|
||||||
msgstr "Выберите местоположение…"
|
msgstr "Выберите местоположение…"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:315
|
#: js/ui/dateMenu.js:317
|
||||||
msgid "Loading…"
|
msgid "Loading…"
|
||||||
msgstr "Загрузка…"
|
msgstr "Загрузка…"
|
||||||
|
|
||||||
#. Translators: %s is a temperature with unit, e.g. "23℃"
|
#. Translators: %s is a temperature with unit, e.g. "23℃"
|
||||||
#: js/ui/dateMenu.js:321
|
#: js/ui/dateMenu.js:323
|
||||||
#, javascript-format
|
#, javascript-format
|
||||||
msgid "Feels like %s."
|
msgid "Feels like %s."
|
||||||
msgstr "Ощущается как %s."
|
msgstr "Ощущается как %s."
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:324
|
#: js/ui/dateMenu.js:326
|
||||||
msgid "Go online for weather information"
|
msgid "Go online for weather information"
|
||||||
msgstr "Подключите интернет для получения информации о погоде"
|
msgstr "Подключите интернет для получения информации о погоде"
|
||||||
|
|
||||||
#: js/ui/dateMenu.js:326
|
#: js/ui/dateMenu.js:328
|
||||||
msgid "Weather information is currently unavailable"
|
msgid "Weather information is currently unavailable"
|
||||||
msgstr "Информация о погоде сейчас недоступна"
|
msgstr "Информация о погоде сейчас недоступна"
|
||||||
|
|
||||||
|
@ -103,6 +103,7 @@ libshell_menu_dep = declare_dependency(link_with: libshell_menu)
|
|||||||
|
|
||||||
libshell_public_headers = [
|
libshell_public_headers = [
|
||||||
'shell-app.h',
|
'shell-app.h',
|
||||||
|
'shell-app-cache.h',
|
||||||
'shell-app-system.h',
|
'shell-app-system.h',
|
||||||
'shell-app-usage.h',
|
'shell-app-usage.h',
|
||||||
'shell-embedded-window.h',
|
'shell-embedded-window.h',
|
||||||
@ -138,6 +139,7 @@ libshell_private_headers = [
|
|||||||
libshell_sources = [
|
libshell_sources = [
|
||||||
'gnome-shell-plugin.c',
|
'gnome-shell-plugin.c',
|
||||||
'shell-app.c',
|
'shell-app.c',
|
||||||
|
'shell-app-cache.c',
|
||||||
'shell-app-system.c',
|
'shell-app-system.c',
|
||||||
'shell-app-usage.c',
|
'shell-app-usage.c',
|
||||||
'shell-embedded-window.c',
|
'shell-embedded-window.c',
|
||||||
|
439
src/shell-app-cache.c
Normal file
439
src/shell-app-cache.c
Normal file
@ -0,0 +1,439 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "shell-app-cache.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SECTION:shell-app-cache
|
||||||
|
* @title: ShellAppCache
|
||||||
|
* @short_description: application information cache
|
||||||
|
*
|
||||||
|
* The #ShellAppCache is responsible for caching information about #GAppInfo
|
||||||
|
* to ensure that the compositor thread never needs to perform disk reads to
|
||||||
|
* access them. All of the work is done off-thread. When the new data has
|
||||||
|
* been loaded, a #ShellAppCache::changed signal is emitted.
|
||||||
|
*
|
||||||
|
* Additionally, the #ShellAppCache caches information about translations for
|
||||||
|
* directories. This allows translation provided in [Desktop Entry] GKeyFiles
|
||||||
|
* to be available when building StLabel and other elements without performing
|
||||||
|
* costly disk reads.
|
||||||
|
*
|
||||||
|
* Various monitors are used to keep this information up to date while the
|
||||||
|
* Shell is running.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define DEFAULT_TIMEOUT_SECONDS 5
|
||||||
|
|
||||||
|
struct _ShellAppCache
|
||||||
|
{
|
||||||
|
GObject parent_instance;
|
||||||
|
|
||||||
|
GMutex mutex;
|
||||||
|
|
||||||
|
GAppInfoMonitor *monitor;
|
||||||
|
GPtrArray *dir_monitors;
|
||||||
|
GHashTable *folders;
|
||||||
|
GCancellable *cancellable;
|
||||||
|
GList *app_infos;
|
||||||
|
|
||||||
|
guint queued_update;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
GList *app_infos;
|
||||||
|
GHashTable *folders;
|
||||||
|
} CacheState;
|
||||||
|
|
||||||
|
G_DEFINE_TYPE (ShellAppCache, shell_app_cache, G_TYPE_OBJECT)
|
||||||
|
|
||||||
|
enum {
|
||||||
|
CHANGED,
|
||||||
|
N_SIGNALS
|
||||||
|
};
|
||||||
|
|
||||||
|
static guint signals [N_SIGNALS];
|
||||||
|
|
||||||
|
static void
|
||||||
|
cache_state_free (CacheState *state)
|
||||||
|
{
|
||||||
|
g_clear_pointer (&state->folders, g_hash_table_unref);
|
||||||
|
g_list_free_full (state->app_infos, g_object_unref);
|
||||||
|
g_slice_free (CacheState, state);
|
||||||
|
}
|
||||||
|
|
||||||
|
static CacheState *
|
||||||
|
cache_state_new (void)
|
||||||
|
{
|
||||||
|
CacheState *state;
|
||||||
|
|
||||||
|
state = g_slice_new0 (CacheState);
|
||||||
|
state->folders = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
||||||
|
|
||||||
|
return g_steal_pointer (&state);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* shell_app_cache_get_default:
|
||||||
|
*
|
||||||
|
* Gets the default #ShellAppCache.
|
||||||
|
*
|
||||||
|
* Returns: (transfer none): a #ShellAppCache
|
||||||
|
*/
|
||||||
|
ShellAppCache *
|
||||||
|
shell_app_cache_get_default (void)
|
||||||
|
{
|
||||||
|
static ShellAppCache *instance;
|
||||||
|
|
||||||
|
if (instance == NULL)
|
||||||
|
{
|
||||||
|
instance = g_object_new (SHELL_TYPE_APP_CACHE, NULL);
|
||||||
|
g_object_add_weak_pointer (G_OBJECT (instance), (gpointer *)&instance);
|
||||||
|
}
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
load_folder (GHashTable *folders,
|
||||||
|
const char *path)
|
||||||
|
{
|
||||||
|
g_autoptr(GDir) dir = NULL;
|
||||||
|
const char *name;
|
||||||
|
|
||||||
|
g_assert (folders != NULL);
|
||||||
|
g_assert (path != NULL);
|
||||||
|
|
||||||
|
dir = g_dir_open (path, 0, NULL);
|
||||||
|
if (dir == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
while ((name = g_dir_read_name (dir)))
|
||||||
|
{
|
||||||
|
g_autofree gchar *filename = NULL;
|
||||||
|
g_autoptr(GKeyFile) keyfile = NULL;
|
||||||
|
|
||||||
|
/* First added wins */
|
||||||
|
if (g_hash_table_contains (folders, name))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
filename = g_build_filename (path, name, NULL);
|
||||||
|
keyfile = g_key_file_new ();
|
||||||
|
|
||||||
|
if (g_key_file_load_from_file (keyfile, filename, G_KEY_FILE_NONE, NULL))
|
||||||
|
g_hash_table_insert (folders,
|
||||||
|
g_strdup (name),
|
||||||
|
g_key_file_get_locale_string (keyfile,
|
||||||
|
"Desktop Entry",
|
||||||
|
"Name",
|
||||||
|
NULL,
|
||||||
|
NULL));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
load_folders (GHashTable *folders)
|
||||||
|
{
|
||||||
|
const char * const *dirs;
|
||||||
|
g_autofree gchar *userdir = NULL;
|
||||||
|
guint i;
|
||||||
|
|
||||||
|
g_assert (folders != NULL);
|
||||||
|
|
||||||
|
userdir = g_build_filename (g_get_user_data_dir (), "desktop-directories", NULL);
|
||||||
|
load_folder (folders, userdir);
|
||||||
|
|
||||||
|
dirs = g_get_system_data_dirs ();
|
||||||
|
for (i = 0; dirs[i] != NULL; i++)
|
||||||
|
{
|
||||||
|
g_autofree gchar *sysdir = g_build_filename (dirs[i], "desktop-directories", NULL);
|
||||||
|
load_folder (folders, sysdir);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
shell_app_cache_worker (GTask *task,
|
||||||
|
gpointer source_object,
|
||||||
|
gpointer task_data,
|
||||||
|
GCancellable *cancellable)
|
||||||
|
{
|
||||||
|
CacheState *state;
|
||||||
|
|
||||||
|
g_assert (G_IS_TASK (task));
|
||||||
|
g_assert (SHELL_IS_APP_CACHE (source_object));
|
||||||
|
|
||||||
|
state = cache_state_new ();
|
||||||
|
state->app_infos = g_app_info_get_all ();
|
||||||
|
load_folders (state->folders);
|
||||||
|
|
||||||
|
g_task_return_pointer (task, state, (GDestroyNotify)cache_state_free);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
apply_update_cb (GObject *object,
|
||||||
|
GAsyncResult *result,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
ShellAppCache *cache = (ShellAppCache *)object;
|
||||||
|
g_autoptr(GError) error = NULL;
|
||||||
|
CacheState *state;
|
||||||
|
|
||||||
|
g_assert (SHELL_IS_APP_CACHE (cache));
|
||||||
|
g_assert (G_IS_TASK (result));
|
||||||
|
g_assert (user_data == NULL);
|
||||||
|
|
||||||
|
state = g_task_propagate_pointer (G_TASK (result), &error);
|
||||||
|
|
||||||
|
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
||||||
|
return;
|
||||||
|
|
||||||
|
g_mutex_lock (&cache->mutex);
|
||||||
|
g_list_free_full (cache->app_infos, g_object_unref);
|
||||||
|
cache->app_infos = g_steal_pointer (&state->app_infos);
|
||||||
|
g_clear_pointer (&cache->folders, g_hash_table_unref);
|
||||||
|
cache->folders = g_steal_pointer (&state->folders);
|
||||||
|
g_mutex_unlock (&cache->mutex);
|
||||||
|
|
||||||
|
g_signal_emit (cache, signals[CHANGED], 0);
|
||||||
|
|
||||||
|
cache_state_free (state);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
shell_app_cache_do_update (gpointer user_data)
|
||||||
|
{
|
||||||
|
ShellAppCache *cache = user_data;
|
||||||
|
g_autoptr(GTask) task = NULL;
|
||||||
|
|
||||||
|
cache->queued_update = 0;
|
||||||
|
|
||||||
|
/* Reset the cancellable state so we don't race with
|
||||||
|
* two updates coming back overlapped and applying the
|
||||||
|
* information in the wrong order.
|
||||||
|
*/
|
||||||
|
g_cancellable_cancel (cache->cancellable);
|
||||||
|
g_clear_object (&cache->cancellable);
|
||||||
|
cache->cancellable = g_cancellable_new ();
|
||||||
|
|
||||||
|
task = g_task_new (cache, cache->cancellable, apply_update_cb, NULL);
|
||||||
|
g_task_set_source_tag (task, shell_app_cache_do_update);
|
||||||
|
g_task_run_in_thread (task, shell_app_cache_worker);
|
||||||
|
|
||||||
|
return G_SOURCE_REMOVE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
shell_app_cache_queue_update (ShellAppCache *self)
|
||||||
|
{
|
||||||
|
g_assert (SHELL_IS_APP_CACHE (self));
|
||||||
|
|
||||||
|
if (self->queued_update != 0)
|
||||||
|
g_source_remove (self->queued_update);
|
||||||
|
|
||||||
|
self->queued_update = g_timeout_add_seconds (DEFAULT_TIMEOUT_SECONDS,
|
||||||
|
shell_app_cache_do_update,
|
||||||
|
self);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
monitor_data_dir (ShellAppCache *self,
|
||||||
|
const gchar *directory)
|
||||||
|
{
|
||||||
|
g_autofree gchar *subdir = NULL;
|
||||||
|
g_autoptr(GFile) file = NULL;
|
||||||
|
g_autoptr(GFileMonitor) monitor = NULL;
|
||||||
|
|
||||||
|
g_assert (SHELL_IS_APP_CACHE (self));
|
||||||
|
|
||||||
|
if (directory == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
subdir = g_build_filename (directory, "desktop-directories", NULL);
|
||||||
|
file = g_file_new_for_path (subdir);
|
||||||
|
monitor = g_file_monitor_directory (file, G_FILE_MONITOR_NONE, NULL, NULL);
|
||||||
|
|
||||||
|
if (monitor != NULL)
|
||||||
|
{
|
||||||
|
g_file_monitor_set_rate_limit (monitor, DEFAULT_TIMEOUT_SECONDS * 1000);
|
||||||
|
g_signal_connect_object (monitor,
|
||||||
|
"changed",
|
||||||
|
G_CALLBACK (shell_app_cache_queue_update),
|
||||||
|
self,
|
||||||
|
G_CONNECT_SWAPPED);
|
||||||
|
g_ptr_array_add (self->dir_monitors, g_steal_pointer (&monitor));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
shell_app_cache_finalize (GObject *object)
|
||||||
|
{
|
||||||
|
ShellAppCache *self = (ShellAppCache *)object;
|
||||||
|
|
||||||
|
g_clear_object (&self->monitor);
|
||||||
|
|
||||||
|
if (self->queued_update)
|
||||||
|
{
|
||||||
|
g_source_remove (self->queued_update);
|
||||||
|
self->queued_update = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_clear_pointer (&self->dir_monitors, g_ptr_array_unref);
|
||||||
|
g_clear_pointer (&self->folders, g_hash_table_unref);
|
||||||
|
g_list_free_full (self->app_infos, g_object_unref);
|
||||||
|
g_mutex_clear (&self->mutex);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (shell_app_cache_parent_class)->finalize (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
shell_app_cache_class_init (ShellAppCacheClass *klass)
|
||||||
|
{
|
||||||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||||||
|
|
||||||
|
object_class->finalize = shell_app_cache_finalize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ShellAppCache::changed:
|
||||||
|
*
|
||||||
|
* The "changed" signal is emitted when the cache has updated
|
||||||
|
* information about installed applications.
|
||||||
|
*/
|
||||||
|
signals [CHANGED] =
|
||||||
|
g_signal_new ("changed",
|
||||||
|
G_TYPE_FROM_CLASS (klass),
|
||||||
|
G_SIGNAL_RUN_LAST,
|
||||||
|
0, NULL, NULL, NULL,
|
||||||
|
G_TYPE_NONE, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
shell_app_cache_init (ShellAppCache *self)
|
||||||
|
{
|
||||||
|
const gchar * const *sysdirs;
|
||||||
|
guint i;
|
||||||
|
|
||||||
|
g_mutex_init (&self->mutex);
|
||||||
|
|
||||||
|
/* Monitor directories for translation changes */
|
||||||
|
self->dir_monitors = g_ptr_array_new_with_free_func (g_object_unref);
|
||||||
|
monitor_data_dir (self, g_get_user_data_dir ());
|
||||||
|
sysdirs = g_get_system_data_dirs ();
|
||||||
|
for (i = 0; sysdirs[i] != NULL; i++)
|
||||||
|
monitor_data_dir (self, sysdirs[i]);
|
||||||
|
|
||||||
|
/* Load translated directory names immediately */
|
||||||
|
self->folders = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
||||||
|
load_folders (self->folders);
|
||||||
|
|
||||||
|
/* Setup AppMonitor to track changes */
|
||||||
|
self->monitor = g_app_info_monitor_get ();
|
||||||
|
g_signal_connect_object (self->monitor,
|
||||||
|
"changed",
|
||||||
|
G_CALLBACK (shell_app_cache_queue_update),
|
||||||
|
self,
|
||||||
|
G_CONNECT_SWAPPED);
|
||||||
|
self->app_infos = g_app_info_get_all ();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* shell_app_cache_get_all:
|
||||||
|
* @cache: (nullable): a #ShellAppCache or %NULL
|
||||||
|
*
|
||||||
|
* Like g_app_info_get_all() but always returns a
|
||||||
|
* cached set of application info so the caller can be
|
||||||
|
* sure that I/O will not happen on the current thread.
|
||||||
|
*
|
||||||
|
* Returns: (transfer full) (element-type GAppInfo):
|
||||||
|
* a newly allocated GList of references to GAppInfos.
|
||||||
|
*/
|
||||||
|
GList *
|
||||||
|
shell_app_cache_get_all (ShellAppCache *cache)
|
||||||
|
{
|
||||||
|
GList *ret;
|
||||||
|
|
||||||
|
if (cache == NULL)
|
||||||
|
cache = shell_app_cache_get_default ();
|
||||||
|
|
||||||
|
g_return_val_if_fail (SHELL_IS_APP_CACHE (cache), NULL);
|
||||||
|
|
||||||
|
g_mutex_lock (&cache->mutex);
|
||||||
|
ret = g_list_copy_deep (cache->app_infos, (GCopyFunc)g_object_ref, NULL);
|
||||||
|
g_mutex_unlock (&cache->mutex);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* shell_app_cache_get_info:
|
||||||
|
* @cache: (nullable): a #ShellAppCache or %NULL
|
||||||
|
* @id: the application id
|
||||||
|
*
|
||||||
|
* A replacement for g_desktop_app_info_new() that will lookup the information
|
||||||
|
* from the cache instead of (re)loading from disk.
|
||||||
|
*
|
||||||
|
* Returns: (nullable) (transfer full): a #GDesktopAppInfo or %NULL
|
||||||
|
*/
|
||||||
|
GDesktopAppInfo *
|
||||||
|
shell_app_cache_get_info (ShellAppCache *cache,
|
||||||
|
const char *id)
|
||||||
|
{
|
||||||
|
GDesktopAppInfo *ret = NULL;
|
||||||
|
const GList *iter;
|
||||||
|
|
||||||
|
if (cache == NULL)
|
||||||
|
cache = shell_app_cache_get_default ();
|
||||||
|
|
||||||
|
g_return_val_if_fail (SHELL_IS_APP_CACHE (cache), NULL);
|
||||||
|
|
||||||
|
g_mutex_lock (&cache->mutex);
|
||||||
|
|
||||||
|
for (iter = cache->app_infos; iter != NULL; iter = iter->next)
|
||||||
|
{
|
||||||
|
GAppInfo *info = iter->data;
|
||||||
|
|
||||||
|
if (g_strcmp0 (id, g_app_info_get_id (info)) == 0)
|
||||||
|
{
|
||||||
|
ret = g_object_ref (G_DESKTOP_APP_INFO (info));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
g_mutex_unlock (&cache->mutex);
|
||||||
|
|
||||||
|
return g_steal_pointer (&ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* shell_app_cache_translate_folder:
|
||||||
|
* @cache: (nullable): a #ShellAppCache or %NULL
|
||||||
|
* @name: the folder name
|
||||||
|
*
|
||||||
|
* Gets the translated folder name for @name if any exists. Otherwise
|
||||||
|
* a copy of @name is returned.
|
||||||
|
*
|
||||||
|
* Returns: the translated string
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
shell_app_cache_translate_folder (ShellAppCache *cache,
|
||||||
|
const char *name)
|
||||||
|
{
|
||||||
|
char *ret;
|
||||||
|
|
||||||
|
if (cache == NULL)
|
||||||
|
cache = shell_app_cache_get_default ();
|
||||||
|
|
||||||
|
g_return_val_if_fail (SHELL_IS_APP_CACHE (cache), NULL);
|
||||||
|
|
||||||
|
if (name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
g_mutex_lock (&cache->mutex);
|
||||||
|
ret = g_strdup (g_hash_table_lookup (cache->folders, name));
|
||||||
|
g_mutex_unlock (&cache->mutex);
|
||||||
|
|
||||||
|
if (ret == NULL)
|
||||||
|
ret = g_strdup (name);
|
||||||
|
|
||||||
|
return g_steal_pointer (&ret);
|
||||||
|
}
|
19
src/shell-app-cache.h
Normal file
19
src/shell-app-cache.h
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
||||||
|
#ifndef __SHELL_APP_CACHE__
|
||||||
|
#define __SHELL_APP_CACHE__
|
||||||
|
|
||||||
|
#include <gio/gio.h>
|
||||||
|
#include <gio/gdesktopappinfo.h>
|
||||||
|
|
||||||
|
#define SHELL_TYPE_APP_CACHE (shell_app_cache_get_type())
|
||||||
|
|
||||||
|
G_DECLARE_FINAL_TYPE (ShellAppCache, shell_app_cache, SHELL, APP_CACHE, GObject)
|
||||||
|
|
||||||
|
ShellAppCache *shell_app_cache_get_default (void);
|
||||||
|
GList *shell_app_cache_get_all (ShellAppCache *cache);
|
||||||
|
GDesktopAppInfo *shell_app_cache_get_info (ShellAppCache *cache,
|
||||||
|
const char *id);
|
||||||
|
char *shell_app_cache_translate_folder (ShellAppCache *cache,
|
||||||
|
const char *name);
|
||||||
|
|
||||||
|
#endif /* __SHELL_APP_CACHE__ */
|
@ -9,11 +9,20 @@
|
|||||||
#include <gio/gio.h>
|
#include <gio/gio.h>
|
||||||
#include <glib/gi18n.h>
|
#include <glib/gi18n.h>
|
||||||
|
|
||||||
|
#include "shell-app-cache.h"
|
||||||
#include "shell-app-private.h"
|
#include "shell-app-private.h"
|
||||||
#include "shell-window-tracker-private.h"
|
#include "shell-window-tracker-private.h"
|
||||||
#include "shell-app-system-private.h"
|
#include "shell-app-system-private.h"
|
||||||
#include "shell-global.h"
|
#include "shell-global.h"
|
||||||
#include "shell-util.h"
|
#include "shell-util.h"
|
||||||
|
#include "st.h"
|
||||||
|
|
||||||
|
/* Rescan for at most RESCAN_TIMEOUT_MS * MAX_RESCAN_RETRIES. That
|
||||||
|
* should be plenty of time for even a slow spinning drive to update
|
||||||
|
* the icon cache.
|
||||||
|
*/
|
||||||
|
#define RESCAN_TIMEOUT_MS 2500
|
||||||
|
#define MAX_RESCAN_RETRIES 6
|
||||||
|
|
||||||
/* Vendor prefixes are something that can be preprended to a .desktop
|
/* Vendor prefixes are something that can be preprended to a .desktop
|
||||||
* file name. Undo this.
|
* file name. Undo this.
|
||||||
@ -50,6 +59,9 @@ struct _ShellAppSystemPrivate {
|
|||||||
GHashTable *running_apps;
|
GHashTable *running_apps;
|
||||||
GHashTable *id_to_app;
|
GHashTable *id_to_app;
|
||||||
GHashTable *startup_wm_class_to_id;
|
GHashTable *startup_wm_class_to_id;
|
||||||
|
|
||||||
|
guint rescan_icons_timeout_id;
|
||||||
|
guint n_rescan_retries;
|
||||||
};
|
};
|
||||||
|
|
||||||
static void shell_app_system_finalize (GObject *object);
|
static void shell_app_system_finalize (GObject *object);
|
||||||
@ -82,12 +94,14 @@ static void
|
|||||||
scan_startup_wm_class_to_id (ShellAppSystem *self)
|
scan_startup_wm_class_to_id (ShellAppSystem *self)
|
||||||
{
|
{
|
||||||
ShellAppSystemPrivate *priv = self->priv;
|
ShellAppSystemPrivate *priv = self->priv;
|
||||||
GList *apps, *l;
|
g_autolist(GAppInfo) all = NULL;
|
||||||
|
const GList *l;
|
||||||
|
|
||||||
g_hash_table_remove_all (priv->startup_wm_class_to_id);
|
g_hash_table_remove_all (priv->startup_wm_class_to_id);
|
||||||
|
|
||||||
apps = g_app_info_get_all ();
|
all = shell_app_cache_get_all (NULL);
|
||||||
for (l = apps; l != NULL; l = l->next)
|
|
||||||
|
for (l = all; l != NULL; l = l->next)
|
||||||
{
|
{
|
||||||
GAppInfo *info = l->data;
|
GAppInfo *info = l->data;
|
||||||
const char *startup_wm_class, *id, *old_id;
|
const char *startup_wm_class, *id, *old_id;
|
||||||
@ -105,8 +119,6 @@ scan_startup_wm_class_to_id (ShellAppSystem *self)
|
|||||||
g_hash_table_insert (priv->startup_wm_class_to_id,
|
g_hash_table_insert (priv->startup_wm_class_to_id,
|
||||||
g_strdup (startup_wm_class), g_strdup (id));
|
g_strdup (startup_wm_class), g_strdup (id));
|
||||||
}
|
}
|
||||||
|
|
||||||
g_list_free_full (apps, g_object_unref);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
@ -119,7 +131,7 @@ app_is_stale (ShellApp *app)
|
|||||||
if (shell_app_is_window_backed (app))
|
if (shell_app_is_window_backed (app))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
info = g_desktop_app_info_new (shell_app_get_id (app));
|
info = shell_app_cache_get_info (NULL, shell_app_get_id (app));
|
||||||
if (!info)
|
if (!info)
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
@ -156,12 +168,52 @@ stale_app_remove_func (gpointer key,
|
|||||||
return app_is_stale (value);
|
return app_is_stale (value);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static gboolean
|
||||||
installed_changed (GAppInfoMonitor *monitor,
|
rescan_icon_theme_cb (gpointer user_data)
|
||||||
gpointer user_data)
|
|
||||||
{
|
{
|
||||||
ShellAppSystem *self = user_data;
|
ShellAppSystemPrivate *priv;
|
||||||
|
ShellAppSystem *self;
|
||||||
|
StTextureCache *texture_cache;
|
||||||
|
gboolean rescanned;
|
||||||
|
|
||||||
|
self = (ShellAppSystem *) user_data;
|
||||||
|
priv = self->priv;
|
||||||
|
|
||||||
|
texture_cache = st_texture_cache_get_default ();
|
||||||
|
rescanned = st_texture_cache_rescan_icon_theme (texture_cache);
|
||||||
|
|
||||||
|
priv->n_rescan_retries++;
|
||||||
|
|
||||||
|
if (rescanned || priv->n_rescan_retries >= MAX_RESCAN_RETRIES)
|
||||||
|
{
|
||||||
|
priv->n_rescan_retries = 0;
|
||||||
|
priv->rescan_icons_timeout_id = 0;
|
||||||
|
return G_SOURCE_REMOVE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return G_SOURCE_CONTINUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
rescan_icon_theme (ShellAppSystem *self)
|
||||||
|
{
|
||||||
|
ShellAppSystemPrivate *priv = self->priv;
|
||||||
|
|
||||||
|
priv->n_rescan_retries = 0;
|
||||||
|
|
||||||
|
if (priv->rescan_icons_timeout_id > 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
priv->rescan_icons_timeout_id = g_timeout_add (RESCAN_TIMEOUT_MS,
|
||||||
|
rescan_icon_theme_cb,
|
||||||
|
self);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
installed_changed (ShellAppCache *cache,
|
||||||
|
ShellAppSystem *self)
|
||||||
|
{
|
||||||
|
rescan_icon_theme (self);
|
||||||
scan_startup_wm_class_to_id (self);
|
scan_startup_wm_class_to_id (self);
|
||||||
|
|
||||||
g_hash_table_foreach_remove (self->priv->id_to_app, stale_app_remove_func, NULL);
|
g_hash_table_foreach_remove (self->priv->id_to_app, stale_app_remove_func, NULL);
|
||||||
@ -173,7 +225,7 @@ static void
|
|||||||
shell_app_system_init (ShellAppSystem *self)
|
shell_app_system_init (ShellAppSystem *self)
|
||||||
{
|
{
|
||||||
ShellAppSystemPrivate *priv;
|
ShellAppSystemPrivate *priv;
|
||||||
GAppInfoMonitor *monitor;
|
ShellAppCache *cache;
|
||||||
|
|
||||||
self->priv = priv = shell_app_system_get_instance_private (self);
|
self->priv = priv = shell_app_system_get_instance_private (self);
|
||||||
|
|
||||||
@ -184,9 +236,9 @@ shell_app_system_init (ShellAppSystem *self)
|
|||||||
|
|
||||||
priv->startup_wm_class_to_id = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
priv->startup_wm_class_to_id = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
||||||
|
|
||||||
monitor = g_app_info_monitor_get ();
|
cache = shell_app_cache_get_default ();
|
||||||
g_signal_connect (monitor, "changed", G_CALLBACK (installed_changed), self);
|
g_signal_connect (cache, "changed", G_CALLBACK (installed_changed), self);
|
||||||
installed_changed (monitor, self);
|
installed_changed (cache, self);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -199,6 +251,9 @@ shell_app_system_finalize (GObject *object)
|
|||||||
g_hash_table_destroy (priv->id_to_app);
|
g_hash_table_destroy (priv->id_to_app);
|
||||||
g_hash_table_destroy (priv->startup_wm_class_to_id);
|
g_hash_table_destroy (priv->startup_wm_class_to_id);
|
||||||
|
|
||||||
|
if (priv->rescan_icons_timeout_id)
|
||||||
|
g_source_remove (priv->rescan_icons_timeout_id);
|
||||||
|
|
||||||
G_OBJECT_CLASS (shell_app_system_parent_class)->finalize (object);
|
G_OBJECT_CLASS (shell_app_system_parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -237,7 +292,7 @@ shell_app_system_lookup_app (ShellAppSystem *self,
|
|||||||
if (app)
|
if (app)
|
||||||
return app;
|
return app;
|
||||||
|
|
||||||
info = g_desktop_app_info_new (id);
|
info = shell_app_cache_get_info (NULL, id);
|
||||||
if (!info)
|
if (!info)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
@ -87,6 +87,8 @@ struct _ShellGlobal {
|
|||||||
/* For sound notifications */
|
/* For sound notifications */
|
||||||
ca_context *sound_context;
|
ca_context *sound_context;
|
||||||
|
|
||||||
|
GHashTable *save_ops;
|
||||||
|
|
||||||
gboolean has_modal;
|
gboolean has_modal;
|
||||||
gboolean frame_timestamps;
|
gboolean frame_timestamps;
|
||||||
gboolean frame_finish_timestamp;
|
gboolean frame_finish_timestamp;
|
||||||
@ -322,6 +324,10 @@ shell_global_init (ShellGlobal *global)
|
|||||||
NULL);
|
NULL);
|
||||||
|
|
||||||
g_strfreev (search_path);
|
g_strfreev (search_path);
|
||||||
|
|
||||||
|
global->save_ops = g_hash_table_new_full (g_file_hash,
|
||||||
|
(GEqualFunc) g_file_equal,
|
||||||
|
g_object_unref, g_object_unref);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -341,6 +347,8 @@ shell_global_finalize (GObject *object)
|
|||||||
g_free (global->imagedir);
|
g_free (global->imagedir);
|
||||||
g_free (global->userdatadir);
|
g_free (global->userdatadir);
|
||||||
|
|
||||||
|
g_hash_table_unref (global->save_ops);
|
||||||
|
|
||||||
G_OBJECT_CLASS(shell_global_parent_class)->finalize (object);
|
G_OBJECT_CLASS(shell_global_parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1757,20 +1765,133 @@ shell_global_get_session_mode (ShellGlobal *global)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
save_variant (GFile *dir,
|
delete_variant_cb (GObject *object,
|
||||||
|
GAsyncResult *result,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
ShellGlobal *global = user_data;
|
||||||
|
GError *error = NULL;
|
||||||
|
|
||||||
|
if (!g_file_delete_finish (G_FILE (object), result, &error))
|
||||||
|
{
|
||||||
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
||||||
|
{
|
||||||
|
g_warning ("Could not delete runtime/persistent state file: %s\n",
|
||||||
|
error->message);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_hash_table_remove (global->save_ops, object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
replace_contents_worker (GTask *task,
|
||||||
|
gpointer source_object,
|
||||||
|
gpointer task_data,
|
||||||
|
GCancellable *cancellable)
|
||||||
|
{
|
||||||
|
GFile *file = source_object;
|
||||||
|
GBytes *bytes = task_data;
|
||||||
|
GError *error = NULL;
|
||||||
|
const gchar *data;
|
||||||
|
gsize len;
|
||||||
|
|
||||||
|
data = g_bytes_get_data (bytes, &len);
|
||||||
|
|
||||||
|
if (!g_file_replace_contents (file, data, len, NULL, FALSE,
|
||||||
|
G_FILE_CREATE_REPLACE_DESTINATION,
|
||||||
|
NULL, cancellable, &error))
|
||||||
|
g_task_return_error (task, g_steal_pointer (&error));
|
||||||
|
else
|
||||||
|
g_task_return_boolean (task, TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
replace_contents_async (GFile *path,
|
||||||
|
GBytes *bytes,
|
||||||
|
GCancellable *cancellable,
|
||||||
|
GAsyncReadyCallback callback,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
g_autoptr(GTask) task = NULL;
|
||||||
|
|
||||||
|
g_assert (G_IS_FILE (path));
|
||||||
|
g_assert (bytes != NULL);
|
||||||
|
g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
|
||||||
|
|
||||||
|
task = g_task_new (path, cancellable, callback, user_data);
|
||||||
|
g_task_set_source_tag (task, replace_contents_async);
|
||||||
|
g_task_set_task_data (task, g_bytes_ref (bytes), (GDestroyNotify)g_bytes_unref);
|
||||||
|
g_task_run_in_thread (task, replace_contents_worker);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
replace_contents_finish (GFile *file,
|
||||||
|
GAsyncResult *result,
|
||||||
|
GError **error)
|
||||||
|
{
|
||||||
|
return g_task_propagate_boolean (G_TASK (result), error);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
replace_variant_cb (GObject *object,
|
||||||
|
GAsyncResult *result,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
ShellGlobal *global = user_data;
|
||||||
|
GError *error = NULL;
|
||||||
|
|
||||||
|
if (!replace_contents_finish (G_FILE (object), result, &error))
|
||||||
|
{
|
||||||
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
||||||
|
{
|
||||||
|
g_warning ("Could not replace runtime/persistent state file: %s\n",
|
||||||
|
error->message);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_hash_table_remove (global->save_ops, object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
save_variant (ShellGlobal *global,
|
||||||
|
GFile *dir,
|
||||||
const char *property_name,
|
const char *property_name,
|
||||||
GVariant *variant)
|
GVariant *variant)
|
||||||
{
|
{
|
||||||
GFile *path = g_file_get_child (dir, property_name);
|
GFile *path = g_file_get_child (dir, property_name);
|
||||||
|
GCancellable *cancellable;
|
||||||
|
|
||||||
|
cancellable = g_hash_table_lookup (global->save_ops, path);
|
||||||
|
g_cancellable_cancel (cancellable);
|
||||||
|
|
||||||
|
cancellable = g_cancellable_new ();
|
||||||
|
g_hash_table_insert (global->save_ops, g_object_ref (path), cancellable);
|
||||||
|
|
||||||
if (variant == NULL || g_variant_get_data (variant) == NULL)
|
if (variant == NULL || g_variant_get_data (variant) == NULL)
|
||||||
(void) g_file_delete (path, NULL, NULL);
|
{
|
||||||
|
g_file_delete_async (path, G_PRIORITY_DEFAULT, cancellable,
|
||||||
|
delete_variant_cb, global);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
gsize size = g_variant_get_size (variant);
|
g_autoptr(GBytes) bytes = NULL;
|
||||||
g_file_replace_contents (path, g_variant_get_data (variant), size,
|
|
||||||
NULL, FALSE, G_FILE_CREATE_REPLACE_DESTINATION,
|
bytes = g_bytes_new_with_free_func (g_variant_get_data (variant),
|
||||||
NULL, NULL, NULL);
|
g_variant_get_size (variant),
|
||||||
|
(GDestroyNotify)g_variant_unref,
|
||||||
|
g_variant_ref (variant));
|
||||||
|
/* g_file_replace_contents_async() can potentially fsync() from the
|
||||||
|
* calling thread when completing the asynchronous task. Instead, we
|
||||||
|
* want to force that fsync() to a thread to avoid blocking the
|
||||||
|
* compositor main loop. Using our own replace_contents_async()
|
||||||
|
* simply executes the operation synchronously from a thread.
|
||||||
|
*/
|
||||||
|
replace_contents_async (path, bytes, cancellable, replace_variant_cb, global);
|
||||||
}
|
}
|
||||||
|
|
||||||
g_object_unref (path);
|
g_object_unref (path);
|
||||||
@ -1824,7 +1945,7 @@ shell_global_set_runtime_state (ShellGlobal *global,
|
|||||||
const char *property_name,
|
const char *property_name,
|
||||||
GVariant *variant)
|
GVariant *variant)
|
||||||
{
|
{
|
||||||
save_variant (global->runtime_state_path, property_name, variant);
|
save_variant (global, global->runtime_state_path, property_name, variant);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1859,7 +1980,7 @@ shell_global_set_persistent_state (ShellGlobal *global,
|
|||||||
const char *property_name,
|
const char *property_name,
|
||||||
GVariant *variant)
|
GVariant *variant)
|
||||||
{
|
{
|
||||||
save_variant (global->userdatadir_path, property_name, variant);
|
save_variant (global, global->userdatadir_path, property_name, variant);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -142,7 +142,7 @@ libst_gir = gnome.generate_gir(libst,
|
|||||||
sources: st_gir_sources,
|
sources: st_gir_sources,
|
||||||
nsversion: '1.0',
|
nsversion: '1.0',
|
||||||
namespace: 'St',
|
namespace: 'St',
|
||||||
includes: ['Clutter-' + mutter_api_version, 'Gtk-3.0'],
|
includes: ['Clutter-' + mutter_api_version, 'Cally-' + mutter_api_version, 'Gtk-3.0'],
|
||||||
dependencies: [mutter_dep],
|
dependencies: [mutter_dep],
|
||||||
include_directories: include_directories('..'),
|
include_directories: include_directories('..'),
|
||||||
extra_args: ['-DST_COMPILATION', '--quiet'],
|
extra_args: ['-DST_COMPILATION', '--quiet'],
|
||||||
|
@ -205,7 +205,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
g_object_class_install_property (object_class,
|
g_object_class_install_property (object_class,
|
||||||
PROP_UPPER,
|
PROP_UPPER,
|
||||||
g_param_spec_double ("upper",
|
g_param_spec_double ("upper",
|
||||||
@ -215,7 +216,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
g_object_class_install_property (object_class,
|
g_object_class_install_property (object_class,
|
||||||
PROP_VALUE,
|
PROP_VALUE,
|
||||||
g_param_spec_double ("value",
|
g_param_spec_double ("value",
|
||||||
@ -225,7 +227,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
g_object_class_install_property (object_class,
|
g_object_class_install_property (object_class,
|
||||||
PROP_STEP_INC,
|
PROP_STEP_INC,
|
||||||
g_param_spec_double ("step-increment",
|
g_param_spec_double ("step-increment",
|
||||||
@ -235,7 +238,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
g_object_class_install_property (object_class,
|
g_object_class_install_property (object_class,
|
||||||
PROP_PAGE_INC,
|
PROP_PAGE_INC,
|
||||||
g_param_spec_double ("page-increment",
|
g_param_spec_double ("page-increment",
|
||||||
@ -245,7 +249,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
g_object_class_install_property (object_class,
|
g_object_class_install_property (object_class,
|
||||||
PROP_PAGE_SIZE,
|
PROP_PAGE_SIZE,
|
||||||
g_param_spec_double ("page-size",
|
g_param_spec_double ("page-size",
|
||||||
@ -255,7 +260,8 @@ st_adjustment_class_init (StAdjustmentClass *klass)
|
|||||||
G_MAXDOUBLE,
|
G_MAXDOUBLE,
|
||||||
0.0,
|
0.0,
|
||||||
ST_PARAM_READWRITE |
|
ST_PARAM_READWRITE |
|
||||||
G_PARAM_CONSTRUCT));
|
G_PARAM_CONSTRUCT |
|
||||||
|
G_PARAM_EXPLICIT_NOTIFY));
|
||||||
/**
|
/**
|
||||||
* StAdjustment::changed:
|
* StAdjustment::changed:
|
||||||
* @self: the #StAdjustment
|
* @self: the #StAdjustment
|
||||||
|
@ -177,15 +177,15 @@ st_bin_get_preferred_height (ClutterActor *self,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
st_bin_dispose (GObject *gobject)
|
st_bin_destroy (ClutterActor *actor)
|
||||||
{
|
{
|
||||||
StBinPrivate *priv = st_bin_get_instance_private (ST_BIN (gobject));
|
StBinPrivate *priv = st_bin_get_instance_private (ST_BIN (actor));
|
||||||
|
|
||||||
if (priv->child)
|
if (priv->child)
|
||||||
clutter_actor_destroy (priv->child);
|
clutter_actor_destroy (priv->child);
|
||||||
g_assert (priv->child == NULL);
|
g_assert (priv->child == NULL);
|
||||||
|
|
||||||
G_OBJECT_CLASS (st_bin_parent_class)->dispose (gobject);
|
CLUTTER_ACTOR_CLASS (st_bin_parent_class)->destroy (actor);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -315,11 +315,11 @@ st_bin_class_init (StBinClass *klass)
|
|||||||
|
|
||||||
gobject_class->set_property = st_bin_set_property;
|
gobject_class->set_property = st_bin_set_property;
|
||||||
gobject_class->get_property = st_bin_get_property;
|
gobject_class->get_property = st_bin_get_property;
|
||||||
gobject_class->dispose = st_bin_dispose;
|
|
||||||
|
|
||||||
actor_class->get_preferred_width = st_bin_get_preferred_width;
|
actor_class->get_preferred_width = st_bin_get_preferred_width;
|
||||||
actor_class->get_preferred_height = st_bin_get_preferred_height;
|
actor_class->get_preferred_height = st_bin_get_preferred_height;
|
||||||
actor_class->allocate = st_bin_allocate;
|
actor_class->allocate = st_bin_allocate;
|
||||||
|
actor_class->destroy = st_bin_destroy;
|
||||||
|
|
||||||
widget_class->popup_menu = st_bin_popup_menu;
|
widget_class->popup_menu = st_bin_popup_menu;
|
||||||
widget_class->navigate_focus = st_bin_navigate_focus;
|
widget_class->navigate_focus = st_bin_navigate_focus;
|
||||||
|
@ -90,7 +90,7 @@ adjustment_value_notify_cb (StAdjustment *adjustment,
|
|||||||
GParamSpec *pspec,
|
GParamSpec *pspec,
|
||||||
StBoxLayout *box)
|
StBoxLayout *box)
|
||||||
{
|
{
|
||||||
clutter_actor_queue_redraw (CLUTTER_ACTOR (box));
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (box));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -490,7 +490,7 @@ st_box_layout_get_paint_volume (ClutterActor *actor,
|
|||||||
ClutterPaintVolume *volume)
|
ClutterPaintVolume *volume)
|
||||||
{
|
{
|
||||||
StBoxLayout *self = ST_BOX_LAYOUT (actor);
|
StBoxLayout *self = ST_BOX_LAYOUT (actor);
|
||||||
gdouble x, y;
|
gdouble x, y, lower, upper;
|
||||||
StBoxLayoutPrivate *priv = self->priv;
|
StBoxLayoutPrivate *priv = self->priv;
|
||||||
StThemeNode *theme_node = st_widget_get_theme_node (ST_WIDGET (actor));
|
StThemeNode *theme_node = st_widget_get_theme_node (ST_WIDGET (actor));
|
||||||
ClutterActorBox allocation_box;
|
ClutterActorBox allocation_box;
|
||||||
@ -505,13 +505,42 @@ st_box_layout_get_paint_volume (ClutterActor *actor,
|
|||||||
* our paint volume on that. */
|
* our paint volume on that. */
|
||||||
if (priv->hadjustment || priv->vadjustment)
|
if (priv->hadjustment || priv->vadjustment)
|
||||||
{
|
{
|
||||||
|
gdouble width, height;
|
||||||
|
|
||||||
clutter_actor_get_allocation_box (actor, &allocation_box);
|
clutter_actor_get_allocation_box (actor, &allocation_box);
|
||||||
st_theme_node_get_content_box (theme_node, &allocation_box, &content_box);
|
st_theme_node_get_content_box (theme_node, &allocation_box, &content_box);
|
||||||
origin.x = content_box.x1 - allocation_box.x1;
|
origin.x = content_box.x1 - allocation_box.x1;
|
||||||
origin.y = content_box.y1 - allocation_box.y2;
|
origin.y = content_box.y1 - allocation_box.y2;
|
||||||
origin.z = 0.f;
|
origin.z = 0.f;
|
||||||
clutter_paint_volume_set_width (volume, content_box.x2 - content_box.x1);
|
|
||||||
clutter_paint_volume_set_height (volume, content_box.y2 - content_box.y1);
|
if (priv->hadjustment)
|
||||||
|
{
|
||||||
|
g_object_get (priv->hadjustment,
|
||||||
|
"lower", &lower,
|
||||||
|
"upper", &upper,
|
||||||
|
NULL);
|
||||||
|
width = upper - lower;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
width = content_box.x2 - content_box.x1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (priv->vadjustment)
|
||||||
|
{
|
||||||
|
g_object_get (priv->vadjustment,
|
||||||
|
"lower", &lower,
|
||||||
|
"upper", &upper,
|
||||||
|
NULL);
|
||||||
|
height = upper - lower;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
height = content_box.y2 - content_box.y1;
|
||||||
|
}
|
||||||
|
|
||||||
|
clutter_paint_volume_set_width (volume, width);
|
||||||
|
clutter_paint_volume_set_height (volume, height);
|
||||||
}
|
}
|
||||||
else if (!CLUTTER_ACTOR_CLASS (st_box_layout_parent_class)->get_paint_volume (actor, volume))
|
else if (!CLUTTER_ACTOR_CLASS (st_box_layout_parent_class)->get_paint_volume (actor, volume))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
@ -248,6 +248,7 @@ st_button_touch_event (ClutterActor *actor,
|
|||||||
if (event->type == CLUTTER_TOUCH_BEGIN && !priv->press_sequence)
|
if (event->type == CLUTTER_TOUCH_BEGIN && !priv->press_sequence)
|
||||||
{
|
{
|
||||||
clutter_input_device_sequence_grab (device, sequence, actor);
|
clutter_input_device_sequence_grab (device, sequence, actor);
|
||||||
|
if (!clutter_event_is_pointer_emulated ((ClutterEvent*) event))
|
||||||
st_button_press (button, device, 0, sequence);
|
st_button_press (button, device, 0, sequence);
|
||||||
return CLUTTER_EVENT_STOP;
|
return CLUTTER_EVENT_STOP;
|
||||||
}
|
}
|
||||||
@ -255,7 +256,9 @@ st_button_touch_event (ClutterActor *actor,
|
|||||||
priv->device == device &&
|
priv->device == device &&
|
||||||
priv->press_sequence == sequence)
|
priv->press_sequence == sequence)
|
||||||
{
|
{
|
||||||
|
if (!clutter_event_is_pointer_emulated ((ClutterEvent*) event))
|
||||||
st_button_release (button, device, mask, 0, sequence);
|
st_button_release (button, device, mask, 0, sequence);
|
||||||
|
|
||||||
clutter_input_device_sequence_ungrab (device, sequence);
|
clutter_input_device_sequence_ungrab (device, sequence);
|
||||||
return CLUTTER_EVENT_STOP;
|
return CLUTTER_EVENT_STOP;
|
||||||
}
|
}
|
||||||
|
@ -906,6 +906,13 @@ st_entry_unmap (ClutterActor *actor)
|
|||||||
CLUTTER_ACTOR_CLASS (st_entry_parent_class)->unmap (actor);
|
CLUTTER_ACTOR_CLASS (st_entry_parent_class)->unmap (actor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
st_entry_get_paint_volume (ClutterActor *actor,
|
||||||
|
ClutterPaintVolume *volume)
|
||||||
|
{
|
||||||
|
return clutter_paint_volume_set_from_allocation (volume, actor);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
st_entry_class_init (StEntryClass *klass)
|
st_entry_class_init (StEntryClass *klass)
|
||||||
{
|
{
|
||||||
@ -923,6 +930,7 @@ st_entry_class_init (StEntryClass *klass)
|
|||||||
actor_class->allocate = st_entry_allocate;
|
actor_class->allocate = st_entry_allocate;
|
||||||
actor_class->paint = st_entry_paint;
|
actor_class->paint = st_entry_paint;
|
||||||
actor_class->unmap = st_entry_unmap;
|
actor_class->unmap = st_entry_unmap;
|
||||||
|
actor_class->get_paint_volume = st_entry_get_paint_volume;
|
||||||
|
|
||||||
actor_class->key_press_event = st_entry_key_press_event;
|
actor_class->key_press_event = st_entry_key_press_event;
|
||||||
actor_class->key_focus_in = st_entry_key_focus_in;
|
actor_class->key_focus_in = st_entry_key_focus_in;
|
||||||
@ -1287,9 +1295,9 @@ st_entry_get_input_hints (StEntry *entry)
|
|||||||
return clutter_text_get_input_hints (CLUTTER_TEXT (priv->entry));
|
return clutter_text_get_input_hints (CLUTTER_TEXT (priv->entry));
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static void
|
||||||
_st_entry_icon_press_cb (ClutterActor *actor,
|
_st_entry_icon_clicked_cb (ClutterClickAction *action,
|
||||||
ClutterButtonEvent *event,
|
ClutterActor *actor,
|
||||||
StEntry *entry)
|
StEntry *entry)
|
||||||
{
|
{
|
||||||
StEntryPrivate *priv = ST_ENTRY_PRIV (entry);
|
StEntryPrivate *priv = ST_ENTRY_PRIV (entry);
|
||||||
@ -1298,8 +1306,6 @@ _st_entry_icon_press_cb (ClutterActor *actor,
|
|||||||
g_signal_emit (entry, entry_signals[PRIMARY_ICON_CLICKED], 0);
|
g_signal_emit (entry, entry_signals[PRIMARY_ICON_CLICKED], 0);
|
||||||
else
|
else
|
||||||
g_signal_emit (entry, entry_signals[SECONDARY_ICON_CLICKED], 0);
|
g_signal_emit (entry, entry_signals[SECONDARY_ICON_CLICKED], 0);
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1309,21 +1315,24 @@ _st_entry_set_icon (StEntry *entry,
|
|||||||
{
|
{
|
||||||
if (*icon)
|
if (*icon)
|
||||||
{
|
{
|
||||||
g_signal_handlers_disconnect_by_func (*icon,
|
clutter_actor_remove_action_by_name (*icon, "entry-icon-action");
|
||||||
_st_entry_icon_press_cb,
|
|
||||||
entry);
|
|
||||||
clutter_actor_remove_child (CLUTTER_ACTOR (entry), *icon);
|
clutter_actor_remove_child (CLUTTER_ACTOR (entry), *icon);
|
||||||
*icon = NULL;
|
*icon = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (new_icon)
|
if (new_icon)
|
||||||
{
|
{
|
||||||
|
ClutterAction *action;
|
||||||
|
|
||||||
*icon = g_object_ref (new_icon);
|
*icon = g_object_ref (new_icon);
|
||||||
|
|
||||||
clutter_actor_set_reactive (*icon, TRUE);
|
clutter_actor_set_reactive (*icon, TRUE);
|
||||||
clutter_actor_add_child (CLUTTER_ACTOR (entry), *icon);
|
clutter_actor_add_child (CLUTTER_ACTOR (entry), *icon);
|
||||||
g_signal_connect (*icon, "button-release-event",
|
|
||||||
G_CALLBACK (_st_entry_icon_press_cb), entry);
|
action = clutter_click_action_new ();
|
||||||
|
clutter_actor_add_action_with_name (*icon, "entry-icon-action", action);
|
||||||
|
g_signal_connect (action, "clicked",
|
||||||
|
G_CALLBACK (_st_entry_icon_clicked_cb), entry);
|
||||||
}
|
}
|
||||||
|
|
||||||
clutter_actor_queue_relayout (CLUTTER_ACTOR (entry));
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (entry));
|
||||||
|
@ -180,6 +180,7 @@ st_label_dispose (GObject *object)
|
|||||||
{
|
{
|
||||||
StLabelPrivate *priv = ST_LABEL (object)->priv;
|
StLabelPrivate *priv = ST_LABEL (object)->priv;
|
||||||
|
|
||||||
|
priv->label = NULL;
|
||||||
g_clear_pointer (&priv->text_shadow_pipeline, cogl_object_unref);
|
g_clear_pointer (&priv->text_shadow_pipeline, cogl_object_unref);
|
||||||
|
|
||||||
G_OBJECT_CLASS (st_label_parent_class)->dispose (object);
|
G_OBJECT_CLASS (st_label_parent_class)->dispose (object);
|
||||||
|
@ -304,6 +304,13 @@ st_scroll_view_pick (ClutterActor *actor,
|
|||||||
clutter_actor_paint (priv->vscroll);
|
clutter_actor_paint (priv->vscroll);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
st_scroll_view_get_paint_volume (ClutterActor *actor,
|
||||||
|
ClutterPaintVolume *volume)
|
||||||
|
{
|
||||||
|
return clutter_paint_volume_set_from_allocation (volume, actor);
|
||||||
|
}
|
||||||
|
|
||||||
static double
|
static double
|
||||||
get_scrollbar_width (StScrollView *scroll,
|
get_scrollbar_width (StScrollView *scroll,
|
||||||
gfloat for_height)
|
gfloat for_height)
|
||||||
@ -793,6 +800,7 @@ st_scroll_view_class_init (StScrollViewClass *klass)
|
|||||||
|
|
||||||
actor_class->paint = st_scroll_view_paint;
|
actor_class->paint = st_scroll_view_paint;
|
||||||
actor_class->pick = st_scroll_view_pick;
|
actor_class->pick = st_scroll_view_pick;
|
||||||
|
actor_class->get_paint_volume = st_scroll_view_get_paint_volume;
|
||||||
actor_class->get_preferred_width = st_scroll_view_get_preferred_width;
|
actor_class->get_preferred_width = st_scroll_view_get_preferred_width;
|
||||||
actor_class->get_preferred_height = st_scroll_view_get_preferred_height;
|
actor_class->get_preferred_height = st_scroll_view_get_preferred_height;
|
||||||
actor_class->allocate = st_scroll_view_allocate;
|
actor_class->allocate = st_scroll_view_allocate;
|
||||||
|
@ -37,6 +37,7 @@ struct _StTextureCachePrivate
|
|||||||
|
|
||||||
/* Things that were loaded with a cache policy != NONE */
|
/* Things that were loaded with a cache policy != NONE */
|
||||||
GHashTable *keyed_cache; /* char * -> CoglTexture* */
|
GHashTable *keyed_cache; /* char * -> CoglTexture* */
|
||||||
|
GHashTable *keyed_surface_cache; /* char * -> cairo_surface_t* */
|
||||||
|
|
||||||
/* Presently this is used to de-duplicate requests for GIcons and async URIs. */
|
/* Presently this is used to de-duplicate requests for GIcons and async URIs. */
|
||||||
GHashTable *outstanding_requests; /* char * -> AsyncTextureLoadData * */
|
GHashTable *outstanding_requests; /* char * -> AsyncTextureLoadData * */
|
||||||
@ -145,6 +146,10 @@ st_texture_cache_init (StTextureCache *self)
|
|||||||
|
|
||||||
self->priv->keyed_cache = g_hash_table_new_full (g_str_hash, g_str_equal,
|
self->priv->keyed_cache = g_hash_table_new_full (g_str_hash, g_str_equal,
|
||||||
g_free, cogl_object_unref);
|
g_free, cogl_object_unref);
|
||||||
|
self->priv->keyed_surface_cache = g_hash_table_new_full (g_str_hash,
|
||||||
|
g_str_equal,
|
||||||
|
g_free,
|
||||||
|
(GDestroyNotify) cairo_surface_destroy);
|
||||||
self->priv->outstanding_requests = g_hash_table_new_full (g_str_hash, g_str_equal,
|
self->priv->outstanding_requests = g_hash_table_new_full (g_str_hash, g_str_equal,
|
||||||
g_free, NULL);
|
g_free, NULL);
|
||||||
self->priv->file_monitors = g_hash_table_new_full (g_file_hash, (GEqualFunc) g_file_equal,
|
self->priv->file_monitors = g_hash_table_new_full (g_file_hash, (GEqualFunc) g_file_equal,
|
||||||
@ -166,6 +171,7 @@ st_texture_cache_dispose (GObject *object)
|
|||||||
}
|
}
|
||||||
|
|
||||||
g_clear_pointer (&self->priv->keyed_cache, g_hash_table_destroy);
|
g_clear_pointer (&self->priv->keyed_cache, g_hash_table_destroy);
|
||||||
|
g_clear_pointer (&self->priv->keyed_surface_cache, g_hash_table_destroy);
|
||||||
g_clear_pointer (&self->priv->outstanding_requests, g_hash_table_destroy);
|
g_clear_pointer (&self->priv->outstanding_requests, g_hash_table_destroy);
|
||||||
g_clear_pointer (&self->priv->file_monitors, g_hash_table_destroy);
|
g_clear_pointer (&self->priv->file_monitors, g_hash_table_destroy);
|
||||||
|
|
||||||
@ -520,6 +526,8 @@ finish_texture_load (AsyncTextureLoadData *data,
|
|||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
texdata = pixbuf_to_cogl_texture (pixbuf);
|
texdata = pixbuf_to_cogl_texture (pixbuf);
|
||||||
|
if (!texdata)
|
||||||
|
goto out;
|
||||||
|
|
||||||
if (data->policy != ST_TEXTURE_CACHE_POLICY_NONE)
|
if (data->policy != ST_TEXTURE_CACHE_POLICY_NONE)
|
||||||
{
|
{
|
||||||
@ -772,13 +780,13 @@ st_texture_cache_load (StTextureCache *cache,
|
|||||||
if (!texture)
|
if (!texture)
|
||||||
{
|
{
|
||||||
texture = load (cache, key, data, error);
|
texture = load (cache, key, data, error);
|
||||||
if (texture)
|
if (texture && policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
||||||
g_hash_table_insert (cache->priv->keyed_cache, g_strdup (key), texture);
|
g_hash_table_insert (cache->priv->keyed_cache, g_strdup (key), texture);
|
||||||
else
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (texture && policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
||||||
cogl_object_ref (texture);
|
cogl_object_ref (texture);
|
||||||
|
|
||||||
return texture;
|
return texture;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -976,7 +984,7 @@ file_changed_cb (GFileMonitor *monitor,
|
|||||||
char *key;
|
char *key;
|
||||||
guint file_hash;
|
guint file_hash;
|
||||||
|
|
||||||
if (event_type != G_FILE_MONITOR_EVENT_CHANGED)
|
if (event_type != G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
file_hash = g_file_hash (file);
|
file_hash = g_file_hash (file);
|
||||||
@ -986,7 +994,7 @@ file_changed_cb (GFileMonitor *monitor,
|
|||||||
g_free (key);
|
g_free (key);
|
||||||
|
|
||||||
key = g_strdup_printf (CACHE_PREFIX_FILE_FOR_CAIRO "%u", file_hash);
|
key = g_strdup_printf (CACHE_PREFIX_FILE_FOR_CAIRO "%u", file_hash);
|
||||||
g_hash_table_remove (cache->priv->keyed_cache, key);
|
g_hash_table_remove (cache->priv->keyed_surface_cache, key);
|
||||||
g_free (key);
|
g_free (key);
|
||||||
|
|
||||||
g_signal_emit (cache, signals[TEXTURE_FILE_CHANGED], 0, file);
|
g_signal_emit (cache, signals[TEXTURE_FILE_CHANGED], 0, file);
|
||||||
@ -1273,6 +1281,9 @@ st_texture_cache_load_file_sync_to_cogl_texture (StTextureCache *cache,
|
|||||||
texdata = pixbuf_to_cogl_texture (pixbuf);
|
texdata = pixbuf_to_cogl_texture (pixbuf);
|
||||||
g_object_unref (pixbuf);
|
g_object_unref (pixbuf);
|
||||||
|
|
||||||
|
if (!texdata)
|
||||||
|
goto out;
|
||||||
|
|
||||||
if (policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
if (policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
||||||
{
|
{
|
||||||
cogl_object_ref (texdata);
|
cogl_object_ref (texdata);
|
||||||
@ -1304,7 +1315,7 @@ st_texture_cache_load_file_sync_to_cairo_surface (StTextureCache *cache,
|
|||||||
|
|
||||||
key = g_strdup_printf (CACHE_PREFIX_FILE_FOR_CAIRO "%u", g_file_hash (file));
|
key = g_strdup_printf (CACHE_PREFIX_FILE_FOR_CAIRO "%u", g_file_hash (file));
|
||||||
|
|
||||||
surface = g_hash_table_lookup (cache->priv->keyed_cache, key);
|
surface = g_hash_table_lookup (cache->priv->keyed_surface_cache, key);
|
||||||
|
|
||||||
if (surface == NULL)
|
if (surface == NULL)
|
||||||
{
|
{
|
||||||
@ -1318,7 +1329,8 @@ st_texture_cache_load_file_sync_to_cairo_surface (StTextureCache *cache,
|
|||||||
if (policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
if (policy == ST_TEXTURE_CACHE_POLICY_FOREVER)
|
||||||
{
|
{
|
||||||
cairo_surface_reference (surface);
|
cairo_surface_reference (surface);
|
||||||
g_hash_table_insert (cache->priv->keyed_cache, g_strdup (key), surface);
|
g_hash_table_insert (cache->priv->keyed_surface_cache,
|
||||||
|
g_strdup (key), surface);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1413,3 +1425,11 @@ st_texture_cache_get_default (void)
|
|||||||
instance = g_object_new (ST_TYPE_TEXTURE_CACHE, NULL);
|
instance = g_object_new (ST_TYPE_TEXTURE_CACHE, NULL);
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
st_texture_cache_rescan_icon_theme (StTextureCache *cache)
|
||||||
|
{
|
||||||
|
StTextureCachePrivate *priv = cache->priv;
|
||||||
|
|
||||||
|
return gtk_icon_theme_rescan_if_needed (priv->icon_theme);
|
||||||
|
}
|
||||||
|
@ -106,4 +106,6 @@ CoglTexture * st_texture_cache_load (StTextureCache *cache,
|
|||||||
void *data,
|
void *data,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
|
||||||
|
gboolean st_texture_cache_rescan_icon_theme (StTextureCache *cache);
|
||||||
|
|
||||||
#endif /* __ST_TEXTURE_CACHE_H__ */
|
#endif /* __ST_TEXTURE_CACHE_H__ */
|
||||||
|
@ -229,9 +229,9 @@ unpremultiply (ClutterColor *color)
|
|||||||
{
|
{
|
||||||
if (color->alpha != 0)
|
if (color->alpha != 0)
|
||||||
{
|
{
|
||||||
color->red = (color->red * 255 + 127) / color->alpha;
|
color->red = MIN((color->red * 255 + 127) / color->alpha, 255);
|
||||||
color->green = (color->green * 255 + 127) / color->alpha;
|
color->green = MIN((color->green * 255 + 127) / color->alpha, 255);
|
||||||
color->blue = (color->blue * 255 + 127) / color->alpha;
|
color->blue = MIN((color->blue * 255 + 127) / color->alpha, 255);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -402,7 +402,7 @@ st_theme_node_lookup_corner (StThemeNode *node,
|
|||||||
return COGL_INVALID_HANDLE;
|
return COGL_INVALID_HANDLE;
|
||||||
|
|
||||||
key = corner_to_string (&corner);
|
key = corner_to_string (&corner);
|
||||||
texture = st_texture_cache_load (cache, key, ST_TEXTURE_CACHE_POLICY_NONE, load_corner, &corner, NULL);
|
texture = st_texture_cache_load (cache, key, ST_TEXTURE_CACHE_POLICY_FOREVER, load_corner, &corner, NULL);
|
||||||
|
|
||||||
if (texture)
|
if (texture)
|
||||||
{
|
{
|
||||||
@ -1414,6 +1414,32 @@ st_theme_node_load_background_image (StThemeNode *node)
|
|||||||
return node->background_texture != COGL_INVALID_HANDLE;
|
return node->background_texture != COGL_INVALID_HANDLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
st_theme_node_invalidate_resources_for_file (StThemeNode *node,
|
||||||
|
GFile *file)
|
||||||
|
{
|
||||||
|
StBorderImage *border_image;
|
||||||
|
gboolean changed = FALSE;
|
||||||
|
GFile *theme_file;
|
||||||
|
|
||||||
|
theme_file = st_theme_node_get_background_image (node);
|
||||||
|
if ((theme_file != NULL) && g_file_equal (theme_file, file))
|
||||||
|
{
|
||||||
|
st_theme_node_invalidate_background_image (node);
|
||||||
|
changed = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
border_image = st_theme_node_get_border_image (node);
|
||||||
|
theme_file = border_image ? st_border_image_get_file (border_image) : NULL;
|
||||||
|
if ((theme_file != NULL) && g_file_equal (theme_file, file))
|
||||||
|
{
|
||||||
|
st_theme_node_invalidate_border_image (node);
|
||||||
|
changed = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return changed;
|
||||||
|
}
|
||||||
|
|
||||||
static void st_theme_node_prerender_shadow (StThemeNodePaintState *state);
|
static void st_theme_node_prerender_shadow (StThemeNodePaintState *state);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -2751,3 +2777,17 @@ st_theme_node_paint_state_invalidate (StThemeNodePaintState *state)
|
|||||||
state->alloc_width = 0;
|
state->alloc_width = 0;
|
||||||
state->alloc_height = 0;
|
state->alloc_height = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
st_theme_node_paint_state_invalidate_for_file (StThemeNodePaintState *state,
|
||||||
|
GFile *file)
|
||||||
|
{
|
||||||
|
if (state->node != NULL &&
|
||||||
|
st_theme_node_invalidate_resources_for_file (state->node, file))
|
||||||
|
{
|
||||||
|
st_theme_node_paint_state_invalidate (state);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
@ -287,6 +287,9 @@ void st_theme_node_paint_state_free (StThemeNodePaintState *state);
|
|||||||
void st_theme_node_paint_state_copy (StThemeNodePaintState *state,
|
void st_theme_node_paint_state_copy (StThemeNodePaintState *state,
|
||||||
StThemeNodePaintState *other);
|
StThemeNodePaintState *other);
|
||||||
void st_theme_node_paint_state_invalidate (StThemeNodePaintState *state);
|
void st_theme_node_paint_state_invalidate (StThemeNodePaintState *state);
|
||||||
|
gboolean st_theme_node_paint_state_invalidate_for_file (StThemeNodePaintState *state,
|
||||||
|
GFile *file);
|
||||||
|
|
||||||
void st_theme_node_paint_state_set_node (StThemeNodePaintState *state,
|
void st_theme_node_paint_state_set_node (StThemeNodePaintState *state,
|
||||||
StThemeNode *node);
|
StThemeNode *node);
|
||||||
|
|
||||||
|
@ -289,44 +289,17 @@ st_widget_texture_cache_changed (StTextureCache *cache,
|
|||||||
{
|
{
|
||||||
StWidget *actor = ST_WIDGET (user_data);
|
StWidget *actor = ST_WIDGET (user_data);
|
||||||
StWidgetPrivate *priv = st_widget_get_instance_private (actor);
|
StWidgetPrivate *priv = st_widget_get_instance_private (actor);
|
||||||
StThemeNode *node = priv->theme_node;
|
|
||||||
StBorderImage *border_image;
|
|
||||||
gboolean changed = FALSE;
|
gboolean changed = FALSE;
|
||||||
GFile *theme_file;
|
int i;
|
||||||
|
|
||||||
if (node == NULL)
|
for (i = 0; i < G_N_ELEMENTS (priv->paint_states); i++)
|
||||||
return;
|
|
||||||
|
|
||||||
theme_file = st_theme_node_get_background_image (node);
|
|
||||||
if ((theme_file != NULL) && g_file_equal (theme_file, file))
|
|
||||||
{
|
{
|
||||||
st_theme_node_invalidate_background_image (node);
|
StThemeNodePaintState *paint_state = &priv->paint_states[i];
|
||||||
changed = TRUE;
|
changed |= st_theme_node_paint_state_invalidate_for_file (paint_state, file);
|
||||||
}
|
}
|
||||||
|
|
||||||
border_image = st_theme_node_get_border_image (node);
|
if (changed && clutter_actor_is_mapped (CLUTTER_ACTOR (actor)))
|
||||||
theme_file = border_image ? st_border_image_get_file (border_image) : NULL;
|
|
||||||
if ((theme_file != NULL) && g_file_equal (theme_file, file))
|
|
||||||
{
|
|
||||||
st_theme_node_invalidate_border_image (node);
|
|
||||||
changed = TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (changed)
|
|
||||||
{
|
|
||||||
/* If we prerender the background / border, we need to update
|
|
||||||
* the paint state. We should probably implement a method to
|
|
||||||
* the theme node to determine this, but for now, just wipe
|
|
||||||
* the entire paint state.
|
|
||||||
*
|
|
||||||
* Use the existing state instead of a new one because it's
|
|
||||||
* assumed the rest of the state will stay the same.
|
|
||||||
*/
|
|
||||||
st_theme_node_paint_state_invalidate (current_paint_state (actor));
|
|
||||||
|
|
||||||
if (clutter_actor_is_mapped (CLUTTER_ACTOR (actor)))
|
|
||||||
clutter_actor_queue_redraw (CLUTTER_ACTOR (actor));
|
clutter_actor_queue_redraw (CLUTTER_ACTOR (actor));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
Loading…
Reference in New Issue
Block a user