Compare commits
	
		
			1 Commits
		
	
	
		
			wip/ewlsh/
			...
			wip/carlos
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | cdd513aef4 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -60,6 +60,7 @@ src/calendar-server/evolution-calendar.desktop | ||||
| src/calendar-server/org.gnome.Shell.CalendarServer.service | ||||
| src/gnome-shell | ||||
| src/gnome-shell-calendar-server | ||||
| src/gnome-shell-extension-prefs | ||||
| src/gnome-shell-extension-tool | ||||
| src/gnome-shell-hotplug-sniffer | ||||
| src/gnome-shell-perf-helper | ||||
|   | ||||
| @@ -1,13 +1,9 @@ | ||||
| include: 'https://gitlab.gnome.org/GNOME/citemplates/raw/master/flatpak/flatpak_ci_initiative.yml' | ||||
|  | ||||
| stages: | ||||
|  - review | ||||
|  - build | ||||
|  - test | ||||
|  - deploy | ||||
|  | ||||
| variables: | ||||
|     BUNDLE: "extensions-git.flatpak" | ||||
|     JS_LOG: "js-report.txt" | ||||
|     POT_LOG: "pot-update.txt" | ||||
|  | ||||
| @@ -18,7 +14,7 @@ variables: | ||||
|         - merge_requests | ||||
|  | ||||
| check_commit_log: | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v4 | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v3 | ||||
|     stage: review | ||||
|     variables: | ||||
|         GIT_DEPTH: "100" | ||||
| @@ -28,10 +24,10 @@ check_commit_log: | ||||
|         - merge_requests | ||||
|  | ||||
| js_check: | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v1 | ||||
|     stage: review | ||||
|     script: | ||||
|         - find js -name '*.js' -exec js68 -c -s '{}' ';' 2>&1 | tee $JS_LOG | ||||
|         - find js -name '*.js' -exec js60 -c -s '{}' ';' 2>&1 | tee $JS_LOG | ||||
|         - (! grep -q . $JS_LOG) | ||||
|     <<: *only_default | ||||
|     artifacts: | ||||
| @@ -40,7 +36,7 @@ js_check: | ||||
|         when: on_failure | ||||
|  | ||||
| eslint: | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v1 | ||||
|     stage: review | ||||
|     script: | ||||
|         - ./.gitlab-ci/run-eslint.sh | ||||
| @@ -50,22 +46,8 @@ eslint: | ||||
|             - reports | ||||
|         when: always | ||||
|  | ||||
| potfile_check: | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 | ||||
|     stage: review | ||||
|     script: | ||||
|         - ./.gitlab-ci/check-potfiles.sh | ||||
|     <<: *only_default | ||||
|  | ||||
| no_template_check: | ||||
|     image: registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 | ||||
|     stage: review | ||||
|     script: | ||||
|         - ./.gitlab-ci/check-template-strings.sh | ||||
|     <<: *only_default | ||||
|  | ||||
| build: | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v4 | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v3 | ||||
|     stage: build | ||||
|     before_script: | ||||
|         - .gitlab-ci/checkout-mutter.sh | ||||
| @@ -83,7 +65,7 @@ build: | ||||
|             - build | ||||
|  | ||||
| test: | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v4 | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v3 | ||||
|     stage: test | ||||
|     variables: | ||||
|         XDG_RUNTIME_DIR: "$CI_PROJECT_DIR/runtime-dir" | ||||
| @@ -100,7 +82,7 @@ test: | ||||
|         when: on_failure | ||||
|  | ||||
| test-pot: | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v4 | ||||
|     image: registry.gitlab.gnome.org/gnome/mutter/master:v3 | ||||
|     stage: test | ||||
|     before_script: | ||||
|         - ninja -C mutter/build install | ||||
| @@ -114,20 +96,3 @@ test-pot: | ||||
|           ' | tee $POT_LOG | ||||
|         - (! grep -q . $POT_LOG) | ||||
|     <<: *only_default | ||||
|  | ||||
| flatpak: | ||||
|     stage: build | ||||
|     variables: | ||||
|         SUBPROJECT: "subprojects/extensions-app" | ||||
|         # Your manifest path | ||||
|         MANIFEST_PATH: "$SUBPROJECT/build-aux/flatpak/org.gnome.Extensions.json" | ||||
|         RUNTIME_REPO: "https://nightly.gnome.org/gnome-nightly.flatpakrepo" | ||||
|         FLATPAK_MODULE: "gnome-extensions-app" | ||||
|         APP_ID: "org.gnome.Extensions" | ||||
|     extends: .flatpak | ||||
|     <<: *only_default | ||||
|  | ||||
| nightly: | ||||
|   extends: '.publish_nightly' | ||||
|   variables: | ||||
|     BUNDLES: '$BUNDLE' | ||||
|   | ||||
| @@ -1,24 +0,0 @@ | ||||
| # Rebuild and push with | ||||
| # | ||||
| #     cd .gitlab-ci/ | ||||
| #     podman build --format docker --no-cache -t registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 . | ||||
| #     podman push registry.gitlab.gnome.org/gnome/gnome-shell/extension-ci:v2 | ||||
| # | ||||
|  | ||||
| FROM registry.fedoraproject.org/fedora:32 | ||||
|  | ||||
| RUN dnf -y update && dnf -y upgrade && \ | ||||
|     dnf install -y 'dnf-command(copr)' git && \ | ||||
|  | ||||
|     # For syntax checks with `find . -name '*.js' -exec js68 -c -s '{}' ';'` | ||||
|     dnf install -y findutils mozjs68-devel && \ | ||||
|  | ||||
|     # For static analysis with eslint | ||||
|     dnf install -y nodejs && \ | ||||
|     npm install -g eslint && \ | ||||
|  | ||||
|     # Shameless plug for my own tooling; useful for generating zip | ||||
|     dnf copr enable -y fmuellner/gnome-shell-ci && \ | ||||
|     dnf install -y gnome-extensions-tool meson && \ | ||||
|  | ||||
|     dnf clean all | ||||
							
								
								
									
										18
									
								
								.gitlab-ci/Dockerfile.extension-ci
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| FROM registry.fedoraproject.org/fedora:latest | ||||
|  | ||||
| RUN dnf -y update && dnf -y upgrade && \ | ||||
|     dnf install -y 'dnf-command(copr)' git && \ | ||||
|  | ||||
|     # For syntax checks with `find . -name '*.js' -exec js60 -c -s '{}' ';'` | ||||
|     dnf install -y findutils mozjs60-devel && \ | ||||
|  | ||||
|     # For static analysis with eslint | ||||
|     dnf install -y nodejs && \ | ||||
|     npm install -g eslint && \ | ||||
|  | ||||
|     # Shameless plug for my own tooling; useful for generating zip | ||||
|     dnf copr enable -y fmuellner/gnome-shell-ci && \ | ||||
|     dnf install -y gnome-extensions-tool meson && \ | ||||
|  | ||||
|     dnf clean all && \ | ||||
|     rm -rf /var/cache/dnf | ||||
| @@ -19,7 +19,7 @@ fi | ||||
| function commit_message_has_url() { | ||||
|   commit=$1 | ||||
|   commit_message=$(git show -s --format='format:%b' $commit) | ||||
|   echo "$commit_message" | grep -qe "\($CI_MERGE_REQUEST_PROJECT_URL/\(-/\)\?\(issues\|merge_requests\)/[0-9]\+\|https://bugzilla.gnome.org/show_bug.cgi?id=[0-9]\+\)" | ||||
|   echo "$commit_message" | grep -qe "\($CI_MERGE_REQUEST_PROJECT_URL/\(issues\|merge_requests\)/[0-9]\+\|https://bugzilla.gnome.org/show_bug.cgi?id=[0-9]\+\)" | ||||
|   return $? | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,31 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| srcdirs="js src subprojects/extensions-tool" | ||||
| globs=('*.js' '*.c') | ||||
|  | ||||
| # find source files that contain gettext keywords | ||||
| files=$(grep -lR ${globs[@]/#/--include=} '\(gettext\|[^I_)]_\)(' $srcdirs) | ||||
|  | ||||
| # filter out excluded files | ||||
| if [ -f po/POTFILES.skip ]; then | ||||
|   files=$(for f in $files; do ! grep -q ^$f po/POTFILES.skip && echo $f; done) | ||||
| fi | ||||
|  | ||||
| # find those that aren't listed in POTFILES.in | ||||
| missing=$(for f in $files; do ! grep -q ^$f po/POTFILES.in && echo $f; done) | ||||
|  | ||||
| if [ ${#missing} -eq 0 ]; then | ||||
|   exit 0 | ||||
| fi | ||||
|  | ||||
| cat >&2 <<EOT | ||||
|  | ||||
| The following files are missing from po/POTFILES.po: | ||||
|  | ||||
| EOT | ||||
| for f in $missing; do | ||||
|   echo "  $f" >&2 | ||||
| done | ||||
| echo >&2 | ||||
|  | ||||
| exit 1 | ||||
| @@ -1,23 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| # find files from POTFILES.in that use js template strings | ||||
| baddies=$(grep -l '${' $(grep ^js po/POTFILES.in)) | ||||
|  | ||||
| if [ ${#baddies} -eq 0 ]; then | ||||
|   exit 0 | ||||
| fi | ||||
|  | ||||
| cat >&2 <<EOT | ||||
|  | ||||
| xgettext cannot handle template strings properly, so we ban their use | ||||
| in files with translatable strings. | ||||
|  | ||||
| The following files are listed in po/POTFILES.in and use template strings: | ||||
|  | ||||
| EOT | ||||
| for f in $baddies; do | ||||
|   echo "  $f" >&2 | ||||
| done | ||||
| echo >&2 | ||||
|  | ||||
| exit 1 | ||||
| @@ -18,14 +18,12 @@ run_eslint() { | ||||
|   local extra_args=ARGS_$1 | ||||
|   local output_var=OUTPUT_$1 | ||||
|   local output=${!output_var} | ||||
|   local cache=.eslintcache-${1,,} | ||||
|  | ||||
|   # ensure output exists even if eslint doesn't report any errors | ||||
|   mkdir -p $(dirname $output) | ||||
|   touch $output | ||||
|  | ||||
|   eslint -f unix --cache --cache-location $cache ${!extra_args} -o $output \ | ||||
|     js subprojects/extensions-app/js | ||||
|   eslint -f unix ${!extra_args} -o $output js | ||||
| } | ||||
|  | ||||
| list_commit_range_additions() { | ||||
|   | ||||
							
								
								
									
										220
									
								
								NEWS
									
									
									
									
									
								
							
							
						
						| @@ -1,223 +1,3 @@ | ||||
| 3.37.2 | ||||
| ====== | ||||
| * Add support for "PrefersNonDefaultGPU" desktop key [Bastien; !1226] | ||||
| * Only start systemd units when running under systemd | ||||
|   [Carlos, Florian; #2755, !1242, !1252] | ||||
| * Fix "ghost" media controls [Bryan; #2776] | ||||
| * Fix zombie sockets from extensions downloader [Michael; #2774] | ||||
| * Update world clocks offsets when timezone changes [Bryan; #2209] | ||||
| * Support scrolling anywhere in slider menu items [Peter; #2795] | ||||
| * Fix "Do Not Disturb" setting getting reset on startup [Florian; #2804] | ||||
| * Only allow updates for extensions that aren't cached [Florian; !1248] | ||||
| * Fix matching notifications by PID [Florian; #2592] | ||||
| * Indicate extension errors in Extensions app [Florian; #2337] | ||||
| * Add clipboard API for querying supported mimetypes [Andy; #2819] | ||||
| * Add preview to color picker [Florian; #451] | ||||
| * Improve world clocks styling [PrOF-kk; #2825] | ||||
| * Remove Frequent view from app picker [Georges; !880] | ||||
| * Fix pad OSD glitches [Carlos; !1290] | ||||
| * Expose actor tree in looking glass [Georges; !1292] | ||||
| * Fixed crashes [Jonas D., Florian; #2709, #2757] | ||||
| * Misc. bug fixes and cleanups [Florian, AsciiWolf, Michael, Piotr, Ting-Wei, | ||||
|   Amr, Alexander, Bryan, Georges, Jonas D., Andy, Björn, Koki, Carlos; !1229, | ||||
|   !1231, !1233, !1235, #2578, #2735, #2751, #2602, #2777, !1249, #2796, !1268, | ||||
|   !1269, !1265, !1245, !1273, #2816, !1274, !1263, !1188, !1276, #2652, !1277, | ||||
|   !1281, #2286, !1267, !1286, !1279, !1288, !1293, !1294, !1291] | ||||
|  | ||||
| Contributors: | ||||
|   AsciiWolf, Michael Catanzaro, Björn Daase, Jonas Dreßler, Bryan Dunsmore, | ||||
|   Koki Fukuda, Carlos Garnacho, Andy Holmes, Amr Ibrahim, Soslan Khubulov, | ||||
|   Ting-Wei Lan, Michael Lass, Alexander Mikhaylenko, Florian Müllner, | ||||
|   Georges Basile Stavracas Neto, Bastien Nocera, PrOF-kk, Peter Simonyi | ||||
|  | ||||
| Translators: | ||||
|   Fabio Tomat [fur], Cheng-Chia Tseng [zh_TW], Yuri Chornoivan [uk], | ||||
|   Dušan Kazik [sk], Piotr Drąg [pl], Soslan Khubulov [os], | ||||
|   Daniel Mustieles [es], Nathan Follens [nl], Bruce Cowan [en_GB], | ||||
|   Florentina Mușat [ro], Milo Casagrande [it], Anders Jonsson [sv], | ||||
|   Charles Monzat [fr], Danial Behzadi [fa], sicklylife [ja], Kukuh Syafaat [id], | ||||
|   Jordi Mas [ca], Emin Tufan Çetin [tr], Jiri Grönroos [fi], Марко Костић [sr], | ||||
|   Christian Kirbach [de], Changwoo Ryu [ko], Matej Urbančič [sl] | ||||
|  | ||||
| 3.37.1 | ||||
| ====== | ||||
| * Improve bluetooth submenu title [Mariana; #2340] | ||||
| * Add openPrefs() convenience method for extensions [Florian; !1163] | ||||
| * Bring back support for empty StIcons [Andre, Jonas D.; !1173, !1178] | ||||
| * Wake up screen when unlocking programmatically [Florian; !1158] | ||||
| * Improve extensions tool error reporting [Florian; #2391] | ||||
| * Improve handling of scale-factor changes [Georges; !1176] | ||||
| * Tone down weekend days with events in calendar [Jakub; #2588] | ||||
| * Fix showing bluetooth submenu when devices were set up [Florian; !1174] | ||||
| * Add support for parental controls filtering [Philip W.; !465] | ||||
| * Provide alternative extension templates [Florian; !812] | ||||
| * Improve weather section's empty state [Mariana; #2179] | ||||
| * Fix translations of folder names [Florian; #2623] | ||||
| * Drop Tweener [Jonas Å.; !1200] | ||||
| * Match ASCII alternatives of system actions [Will; #2688] | ||||
| * Fix delay on lock screen after entering wrong password [Jonas D.; #2655] | ||||
| * Use globalThis instead of window [Andy; #2322] | ||||
| * Inhibit remote access when disabled by session mode [Jonas Å.; !1210] | ||||
| * Improve calendar-server performance [Milan; #1875] | ||||
| * Add gnome-shell-extension-prefs wrapper for compatibility [Florian; !1220] | ||||
| * Fix stuck lock screen after unlock [Jonas D., Florian; #2446] | ||||
| * Fixed crashes [Jonas D., Florian, Carlos; #2584, #2625, !1223, !1218] | ||||
| * Misc. bug fixes and cleanups [Florian, Jonas Å., Marco, Andre, Georges, | ||||
|   Jonas D., Jan, Philip Ch.,, Xiaoguang, Will, Jordan, Matthew, qarmin; | ||||
|   !1126, !1155, !1156, !1165, !1168, !1169, #2551, #2563, !1172, !1175, !1179, | ||||
|   !1160, #2562, #2578, !1184, #2559, !1186, #2607, !1191, !1194, !1199, !1203, | ||||
|   #2649, #2628, !1205, !1206, !1208, !1207, !1211, !1214, !1213, !1192, !1217, | ||||
|   !1219, #1615, #2691, !1094, !1177] | ||||
|  | ||||
| Contributors: | ||||
|   Marco Trevisan (Treviño), Philip Chimento, Milan Crha, Jonas Dreßler, | ||||
|   Carlos Garnacho, Andy Holmes, Matthew Leeds, Andre Moreira Magalhaes, | ||||
|   Florian Müllner, Georges Basile Stavracas Neto, Jordan Petridis, | ||||
|   Mariana Picolo, Jakub Steiner, Will Thompson, Jan Tojnar, Xiaoguang Wang, | ||||
|   Philip Withnall, qarmin, Jonas Ådahl | ||||
|  | ||||
| Translators: | ||||
|   Fabio Tomat [fur], Cheng-Chia Tseng [zh_TW], Danial Behzadi [fa], | ||||
|   Jiri Grönroos [fi], Ibai Oihanguren Sala [eu], Марко Костић [sr], | ||||
|   Rūdolfs Mazurs [lv], Yuri Chornoivan [uk], Carmen Bianca BAKKER [eo], | ||||
|   Dingzhong Chen [zh_CN], Rafael Fontenelle [pt_BR], Petr Kovář [cs], | ||||
|   Asier Sarasua Garmendia [eu], Daniel Mustieles [es], Emin Tufan Çetin [tr] | ||||
|  | ||||
| 3.36.0 | ||||
| ====== | ||||
| * Fix off-by-1900 error in date conversions [Florian; !1061] | ||||
| * Fix crash on startup with topIcons* extension enabled [Florian; #2308] | ||||
| * Don't require gsd-xsettings for X11 support on wayland [Olivier; !1065] | ||||
| * Fix ibus support in Xorg session [Carlos; #1690] | ||||
| * Improve Extensions D-Bus API [Florian; !1074] | ||||
| * Allow session modes to specify alternative resource name [Marco; !1063] | ||||
| * Fix link to location settings in aggregate menu [Sebastian; #2316] | ||||
| * Fix illegible app folder titles with light theme [ub; !1059] | ||||
| * Really fix visual glitch in sliders [Jonas; #1569] | ||||
|  | ||||
| Contributors: | ||||
|   Marco Trevisan (Treviño), Jonas Dreßler, Olivier Fourdan, Carlos Garnacho, | ||||
|   Sebastian Keller, Florian Müllner, ub | ||||
|  | ||||
| Translators: | ||||
|   Aman Alam [pa], Goran Vidović [hr], Aurimas Černius [lt], | ||||
|   Milo Casagrande [it], Daniel Korostil [uk], sicklylife [ja], | ||||
|   Marek Černocký [cs], Nathan Follens [nl] | ||||
|  | ||||
| 3.35.92 | ||||
| ======= | ||||
| * Plug a memory leak [Jonas D.; !1015] | ||||
| * Fix missing "back" button on login screen [Florian; #2228] | ||||
| * Fix width of window preview titles in overview [Jonas D.; #58] | ||||
| * Fix looking glass text with light style variant [Feichtmeier; !1023] | ||||
| * Center unlock entry [Florian; !1021] | ||||
| * Hide overlay scrollbars in notification popup [Jonas D.; !1013] | ||||
| * Work around add_actor() slowness in icon spring animation [Daniel; !1002] | ||||
| * Add disable-animations heuristics [Jonas Å.; !757] | ||||
| * Fix visual glitches in on-screen keyboard [Carlos; #2214] | ||||
| * Fix clearing changed textures from cache [Florian; #2244] | ||||
| * Fix visual glitch in sliders [Daniel; #1569] | ||||
| * Stop using dedicated lock screen background [Florian; !1001] | ||||
| * Fix entries disappearing after authentication errors [Florian; #2236] | ||||
| * Fix crash when animations are disabled [Florian; #2255] | ||||
| * Fix passing pointer events to clients when magnified [Jonas D.; !993] | ||||
| * Fix keynav on new lock screen [Florian; #2210] | ||||
| * Avoid short-lived allocations on actor removal [Christian; #2263] | ||||
| * Fix super-sized default avatars in user list [Florian, Sam; #2242] | ||||
| * Leave overview when locking the screen [Jonas D.; !1043] | ||||
| * Hide message list on login screen [Florian; #2241] | ||||
| * Avoid IO on the main thread [Christian, Florian; !1050, !1051] | ||||
| * Fix window animations getting stuck when client doesn't respond [Jonas; !1055] | ||||
| * Only subscribe to touchpad events for touchpad gestures [Daniel; !925] | ||||
| * Start X11 session services before Xwayland clients [Carlos; !836, !1056] | ||||
| * Only show switch-user button with unlock prompt [Florian; !1029] | ||||
| * Misc. bug fixes and cleanups [Jonas D., Florian, Georges, Jonas Å., Daniel, | ||||
|   Jakub, Philippe; !1018, !1020, !1024, !1027, !1026, !1022, !1031, !1035, | ||||
|   !1032, !1025, !1039, #2157, !1037, !1042, !1047, !1048, #2270, !1046, | ||||
|   !167, !1016] | ||||
|  | ||||
| Contributors: | ||||
|   Jonas Dreßler, Feichtmeier, Carlos Garnacho, Christian Hergert, Sam Hewitt, | ||||
|   Florian Müllner, Georges Basile Stavracas Neto, Jakub Steiner, Philippe Troin, | ||||
|   Daniel van Vugt, Jonas Ådahl | ||||
|  | ||||
| Translators: | ||||
|   Danial Behzadi [fa], Efstathios Iosifidis [el], Daniel Mustieles [es], | ||||
|   Sabri Ünal [tr], sicklylife [ja], Piotr Drąg [pl], Jordi Mas [ca], | ||||
|   Anders Jonsson [sv], Chao-Hsiung Liao [zh_TW], Asier Sarasua Garmendia [eu], | ||||
|   Rafael Fontenelle [pt_BR], Марко Костић [sr], Changwoo Ryu [ko], | ||||
|   Charles Monzat [fr], Jiri Grönroos [fi], Jor Teron [mjw], Bruce Cowan [en_GB], | ||||
|   Emin Tufan Çetin [tr], Alan Mortensen [da], Balázs Úr [hu], Fran Dieguez [gl], | ||||
|   Kukuh Syafaat [id] | ||||
|  | ||||
| 3.35.91 | ||||
| ======= | ||||
| * Improve magnifier [Carlos; !984] | ||||
| * Only enable OSK automatically if touch-mode is enabled [Carlos; #872] | ||||
| * Merge screen shield and unlock dialog to new lock screen [Georges; !872] | ||||
| * Improve ShellBlur effect [Jonas; !991] | ||||
| * Adapt user avatar for new lock screen [Umang, Georges; !922] | ||||
| * Animate prompt transition on lock screen [Florian; !972] | ||||
| * Reduce font-size in dialog titles if text doesn't fit [Jonas; !1012] | ||||
| * Various lock screen improvements and bug fixes [Jakub, Florian, Georges; | ||||
|   !996, !997, !999, #2212, !998, !1006, #2215, #2213] | ||||
| * Misc. bug fixes and cleanups [Daniel, Florian, Jakub, nana-4, Jonas; #2170, | ||||
|   #2167, !936, !988, #2187, !994, !995, !938, #2194, #2203, !1004, !977, !1014] | ||||
|  | ||||
| Contributors: | ||||
|   Jonas Dreßler, Carlos Garnacho, Umang Jain, Daniel Mustieles, Florian Müllner, | ||||
|   Georges Basile Stavracas Neto, Jakub Steiner, Daniel van Vugt, nana-4 | ||||
|  | ||||
| Translators: | ||||
|   Daniel Mustieles [es, pt_BR], Rafael Fontenelle [pt_BR], Danial Behzadi [fa], | ||||
|   Anders Jonsson [sv], Asier Sarasua Garmendia [eu], Aurimas Černius [lt], | ||||
|   Bruce Cowan [en_GB], sicklylife [ja], Fran Dieguez [gl], Kukuh Syafaat [id], | ||||
|   Emin Tufan Çetin [tr], Jiri Grönroos [fi], Jordi Mas [ca], Claude Paroz [fr], | ||||
|   Ask Hjorth Larsen [da], Марко Костић [sr], Piotr Drąg [pl], | ||||
|   Charles Monzat [fr], Balázs Úr [hu] | ||||
|  | ||||
| 3.35.90 | ||||
| ======= | ||||
| * Update default favorite apps [Michael; !907] | ||||
| * Add Shell.Blur effect [Georges; !864, !924] | ||||
| * Overhaul scroll/swipe gestures [Alexander; !821, !825, !826] | ||||
| * Fix VPN connections when delaying request [Florian; #2008] | ||||
| * Overhaul theme [Sam, Jakub, nana-4; !904, !931, !957] | ||||
| * Improve visual appearance of Weather integration [Florian; #1143] | ||||
| * Implement new system dialog designs [Jonas; #1343] | ||||
| * Animate position changes of app icons [Georges; !882] | ||||
| * Add St.Viewport [Georges; !929] | ||||
| * Make app folders behave as dialogs [Georges; !896] | ||||
| * Add do-not-disturb functionality to calendar popup [Florian; #239] | ||||
| * Show hint actor in focused entries [Jonas; !944] | ||||
| * Switch screen-recorder back to VP8 [Björn; #256] | ||||
| * Allow to run perf-tool as wayland compositor [Olivier; !941] | ||||
| * Handle extension updates [Florian; !945] | ||||
| * Animate showing and hiding caps-lock warning [Jonas; !952] | ||||
| * Support "auto" lengths in CSS [Florian; !971] | ||||
| * Turn extension-prefs into the offical Extensions app [Florian; #1968] | ||||
| * Sandbox the portal helper [Michael; !983] | ||||
| * Misc. bug fixes and cleanups [Florian, Björn, Jakub, Alexander, Daniel V., | ||||
|   Jonas, nana-4, Carlos, Sebastian, Daniel G., Georges, Piotr; !918, !917, | ||||
|   !919, !920, #763, #791659, !927, #2091, !930, !926, !888, !934, !168, #2133, | ||||
|   #682, #2142, #2131, !943, #2132, #1958, #2146, !951, #1779, #2130, !964, | ||||
|   !965, !948, #2151, #1746, !967, !760, !968, !970, !973, #2169, #2176, !978, | ||||
|   !980, !979, #2177, !981, #2180, !974] | ||||
|  | ||||
| Contributors: | ||||
|   Michael Catanzaro, Björn Daase, Jonas Dreßler, Piotr Drąg, Olivier Fourdan, | ||||
|   Carlos Garnacho, Sam Hewitt, Sebastian Keller, Andre Klapper, | ||||
|   Alexander Mikhaylenko, Daniel García Moreno, Florian Müllner, | ||||
|   Georges Basile Stavracas Neto, Jakub Steiner, Daniel van Vugt, nana-4 | ||||
|  | ||||
| Translators: | ||||
|   Asier Sarasua Garmendia [eu], Daniel Mustieles [es], Andrej Shadura [sk], | ||||
|   Carmen Bianca BAKKER [eo], Sucipto [id], Dušan Kazik [sk], Goran Vidović [hr], | ||||
|   sicklylife [ja], Kukuh Syafaat [id], Yi-Jyun Pan [zh_TW], | ||||
|   Rafael Fontenelle [pt_BR], Jordi Mas [ca], Jiri Grönroos [fi], | ||||
|   Fabio Tomat [fur], Umarzuki Bin Mochlis Moktar [ms], Daniel Korostil [uk], | ||||
|   Jor Teron [mjw], Anders Jonsson [sv], Aurimas Černius [lt] | ||||
|  | ||||
| 3.35.3 | ||||
| ====== | ||||
| * Add discrete GPU support for NVidia drivers [Bastien; #1810] | ||||
|   | ||||
| @@ -161,16 +161,12 @@ def convert_file(source_file, destination_path): | ||||
|     try: | ||||
|         xkb_name = locale_to_xkb(root["locale"], root["name"]) | ||||
|     except KeyError as e: | ||||
|         logging.warning(e) | ||||
|         logging.warn(e) | ||||
|         return False | ||||
|     destination_file = os.path.join(destination_path, xkb_name + ".json") | ||||
| 
 | ||||
|     try: | ||||
|         with open(destination_file, 'x', encoding="utf-8") as dest_fd: | ||||
|             json.dump(root, dest_fd, ensure_ascii=False, indent=2, sort_keys=True) | ||||
|     except FileExistsError as e: | ||||
|         logging.info("File %s exists, not updating", destination_file) | ||||
|         return False | ||||
|     with open(destination_file, 'w', encoding="utf-8") as dest_fd: | ||||
|         json.dump(root, dest_fd, ensure_ascii=False, indent=2, sort_keys=True) | ||||
| 
 | ||||
|     logging.debug("written %s", destination_file) | ||||
| 
 | ||||
| @@ -1,19 +1,12 @@ | ||||
| <node> | ||||
|   <interface name="org.gnome.Shell.CalendarServer"> | ||||
|     <method name="SetTimeRange"> | ||||
|       <arg type="x" name="since" direction="in"/> | ||||
|       <arg type="x" name="until" direction="in"/> | ||||
|       <arg type="b" name="force_reload" direction="in"/> | ||||
|     <method name="GetEvents"> | ||||
|       <arg type="x" direction="in" /> | ||||
|       <arg type="x" direction="in" /> | ||||
|       <arg type="b" direction="in" /> | ||||
|       <arg type="a(sssbxxa{sv})" direction="out" /> | ||||
|     </method> | ||||
|     <signal name="EventsAddedOrUpdated"> | ||||
|       <arg type="a(ssbxxa{sv})" name="events" direction="out"/> | ||||
|     </signal> | ||||
|     <signal name="EventsRemoved"> | ||||
|       <arg type="as" name="ids" direction="out"/> | ||||
|     </signal> | ||||
|     <signal name="ClientDisappeared"> | ||||
|       <arg type="s" name="source_uid" direction="out"/> | ||||
|     </signal> | ||||
|     <property name="HasCalendars" type="b" access="read" /> | ||||
|     <signal name="Changed" /> | ||||
|   </interface> | ||||
| </node> | ||||
|   | ||||
| @@ -199,37 +199,14 @@ | ||||
|  | ||||
|     <!-- | ||||
|         LaunchExtensionPrefs: | ||||
|         Deprecated for OpenExtensionPrefs | ||||
|         @uuid: The UUID of the extension | ||||
|  | ||||
|         Launch preferences of an extension. | ||||
|     --> | ||||
|     <method name="LaunchExtensionPrefs"> | ||||
|       <arg type="s" direction="in" name="uuid"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         OpenExtensionPrefs: | ||||
|         @uuid: The UUID of the extension | ||||
|         @parent_window: Identifier for the application window | ||||
|         @options: Vardict with further options | ||||
|  | ||||
|         Opens the prefs dialog of extension @uuid. | ||||
|  | ||||
|         The following @options are recognized: | ||||
|  | ||||
|         <variablelist> | ||||
|           <varlistentry> | ||||
|             <term>modal b</term> | ||||
|             <listitem> | ||||
|               <para>Whether the prefs window should be modal, default: false</para> | ||||
|             </listitem> | ||||
|           </varlistentry> | ||||
|         </variablelist> | ||||
|     --> | ||||
|     <method name="OpenExtensionPrefs"> | ||||
|       <arg type="s" direction="in" name="uuid"/> | ||||
|       <arg type="s" direction="in" name="parent_window"/> | ||||
|       <arg type="a{sv}" direction="in" name="options"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         CheckForUpdates: | ||||
|         Update all extensions for which updates are available | ||||
| @@ -257,11 +234,5 @@ | ||||
|     --> | ||||
|     <property name="ShellVersion" type="s" access="read"/> | ||||
|  | ||||
|     <!-- | ||||
|         UserExtensionsEnabled: | ||||
|         Whether user extensions are enabled | ||||
|     --> | ||||
|     <property name="UserExtensionsEnabled" type="b" access="readwrite"/> | ||||
|  | ||||
|   </interface> | ||||
| </node> | ||||
|   | ||||
| @@ -57,19 +57,5 @@ | ||||
|     <method name="GetWindows"> | ||||
|       <arg name="windows" direction="out" type="a{ta{sv}}" /> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|        AnimationsEnabled: | ||||
|        @short_description: Whether the shell animations are enabled | ||||
|  | ||||
|        By default determined by the org.gnome.desktop.interface enable-animations | ||||
|        gsetting, but may be overridden, e.g. if there is an active screen cast or | ||||
|        remote desktop session that asked for animations to be disabled. | ||||
|  | ||||
|        Since: 2 | ||||
|     --> | ||||
|     <property name="AnimationsEnabled" type="b" access="read"/> | ||||
|  | ||||
|     <property name="version" type="u" access="read"/> | ||||
|   </interface> | ||||
| </node> | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| <gresources> | ||||
|   <gresource prefix="/org/gnome/shell/dbus-interfaces"> | ||||
|     <file preprocess="xml-stripblanks">net.hadess.SensorProxy.xml</file> | ||||
|     <file preprocess="xml-stripblanks">net.hadess.SwitcherooControl.xml</file> | ||||
|     <file preprocess="xml-stripblanks">org.freedesktop.Application.xml</file> | ||||
|     <file preprocess="xml-stripblanks">org.freedesktop.bolt1.Device.xml</file> | ||||
|     <file preprocess="xml-stripblanks">org.freedesktop.bolt1.Manager.xml</file> | ||||
|   | ||||
							
								
								
									
										8
									
								
								data/gnome-shell-extension-prefs.desktop.in.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| [Desktop Entry] | ||||
| Type=Application | ||||
| Name=Shell Extensions | ||||
| Comment=Configure GNOME Shell Extensions | ||||
| Exec=@bindir@/gnome-shell-extension-prefs %u | ||||
| Categories=GNOME;GTK; | ||||
| OnlyShowIn=GNOME; | ||||
| NoDisplay=true | ||||
| @@ -6,13 +6,19 @@ | ||||
|     <file>checkbox-off-focused.svg</file> | ||||
|     <file>checkbox-off.svg</file> | ||||
|     <file>checkbox.svg</file> | ||||
|     <file alias="icons/color-pick.svg">color-pick.svg</file> | ||||
|     <file>dash-placeholder.svg</file> | ||||
|     <file>gnome-shell.css</file> | ||||
|     <file>gnome-shell-high-contrast.css</file> | ||||
|     <file>key-enter.svg</file> | ||||
|     <file>key-hide.svg</file> | ||||
|     <file>key-layout.svg</file> | ||||
|     <file>key-shift.svg</file> | ||||
|     <file>key-shift-uppercase.svg</file> | ||||
|     <file>key-shift-latched-uppercase.svg</file> | ||||
|     <file alias="icons/message-indicator-symbolic.svg">message-indicator-symbolic.svg</file> | ||||
|     <file>no-events.svg</file> | ||||
|     <file>no-notifications.svg</file> | ||||
|     <file>noise-texture.png</file> | ||||
|     <file>pad-osd.css</file> | ||||
|     <file alias="icons/eye-open-negative-filled-symbolic.svg">eye-open-negative-filled-symbolic.svg</file> | ||||
|     <file alias="icons/eye-not-looking-symbolic.svg">eye-not-looking-symbolic.svg</file> | ||||
| @@ -20,11 +26,6 @@ | ||||
|     <file alias="icons/pointer-drag-symbolic.svg">pointer-drag-symbolic.svg</file> | ||||
|     <file alias="icons/pointer-primary-click-symbolic.svg">pointer-primary-click-symbolic.svg</file> | ||||
|     <file alias="icons/pointer-secondary-click-symbolic.svg">pointer-secondary-click-symbolic.svg</file> | ||||
|     <file alias="icons/keyboard-caps-lock-filled-symbolic.svg">keyboard-caps-lock-filled-symbolic.svg</file> | ||||
|     <file alias="icons/keyboard-enter-symbolic.svg">keyboard-enter-symbolic.svg</file> | ||||
|     <file alias="icons/keyboard-hide-symbolic.svg">keyboard-hide-symbolic.svg</file> | ||||
|     <file alias="icons/keyboard-layout-filled-symbolic.svg">keyboard-layout-filled-symbolic.svg</file> | ||||
|     <file alias="icons/keyboard-shift-filled-symbolic.svg">keyboard-shift-filled-symbolic.svg</file> | ||||
|     <file>process-working.svg</file> | ||||
|     <file>toggle-off.svg</file> | ||||
|     <file>toggle-off-dark.svg</file> | ||||
|   | ||||
| @@ -19,10 +19,6 @@ Before=gnome-session-initialized.target | ||||
| [Service] | ||||
| Type=notify | ||||
| ExecStart=@bindir@/gnome-shell | ||||
|  | ||||
| # unset some environment variables that were set by the shell and won't work now that the shell is gone | ||||
| ExecStopPost=-systemctl --user unset-environment GNOME_SETUP_DISPLAY WAYLAND_DISPLAY DISPLAY XAUTHORITY | ||||
|  | ||||
| # Exit code 1 means we are probably *not* dealing with an extension failure | ||||
| SuccessExitStatus=1 | ||||
| # On wayland we cannot restart | ||||
|   | ||||
| Before Width: | Height: | Size: 6.1 KiB | 
| @@ -1 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M6.5 1.031c-.371 0-.742-.035-1.11.016-.367.05-.73.203-.972.476-.125.141-.215.309-.266.485-.047.18-.054.367-.02.55.032.184.102.356.192.516.09.164.203.309.317.457L5 4H2a1.8 1.8 0 00-.41.035.791.791 0 00-.36.195.791.791 0 00-.195.36C1 4.723 1 4.863 1 5v2.75l.77-.344c.265-.117.542-.23.832-.242.289-.016.586.074.812.254.227.18.383.441.465.723.082.277.101.57.121.859.02.316.04.637-.016.95-.058.312-.199.616-.43.831a1.264 1.264 0 01-.874.32c-.317-.007-.618-.128-.91-.257L1 10.5V14c0 .137.004.277.035.41a.791.791 0 00.195.36c.098.097.227.16.36.195.133.035.273.035.41.035h3l-.328-.68c-.14-.293-.274-.597-.29-.922-.015-.32.095-.652.31-.894.214-.242.523-.39.84-.453.316-.067.644-.059.968-.059.324 0 .652-.008.969.059.316.062.625.21.84.453.214.242.324.574.308.894-.015.325-.148.63-.289.922L8 15h3a1.8 1.8 0 00.41-.035.791.791 0 00.36-.195.791.791 0 00.195-.36C12 14.277 12 14.137 12 14v-3.563l.703.297c.29.125.59.239.902.246.313.004.63-.101.864-.308.238-.203.386-.496.46-.8C15 9.565 15 9.25 15 8.937c0-.313 0-.63-.07-.934-.075-.305-.223-.598-.461-.8a1.288 1.288 0 00-.864-.31c-.312.008-.613.122-.902.247L12 7.437V5a1.8 1.8 0 00-.035-.41.791.791 0 00-.195-.36.791.791 0 00-.36-.195C11.277 4 11.137 4 11 4H8l.36-.469c.113-.148.226-.293.316-.457.09-.16.16-.332.191-.515a1.248 1.248 0 00-.02-.551 1.256 1.256 0 00-.265-.485c-.242-.273-.605-.425-.973-.476-.367-.05-.738-.016-1.109-.016zm0 0" fill="#474747"/></svg> | ||||
| Before Width: | Height: | Size: 1.4 KiB | 
| @@ -1 +0,0 @@ | ||||
| install_subdir('hicolor', install_dir: icondir) | ||||
| @@ -1,6 +1,6 @@ | ||||
| desktop_files = [ | ||||
|   'org.gnome.Shell.desktop', | ||||
|   'org.gnome.Shell.Extensions.desktop', | ||||
|   'gnome-shell-extension-prefs.desktop' | ||||
| ] | ||||
| service_files = [] | ||||
|  | ||||
| @@ -43,7 +43,6 @@ endforeach | ||||
|  | ||||
|  | ||||
| subdir('dbus-interfaces') | ||||
| subdir('icons') | ||||
| subdir('theme') | ||||
|  | ||||
| data_resources = [ | ||||
|   | ||||
| @@ -1,10 +0,0 @@ | ||||
| [Desktop Entry] | ||||
| Type=Application | ||||
| # Keep in sync with subprojects/extensions-app | ||||
| Name=Extensions | ||||
| # Translators: Do NOT translate or transliterate this text (this is an icon file name)! | ||||
| Icon=org.gnome.Shell.Extensions | ||||
| # Never launch this, just provide name+icon to portal dialog | ||||
| Exec=false | ||||
| OnlyShowIn=GNOME; | ||||
| NoDisplay=true | ||||
| @@ -12,9 +12,7 @@ | ||||
|             "w" | ||||
|           ], | ||||
|           [ | ||||
|             "e", | ||||
|             "é", | ||||
|             "ë" | ||||
|             "e" | ||||
|           ], | ||||
|           [ | ||||
|             "r" | ||||
| @@ -23,58 +21,30 @@ | ||||
|             "t" | ||||
|           ], | ||||
|           [ | ||||
|             "y", | ||||
|             "ý", | ||||
|             "ÿ" | ||||
|             "y" | ||||
|           ], | ||||
|           [ | ||||
|             "u", | ||||
|             "ú", | ||||
|             "ü", | ||||
|             "û", | ||||
|             "ù", | ||||
|             "ū" | ||||
|             "u" | ||||
|           ], | ||||
|           [ | ||||
|             "i", | ||||
|             "í", | ||||
|             "ï" | ||||
|             "i" | ||||
|           ], | ||||
|           [ | ||||
|             "o", | ||||
|             "ó", | ||||
|             "ô", | ||||
|             "ò", | ||||
|             "õ", | ||||
|             "œ", | ||||
|             "ō" | ||||
|             "o" | ||||
|           ], | ||||
|           [ | ||||
|             "p" | ||||
|           ], | ||||
|           [ | ||||
|             "å" | ||||
|           ] | ||||
|         ], | ||||
|         [ | ||||
|           [ | ||||
|             "a", | ||||
|             "á", | ||||
|             "ä", | ||||
|             "à", | ||||
|             "â", | ||||
|             "ã", | ||||
|             "ā" | ||||
|             "a" | ||||
|           ], | ||||
|           [ | ||||
|             "s", | ||||
|             "ß", | ||||
|             "ś", | ||||
|             "š" | ||||
|             "s" | ||||
|           ], | ||||
|           [ | ||||
|             "d", | ||||
|             "ð" | ||||
|             "d" | ||||
|           ], | ||||
|           [ | ||||
|             "f" | ||||
| @@ -92,16 +62,7 @@ | ||||
|             "k" | ||||
|           ], | ||||
|           [ | ||||
|             "l", | ||||
|             "ł" | ||||
|           ], | ||||
|           [ | ||||
|             "ø", | ||||
|             "ö" | ||||
|           ], | ||||
|           [ | ||||
|             "æ", | ||||
|             "ä" | ||||
|             "l" | ||||
|           ] | ||||
|         ], | ||||
|         [ | ||||
| @@ -121,9 +82,7 @@ | ||||
|             "b" | ||||
|           ], | ||||
|           [ | ||||
|             "n", | ||||
|             "ñ", | ||||
|             "ń" | ||||
|             "n" | ||||
|           ], | ||||
|           [ | ||||
|             "m" | ||||
| @@ -162,9 +121,7 @@ | ||||
|             "W" | ||||
|           ], | ||||
|           [ | ||||
|             "E", | ||||
|             "É", | ||||
|             "Ë" | ||||
|             "E" | ||||
|           ], | ||||
|           [ | ||||
|             "R" | ||||
| @@ -173,58 +130,30 @@ | ||||
|             "T" | ||||
|           ], | ||||
|           [ | ||||
|             "Y", | ||||
|             "Ý", | ||||
|             "Ÿ" | ||||
|             "Y" | ||||
|           ], | ||||
|           [ | ||||
|             "U", | ||||
|             "Ú", | ||||
|             "Ü", | ||||
|             "Û", | ||||
|             "Ù", | ||||
|             "Ū" | ||||
|             "U" | ||||
|           ], | ||||
|           [ | ||||
|             "I", | ||||
|             "Í", | ||||
|             "Ï" | ||||
|             "I" | ||||
|           ], | ||||
|           [ | ||||
|             "O", | ||||
|             "Ó", | ||||
|             "Ô", | ||||
|             "Ò", | ||||
|             "Õ", | ||||
|             "Œ", | ||||
|             "Ō" | ||||
|             "O" | ||||
|           ], | ||||
|           [ | ||||
|             "P" | ||||
|           ], | ||||
|           [ | ||||
|             "Å" | ||||
|           ] | ||||
|         ], | ||||
|         [ | ||||
|           [ | ||||
|             "A", | ||||
|             "Á", | ||||
|             "Ä", | ||||
|             "À", | ||||
|             "Â", | ||||
|             "Ã", | ||||
|             "Ā" | ||||
|             "A" | ||||
|           ], | ||||
|           [ | ||||
|             "S", | ||||
|             "SS", | ||||
|             "Ś", | ||||
|             "Š" | ||||
|             "S" | ||||
|           ], | ||||
|           [ | ||||
|             "D", | ||||
|             "Ð" | ||||
|             "D" | ||||
|           ], | ||||
|           [ | ||||
|             "F" | ||||
| @@ -242,16 +171,7 @@ | ||||
|             "K" | ||||
|           ], | ||||
|           [ | ||||
|             "L", | ||||
|             "Ł" | ||||
|           ], | ||||
|           [ | ||||
|             "Ø", | ||||
|             "Ö" | ||||
|           ], | ||||
|           [ | ||||
|             "Æ", | ||||
|             "Ä" | ||||
|             "L" | ||||
|           ] | ||||
|         ], | ||||
|         [ | ||||
| @@ -271,9 +191,7 @@ | ||||
|             "B" | ||||
|           ], | ||||
|           [ | ||||
|             "N", | ||||
|             "Ñ", | ||||
|             "Ń" | ||||
|             "N" | ||||
|           ], | ||||
|           [ | ||||
|             "M" | ||||
| @@ -359,10 +277,10 @@ | ||||
|             "#" | ||||
|           ], | ||||
|           [ | ||||
|             "€", | ||||
|             "$", | ||||
|             "¢", | ||||
|             "£", | ||||
|             "$", | ||||
|             "€", | ||||
|             "¥", | ||||
|             "₱" | ||||
|           ], | ||||
| @@ -500,16 +418,15 @@ | ||||
|           [ | ||||
|             "£" | ||||
|           ], | ||||
|           [ | ||||
|             "¢" | ||||
|           ], | ||||
|           [ | ||||
|             "€" | ||||
|           ], | ||||
|           [ | ||||
|             "¥" | ||||
|           ], | ||||
|           [ | ||||
|             "$", | ||||
|             "¢" | ||||
|           ], | ||||
|           [ | ||||
|             "¢" | ||||
|           ], | ||||
|           [ | ||||
|             "^", | ||||
|             "↑", | ||||
| @@ -587,4 +504,4 @@ | ||||
|   ], | ||||
|   "locale": "nb", | ||||
|   "name": "Norwegian Bokmål" | ||||
| } | ||||
| } | ||||
| @@ -1,94 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="5.4116011mm" | ||||
|    height="5.1374583mm" | ||||
|    viewBox="0 0 5.4116011 5.1374583" | ||||
|    version="1.1" | ||||
|    id="svg5595" | ||||
|    inkscape:version="0.92.4 (unknown)" | ||||
|    sodipodi:docname="color-pick.svg"> | ||||
|   <defs | ||||
|      id="defs5589"> | ||||
|     <filter | ||||
|        inkscape:collect="always" | ||||
|        x="-0.10291173" | ||||
|        width="1.2058235" | ||||
|        y="-0.065432459" | ||||
|        height="1.1308649" | ||||
|        id="filter5601" | ||||
|        style="color-interpolation-filters:sRGB"> | ||||
|       <feGaussianBlur | ||||
|          inkscape:collect="always" | ||||
|          stdDeviation="0.610872" | ||||
|          id="feGaussianBlur5603" /> | ||||
|     </filter> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="15.839192" | ||||
|      inkscape:cx="39.387731" | ||||
|      inkscape:cy="12.554326" | ||||
|      inkscape:document-units="mm" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1920" | ||||
|      inkscape:window-height="1016" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      fit-margin-top="0" | ||||
|      fit-margin-left="0" | ||||
|      fit-margin-right="0" | ||||
|      fit-margin-bottom="0" /> | ||||
|   <metadata | ||||
|      id="metadata5592"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-103.12753,-146.26461)"> | ||||
|     <circle | ||||
|        r="8.4810486" | ||||
|        cy="9.82623" | ||||
|        cx="10.226647" | ||||
|        id="circle7584" | ||||
|        style="color:#000000;display:inline;overflow:visible;opacity:0.6;vector-effect:none;fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:10;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;filter:url(#filter5601)" | ||||
|        transform="matrix(0.26458333,0,0,0.26458333,103.12753,146.26461)" /> | ||||
|     <path | ||||
|        style="color:#000000;display:inline;overflow:visible;opacity:1;vector-effect:none;fill:#000000;fill-opacity:1;stroke:#ffffff;stroke-width:0.26399338;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:10;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal" | ||||
|        d="m 108.07728,148.64122 c 0,1.2393 -1.00465,2.24394 -2.24395,2.24394 -1.23929,0 -2.24716,-1.00465 -2.25221,-2.24394 l -0.009,-2.24458 2.26136,6.4e-4 c 1.2393,3.4e-4 2.24395,1.00464 2.24395,2.24394 z" | ||||
|        id="path7523-7" | ||||
|        inkscape:connector-curvature="0" | ||||
|        sodipodi:nodetypes="ssscss" /> | ||||
|     <circle | ||||
|        style="color:#000000;display:inline;overflow:visible;opacity:1;vector-effect:none;fill:#50dbb5;fill-opacity:1;stroke:none;stroke-width:0.36885914;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:10;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal" | ||||
|        id="path7482-1" | ||||
|        cx="105.83707" | ||||
|        cy="148.64352" | ||||
|        r="1.844296" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.7 KiB | 
| @@ -19,13 +19,13 @@ $error_color: #ff8080; | ||||
| $success_color: if($variant == 'light', #33d17a, darken(#33d17a, 10%)); | ||||
| $destructive_color: if($variant == 'light', #e01b24, darken(#e01b24, 10%)); | ||||
|  | ||||
| $osd_fg_color: #eeeeec; | ||||
| $osd_text_color: white; | ||||
| $osd_bg_color: transparentize(darken(desaturate(#3d3846, 100%), 12%),0.04); | ||||
| $osd_fg_color: $fg_color; | ||||
| $osd_text_color: if($variant == 'light', #000, #fff); | ||||
| $osd_bg_color: if($variant == 'light', rgba(255,255,255,0.9), transparentize(darken(desaturate(#3d3846, 100%), 12%),0.04)); | ||||
| $osd_insensitive_bg_color: transparentize(mix($osd_fg_color, opacify($osd_bg_color, 1), 10%), 0.5); | ||||
| $osd_insensitive_fg_color: mix($osd_fg_color, opacify($osd_bg_color, 1), 50%); | ||||
| $osd_borders_color: transparentize(black, 0.3); | ||||
| $osd_outer_borders_color: transparentize(white, 0.84); | ||||
| $osd_borders_color: if($variant == 'light', rgba(255,255,255,0.1), rgba(0,0,0,0.7)); | ||||
| $osd_outer_borders_color: if($variant == 'light', rgba(0,0,0,0.1), lighten($osd_bg_color, 7%)); | ||||
|  | ||||
| $shadow_color: if($variant == 'light', rgba(0,0,0,0.1), rgba(0,0,0,0.2)); | ||||
|  | ||||
| @@ -40,4 +40,4 @@ $backdrop_bg_color: $bg_color; | ||||
| $backdrop_fg_color: mix($fg_color, $backdrop_bg_color, 80%); | ||||
| $backdrop_insensitive_color: if($variant =='light', darken($backdrop_bg_color,15%), lighten($backdrop_bg_color,15%)); | ||||
| $backdrop_borders_color: mix($borders_color, $bg_color, 90%); | ||||
| $backdrop_dark_fill: mix($backdrop_borders_color,$backdrop_bg_color, 35%); | ||||
| $backdrop_dark_fill: mix($backdrop_borders_color,$backdrop_bg_color, 35%); | ||||
| @@ -54,8 +54,8 @@ $base_font_size: 11; | ||||
| $text_shadow_color: if($variant == 'light', rgba(255,255,255,0.3), rgba(0,0,0,0.2)); | ||||
|  | ||||
| // icons | ||||
| $base_icon_size: 1.09em; | ||||
| // $base_icon_size: 16px; | ||||
| // $base_icon_size: 1.09em; | ||||
| $base_icon_size: 16px; | ||||
|  | ||||
| // Stage | ||||
| stage { | ||||
| @@ -91,9 +91,13 @@ stage { | ||||
|  | ||||
| // icon tiles | ||||
| %icon_tile { | ||||
|   background-color: transparent; // no background | ||||
|   color: $osd_fg_color; | ||||
|   border-radius: $base_border_radius + 4px; | ||||
|   padding: $base_padding; | ||||
|   border: 2px solid transparent; | ||||
|   border-width: 2px; | ||||
|   border-style: solid; | ||||
|   border-color: transparent; | ||||
|   transition-duration: 100ms; | ||||
|   text-align: center; | ||||
| } | ||||
| @@ -151,17 +155,14 @@ stage { | ||||
|  | ||||
|  | ||||
| // notification styling | ||||
| @mixin notification_bubble($flat: false) { | ||||
| %notification_bubble { | ||||
|   border-width: 1px; | ||||
|   border-style: solid; | ||||
|   border-radius: $base_border_radius + 2px; | ||||
|   border-radius:$base_border_radius + 2px; | ||||
|   padding: 0; | ||||
|   margin: $base_margin; | ||||
|  | ||||
|   @if $flat { | ||||
|     @include button(undecorated); | ||||
|   } @else { | ||||
|     @include button(normal); | ||||
|   } | ||||
|   @include button(normal); | ||||
|  | ||||
|   &:focus { | ||||
|     @include button(focus); | ||||
|   | ||||
| @@ -137,7 +137,7 @@ | ||||
|   // normal button | ||||
|   @if $t==normal { | ||||
|     color: $tc; | ||||
|     background-color: lighten($c, 3%); | ||||
|     background-color: lighten($c, 3%) !important; | ||||
|     border-color: draw_border_color($c); | ||||
|     @include draw_shadows($button_shadow); | ||||
|     // box-shadow: 0 1px 1px 0 rgba(0,0,0,0.1); | ||||
| @@ -150,14 +150,14 @@ | ||||
|     color: $tc; | ||||
|     text-shadow: 0 1px $text_shadow_color; | ||||
|     icon-shadow: 0 1px $text_shadow_color; | ||||
|     box-shadow: inset 0 0 0 2px transparentize($selected_bg_color, 0.4); | ||||
|     box-shadow: inset 0 0 0 2px transparentize($selected_bg_color, 0.7); | ||||
|     //border-color: $selected_bg_color; | ||||
|   } | ||||
|  | ||||
|   // hover button | ||||
|   @else if $t==hover { | ||||
|     color: $tc; | ||||
|     background-color: lighten($c, if($variant == 'light', 8%, 5%)); | ||||
|     background-color: lighten($c, if($variant == 'light', 8%, 5%)) !important; | ||||
|     border-color: if($variant == 'light', draw_border_color(lighten($c, 7%)), draw_border_color($c)); | ||||
|     @include draw_shadows($button_shadow); | ||||
|     text-shadow: 0 1px $text_shadow_color; | ||||
| @@ -167,7 +167,7 @@ | ||||
|   // active button | ||||
|   @else if $t==active { | ||||
|     color: $tc; | ||||
|     background-color: darken($c,3%); | ||||
|     background-color: darken($c,3%) !important; | ||||
|     border-color: draw_border_color(if($variant == 'light', $c, darken($c,7%))); | ||||
|     text-shadow: none; | ||||
|     icon-shadow: none; | ||||
| @@ -178,7 +178,7 @@ | ||||
|   @else if $t==insensitive { | ||||
|     color: $insensitive_fg_color; | ||||
|     border-color: $insensitive_borders_color; | ||||
|     background-color: $insensitive_bg_color; | ||||
|     background-color: $insensitive_bg_color !important; | ||||
|     box-shadow: none; | ||||
|     text-shadow: none; | ||||
|     icon-shadow: none; | ||||
| @@ -194,38 +194,3 @@ | ||||
|     icon-shadow: none; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // overview icons | ||||
| @mixin overview-icon($color) { | ||||
|   .overview-icon { | ||||
|     @extend %icon_tile; | ||||
|     color: $color; | ||||
|   } | ||||
|  | ||||
|   &:hover, | ||||
|   &:selected { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($color, .9); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:focus { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($color, .7); | ||||
|       // border-color: $selected_bg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:drop { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($selected_bg_color, .15); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:active, | ||||
|   &:checked { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize(darken($osd_bg_color, 10%), .5); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -5,47 +5,34 @@ | ||||
| // | ||||
|  | ||||
| /* WIDGETS */ | ||||
|  | ||||
| // Primary widgets | ||||
| @import 'widgets/base'; | ||||
| @import 'widgets/entries'; | ||||
| @import 'widgets/app-grid'; | ||||
| @import 'widgets/app-switcher'; | ||||
| @import 'widgets/buttons'; | ||||
| @import 'widgets/check-box'; | ||||
| @import 'widgets/switches'; | ||||
| @import 'widgets/slider'; | ||||
| @import 'widgets/scrollbars'; | ||||
| // Popovers | ||||
| @import 'widgets/popovers'; | ||||
| @import 'widgets/calendar'; | ||||
| @import 'widgets/message-list'; | ||||
| @import 'widgets/ibus-popup'; | ||||
| // Notifications | ||||
| @import 'widgets/notifications'; | ||||
| @import 'widgets/hotplug'; | ||||
| // Dialogs | ||||
| @import 'widgets/dialogs'; | ||||
| @import 'widgets/network-dialog'; | ||||
| // OSDs | ||||
| @import 'widgets/osd'; | ||||
| @import 'widgets/switcher-popup'; | ||||
| @import 'widgets/workspace-switcher'; | ||||
| // Panel | ||||
| @import 'widgets/panel'; | ||||
| @import 'widgets/check-box'; | ||||
| @import 'widgets/corner-ripple'; | ||||
| // Overview | ||||
| @import 'widgets/dash'; | ||||
| @import 'widgets/dialogs'; | ||||
| @import 'widgets/entries'; | ||||
| @import 'widgets/hotplug'; | ||||
| @import 'widgets/ibus-popup'; | ||||
| @import 'widgets/keyboard'; | ||||
| @import 'widgets/login-dialog'; | ||||
| @import 'widgets/looking-glass'; | ||||
| @import 'widgets/message-list'; | ||||
| @import 'widgets/notifications'; | ||||
| @import 'widgets/misc'; | ||||
| @import 'widgets/network-dialog'; | ||||
| @import 'widgets/osd'; | ||||
| @import 'widgets/overview'; | ||||
| @import 'widgets/window-picker'; | ||||
| @import 'widgets/panel'; | ||||
| @import 'widgets/popovers'; | ||||
| @import 'widgets/screen-shield'; | ||||
| @import 'widgets/scrollbars'; | ||||
| @import 'widgets/search-entry'; | ||||
| @import 'widgets/search-results'; | ||||
| @import 'widgets/app-grid'; | ||||
| @import 'widgets/dash'; | ||||
| @import 'widgets/workspace-thumbnails'; | ||||
| // A11y / misc | ||||
| @import 'widgets/a11y'; | ||||
| @import 'widgets/misc'; | ||||
| @import 'widgets/slider'; | ||||
| @import 'widgets/switches'; | ||||
| @import 'widgets/tiled-previews'; | ||||
| @import 'widgets/keyboard'; | ||||
| @import 'widgets/looking-glass'; | ||||
| // Lock / login screens | ||||
| @import 'widgets/login-dialog'; | ||||
| @import 'widgets/screen-shield'; | ||||
| @import 'widgets/window-picker'; | ||||
| @import 'widgets/workspace-switcher'; | ||||
|   | ||||
| @@ -1,24 +0,0 @@ | ||||
| // Pointer location | ||||
| .ripple-pointer-location { | ||||
|   width: $ripple_size; | ||||
|   height: $ripple_size; | ||||
|   border-radius: $ripple_size * 0.5; // radius equals the size of the box to give us the curve | ||||
|   background-color: lighten(transparentize($selected_bg_color, 0.7), 30%); | ||||
|   box-shadow: 0 0 2px 2px lighten($selected_bg_color, 20%); | ||||
| } | ||||
|  | ||||
| // Pointer accessibility notifications | ||||
| .pie-timer { | ||||
|   width: 60px; | ||||
|   height: 60px; | ||||
|   -pie-border-width: 3px; | ||||
|   -pie-border-color: $selected_bg_color; | ||||
|   -pie-background-color: lighten(transparentize($selected_bg_color, 0.7), 40%); | ||||
| } | ||||
|  | ||||
| // Screen zoom/Magnifier | ||||
| .magnifier-zoom-region { | ||||
|   border: 2px solid $selected_bg_color; | ||||
|  | ||||
|   &.full-screen { border-width: 0; } | ||||
| } | ||||
| @@ -1,60 +1,110 @@ | ||||
| /* App Grid */ | ||||
|  | ||||
| $app_icon_size: 96px; | ||||
| $app_icon_padding: 24px; | ||||
|  | ||||
| // app icons | ||||
| .icon-grid { | ||||
|   row-spacing: $base_spacing * 6; | ||||
|   column-spacing: $base_spacing * 6; | ||||
|   max-row-spacing: $base_spacing * 12; | ||||
|   max-column-spacing: $base_spacing * 12; | ||||
|   -shell-grid-horizontal-item-size: $app_icon_size + $app_icon_padding * 2; | ||||
|   -shell-grid-vertical-item-size: $app_icon_size + $app_icon_padding * 2; | ||||
|   spacing: $base_spacing * 6; | ||||
|  | ||||
|   .overview-icon { | ||||
|     icon-size: $app_icon_size; | ||||
|     StIcon { margin-bottom: $base_margin; } // margin on icon so label isn't close | ||||
|   } | ||||
| } | ||||
|  | ||||
| //.app-display { spacing: 20px; } | ||||
|  | ||||
| /* App Icons */ | ||||
|  | ||||
| $app_grid_fg_color: #fff; | ||||
|  | ||||
| // Outline for low res icons | ||||
| .lowres-icon { | ||||
|   icon-shadow: 0 1px 2px rgba(0,0,0,0.3); | ||||
| } | ||||
|  | ||||
| // Dropshadow for large icons | ||||
| .icon-dropshadow { | ||||
|   icon-shadow: 0 1px 2px rgba(0,0,0,0.4); | ||||
| } | ||||
|  | ||||
| // Icon tiles in the app grid | ||||
| .app-well-app, | ||||
| %app-well-app { | ||||
|   @include overview-icon($app_grid_fg_color); | ||||
| .app-folder { | ||||
|  | ||||
|   .overview-icon.overview-icon-with-label { | ||||
|     padding: 10px 8px 5px 8px; | ||||
|   .overview-icon { | ||||
|     @extend %icon_tile; | ||||
|     color: $app_grid_fg_color !important; | ||||
|   } | ||||
|  | ||||
|     > StBoxLayout { | ||||
|       spacing: $base_spacing; | ||||
|   &:selected { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($osd_bg_color,0.7); | ||||
|       color: $app_grid_fg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:hover, | ||||
|   &:focus, | ||||
|   &:selected { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($osd_fg_color,0.9); | ||||
|       color: $osd_fg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:focus { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($osd_fg_color,0.7  ); | ||||
|       // border-color: $selected_bg_color; | ||||
|       color: $app_grid_fg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:drop { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($selected_bg_color,.15); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:active, | ||||
|   &:checked { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize(darken($osd_bg_color,10%), 0.5); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* App Folders */ | ||||
| .app-well-app.app-folder { | ||||
|   background-color: transparentize($osd_bg_color, 0.8); | ||||
|   border-radius: $base_border_radius + 4px; // same as %icon_tile | ||||
| .app-folder { | ||||
|   .overview-icon { | ||||
|     @extend %icon_tile; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // expanded folder | ||||
| .app-folder-dialog { | ||||
|   border-radius: $modal_radius * 1.5; | ||||
|   border: 1px solid $osd_outer_borders_color; | ||||
|   border-radius: 8px; | ||||
|   spacing: 24px; | ||||
|   background-color: transparentize(darken($osd_bg_color,10%), 0.05); | ||||
|   padding: 12px; | ||||
|  | ||||
|   & .folder-name-container { | ||||
|     padding: 24px 36px 0; | ||||
|     padding: 12px 18px; | ||||
|     spacing: 12px; | ||||
|  | ||||
|     & .folder-name-label, | ||||
|     & .folder-name-entry { | ||||
|       font-size: 18pt; | ||||
|       font-weight: 800; | ||||
|       font-weight: bold; | ||||
|     } | ||||
|  | ||||
|     & .folder-name-entry { width: 300px } | ||||
|  | ||||
|     /* FIXME: this is to keep the label in sync with the entry */ | ||||
|     & .folder-name-label { padding: 5px 7px; color: $osd_fg_color; } | ||||
|     & .folder-name-label { padding: 5px 7px } | ||||
|  | ||||
|     & .edit-folder-button { | ||||
|       @extend %button; | ||||
| @@ -67,24 +117,11 @@ $app_grid_fg_color: #fff; | ||||
|       & > StIcon { icon-size: 16px } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   & .icon-grid { | ||||
|     row-spacing: $base_spacing * 2; | ||||
|     column-spacing: $base_spacing * 5; | ||||
|   } | ||||
|  | ||||
|   & .page-indicators { | ||||
|     margin-bottom: 18px; | ||||
|  | ||||
|     .page-indicator { | ||||
|       padding: 15px 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| .app-folder-dialog-container { | ||||
|   padding: 12px; | ||||
|   width: 620px; | ||||
|   height: 620px; | ||||
|   width: 800px; | ||||
|   height: 600px; | ||||
| } | ||||
|  | ||||
| .app-folder-icon { | ||||
| @@ -130,11 +167,20 @@ $app_grid_fg_color: #fff; | ||||
| } | ||||
|  | ||||
| // Some hacks I don't even know | ||||
| .all-apps { | ||||
| .all-apps, | ||||
| .frequent-apps > StBoxLayout { | ||||
|   // horizontal padding to make sure scrollbars or dash don't overlap content | ||||
|   padding: 0px 88px 10px 88px; | ||||
| } | ||||
|  | ||||
| .app-well-app > .overview-icon.overview-icon-with-label { | ||||
|   padding: 10px 8px 5px 8px; | ||||
|   spacing: $base_spacing; | ||||
| } | ||||
|  | ||||
| // Label when no frequent apps | ||||
| .no-frequent-applications-label { @extend %status_text; } | ||||
|  | ||||
| // shutdown and other actions in the grid | ||||
| .system-action-icon { | ||||
|   background-color: rgba(0,0,0,0.8); | ||||
| @@ -142,3 +188,50 @@ $app_grid_fg_color: #fff; | ||||
|   border-radius: 99px; | ||||
|   icon-size: $app_icon_size * 0.5; | ||||
| } | ||||
|  | ||||
| /* Frequent | All toggle */ | ||||
|  | ||||
| // container | ||||
| .app-view-controls {  | ||||
|   padding-bottom: 32px; | ||||
| } | ||||
|  | ||||
| // buttons | ||||
| .app-view-control {  | ||||
|   padding: $base_padding $base_padding*5; | ||||
|   margin: 0; | ||||
|   background-color: transparentize($osd_bg_color, 0.5); | ||||
|   border-width: 1px; | ||||
|   color: darken($osd_fg_color, 25%); | ||||
|  | ||||
|   &:hover { | ||||
|     background-color: transparentize($osd_bg_color, 0.5) !important; | ||||
|     box-shadow:none !important; | ||||
|     color: darken($osd_fg_color, 25%); | ||||
|   } | ||||
|  | ||||
|   &:active { | ||||
|     box-shadow: none; | ||||
|     background-color: $selected_bg_color !important; | ||||
|     &:hover { | ||||
|       background-color: lighten($selected_bg_color, 11%) !important; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:checked { | ||||
|     background-color: $selected_bg_color !important; | ||||
|     color: $selected_fg_color; | ||||
|     box-shadow: none; | ||||
|     &:active { background-color: darken($selected_bg_color, 4%) !important; } | ||||
|     &:hover { background-color: lighten($selected_bg_color, 7%) !important; } | ||||
|   } | ||||
|  | ||||
|   &:first-child { | ||||
|     border-right-width: 0 !important; | ||||
|     border-radius: $base_border_radius 0 0 $base_border_radius; | ||||
|   } | ||||
|  | ||||
|   &:last-child { | ||||
|     border-radius: 0 $base_border_radius $base_border_radius 0; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -15,11 +15,13 @@ | ||||
|     border: 1px solid transparent; | ||||
| 
 | ||||
|     &:outlined { | ||||
|       background-color: transparentize($osd_fg_color, 0.7); | ||||
|       border: 1px solid darken($borders_color,5%); | ||||
|       background-color: transparentize($osd_fg_color, 0.9); | ||||
|       box-shadow: inset 0 2px 2px 0 rgba(0,0,0,0.4); | ||||
|     } | ||||
| 
 | ||||
|     &:selected { | ||||
|       background-color: transparentize($osd_fg_color, 0.7); | ||||
|       background-color: transparentize($osd_fg_color, 0.9); | ||||
|       color: $osd_fg_color; | ||||
|     } | ||||
|   } | ||||
| @@ -50,16 +52,4 @@ | ||||
|   &:highlighted { | ||||
|     color: $fg_color; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // Input Source Switcher | ||||
| .input-source-switcher-symbol { | ||||
|   font-size: 34pt; | ||||
|   width: 96px; | ||||
|   height: 96px; | ||||
| } | ||||
| 
 | ||||
| // Window cycler highlight | ||||
| .cycler-highlight { | ||||
|   border: 5px solid $selected_bg_color; | ||||
| } | ||||
| } | ||||
| @@ -1,18 +0,0 @@ | ||||
| // Links | ||||
| .shell-link { | ||||
|   color: $link_color; | ||||
|  | ||||
|   &:hover { | ||||
|     color: lighten($link_color, 10%); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Outline for low res icons | ||||
| .lowres-icon { | ||||
|   icon-shadow: 0 1px 2px rgba(black, 0.3); | ||||
| } | ||||
|  | ||||
| // Dropshadow for large icons | ||||
| .icon-dropshadow { | ||||
|   icon-shadow: 0 1px 2px rgba(black, 0.4); | ||||
| } | ||||
| @@ -1,65 +1,87 @@ | ||||
| /* Date/Time Menu */ | ||||
|  | ||||
| .clock-display-box { | ||||
|   spacing: $base_spacing / 2; | ||||
|  | ||||
|   .clock { | ||||
|     padding-left: $base_padding; | ||||
|     padding-right: $base_padding; | ||||
|   } | ||||
| } | ||||
| .clock-display-box { spacing: $base_spacing; } | ||||
|  | ||||
| // overall menu | ||||
| #calendarArea { | ||||
|   padding:0; | ||||
|   margin:0; | ||||
| } | ||||
|  | ||||
| // Calendar menu side column | ||||
| .datemenu-calendar-column { | ||||
|   spacing: $base_spacing; | ||||
|   spacing: 0; | ||||
|   border: 0 solid $bubble_borders_color; | ||||
|   padding: 0 $base_padding * 2; | ||||
|   padding: $base_padding * 2; | ||||
|   padding-bottom: 3em; // account for the notifications clear button | ||||
|   padding-top:0; | ||||
|  | ||||
|   &:ltr {margin-right: $base_margin * 2; border-left-width: 1px; } | ||||
|   &:rtl {margin-left: $base_margin * 2; border-right-width: 1px; } | ||||
|  | ||||
|   // today button (the date) | ||||
|   .datemenu-today-button { | ||||
|     padding: $base_padding * 1.5; | ||||
|     margin: $base_margin; | ||||
|     border: 1px solid transparent; | ||||
|     border-radius: $base_border_radius + 2px; | ||||
|  | ||||
|     &:hover { @include button(hover);} | ||||
|     &:focus { @include button(focus);} | ||||
|  | ||||
|     &:active { | ||||
|       @include button(active); | ||||
|     } | ||||
|  | ||||
|     // weekday label | ||||
|     .day-label { | ||||
|       @include fontsize($base_font_size+1); | ||||
|       font-weight: bold; | ||||
|     } | ||||
|  | ||||
|     // date label | ||||
|     .date-label { | ||||
|       @include fontsize($base_font_size+7); | ||||
|       font-weight: 1000; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // calendar | ||||
|   .calendar { | ||||
|     @extend %notification_bubble; | ||||
|     margin:$base_margin !important; | ||||
|     margin-bottom: $base_padding + $base_margin !important; | ||||
|     padding:$base_padding !important; | ||||
|  | ||||
|     // more below for sub-elements | ||||
|   } | ||||
|  | ||||
|   .datemenu-displays-section { | ||||
|     margin:0; | ||||
|   } | ||||
|  | ||||
|   .datemenu-displays-box { | ||||
|     spacing: $base_spacing; | ||||
|     margin:0; | ||||
|  | ||||
|     // world clocks and weather | ||||
|     .world-clocks-button, | ||||
|     .weather-button { | ||||
|       @extend %notification_bubble; | ||||
|       padding:$base_padding !important; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .events-section-title { | ||||
|   @include notification_bubble($flat: true); | ||||
|   color: desaturate(darken($fg_color,40%), 10%); | ||||
|   font-weight: bold; | ||||
|   border-radius: 4px; | ||||
|   padding: .4em; | ||||
| } | ||||
|  | ||||
| /* today button (the date) */ | ||||
| .datemenu-today-button { | ||||
|   @include notification_bubble($flat: true); | ||||
|   padding: $base_padding * 1.5; | ||||
|  | ||||
|   // weekday label | ||||
|   .day-label { | ||||
|     @include fontsize($base_font_size+1); | ||||
|     font-weight: bold; | ||||
|   } | ||||
|  | ||||
|   // date label | ||||
|   .date-label { | ||||
|     @include fontsize($base_font_size+7); | ||||
|     font-weight: 1000; | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* Calendar */ | ||||
| .calendar { | ||||
|   @include notification_bubble; | ||||
|   padding: $base_padding; | ||||
|  | ||||
|   // month | ||||
|   .calendar-month-label { | ||||
| @@ -110,7 +132,6 @@ | ||||
|       @include fontsize($base_font_size - 4); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .calendar-day { //border collapse hack - see calendar.js | ||||
|     border-width: 0; | ||||
|   } | ||||
| @@ -119,12 +140,8 @@ | ||||
|     border-top-width: 1px; | ||||
|   } | ||||
|  | ||||
|   .calendar-day-left { | ||||
|     border-left-width: 1px; | ||||
|   } | ||||
|  | ||||
|   .calendar-day-left { border-left-width: 1px; } | ||||
|   .calendar-work-day {} | ||||
|  | ||||
|   .calendar-nonwork-day { | ||||
|     color: $insensitive_fg_color; | ||||
|   } | ||||
| @@ -144,24 +161,22 @@ | ||||
|     &:active,&:selected { | ||||
|       background-color: $selected_bg_color; | ||||
|       color: $selected_fg_color; | ||||
|  | ||||
|       &:hover,&:focus { | ||||
|         background-color:lighten($selected_bg_color, 3%); | ||||
|         color: $selected_fg_color; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   } | ||||
|   .calendar-day-with-events { | ||||
|     color: lighten($fg_color,10%); | ||||
|     font-weight: bold; | ||||
|     background-image: url("resource:///org/gnome/shell/theme/calendar-today.svg"); | ||||
|     &.calendar-work-day { | ||||
|        color: lighten($fg_color,10%); | ||||
|        font-weight: bold; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .calendar-other-month-day { | ||||
|     color: transparentize($fg_color ,0.5); | ||||
|     opacity: 0.5; | ||||
|   } | ||||
|  | ||||
|   .calendar-week-number { | ||||
| @@ -177,89 +192,15 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* Events */ | ||||
| .events-button { | ||||
|   @include notification_bubble; | ||||
|   padding: $base_padding * 2; | ||||
|  | ||||
|   .events-box { | ||||
|     spacing: $base_spacing; | ||||
|   } | ||||
|  | ||||
|   .events-list { | ||||
|     spacing: 2 * $base_spacing; | ||||
|   } | ||||
|  | ||||
|   .events-title { | ||||
|     color: desaturate(darken($fg_color,40%), 10%); | ||||
|     font-weight: bold; | ||||
|     margin-bottom: $base_margin; | ||||
|   } | ||||
|  | ||||
|   .event-time { | ||||
|     color: darken($fg_color,20%); | ||||
|     font-feature-settings: "tnum"; | ||||
|     @include fontsize($base_font_size - 1); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* World clocks */ | ||||
| .world-clocks-button { | ||||
|   @include notification_bubble; | ||||
|   padding: $base_padding * 2; | ||||
|  | ||||
|   .world-clocks-grid { | ||||
|     spacing-rows: $base_spacing; | ||||
|     spacing-columns: $base_spacing * 2; | ||||
|   } | ||||
|  | ||||
|   // title | ||||
|   .world-clocks-header { | ||||
|     color: desaturate(darken($fg_color,40%), 10%); | ||||
|     font-weight: bold; | ||||
|   } | ||||
|  | ||||
|   // city label | ||||
|   .world-clocks-city { | ||||
|     color: $fg_color; | ||||
|     @include fontsize($base_font_size); | ||||
|     font-weight: normal; | ||||
|   } | ||||
|  | ||||
|   // timezone time | ||||
|   .world-clocks-time { | ||||
|     font-weight: bold; | ||||
|     color: $fg_color; | ||||
|     font-feature-settings: "tnum"; | ||||
|     @include fontsize($base_font_size); | ||||
|     text-align: right; | ||||
|   } | ||||
|  | ||||
|   // timezone offset label | ||||
|   .world-clocks-timezone { | ||||
|     color: darken($fg_color,20%); | ||||
|     font-feature-settings: "tnum"; | ||||
|     @include fontsize($base_font_size - 1); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* Weather */ | ||||
| .weather-button { | ||||
|   @include notification_bubble; | ||||
|   padding: $base_padding * 2; | ||||
|  | ||||
|   .weather-box { | ||||
|     spacing: $base_spacing + $base_margin; | ||||
|   } | ||||
|  | ||||
|   .weather-header-box { | ||||
|     spacing: $base_spacing; | ||||
|   } | ||||
| .weather-box { | ||||
|   spacing: $base_spacing; | ||||
|   padding:$base_padding; | ||||
|  | ||||
|   .weather-header { | ||||
|     color: desaturate(darken($fg_color,40%), 10%); | ||||
|     font-weight: bold; | ||||
|  | ||||
|     &.location { | ||||
|       font-weight: normal; | ||||
|       @include fontsize($base_font_size - 1); | ||||
| @@ -267,6 +208,7 @@ | ||||
|   } | ||||
|  | ||||
|   .weather-grid { | ||||
|     margin-top: $base_margin; | ||||
|     spacing-rows: $base_spacing; | ||||
|     spacing-columns: $base_spacing * 2; | ||||
|   } | ||||
| @@ -288,3 +230,39 @@ | ||||
|     font-weight: bold; | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* World clocks */ | ||||
| .world-clocks-grid { | ||||
|   padding:$base_padding; | ||||
|   spacing-rows: $base_spacing; | ||||
|   spacing-columns: $base_spacing * 2; | ||||
|  | ||||
|   // title | ||||
|   .world-clocks-header { | ||||
|     color: desaturate(darken($fg_color,40%), 10%); | ||||
|     font-weight: bold; | ||||
|   } | ||||
|  | ||||
|   // city label | ||||
|   .world-clocks-city { | ||||
|     color: $fg_color; | ||||
|     @include fontsize($base_font_size); | ||||
|     font-weight: normal; | ||||
|   } | ||||
|  | ||||
|   // timezone time | ||||
|   .world-clocks-time { | ||||
|     font-weight: bold; | ||||
|     color: $fg_color; | ||||
|     font-feature-settings: "lnum"; | ||||
|     @include fontsize($base_font_size); | ||||
|     text-align: right; | ||||
|   } | ||||
|  | ||||
|   // timezone offset label | ||||
|   .world-clocks-timezone { | ||||
|     color: darken($fg_color,20%); | ||||
|     font-feature-settings: "tnum"; | ||||
|     @include fontsize($base_font_size - 1); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -13,3 +13,12 @@ $ripple_size: 50px; | ||||
|   // just a simple change to the border radius position | ||||
|   &:rtl { border-radius: 0 0 0 $ripple_size + 2px; } | ||||
| } | ||||
|  | ||||
| // Pointer location | ||||
| .ripple-pointer-location { | ||||
|   width: $ripple_size; | ||||
|   height: $ripple_size; | ||||
|   border-radius: $ripple_size * 0.5; // radius equals the size of the box to give us the curve | ||||
|   background-color: lighten(transparentize($selected_bg_color, 0.7), 30%); | ||||
|   box-shadow: 0 0 2px 2px lighten($selected_bg_color, 20%); | ||||
| } | ||||
|   | ||||
| @@ -9,11 +9,12 @@ $dash_border_radius: $modal_radius * 1.5; | ||||
|   @include fontsize($base_font_size - 2); | ||||
|   padding: ($dash_spacing / 2) 0; | ||||
|  | ||||
|   //fixme: can't have non uniform borders :( | ||||
|   border-radius: 0 $dash_border_radius $dash_border_radius 0;  | ||||
|   border-left-width: 0; | ||||
|   &:rtl { | ||||
|   border-left-width: 0 !important; | ||||
|   &:rtl {  | ||||
|     border-radius: $dash_border_radius 0 0 $dash_border_radius; | ||||
|     border-right-width: 0; | ||||
|     border-right-width: 0 !important; | ||||
|   } | ||||
|  | ||||
|   .placeholder { | ||||
| @@ -48,13 +49,36 @@ $dash_border_radius: $modal_radius * 1.5; | ||||
|  | ||||
| // Show apps button | ||||
| .show-apps { | ||||
|   @include overview-icon($osd_fg_color); | ||||
|   color: $osd_fg_color; | ||||
|  | ||||
|   & .overview-icon { | ||||
|     @extend %icon_tile; | ||||
|     color: $osd_fg_color; | ||||
|   } | ||||
|  | ||||
|   &:hover, | ||||
|   &:focus, | ||||
|   &:checked { | ||||
|   &:selected { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($osd_fg_color,0.9); | ||||
|       color: $osd_fg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:drop .overview-icon { | ||||
|     background-color: transparentize($selected_bg_color,.15); | ||||
|   } | ||||
|  | ||||
|   &:active, &:checked { | ||||
|     .overview-icon { | ||||
|       background-color: darken($osd_bg_color,10%); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:checked, &:focus { | ||||
|      .show-apps-icon { | ||||
|       color: $fg_color; | ||||
|       transition-duration: 100ms; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,5 @@ | ||||
| /* Modal Dialogs */ | ||||
|  | ||||
| .headline { | ||||
|   @include fontsize($base_font_size + 1); | ||||
| } | ||||
|  | ||||
| .modal-dialog { | ||||
|   border-radius: $modal_radius; | ||||
|   @extend %bubble_panel; | ||||
| @@ -19,7 +15,12 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .mount-dialog-subject { | ||||
|   @include fontsize($base_font_size + 3); | ||||
| } | ||||
|  | ||||
| /* End Session Dialog */ | ||||
|  | ||||
| .end-session-dialog { | ||||
|   width: 30em; | ||||
|  | ||||
| @@ -37,11 +38,6 @@ | ||||
|     text-align: center; | ||||
|     font-size: 18pt; | ||||
|     font-weight: 800; | ||||
|  | ||||
|     &.leightweight { | ||||
|       font-size: 13pt; | ||||
|       font-weight: 800; | ||||
|     } | ||||
|   } | ||||
|   .message-dialog-description { text-align: center; } | ||||
| } | ||||
| @@ -88,58 +84,76 @@ | ||||
| /* Password or Authentication Dialog */ | ||||
|  | ||||
| .prompt-dialog { | ||||
|   width: 28em; | ||||
|   //this is the width of the entire modal popup | ||||
|   width: 34em; | ||||
|  | ||||
|   .modal-dialog-content-box { | ||||
|     margin-bottom: 24px; | ||||
|   } | ||||
|   .message-dialog-content { spacing: $base_spacing * 4; } | ||||
|   .message-dialog-title { color: lighten($fg_color,15%); } | ||||
| } | ||||
|  | ||||
| .prompt-dialog-password-grid { | ||||
|   spacing-rows: 8px; | ||||
|   spacing-columns: 4px; | ||||
|  | ||||
|   .prompt-dialog-password-entry { | ||||
|     width: auto; | ||||
|  | ||||
|     // 4px (spacing) + 16px (spinner-width) | ||||
|     &:ltr { margin-left: 20px; } | ||||
|     &:rtl { margin-right: 20px; } | ||||
|   } | ||||
| .prompt-dialog-description:rtl { | ||||
|   text-align: right; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-password-layout { | ||||
|   spacing: 8px; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-password-entry { | ||||
|   width: 20em; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-error-label, | ||||
| .prompt-dialog-info-label, | ||||
| .prompt-dialog-null-label { | ||||
|   text-align: center; | ||||
|   @include fontsize($base_font_size - 1); | ||||
| .prompt-dialog-password-box { | ||||
|   spacing: 1em; | ||||
|   padding-bottom: 1em; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-error-label { | ||||
|   @include fontsize($base_font_size - 1); | ||||
|   color: $warning_color; | ||||
|   padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-info-label { | ||||
|   @include fontsize($base_font_size - 1); | ||||
|   padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-null-label { | ||||
|   @include fontsize($base_font_size - 1); | ||||
|   padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-pim-box { | ||||
|   spacing: 1em; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-grid { | ||||
|   spacing-rows: 15px; | ||||
|   spacing-columns: 1em; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-keyfiles-box { | ||||
|   spacing: 1em; | ||||
| } | ||||
|  | ||||
| .prompt-dialog-button.button { | ||||
|   padding: 8px; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Polkit Dialog */ | ||||
|  | ||||
| .polkit-dialog-user-layout { | ||||
|   text-align: center; | ||||
|   spacing: 8px; | ||||
|   margin-bottom: 6px; | ||||
|  | ||||
|   .polkit-dialog-user-icon { | ||||
|     border-radius: 99px; | ||||
|     background-size: contain; | ||||
|   padding-left: 10px; | ||||
|   spacing: 10px; | ||||
|   &:rtl { | ||||
|     padding-left: 0px; | ||||
|     padding-right: 10px; | ||||
|   } | ||||
|   .polkit-dialog-user-root-label { color: $warning_color; } | ||||
| } | ||||
|  | ||||
| .polkit-dialog-user-root-label { | ||||
|   color: $warning_color; | ||||
| } | ||||
|  | ||||
| .polkit-dialog-user-icon { | ||||
|   border-radius: 99px; | ||||
|   background-size: contain; | ||||
|   width: 48px; | ||||
|   height: 48px; | ||||
| } | ||||
|  | ||||
| /* Audio selection dialog */ | ||||
| @@ -166,3 +180,20 @@ | ||||
| .audio-selection-device-icon { | ||||
|   icon-size: $base_icon_size * 4; | ||||
| } | ||||
|  | ||||
| /* Access Dialog */ | ||||
| .access-dialog { | ||||
|   spacing: 30px; | ||||
| } | ||||
|  | ||||
| /* Network Agent Dialog */ | ||||
|  | ||||
| .network-dialog-secret-table { | ||||
|   spacing-rows: 15px; | ||||
|   spacing-columns: 1em; | ||||
| } | ||||
|  | ||||
| .keyring-dialog-control-table { | ||||
|   spacing-rows: 15px; | ||||
|   spacing-columns: 1em; | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,4 @@ | ||||
| // IBus Candidate Popup | ||||
|  | ||||
| .candidate-popup-boxpointer { | ||||
|   @extend .popup-menu-boxpointer; | ||||
| } | ||||
|  | ||||
| .candidate-popup-content { | ||||
|   padding: 0.5em; | ||||
|   spacing: 0.3em; | ||||
|   | ||||
| @@ -10,7 +10,7 @@ $default_key_bg_color: if($variant=='light', darken($osd_bg_color, 11%), lighten | ||||
| // draw keys using button function | ||||
| #keyboard { | ||||
|   background-color: transparentize(if($variant=='light', darken($bg_color, 5%), darken($bg_color, 8%)), 0.1); | ||||
|   box-shadow: inset 0 1px 0 0 $osd_outer_borders_color; | ||||
|   box-shadow: inset 0 1px 0 0 $osd_outer_borders_color !important; | ||||
|  | ||||
|   .page-indicator { | ||||
|     padding: $base_padding; | ||||
| @@ -52,6 +52,10 @@ $default_key_bg_color: if($variant=='light', darken($osd_bg_color, 11%), lighten | ||||
|  | ||||
|   // non-character keys | ||||
|   &.default-key { | ||||
|  | ||||
|     // size of the icon asset | ||||
|     background-size: 24px; | ||||
|  | ||||
|     @include button(normal, $c:$default_key_bg_color); | ||||
|     &:hover, &:checked {@include button(hover, $c: $default_key_bg_color);} | ||||
|     &:active { @include button(active, $c: $default_key_bg_color);} | ||||
| @@ -59,14 +63,19 @@ $default_key_bg_color: if($variant=='light', darken($osd_bg_color, 11%), lighten | ||||
|  | ||||
|   // enter key is suggested-action | ||||
|   &.enter-key { | ||||
|     background-image: url("resource:///org/gnome/shell/theme/key-enter.svg"); | ||||
|  | ||||
|     @include button(normal, $c:$selected_bg_color, $tc:$selected_fg_color); | ||||
|     &:hover, &:checked { @include button(hover, $c: lighten($selected_bg_color, 3%));} | ||||
|     &:active {@include button(active, $c: darken($selected_bg_color, 2%));} | ||||
|   } | ||||
|  | ||||
|   &.shift-key-uppercase { color: $selected_bg_color } | ||||
|  | ||||
|   StIcon { icon-size: 1.125em; } | ||||
|   // key assets | ||||
|   &.shift-key-lowercase {background-image: url("resource:///org/gnome/shell/theme/key-shift.svg");} | ||||
|   &.shift-key-uppercase {background-image: url("resource:///org/gnome/shell/theme/key-shift-uppercase.svg");} | ||||
|   &.shift-key-uppercase:latched {background-image: url("resource:///org/gnome/shell/theme/key-shift-latched-uppercase.svg");} | ||||
|   &.hide-key {background-image: url("resource:///org/gnome/shell/theme/key-hide.svg");} | ||||
|   &.layout-key {background-image: url("resource:///org/gnome/shell/theme/key-layout.svg");} | ||||
| } | ||||
|  | ||||
| // long press on a key popup | ||||
| @@ -112,4 +121,4 @@ $default_key_bg_color: if($variant=='light', darken($osd_bg_color, 11%), lighten | ||||
|   @include fontsize($base_font_size + 3); | ||||
|   spacing: 12px; | ||||
|   min-height: 20pt; | ||||
| } | ||||
| } | ||||
| @@ -68,30 +68,12 @@ | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .cancel-button, | ||||
|   .switch-user-button, | ||||
|   .login-dialog-session-list-button { | ||||
|     padding: 0; | ||||
|     border-radius: 99px; | ||||
|     width: $base_icon_size * 2; | ||||
|     height: $base_icon_size * 2; | ||||
|     border-color: transparentize($bg_color,0.7); | ||||
|     background-color: transparentize($bg_color,0.7); | ||||
|  | ||||
|     StIcon { icon-size: $base_icon_size; } | ||||
|   } | ||||
|  | ||||
|   .caps-lock-warning-label, | ||||
|   .login-dialog-message-warning { | ||||
|     color: $osd_fg_color; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .login-dialog-logo-bin { padding: 24px 0px; } | ||||
| .login-dialog-banner { color: darken($osd_fg_color,10%); } | ||||
| .login-dialog-button-box { width: 23em; spacing: 5px; } | ||||
| .login-dialog-message { text-align: center; } | ||||
| .login-dialog-button-box { spacing: 5px; } | ||||
| .login-dialog-message-warning { color: $warning_color; } | ||||
| .login-dialog-message-hint { padding-top: 0; padding-bottom: 20px; } | ||||
| .login-dialog-user-selection-box { padding: 100px 0px; } | ||||
| .login-dialog-not-listed-label { | ||||
| @@ -118,7 +100,7 @@ | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item { | ||||
|   border-radius: $base_border_radius + 4px; | ||||
|   border-radius: 5px; | ||||
|   padding: 6px; | ||||
|   color: darken($osd_fg_color,30%); | ||||
|   &:ltr .user-widget { padding-right: 1em; } | ||||
| @@ -131,25 +113,18 @@ | ||||
|   &:focus .login-dialog-timed-login-indicator { background-color: $selected_fg_color; } | ||||
| } | ||||
|  | ||||
| .login-dialog-username, | ||||
| .user-widget-label { | ||||
|   color: $osd_fg_color; | ||||
| } | ||||
|  | ||||
| .user-widget.horizontal .user-widget-label { | ||||
|   @include fontsize($base_font_size + 2); | ||||
|   font-weight: bold; | ||||
|   text-align: left; | ||||
|   padding-left: 15px; | ||||
|  | ||||
|   &:ltr { padding-left: 14px; } | ||||
|   &:rtl { padding-right: 14px; } | ||||
| } | ||||
|  | ||||
| .user-widget.vertical .user-widget-label { | ||||
|   @include fontsize($base_font_size + 5); | ||||
|   text-align: center; | ||||
|   font-weight: normal; | ||||
|   padding-top: 16px; | ||||
| .user-widget-label { | ||||
|   &:ltr { padding-left: 14px; } | ||||
|   &:rtl { padding-right: 14px; } | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-layout { | ||||
| @@ -159,12 +134,18 @@ | ||||
|   width: 23em; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-entry { | ||||
|   height: 1.5em; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-label { | ||||
|   color: darken($osd_fg_color, 20%); | ||||
|   @include fontsize($base_font_size + 1); | ||||
|   padding-top: 1em; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button StIcon { | ||||
|   icon-size: 1.25em; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button { | ||||
|   color: darken($osd_fg_color,30%); | ||||
|   &:hover,&:focus { color: $osd_fg_color; } | ||||
|   &:active { color: darken($osd_fg_color, 50%); } | ||||
| } | ||||
| @@ -1,20 +1,18 @@ | ||||
| /* Looking Glass */ | ||||
|  | ||||
| $text_fg_color: #ccc; | ||||
|  | ||||
| // Dialog | ||||
| #LookingGlassDialog { | ||||
|   background-color: $osd_bg_color; | ||||
|   spacing: $base_spacing; | ||||
|   padding: 4px; | ||||
|   border: 1px solid transparentize($osd_fg_color, 0.8); | ||||
|   border-radius: $base_border_radius; | ||||
|   color: $osd_fg_color; | ||||
|   border: 2px solid transparentize($osd_fg_color, 0.8); | ||||
|   border-top-width:0; | ||||
|   border-radius: 0 0 $base_border_radius $base_border_radius; | ||||
|  | ||||
|   & > #Toolbar { | ||||
|     border: none; | ||||
|     border-radius: $base_border_radius; | ||||
|     background-color: $osd_bg_color; | ||||
|     background-color: darken($osd_bg_color, 10%); | ||||
|   } | ||||
|  | ||||
|   .labels { spacing: $base_spacing; } | ||||
| @@ -22,18 +20,19 @@ $text_fg_color: #ccc; | ||||
|     -natural-hpadding: $base_padding * 2; | ||||
|     -minimum-hpadding: 6px; | ||||
|     font-weight: bold; | ||||
|     color: darken($osd_fg_color, 15%); | ||||
|     color: $fg_color; | ||||
|     transition-duration: 100ms; | ||||
|     padding-left: .3em; | ||||
|     padding-right: .3em; | ||||
|     border-bottom-width: 2px; | ||||
|     &:hover { | ||||
|       color: $osd_fg_color; | ||||
|       color: white; | ||||
|       text-shadow: black 0px 2px 2px; | ||||
|     } | ||||
|     &:selected { | ||||
|       border-bottom-width: 2px; | ||||
|       box-shadow: inset 0 -2px 0 0 lighten($selected_bg_color, 5%); | ||||
|       color: $osd_fg_color; | ||||
|       border-color: lighten($selected_bg_color,5%); | ||||
|       color: white; | ||||
|       text-shadow: black 0px 2px 2px; | ||||
|     } | ||||
|   } | ||||
|   StBoxLayout#EvalBox { padding: 4px; spacing: $base_spacing; } | ||||
| @@ -42,22 +41,12 @@ $text_fg_color: #ccc; | ||||
|  | ||||
| .lg-dialog { | ||||
|   StEntry { | ||||
|     background-color: transparentize(lighten($osd_bg_color, 5%), 0.4); | ||||
|     color: $osd_fg_color; | ||||
|     border-color: transparentize($osd_fg_color, 0.8); | ||||
|     min-height: 22px; | ||||
|     selection-background-color: $selected_bg_color; | ||||
|     selected-color: $selected_fg_color; | ||||
|     selection-background-color: #bbbbbb; | ||||
|     selected-color: $osd_bg_color; | ||||
|   } | ||||
|   .shell-link { | ||||
|     color: $link_color; | ||||
|     &:hover { color: lighten($link_color, 10%); } | ||||
|     &:active { color: darken($link_color, 10%); } | ||||
|    } | ||||
|   .actor-link { | ||||
|     color: $text_fg_color; | ||||
|     &:hover { color: lighten($text_fg_color, 20%); } | ||||
|     &:active { color: darken($text_fg_color, 20%); } | ||||
|     color: #999999; | ||||
|     &:hover { color: #dddddd; } | ||||
|    } | ||||
| } | ||||
|  | ||||
| @@ -71,7 +60,7 @@ $text_fg_color: #ccc; | ||||
| } | ||||
|  | ||||
| .lg-obj-inspector-button { | ||||
|     border: 1px solid $osd_borders_color; | ||||
|     border: 1px solid gray; | ||||
|     padding: 4px; | ||||
|     border-radius: $base_border_radius; | ||||
|     &:hover { border: 1px solid #ffffff; } | ||||
| @@ -86,8 +75,7 @@ $text_fg_color: #ccc; | ||||
| } | ||||
|  | ||||
| .lg-extension { | ||||
|     border: 1px solid lighten($osd_borders_color, 5%); | ||||
|     background-color: lighten($osd_bg_color, 5%); | ||||
|     border: 1px solid $osd_borders_color; | ||||
|     border-radius: $base_border_radius; | ||||
|     padding: 4px; | ||||
| } | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
|  | ||||
| .message-list-sections { | ||||
|   spacing: $base_spacing; | ||||
|   margin: 0 $base_margin * 4; // to account for scrollbar | ||||
|   margin: $base_margin * 4; // to account for scrollbar | ||||
| } | ||||
|  | ||||
| .message-list-section, | ||||
| @@ -19,61 +19,40 @@ | ||||
|   spacing: $base_spacing; | ||||
| } | ||||
|  | ||||
| .message-list-section-list { | ||||
|   &:ltr {padding:0;} | ||||
|   &:rtl {padding:0;} | ||||
| } | ||||
|  | ||||
| // do-not-disturb + clear button | ||||
| .message-list-controls { | ||||
|   margin: ($base_margin * 2) ($base_margin * 4) 0; | ||||
|   // NOTE: remove the padding if notification_bubble could remove margin for drop shadow | ||||
|   padding: $base_margin; | ||||
|   spacing: $base_spacing * 2; | ||||
|   margin: $base_margin $base_margin*2; | ||||
|   spacing: $base_spacing; | ||||
| } | ||||
|  | ||||
| // message bubbles | ||||
| .message { | ||||
|   @include notification_bubble; | ||||
|   @extend %notification_bubble; | ||||
|  | ||||
|   // icon container | ||||
|   .message-icon-bin { | ||||
|     padding: ($base_padding * 3) 0 ($base_padding * 3) ($base_padding * 2); | ||||
|  | ||||
|     &:rtl { | ||||
|       padding: ($base_padding * 3) ($base_padding * 2) ($base_padding * 3) 0; | ||||
|     } | ||||
|  | ||||
|     // icon size and color | ||||
|     > StIcon { | ||||
|       icon-size: $base_icon_size*2; // 32px | ||||
|       -st-icon-style: symbolic; | ||||
|     } | ||||
|  | ||||
|     // fallback | ||||
|     > .fallback-app-icon { | ||||
|       width: $base_icon_size; | ||||
|       height: $base_icon_size; | ||||
|     } | ||||
|   // title | ||||
|   .message-title { | ||||
|     color: $fg_color; | ||||
|     font-weight: bold; | ||||
|     margin-bottom:4px; | ||||
|   } | ||||
|  | ||||
|   // content | ||||
|   .message-content { | ||||
|     padding: $base_padding + $base_margin * 2; | ||||
|     spacing: 4px; | ||||
|   } | ||||
|  | ||||
|   // title | ||||
|   .message-title { | ||||
|     font-weight: bold; | ||||
|   } | ||||
|  | ||||
|   // secondary container in title box | ||||
|   .message-secondary-bin { | ||||
|     padding: 0 $base_margin * 2; | ||||
|  | ||||
|     // notification time stamp | ||||
|     > .event-time { | ||||
|       color: transparentize($fg_color, 0.5); | ||||
|       @include fontsize($base_font_size - 2); | ||||
|       text-align: right; | ||||
|       /* HACK: the label should be baseline-aligned with a 1em label, fake this with some bottom padding */ | ||||
|       padding-bottom: 0.13em; | ||||
|     color: darken($fg_color, 10%); | ||||
|     padding: $base_padding 0; | ||||
|     margin:$base_margin * 2; | ||||
|     &:ltr { | ||||
|       margin-left: $base_margin; | ||||
|       padding-right:$base_padding; | ||||
|     } | ||||
|     &:rtl { | ||||
|       margin-right: $base_margin; | ||||
|       padding-left:$base_padding; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -84,15 +63,48 @@ | ||||
|     &:active { color: if($variant=='light', lighten($fg_color, 40%), darken($fg_color, 20%)); } | ||||
|   } | ||||
|  | ||||
|   // body | ||||
|   .message-body { | ||||
|     color: darken($fg_color, 10%); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // URLs in messages | ||||
| .url-highlighter { | ||||
|   link-color: $link_color; | ||||
|   // icon container | ||||
|   .message-icon-bin { | ||||
|     padding: $base_padding; | ||||
|     margin:$base_padding 0; | ||||
|  | ||||
|     &:rtl { | ||||
|       // padding: $base_padding; | ||||
|     } | ||||
|  | ||||
|     // icon size and color | ||||
|     > StIcon { | ||||
|       color: $fg_color; | ||||
|       icon-size: $base_icon_size*2; // 32px | ||||
|       -st-icon-style: symbolic; | ||||
|  | ||||
|       padding:0; | ||||
|       margin:$base_padding; | ||||
|     } | ||||
|  | ||||
|     // fallback | ||||
|     > .fallback-app-icon { | ||||
|       width: $base_icon_size; | ||||
|       height: $base_icon_size; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // secondary container in title box | ||||
|   .message-secondary-bin { | ||||
|     padding: 0; | ||||
|  | ||||
|     // notification time stamp | ||||
|     > .event-time { | ||||
|       color: transparentize($fg_color, 0.5); | ||||
|       @include fontsize($base_font_size - 2); | ||||
|       text-align: right; | ||||
|       margin: 0 $base_margin * 2; | ||||
|       /* HACK: the label should be baseline-aligned with a 1em label, fake this with some bottom padding */ | ||||
|       padding-bottom: $base_padding; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
| /* Media Controls */ | ||||
| @@ -113,9 +125,9 @@ | ||||
|  | ||||
|   &:insensitive { color: darken($fg_color,40%); } | ||||
|    | ||||
|   // fix border-radius for last button | ||||
|   &:last-child:ltr { border-radius: 0 $base_border_radius+2 $base_border_radius+2 0; } | ||||
|   &:last-child:rtl { border-radius: $base_border_radius+2 0 0 $base_border_radius+2; } | ||||
|   // fix border-radius for last button on hover | ||||
|   &:last-child:ltr { &:hover {border-radius:  0 $base_border_radius+2 $base_border_radius+2 0;} } | ||||
|   &:last-child:rtl { &:hover {border-radius: $base_border_radius+2 0 0 $base_border_radius+2;} } | ||||
| } | ||||
|  | ||||
| // album-art | ||||
| @@ -130,5 +142,6 @@ | ||||
|     border: 1px solid transparent; | ||||
|     border-radius: $base_border_radius; | ||||
|     icon-size: $base_icon_size * 2 !important; | ||||
|     padding: $base_padding * 2; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,32 @@ | ||||
| // Links/URLs | ||||
| .shell-link { | ||||
|   color: $link_color; | ||||
|   &:hover { color: lighten($link_color,10%); } | ||||
| } | ||||
|  | ||||
| .url-highlighter { link-color: $link_color; } | ||||
|  | ||||
| // Rubberband for select-area screenshots | ||||
| .select-area-rubberband { | ||||
|   background-color: transparentize($selected_bg_color,0.7); | ||||
|   border: 1px solid $selected_bg_color; | ||||
| } | ||||
|  | ||||
| // Pointer accessibility notifications | ||||
| .pie-timer { | ||||
|   width: 60px; | ||||
|   height: 60px; | ||||
|   -pie-border-width: 3px; | ||||
|   -pie-border-color: $selected_bg_color; | ||||
|   -pie-background-color: lighten(transparentize($selected_bg_color, 0.7), 40%); | ||||
| } | ||||
|  | ||||
| // Screen zoom/Magnifier | ||||
| .magnifier-zoom-region { | ||||
|   border: 2px solid $selected_bg_color; | ||||
|   &.full-screen { border-width: 0; } | ||||
| } | ||||
|  | ||||
| // User icon | ||||
| .user-icon { | ||||
|   background-size: contain; | ||||
| @@ -12,33 +35,22 @@ | ||||
|   &:hover { | ||||
|     color: lighten($osd_fg_color,30%); | ||||
|   } | ||||
|  | ||||
|   & StIcon { | ||||
|     background-color: transparentize($osd_fg_color,0.95); | ||||
|     border-radius: 99px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .user-widget.vertical .user-icon { | ||||
|   icon-size: $base_icon_size * 6; // 128px | ||||
|  | ||||
|   & StIcon { | ||||
|     padding: $base_padding * 3 + 2px; // 20px | ||||
|     padding-top: $base_padding * 3; // 18 px | ||||
|     padding-bottom: $base_padding * 3 + 4px; // 22px | ||||
|     width: $base_icon_size * 5.5; height:  $base_icon_size * 5.5; // 88px; | ||||
|   } | ||||
| // Input Source Switcher | ||||
| .input-source-switcher-symbol { | ||||
|   font-size: 34pt; | ||||
|   width: 96px; | ||||
|   height: 96px; | ||||
| } | ||||
|  | ||||
| .user-widget.horizontal .user-icon { | ||||
|   icon-size: $base_icon_size * 4; // 64px | ||||
|  | ||||
|   & StIcon { | ||||
|     padding: $base_padding * 2 ; // 12px | ||||
|     width: $base_icon_size * 2.5; height:  $base_icon_size * 2.5; // 40px; | ||||
|   } | ||||
| // Window cycler highlight | ||||
| .cycler-highlight { | ||||
|   border: 5px solid $selected_bg_color; | ||||
| } | ||||
|  | ||||
| // Text | ||||
| .headline { @include fontsize($base_font_size + 1); } | ||||
| .lightbox { background-color: black; } | ||||
| .flashspot { background-color: white; } | ||||
|  | ||||
| @@ -48,8 +60,8 @@ | ||||
|  | ||||
| // Caps-lock warning | ||||
| .caps-lock-warning-label { | ||||
|   text-align: center; | ||||
|   padding-bottom: 8px; | ||||
|   padding-left: 6.2em; | ||||
|   @include fontsize($base_font_size - 1); | ||||
|   color: $warning_color; | ||||
| } | ||||
|   | ||||
| @@ -7,8 +7,39 @@ $notification_banner_width: 34em; | ||||
| .notification-banner { | ||||
|   min-height: $notification_banner_height; | ||||
|   width: $notification_banner_width; | ||||
|   @include fontsize($base_font_size); | ||||
|   margin: $base_margin; | ||||
|   border-radius: $modal_radius; | ||||
|  | ||||
|   .message-title { color: $fg_color } | ||||
|   .message-content { color: $fg_color; } | ||||
|  | ||||
|   &:hover { background: $bg_color; } | ||||
|   &, &:focus, &:active { | ||||
|     background-color: $bg_color; | ||||
|     .message-title { color: $fg_color } | ||||
|     .message-content { color: $fg_color; } | ||||
|   } | ||||
|  | ||||
|   // icon | ||||
|   .message-icon-bin > StIcon { | ||||
|     icon-size: $base_icon_size * 2; | ||||
|     color: $fg_color; | ||||
|   } | ||||
|  | ||||
|   .notification-icon { | ||||
|     padding: 5px; | ||||
|   } | ||||
|  | ||||
|   .notification-content { | ||||
|     padding: 5px; | ||||
|     spacing: 5px; | ||||
|   } | ||||
|  | ||||
|   .secondary-icon { icon-size: $base_icon_size; } | ||||
|  | ||||
|   .notification-actions { | ||||
|     padding-top: 0; | ||||
|     spacing: 0; | ||||
|   } | ||||
|  | ||||
| @@ -32,6 +63,8 @@ $notification_banner_width: 34em; | ||||
|   border-radius: 0.9em; // should be 0.8 but whatever; wish I could do 50%; | ||||
| } | ||||
|  | ||||
| .secondary-icon { icon-size: $base_icon_size; } | ||||
|  | ||||
| // chat bubbles | ||||
| .chat-body { spacing: 5px; } | ||||
| .chat-response { margin: 5px; } | ||||
| @@ -54,4 +87,4 @@ $notification_banner_width: 34em; | ||||
|   font-weight: bold; | ||||
|   color: lighten($fg_color,18%); | ||||
|   &:rtl { padding-left: 0; padding-right: 4px; } | ||||
| } | ||||
| } | ||||
| @@ -2,8 +2,8 @@ | ||||
| // a.k.a. the panel | ||||
|  | ||||
| $panel_corner_radius: $base_border_radius+1; | ||||
| $panel_bg_color: #000; | ||||
| $panel_fg_color: #ccc; | ||||
| $panel_bg_color: if($variant == 'light', rgba(0,0,0,0.9), #000); | ||||
| $panel_fg_color: if($variant == 'light', darken($fg_color, 15%), darken($fg_color, 10%)); | ||||
| $panel_height: 1.86em; | ||||
|  | ||||
|  | ||||
| @@ -15,7 +15,8 @@ $panel_height: 1.86em; | ||||
|  | ||||
|   // transparent panel on lock & login screens | ||||
|   &.unlock-screen, | ||||
|   &.login-screen { | ||||
|   &.login-screen, | ||||
|   &.lock-screen { | ||||
|     background-color: transparent; | ||||
|  | ||||
|     .panel-corner { | ||||
| @@ -54,6 +55,9 @@ $panel_height: 1.86em; | ||||
|     } | ||||
|  | ||||
|     &:active, &:overview, &:focus, &:checked { | ||||
|       background-color: $panel_bg_color; // Trick due to St limitations. It needs a background to draw a box-shadow | ||||
|       box-shadow: inset 0 -2px 0 0 lighten($selected_bg_color,5%); | ||||
|  | ||||
|       color: lighten($panel_fg_color, 20%); | ||||
|     } | ||||
|  | ||||
| @@ -73,32 +77,13 @@ $panel_height: 1.86em; | ||||
|  | ||||
|     // lock & login screen styles | ||||
|     .unlock-screen &, | ||||
|     .login-screen & { | ||||
|     .login-screen &, | ||||
|     .lock-screen & { | ||||
|       color: lighten($fg_color, 10%); | ||||
|       &:focus, &:hover, &:active { color: lighten($fg_color, 10%); } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .panel-button { | ||||
|     &:active, &:overview, &:focus, &:checked { | ||||
|       // Trick due to St limitations. It needs a background to draw a box-shadow | ||||
|       background-color: rgba(0, 0, 0, 0.01); | ||||
|       box-shadow: inset 0 -2px 0 0 lighten($selected_bg_color,5%); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .panel-button.clock-display { | ||||
|     // Move highlight from .panel-button to .clock | ||||
|     &:active, &:overview, &:focus, &:checked { | ||||
|       box-shadow: none; | ||||
|  | ||||
|       .clock { | ||||
|         background-color: rgba(0, 0, 0, 0.01); | ||||
|         box-shadow: inset 0 -2px 0 0 lighten($selected_bg_color,5%); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .panel-status-indicators-box, | ||||
|   .panel-status-menu-box { | ||||
|     spacing: 2px; | ||||
|   | ||||
| @@ -3,7 +3,8 @@ | ||||
| $popover_arrow_height: 12px; | ||||
|  | ||||
| //.the popover itself | ||||
| .popup-menu-boxpointer { | ||||
| .popup-menu-boxpointer, | ||||
| .candidate-popup-boxpointer { | ||||
|   -arrow-border-radius: $base_border_radius+4; | ||||
|   -arrow-background-color: $bg_color; | ||||
|   -arrow-border-width: 1px; | ||||
| @@ -82,20 +83,11 @@ $popover_arrow_height: 12px; | ||||
|  | ||||
| // separator | ||||
| .popup-separator-menu-item { | ||||
|   padding: 0; | ||||
|  | ||||
|   .popup-separator-menu-item-separator { | ||||
|     //-margin-horizontal: 24px; | ||||
|     height: 1px; //not really the whole box | ||||
|     margin: 6px 64px; | ||||
|     background-color: lighten($borders_color, 2%); | ||||
|     .popup-sub-menu & { //submenu separators | ||||
|       margin: 0 64px 0 32px; | ||||
|       @if $variant == 'dark' { | ||||
|         background-color: lighten($bg_color,10%); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   //-margin-horizontal: 24px; | ||||
|   height: 1px; //not really the whole box | ||||
|   margin: 6px 64px; | ||||
|   background-color: lighten($borders_color, 2%); | ||||
|   border: none !important; | ||||
| } | ||||
|  | ||||
| // desktop background menu | ||||
| @@ -126,4 +118,4 @@ $popover_arrow_height: 12px; | ||||
|       margin-right: $base_icon_size; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| } | ||||
| @@ -1,64 +1,68 @@ | ||||
| /* Screen Shield */ | ||||
|  | ||||
| .unlock-dialog-clock { | ||||
|   color: white; | ||||
|   font-weight: 300; | ||||
|   text-align: center; | ||||
|   spacing: 24px; | ||||
|   padding-bottom: 2.5em; | ||||
| $_screenshield_shadow: 0px 0px 6px rgba(0, 0, 0, 0.726); | ||||
|  | ||||
| .screen-shield-arrows { | ||||
|   padding-bottom: 3em; | ||||
| } | ||||
|  | ||||
| .unlock-dialog-clock-time { | ||||
|   font-size: 64pt; | ||||
|   padding-top: 42px; | ||||
| .screen-shield-arrows Gjs_Arrow { | ||||
|   color: white; | ||||
|   width: 80px; | ||||
|   height: 48px; | ||||
|   -arrow-thickness: 12px; | ||||
|   -arrow-shadow: $_screenshield_shadow; | ||||
| } | ||||
|  | ||||
| .screen-shield-clock { | ||||
|   color: white; | ||||
|   text-shadow: $_screenshield_shadow; | ||||
|   font-weight: bold; | ||||
|   text-align: center; | ||||
|   padding-bottom: 1.5em; | ||||
| } | ||||
|  | ||||
| .screen-shield-clock-time { | ||||
|   font-size: 72pt; | ||||
|   text-shadow: $_screenshield_shadow; | ||||
|   font-feature-settings: "tnum"; | ||||
| } | ||||
|  | ||||
| .unlock-dialog-clock-date { | ||||
|   font-size: 16pt; | ||||
| .screen-shield-clock-date {  | ||||
|   font-size: 28pt; | ||||
|   font-weight: normal; | ||||
| } | ||||
|  | ||||
| .unlock-dialog-clock-hint { | ||||
|   font-weight: normal; | ||||
|   padding-top: 48px; | ||||
| } | ||||
|  | ||||
| .unlock-dialog-notifications-container { | ||||
|   margin: 12px 0; | ||||
| .screen-shield-notifications-container { | ||||
|   spacing: 6px; | ||||
|   width: 23em; | ||||
|   width: 30em; | ||||
|   background-color: transparent; | ||||
|   max-height: 500px; | ||||
|   .summary-notification-stack-scrollview { | ||||
|     padding-top: 0; | ||||
|     padding-bottom: 0; | ||||
|   } | ||||
|  | ||||
|   .notification, | ||||
|   .unlock-dialog-notification-source { | ||||
|   .screen-shield-notification-source { | ||||
|     padding: 12px 6px; | ||||
|     border: none; | ||||
|     background-color: transparentize($osd_bg_color,0.7); | ||||
|     border: 1px solid $osd_outer_borders_color; | ||||
|     background-color: transparentize($osd_bg_color,0.5); | ||||
|     color: $osd_fg_color; | ||||
|     border-radius: $modal_radius; | ||||
|  | ||||
|     &.critical { background-color: transparentize($osd_bg_color,0.1) } | ||||
|     border-radius: 4px; | ||||
|   } | ||||
|   .notification { margin-right: 15px; } //compensate for space allocated to the scrollbar | ||||
| } | ||||
|  | ||||
| .unlock-dialog-notification-label { | ||||
|  | ||||
| .screen-shield-notification-label { | ||||
|   font-weight: bold; | ||||
|   padding: 0px 0px 0px 12px; | ||||
| } | ||||
|  | ||||
| .unlock-dialog-notification-count-text { | ||||
|   weight: bold; | ||||
|   padding: 0 6px; | ||||
|   color: $osd_bg_color; | ||||
|   background-color: transparentize($osd_fg_color, 0.7); | ||||
|   border-radius: 99px; | ||||
|   margin-right: 12px; | ||||
| .screen-shield-notification-count-text { padding: 0px 0px 0px 12px; } | ||||
|  | ||||
| } | ||||
| #panel.lock-screen { background-color: transparentize($osd_bg_color, 0.5); } | ||||
|  | ||||
| .screen-shield-background { //just the shadow, really | ||||
|   background: black; | ||||
| @@ -66,13 +70,14 @@ | ||||
| } | ||||
|  | ||||
| #lockDialogGroup { | ||||
|   background-color: lighten(#2e3436, 8%); | ||||
|   background: lighten(#2e3436, 8%) url(resource:///org/gnome/shell/theme/noise-texture.png); | ||||
|   background-repeat: repeat; | ||||
| } | ||||
|  | ||||
| #unlockDialogNotifications { | ||||
| #screenShieldNotifications { | ||||
|   StButton#vhandle, StButton#hhandle { | ||||
|     background-color: transparentize($bg_color,0.7); | ||||
|     &:hover, &:focus { background-color: transparentize($bg_color,0.5); } | ||||
|     &:active { background-color: transparentize($selected_bg_color,0.5); } | ||||
|   } | ||||
| } | ||||
| } | ||||
| @@ -3,16 +3,19 @@ | ||||
| // search overview container | ||||
| #searchResultsContent { | ||||
|   max-width: 1024px; | ||||
|   spacing: $base_margin * 2; | ||||
| } | ||||
|  | ||||
| // search results sections "the boxes" | ||||
| .search-section { | ||||
|   // This should be equal to #searchResultsContent spacing | ||||
|   spacing: $base_margin * 2; | ||||
|  | ||||
|   padding:0 !important; | ||||
|   margin:0 !important; | ||||
|   background-color:transparent; | ||||
|   box-shadow:none; | ||||
|   border:none; | ||||
|   // separator | ||||
|   .search-section-separator {  | ||||
|     // margin-top: $base_padding * 2; | ||||
|     // height: 1px; | ||||
|     // background-color: $osd_outer_borders_color; | ||||
|     height: 0; | ||||
| @@ -29,24 +32,8 @@ | ||||
|   text-shadow: 0 1px if($variant == 'light', rgba(255,255,255,0.2), rgba(0,0,0,0.2)); | ||||
|   color: $osd_fg_color; | ||||
|   padding: $base_padding * 3; | ||||
|   // This is the space between the provider icon and the results container | ||||
|   spacing: $base_margin * 2; | ||||
| } | ||||
|  | ||||
| %search-section-content-item { | ||||
|   @extend %icon_tile; | ||||
|  | ||||
|   &:focus, | ||||
|   &:hover, | ||||
|   &:selected { | ||||
|     background-color: transparentize($osd_fg_color, .9); | ||||
|     transition-duration: 200ms; | ||||
|   } | ||||
|  | ||||
|   &:active, | ||||
|   &:checked { | ||||
|     background-color: transparentize(darken($osd_bg_color, 10%), .1); | ||||
|   } | ||||
|   margin: $base_margin 0; | ||||
|   spacing: 0; | ||||
| } | ||||
|  | ||||
| // "no results" text | ||||
| @@ -54,18 +41,56 @@ | ||||
|   @extend %status_text; | ||||
| } | ||||
|  | ||||
| .grid-search-results { | ||||
|   spacing: $base_spacing * 6; | ||||
| } | ||||
|  | ||||
| // Search results with icons | ||||
| .grid-search-result { | ||||
|   @extend %app-well-app; | ||||
|   > .overview-icon { | ||||
|     @extend %icon_tile; | ||||
|     color: $osd_fg_color; | ||||
|   } | ||||
|  | ||||
|   > .overview-icon.overview-icon-with-label { | ||||
|     padding: 10px 8px 5px 8px; | ||||
|     spacing: $base_spacing; | ||||
|   } | ||||
|  | ||||
|   &:hover, | ||||
|   &:focus, | ||||
|   &:selected { | ||||
|     .overview-icon { | ||||
|       background-color: transparentize($osd_bg_color,0.8); | ||||
|       color: $osd_fg_color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &:drop .overview-icon { | ||||
|     background-color: transparentize($selected_bg_color,.15); | ||||
|   } | ||||
|  | ||||
|   &:active .overview-icon, | ||||
|   &:checked .overview-icon { | ||||
|     background-color: transparentize(darken($osd_bg_color,10%), 0.5); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // search result provider | ||||
| .search-provider-icon { | ||||
|   @extend %search-section-content-item; | ||||
|   @extend %icon_tile; | ||||
|  | ||||
|   padding: $base_padding; | ||||
|   spacing: 0; | ||||
|   margin-right: $base_margin * 2; | ||||
|  | ||||
|   &:focus, | ||||
|   &:selected, | ||||
|   &:hover { | ||||
|     background-color: transparentize($osd_fg_color,.9); | ||||
|     transition-duration: 200ms; | ||||
|   } | ||||
|  | ||||
|   &:active, | ||||
|   &:checked { | ||||
|     background-color: transparentize(darken($osd_bg_color,10%),.1); | ||||
|   } | ||||
|  | ||||
|   // content | ||||
|   .list-search-provider-content { | ||||
| @@ -88,16 +113,34 @@ | ||||
|  | ||||
| // search result listitem | ||||
| .list-search-result { | ||||
|   @extend %search-section-content-item; | ||||
|   @extend %icon_tile; | ||||
|   spacing: 0; | ||||
|   padding: $base_padding; | ||||
|   color: $osd_fg_color; | ||||
|  | ||||
|   border-radius: $base_border_radius + 2px !important; | ||||
|  | ||||
|   &:focus, | ||||
|   &:selected, | ||||
|   &:hover { | ||||
|     background-color: transparentize($osd_fg_color,.9); | ||||
|     transition-duration: 200ms; | ||||
|   } | ||||
|   &:active, | ||||
|   &:checked { | ||||
|     background-color: transparentize(darken($osd_bg_color,10%),.1); | ||||
|   } | ||||
|  | ||||
|   // content | ||||
|   .list-search-result-content { | ||||
|     spacing: $base_padding; | ||||
|     spacing: 0; | ||||
|   } | ||||
|  | ||||
|   // list item title (with leading icon) | ||||
|   // list item title | ||||
|   .list-search-result-title { | ||||
|     color: $osd_fg_color; | ||||
|     spacing: $base_spacing * 2; | ||||
|     padding-right: $base_padding; | ||||
|     // font-weight: bold; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,13 @@ | ||||
| /* Window Picker */ | ||||
|  | ||||
| $window_picker_spacing: $base_spacing * 2; // 16px | ||||
| $window_picker_padding: $base_padding * 2; // 16px | ||||
| $window_picker_spacing: $base_spacing * 8; // 48px | ||||
| $window_picker_padding: $base_padding * 10; // 60px | ||||
|  | ||||
| $window_thumbnail_border_color:transparentize($selected_fg_color, 0.65); | ||||
|  | ||||
| $window_close_button_size: 24px; | ||||
| $window_close_button_padding: 3px; | ||||
|  | ||||
| $window_clone_border_size: 6px; | ||||
|  | ||||
| // Window picker | ||||
| .window-picker { | ||||
| @@ -23,7 +22,7 @@ $window_clone_border_size: 6px; | ||||
|  | ||||
| // Borders on window thumbnails | ||||
| .window-clone-border { | ||||
|   border-width: $window_clone_border_size; | ||||
|   border-width: 6px; | ||||
|   border-style: solid; | ||||
|   border-color: $window_thumbnail_border_color; | ||||
|   border-radius: $base_border_radius + 2; | ||||
| @@ -55,6 +54,8 @@ $window_clone_border_size: 6px; | ||||
|   width: $window_close_button_size; | ||||
|   box-shadow: -1px 1px 5px 0px rgba(0,0,0,0.5); | ||||
|  | ||||
|   -shell-close-overlap: $window_close_button_size * 0.5; | ||||
|  | ||||
|   &:hover { | ||||
|     background-color: lighten($selected_bg_color, 5%); | ||||
|   } | ||||
| @@ -62,4 +63,4 @@ $window_clone_border_size: 6px; | ||||
|   &:active { | ||||
|     background-color: darken($selected_bg_color, 5%); | ||||
|   } | ||||
| } | ||||
| } | ||||
| @@ -16,6 +16,7 @@ | ||||
| } | ||||
|  | ||||
| .ws-switcher-box { | ||||
|   // background: transparent; | ||||
|   background: transparent; | ||||
|   height: 50px; | ||||
|   background-size: 32px; | ||||
| @@ -28,9 +29,44 @@ | ||||
| .ws-switcher-active-down, | ||||
| .ws-switcher-active-left, | ||||
| .ws-switcher-active-right { | ||||
|   height: 52px; | ||||
|   background-color: $selected_bg_color; | ||||
|   border: 1px solid if($variant=='light', darken($selected_bg_color, 8%), lighten($selected_bg_color, 5%)); | ||||
|   border-radius: $base_border_radius + 3px; | ||||
|   color: $selected_fg_color; | ||||
| } | ||||
|  | ||||
| /* Workspace pager */ | ||||
|  | ||||
| // thumbnails in overview | ||||
| .workspace-thumbnails {  | ||||
|   @extend %overview_panel; | ||||
|   visible-width: 32px; //amount visible before hover | ||||
|   spacing: $base_spacing; | ||||
|   padding: $base_padding; | ||||
|  | ||||
|   border-radius: $modal_radius 0 0 $modal_radius; | ||||
|   border-right-width: 0 !important; | ||||
|   //fixme: can't have non uniform borders :( | ||||
|   border-top-left-radius:0 !important; | ||||
|   border-bottom-left-radius:0 !important; | ||||
|  | ||||
|   &:rtl {  | ||||
|     border-radius: 0 $modal_radius $modal_radius 0;  | ||||
|     border-left-width: 0 !important; | ||||
|   } | ||||
|  | ||||
|   // drag and drop indicator | ||||
|   .placeholder { | ||||
|     background-image: url("resource:///org/gnome/shell/theme/dash-placeholder.svg"); | ||||
|     background-size: contain; | ||||
|     height: 24px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // selected indicator | ||||
| .workspace-thumbnail-indicator { | ||||
|   border: 3px solid $selected_bg_color; | ||||
|   border-radius: 3px; | ||||
|   padding: 0px; | ||||
|   // background-color: transparentize($selected_bg_color, 0.9); | ||||
| } | ||||
| @@ -1,32 +0,0 @@ | ||||
| /* Workspace pager */ | ||||
|  | ||||
| // thumbnails in overview | ||||
| .workspace-thumbnails { | ||||
|   @extend %overview_panel; | ||||
|   visible-width: 32px; //amount visible before hover | ||||
|   spacing: $base_spacing; | ||||
|   padding: $base_padding; | ||||
|  | ||||
|   border-radius: $modal_radius 0 0 $modal_radius; | ||||
|   border-right-width: 0; | ||||
|  | ||||
|   &:rtl { | ||||
|     border-radius: 0 $modal_radius $modal_radius 0; | ||||
|     border-left-width: 0; | ||||
|   } | ||||
|  | ||||
|   // drag and drop indicator | ||||
|   .placeholder { | ||||
|     background-image: url("resource:///org/gnome/shell/theme/dash-placeholder.svg"); | ||||
|     background-size: contain; | ||||
|     height: 24px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // selected indicator | ||||
| .workspace-thumbnail-indicator { | ||||
|   border: 3px solid $selected_bg_color; | ||||
|   border-radius: 3px; | ||||
|   padding: 0px; | ||||
|   // background-color: transparentize($selected_bg_color, 0.9); | ||||
| } | ||||
							
								
								
									
										4
									
								
								data/theme/key-enter.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" class="keyboard-key" width="24" height="24"> | ||||
|     <path overflow="visible" font-weight="400" style="line-height:normal;-inkscape-font-specification:'Bitstream Vera Sans';text-indent:0;text-align:start;text-decoration-line:none;text-transform:none;marker:none" d="M10 23H8.5c-.398 0-.796-.14-1.079-.422L.345 15.5l7.078-7.078C7.704 8.14 8.102 8 8.5 8H10v1.5c0 .398-.14.796-.422 1.079L4.657 15.5l4.921 4.922c.282.282.422.68.422 1.078z" color="#000" font-family="Bitstream Vera Sans" fill="#fff"/> | ||||
|     <path overflow="visible" d="M22 1.5v9a5 5 0 01-5 5H4" style="marker:none" color="#000" fill="none" stroke="#fff" stroke-width="3"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 676 B | 
							
								
								
									
										3
									
								
								data/theme/key-hide.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| <svg class="keyboard-key" xmlns="http://www.w3.org/2000/svg" width="24" height="24"> | ||||
|     <path d="M12 20.875L.562 9.438C.171 9.046 0 8.51 0 8V6h2c.511 0 1.046.17 1.438.563L12 15.125l8.563-8.562C20.953 6.17 21.488 6 22 6h2v2c0 .511-.17 1.046-.563 1.438z" fill="#e5e5e5"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 278 B | 
							
								
								
									
										5
									
								
								data/theme/key-layout.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" class="keyboard-key" width="24" height="24"> | ||||
|     <path d="M4.5 2v21" fill="#e5e5e5" fill-rule="evenodd" stroke="#e5e5e5" stroke-width="3"/> | ||||
|     <path d="M4 12h6l2 4h8V6h-6l-2-4H4z" fill="none" stroke="#e5e5e5" stroke-width="2" stroke-linejoin="round"/> | ||||
|     <path d="M4 12h6l2 4h8V6h-6l-2-4H4z" fill="#e5e5e5" fill-rule="evenodd"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 378 B | 
							
								
								
									
										3
									
								
								data/theme/key-shift-latched-uppercase.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| <svg class="keyboard-key" xmlns="http://www.w3.org/2000/svg" width="24" height="24"> | ||||
|     <path style="marker:none" d="M12 0L2 12h6v6h8v-6h6zM8 21v3h8v-3z" color="#000" overflow="visible" fill="#3584e4"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 211 B | 
							
								
								
									
										3
									
								
								data/theme/key-shift-uppercase.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| <svg class="keyboard-key" xmlns="http://www.w3.org/2000/svg" width="24" height="24"> | ||||
|     <path d="M8 22v-8H2L12 2l10 12h-6v8z" style="marker:none" color="#000" overflow="visible" fill="#3584e4"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 203 B | 
							
								
								
									
										3
									
								
								data/theme/key-shift.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| <svg class="keyboard-key" xmlns="http://www.w3.org/2000/svg" width="24" height="24"> | ||||
|     <path d="M8 22v-8H2L12 2l10 12h-6v8z" style="marker:none" color="#000" overflow="visible" fill="#bebebe"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 203 B | 
| @@ -1,6 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"> | ||||
|     <g fill="#2e3436"> | ||||
|         <path d="M6 8H2.937l5.126-5.781L13.186 8H10v2H6z" style="marker:none" color="#000" overflow="visible"/> | ||||
|         <path d="M6 11h4v2H6z" style="marker:none"/> | ||||
|     </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 268 B | 
| @@ -1,6 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"> | ||||
|     <g font-weight="400" fill="#2e3436"> | ||||
|         <path d="M11.994 3v4.004c.002.666-.183.72-.445.852-.262.13-.555.144-.555.144H4v2h6.994s.71.014 1.45-.355c.738-.37 1.552-1.313 1.55-2.645V3z" style="line-height:normal;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000;text-transform:none;text-orientation:mixed;white-space:normal;shape-padding:0;isolation:auto;mix-blend-mode:normal;solid-color:#000;solid-opacity:1;marker:none" color="#000" font-family="sans-serif" overflow="visible"/> | ||||
|         <path d="M6 12v-1c0-.257-.13-.528-.313-.719l-1.28-1.303 1.28-1.26C5.87 7.529 6 7.258 6 7V6H5c-.31 0-.552.09-.75.281L1.594 8.978l2.656 2.74c.198.192.44.282.75.282z" style="line-height:normal;-inkscape-font-specification:'Bitstream Vera Sans';text-indent:0;text-align:start;text-decoration-line:none;text-transform:none;marker:none" color="#bebebe" font-family="Bitstream Vera Sans" overflow="visible"/> | ||||
|     </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 1.2 KiB | 
| @@ -1,7 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"> | ||||
|     <g color="#000" fill="#2e3436"> | ||||
|         <path d="M4.707 5.293L3.293 6.707 8 11.414l4.707-4.707-1.414-1.414L8 8.586z" style="line-height:normal;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000;text-transform:none;text-orientation:mixed;white-space:normal;shape-padding:0;isolation:auto;mix-blend-mode:normal;solid-color:#000;solid-opacity:1;marker:none" font-weight="400" font-family="sans-serif" overflow="visible"/> | ||||
|         <path d="M12 6V5h1v1zM3 6V5h1v1z" style="marker:none" overflow="visible"/> | ||||
|         <path d="M3 6c0-.554.446-1 1-1s1 .446 1 1-.446 1-1 1-1-.446-1-1zM11 6c0-.554.446-1 1-1s1 .446 1 1-.446 1-1 1-1-.446-1-1z" style="marker:none" overflow="visible"/> | ||||
|     </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 990 B | 
| @@ -1,7 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"> | ||||
|     <g fill="#2e3436" fill-rule="evenodd"> | ||||
|         <path d="M2 1v14h2V1z" style="line-height:normal;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000;text-transform:none;text-orientation:mixed;white-space:normal;shape-padding:0;isolation:auto;mix-blend-mode:normal;solid-color:#000;solid-opacity:1" color="#000" font-weight="400" font-family="sans-serif" overflow="visible"/> | ||||
|         <path d="M3 1a1 1 0 00-1 1v6a1 1 0 001 1h3.383l.722 1.447A1 1 0 008 11h5a1 1 0 001-1V4a1 1 0 00-1-1H9.617l-.722-1.447A1 1 0 008 1zm1 2h3.383l.722 1.447A1 1 0 009 5h3v4H8.617l-.722-1.447A1 1 0 007 7H4z" style="line-height:normal;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000;text-transform:none;text-orientation:mixed;white-space:normal;shape-padding:0;isolation:auto;mix-blend-mode:normal;solid-color:#000;solid-opacity:1" color="#000" font-weight="400" font-family="sans-serif" overflow="visible"/> | ||||
|         <path d="M3 8h4l1 2h5V4H9L8 2H3z"/> | ||||
|     </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 1.4 KiB | 
| @@ -1,3 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"> | ||||
|     <path d="M6 13V9H2.937l5.126-5.781L13.186 9H10v4z" style="marker:none" color="#000" overflow="visible" fill="#2e3436"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 195 B | 
| @@ -6,9 +6,8 @@ theme_sources = files([ | ||||
|   'gnome-shell-sass/_drawing.scss', | ||||
|   'gnome-shell-sass/_high-contrast-colors.scss', | ||||
|   'gnome-shell-sass/_widgets.scss', | ||||
|   'gnome-shell-sass/widgets/_a11y.scss', | ||||
|   'gnome-shell-sass/widgets/_app-grid.scss', | ||||
|   'gnome-shell-sass/widgets/_base.scss', | ||||
|   'gnome-shell-sass/widgets/_app-switcher.scss', | ||||
|   'gnome-shell-sass/widgets/_buttons.scss', | ||||
|   'gnome-shell-sass/widgets/_calendar.scss', | ||||
|   'gnome-shell-sass/widgets/_check-box.scss', | ||||
| @@ -34,12 +33,10 @@ theme_sources = files([ | ||||
|   'gnome-shell-sass/widgets/_search-entry.scss', | ||||
|   'gnome-shell-sass/widgets/_search-results.scss', | ||||
|   'gnome-shell-sass/widgets/_slider.scss', | ||||
|   'gnome-shell-sass/widgets/_switcher-popup.scss', | ||||
|   'gnome-shell-sass/widgets/_switches.scss', | ||||
|   'gnome-shell-sass/widgets/_tiled-previews.scss', | ||||
|   'gnome-shell-sass/widgets/_window-picker.scss', | ||||
|   'gnome-shell-sass/widgets/_workspace-switcher.scss', | ||||
|   'gnome-shell-sass/widgets/_workspace-thumbnails.scss' | ||||
|   'gnome-shell-sass/widgets/_workspace-switcher.scss' | ||||
| ]) | ||||
|  | ||||
| styles = [ | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								data/theme/noise-texture.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 78 KiB | 
| @@ -1 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="46" height="22"><defs><linearGradient id="a"><stop offset="0" stop-color="#39393a"/><stop offset="1" stop-color="#302f30"/></linearGradient><linearGradient xlink:href="#a" id="b" x1="53" y1="294.429" x2="53" y2="309.804" gradientUnits="userSpaceOnUse" gradientTransform="translate(-42.76)"/></defs><g transform="translate(0 -291.18)" stroke-width="1.085" stroke="#151515"><rect style="marker:none" width="44.446" height="20.911" x=".625" y="291.715" rx="10.455" ry="10.073" fill="#282828"/><rect ry="10.455" rx="10.455" y="291.715" x=".543" height="20.911" width="21.143" style="marker:none" fill="url(#b)"/></g></svg> | ||||
| <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="46" height="22"><defs><linearGradient id="a"><stop offset="0" stop-color="#39393a"/><stop offset="1" stop-color="#302f30"/></linearGradient><linearGradient xlink:href="#a" id="b" x1="53" y1="294.429" x2="53" y2="309.804" gradientUnits="userSpaceOnUse" gradientTransform="translate(-42.76)"/></defs><g transform="translate(0 -291.18)" stroke-width="1.085"><rect style="marker:none" width="44.446" height="20.911" x=".625" y="291.715" rx="10.455" ry="10.073" fill="#323233" stroke="#272728"/><rect ry="10.455" rx="10.455" y="291.715" x=".543" height="20.911" width="21.143" style="marker:none" fill="url(#b)" stroke="#151515"/></g></svg> | ||||
| Before Width: | Height: | Size: 708 B After Width: | Height: | Size: 725 B | 
| @@ -1,9 +1,10 @@ | ||||
| #!/bin/env bash | ||||
|  | ||||
| CLDR_LAYOUTS_TARBALL="http://www.unicode.org/Public/cldr/latest/keyboards.zip" | ||||
| CLDR2JSON_GIT="git://repo.or.cz/cldr2json.git" | ||||
|  | ||||
| WORKDIR=".osk-layout-workbench" | ||||
| CLDR2JSON="cldr2json/cldr2json.py" | ||||
| CLDR2JSON="$WORKDIR/cldr2json/cldr2json.py" | ||||
| SRCDIR="$WORKDIR/keyboards/android" | ||||
| DESTDIR="osk-layouts" | ||||
| GRESOURCE_FILE="gnome-shell-osk-layouts.gresource.xml" | ||||
| @@ -19,6 +20,7 @@ mkdir -p "osk-layouts" | ||||
| # Download stuff on the work dir | ||||
| pushd $WORKDIR | ||||
| gio copy $CLDR_LAYOUTS_TARBALL . | ||||
| git clone $CLDR2JSON_GIT | ||||
| unzip keyboards.zip | ||||
| popd | ||||
|  | ||||
|   | ||||
| @@ -1,5 +0,0 @@ | ||||
| imports.package.start({ | ||||
|     name: '@PACKAGE_NAME@', | ||||
|     prefix: '@prefix@', | ||||
|     libdir: '@libdir@', | ||||
| }); | ||||
| @@ -1,3 +0,0 @@ | ||||
| [D-BUS Service] | ||||
| Name=@service@ | ||||
| Exec=@gjs@ @pkgdatadir@/@service@ | ||||
| @@ -1,177 +0,0 @@ | ||||
| /* exported DBusService, ServiceImplementation */ | ||||
|  | ||||
| const { Gio, GLib } = imports.gi; | ||||
|  | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const IDLE_SHUTDOWN_TIME = 2; // s | ||||
|  | ||||
| var ServiceImplementation = class { | ||||
|     constructor(info, objectPath) { | ||||
|         this._objectPath = objectPath; | ||||
|         this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(info, this); | ||||
|  | ||||
|         this._injectTracking('return_dbus_error'); | ||||
|         this._injectTracking('return_error_literal'); | ||||
|         this._injectTracking('return_gerror'); | ||||
|         this._injectTracking('return_value'); | ||||
|         this._injectTracking('return_value_with_unix_fd_list'); | ||||
|  | ||||
|         this._senders = new Map(); | ||||
|         this._holdCount = 0; | ||||
|  | ||||
|         this._hasSignals = this._dbusImpl.get_info().signals.length > 0; | ||||
|         this._shutdownTimeoutId = 0; | ||||
|  | ||||
|         // subclasses may override this to disable automatic shutdown | ||||
|         this._autoShutdown = true; | ||||
|     } | ||||
|  | ||||
|     // subclasses may override this to own additional names | ||||
|     register() { | ||||
|     } | ||||
|  | ||||
|     export() { | ||||
|         this._dbusImpl.export(Gio.DBus.session, this._objectPath); | ||||
|     } | ||||
|  | ||||
|     unexport() { | ||||
|         this._dbusImpl.unexport(); | ||||
|     } | ||||
|  | ||||
|     hold() { | ||||
|         this._holdCount++; | ||||
|     } | ||||
|  | ||||
|     release() { | ||||
|         if (this._holdCount === 0) { | ||||
|             logError(new Error('Unmatched call to release()')); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._holdCount--; | ||||
|  | ||||
|         if (this._holdCount === 0) | ||||
|             this._queueShutdownCheck(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * _handleError: | ||||
|      * @param {Gio.DBusMethodInvocation} | ||||
|      * @param {Error} | ||||
|      * | ||||
|      * Complete @invocation with an appropriate error if @error is set; | ||||
|      * useful for implementing early returns from method implementations. | ||||
|      * | ||||
|      * @returns {bool} - true if @invocation was completed | ||||
|      */ | ||||
|  | ||||
|     _handleError(invocation, error) { | ||||
|         if (error === null) | ||||
|             return false; | ||||
|  | ||||
|         if (error instanceof GLib.Error) { | ||||
|             invocation.return_gerror(error); | ||||
|         } else { | ||||
|             let name = error.name; | ||||
|             if (!name.includes('.')) // likely a normal JS error | ||||
|                 name = `org.gnome.gjs.JSError.${name}`; | ||||
|             invocation.return_dbus_error(name, error.message); | ||||
|         } | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     _maybeShutdown() { | ||||
|         if (!this._autoShutdown) | ||||
|             return; | ||||
|  | ||||
|         if (this._holdCount > 0) | ||||
|             return; | ||||
|  | ||||
|         this.emit('shutdown'); | ||||
|     } | ||||
|  | ||||
|     _queueShutdownCheck() { | ||||
|         if (this._shutdownTimeoutId) | ||||
|             GLib.source_remove(this._shutdownTimeoutId); | ||||
|  | ||||
|         this._shutdownTimeoutId = GLib.timeout_add_seconds( | ||||
|             GLib.PRIORITY_DEFAULT, IDLE_SHUTDOWN_TIME, | ||||
|             () => { | ||||
|                 this._shutdownTimeoutId = 0; | ||||
|                 this._maybeShutdown(); | ||||
|  | ||||
|                 return GLib.SOURCE_REMOVE; | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     _trackSender(sender) { | ||||
|         if (this._senders.has(sender)) | ||||
|             return; | ||||
|  | ||||
|         this.hold(); | ||||
|         this._senders.set(sender, | ||||
|             this._dbusImpl.get_connection().watch_name( | ||||
|                 sender, | ||||
|                 Gio.BusNameWatcherFlags.NONE, | ||||
|                 null, | ||||
|                 () => this._untrackSender(sender))); | ||||
|     } | ||||
|  | ||||
|     _untrackSender(sender) { | ||||
|         const id = this._senders.get(sender); | ||||
|  | ||||
|         if (id) | ||||
|             this._dbusImpl.get_connection().unwatch_name(id); | ||||
|  | ||||
|         if (this._senders.delete(sender)) | ||||
|             this.release(); | ||||
|     } | ||||
|  | ||||
|     _injectTracking(methodName) { | ||||
|         const { prototype } = Gio.DBusMethodInvocation; | ||||
|         const origMethod = prototype[methodName]; | ||||
|         const that = this; | ||||
|  | ||||
|         prototype[methodName] = function (...args) { | ||||
|             origMethod.apply(this, args); | ||||
|  | ||||
|             if (that._hasSignals) | ||||
|                 that._trackSender(this.get_sender()); | ||||
|  | ||||
|             that._queueShutdownCheck(); | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(ServiceImplementation.prototype); | ||||
|  | ||||
| var DBusService = class { | ||||
|     constructor(name, service) { | ||||
|         this._name = name; | ||||
|         this._service = service; | ||||
|         this._loop = new GLib.MainLoop(null, false); | ||||
|  | ||||
|         this._service.connect('shutdown', () => this._loop.quit()); | ||||
|     } | ||||
|  | ||||
|     run() { | ||||
|         // Bail out when not running under gnome-shell | ||||
|         Gio.DBus.watch_name(Gio.BusType.SESSION, | ||||
|             'org.gnome.Shell', | ||||
|             Gio.BusNameWatcherFlags.NONE, | ||||
|             null, | ||||
|             () => this._loop.quit()); | ||||
|  | ||||
|         this._service.register(); | ||||
|  | ||||
|         Gio.DBus.own_name(Gio.BusType.SESSION, | ||||
|             this._name, | ||||
|             Gio.BusNameOwnerFlags.REPLACE, | ||||
|             () => this._service.export(), | ||||
|             null, | ||||
|             () => this._loop.quit()); | ||||
|  | ||||
|         this._loop.run(); | ||||
|     } | ||||
| }; | ||||
| @@ -1,2 +0,0 @@ | ||||
| .expander-frame > * { border-top-width: 0; } | ||||
| .expander-toolbar { border: 0 solid @borders; border-top-width: 1px; } | ||||
| @@ -1,274 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported ExtensionsService */ | ||||
|  | ||||
| const { Gdk, Gio, GLib, GObject, Gtk, Shew } = imports.gi; | ||||
|  | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
|  | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
| const { ServiceImplementation } = imports.dbusService; | ||||
|  | ||||
| const ExtensionsIface = loadInterfaceXML('org.gnome.Shell.Extensions'); | ||||
| const ExtensionsProxy = Gio.DBusProxy.makeProxyWrapper(ExtensionsIface); | ||||
|  | ||||
| var ExtensionsService = class extends ServiceImplementation { | ||||
|     constructor() { | ||||
|         super(ExtensionsIface, '/org/gnome/Shell/Extensions'); | ||||
|  | ||||
|         this._proxy = new ExtensionsProxy(Gio.DBus.session, | ||||
|             'org.gnome.Shell', '/org/gnome/Shell'); | ||||
|  | ||||
|         this._proxy.connectSignal('ExtensionStateChanged', | ||||
|             (proxy, sender, params) => { | ||||
|                 this._dbusImpl.emit_signal('ExtensionStateChanged', | ||||
|                     new GLib.Variant('(sa{sv})', params)); | ||||
|             }); | ||||
|  | ||||
|         this._proxy.connect('g-properties-changed', () => { | ||||
|             this._dbusImpl.emit_property_changed('UserExtensionsEnabled', | ||||
|                 new GLib.Variant('b', this._proxy.UserExtensionsEnabled)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     get ShellVersion() { | ||||
|         return this._proxy.ShellVersion; | ||||
|     } | ||||
|  | ||||
|     get UserExtensionsEnabled() { | ||||
|         return this._proxy.UserExtensionsEnabled; | ||||
|     } | ||||
|  | ||||
|     set UserExtensionsEnabled(enable) { | ||||
|         this._proxy.UserExtensionsEnabled = enable; | ||||
|     } | ||||
|  | ||||
|     ListExtensionsAsync(params, invocation) { | ||||
|         this._proxy.ListExtensionsRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(a{sa{sv}})', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     GetExtensionInfoAsync(params, invocation) { | ||||
|         this._proxy.GetExtensionInfoRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(a{sv})', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     GetExtensionErrorsAsync(params, invocation) { | ||||
|         this._proxy.GetExtensionErrorsRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(as)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     InstallRemoteExtensionAsync(params, invocation) { | ||||
|         this._proxy.InstallRemoteExtensionRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(s)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     UninstallExtensionAsync(params, invocation) { | ||||
|         this._proxy.UninstallExtensionRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(b)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     EnableExtensionAsync(params, invocation) { | ||||
|         this._proxy.EnableExtensionRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(b)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     DisableExtensionAsync(params, invocation) { | ||||
|         this._proxy.DisableExtensionRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(b)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     LaunchExtensionPrefsAsync([uuid], invocation) { | ||||
|         this.OpenExtensionPrefsAsync([uuid, '', {}], invocation); | ||||
|     } | ||||
|  | ||||
|     OpenExtensionPrefsAsync(params, invocation) { | ||||
|         const [uuid, parentWindow, options] = params; | ||||
|  | ||||
|         this._proxy.GetExtensionInfoRemote(uuid, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             const [serialized] = res; | ||||
|             const extension = ExtensionUtils.deserializeExtension(serialized); | ||||
|  | ||||
|             const window = new ExtensionPrefsDialog(extension); | ||||
|             window.realize(); | ||||
|  | ||||
|             let externalWindow = null; | ||||
|  | ||||
|             if (parentWindow) | ||||
|                 externalWindow = Shew.ExternalWindow.new_from_handle(parentWindow); | ||||
|  | ||||
|             if (externalWindow) | ||||
|                 externalWindow.set_parent_of(window.window); | ||||
|  | ||||
|             if (options.modal) | ||||
|                 window.modal = options.modal.get_boolean(); | ||||
|  | ||||
|             window.connect('destroy', () => this.release()); | ||||
|             this.hold(); | ||||
|  | ||||
|             window.show(); | ||||
|  | ||||
|             invocation.return_value(null); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     CheckForUpdatesAsync(params, invocation) { | ||||
|         this._proxy.CheckForUpdatesRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(null); | ||||
|         }); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| var ExtensionPrefsDialog = GObject.registerClass({ | ||||
|     GTypeName: 'ExtensionPrefsDialog', | ||||
|     Template: 'resource:///org/gnome/Shell/Extensions/ui/extension-prefs-dialog.ui', | ||||
|     InternalChildren: [ | ||||
|         'headerBar', | ||||
|         'stack', | ||||
|         'expander', | ||||
|         'expanderArrow', | ||||
|         'revealer', | ||||
|         'errorView', | ||||
|     ], | ||||
| }, class ExtensionPrefsDialog extends Gtk.Window { | ||||
|     _init(extension) { | ||||
|         super._init(); | ||||
|  | ||||
|         this._uuid = extension.uuid; | ||||
|         this._url = extension.metadata.url || ''; | ||||
|  | ||||
|         this._headerBar.title = extension.metadata.name; | ||||
|  | ||||
|         this._actionGroup = new Gio.SimpleActionGroup(); | ||||
|         this.insert_action_group('win', this._actionGroup); | ||||
|  | ||||
|         this._initActions(); | ||||
|         this._addCustomStylesheet(); | ||||
|  | ||||
|         this._gesture = new Gtk.GestureMultiPress({ | ||||
|             widget: this._expander, | ||||
|             button: 0, | ||||
|             exclusive: true, | ||||
|         }); | ||||
|  | ||||
|         this._gesture.connect('released', (gesture, nPress) => { | ||||
|             if (nPress === 1) | ||||
|                 this._revealer.reveal_child = !this._revealer.reveal_child; | ||||
|         }); | ||||
|  | ||||
|         this._revealer.connect('notify::reveal-child', () => { | ||||
|             this._expanderArrow.icon_name = this._revealer.reveal_child | ||||
|                 ? 'pan-down-symbolic' | ||||
|                 : 'pan-end-symbolic'; | ||||
|         }); | ||||
|  | ||||
|         try { | ||||
|             ExtensionUtils.installImporter(extension); | ||||
|  | ||||
|             // give extension prefs access to their own extension object | ||||
|             ExtensionUtils.getCurrentExtension = () => extension; | ||||
|  | ||||
|             const prefsModule = extension.imports.prefs; | ||||
|             prefsModule.init(extension.metadata); | ||||
|  | ||||
|             const widget = prefsModule.buildPrefsWidget(); | ||||
|             this._stack.add(widget); | ||||
|             this._stack.visible_child = widget; | ||||
|         } catch (e) { | ||||
|             this._setError(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _setError(exc) { | ||||
|         this._errorView.buffer.text = `${exc}\n\nStack trace:\n`; | ||||
|         // Indent stack trace. | ||||
|         this._errorView.buffer.text += | ||||
|             exc.stack.split('\n').map(line => `  ${line}`).join('\n'); | ||||
|  | ||||
|         // markdown for pasting in gitlab issues | ||||
|         let lines = [ | ||||
|             `The settings of extension ${this._uuid} had an error:`, | ||||
|             '```', | ||||
|             `${exc}`, | ||||
|             '```', | ||||
|             '', | ||||
|             'Stack trace:', | ||||
|             '```', | ||||
|             exc.stack.replace(/\n$/, ''), // stack without trailing newline | ||||
|             '```', | ||||
|             '', | ||||
|         ]; | ||||
|         this._errorMarkdown = lines.join('\n'); | ||||
|         this._actionGroup.lookup('copy-error').enabled = true; | ||||
|     } | ||||
|  | ||||
|     _initActions() { | ||||
|         let action; | ||||
|  | ||||
|         action = new Gio.SimpleAction({ | ||||
|             name: 'copy-error', | ||||
|             enabled: false, | ||||
|         }); | ||||
|         action.connect('activate', () => { | ||||
|             const clipboard = Gtk.Clipboard.get_default(this.get_display()); | ||||
|             clipboard.set_text(this._errorMarkdown, -1); | ||||
|         }); | ||||
|         this._actionGroup.add_action(action); | ||||
|  | ||||
|         action = new Gio.SimpleAction({ | ||||
|             name: 'show-url', | ||||
|             enabled: this._url !== '', | ||||
|         }); | ||||
|         action.connect('activate', () => { | ||||
|             Gio.AppInfo.launch_default_for_uri(this._url, | ||||
|                 this.get_display().get_app_launch_context()); | ||||
|         }); | ||||
|         this._actionGroup.add_action(action); | ||||
|     } | ||||
|  | ||||
|     _addCustomStylesheet() { | ||||
|         let provider = new Gtk.CssProvider(); | ||||
|         let uri = 'resource:///org/gnome/Shell/Extensions/css/application.css'; | ||||
|         try { | ||||
|             provider.load_from_file(Gio.File.new_for_uri(uri)); | ||||
|         } catch (e) { | ||||
|             logError(e, 'Failed to add application style'); | ||||
|         } | ||||
|         Gtk.StyleContext.add_provider_for_screen(Gdk.Screen.get_default(), | ||||
|             provider, | ||||
|             Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION); | ||||
|     } | ||||
| }); | ||||
| @@ -1,20 +0,0 @@ | ||||
| /* exported main */ | ||||
|  | ||||
| imports.gi.versions.Gdk = '3.0'; | ||||
| imports.gi.versions.Gtk = '3.0'; | ||||
|  | ||||
| const { Gtk } = imports.gi; | ||||
| const pkg = imports.package; | ||||
|  | ||||
| const { DBusService } = imports.dbusService; | ||||
| const { ExtensionsService } = imports.extensionsService; | ||||
|  | ||||
| function main() { | ||||
|     Gtk.init(null); | ||||
|     pkg.initFormat(); | ||||
|  | ||||
|     const service = new DBusService( | ||||
|         'org.gnome.Shell.Extensions', | ||||
|         new ExtensionsService()); | ||||
|     service.run(); | ||||
| } | ||||
| @@ -1,197 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!-- Generated with glade 3.22.1 --> | ||||
| <interface> | ||||
|   <requires lib="gtk+" version="3.20"/> | ||||
|   <template class="ExtensionPrefsDialog" parent="GtkWindow"> | ||||
|     <property name="default_width">600</property> | ||||
|     <property name="default_height">400</property> | ||||
|     <child type="titlebar"> | ||||
|       <object class="GtkHeaderBar" id="headerBar"> | ||||
|         <property name="visible">True</property> | ||||
|         <property name="show_close_button">True</property> | ||||
|       </object> | ||||
|     </child> | ||||
|     <child> | ||||
|       <object class="GtkStack" id="stack"> | ||||
|         <property name="visible">True</property> | ||||
|         <property name="can_focus">False</property> | ||||
|         <child> | ||||
|           <object class="GtkScrolledWindow"> | ||||
|             <property name="visible">True</property> | ||||
|             <property name="hscrollbar_policy">never</property> | ||||
|             <property name="propagate_natural_height">True</property> | ||||
|             <child> | ||||
|               <object class="GtkViewport"> | ||||
|                 <property name="visible">True</property> | ||||
|                 <child> | ||||
|                   <object class="GtkBox"> | ||||
|                     <property name="visible">True</property> | ||||
|                     <property name="orientation">vertical</property> | ||||
|                     <property name="margin">100</property> | ||||
|                     <property name="margin_bottom">60</property> | ||||
|                     <property name="spacing">12</property> | ||||
|                     <child> | ||||
|                       <object class="GtkLabel"> | ||||
|                         <property name="visible">True</property> | ||||
|                         <property name="label" translatable="yes">Something’s gone wrong</property> | ||||
|                         <attributes> | ||||
|                           <attribute name="scale" value="1.44"/> <!-- x-large --> | ||||
|                         </attributes> | ||||
|                         <style> | ||||
|                           <class name="dim-label"/> | ||||
|                         </style> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                     <child> | ||||
|                       <object class="GtkLabel"> | ||||
|                         <property name="visible">True</property> | ||||
|                         <property name="label" translatable="yes">We’re very sorry, but there’s been a problem: the settings for this extension can’t be displayed. We recommend that you report the issue to the extension authors.</property> | ||||
|                         <property name="justify">center</property> | ||||
|                         <property name="wrap">True</property> | ||||
|                         <property name="xalign">0.5</property> | ||||
|                         <property name="yalign">0.5</property> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                     <child> | ||||
|                       <object class="GtkBox"> | ||||
|                         <property name="visible">True</property> | ||||
|                         <property name="orientation">vertical</property> | ||||
|                         <property name="margin_top">12</property> | ||||
|                         <child> | ||||
|                           <object class="GtkFrame" id="expander"> | ||||
|                             <property name="visible">True</property> | ||||
|                             <property name="hexpand">True</property> | ||||
|                             <property name="shadow_type">in</property> | ||||
|                             <child> | ||||
|                               <object class="GtkEventBox"> | ||||
|                                 <property name="visible">True</property> | ||||
|                                 <child> | ||||
|                                   <object class="GtkBox"> | ||||
|                                     <property name="visible">True</property> | ||||
|                                     <property name="margin">12</property> | ||||
|                                     <property name="spacing">6</property> | ||||
|                                     <child> | ||||
|                                       <object class="GtkImage" id="expanderArrow"> | ||||
|                                         <property name="visible">True</property> | ||||
|                                         <property name="icon_name">pan-end-symbolic</property> | ||||
|                                       </object> | ||||
|                                     </child> | ||||
|                                     <child> | ||||
|                                       <object class="GtkLabel"> | ||||
|                                         <property name="visible">True</property> | ||||
|                                         <property name="label" translatable="yes">Technical Details</property> | ||||
|                                       </object> | ||||
|                                     </child> | ||||
|                                   </object> | ||||
|                                 </child> | ||||
|                               </object> | ||||
|                             </child> | ||||
|                           </object> | ||||
|                         </child> | ||||
|                         <child> | ||||
|                           <object class="GtkRevealer" id="revealer"> | ||||
|                             <property name="visible">True</property> | ||||
|                             <child> | ||||
|                               <object class="GtkFrame"> | ||||
|                                 <property name="visible">True</property> | ||||
|                                 <property name="shadow_type">in</property> | ||||
|                                 <style> | ||||
|                                   <class name="expander-frame"/> | ||||
|                                 </style> | ||||
|                                 <child> | ||||
|                                   <object class="GtkBox"> | ||||
|                                     <property name="visible">True</property> | ||||
|                                     <property name="orientation">vertical</property> | ||||
|                                     <child> | ||||
|                                       <object class="GtkTextView" id="errorView"> | ||||
|                                         <property name="visible">True</property> | ||||
|                                         <property name="can_focus">True</property> | ||||
|                                         <property name="monospace">True</property> | ||||
|                                         <property name="editable">False</property> | ||||
|                                         <property name="wrap_mode">word</property> | ||||
|                                         <property name="left_margin">12</property> | ||||
|                                         <property name="right_margin">12</property> | ||||
|                                         <property name="top_margin">12</property> | ||||
|                                         <property name="bottom_margin">12</property> | ||||
|                                       </object> | ||||
|                                     </child> | ||||
|                                     <child> | ||||
|                                       <object class="GtkToolbar"> | ||||
|                                         <property name="visible">True</property> | ||||
|                                         <style> | ||||
|                                           <class name="expander-toolbar"/> | ||||
|                                         </style> | ||||
|                                         <child> | ||||
|                                           <object class="GtkToolItem"> | ||||
|                                             <property name="visible">True</property> | ||||
|                                             <child> | ||||
|                                               <object class="GtkButton"> | ||||
|                                                 <property name="visible">True</property> | ||||
|                                                 <property name="can_focus">True</property> | ||||
|                                                 <property name="receives_default">True</property> | ||||
|                                                 <property name="action_name">win.copy-error</property> | ||||
|                                                 <style> | ||||
|                                                   <class name="flat"/> | ||||
|                                                   <class name="image-button"/> | ||||
|                                                 </style> | ||||
|                                                 <child> | ||||
|                                                   <object class="GtkImage"> | ||||
|                                                     <property name="visible">True</property> | ||||
|                                                     <property name="icon_name">edit-copy-symbolic</property> | ||||
|                                                   </object> | ||||
|                                                 </child> | ||||
|                                               </object> | ||||
|                                             </child> | ||||
|                                           </object> | ||||
|                                         </child> | ||||
|                                         <child> | ||||
|                                           <object class="GtkSeparatorToolItem"> | ||||
|                                             <property name="visible">True</property> | ||||
|                                             <property name="draw">False</property> | ||||
|                                           </object> | ||||
|                                           <packing> | ||||
|                                             <property name="expand">True</property> | ||||
|                                           </packing> | ||||
|                                         </child> | ||||
|                                         <child> | ||||
|                                           <object class="GtkToolItem"> | ||||
|                                             <property name="visible">True</property> | ||||
|                                             <child> | ||||
|                                               <object class="GtkButton" id="homeButton"> | ||||
|                                                 <property name="visible" | ||||
|                                                           bind-source="homeButton" | ||||
|                                                           bind-property="sensitive" | ||||
|                                                           bind-flags="sync-create"/> | ||||
|                                                 <property name="label" translatable="yes">Homepage</property> | ||||
|                                                 <property name="tooltip_text" translatable="yes">Visit extension homepage</property> | ||||
|                                                 <property name="can_focus">True</property> | ||||
|                                                 <property name="receives_default">True</property> | ||||
|                                                 <property name="no_show_all">True</property> | ||||
|                                                 <property name="action_name">win.show-url</property> | ||||
|                                                 <style> | ||||
|                                                   <class name="flat"/> | ||||
|                                                 </style> | ||||
|                                               </object> | ||||
|                                             </child> | ||||
|                                           </object> | ||||
|                                         </child> | ||||
|                                       </object> | ||||
|                                     </child> | ||||
|                                   </object> | ||||
|                                 </child> | ||||
|                               </object> | ||||
|                             </child> | ||||
|                           </object> | ||||
|                         </child> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                   </object> | ||||
|                 </child> | ||||
|               </object> | ||||
|             </child> | ||||
|           </object> | ||||
|         </child> | ||||
|       </object> | ||||
|     </child> | ||||
|   </template> | ||||
| </interface> | ||||
| @@ -1,42 +0,0 @@ | ||||
| launcherconf = configuration_data() | ||||
| launcherconf.set('PACKAGE_NAME', meson.project_name()) | ||||
| launcherconf.set('prefix', prefix) | ||||
| launcherconf.set('libdir', libdir) | ||||
|  | ||||
| dbus_services = { | ||||
|   'org.gnome.Shell.Extensions': 'extensions', | ||||
|   'org.gnome.Shell.Notifications': 'notifications', | ||||
| } | ||||
|  | ||||
| config_dir = '@0@/..'.format(meson.current_build_dir()) | ||||
|  | ||||
| foreach service, dir : dbus_services | ||||
|   configure_file( | ||||
|     input: 'dbus-service.in', | ||||
|     output: service, | ||||
|     configuration: launcherconf, | ||||
|     install_dir: pkgdatadir, | ||||
|   ) | ||||
|  | ||||
|   serviceconf = configuration_data() | ||||
|   serviceconf.set('service', service) | ||||
|   serviceconf.set('gjs', gjs.path()) | ||||
|   serviceconf.set('pkgdatadir', pkgdatadir) | ||||
|  | ||||
|   configure_file( | ||||
|     input: 'dbus-service.service.in', | ||||
|     output: service + '.service', | ||||
|     configuration: serviceconf, | ||||
|     install_dir: servicedir | ||||
|   ) | ||||
|  | ||||
|   gnome.compile_resources( | ||||
|     service + '.src', | ||||
|     service + '.src.gresource.xml', | ||||
|     dependencies: [config_js], | ||||
|     source_dir: ['.', '..', dir, config_dir], | ||||
|     gresource_bundle: true, | ||||
|     install: true, | ||||
|     install_dir: pkgdatadir | ||||
|   ) | ||||
| endforeach | ||||
| @@ -1,11 +0,0 @@ | ||||
| /* exported main */ | ||||
|  | ||||
| const { DBusService } = imports.dbusService; | ||||
| const { NotificationDaemon } = imports.notificationDaemon; | ||||
|  | ||||
| function main() { | ||||
|     const service = new DBusService( | ||||
|         'org.gnome.Shell.Notifications', | ||||
|         new NotificationDaemon()); | ||||
|     service.run(); | ||||
| } | ||||
| @@ -1,105 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported NotificationDaemon */ | ||||
|  | ||||
| const { Gio, GLib } = imports.gi; | ||||
|  | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
| const { ServiceImplementation } = imports.dbusService; | ||||
|  | ||||
| const NotificationsIface = loadInterfaceXML('org.freedesktop.Notifications'); | ||||
| const NotificationsProxy = Gio.DBusProxy.makeProxyWrapper(NotificationsIface); | ||||
|  | ||||
| Gio._promisify(Gio.DBusConnection.prototype, 'call', 'call_finish'); | ||||
|  | ||||
| var NotificationDaemon = class extends ServiceImplementation { | ||||
|     constructor() { | ||||
|         super(NotificationsIface, '/org/freedesktop/Notifications'); | ||||
|  | ||||
|         this._autoShutdown = false; | ||||
|  | ||||
|         this._proxy = new NotificationsProxy(Gio.DBus.session, | ||||
|             'org.gnome.Shell', | ||||
|             '/org/freedesktop/Notifications', | ||||
|             (proxy, error) => { | ||||
|                 if (error) | ||||
|                     log(error.message); | ||||
|             }); | ||||
|  | ||||
|         this._proxy.connectSignal('ActionInvoked', | ||||
|             (proxy, sender, params) => { | ||||
|                 this._dbusImpl.emit_signal('ActionInvoked', | ||||
|                     new GLib.Variant('(us)', params)); | ||||
|             }); | ||||
|         this._proxy.connectSignal('NotificationClosed', | ||||
|             (proxy, sender, params) => { | ||||
|                 this._dbusImpl.emit_signal('NotificationClosed', | ||||
|                     new GLib.Variant('(uu)', params)); | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     register() { | ||||
|         Gio.DBus.session.own_name( | ||||
|             'org.freedesktop.Notifications', | ||||
|             Gio.BusNameOwnerFlags.REPLACE, | ||||
|             null, null); | ||||
|     } | ||||
|  | ||||
|     async NotifyAsync(params, invocation) { | ||||
|         const pid = await this._getSenderPid(invocation.get_sender()); | ||||
|         const hints = params[6]; | ||||
|  | ||||
|         params[6] = { | ||||
|             ...hints, | ||||
|             'sender-pid': new GLib.Variant('u', pid), | ||||
|         }; | ||||
|  | ||||
|         this._proxy.NotifyRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(u)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     CloseNotificationAsync(params, invocation) { | ||||
|         this._proxy.CloseNotificationRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(null); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     GetCapabilitiesAsync(params, invocation) { | ||||
|         this._proxy.GetCapabilitiesRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(as)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     GetServerInformationAsync(params, invocation) { | ||||
|         this._proxy.GetServerInformationRemote(...params, (res, error) => { | ||||
|             if (this._handleError(invocation, error)) | ||||
|                 return; | ||||
|  | ||||
|             invocation.return_value(new GLib.Variant('(ssss)', res)); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     async _getSenderPid(sender) { | ||||
|         const res = await Gio.DBus.session.call( | ||||
|             'org.freedesktop.DBus', | ||||
|             '/', | ||||
|             'org.freedesktop.DBus', | ||||
|             'GetConnectionUnixProcessID', | ||||
|             new GLib.Variant('(s)', [sender]), | ||||
|             new GLib.VariantType('(u)'), | ||||
|             Gio.DBusCallFlags.NONE, | ||||
|             -1, | ||||
|             null); | ||||
|         const [pid] = res.deepUnpack(); | ||||
|         return pid; | ||||
|     } | ||||
| }; | ||||
| @@ -1,18 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <gresources> | ||||
|   <gresource prefix="/org/gnome/Shell/Extensions/js"> | ||||
|     <file>main.js</file> | ||||
|     <file>extensionsService.js</file> | ||||
|     <file>dbusService.js</file> | ||||
|  | ||||
|     <file>misc/config.js</file> | ||||
|     <file>misc/extensionUtils.js</file> | ||||
|     <file>misc/fileUtils.js</file> | ||||
|     <file>misc/params.js</file> | ||||
|   </gresource> | ||||
|  | ||||
|   <gresource prefix="/org/gnome/Shell/Extensions"> | ||||
|     <file>css/application.css</file> | ||||
|     <file>ui/extension-prefs-dialog.ui</file> | ||||
|   </gresource> | ||||
| </gresources> | ||||
| @@ -1,11 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <gresources> | ||||
|   <gresource prefix="/org/gnome/Shell/Notifications/js"> | ||||
|     <file>main.js</file> | ||||
|     <file>notificationDaemon.js</file> | ||||
|     <file>dbusService.js</file> | ||||
|  | ||||
|     <file>misc/config.js</file> | ||||
|     <file>misc/fileUtils.js</file> | ||||
|   </gresource> | ||||
| </gresources> | ||||
							
								
								
									
										695
									
								
								js/extensionPrefs/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,695 @@ | ||||
| /* exported main */ | ||||
| imports.gi.versions.Gdk = '3.0'; | ||||
| imports.gi.versions.Gtk = '3.0'; | ||||
|  | ||||
| const Gettext = imports.gettext; | ||||
| const { Gdk, GLib, Gio, GObject, Gtk, Pango } = imports.gi; | ||||
| const Format = imports.format; | ||||
|  | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
|  | ||||
| const { ExtensionState } = ExtensionUtils; | ||||
|  | ||||
| const GnomeShellIface = loadInterfaceXML('org.gnome.Shell.Extensions'); | ||||
| const GnomeShellProxy = Gio.DBusProxy.makeProxyWrapper(GnomeShellIface); | ||||
|  | ||||
| function stripPrefix(string, prefix) { | ||||
|     if (string.slice(0, prefix.length) == prefix) | ||||
|         return string.slice(prefix.length); | ||||
|     return string; | ||||
| } | ||||
|  | ||||
| var Application = GObject.registerClass( | ||||
| class Application extends Gtk.Application { | ||||
|     _init() { | ||||
|         GLib.set_prgname('gnome-shell-extension-prefs'); | ||||
|         super._init({ | ||||
|             application_id: 'org.gnome.shell.ExtensionPrefs', | ||||
|             flags: Gio.ApplicationFlags.HANDLES_COMMAND_LINE, | ||||
|         }); | ||||
|  | ||||
|         this._startupUuid = null; | ||||
|         this._loaded = false; | ||||
|         this._skipMainWindow = false; | ||||
|         this._shellProxy = null; | ||||
|     } | ||||
|  | ||||
|     get shellProxy() { | ||||
|         return this._shellProxy; | ||||
|     } | ||||
|  | ||||
|     _showPrefs(uuid) { | ||||
|         let row = this._extensionSelector.get_children().find(c => { | ||||
|             return c.uuid === uuid && c.hasPrefs; | ||||
|         }); | ||||
|  | ||||
|         if (!row) | ||||
|             return false; | ||||
|  | ||||
|         let widget; | ||||
|  | ||||
|         try { | ||||
|             widget = row.prefsModule.buildPrefsWidget(); | ||||
|         } catch (e) { | ||||
|             widget = this._buildErrorUI(row, e); | ||||
|         } | ||||
|  | ||||
|         let dialog = new Gtk.Window({ | ||||
|             modal: !this._skipMainWindow, | ||||
|             type_hint: Gdk.WindowTypeHint.DIALOG, | ||||
|         }); | ||||
|         dialog.set_titlebar(new Gtk.HeaderBar({ | ||||
|             show_close_button: true, | ||||
|             title: row.name, | ||||
|             visible: true, | ||||
|         })); | ||||
|  | ||||
|         if (this._skipMainWindow) { | ||||
|             this.add_window(dialog); | ||||
|             if (this._window) | ||||
|                 this._window.destroy(); | ||||
|             this._window = dialog; | ||||
|             this._window.window_position = Gtk.WindowPosition.CENTER; | ||||
|         } else { | ||||
|             dialog.transient_for = this._window; | ||||
|         } | ||||
|  | ||||
|         dialog.set_default_size(600, 400); | ||||
|         dialog.add(widget); | ||||
|         dialog.show(); | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     _buildErrorUI(row, exc) { | ||||
|         let scroll = new Gtk.ScrolledWindow({ | ||||
|             hscrollbar_policy: Gtk.PolicyType.NEVER, | ||||
|             propagate_natural_height: true, | ||||
|         }); | ||||
|  | ||||
|         let box = new Gtk.Box({ | ||||
|             orientation: Gtk.Orientation.VERTICAL, | ||||
|             spacing: 12, | ||||
|             margin: 100, | ||||
|             margin_bottom: 60, | ||||
|         }); | ||||
|         scroll.add(box); | ||||
|  | ||||
|         let label = new Gtk.Label({ | ||||
|             label: '<span size="x-large">%s</span>'.format(_("Something’s gone wrong")), | ||||
|             use_markup: true, | ||||
|         }); | ||||
|         label.get_style_context().add_class(Gtk.STYLE_CLASS_DIM_LABEL); | ||||
|         box.add(label); | ||||
|  | ||||
|         label = new Gtk.Label({ | ||||
|             label: _("We’re very sorry, but there’s been a problem: the settings for this extension can’t be displayed. We recommend that you report the issue to the extension authors."), | ||||
|             justify: Gtk.Justification.CENTER, | ||||
|             wrap: true, | ||||
|         }); | ||||
|         box.add(label); | ||||
|  | ||||
|         let expander = new Expander({ | ||||
|             label: _("Technical Details"), | ||||
|             margin_top: 12, | ||||
|         }); | ||||
|         box.add(expander); | ||||
|  | ||||
|         let errortext = `${exc}\n\nStack trace:\n${ | ||||
|             // Indent stack trace. | ||||
|             exc.stack.split('\n').map(line => `  ${line}`).join('\n') | ||||
|         }`; | ||||
|  | ||||
|         let buffer = new Gtk.TextBuffer({ text: errortext }); | ||||
|         let textview = new Gtk.TextView({ | ||||
|             buffer, | ||||
|             wrap_mode: Gtk.WrapMode.WORD, | ||||
|             monospace: true, | ||||
|             editable: false, | ||||
|             top_margin: 12, | ||||
|             bottom_margin: 12, | ||||
|             left_margin: 12, | ||||
|             right_margin: 12, | ||||
|         }); | ||||
|  | ||||
|         let toolbar = new Gtk.Toolbar(); | ||||
|         let provider = new Gtk.CssProvider(); | ||||
|         provider.load_from_data(`* { | ||||
|             border: 0 solid @borders; | ||||
|             border-top-width: 1px; | ||||
|         }`); | ||||
|         toolbar.get_style_context().add_provider( | ||||
|             provider, | ||||
|             Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION | ||||
|         ); | ||||
|  | ||||
|         let copyButton = new Gtk.ToolButton({ | ||||
|             icon_name: 'edit-copy-symbolic', | ||||
|             tooltip_text: _("Copy Error"), | ||||
|         }); | ||||
|         toolbar.add(copyButton); | ||||
|  | ||||
|         copyButton.connect('clicked', w => { | ||||
|             let clipboard = Gtk.Clipboard.get_default(w.get_display()); | ||||
|             // markdown for pasting in gitlab issues | ||||
|             let lines = [ | ||||
|                 `The settings of extension ${row.uuid} had an error:`, | ||||
|                 '```', // '`' (xgettext throws up on odd number of backticks) | ||||
|                 `${exc}`, | ||||
|                 '```', // '`' | ||||
|                 '', | ||||
|                 'Stack trace:', | ||||
|                 '```', // '`' | ||||
|                 exc.stack.replace(/\n$/, ''), // stack without trailing newline | ||||
|                 '```', // '`' | ||||
|                 '', | ||||
|             ]; | ||||
|             clipboard.set_text(lines.join('\n'), -1); | ||||
|         }); | ||||
|  | ||||
|         let spacing = new Gtk.SeparatorToolItem({ draw: false }); | ||||
|         toolbar.add(spacing); | ||||
|         toolbar.child_set_property(spacing, "expand", true); | ||||
|  | ||||
|         let urlButton = new Gtk.ToolButton({ | ||||
|             label: _("Homepage"), | ||||
|             tooltip_text: _("Visit extension homepage"), | ||||
|             no_show_all: true, | ||||
|             visible: row.url != null, | ||||
|         }); | ||||
|         toolbar.add(urlButton); | ||||
|  | ||||
|         urlButton.connect('clicked', w => { | ||||
|             let context = w.get_display().get_app_launch_context(); | ||||
|             Gio.AppInfo.launch_default_for_uri(row.url, context); | ||||
|         }); | ||||
|  | ||||
|         let expandedBox = new Gtk.Box({ | ||||
|             orientation: Gtk.Orientation.VERTICAL, | ||||
|         }); | ||||
|         expandedBox.add(textview); | ||||
|         expandedBox.add(toolbar); | ||||
|  | ||||
|         expander.add(expandedBox); | ||||
|  | ||||
|         scroll.show_all(); | ||||
|         return scroll; | ||||
|     } | ||||
|  | ||||
|     _buildUI() { | ||||
|         this._window = new Gtk.ApplicationWindow({ application: this, | ||||
|                                                    window_position: Gtk.WindowPosition.CENTER }); | ||||
|  | ||||
|         this._window.set_default_size(800, 500); | ||||
|  | ||||
|         this._titlebar = new Gtk.HeaderBar({ show_close_button: true, | ||||
|                                              title: _("Shell Extensions") }); | ||||
|         this._window.set_titlebar(this._titlebar); | ||||
|  | ||||
|         let killSwitch = new Gtk.Switch({ valign: Gtk.Align.CENTER }); | ||||
|         this._titlebar.pack_end(killSwitch); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|         this._settings.bind('disable-user-extensions', killSwitch, 'active', | ||||
|                             Gio.SettingsBindFlags.DEFAULT | | ||||
|                             Gio.SettingsBindFlags.INVERT_BOOLEAN); | ||||
|  | ||||
|         this._mainStack = new Gtk.Stack({ | ||||
|             transition_type: Gtk.StackTransitionType.CROSSFADE, | ||||
|         }); | ||||
|         this._window.add(this._mainStack); | ||||
|  | ||||
|         let scroll = new Gtk.ScrolledWindow({ hscrollbar_policy: Gtk.PolicyType.NEVER }); | ||||
|  | ||||
|         this._extensionSelector = new Gtk.ListBox({ selection_mode: Gtk.SelectionMode.NONE }); | ||||
|         this._extensionSelector.set_sort_func(this._sortList.bind(this)); | ||||
|         this._extensionSelector.set_header_func(this._updateHeader.bind(this)); | ||||
|  | ||||
|         scroll.add(this._extensionSelector); | ||||
|  | ||||
|         this._mainStack.add_named(scroll, 'listing'); | ||||
|         this._mainStack.add_named(new EmptyPlaceholder(), 'placeholder'); | ||||
|  | ||||
|         this._shellProxy = new GnomeShellProxy(Gio.DBus.session, 'org.gnome.Shell', '/org/gnome/Shell'); | ||||
|         this._shellProxy.connectSignal('ExtensionStateChanged', | ||||
|             this._onExtensionStateChanged.bind(this)); | ||||
|  | ||||
|         this._window.show_all(); | ||||
|     } | ||||
|  | ||||
|     _sortList(row1, row2) { | ||||
|         return row1.name.localeCompare(row2.name); | ||||
|     } | ||||
|  | ||||
|     _updateHeader(row, before) { | ||||
|         if (!before || row.get_header()) | ||||
|             return; | ||||
|  | ||||
|         let sep = new Gtk.Separator({ orientation: Gtk.Orientation.HORIZONTAL }); | ||||
|         row.set_header(sep); | ||||
|     } | ||||
|  | ||||
|     _findExtensionRow(uuid) { | ||||
|         return this._extensionSelector.get_children().find(c => c.uuid === uuid); | ||||
|     } | ||||
|  | ||||
|     _onExtensionStateChanged(proxy, senderName, [uuid, newState]) { | ||||
|         let extension = ExtensionUtils.deserializeExtension(newState); | ||||
|         let row = this._findExtensionRow(uuid); | ||||
|  | ||||
|         if (row) { | ||||
|             if (extension.state === ExtensionState.UNINSTALLED) | ||||
|                 row.destroy(); | ||||
|             return; // we only deal with new and deleted extensions here | ||||
|         } | ||||
|         this._addExtensionRow(extension); | ||||
|     } | ||||
|  | ||||
|     _scanExtensions() { | ||||
|         this._shellProxy.ListExtensionsRemote(([extensionsMap], e) => { | ||||
|             if (e) { | ||||
|                 if (e instanceof Gio.DBusError) { | ||||
|                     log(`Failed to connect to shell proxy: ${e}`); | ||||
|                     this._mainStack.add_named(new NoShellPlaceholder(), 'noshell'); | ||||
|                     this._mainStack.visible_child_name = 'noshell'; | ||||
|                 } else { | ||||
|                     throw e; | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             for (let uuid in extensionsMap) { | ||||
|                 let extension = ExtensionUtils.deserializeExtension(extensionsMap[uuid]); | ||||
|                 this._addExtensionRow(extension); | ||||
|             } | ||||
|             this._extensionsLoaded(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _addExtensionRow(extension) { | ||||
|         let row = new ExtensionRow(extension); | ||||
|  | ||||
|         row.prefsButton.connect('clicked', () => { | ||||
|             this._showPrefs(row.uuid); | ||||
|         }); | ||||
|  | ||||
|         row.show_all(); | ||||
|         this._extensionSelector.add(row); | ||||
|     } | ||||
|  | ||||
|     _extensionsLoaded() { | ||||
|         if (this._extensionSelector.get_children().length > 0) | ||||
|             this._mainStack.visible_child_name = 'listing'; | ||||
|         else | ||||
|             this._mainStack.visible_child_name = 'placeholder'; | ||||
|  | ||||
|         if (this._startupUuid) | ||||
|             this._showPrefs(this._startupUuid); | ||||
|         this._startupUuid = null; | ||||
|         this._skipMainWindow = false; | ||||
|         this._loaded = true; | ||||
|     } | ||||
|  | ||||
|     vfunc_activate() { | ||||
|         this._window.present(); | ||||
|     } | ||||
|  | ||||
|     vfunc_startup() { | ||||
|         super.vfunc_startup(); | ||||
|  | ||||
|         this._buildUI(); | ||||
|         this._scanExtensions(); | ||||
|     } | ||||
|  | ||||
|     vfunc_command_line(commandLine) { | ||||
|         this.activate(); | ||||
|         let args = commandLine.get_arguments(); | ||||
|  | ||||
|         if (args.length) { | ||||
|             let uuid = args[0]; | ||||
|  | ||||
|             this._skipMainWindow = true; | ||||
|  | ||||
|             // Strip off "extension:///" prefix which fakes a URI, if it exists | ||||
|             uuid = stripPrefix(uuid, "extension:///"); | ||||
|  | ||||
|             if (!this._loaded) | ||||
|                 this._startupUuid = uuid; | ||||
|             else if (!this._showPrefs(uuid)) | ||||
|                 this._skipMainWindow = false; | ||||
|         } | ||||
|         return 0; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var Expander = GObject.registerClass({ | ||||
|     Properties: { | ||||
|         'label': GObject.ParamSpec.string( | ||||
|             'label', 'label', 'label', | ||||
|             GObject.ParamFlags.READWRITE, | ||||
|             null | ||||
|         ), | ||||
|     }, | ||||
| }, class Expander extends Gtk.Box { | ||||
|     _init(params = {}) { | ||||
|         this._labelText = null; | ||||
|  | ||||
|         super._init(Object.assign(params, { | ||||
|             orientation: Gtk.Orientation.VERTICAL, | ||||
|             spacing: 0, | ||||
|         })); | ||||
|  | ||||
|         this._frame = new Gtk.Frame({ | ||||
|             shadow_type: Gtk.ShadowType.IN, | ||||
|             hexpand: true, | ||||
|         }); | ||||
|  | ||||
|         let eventBox = new Gtk.EventBox(); | ||||
|         this._frame.add(eventBox); | ||||
|  | ||||
|         let hbox = new Gtk.Box({ | ||||
|             spacing: 6, | ||||
|             margin: 12, | ||||
|         }); | ||||
|         eventBox.add(hbox); | ||||
|  | ||||
|         this._arrow = new Gtk.Image({ | ||||
|             icon_name: 'pan-end-symbolic', | ||||
|         }); | ||||
|         hbox.add(this._arrow); | ||||
|  | ||||
|         this._label = new Gtk.Label({ label: this._labelText }); | ||||
|         hbox.add(this._label); | ||||
|  | ||||
|         this._revealer = new Gtk.Revealer(); | ||||
|  | ||||
|         this._childBin = new Gtk.Frame({ | ||||
|             shadow_type: Gtk.ShadowType.IN, | ||||
|         }); | ||||
|         this._revealer.add(this._childBin); | ||||
|  | ||||
|         // Directly chain up to parent for internal children | ||||
|         super.add(this._frame); | ||||
|         super.add(this._revealer); | ||||
|  | ||||
|         let provider = new Gtk.CssProvider(); | ||||
|         provider.load_from_data('* { border-top-width: 0; }'); | ||||
|         this._childBin.get_style_context().add_provider( | ||||
|             provider, | ||||
|             Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION | ||||
|         ); | ||||
|  | ||||
|         this._gesture = new Gtk.GestureMultiPress({ | ||||
|             widget: this._frame, | ||||
|             button: 0, | ||||
|             exclusive: true, | ||||
|         }); | ||||
|         this._gesture.connect('released', (gesture, nPress) => { | ||||
|             if (nPress == 1) | ||||
|                 this._revealer.reveal_child = !this._revealer.reveal_child; | ||||
|         }); | ||||
|         this._revealer.connect('notify::reveal-child', () => { | ||||
|             if (this._revealer.reveal_child) | ||||
|                 this._arrow.icon_name = 'pan-down-symbolic'; | ||||
|             else | ||||
|                 this._arrow.icon_name = 'pan-end-symbolic'; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     get label() { | ||||
|         return this._labelText; | ||||
|     } | ||||
|  | ||||
|     set label(text) { | ||||
|         if (this._labelText == text) | ||||
|             return; | ||||
|  | ||||
|         if (this._label) | ||||
|             this._label.label = text; | ||||
|         this._labelText = text; | ||||
|         this.notify('label'); | ||||
|     } | ||||
|  | ||||
|     add(child) { | ||||
|         // set expanded child | ||||
|         this._childBin.get_children().forEach(c => { | ||||
|             this._childBin.remove(c); | ||||
|         }); | ||||
|  | ||||
|         if (child) | ||||
|             this._childBin.add(child); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var EmptyPlaceholder = GObject.registerClass( | ||||
| class EmptyPlaceholder extends Gtk.Box { | ||||
|     _init() { | ||||
|         super._init({ | ||||
|             orientation: Gtk.Orientation.VERTICAL, | ||||
|             spacing: 6, | ||||
|             margin: 32, | ||||
|         }); | ||||
|  | ||||
|         let image = new Gtk.Image({ | ||||
|             icon_name: 'application-x-addon-symbolic', | ||||
|             pixel_size: 96, | ||||
|             visible: true, | ||||
|             vexpand: true, | ||||
|             valign: Gtk.Align.END, | ||||
|         }); | ||||
|         image.get_style_context().add_class(Gtk.STYLE_CLASS_DIM_LABEL); | ||||
|         this.add(image); | ||||
|  | ||||
|         let label = new Gtk.Label({ | ||||
|             label: `<b><span size="x-large">${_("No Extensions Installed")}</span></b>`, | ||||
|             use_markup: true, | ||||
|             visible: true, | ||||
|         }); | ||||
|         label.get_style_context().add_class(Gtk.STYLE_CLASS_DIM_LABEL); | ||||
|         this.add(label); | ||||
|  | ||||
|         let appInfo = Gio.DesktopAppInfo.new('org.gnome.Software.desktop'); | ||||
|  | ||||
|         let desc = new Gtk.Label({ | ||||
|             label: _("Extensions can be installed through Software or <a href=\"https://extensions.gnome.org\">extensions.gnome.org</a>."), | ||||
|             use_markup: true, | ||||
|             wrap: true, | ||||
|             justify: Gtk.Justification.CENTER, | ||||
|             visible: true, | ||||
|             max_width_chars: 50, | ||||
|             hexpand: true, | ||||
|             vexpand: appInfo == null, | ||||
|             halign: Gtk.Align.CENTER, | ||||
|             valign: Gtk.Align.START, | ||||
|         }); | ||||
|         this.add(desc); | ||||
|  | ||||
|         if (appInfo) { | ||||
|             let button = new Gtk.Button({ | ||||
|                 label: _("Browse in Software"), | ||||
|                 image: new Gtk.Image({ | ||||
|                     icon_name: "org.gnome.Software-symbolic", | ||||
|                 }), | ||||
|                 always_show_image: true, | ||||
|                 margin_top: 12, | ||||
|                 visible: true, | ||||
|                 halign: Gtk.Align.CENTER, | ||||
|                 valign: Gtk.Align.START, | ||||
|                 vexpand: true, | ||||
|             }); | ||||
|             this.add(button); | ||||
|  | ||||
|             button.connect('clicked', w => { | ||||
|                 let context = w.get_display().get_app_launch_context(); | ||||
|                 appInfo.launch([], context); | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var NoShellPlaceholder = GObject.registerClass( | ||||
| class NoShellPlaceholder extends Gtk.Box { | ||||
|     _init() { | ||||
|         super._init({ | ||||
|             orientation: Gtk.Orientation.VERTICAL, | ||||
|             spacing: 12, | ||||
|             margin: 100, | ||||
|             margin_bottom: 60, | ||||
|         }); | ||||
|  | ||||
|         let label = new Gtk.Label({ | ||||
|             label: '<span size="x-large">%s</span>'.format( | ||||
|                 _("Something’s gone wrong")), | ||||
|             use_markup: true, | ||||
|         }); | ||||
|         label.get_style_context().add_class(Gtk.STYLE_CLASS_DIM_LABEL); | ||||
|         this.add(label); | ||||
|  | ||||
|         label = new Gtk.Label({ | ||||
|             label: _("We’re very sorry, but it was not possible to get the list of installed extensions. Make sure you are logged into GNOME and try again."), | ||||
|             justify: Gtk.Justification.CENTER, | ||||
|             wrap: true, | ||||
|         }); | ||||
|         this.add(label); | ||||
|  | ||||
|         this.show_all(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var DescriptionLabel = GObject.registerClass( | ||||
| class DescriptionLabel extends Gtk.Label { | ||||
|     vfunc_get_preferred_height_for_width(width) { | ||||
|         // Hack: Request the maximum height allowed by the line limit | ||||
|         if (this.lines > 0) | ||||
|             return super.vfunc_get_preferred_height_for_width(0); | ||||
|         return super.vfunc_get_preferred_height_for_width(width); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var ExtensionRow = GObject.registerClass( | ||||
| class ExtensionRow extends Gtk.ListBoxRow { | ||||
|     _init(extension) { | ||||
|         super._init(); | ||||
|  | ||||
|         this._app = Gio.Application.get_default(); | ||||
|         this._extension = extension; | ||||
|         this._prefsModule = null; | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|  | ||||
|         this._buildUI(); | ||||
|  | ||||
|         this._extensionStateChangedId = this._app.shellProxy.connectSignal( | ||||
|             'ExtensionStateChanged', (p, sender, [uuid, newState]) => { | ||||
|                 if (this.uuid !== uuid) | ||||
|                     return; | ||||
|  | ||||
|                 this._extension = ExtensionUtils.deserializeExtension(newState); | ||||
|                 let state = this._extension.state == ExtensionState.ENABLED; | ||||
|  | ||||
|                 this._switch.block_signal_handler(this._notifyActiveId); | ||||
|                 this._switch.state = state; | ||||
|                 this._switch.unblock_signal_handler(this._notifyActiveId); | ||||
|  | ||||
|                 this._switch.sensitive = this._canToggle(); | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     get uuid() { | ||||
|         return this._extension.uuid; | ||||
|     } | ||||
|  | ||||
|     get name() { | ||||
|         return this._extension.metadata.name; | ||||
|     } | ||||
|  | ||||
|     get hasPrefs() { | ||||
|         return this._extension.hasPrefs; | ||||
|     } | ||||
|  | ||||
|     get url() { | ||||
|         return this._extension.metadata.url; | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
|         if (!this._app.shellProxy) | ||||
|             return; | ||||
|  | ||||
|         if (this._extensionStateChangedId) | ||||
|             this._app.shellProxy.disconnectSignal(this._extensionStateChangedId); | ||||
|         this._extensionStateChangedId = 0; | ||||
|     } | ||||
|  | ||||
|     _buildUI() { | ||||
|         let hbox = new Gtk.Box({ orientation: Gtk.Orientation.HORIZONTAL, | ||||
|                                  hexpand: true, margin_end: 24, spacing: 24, | ||||
|                                  margin: 12 }); | ||||
|         this.add(hbox); | ||||
|  | ||||
|         let vbox = new Gtk.Box({ orientation: Gtk.Orientation.VERTICAL, | ||||
|                                  spacing: 6, hexpand: true }); | ||||
|         hbox.add(vbox); | ||||
|  | ||||
|         let name = GLib.markup_escape_text(this.name, -1); | ||||
|         let label = new Gtk.Label({ label: '<b>' + name + '</b>', | ||||
|                                     use_markup: true, | ||||
|                                     halign: Gtk.Align.START }); | ||||
|         vbox.add(label); | ||||
|  | ||||
|         let desc = this._extension.metadata.description.split('\n')[0]; | ||||
|         label = new DescriptionLabel({ label: desc, wrap: true, lines: 2, | ||||
|                                        ellipsize: Pango.EllipsizeMode.END, | ||||
|                                        xalign: 0, yalign: 0 }); | ||||
|         vbox.add(label); | ||||
|  | ||||
|         let button = new Gtk.Button({ valign: Gtk.Align.CENTER, | ||||
|                                       visible: this.hasPrefs, | ||||
|                                       no_show_all: true }); | ||||
|         button.set_image(new Gtk.Image({ icon_name: 'emblem-system-symbolic', | ||||
|                                          icon_size: Gtk.IconSize.BUTTON, | ||||
|                                          visible: true })); | ||||
|         button.get_style_context().add_class('circular'); | ||||
|         hbox.add(button); | ||||
|  | ||||
|         this.prefsButton = button; | ||||
|  | ||||
|         this._switch = new Gtk.Switch({ | ||||
|             valign: Gtk.Align.CENTER, | ||||
|             sensitive: this._canToggle(), | ||||
|             state: this._extension.state === ExtensionState.ENABLED, | ||||
|         }); | ||||
|         this._notifyActiveId = this._switch.connect('notify::active', () => { | ||||
|             if (this._switch.active) | ||||
|                 this._app.shellProxy.EnableExtensionRemote(this.uuid); | ||||
|             else | ||||
|                 this._app.shellProxy.DisableExtensionRemote(this.uuid); | ||||
|         }); | ||||
|         this._switch.connect('state-set', () => true); | ||||
|         hbox.add(this._switch); | ||||
|     } | ||||
|  | ||||
|     _canToggle() { | ||||
|         return this._extension.canChange; | ||||
|     } | ||||
|  | ||||
|     get prefsModule() { | ||||
|         // give extension prefs access to their own extension object | ||||
|         ExtensionUtils.getCurrentExtension = () => this._extension; | ||||
|  | ||||
|         if (!this._prefsModule) { | ||||
|             ExtensionUtils.installImporter(this._extension); | ||||
|  | ||||
|             this._prefsModule = this._extension.imports.prefs; | ||||
|             this._prefsModule.init(this._extension.metadata); | ||||
|         } | ||||
|  | ||||
|         return this._prefsModule; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function initEnvironment() { | ||||
|     // Monkey-patch in a "global" object that fakes some Shell utilities | ||||
|     // that ExtensionUtils depends on. | ||||
|     window.global = { | ||||
|         log(...args) { | ||||
|             print(args.join(', ')); | ||||
|         }, | ||||
|  | ||||
|         logError(s) { | ||||
|             log(`ERROR: ${s}`); | ||||
|         }, | ||||
|  | ||||
|         userdatadir: GLib.build_filenamev([GLib.get_user_data_dir(), 'gnome-shell']), | ||||
|     }; | ||||
|  | ||||
|     String.prototype.format = Format.format; | ||||
| } | ||||
|  | ||||
| function main(argv) { | ||||
|     initEnvironment(); | ||||
|  | ||||
|     new Application().run(argv); | ||||
| } | ||||
| @@ -17,6 +17,10 @@ var DEFAULT_BUTTON_WELL_ANIMATION_TIME = 300; | ||||
|  | ||||
| var MESSAGE_FADE_OUT_ANIMATION_TIME = 500; | ||||
|  | ||||
| const WIGGLE_OFFSET = 6; | ||||
| const WIGGLE_DURATION = 65; | ||||
| const N_WIGGLES = 3; | ||||
|  | ||||
| var AuthPromptMode = { | ||||
|     UNLOCK_ONLY: 0, | ||||
|     UNLOCK_OR_LOG_IN: 1, | ||||
| @@ -47,15 +51,12 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         super._init({ | ||||
|             style_class: 'login-dialog-prompt-layout', | ||||
|             vertical: true, | ||||
|             x_expand: true, | ||||
|             x_align: Clutter.ActorAlign.CENTER, | ||||
|         }); | ||||
|  | ||||
|         this.verificationStatus = AuthPromptStatus.NOT_VERIFYING; | ||||
|  | ||||
|         this._gdmClient = gdmClient; | ||||
|         this._mode = mode; | ||||
|         this._defaultButtonWellActor = null; | ||||
|  | ||||
|         let reauthenticationOnly; | ||||
|         if (this._mode == AuthPromptMode.UNLOCK_ONLY) | ||||
| @@ -74,6 +75,15 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this._userVerifier.connect('ovirt-user-authenticated', this._onOVirtUserAuthenticated.bind(this)); | ||||
|         this.smartcardDetected = this._userVerifier.smartcardDetected; | ||||
|  | ||||
|         this.connect('next', () => { | ||||
|             this.updateSensitivity(false); | ||||
|             this.startSpinning(); | ||||
|             if (this._queryingService) | ||||
|                 this._userVerifier.answerQuery(this._queryingService, this._entry.text); | ||||
|             else | ||||
|                 this._preemptiveAnswer = this._entry.text; | ||||
|         }); | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|  | ||||
|         this._userWell = new St.Bin({ | ||||
| @@ -81,75 +91,19 @@ var AuthPrompt = GObject.registerClass({ | ||||
|             y_expand: true, | ||||
|         }); | ||||
|         this.add_child(this._userWell); | ||||
|  | ||||
|         this._hasCancelButton = this._mode === AuthPromptMode.UNLOCK_OR_LOG_IN; | ||||
|  | ||||
|         this._initEntryRow(); | ||||
|  | ||||
|         let capsLockPlaceholder = new St.Label(); | ||||
|         this.add_child(capsLockPlaceholder); | ||||
|  | ||||
|         this._capsLockWarningLabel = new ShellEntry.CapsLockWarning({ | ||||
|             x_expand: true, | ||||
|             x_align: Clutter.ActorAlign.CENTER, | ||||
|         }); | ||||
|         this.add_child(this._capsLockWarningLabel); | ||||
|  | ||||
|         this._capsLockWarningLabel.bind_property('visible', | ||||
|             capsLockPlaceholder, 'visible', | ||||
|             GObject.BindingFlags.SYNC_CREATE | GObject.BindingFlags.INVERT_BOOLEAN); | ||||
|  | ||||
|         this._message = new St.Label({ | ||||
|             opacity: 0, | ||||
|             styleClass: 'login-dialog-message', | ||||
|         this._label = new St.Label({ | ||||
|             style_class: 'login-dialog-prompt-label', | ||||
|             x_expand: false, | ||||
|             y_expand: true, | ||||
|             x_expand: true, | ||||
|             y_align: Clutter.ActorAlign.START, | ||||
|             x_align: Clutter.ActorAlign.CENTER, | ||||
|         }); | ||||
|         this._message.clutter_text.line_wrap = true; | ||||
|         this._message.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this.add_child(this._message); | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
|         this._userVerifier.destroy(); | ||||
|         this._userVerifier = null; | ||||
|     } | ||||
|  | ||||
|     vfunc_key_press_event(keyPressEvent) { | ||||
|         if (keyPressEvent.keyval == Clutter.KEY_Escape) | ||||
|             this.cancel(); | ||||
|         return super.vfunc_key_press_event(keyPressEvent); | ||||
|     } | ||||
|  | ||||
|     _initEntryRow() { | ||||
|         this._mainBox = new St.BoxLayout({ | ||||
|             style_class: 'login-dialog-button-box', | ||||
|             vertical: false, | ||||
|         }); | ||||
|         this.add_child(this._mainBox); | ||||
|  | ||||
|         this.cancelButton = new St.Button({ | ||||
|             style_class: 'modal-dialog-button button cancel-button', | ||||
|             accessible_name: _('Cancel'), | ||||
|             button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|             reactive: this._hasCancelButton, | ||||
|             can_focus: this._hasCancelButton, | ||||
|             x_align: Clutter.ActorAlign.START, | ||||
|             y_align: Clutter.ActorAlign.CENTER, | ||||
|             child: new St.Icon({ icon_name: 'go-previous-symbolic' }), | ||||
|         }); | ||||
|         if (this._hasCancelButton) | ||||
|             this.cancelButton.connect('clicked', () => this.cancel()); | ||||
|         else | ||||
|             this.cancelButton.opacity = 0; | ||||
|         this._mainBox.add_child(this.cancelButton); | ||||
|         this.add_child(this._label); | ||||
|  | ||||
|         let entryParams = { | ||||
|             style_class: 'login-dialog-prompt-entry', | ||||
|             can_focus: true, | ||||
|             x_expand: true, | ||||
|             x_expand: false, | ||||
|             y_expand: true, | ||||
|         }; | ||||
|  | ||||
|         this._entry = null; | ||||
| @@ -161,57 +115,104 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         ShellEntry.addContextMenu(this._passwordEntry, { actionMode: Shell.ActionMode.NONE }); | ||||
|  | ||||
|         this._entry = this._passwordEntry; | ||||
|         this._mainBox.add_child(this._entry); | ||||
|         this.add_child(this._entry); | ||||
|  | ||||
|         this._entry.grab_key_focus(); | ||||
|  | ||||
|         [this._textEntry, this._passwordEntry].forEach(entry => { | ||||
|             entry.clutter_text.connect('text-changed', () => { | ||||
|                 if (!this._userVerifier.hasPendingMessages) | ||||
|                     this._fadeOutMessage(); | ||||
|             }); | ||||
|         this._capsLockWarningLabel = new ShellEntry.CapsLockWarning(); | ||||
|         this.add_child(this._capsLockWarningLabel); | ||||
|  | ||||
|             entry.clutter_text.connect('activate', () => { | ||||
|                 let shouldSpin = entry === this._passwordEntry; | ||||
|                 if (entry.reactive) | ||||
|                     this._activateNext(shouldSpin); | ||||
|             }); | ||||
|         this._message = new St.Label({ | ||||
|             opacity: 0, | ||||
|             styleClass: 'login-dialog-message', | ||||
|             x_expand: false, | ||||
|             y_expand: true, | ||||
|             y_align: Clutter.ActorAlign.START, | ||||
|         }); | ||||
|         this._message.clutter_text.line_wrap = true; | ||||
|         this._message.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this.add_child(this._message); | ||||
|  | ||||
|         this._buttonBox = new St.BoxLayout({ | ||||
|             style_class: 'login-dialog-button-box', | ||||
|             vertical: false, | ||||
|             y_align: Clutter.ActorAlign.END, | ||||
|         }); | ||||
|         this.add_child(this._buttonBox); | ||||
|  | ||||
|         this._defaultButtonWell = new St.Widget({ | ||||
|             layout_manager: new Clutter.BinLayout(), | ||||
|             x_align: Clutter.ActorAlign.END, | ||||
|             y_align: Clutter.ActorAlign.CENTER, | ||||
|         }); | ||||
|         this._defaultButtonWell.add_constraint(new Clutter.BindConstraint({ | ||||
|             source: this.cancelButton, | ||||
|             coordinate: Clutter.BindCoordinate.WIDTH, | ||||
|         })); | ||||
|         this._mainBox.add_child(this._defaultButtonWell); | ||||
|  | ||||
|         this._initButtons(); | ||||
|  | ||||
|         this._spinner = new Animation.Spinner(DEFAULT_BUTTON_WELL_ICON_SIZE); | ||||
|         this._spinner.opacity = 0; | ||||
|         this._spinner.show(); | ||||
|         this._defaultButtonWell.add_child(this._spinner); | ||||
|     } | ||||
|  | ||||
|     _activateNext(shouldSpin) { | ||||
|         this.updateSensitivity(false); | ||||
|     _onDestroy() { | ||||
|         this._userVerifier.destroy(); | ||||
|         this._userVerifier = null; | ||||
|     } | ||||
|  | ||||
|         if (shouldSpin) | ||||
|             this.startSpinning(); | ||||
|     vfunc_key_press_event(keyPressEvent) { | ||||
|         if (keyPressEvent.keyval == Clutter.KEY_Escape) | ||||
|             this.cancel(); | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     } | ||||
|  | ||||
|         if (this._queryingService) | ||||
|             this._userVerifier.answerQuery(this._queryingService, this._entry.text); | ||||
|         else | ||||
|             this._preemptiveAnswer = this._entry.text; | ||||
|     _initButtons() { | ||||
|         this.cancelButton = new St.Button({ | ||||
|             style_class: 'modal-dialog-button button', | ||||
|             button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|             reactive: true, | ||||
|             can_focus: true, | ||||
|             label: _("Cancel"), | ||||
|             x_expand: true, | ||||
|             x_align: Clutter.ActorAlign.START, | ||||
|             y_align: Clutter.ActorAlign.END, | ||||
|         }); | ||||
|         this.cancelButton.connect('clicked', () => this.cancel()); | ||||
|         this._buttonBox.add_child(this.cancelButton); | ||||
|  | ||||
|         this.emit('next'); | ||||
|         this._buttonBox.add_child(this._defaultButtonWell); | ||||
|         this.nextButton = new St.Button({ | ||||
|             style_class: 'modal-dialog-button button', | ||||
|             button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|             reactive: true, | ||||
|             can_focus: true, | ||||
|             label: _("Next"), | ||||
|             x_align: Clutter.ActorAlign.END, | ||||
|             y_align: Clutter.ActorAlign.END, | ||||
|         }); | ||||
|         this.nextButton.connect('clicked', () => this.emit('next')); | ||||
|         this.nextButton.add_style_pseudo_class('default'); | ||||
|         this._buttonBox.add_child(this.nextButton); | ||||
|  | ||||
|         this._updateNextButtonSensitivity(this._entry.text.length > 0); | ||||
|  | ||||
|         this._entry.clutter_text.connect('text-changed', () => { | ||||
|             if (!this._userVerifier.hasPendingMessages) | ||||
|                 this._fadeOutMessage(); | ||||
|  | ||||
|             this._updateNextButtonSensitivity(this._entry.text.length > 0 || this.verificationStatus == AuthPromptStatus.VERIFYING); | ||||
|         }); | ||||
|         this._entry.clutter_text.connect('activate', () => { | ||||
|             if (this.nextButton.reactive) | ||||
|                 this.emit('next'); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _updateEntry(secret) { | ||||
|         if (secret && this._entry !== this._passwordEntry) { | ||||
|             this._mainBox.replace_child(this._entry, this._passwordEntry); | ||||
|         if (secret && (this._entry != this._passwordEntry)) { | ||||
|             this.replace_child(this._entry, this._passwordEntry); | ||||
|             this._entry = this._passwordEntry; | ||||
|         } else if (!secret && this._entry !== this._textEntry) { | ||||
|             this._mainBox.replace_child(this._entry, this._textEntry); | ||||
|         } else if (!secret && (this._entry != this._textEntry)) { | ||||
|             this.replace_child(this._entry, this._textEntry); | ||||
|             this._entry = this._textEntry; | ||||
|         } | ||||
|         this._capsLockWarningLabel.visible = secret; | ||||
| @@ -229,14 +230,16 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         } | ||||
|  | ||||
|         this._updateEntry(secret); | ||||
|         this.setQuestion(question); | ||||
|  | ||||
|         // Hack: The question string comes directly from PAM, if it's "Password:" | ||||
|         // we replace it with our own to allow localization, if it's something | ||||
|         // else we remove the last colon and any trailing or leading spaces. | ||||
|         if (question === 'Password:' || question === 'Password: ') | ||||
|             this.setQuestion(_('Password')); | ||||
|         else | ||||
|             this.setQuestion(question.replace(/: *$/, '').trim()); | ||||
|         if (secret) { | ||||
|             if (this._userVerifier.reauthenticating) | ||||
|                 this.nextButton.label = _("Unlock"); | ||||
|             else | ||||
|                 this.nextButton.label = C_("button", "Sign In"); | ||||
|         } else { | ||||
|             this.nextButton.label = _("Next"); | ||||
|         } | ||||
|  | ||||
|         this.updateSensitivity(true); | ||||
|         this.emit('prompted'); | ||||
| @@ -279,14 +282,17 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this.setActorInDefaultButtonWell(null); | ||||
|         this.verificationStatus = AuthPromptStatus.VERIFICATION_FAILED; | ||||
|  | ||||
|         Util.wiggle(this._entry); | ||||
|         Util.wiggle(this._entry, { | ||||
|             offset: WIGGLE_OFFSET, | ||||
|             duration: WIGGLE_DURATION, | ||||
|             wiggleCount: N_WIGGLES, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _onVerificationComplete() { | ||||
|         this.setActorInDefaultButtonWell(null); | ||||
|         this.verificationStatus = AuthPromptStatus.VERIFICATION_SUCCEEDED; | ||||
|         this.cancelButton.reactive = false; | ||||
|         this.cancelButton.can_focus = false; | ||||
|     } | ||||
|  | ||||
|     _onReset() { | ||||
| @@ -294,6 +300,10 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this.reset(); | ||||
|     } | ||||
|  | ||||
|     addActorToDefaultButtonWell(actor) { | ||||
|         this._defaultButtonWell.add_child(actor); | ||||
|     } | ||||
|  | ||||
|     setActorInDefaultButtonWell(actor, animate) { | ||||
|         if (!this._defaultButtonWellActor && | ||||
|             !actor) | ||||
| @@ -373,9 +383,11 @@ var AuthPrompt = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     setQuestion(question) { | ||||
|         this._entry.hint_text = question; | ||||
|         this._label.set_text(question); | ||||
|  | ||||
|         this._label.show(); | ||||
|         this._entry.show(); | ||||
|  | ||||
|         this._entry.grab_key_focus(); | ||||
|     } | ||||
|  | ||||
| @@ -423,14 +435,15 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _updateNextButtonSensitivity(sensitive) { | ||||
|         this.nextButton.reactive = sensitive; | ||||
|         this.nextButton.can_focus = sensitive; | ||||
|     } | ||||
|  | ||||
|     updateSensitivity(sensitive) { | ||||
|         if (this._entry.reactive === sensitive) | ||||
|             return; | ||||
|  | ||||
|         this._updateNextButtonSensitivity(sensitive && (this._entry.text.length > 0 || this.verificationStatus == AuthPromptStatus.VERIFYING)); | ||||
|         this._entry.reactive = sensitive; | ||||
|  | ||||
|         if (sensitive) | ||||
|             this._entry.grab_key_focus(); | ||||
|         this._entry.clutter_text.editable = sensitive; | ||||
|     } | ||||
|  | ||||
|     vfunc_hide() { | ||||
| @@ -449,18 +462,18 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         if (oldChild) | ||||
|             oldChild.destroy(); | ||||
|  | ||||
|         let userWidget = new UserWidget.UserWidget(user, Clutter.Orientation.VERTICAL); | ||||
|         this._userWell.set_child(userWidget); | ||||
|  | ||||
|         if (!user) | ||||
|             this._updateEntry(false); | ||||
|         if (user) { | ||||
|             let userWidget = new UserWidget.UserWidget(user); | ||||
|             userWidget.x_align = Clutter.ActorAlign.START; | ||||
|             this._userWell.set_child(userWidget); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     reset() { | ||||
|         let oldStatus = this.verificationStatus; | ||||
|         this.verificationStatus = AuthPromptStatus.NOT_VERIFYING; | ||||
|         this.cancelButton.reactive = this._hasCancelButton; | ||||
|         this.cancelButton.can_focus = this._hasCancelButton; | ||||
|         this.cancelButton.reactive = true; | ||||
|         this.nextButton.label = _("Next"); | ||||
|         this._preemptiveAnswer = null; | ||||
|  | ||||
|         if (this._userVerifier) | ||||
| @@ -470,7 +483,6 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this.clear(); | ||||
|         this._message.opacity = 0; | ||||
|         this.setUser(null); | ||||
|         this._updateEntry(true); | ||||
|         this.stopSpinning(); | ||||
|  | ||||
|         if (oldStatus == AuthPromptStatus.VERIFICATION_FAILED) | ||||
|   | ||||
| @@ -35,6 +35,8 @@ const UserWidget = imports.ui.userWidget; | ||||
| const _FADE_ANIMATION_TIME = 250; | ||||
| const _SCROLL_ANIMATION_TIME = 500; | ||||
| const _TIMED_LOGIN_IDLE_THRESHOLD = 5.0; | ||||
| const _LOGO_ICON_HEIGHT = 48; | ||||
| const _MAX_BOTTOM_MENU_ITEMS = 5; | ||||
|  | ||||
| var UserListItem = GObject.registerClass({ | ||||
|     Signals: { 'activate': {} }, | ||||
| @@ -177,7 +179,6 @@ var UserList = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     vfunc_key_focus_in() { | ||||
|         super.vfunc_key_focus_in(); | ||||
|         this._moveFocusToItems(); | ||||
|     } | ||||
|  | ||||
| @@ -311,21 +312,28 @@ var SessionMenuButton = GObject.registerClass({ | ||||
|     _init() { | ||||
|         let gearIcon = new St.Icon({ icon_name: 'emblem-system-symbolic' }); | ||||
|         let button = new St.Button({ | ||||
|             style_class: 'modal-dialog-button button login-dialog-session-list-button', | ||||
|             style_class: 'login-dialog-session-list-button', | ||||
|             reactive: true, | ||||
|             track_hover: true, | ||||
|             can_focus: true, | ||||
|             accessible_name: _("Choose Session"), | ||||
|             accessible_role: Atk.Role.MENU, | ||||
|             x_align: Clutter.ActorAlign.CENTER, | ||||
|             y_align: Clutter.ActorAlign.CENTER, | ||||
|             child: gearIcon, | ||||
|         }); | ||||
|  | ||||
|         super._init({ child: button }); | ||||
|         this._button = button; | ||||
|  | ||||
|         this._menu = new PopupMenu.PopupMenu(this._button, 0, St.Side.BOTTOM); | ||||
|         let side = St.Side.TOP; | ||||
|         let align = 0; | ||||
|         if (Gdm.get_session_ids().length > _MAX_BOTTOM_MENU_ITEMS) { | ||||
|             if (this.text_direction == Clutter.TextDirection.RTL) | ||||
|                 side = St.Side.RIGHT; | ||||
|             else | ||||
|                 side = St.Side.LEFT; | ||||
|             align = 0.5; | ||||
|         } | ||||
|         this._menu = new PopupMenu.PopupMenu(this._button, align, side); | ||||
|         Main.uiGroup.add_actor(this._menu.actor); | ||||
|         this._menu.actor.hide(); | ||||
|  | ||||
| @@ -350,7 +358,6 @@ var SessionMenuButton = GObject.registerClass({ | ||||
|     updateSensitivity(sensitive) { | ||||
|         this._button.reactive = sensitive; | ||||
|         this._button.can_focus = sensitive; | ||||
|         this.opacity = sensitive ? 255 : 0; | ||||
|         this._menu.close(BoxPointer.PopupAnimation.NONE); | ||||
|     } | ||||
|  | ||||
| @@ -402,10 +409,7 @@ var SessionMenuButton = GObject.registerClass({ | ||||
| }); | ||||
|  | ||||
| var LoginDialog = GObject.registerClass({ | ||||
|     Signals: { | ||||
|         'failed': {}, | ||||
|         'wake-up-screen': {}, | ||||
|     }, | ||||
|     Signals: { 'failed': {} }, | ||||
| }, class LoginDialog extends St.Widget { | ||||
|     _init(parentActor) { | ||||
|         super._init({ style_class: 'login-dialog', visible: false }); | ||||
| @@ -421,13 +425,13 @@ var LoginDialog = GObject.registerClass({ | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema_id: GdmUtil.LOGIN_SCREEN_SCHEMA }); | ||||
|  | ||||
|         this._settings.connect('changed::%s'.format(GdmUtil.BANNER_MESSAGE_KEY), | ||||
|         this._settings.connect(`changed::${GdmUtil.BANNER_MESSAGE_KEY}`, | ||||
|                                this._updateBanner.bind(this)); | ||||
|         this._settings.connect('changed::%s'.format(GdmUtil.BANNER_MESSAGE_TEXT_KEY), | ||||
|         this._settings.connect(`changed::${GdmUtil.BANNER_MESSAGE_TEXT_KEY}`, | ||||
|                                this._updateBanner.bind(this)); | ||||
|         this._settings.connect('changed::%s'.format(GdmUtil.DISABLE_USER_LIST_KEY), | ||||
|         this._settings.connect(`changed::${GdmUtil.DISABLE_USER_LIST_KEY}`, | ||||
|                                this._updateDisableUserList.bind(this)); | ||||
|         this._settings.connect('changed::%s'.format(GdmUtil.LOGO_KEY), | ||||
|         this._settings.connect(`changed::${GdmUtil.LOGO_KEY}`, | ||||
|                                this._updateLogo.bind(this)); | ||||
|  | ||||
|         this._textureCache = St.TextureCache.get_default(); | ||||
| @@ -456,6 +460,7 @@ var LoginDialog = GObject.registerClass({ | ||||
|         let notListedLabel = new St.Label({ | ||||
|             text: _("Not listed?"), | ||||
|             style_class: 'login-dialog-not-listed-label', | ||||
|             x_align: Clutter.ActorAlign.START, | ||||
|         }); | ||||
|         this._notListedButton = new St.Button({ | ||||
|             style_class: 'login-dialog-not-listed-button', | ||||
| @@ -463,7 +468,6 @@ var LoginDialog = GObject.registerClass({ | ||||
|             can_focus: true, | ||||
|             child: notListedLabel, | ||||
|             reactive: true, | ||||
|             x_align: Clutter.ActorAlign.START, | ||||
|         }); | ||||
|  | ||||
|         this._notListedButton.connect('clicked', this._hideUserListAskForUsernameAndBeginVerification.bind(this)); | ||||
| @@ -488,15 +492,6 @@ var LoginDialog = GObject.registerClass({ | ||||
|         bannerBox.add_child(this._bannerLabel); | ||||
|         this._updateBanner(); | ||||
|  | ||||
|         this._sessionMenuButton = new SessionMenuButton(); | ||||
|         this._sessionMenuButton.connect('session-activated', | ||||
|             (list, sessionId) => { | ||||
|                 this._greeter.call_select_session_sync(sessionId, null); | ||||
|             }); | ||||
|         this._sessionMenuButton.opacity = 0; | ||||
|         this._sessionMenuButton.show(); | ||||
|         this.add_child(this._sessionMenuButton); | ||||
|  | ||||
|         this._logoBin = new St.Widget({ style_class: 'login-dialog-logo-bin', | ||||
|                                         x_align: Clutter.ActorAlign.CENTER, | ||||
|                                         y_align: Clutter.ActorAlign.END }); | ||||
| @@ -510,6 +505,16 @@ var LoginDialog = GObject.registerClass({ | ||||
|             this._onUserListActivated(item); | ||||
|         }); | ||||
|  | ||||
|  | ||||
|         this._sessionMenuButton = new SessionMenuButton(); | ||||
|         this._sessionMenuButton.connect('session-activated', | ||||
|             (list, sessionId) => { | ||||
|                 this._greeter.call_select_session_sync(sessionId, null); | ||||
|             }); | ||||
|         this._sessionMenuButton.opacity = 0; | ||||
|         this._sessionMenuButton.show(); | ||||
|         this._authPrompt.addActorToDefaultButtonWell(this._sessionMenuButton); | ||||
|  | ||||
|         this._disableUserList = undefined; | ||||
|         this._userListLoaded = false; | ||||
|  | ||||
| @@ -554,23 +559,6 @@ var LoginDialog = GObject.registerClass({ | ||||
|         return actorBox; | ||||
|     } | ||||
|  | ||||
|     _getSessionMenuButtonAllocation(dialogBox) { | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [, , natWidth, natHeight] = this._sessionMenuButton.get_preferred_size(); | ||||
|  | ||||
|         if (this.get_text_direction() === Clutter.TextDirection.RTL) | ||||
|             actorBox.x1 = dialogBox.x1 + natWidth; | ||||
|         else | ||||
|             actorBox.x1 = dialogBox.x2 - (natWidth * 2); | ||||
|  | ||||
|         actorBox.y1 = dialogBox.y2 - (natHeight * 2); | ||||
|         actorBox.x2 = actorBox.x1 + natWidth; | ||||
|         actorBox.y2 = actorBox.y1 + natHeight; | ||||
|  | ||||
|         return actorBox; | ||||
|     } | ||||
|  | ||||
|     _getCenterActorAllocation(dialogBox, actor) { | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|  | ||||
| @@ -589,8 +577,8 @@ var LoginDialog = GObject.registerClass({ | ||||
|         return actorBox; | ||||
|     } | ||||
|  | ||||
|     vfunc_allocate(dialogBox) { | ||||
|         this.set_allocation(dialogBox); | ||||
|     vfunc_allocate(dialogBox, flags) { | ||||
|         this.set_allocation(dialogBox, flags); | ||||
|  | ||||
|         let themeNode = this.get_theme_node(); | ||||
|         dialogBox = themeNode.get_content_box(dialogBox); | ||||
| @@ -627,10 +615,6 @@ var LoginDialog = GObject.registerClass({ | ||||
|             logoHeight = logoAllocation.y2 - logoAllocation.y1; | ||||
|         } | ||||
|  | ||||
|         let sessionMenuButtonAllocation = null; | ||||
|         if (this._sessionMenuButton.visible) | ||||
|             sessionMenuButtonAllocation = this._getSessionMenuButtonAllocation(dialogBox); | ||||
|  | ||||
|         // Then figure out if we're overly constrained and need to | ||||
|         // try a different layout, or if we have what extra space we | ||||
|         // can hand out | ||||
| @@ -719,19 +703,16 @@ var LoginDialog = GObject.registerClass({ | ||||
|  | ||||
|         // Finally hand out the allocations | ||||
|         if (bannerAllocation) | ||||
|             this._bannerView.allocate(bannerAllocation); | ||||
|             this._bannerView.allocate(bannerAllocation, flags); | ||||
|  | ||||
|         if (authPromptAllocation) | ||||
|             this._authPrompt.allocate(authPromptAllocation); | ||||
|             this._authPrompt.allocate(authPromptAllocation, flags); | ||||
|  | ||||
|         if (userSelectionAllocation) | ||||
|             this._userSelectionBox.allocate(userSelectionAllocation); | ||||
|             this._userSelectionBox.allocate(userSelectionAllocation, flags); | ||||
|  | ||||
|         if (logoAllocation) | ||||
|             this._logoBin.allocate(logoAllocation); | ||||
|  | ||||
|         if (sessionMenuButtonAllocation) | ||||
|             this._sessionMenuButton.allocate(sessionMenuButtonAllocation); | ||||
|             this._logoBin.allocate(logoAllocation, flags); | ||||
|     } | ||||
|  | ||||
|     _ensureUserListLoaded() { | ||||
| @@ -810,13 +791,12 @@ var LoginDialog = GObject.registerClass({ | ||||
|             return; | ||||
|  | ||||
|         this._logoBin.destroy_all_children(); | ||||
|         const [valid, resourceScale] = this._logoBin.get_resource_scale(); | ||||
|         if (this._logoFile && valid) { | ||||
|         if (this._logoFile && this._logoBin.resource_scale > 0) { | ||||
|             let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|             this._logoBin.add_child(this._textureCache.load_file_async(this._logoFile, | ||||
|                                                                        -1, -1, | ||||
|                                                                        -1, _LOGO_ICON_HEIGHT, | ||||
|                                                                        scaleFactor, | ||||
|                                                                        resourceScale)); | ||||
|                                                                        this._logoBin.resource_scale)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -828,10 +808,12 @@ var LoginDialog = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     _onPrompted() { | ||||
|         const showSessionMenu = this._shouldShowSessionMenuButton(); | ||||
|  | ||||
|         this._sessionMenuButton.updateSensitivity(showSessionMenu); | ||||
|         this._sessionMenuButton.visible = showSessionMenu; | ||||
|         if (this._shouldShowSessionMenuButton()) { | ||||
|             this._sessionMenuButton.updateSensitivity(true); | ||||
|             this._authPrompt.setActorInDefaultButtonWell(this._sessionMenuButton); | ||||
|         } else { | ||||
|             this._sessionMenuButton.updateSensitivity(false); | ||||
|         } | ||||
|         this._showPrompt(); | ||||
|     } | ||||
|  | ||||
| @@ -914,8 +896,7 @@ var LoginDialog = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     _askForUsernameAndBeginVerification() { | ||||
|         this._authPrompt.setUser(null); | ||||
|         this._authPrompt.setQuestion(_('Username')); | ||||
|         this._authPrompt.setQuestion(_("Username: ")); | ||||
|  | ||||
|         this._showRealmLoginHint(this._realmManager.loginFormat); | ||||
|  | ||||
| @@ -929,6 +910,7 @@ var LoginDialog = GObject.registerClass({ | ||||
|                 let answer = this._authPrompt.getAnswer(); | ||||
|                 this._user = this._userManager.get_user(answer); | ||||
|                 this._authPrompt.clear(); | ||||
|                 this._authPrompt.startSpinning(); | ||||
|                 this._authPrompt.begin({ userName: answer }); | ||||
|                 this._updateCancelButton(); | ||||
|             }); | ||||
| @@ -1140,7 +1122,6 @@ var LoginDialog = GObject.registerClass({ | ||||
|         this._authPrompt.hide(); | ||||
|         this._hideBannerView(); | ||||
|         this._sessionMenuButton.close(); | ||||
|         this._sessionMenuButton.hide(); | ||||
|         this._setUserListExpanded(true); | ||||
|         this._notListedButton.show(); | ||||
|         this._userList.grab_key_focus(); | ||||
| @@ -1244,18 +1225,13 @@ var LoginDialog = GObject.registerClass({ | ||||
|         return GLib.SOURCE_REMOVE; | ||||
|     } | ||||
|  | ||||
|     activate() { | ||||
|         this._userList.grab_key_focus(); | ||||
|         this.show(); | ||||
|     } | ||||
|  | ||||
|     open() { | ||||
|         Main.ctrlAltTabManager.addGroup(this, | ||||
|                                         _("Login Window"), | ||||
|                                         'dialog-password-symbolic', | ||||
|                                         { sortGroup: CtrlAltTab.SortGroup.MIDDLE }); | ||||
|         this.activate(); | ||||
|  | ||||
|         this._userList.grab_key_focus(); | ||||
|         this.show(); | ||||
|         this.opacity = 0; | ||||
|  | ||||
|         Main.pushModal(this, { actionMode: Shell.ActionMode.LOGIN_SCREEN }); | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| /* exported BANNER_MESSAGE_KEY, BANNER_MESSAGE_TEXT_KEY, LOGO_KEY, | ||||
|             DISABLE_USER_LIST_KEY, fadeInActor, fadeOutActor, cloneAndFadeOutActor */ | ||||
|  | ||||
| const { Clutter, Gdm, Gio, GLib } = imports.gi; | ||||
| const { Clutter, Gio, GLib } = imports.gi; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Batch = imports.gdm.batch; | ||||
| @@ -12,15 +12,6 @@ const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const SmartcardManager = imports.misc.smartcardManager; | ||||
|  | ||||
| Gio._promisify(Gdm.Client.prototype, | ||||
|     'open_reauthentication_channel', 'open_reauthentication_channel_finish'); | ||||
| Gio._promisify(Gdm.Client.prototype, | ||||
|     'get_user_verifier', 'get_user_verifier_finish'); | ||||
| Gio._promisify(Gdm.UserVerifierProxy.prototype, | ||||
|     'call_begin_verification_for_user', 'call_begin_verification_for_user_finish'); | ||||
| Gio._promisify(Gdm.UserVerifierProxy.prototype, | ||||
|     'call_begin_verification', 'call_begin_verification_finish'); | ||||
|  | ||||
| var PASSWORD_SERVICE_NAME = 'gdm-password'; | ||||
| var FINGERPRINT_SERVICE_NAME = 'gdm-fingerprint'; | ||||
| var SMARTCARD_SERVICE_NAME = 'gdm-smartcard'; | ||||
| @@ -177,12 +168,14 @@ var ShellUserVerifier = class { | ||||
|  | ||||
|         this._checkForFingerprintReader(); | ||||
|  | ||||
|         // If possible, reauthenticate an already running session, | ||||
|         // so any session specific credentials get updated appropriately | ||||
|         if (userName) | ||||
|             this._openReauthenticationChannel(userName); | ||||
|         else | ||||
|             this._getUserVerifier(); | ||||
|         if (userName) { | ||||
|             // If possible, reauthenticate an already running session, | ||||
|             // so any session specific credentials get updated appropriately | ||||
|             this._client.open_reauthentication_channel(userName, this._cancellable, | ||||
|                                                        this._reauthenticationChannelOpened.bind(this)); | ||||
|         } else { | ||||
|             this._client.get_user_verifier(this._cancellable, this._userVerifierGot.bind(this)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     cancel() { | ||||
| @@ -346,11 +339,10 @@ var ShellUserVerifier = class { | ||||
|         this._verificationFailed(false); | ||||
|     } | ||||
|  | ||||
|     async _openReauthenticationChannel(userName) { | ||||
|     _reauthenticationChannelOpened(client, result) { | ||||
|         try { | ||||
|             this._clearUserVerifier(); | ||||
|             this._userVerifier = await this._client.open_reauthentication_channel( | ||||
|                 userName, this._cancellable); | ||||
|             this._userVerifier = client.open_reauthentication_channel_finish(result); | ||||
|         } catch (e) { | ||||
|             if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                 return; | ||||
| @@ -359,7 +351,8 @@ var ShellUserVerifier = class { | ||||
|                 // Gdm emits org.freedesktop.DBus.Error.AccessDenied when there | ||||
|                 // is no session to reauthenticate. Fall back to performing | ||||
|                 // verification from this login session | ||||
|                 this._getUserVerifier(); | ||||
|                 client.get_user_verifier(this._cancellable, | ||||
|                                          this._userVerifierGot.bind(this)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
| @@ -373,11 +366,10 @@ var ShellUserVerifier = class { | ||||
|         this._hold.release(); | ||||
|     } | ||||
|  | ||||
|     async _getUserVerifier() { | ||||
|     _userVerifierGot(client, result) { | ||||
|         try { | ||||
|             this._clearUserVerifier(); | ||||
|             this._userVerifier = | ||||
|                 await this._client.get_user_verifier(this._cancellable); | ||||
|             this._userVerifier = client.get_user_verifier_finish(result); | ||||
|         } catch (e) { | ||||
|             if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                 return; | ||||
| @@ -429,25 +421,35 @@ var ShellUserVerifier = class { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async _startService(serviceName) { | ||||
|     _startService(serviceName) { | ||||
|         this._hold.acquire(); | ||||
|         try { | ||||
|             if (this._userName) { | ||||
|                 await this._userVerifier.call_begin_verification_for_user( | ||||
|                     serviceName, this._userName, this._cancellable); | ||||
|             } else { | ||||
|                 await this._userVerifier.call_begin_verification( | ||||
|                     serviceName, this._cancellable); | ||||
|             } | ||||
|         } catch (e) { | ||||
|             if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                 return; | ||||
|             this._reportInitError(this._userName | ||||
|                 ? 'Failed to start verification for user' | ||||
|                 : 'Failed to start verification', e); | ||||
|             return; | ||||
|         if (this._userName) { | ||||
|             this._userVerifier.call_begin_verification_for_user(serviceName, this._userName, this._cancellable, (obj, result) => { | ||||
|                 try { | ||||
|                     obj.call_begin_verification_for_user_finish(result); | ||||
|                 } catch (e) { | ||||
|                     if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                         return; | ||||
|                     this._reportInitError('Failed to start verification for user', e); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 this._hold.release(); | ||||
|             }); | ||||
|         } else { | ||||
|             this._userVerifier.call_begin_verification(serviceName, this._cancellable, (obj, result) => { | ||||
|                 try { | ||||
|                     obj.call_begin_verification_finish(result); | ||||
|                 } catch (e) { | ||||
|                     if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                         return; | ||||
|                     this._reportInitError('Failed to start verification', e); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 this._hold.release(); | ||||
|             }); | ||||
|         } | ||||
|         this._hold.release(); | ||||
|     } | ||||
|  | ||||
|     _beginVerification() { | ||||
|   | ||||
| @@ -23,7 +23,6 @@ | ||||
|     <file>misc/modemManager.js</file> | ||||
|     <file>misc/objectManager.js</file> | ||||
|     <file>misc/params.js</file> | ||||
|     <file>misc/parentalControlsManager.js</file> | ||||
|     <file>misc/permissionStore.js</file> | ||||
|     <file>misc/smartcardManager.js</file> | ||||
|     <file>misc/systemActions.js</file> | ||||
| @@ -102,13 +101,13 @@ | ||||
|     <file>ui/swipeTracker.js</file> | ||||
|     <file>ui/switcherPopup.js</file> | ||||
|     <file>ui/switchMonitor.js</file> | ||||
|     <file>ui/tweener.js</file> | ||||
|     <file>ui/unlockDialog.js</file> | ||||
|     <file>ui/userWidget.js</file> | ||||
|     <file>ui/viewSelector.js</file> | ||||
|     <file>ui/windowAttentionHandler.js</file> | ||||
|     <file>ui/windowMenu.js</file> | ||||
|     <file>ui/windowManager.js</file> | ||||
|     <file>ui/windowPreview.js</file> | ||||
|     <file>ui/workspace.js</file> | ||||
|     <file>ui/workspaceSwitcherPopup.js</file> | ||||
|     <file>ui/workspaceThumbnail.js</file> | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| subdir('misc') | ||||
| subdir('dbusServices') | ||||
|  | ||||
| js_resources = gnome.compile_resources( | ||||
|   'js-resources', 'js-resources.gresource.xml', | ||||
| @@ -14,3 +13,10 @@ portal_resources = gnome.compile_resources( | ||||
|   c_name: 'portal_js_resources', | ||||
|   dependencies: [config_js] | ||||
| ) | ||||
|  | ||||
| prefs_resources = gnome.compile_resources( | ||||
|   'prefs-resources', 'prefs-resources.gresource.xml', | ||||
|   source_dir: ['.', meson.current_build_dir()], | ||||
|   c_name: 'prefs_js_resources', | ||||
|   dependencies: [config_js] | ||||
| ) | ||||
|   | ||||
| @@ -15,5 +15,6 @@ var LOCALEDIR = '@datadir@/locale'; | ||||
| /* other standard directories */ | ||||
| var LIBEXECDIR = '@libexecdir@'; | ||||
| var PKGDATADIR = '@datadir@/@PACKAGE_NAME@'; | ||||
| var VPNDIR = '@vpndir@'; | ||||
| /* g-i package versions */ | ||||
| var LIBMUTTER_API_VERSION = '@LIBMUTTER_API_VERSION@' | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported ExtensionState, ExtensionType, getCurrentExtension, | ||||
|    getSettings, initTranslations, openPrefs, isOutOfDate, | ||||
|    installImporter, serializeExtension, deserializeExtension */ | ||||
|    getSettings, initTranslations, isOutOfDate, installImporter, | ||||
|    serializeExtension, deserializeExtension */ | ||||
|  | ||||
| // Common utils for the extension system and the extension | ||||
| // preferences tool | ||||
| @@ -153,27 +153,6 @@ function getSettings(schema) { | ||||
|     return new Gio.Settings({ settings_schema: schemaObj }); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * openPrefs: | ||||
|  * | ||||
|  * Open the preference dialog of the current extension | ||||
|  */ | ||||
| function openPrefs() { | ||||
|     const extension = getCurrentExtension(); | ||||
|  | ||||
|     if (!extension) | ||||
|         throw new Error('openPrefs() can only be called from extensions'); | ||||
|  | ||||
|     try { | ||||
|         const extensionManager = imports.ui.main.extensionManager; | ||||
|         extensionManager.openExtensionPrefs(extension.uuid, '', {}); | ||||
|     } catch (e) { | ||||
|         if (e.name === 'ImportError') | ||||
|             throw new Error('openPrefs() cannot be called from preferences'); | ||||
|         logError(e, 'Failed to open extension preferences'); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * versionCheck: | ||||
|  * @param {string[]} required - an array of versions we're compatible with | ||||
|   | ||||
| @@ -76,15 +76,19 @@ function loadInterfaceXML(iface) { | ||||
|         _ifaceResource._register(); | ||||
|     } | ||||
|  | ||||
|     let xml = null; | ||||
|     let uri = `resource:///org/gnome/shell/dbus-interfaces/${iface}.xml`; | ||||
|     let f = Gio.File.new_for_uri(uri); | ||||
|  | ||||
|     try { | ||||
|         let [ok_, bytes] = f.load_contents(null); | ||||
|         return imports.byteArray.toString(bytes); | ||||
|         if (bytes instanceof Uint8Array) | ||||
|             xml = imports.byteArray.toString(bytes); | ||||
|         else | ||||
|             xml = bytes.toString(); | ||||
|     } catch (e) { | ||||
|         log(`Failed to load D-Bus interface ${iface}`); | ||||
|     } | ||||
|  | ||||
|     return null; | ||||
|     return xml; | ||||
| } | ||||
|   | ||||
| @@ -1,20 +1,11 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported getIBusManager */ | ||||
|  | ||||
| const { Gio, GLib, IBus, Meta } = imports.gi; | ||||
| const { Gio, GLib, IBus } = imports.gi; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const IBusCandidatePopup = imports.ui.ibusCandidatePopup; | ||||
|  | ||||
| Gio._promisify(IBus.Bus.prototype, | ||||
|     'list_engines_async', 'list_engines_async_finish'); | ||||
| Gio._promisify(IBus.Bus.prototype, | ||||
|     'request_name_async', 'request_name_async_finish'); | ||||
| Gio._promisify(IBus.Bus.prototype, | ||||
|     'get_global_engine_async', 'get_global_engine_async_finish'); | ||||
| Gio._promisify(IBus.Bus.prototype, | ||||
|     'set_global_engine_async', 'set_global_engine_async_finish'); | ||||
|  | ||||
| // Ensure runtime version matches | ||||
| _checkIBusVersion(1, 5, 2); | ||||
|  | ||||
| @@ -64,18 +55,13 @@ var IBusManager = class { | ||||
|         this._ibus.set_watch_ibus_signal(true); | ||||
|         this._ibus.connect('global-engine-changed', this._engineChanged.bind(this)); | ||||
|  | ||||
|         this._spawn(Meta.is_wayland_compositor() ? [] : ['--xim']); | ||||
|         this._spawn(); | ||||
|     } | ||||
|  | ||||
|     _spawn(extraArgs = []) { | ||||
|         try { | ||||
|             let cmdLine = ['ibus-daemon', '--panel', 'disable', ...extraArgs]; | ||||
|             let launcher = Gio.SubprocessLauncher.new(Gio.SubprocessFlags.NONE); | ||||
|             // Forward the right X11 Display for ibus-x11 | ||||
|             let display = GLib.getenv('GNOME_SETUP_DISPLAY'); | ||||
|             if (display) | ||||
|                 launcher.setenv('DISPLAY', display, true); | ||||
|             launcher.spawnv(cmdLine); | ||||
|             Gio.Subprocess.new(cmdLine, Gio.SubprocessFlags.NONE); | ||||
|         } catch (e) { | ||||
|             log(`Failed to launch ibus-daemon: ${e.message}`); | ||||
|         } | ||||
| @@ -111,14 +97,16 @@ var IBusManager = class { | ||||
|  | ||||
|     _onConnected() { | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|         this._initEngines(); | ||||
|         this._initPanelService(); | ||||
|         this._ibus.list_engines_async(-1, this._cancellable, | ||||
|             this._initEngines.bind(this)); | ||||
|         this._ibus.request_name_async(IBus.SERVICE_PANEL, | ||||
|             IBus.BusNameFlag.REPLACE_EXISTING, -1, this._cancellable, | ||||
|             this._initPanelService.bind(this)); | ||||
|     } | ||||
|  | ||||
|     async _initEngines() { | ||||
|     _initEngines(ibus, result) { | ||||
|         try { | ||||
|             const enginesList = | ||||
|                 await this._ibus.list_engines_async(-1, this._cancellable); | ||||
|             let enginesList = this._ibus.list_engines_async_finish(result); | ||||
|             for (let i = 0; i < enginesList.length; ++i) { | ||||
|                 let name = enginesList[i].get_name(); | ||||
|                 this._engines.set(name, enginesList[i]); | ||||
| @@ -133,52 +121,56 @@ var IBusManager = class { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async _initPanelService() { | ||||
|     _initPanelService(ibus, result) { | ||||
|         let success = false; | ||||
|         try { | ||||
|             await this._ibus.request_name_async(IBus.SERVICE_PANEL, | ||||
|                 IBus.BusNameFlag.REPLACE_EXISTING, -1, this._cancellable); | ||||
|             success = !!this._ibus.request_name_async_finish(result); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|                 logError(e); | ||||
|                 this._clear(); | ||||
|             if (e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                 return; | ||||
|             logError(e); | ||||
|         } | ||||
|  | ||||
|         if (success) { | ||||
|             this._panelService = new IBus.PanelService({ connection: this._ibus.get_connection(), | ||||
|                                                          object_path: IBus.PATH_PANEL }); | ||||
|             this._candidatePopup.setPanelService(this._panelService); | ||||
|             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); | ||||
|             }); | ||||
|             this._panelService.connect('focus-in', (panel, path) => { | ||||
|                 if (!GLib.str_has_suffix(path, '/InputContext_1')) | ||||
|                     this.emit('focus-in'); | ||||
|             }); | ||||
|             this._panelService.connect('focus-out', () => this.emit('focus-out')); | ||||
|  | ||||
|             try { | ||||
|                 // IBus versions older than 1.5.10 have a bug which | ||||
|                 // causes spurious set-content-type emissions when | ||||
|                 // switching input focus that temporarily lose purpose | ||||
|                 // and hints defeating its intended semantics and | ||||
|                 // confusing users. We thus don't use it in that case. | ||||
|                 _checkIBusVersion(1, 5, 10); | ||||
|                 this._panelService.connect('set-content-type', this._setContentType.bind(this)); | ||||
|             } catch (e) { | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._panelService = new IBus.PanelService({ | ||||
|             connection: this._ibus.get_connection(), | ||||
|             object_path: IBus.PATH_PANEL, | ||||
|         }); | ||||
|         this._candidatePopup.setPanelService(this._panelService); | ||||
|         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); | ||||
|         }); | ||||
|         this._panelService.connect('focus-in', (panel, path) => { | ||||
|             if (!GLib.str_has_suffix(path, '/InputContext_1')) | ||||
|                 this.emit('focus-in'); | ||||
|         }); | ||||
|         this._panelService.connect('focus-out', () => this.emit('focus-out')); | ||||
|  | ||||
|         try { | ||||
|             // IBus versions older than 1.5.10 have a bug which | ||||
|             // causes spurious set-content-type emissions when | ||||
|             // switching input focus that temporarily lose purpose | ||||
|             // and hints defeating its intended semantics and | ||||
|             // confusing users. We thus don't use it in that case. | ||||
|             _checkIBusVersion(1, 5, 10); | ||||
|             this._panelService.connect('set-content-type', this._setContentType.bind(this)); | ||||
|         } catch (e) { | ||||
|         } | ||||
|         this._updateReadiness(); | ||||
|  | ||||
|         try { | ||||
|             // If an engine is already active we need to get its properties | ||||
|             const engine = | ||||
|                 await this._ibus.get_global_engine_async(-1, this._cancellable); | ||||
|             this._engineChanged(this._ibus, engine.get_name()); | ||||
|         } catch (e) { | ||||
|             this._ibus.get_global_engine_async(-1, this._cancellable, (_bus, res) => { | ||||
|                 let engine; | ||||
|                 try { | ||||
|                     engine = this._ibus.get_global_engine_async_finish(res); | ||||
|                     if (!engine) | ||||
|                         return; | ||||
|                 } catch (e) { | ||||
|                     return; | ||||
|                 } | ||||
|                 this._engineChanged(this._ibus, engine.get_name()); | ||||
|             }); | ||||
|             this._updateReadiness(); | ||||
|         } else { | ||||
|             this._clear(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -227,7 +219,7 @@ var IBusManager = class { | ||||
|         return this._engines.get(id); | ||||
|     } | ||||
|  | ||||
|     async setEngine(id, callback) { | ||||
|     setEngine(id, callback) { | ||||
|         // Send id even if id == this._currentEngineName | ||||
|         // because 'properties-registered' signal can be emitted | ||||
|         // while this._ibusSources == null on a lock screen. | ||||
| @@ -237,16 +229,18 @@ var IBusManager = class { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             await this._ibus.set_global_engine_async(id, | ||||
|                 this._MAX_INPUT_SOURCE_ACTIVATION_TIME, | ||||
|                 this._cancellable); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                 logError(e); | ||||
|         } | ||||
|         if (callback) | ||||
|             callback(); | ||||
|         this._ibus.set_global_engine_async(id, | ||||
|             this._MAX_INPUT_SOURCE_ACTIVATION_TIME, | ||||
|             this._cancellable, (_bus, res) => { | ||||
|                 try { | ||||
|                     this._ibus.set_global_engine_async_finish(res); | ||||
|                 } catch (e) { | ||||
|                     if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) | ||||
|                         logError(e); | ||||
|                 } | ||||
|                 if (callback) | ||||
|                     callback(); | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     preloadEngines(ids) { | ||||
|   | ||||
| @@ -4,9 +4,6 @@ const { Clutter, GLib, Gio, GObject, IBus } = imports.gi; | ||||
|  | ||||
| const Keyboard = imports.ui.status.keyboard; | ||||
|  | ||||
| Gio._promisify(IBus.Bus.prototype, | ||||
|     'create_input_context_async', 'create_input_context_async_finish'); | ||||
|  | ||||
| var HIDE_PANEL_TIME = 50; | ||||
|  | ||||
| var InputMethod = GObject.registerClass( | ||||
| @@ -49,11 +46,15 @@ class InputMethod extends Clutter.InputMethod { | ||||
|         this._currentSource = this._inputSourceManager.currentSource; | ||||
|     } | ||||
|  | ||||
|     async _onConnected() { | ||||
|     _onConnected() { | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|         this._ibus.create_input_context_async('gnome-shell', -1, | ||||
|             this._cancellable, this._setContext.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _setContext(bus, res) { | ||||
|         try { | ||||
|             this._context = await this._ibus.create_input_context_async( | ||||
|                 'gnome-shell', -1, this._cancellable); | ||||
|             this._context = this._ibus.create_input_context_async_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|                 logError(e); | ||||
| @@ -96,13 +97,8 @@ class InputMethod extends Clutter.InputMethod { | ||||
|         this.commit(text.get_text()); | ||||
|     } | ||||
|  | ||||
|     _onDeleteSurroundingText(_context, offset, nchars) { | ||||
|         try { | ||||
|             this.delete_surrounding(offset, nchars); | ||||
|         } catch (e) { | ||||
|             // We may get out of bounds for negative offset on older mutter | ||||
|             this.delete_surrounding(0, nchars + offset); | ||||
|         } | ||||
|     _onDeleteSurroundingText() { | ||||
|         this.delete_surrounding(); | ||||
|     } | ||||
|  | ||||
|     _onUpdatePreeditText(_context, text, pos, visible) { | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| /* exported IntrospectService */ | ||||
| const { Gio, GLib, Meta, Shell, St } = imports.gi; | ||||
| const { Gio, GLib, Meta, Shell } = imports.gi; | ||||
|  | ||||
| const INTROSPECT_SCHEMA = 'org.gnome.shell'; | ||||
| const INTROSPECT_KEY = 'introspect'; | ||||
| const APP_WHITELIST = ['org.freedesktop.impl.portal.desktop.gtk']; | ||||
|  | ||||
| const INTROSPECT_DBUS_API_VERSION = 2; | ||||
|  | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
|  | ||||
| const IntrospectDBusIface = loadInterfaceXML('org.gnome.Shell.Introspect'); | ||||
| @@ -24,7 +22,6 @@ var IntrospectService = class { | ||||
|         this._runningApplicationsDirty = true; | ||||
|         this._activeApplication = null; | ||||
|         this._activeApplicationDirty = true; | ||||
|         this._animationsEnabled = true; | ||||
|  | ||||
|         this._appSystem = Shell.AppSystem.get_default(); | ||||
|         this._appSystem.connect('app-state-changed', | ||||
| @@ -33,9 +30,7 @@ var IntrospectService = class { | ||||
|                                     this._syncRunningApplications(); | ||||
|                                 }); | ||||
|  | ||||
|         this._introspectSettings = new Gio.Settings({ | ||||
|             schema_id: INTROSPECT_SCHEMA, | ||||
|         }); | ||||
|         this._settings = new Gio.Settings({ schema_id: INTROSPECT_SCHEMA }); | ||||
|  | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         tracker.connect('notify::focus-app', | ||||
| @@ -54,11 +49,6 @@ var IntrospectService = class { | ||||
|                 (conn, name, owner) => this._whitelistMap.set(name, owner), | ||||
|                 (conn, name) => this._whitelistMap.delete(name)); | ||||
|         }); | ||||
|  | ||||
|         this._settings = St.Settings.get(); | ||||
|         this._settings.connect('notify::enable-animations', | ||||
|             this._syncAnimationsEnabled.bind(this)); | ||||
|         this._syncAnimationsEnabled(); | ||||
|     } | ||||
|  | ||||
|     _isStandaloneApp(app) { | ||||
| @@ -66,7 +56,7 @@ var IntrospectService = class { | ||||
|     } | ||||
|  | ||||
|     _isIntrospectEnabled() { | ||||
|         return this._introspectSettings.get_boolean(INTROSPECT_KEY); | ||||
|         return this._settings.get_boolean(INTROSPECT_KEY); | ||||
|     } | ||||
|  | ||||
|     _isSenderWhitelisted(sender) { | ||||
| @@ -129,18 +119,9 @@ var IntrospectService = class { | ||||
|                 type == Meta.WindowType.UTILITY; | ||||
|     } | ||||
|  | ||||
|     _isInvocationAllowed(invocation) { | ||||
|         if (this._isIntrospectEnabled()) | ||||
|             return true; | ||||
|  | ||||
|         if (this._isSenderWhitelisted(invocation.get_sender())) | ||||
|             return true; | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     GetRunningApplicationsAsync(params, invocation) { | ||||
|         if (!this._isInvocationAllowed(invocation)) { | ||||
|         if (!this._isIntrospectEnabled() && | ||||
|             !this._isSenderWhitelisted(invocation.get_sender())) { | ||||
|             invocation.return_error_literal(Gio.DBusError, | ||||
|                                             Gio.DBusError.ACCESS_DENIED, | ||||
|                                             'App introspection not allowed'); | ||||
| @@ -155,7 +136,8 @@ var IntrospectService = class { | ||||
|         let apps = this._appSystem.get_running(); | ||||
|         let windowsList = {}; | ||||
|  | ||||
|         if (!this._isInvocationAllowed(invocation)) { | ||||
|         if (!this._isIntrospectEnabled() && | ||||
|             !this._isSenderWhitelisted(invocation.get_sender())) { | ||||
|             invocation.return_error_literal(Gio.DBusError, | ||||
|                                             Gio.DBusError.ACCESS_DENIED, | ||||
|                                             'App introspection not allowed'); | ||||
| @@ -199,21 +181,4 @@ var IntrospectService = class { | ||||
|         } | ||||
|         invocation.return_value(new GLib.Variant('(a{ta{sv}})', [windowsList])); | ||||
|     } | ||||
|  | ||||
|     _syncAnimationsEnabled() { | ||||
|         let wasAnimationsEnabled = this._animationsEnabled; | ||||
|         this._animationsEnabled = this._settings.enable_animations; | ||||
|         if (wasAnimationsEnabled !== this._animationsEnabled) { | ||||
|             let variant = new GLib.Variant('b', this._animationsEnabled); | ||||
|             this._dbusImpl.emit_property_changed('AnimationsEnabled', variant); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     get AnimationsEnabled() { | ||||
|         return this._animationsEnabled; | ||||
|     } | ||||
|  | ||||
|     get version() { | ||||
|         return INTROSPECT_DBUS_API_VERSION; | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -24,7 +24,8 @@ function getCompletions(text, commandHeader, globalCompletionList) { | ||||
|             [expr_, base, attrHead] = matches; | ||||
|  | ||||
|             methods = getPropertyNamesFromExpression(base, commandHeader).filter( | ||||
|                 attr => attr.slice(0, attrHead.length) === attrHead); | ||||
|                 attr => attr.slice(0, attrHead.length) == attrHead | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         // Look for the empty expression or partially entered words | ||||
| @@ -33,7 +34,8 @@ function getCompletions(text, commandHeader, globalCompletionList) { | ||||
|         if (text == '' || matches) { | ||||
|             [expr_, attrHead] = matches; | ||||
|             methods = globalCompletionList.filter( | ||||
|                 attr => attr.slice(0, attrHead.length) === attrHead); | ||||
|                 attr => attr.slice(0, attrHead.length) == attrHead | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported getKeyboardManager, holdKeyboard, releaseKeyboard */ | ||||
|  | ||||
| const { GLib, GnomeDesktop } = imports.gi; | ||||
| const { GLib, GnomeDesktop, Meta } = imports.gi; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| @@ -62,11 +62,11 @@ var KeyboardManager = class { | ||||
|             return; | ||||
|  | ||||
|         this._currentKeymap = { layouts, variants, options }; | ||||
|         global.backend.set_keymap(layouts, variants, options); | ||||
|         Meta.get_backend().set_keymap(layouts, variants, options); | ||||
|     } | ||||
|  | ||||
|     _applyLayoutGroupIndex(idx) { | ||||
|         global.backend.lock_layout_group(idx); | ||||
|         Meta.get_backend().lock_layout_group(idx); | ||||
|     } | ||||
|  | ||||
|     apply(id) { | ||||
|   | ||||
| @@ -50,22 +50,25 @@ function canLock() { | ||||
| } | ||||
|  | ||||
|  | ||||
| async function registerSessionWithGDM() { | ||||
| function registerSessionWithGDM() { | ||||
|     log("Registering session with GDM"); | ||||
|     try { | ||||
|         await Gio.DBus.system.call( | ||||
|             'org.gnome.DisplayManager', | ||||
|             '/org/gnome/DisplayManager/Manager', | ||||
|             'org.gnome.DisplayManager.Manager', | ||||
|             'RegisterSession', | ||||
|             GLib.Variant.new('(a{sv})', [{}]), null, | ||||
|             Gio.DBusCallFlags.NONE, -1, null); | ||||
|     } catch (e) { | ||||
|         if (!e.matches(Gio.DBusError, Gio.DBusError.UNKNOWN_METHOD)) | ||||
|             log(`Error registering session with GDM: ${e.message}`); | ||||
|         else | ||||
|             log('Not calling RegisterSession(): method not exported, GDM too old?'); | ||||
|     } | ||||
|     Gio.DBus.system.call('org.gnome.DisplayManager', | ||||
|                          '/org/gnome/DisplayManager/Manager', | ||||
|                          'org.gnome.DisplayManager.Manager', | ||||
|                          'RegisterSession', | ||||
|                          GLib.Variant.new('(a{sv})', [{}]), null, | ||||
|                          Gio.DBusCallFlags.NONE, -1, null, | ||||
|         (source, result) => { | ||||
|             try { | ||||
|                 source.call_finish(result); | ||||
|             } catch (e) { | ||||
|                 if (!e.matches(Gio.DBusError, Gio.DBusError.UNKNOWN_METHOD)) | ||||
|                     log(`Error registering session with GDM: ${e.message}`); | ||||
|                 else | ||||
|                     log("Not calling RegisterSession(): method not exported, GDM too old?"); | ||||
|             } | ||||
|         } | ||||
|     ); | ||||
| } | ||||
|  | ||||
| let _loginManager = null; | ||||
| @@ -171,19 +174,24 @@ var LoginManagerSystemd = class { | ||||
|         this._proxy.SuspendRemote(true); | ||||
|     } | ||||
|  | ||||
|     async inhibit(reason, callback) { | ||||
|         try { | ||||
|             const inVariant = new GLib.Variant('(ssss)', | ||||
|                 ['sleep', 'GNOME Shell', reason, 'delay']); | ||||
|             const [outVariant_, fdList] = | ||||
|                 await this._proxy.call_with_unix_fd_list('Inhibit', | ||||
|                     inVariant, 0, -1, null, null); | ||||
|             const [fd] = fdList.steal_fds(); | ||||
|             callback(new Gio.UnixInputStream({ fd })); | ||||
|         } catch (e) { | ||||
|             logError(e, 'Error getting systemd inhibitor'); | ||||
|             callback(null); | ||||
|         } | ||||
|     inhibit(reason, callback) { | ||||
|         let inVariant = GLib.Variant.new('(ssss)', | ||||
|                                          ['sleep', | ||||
|                                           'GNOME Shell', | ||||
|                                           reason, | ||||
|                                           'delay']); | ||||
|         this._proxy.call_with_unix_fd_list('Inhibit', inVariant, 0, -1, null, null, | ||||
|             (proxy, result) => { | ||||
|                 let fd = -1; | ||||
|                 try { | ||||
|                     let [outVariant_, fdList] = proxy.call_with_unix_fd_list_finish(result); | ||||
|                     fd = fdList.steal_fds()[0]; | ||||
|                     callback(new Gio.UnixInputStream({ fd })); | ||||
|                 } catch (e) { | ||||
|                     logError(e, "Error getting systemd inhibitor"); | ||||
|                     callback(null); | ||||
|                 } | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     _prepareForSleep(proxy, sender, [aboutToSuspend]) { | ||||
|   | ||||
| @@ -7,6 +7,7 @@ jsconf.set10('HAVE_BLUETOOTH', bt_dep.found()) | ||||
| jsconf.set10('HAVE_NETWORKMANAGER', have_networkmanager) | ||||
| jsconf.set('datadir', datadir) | ||||
| jsconf.set('libexecdir', libexecdir) | ||||
| jsconf.set('vpndir', vpndir) | ||||
|  | ||||
| config_js = configure_file( | ||||
|   input: 'config.js.in', | ||||
|   | ||||
| @@ -223,7 +223,7 @@ var BroadbandModem = GObject.registerClass({ | ||||
| }, class BroadbandModem extends ModemBase { | ||||
|     _init(path, capabilities) { | ||||
|         super._init({ capabilities }); | ||||
|         this._proxy = new BroadbandModemProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|         this._proxy = new BroadbandModemProxy(Gio.DBus.system, 'org.freedesktop.ModemManager', path); | ||||
|         this._proxy_3gpp = new BroadbandModem3gppProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|         this._proxy_cdma = new BroadbandModemCdmaProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|  | ||||
| @@ -249,7 +249,7 @@ var BroadbandModem = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     _reloadSignalQuality() { | ||||
|         let [quality, recent_] = this._proxy.SignalQuality; | ||||
|         let [quality, recent_] = this.SignalQuality; | ||||
|         this._setSignalQuality(quality); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -57,7 +57,9 @@ var ObjectManager = class { | ||||
|         // Start out inhibiting load until at least the proxy | ||||
|         // manager is loaded and the remote objects are fetched | ||||
|         this._numLoadInhibitors = 1; | ||||
|         this._initManagerProxy(); | ||||
|         this._managerProxy.init_async(GLib.PRIORITY_DEFAULT, | ||||
|                                       this._cancellable, | ||||
|                                       this._onManagerProxyLoaded.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _tryToCompleteLoad() { | ||||
| @@ -71,7 +73,7 @@ var ObjectManager = class { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async _addInterface(objectPath, interfaceName, onFinished) { | ||||
|     _addInterface(objectPath, interfaceName, onFinished) { | ||||
|         let info = this._interfaceInfos[interfaceName]; | ||||
|  | ||||
|         if (!info) { | ||||
| @@ -87,38 +89,40 @@ var ObjectManager = class { | ||||
|                                         g_interface_info: info, | ||||
|                                         g_flags: Gio.DBusProxyFlags.DO_NOT_AUTO_START }); | ||||
|  | ||||
|         try { | ||||
|             await proxy.init_async(GLib.PRIORITY_DEFAULT, this._cancellable); | ||||
|         } catch (e) { | ||||
|             logError(e, `could not initialize proxy for interface ${interfaceName}`); | ||||
|         proxy.init_async(GLib.PRIORITY_DEFAULT, this._cancellable, (initable, result) => { | ||||
|             try { | ||||
|                 initable.init_finish(result); | ||||
|             } catch (e) { | ||||
|                 logError(e, `could not initialize proxy for interface ${interfaceName}`); | ||||
|  | ||||
|                 if (onFinished) | ||||
|                     onFinished(); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             let isNewObject; | ||||
|             if (!this._objects[objectPath]) { | ||||
|                 this._objects[objectPath] = {}; | ||||
|                 isNewObject = true; | ||||
|             } else { | ||||
|                 isNewObject = false; | ||||
|             } | ||||
|  | ||||
|             this._objects[objectPath][interfaceName] = proxy; | ||||
|  | ||||
|             if (!this._interfaces[interfaceName]) | ||||
|                 this._interfaces[interfaceName] = []; | ||||
|  | ||||
|             this._interfaces[interfaceName].push(proxy); | ||||
|  | ||||
|             if (isNewObject) | ||||
|                 this.emit('object-added', objectPath); | ||||
|  | ||||
|             this.emit('interface-added', interfaceName, proxy); | ||||
|  | ||||
|             if (onFinished) | ||||
|                 onFinished(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let isNewObject; | ||||
|         if (!this._objects[objectPath]) { | ||||
|             this._objects[objectPath] = {}; | ||||
|             isNewObject = true; | ||||
|         } else { | ||||
|             isNewObject = false; | ||||
|         } | ||||
|  | ||||
|         this._objects[objectPath][interfaceName] = proxy; | ||||
|  | ||||
|         if (!this._interfaces[interfaceName]) | ||||
|             this._interfaces[interfaceName] = []; | ||||
|  | ||||
|         this._interfaces[interfaceName].push(proxy); | ||||
|  | ||||
|         if (isNewObject) | ||||
|             this.emit('object-added', objectPath); | ||||
|  | ||||
|         this.emit('interface-added', interfaceName, proxy); | ||||
|  | ||||
|         if (onFinished) | ||||
|             onFinished(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _removeInterface(objectPath, interfaceName) { | ||||
| @@ -147,10 +151,9 @@ var ObjectManager = class { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async _initManagerProxy() { | ||||
|     _onManagerProxyLoaded(initable, result) { | ||||
|         try { | ||||
|             await this._managerProxy.init_async( | ||||
|                 GLib.PRIORITY_DEFAULT, this._cancellable); | ||||
|             initable.init_finish(result); | ||||
|         } catch (e) { | ||||
|             logError(e, `could not initialize object manager for object ${this._serviceName}`); | ||||
|  | ||||
|   | ||||
| @@ -1,146 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| // | ||||
| // Copyright (C) 2018, 2019, 2020 Endless Mobile, Inc. | ||||
| // | ||||
| // This is a GNOME Shell component to wrap the interactions over | ||||
| // D-Bus with the malcontent library. | ||||
| // | ||||
| // Licensed under the GNU General Public License Version 2 | ||||
| // | ||||
| // This program is free software; you can redistribute it and/or | ||||
| // modify it under the terms of the GNU General Public License | ||||
| // as published by the Free Software Foundation; either version 2 | ||||
| // of the License, or (at your option) any later version. | ||||
| // | ||||
| // This program is distributed in the hope that it will be useful, | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| // GNU General Public License for more details. | ||||
| // | ||||
| // You should have received a copy of the GNU General Public License | ||||
| // along with this program; if not, write to the Free Software | ||||
| // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA. | ||||
|  | ||||
| /* exported getDefault */ | ||||
|  | ||||
| const { Gio, GObject, Shell } = imports.gi; | ||||
|  | ||||
| // We require libmalcontent ≥ 0.6.0 | ||||
| const HAVE_MALCONTENT = imports.package.checkSymbol( | ||||
|     'Malcontent', '0', 'ManagerGetValueFlags'); | ||||
|  | ||||
| var Malcontent = null; | ||||
| if (HAVE_MALCONTENT) { | ||||
|     Malcontent = imports.gi.Malcontent; | ||||
|     Gio._promisify(Malcontent.Manager.prototype, 'get_app_filter_async', 'get_app_filter_finish'); | ||||
| } | ||||
|  | ||||
| let _singleton = null; | ||||
|  | ||||
| function getDefault() { | ||||
|     if (_singleton === null) | ||||
|         _singleton = new ParentalControlsManager(); | ||||
|  | ||||
|     return _singleton; | ||||
| } | ||||
|  | ||||
| // A manager class which provides cached access to the constructing user’s | ||||
| // parental controls settings. It’s possible for the user’s parental controls | ||||
| // to change at runtime if the Parental Controls application is used by an | ||||
| // administrator from within the user’s session. | ||||
| var ParentalControlsManager = GObject.registerClass({ | ||||
|     Signals: { | ||||
|         'app-filter-changed': {}, | ||||
|     }, | ||||
| }, class ParentalControlsManager extends GObject.Object { | ||||
|     _init() { | ||||
|         super._init(); | ||||
|  | ||||
|         this._initialized = false; | ||||
|         this._disabled = false; | ||||
|         this._appFilter = null; | ||||
|  | ||||
|         this._initializeManager(); | ||||
|     } | ||||
|  | ||||
|     async _initializeManager() { | ||||
|         if (!HAVE_MALCONTENT) { | ||||
|             log('Skipping parental controls support as it’s disabled'); | ||||
|             this._initialized = true; | ||||
|             this.emit('app-filter-changed'); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         log(`Getting parental controls for user ${Shell.util_get_uid()}`); | ||||
|         try { | ||||
|             const connection = await Gio.DBus.get(Gio.BusType.SYSTEM, null); | ||||
|             this._manager = new Malcontent.Manager({ connection }); | ||||
|             this._appFilter = await this._manager.get_app_filter_async( | ||||
|                 Shell.util_get_uid(), | ||||
|                 Malcontent.ManagerGetValueFlags.NONE, | ||||
|                 null); | ||||
|         } catch (e) { | ||||
|             if (e.matches(Malcontent.ManagerError, Malcontent.ManagerError.DISABLED)) { | ||||
|                 log('Parental controls globally disabled'); | ||||
|                 this._disabled = true; | ||||
|             } else { | ||||
|                 logError(e, 'Failed to get parental controls settings'); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this._manager.connect('app-filter-changed', this._onAppFilterChanged.bind(this)); | ||||
|  | ||||
|         // Signal initialisation is complete. | ||||
|         this._initialized = true; | ||||
|         this.emit('app-filter-changed'); | ||||
|     } | ||||
|  | ||||
|     async _onAppFilterChanged(manager, uid) { | ||||
|         // Emit 'changed' signal only if app-filter is changed for currently logged-in user. | ||||
|         let currentUid = Shell.util_get_uid(); | ||||
|         if (currentUid !== uid) | ||||
|             return; | ||||
|  | ||||
|         try { | ||||
|             this._appFilter = await this._manager.get_app_filter_async( | ||||
|                 currentUid, | ||||
|                 Malcontent.ManagerGetValueFlags.NONE, | ||||
|                 null); | ||||
|             this.emit('app-filter-changed'); | ||||
|         } catch (e) { | ||||
|             // Log an error and keep the old app filter. | ||||
|             logError(e, `Failed to get new MctAppFilter for uid ${Shell.util_get_uid()} on app-filter-changed`); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     get initialized() { | ||||
|         return this._initialized; | ||||
|     } | ||||
|  | ||||
|     // Calculate whether the given app (a Gio.DesktopAppInfo) should be shown | ||||
|     // on the desktop, in search results, etc. The app should be shown if: | ||||
|     //  - The .desktop file doesn’t say it should be hidden. | ||||
|     //  - The executable from the .desktop file’s Exec line isn’t blacklisted in | ||||
|     //    the user’s parental controls. | ||||
|     //  - None of the flatpak app IDs from the X-Flatpak and the | ||||
|     //    X-Flatpak-RenamedFrom lines are blacklisted in the user’s parental | ||||
|     //    controls. | ||||
|     shouldShowApp(appInfo) { | ||||
|         // Quick decision? | ||||
|         if (!appInfo.should_show()) | ||||
|             return false; | ||||
|  | ||||
|         // Are parental controls enabled (at configure time or runtime)? | ||||
|         if (!HAVE_MALCONTENT || this._disabled) | ||||
|             return true; | ||||
|  | ||||
|         // Have we finished initialising yet? | ||||
|         if (!this.initialized) { | ||||
|             log(`Warning: Hiding app because parental controls not yet initialised: ${appInfo.get_id()}`); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         return this._appFilter.is_appinfo_allowed(appInfo); | ||||
|     } | ||||
| }); | ||||
| @@ -83,17 +83,13 @@ const SystemActions = GObject.registerClass({ | ||||
|         this._canHavePowerOff = true; | ||||
|         this._canHaveSuspend = true; | ||||
|  | ||||
|         function tokenizeKeywords(keywords) { | ||||
|             return keywords.split(';').map(keyword => GLib.str_tokenize_and_fold(keyword, null)).flat(2); | ||||
|         } | ||||
|  | ||||
|         this._actions = new Map(); | ||||
|         this._actions.set(POWER_OFF_ACTION_ID, { | ||||
|             // Translators: The name of the power-off action in search | ||||
|             name: C_("search-result", "Power Off"), | ||||
|             iconName: 'system-shutdown-symbolic', | ||||
|             // Translators: A list of keywords that match the power-off action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('power off;shutdown;reboot;restart;halt;stop')), | ||||
|             keywords: _('power off;shutdown;reboot;restart;halt;stop').split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|         this._actions.set(LOCK_SCREEN_ACTION_ID, { | ||||
| @@ -101,15 +97,15 @@ const SystemActions = GObject.registerClass({ | ||||
|             name: C_("search-result", "Lock Screen"), | ||||
|             iconName: 'system-lock-screen-symbolic', | ||||
|             // Translators: A list of keywords that match the lock screen action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('lock screen')), | ||||
|             keywords: _("lock screen").split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|         this._actions.set(LOGOUT_ACTION_ID, { | ||||
|             // Translators: The name of the logout action in search | ||||
|             name: C_("search-result", "Log Out"), | ||||
|             iconName: 'system-log-out-symbolic', | ||||
|             iconName: 'application-exit-symbolic', | ||||
|             // Translators: A list of keywords that match the logout action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('logout;log out;sign off')), | ||||
|             keywords: _("logout;log out;sign off").split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|         this._actions.set(SUSPEND_ACTION_ID, { | ||||
| @@ -117,7 +113,7 @@ const SystemActions = GObject.registerClass({ | ||||
|             name: C_("search-result", "Suspend"), | ||||
|             iconName: 'media-playback-pause-symbolic', | ||||
|             // Translators: A list of keywords that match the suspend action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('suspend;sleep')), | ||||
|             keywords: _("suspend;sleep").split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|         this._actions.set(SWITCH_USER_ACTION_ID, { | ||||
| @@ -125,14 +121,14 @@ const SystemActions = GObject.registerClass({ | ||||
|             name: C_("search-result", "Switch User"), | ||||
|             iconName: 'system-switch-user-symbolic', | ||||
|             // Translators: A list of keywords that match the switch user action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('switch user')), | ||||
|             keywords: _("switch user").split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|         this._actions.set(LOCK_ORIENTATION_ACTION_ID, { | ||||
|             name: '', | ||||
|             iconName: '', | ||||
|             // Translators: A list of keywords that match the lock orientation action, separated by semicolons | ||||
|             keywords: tokenizeKeywords(_('lock orientation;unlock orientation;screen;rotation')), | ||||
|             keywords: _("lock orientation;unlock orientation;screen;rotation").split(/[; ]/), | ||||
|             available: false, | ||||
|         }); | ||||
|  | ||||
| @@ -155,17 +151,17 @@ const SystemActions = GObject.registerClass({ | ||||
|         this._userManager.connect('user-removed', | ||||
|                                   () => this._updateMultiUser()); | ||||
|  | ||||
|         this._lockdownSettings.connect('changed::%s'.format(DISABLE_USER_SWITCH_KEY), | ||||
|         this._lockdownSettings.connect(`changed::${DISABLE_USER_SWITCH_KEY}`, | ||||
|                                        () => this._updateSwitchUser()); | ||||
|         this._lockdownSettings.connect('changed::%s'.format(DISABLE_LOG_OUT_KEY), | ||||
|         this._lockdownSettings.connect(`changed::${DISABLE_LOG_OUT_KEY}`, | ||||
|                                        () => this._updateLogout()); | ||||
|         global.settings.connect('changed::%s'.format(ALWAYS_SHOW_LOG_OUT_KEY), | ||||
|         global.settings.connect(`changed::${ALWAYS_SHOW_LOG_OUT_KEY}`, | ||||
|                                 () => this._updateLogout()); | ||||
|  | ||||
|         this._lockdownSettings.connect('changed::%s'.format(DISABLE_LOCK_SCREEN_KEY), | ||||
|         this._lockdownSettings.connect(`changed::${DISABLE_LOCK_SCREEN_KEY}`, | ||||
|                                        () => this._updateLockScreen()); | ||||
|  | ||||
|         this._lockdownSettings.connect('changed::%s'.format(DISABLE_LOG_OUT_KEY), | ||||
|         this._lockdownSettings.connect(`changed::${DISABLE_LOG_OUT_KEY}`, | ||||
|                                        () => this._updateHaveShutdown()); | ||||
|  | ||||
|         this.forceUpdate(); | ||||
| @@ -281,7 +277,7 @@ const SystemActions = GObject.registerClass({ | ||||
|  | ||||
|     getMatchingActions(terms) { | ||||
|         // terms is a list of strings | ||||
|         terms = terms.map(term => GLib.str_tokenize_and_fold(term, null)[0]); | ||||
|         terms = terms.map(term => term.toLowerCase()); | ||||
|  | ||||
|         let results = []; | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported findUrls, spawn, spawnCommandLine, spawnApp, trySpawnCommandLine, | ||||
|             formatTime, formatTimeSpan, createTimeLabel, insertSorted, | ||||
|             ensureActorVisibleInScrollView, wiggle */ | ||||
|             makeCloseButton, ensureActorVisibleInScrollView, wiggle */ | ||||
|  | ||||
| const { Clutter, Gio, GLib, Shell, St, GnomeDesktop } = imports.gi; | ||||
| const { Clutter, Gio, GLib, GObject, Shell, St, GnomeDesktop } = imports.gi; | ||||
| const Gettext = imports.gettext; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| @@ -11,17 +11,13 @@ const Params = imports.misc.params; | ||||
|  | ||||
| var SCROLL_TIME = 100; | ||||
|  | ||||
| const WIGGLE_OFFSET = 6; | ||||
| const WIGGLE_DURATION = 65; | ||||
| const N_WIGGLES = 3; | ||||
|  | ||||
| // http://daringfireball.net/2010/07/improved_regex_for_matching_urls | ||||
| const _balancedParens = '\\([^\\s()<>]+\\)'; | ||||
| const _leadingJunk = '[\\s`(\\[{\'\\"<\u00AB\u201C\u2018]'; | ||||
| const _notTrailingJunk = '[^\\s`!()\\[\\]{};:\'\\".,<>?\u00AB\u00BB\u200E\u200F\u201C\u201D\u2018\u2019\u202A\u202C]'; | ||||
|  | ||||
| const _urlRegexp = new RegExp( | ||||
|     '(^|%s)'.format(_leadingJunk) + | ||||
|     `(^|${_leadingJunk})` + | ||||
|     '(' + | ||||
|         '(?:' + | ||||
|             '(?:http|https|ftp)://' +             // scheme:// | ||||
| @@ -33,12 +29,12 @@ const _urlRegexp = new RegExp( | ||||
|         '(?:' +                                   // one or more: | ||||
|             '[^\\s()<>]+' +                       // run of non-space non-() | ||||
|             '|' +                                 // or | ||||
|             '%s'.format(_balancedParens) +        // balanced parens | ||||
|             `${_balancedParens}` +                // balanced parens | ||||
|         ')+' + | ||||
|         '(?:' +                                   // end with: | ||||
|             '%s'.format(_balancedParens) +        // balanced parens | ||||
|             `${_balancedParens}` +                // balanced parens | ||||
|             '|' +                                 // or | ||||
|             '%s'.format(_notTrailingJunk) +       // last non-junk char | ||||
|             `${_notTrailingJunk}` +               // last non-junk char | ||||
|         ')' + | ||||
|     ')', 'gi'); | ||||
|  | ||||
| @@ -153,7 +149,7 @@ function trySpawnCommandLine(commandLine) { | ||||
|     } catch (err) { | ||||
|         // Replace "Error invoking GLib.shell_parse_argv: " with | ||||
|         // something nicer | ||||
|         err.message = err.message.replace(/[^:]*: /, '%s\n'.format(_('Could not parse command:'))); | ||||
|         err.message = err.message.replace(/[^:]*: /, `${_("Could not parse command:")}\n`); | ||||
|         throw err; | ||||
|     } | ||||
|  | ||||
| @@ -363,6 +359,51 @@ function insertSorted(array, val, cmp) { | ||||
|     return pos; | ||||
| } | ||||
|  | ||||
| var CloseButton = GObject.registerClass( | ||||
| class CloseButton extends St.Button { | ||||
|     _init(boxpointer) { | ||||
|         super._init({ | ||||
|             style_class: 'notification-close', | ||||
|             x_expand: true, | ||||
|             y_expand: true, | ||||
|             x_align: Clutter.ActorAlign.END, | ||||
|             y_align: Clutter.ActorAlign.START, | ||||
|         }); | ||||
|  | ||||
|         this._boxPointer = boxpointer; | ||||
|         if (boxpointer) | ||||
|             this._boxPointer.connect('arrow-side-changed', this._sync.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _computeBoxPointerOffset() { | ||||
|         if (!this._boxPointer || !this._boxPointer.get_stage()) | ||||
|             return 0; | ||||
|  | ||||
|         let side = this._boxPointer.arrowSide; | ||||
|         if (side == St.Side.TOP) | ||||
|             return this._boxPointer.getArrowHeight(); | ||||
|         else | ||||
|             return 0; | ||||
|     } | ||||
|  | ||||
|     _sync() { | ||||
|         let themeNode = this.get_theme_node(); | ||||
|  | ||||
|         let offY = this._computeBoxPointerOffset(); | ||||
|         this.translation_x = themeNode.get_length('-shell-close-overlap-x'); | ||||
|         this.translation_y = themeNode.get_length('-shell-close-overlap-y') + offY; | ||||
|     } | ||||
|  | ||||
|     vfunc_style_changed() { | ||||
|         this._sync(); | ||||
|         super.vfunc_style_changed(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function makeCloseButton(boxpointer) { | ||||
|     return new CloseButton(boxpointer); | ||||
| } | ||||
|  | ||||
| function ensureActorVisibleInScrollView(scrollView, actor) { | ||||
|     let adjustment = scrollView.vscroll.adjustment; | ||||
|     let [value, lower_, upper, stepIncrement_, pageIncrement_, pageSize] = adjustment.get_values(); | ||||
| @@ -400,13 +441,10 @@ function ensureActorVisibleInScrollView(scrollView, actor) { | ||||
| } | ||||
|  | ||||
| function wiggle(actor, params) { | ||||
|     if (!St.Settings.get().enable_animations) | ||||
|         return; | ||||
|  | ||||
|     params = Params.parse(params, { | ||||
|         offset: WIGGLE_OFFSET, | ||||
|         duration: WIGGLE_DURATION, | ||||
|         wiggleCount: N_WIGGLES, | ||||
|         offset: 0, | ||||
|         duration: 0, | ||||
|         wiggleCount: 0, | ||||
|     }); | ||||
|     actor.translation_x = 0; | ||||
|  | ||||
|   | ||||
| @@ -7,8 +7,6 @@ const PermissionStore = imports.misc.permissionStore; | ||||
|  | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
|  | ||||
| Gio._promisify(Geoclue.Simple, 'new', 'new_finish'); | ||||
|  | ||||
| const WeatherIntegrationIface = loadInterfaceXML('org.gnome.Shell.WeatherIntegration'); | ||||
|  | ||||
| const WEATHER_BUS_NAME = 'org.gnome.Weather'; | ||||
| @@ -81,7 +79,16 @@ var WeatherClient = class { | ||||
|         this._weatherApp = null; | ||||
|         this._weatherProxy = null; | ||||
|  | ||||
|         this._createWeatherProxy(); | ||||
|         let nodeInfo = Gio.DBusNodeInfo.new_for_xml(WeatherIntegrationIface); | ||||
|         Gio.DBusProxy.new( | ||||
|             Gio.DBus.session, | ||||
|             Gio.DBusProxyFlags.DO_NOT_AUTO_START | Gio.DBusProxyFlags.GET_INVALIDATED_PROPERTIES, | ||||
|             nodeInfo.lookup_interface(WEATHER_INTEGRATION_IFACE), | ||||
|             WEATHER_BUS_NAME, | ||||
|             WEATHER_OBJECT_PATH, | ||||
|             WEATHER_INTEGRATION_IFACE, | ||||
|             null, | ||||
|             this._onWeatherProxyReady.bind(this)); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ | ||||
|             schema_id: 'org.gnome.shell.weather', | ||||
| @@ -139,17 +146,9 @@ var WeatherClient = class { | ||||
|                (!this._needsAuth || this._weatherAuthorized); | ||||
|     } | ||||
|  | ||||
|     async _createWeatherProxy() { | ||||
|         const nodeInfo = Gio.DBusNodeInfo.new_for_xml(WeatherIntegrationIface); | ||||
|     _onWeatherProxyReady(o, res) { | ||||
|         try { | ||||
|             this._weatherProxy = await Gio.DBusProxy.new( | ||||
|                 Gio.DBus.session, | ||||
|                 Gio.DBusProxyFlags.DO_NOT_AUTO_START | Gio.DBusProxyFlags.GET_INVALIDATED_PROPERTIES, | ||||
|                 nodeInfo.lookup_interface(WEATHER_INTEGRATION_IFACE), | ||||
|                 WEATHER_BUS_NAME, | ||||
|                 WEATHER_OBJECT_PATH, | ||||
|                 WEATHER_INTEGRATION_IFACE, | ||||
|                 null); | ||||
|             this._weatherProxy = Gio.DBusProxy.new_finish(res); | ||||
|         } catch (e) { | ||||
|             log(`Failed to create GNOME Weather proxy: ${e}`); | ||||
|             return; | ||||
| @@ -240,23 +239,25 @@ var WeatherClient = class { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async _startGClueService() { | ||||
|     _startGClueService() { | ||||
|         if (this._gclueStarting) | ||||
|             return; | ||||
|  | ||||
|         this._gclueStarting = true; | ||||
|  | ||||
|         try { | ||||
|             this._gclueService = await Geoclue.Simple.new( | ||||
|                 'org.gnome.Shell', Geoclue.AccuracyLevel.CITY, null); | ||||
|         } catch (e) { | ||||
|             log(`Failed to connect to Geoclue2 service: ${e.message}`); | ||||
|             this._setLocation(this._mostRecentLocation); | ||||
|             return; | ||||
|         } | ||||
|         this._gclueStarted = true; | ||||
|         this._gclueService.get_client().distance_threshold = 100; | ||||
|         this._updateLocationMonitoring(); | ||||
|         Geoclue.Simple.new('org.gnome.Shell', Geoclue.AccuracyLevel.CITY, null, | ||||
|             (o, res) => { | ||||
|                 try { | ||||
|                     this._gclueService = Geoclue.Simple.new_finish(res); | ||||
|                 } catch (e) { | ||||
|                     log(`Failed to connect to Geoclue2 service: ${e.message}`); | ||||
|                     this._setLocation(this._mostRecentLocation); | ||||
|                     return; | ||||
|                 } | ||||
|                 this._gclueStarted = true; | ||||
|                 this._gclueService.get_client().distance_threshold = 100; | ||||
|                 this._updateLocationMonitoring(); | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     _onGClueLocationChanged() { | ||||
|   | ||||