Compare commits
	
		
			186 Commits
		
	
	
		
			wip/carlos
			...
			wip/exalm/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | c3aa25e4e3 | ||
|   | cfd792fe86 | ||
|   | 7bdcc503cb | ||
|   | 856a4d27e0 | ||
|   | 9d1fe27221 | ||
|   | 407abeb178 | ||
|   | 95fc8021d2 | ||
|   | 060cbb65f6 | ||
|   | 1759e27ef7 | ||
|   | 4420f52080 | ||
|   | b4082063de | ||
|   | bde974087a | ||
|   | d93b51e135 | ||
|   | 730a68dffc | ||
|   | f06c257952 | ||
|   | 7afab2c28c | ||
|   | 16a18f2ae7 | ||
|   | 704e08dc08 | ||
|   | 60311aa4d1 | ||
|   | cceb74706a | ||
|   | 578ae29ed2 | ||
|   | 701d110493 | ||
|   | d1cbf6c7a9 | ||
|   | b64ce217e4 | ||
|   | 783dbe2aa9 | ||
|   | f4ce1cf462 | ||
|   | d04d6e069d | ||
|   | 40de201056 | ||
|   | 18234ea91a | ||
|   | e3f3297cba | ||
|   | e1ea1d2954 | ||
|   | 8e23ff8111 | ||
|   | 8a50a8e64c | ||
|   | fae207811a | ||
|   | 1d86424942 | ||
|   | 2bc8175219 | ||
|   | 1e31caf0b8 | ||
|   | 8e24ac6b26 | ||
|   | 427b9ac75f | ||
|   | f50205e9b4 | ||
|   | 168cfdd86b | ||
|   | d339c94c18 | ||
|   | 97509bf1d2 | ||
|   | 4a9c2ee805 | ||
|   | 51e1e6d15c | ||
|   | f0d2509dc3 | ||
|   | 95bb194356 | ||
|   | de8b43a45d | ||
|   | 82be010fd8 | ||
|   | 2a0c116757 | ||
|   | 6cdaec4001 | ||
|   | ed4baec40f | ||
|   | 9e8883c922 | ||
|   | 9bb64da895 | ||
|   | 049f348e25 | ||
|   | be190cc4d9 | ||
|   | 0893789b34 | ||
|   | 7b5c6b657a | ||
|   | 9363fd3524 | ||
|   | 0ad242a81e | ||
|   | 7031449f01 | ||
|   | 06df79286d | ||
|   | 9297d87775 | ||
|   | ef56b14553 | ||
|   | 30ff76272e | ||
|   | 52a7481ba6 | ||
|   | 14cfd74f08 | ||
|   | fa97f7141b | ||
|   | 03bcd4c05b | ||
|   | 89574abc83 | ||
|   | 602078cbde | ||
|   | 96f63b08c2 | ||
|   | d66cd0d206 | ||
|   | ff13ad9e71 | ||
|   | 5e4b65d37e | ||
|   | 51fd0875d1 | ||
|   | 75f8903c7e | ||
|   | 124eb1ca18 | ||
|   | b5d925817f | ||
|   | c134091268 | ||
|   | ff89693998 | ||
|   | c26860dcb7 | ||
|   | 6757c7d20a | ||
|   | 751189253a | ||
|   | 21187a4cec | ||
|   | b7db56ca9b | ||
|   | 261d36ba72 | ||
|   | 1e72874192 | ||
|   | f29dbd1f18 | ||
|   | b60836932a | ||
|   | bf47d1b22d | ||
|   | 4ea0fca4fc | ||
|   | 809f820cd4 | ||
|   | 75235624b2 | ||
|   | 5ea54426b9 | ||
|   | 4aabcd9e7d | ||
|   | 918b3eeb42 | ||
|   | 482c655590 | ||
|   | 86b5a43008 | ||
|   | d7cb2eeebc | ||
|   | 770231c2d7 | ||
|   | ecdf62d63e | ||
|   | d885486397 | ||
|   | 1524abc947 | ||
|   | cb9842e4a4 | ||
|   | 140ab4dec1 | ||
|   | 845122497b | ||
|   | d51a622fc0 | ||
|   | e90466347a | ||
|   | 73f8c1c482 | ||
|   | 0ff75941ea | ||
|   | e4bb2037ca | ||
|   | 481014ac9e | ||
|   | 6dbcb5f9da | ||
|   | c1f06daf88 | ||
|   | 0717f76362 | ||
|   | da738988cd | ||
|   | e86e88ea47 | ||
|   | ae338af1e8 | ||
|   | 3f9cc0ed37 | ||
|   | 8f8ecdb983 | ||
|   | 8f547c9d5d | ||
|   | 821f3e8ddf | ||
|   | 81be25bbd6 | ||
|   | 26d27fdbf8 | ||
|   | c7e597cf72 | ||
|   | e4db68a1da | ||
|   | 980a90f8fb | ||
|   | 87e4bf52b7 | ||
|   | a368df61ac | ||
|   | 5e66b104dc | ||
|   | 08a5f41505 | ||
|   | ec36762309 | ||
|   | aa70020bc8 | ||
|   | 1e77e6fc79 | ||
|   | 68203e7091 | ||
|   | cff0752bcc | ||
|   | 45d8e11123 | ||
|   | 5aee714b70 | ||
|   | 3c3c3b7c69 | ||
|   | 6ba2913075 | ||
|   | 75c4e1cd63 | ||
|   | fb4a4ca4a2 | ||
|   | 4c2c1297be | ||
|   | c5dbdad5fc | ||
|   | 4e05bcd3b6 | ||
|   | 3555550d5e | ||
|   | 8e05fa2728 | ||
|   | d7c3050e2d | ||
|   | 89ba8562c3 | ||
|   | 82da73baff | ||
|   | b2eeda9b46 | ||
|   | 0db41a3773 | ||
|   | 8d7f7e61dd | ||
|   | 3f4b253dac | ||
|   | e4cbe5126a | ||
|   | 8357739ef8 | ||
|   | 6bef9334b7 | ||
|   | 8c49f45ac8 | ||
|   | 46600740fe | ||
|   | c281e868a0 | ||
|   | 039431a73f | ||
|   | d4f8ea1c53 | ||
|   | 96f5e2b33e | ||
|   | 93a542d52c | ||
|   | f4fcba74ff | ||
|   | cadbf7cd8b | ||
|   | dd8e1aef51 | ||
|   | 15dc37a139 | ||
|   | c23ad83c59 | ||
|   | 4a6f550acb | ||
|   | 8078d78c30 | ||
|   | d7185d71c6 | ||
|   | 522ecba180 | ||
|   | 9b22f6183f | ||
|   | b2c35e4fb0 | ||
|   | af543daf1c | ||
|   | baeb4079ee | ||
|   | f91f9801b4 | ||
|   | c5634335b0 | ||
|   | 2b4317349f | ||
|   | 96bfd1f8be | ||
|   | 33ab53068e | ||
|   | 40123ae6da | ||
|   | df149524d4 | ||
|   | f0ee9cdcf8 | 
							
								
								
									
										35
									
								
								NEWS
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								NEWS
									
									
									
									
									
								
							| @@ -1,3 +1,38 @@ | ||||
| 3.37.3 | ||||
| ====== | ||||
| * Refactor and clean up window picker | ||||
|   [Jonas D., Florian; !1297, !1298, !1305, !1345, !1353] | ||||
| * Move calendar events out of notifications list [Florian; !1282] | ||||
| * Refine app folder dialogs [Georges; !1301] | ||||
| * Hide switch-user button on lock screen if unsupported [Chingkai; #2687] | ||||
| * Refactor and clean up app picker pagination [Georges; !1271] | ||||
| * Add API to retrieve specified mimetypes from clipboards [Carlos; !1321] | ||||
| * Support prepending workspace with horizontal layouts [Florian; #2916] | ||||
| * Update microphone icon on input volume changes [fludixx; #2902] | ||||
| * Cache labels on GPU [Daniel; !1329] | ||||
| * Fix regressions in redesigned modal dialogs [Florian, Jonas D.; #2491, !1336] | ||||
| * Use GIcon for all application icons [Florian; !1342] | ||||
| * Support pre-authenticated logins in vmware environments [yun341; #1983] | ||||
| * Better support sandboxed apps with multiple .desktop files [Florian; #219] | ||||
| * Fix on-screen keyboard size in portrait orientation [Florian; #2349] | ||||
| * Plugged leaks [Sebastian, Daniel, Florian; !1306, !1319, !1341] | ||||
| * Misc. bug fixes and cleanups [Jonas D., Georges, Marco, Florian, Sebastian, | ||||
|   MOZGIII, Daniel, Mariana, Jonas Å.; !1296, !1295, #2643, !1300, !1309, | ||||
|   !1119, #2901, !1313, !1251, !1285, !1307, !1318, !1310, !1320, !1327, !1315, | ||||
|   !1289, !1331, !1332, !1333, !1334, !1340, !1287, !1308, !1346, !1299, !1343, | ||||
|   !1351, !1352, !1322] | ||||
|  | ||||
| Contributors: | ||||
|   Marco Trevisan (Treviño), Chingkai, Jonas Dreßler, Carlos Garnacho, | ||||
|   Sebastian Keller, MOZGIII, Florian Müllner, Georges Basile Stavracas Neto, | ||||
|   Mariana Picolo, Daniel van Vugt, fludixx, yun341, Jonas Ådahl | ||||
|  | ||||
| Translators: | ||||
|   Daniel Mustieles [es], Boyuan Yang [zh_CN], Yuri Chornoivan [uk], | ||||
|   Jordi Mas [ca], sicklylife [ja], Emin Tufan Çetin [tr], | ||||
|   Baurzhan Muftakhidinov [kk], Florentina Mușat [ro], Aurimas Černius [lt], | ||||
|   Rūdolfs Mazurs [lv] | ||||
|  | ||||
| 3.37.2 | ||||
| ====== | ||||
| * Add support for "PrefersNonDefaultGPU" desktop key [Bastien; !1226] | ||||
|   | ||||
| @@ -6,25 +6,25 @@ | ||||
|     <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 alias="icons/scalable/actions/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 alias="icons/message-indicator-symbolic.svg">message-indicator-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/message-indicator-symbolic.svg">message-indicator-symbolic.svg</file> | ||||
|     <file>no-events.svg</file> | ||||
|     <file>no-notifications.svg</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> | ||||
|     <file alias="icons/pointer-double-click-symbolic.svg">pointer-double-click-symbolic.svg</file> | ||||
|     <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 alias="icons/scalable/status/eye-open-negative-filled-symbolic.svg">eye-open-negative-filled-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/eye-not-looking-symbolic.svg">eye-not-looking-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/actions/pointer-double-click-symbolic.svg">pointer-double-click-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/actions/pointer-drag-symbolic.svg">pointer-drag-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/actions/pointer-primary-click-symbolic.svg">pointer-primary-click-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/actions/pointer-secondary-click-symbolic.svg">pointer-secondary-click-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/keyboard-caps-lock-filled-symbolic.svg">keyboard-caps-lock-filled-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/keyboard-enter-symbolic.svg">keyboard-enter-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/keyboard-hide-symbolic.svg">keyboard-hide-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/keyboard-layout-filled-symbolic.svg">keyboard-layout-filled-symbolic.svg</file> | ||||
|     <file alias="icons/scalable/status/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> | ||||
|   | ||||
| @@ -109,6 +109,17 @@ | ||||
|         the shell. | ||||
|       </description> | ||||
|     </key> | ||||
|     <key name="app-picker-layout" type="aa{sv}"> | ||||
|       <default>[]</default> | ||||
|       <summary>Layout of the app picker</summary> | ||||
|       <description> | ||||
|         Layout of the app picker. Each entry in the array is a page. Pages are | ||||
|         stored in the order they appear in GNOME Shell. Each page contains an | ||||
|         “application id” → 'data' pair. Currently, the following values are | ||||
|         stored as 'data': | ||||
|           • “position”: the position of the application icon in the page | ||||
|       </description> | ||||
|     </key> | ||||
|     <child name="keybindings" schema="org.gnome.shell.keybindings"/> | ||||
|   </schema> | ||||
|  | ||||
|   | ||||
| @@ -1,17 +1,13 @@ | ||||
| /* App Grid */ | ||||
|  | ||||
| $app_icon_size: 96px; | ||||
| $app_icon_padding: 24px; | ||||
|  | ||||
| // app icons | ||||
| .icon-grid { | ||||
|   -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; | ||||
|   } | ||||
|   row-spacing: $base_spacing * 6; | ||||
|   column-spacing: $base_spacing * 6; | ||||
|   max-row-spacing: $base_spacing * 12; | ||||
|   max-column-spacing: $base_spacing * 12; | ||||
| } | ||||
|  | ||||
| /* App Icons */ | ||||
| @@ -42,8 +38,8 @@ $app_grid_fg_color: #fff; | ||||
| .app-folder-dialog { | ||||
|   border-radius: $modal_radius * 1.5; | ||||
|   border: 1px solid $osd_outer_borders_color; | ||||
|   spacing: 12px; | ||||
|   background-color: transparentize(darken($osd_bg_color,10%), 0.05); | ||||
|   padding: 12px; | ||||
|  | ||||
|   & .folder-name-container { | ||||
|     padding: 24px 36px 0; | ||||
| @@ -52,7 +48,7 @@ $app_grid_fg_color: #fff; | ||||
|     & .folder-name-label, | ||||
|     & .folder-name-entry { | ||||
|       font-size: 18pt; | ||||
|       font-weight: bold; | ||||
|       font-weight: 800; | ||||
|     } | ||||
|  | ||||
|     & .folder-name-entry { width: 300px } | ||||
| @@ -71,11 +67,24 @@ $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: 800px; | ||||
|   height: 600px; | ||||
|   width: 620px; | ||||
|   height: 620px; | ||||
| } | ||||
|  | ||||
| .app-folder-icon { | ||||
|   | ||||
| @@ -1,19 +1,20 @@ | ||||
| /* Window Picker */ | ||||
|  | ||||
| $window_picker_spacing: $base_spacing * 2; // 16px | ||||
| $window_picker_padding: $base_padding * 2; // 16px | ||||
| $window_picker_spacing: $base_spacing; // 6px | ||||
| $window_picker_padding: $base_padding * 2; // 12px | ||||
|  | ||||
| $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 { | ||||
|   // Space between window thumbnails | ||||
|   -horizontal-spacing: $window_picker_spacing; | ||||
|   -vertical-spacing: $window_picker_spacing; | ||||
|   spacing: $window_picker_spacing; | ||||
|  | ||||
|   // Padding for container around window thumbnails | ||||
|   padding: $window_picker_padding; | ||||
|  | ||||
| @@ -22,7 +23,7 @@ $window_close_button_padding: 3px; | ||||
|  | ||||
| // Borders on window thumbnails | ||||
| .window-clone-border { | ||||
|   border-width: 6px; | ||||
|   border-width: $window_clone_border_size; | ||||
|   border-style: solid; | ||||
|   border-color: $window_thumbnail_border_color; | ||||
|   border-radius: $base_border_radius + 2; | ||||
| @@ -54,8 +55,6 @@ $window_close_button_padding: 3px; | ||||
|   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%); | ||||
|   } | ||||
|   | ||||
| @@ -71,7 +71,7 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this._userVerifier.connect('verification-complete', this._onVerificationComplete.bind(this)); | ||||
|         this._userVerifier.connect('reset', this._onReset.bind(this)); | ||||
|         this._userVerifier.connect('smartcard-status-changed', this._onSmartcardStatusChanged.bind(this)); | ||||
|         this._userVerifier.connect('ovirt-user-authenticated', this._onOVirtUserAuthenticated.bind(this)); | ||||
|         this._userVerifier.connect('credential-manager-authenticated', this._onCredentialManagerAuthenticated.bind(this)); | ||||
|         this.smartcardDetected = this._userVerifier.smartcardDetected; | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
| @@ -242,7 +242,7 @@ var AuthPrompt = GObject.registerClass({ | ||||
|         this.emit('prompted'); | ||||
|     } | ||||
|  | ||||
|     _onOVirtUserAuthenticated() { | ||||
|     _onCredentialManagerAuthenticated() { | ||||
|         if (this.verificationStatus != AuthPromptStatus.VERIFICATION_SUCCEEDED) | ||||
|             this.reset(); | ||||
|     } | ||||
|   | ||||
							
								
								
									
										24
									
								
								js/gdm/credentialManager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								js/gdm/credentialManager.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported CredentialManager */ | ||||
|  | ||||
| class CredentialManager { | ||||
|     constructor(service) { | ||||
|         this._token = null; | ||||
|         this._service = service; | ||||
|         this._authenticatedSignalId = null; | ||||
|     } | ||||
|  | ||||
|     get token() { | ||||
|         return this._token; | ||||
|     } | ||||
|  | ||||
|     set token(t) { | ||||
|         this._token = t; | ||||
|         if (this._token) | ||||
|             this.emit('user-authenticated', this._token); | ||||
|     } | ||||
|  | ||||
|     get service() { | ||||
|         return this._service; | ||||
|     } | ||||
| } | ||||
| @@ -810,8 +810,8 @@ var LoginDialog = GObject.registerClass({ | ||||
|             return; | ||||
|  | ||||
|         this._logoBin.destroy_all_children(); | ||||
|         const [valid, resourceScale] = this._logoBin.get_resource_scale(); | ||||
|         if (this._logoFile && valid) { | ||||
|         const resourceScale = this._logoBin.get_resource_scale(); | ||||
|         if (this._logoFile) { | ||||
|             let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|             this._logoBin.add_child(this._textureCache.load_file_async(this._logoFile, | ||||
|                                                                        -1, -1, | ||||
|   | ||||
| @@ -3,6 +3,9 @@ | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Signals = imports.signals; | ||||
| const Credential = imports.gdm.credentialManager; | ||||
|  | ||||
| var SERVICE_NAME = 'gdm-ovirtcred'; | ||||
|  | ||||
| const OVirtCredentialsIface = ` | ||||
| <node> | ||||
| @@ -28,30 +31,14 @@ function OVirtCredentials() { | ||||
|     return self; | ||||
| } | ||||
|  | ||||
| var OVirtCredentialsManager = class { | ||||
| var OVirtCredentialsManager = class OVirtCredentialsManager extends Credential.CredentialManager { | ||||
|     constructor() { | ||||
|         this._token = null; | ||||
|  | ||||
|         super(SERVICE_NAME); | ||||
|         this._credentials = new OVirtCredentials(); | ||||
|         this._credentials.connectSignal('UserAuthenticated', | ||||
|                                         this._onUserAuthenticated.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _onUserAuthenticated(proxy, sender, [token]) { | ||||
|         this._token = token; | ||||
|         this.emit('user-authenticated', token); | ||||
|     } | ||||
|  | ||||
|     hasToken() { | ||||
|         return this._token != null; | ||||
|     } | ||||
|  | ||||
|     getToken() { | ||||
|         return this._token; | ||||
|     } | ||||
|  | ||||
|     resetToken() { | ||||
|         this._token = null; | ||||
|             (proxy, sender, [token]) => { | ||||
|                 this.token = token; | ||||
|             }); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(OVirtCredentialsManager.prototype); | ||||
|   | ||||
| @@ -8,6 +8,7 @@ const Signals = imports.signals; | ||||
| const Batch = imports.gdm.batch; | ||||
| const Fprint = imports.gdm.fingerprint; | ||||
| const OVirt = imports.gdm.oVirt; | ||||
| const Vmware = imports.gdm.vmware; | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const SmartcardManager = imports.misc.smartcardManager; | ||||
| @@ -24,7 +25,6 @@ Gio._promisify(Gdm.UserVerifierProxy.prototype, | ||||
| var PASSWORD_SERVICE_NAME = 'gdm-password'; | ||||
| var FINGERPRINT_SERVICE_NAME = 'gdm-fingerprint'; | ||||
| var SMARTCARD_SERVICE_NAME = 'gdm-smartcard'; | ||||
| var OVIRT_SERVICE_NAME = 'gdm-ovirtcred'; | ||||
| var FADE_ANIMATION_TIME = 160; | ||||
| var CLONE_FADE_ANIMATION_TIME = 250; | ||||
|  | ||||
| @@ -160,13 +160,20 @@ var ShellUserVerifier = class { | ||||
|  | ||||
|         this._failCounter = 0; | ||||
|  | ||||
|         this._oVirtCredentialsManager = OVirt.getOVirtCredentialsManager(); | ||||
|         this._credentialManagers = {}; | ||||
|         this._credentialManagers[OVirt.SERVICE_NAME] = OVirt.getOVirtCredentialsManager(); | ||||
|         this._credentialManagers[Vmware.SERVICE_NAME] = Vmware.getVmwareCredentialsManager(); | ||||
|  | ||||
|         if (this._oVirtCredentialsManager.hasToken()) | ||||
|             this._oVirtUserAuthenticated(this._oVirtCredentialsManager.getToken()); | ||||
|         for (let service in this._credentialManagers) { | ||||
|             if (this._credentialManagers[service].token) { | ||||
|                 this._onCredentialManagerAuthenticated(this._credentialManagers[service], | ||||
|                     this._credentialManagers[service].token); | ||||
|             } | ||||
|  | ||||
|         this._oVirtUserAuthenticatedId = this._oVirtCredentialsManager.connect('user-authenticated', | ||||
|                                                                                this._oVirtUserAuthenticated.bind(this)); | ||||
|             this._credentialManagers[service]._authenticatedSignalId = | ||||
|                 this._credentialManagers[service].connect('user-authenticated', | ||||
|                                                           this._onCredentialManagerAuthenticated.bind(this)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     begin(userName, hold) { | ||||
| @@ -222,8 +229,11 @@ var ShellUserVerifier = class { | ||||
|         this._smartcardManager.disconnect(this._smartcardRemovedId); | ||||
|         this._smartcardManager = null; | ||||
|  | ||||
|         this._oVirtCredentialsManager.disconnect(this._oVirtUserAuthenticatedId); | ||||
|         this._oVirtCredentialsManager = null; | ||||
|         for (let service in this._credentialManagers) { | ||||
|             let credentialManager = this._credentialManagers[service]; | ||||
|             credentialManager.disconnect(credentialManager._authenticatedSignalId); | ||||
|             credentialManager = null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     answerQuery(serviceName, answer) { | ||||
| @@ -311,9 +321,9 @@ var ShellUserVerifier = class { | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     _oVirtUserAuthenticated(_token) { | ||||
|         this._preemptingService = OVIRT_SERVICE_NAME; | ||||
|         this.emit('ovirt-user-authenticated'); | ||||
|     _onCredentialManagerAuthenticated(credentialManager, _token) { | ||||
|         this._preemptingService = credentialManager.service; | ||||
|         this.emit('credential-manager-authenticated'); | ||||
|     } | ||||
|  | ||||
|     _checkForSmartcard() { | ||||
| @@ -490,9 +500,12 @@ var ShellUserVerifier = class { | ||||
|         if (!this.serviceIsForeground(serviceName)) | ||||
|             return; | ||||
|  | ||||
|         if (serviceName == OVIRT_SERVICE_NAME) { | ||||
|             // The only question asked by this service is "Token?" | ||||
|             this.answerQuery(serviceName, this._oVirtCredentialsManager.getToken()); | ||||
|         let token = null; | ||||
|         if (this._credentialManagers[serviceName]) | ||||
|             token = this._credentialManagers[serviceName].token; | ||||
|  | ||||
|         if (token) { | ||||
|             this.answerQuery(serviceName, token); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -560,8 +573,10 @@ var ShellUserVerifier = class { | ||||
|         // If the login failed with the preauthenticated oVirt credentials | ||||
|         // then discard the credentials and revert to default authentication | ||||
|         // mechanism. | ||||
|         if (this.serviceIsForeground(OVIRT_SERVICE_NAME)) { | ||||
|             this._oVirtCredentialsManager.resetToken(); | ||||
|         let foregroundService = Object.keys(this._credentialManagers).find(service => | ||||
|             this.serviceIsForeground(service)); | ||||
|         if (foregroundService) { | ||||
|             this._credentialManagers[foregroundService].token = null; | ||||
|             this._preemptingService = null; | ||||
|             this._verificationFailed(false); | ||||
|             return; | ||||
|   | ||||
							
								
								
									
										54
									
								
								js/gdm/vmware.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								js/gdm/vmware.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported getVmwareCredentialsManager */ | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Signals = imports.signals; | ||||
| const Credential = imports.gdm.credentialManager; | ||||
|  | ||||
| const dbusPath = '/org/vmware/viewagent/Credentials'; | ||||
| const dbusInterface = 'org.vmware.viewagent.Credentials'; | ||||
|  | ||||
| var SERVICE_NAME = 'gdm-vmwcred'; | ||||
|  | ||||
| const VmwareCredentialsIface = '<node> \ | ||||
| <interface name="' + dbusInterface + '"> \ | ||||
| <signal name="UserAuthenticated"> \ | ||||
|     <arg type="s" name="token"/> \ | ||||
| </signal> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
|  | ||||
| const VmwareCredentialsInfo = Gio.DBusInterfaceInfo.new_for_xml(VmwareCredentialsIface); | ||||
|  | ||||
| let _vmwareCredentialsManager = null; | ||||
|  | ||||
| function VmwareCredentials() { | ||||
|     var self = new Gio.DBusProxy({ g_connection: Gio.DBus.session, | ||||
|                                    g_interface_name: VmwareCredentialsInfo.name, | ||||
|                                    g_interface_info: VmwareCredentialsInfo, | ||||
|                                    g_name: dbusInterface, | ||||
|                                    g_object_path: dbusPath, | ||||
|                                    g_flags: Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES }); | ||||
|     self.init(null); | ||||
|     return self; | ||||
| } | ||||
|  | ||||
| var VmwareCredentialsManager = class VmwareCredentialsManager extends Credential.CredentialManager { | ||||
|     constructor() { | ||||
|         super(SERVICE_NAME); | ||||
|         this._credentials = new VmwareCredentials(); | ||||
|         this._credentials.connectSignal('UserAuthenticated', | ||||
|             (proxy, sender, [token]) => { | ||||
|                 this.token = token; | ||||
|             }); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(VmwareCredentialsManager.prototype); | ||||
|  | ||||
| function getVmwareCredentialsManager() { | ||||
|     if (!_vmwareCredentialsManager) | ||||
|         _vmwareCredentialsManager = new VmwareCredentialsManager(); | ||||
|  | ||||
|     return _vmwareCredentialsManager; | ||||
| } | ||||
| @@ -6,6 +6,8 @@ | ||||
|     <file>gdm/fingerprint.js</file> | ||||
|     <file>gdm/loginDialog.js</file> | ||||
|     <file>gdm/oVirt.js</file> | ||||
|     <file>gdm/credentialManager.js</file> | ||||
|     <file>gdm/vmware.js</file> | ||||
|     <file>gdm/realmd.js</file> | ||||
|     <file>gdm/util.js</file> | ||||
|  | ||||
| @@ -108,7 +110,9 @@ | ||||
|     <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/workspaceAnimation.js</file> | ||||
|     <file>ui/workspaceSwitcherPopup.js</file> | ||||
|     <file>ui/workspaceThumbnail.js</file> | ||||
|     <file>ui/workspacesView.js</file> | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported findUrls, spawn, spawnCommandLine, spawnApp, trySpawnCommandLine, | ||||
|             formatTime, formatTimeSpan, createTimeLabel, insertSorted, | ||||
|             makeCloseButton, ensureActorVisibleInScrollView, wiggle */ | ||||
|             ensureActorVisibleInScrollView, wiggle */ | ||||
|  | ||||
| const { Clutter, Gio, GLib, GObject, Shell, St, GnomeDesktop } = imports.gi; | ||||
| const { Clutter, Gio, GLib, Shell, St, GnomeDesktop } = imports.gi; | ||||
| const Gettext = imports.gettext; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| @@ -363,51 +363,6 @@ 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(); | ||||
|   | ||||
| @@ -852,6 +852,9 @@ class AppSwitcher extends SwitcherPopup.SwitcherList { | ||||
|         if (index === -1) | ||||
|             return; | ||||
|  | ||||
|         this._arrows[index].destroy(); | ||||
|         this._arrows.splice(index, 1); | ||||
|  | ||||
|         this.icons.splice(index, 1); | ||||
|         this.removeItem(index); | ||||
|     } | ||||
|   | ||||
| @@ -60,7 +60,7 @@ class Animation extends St.Bin { | ||||
|     } | ||||
|  | ||||
|     _loadFile(file, width, height) { | ||||
|         let [validResourceScale, resourceScale] = this.get_resource_scale(); | ||||
|         const resourceScale = this.get_resource_scale(); | ||||
|         let wasPlaying = this._isPlaying; | ||||
|  | ||||
|         if (this._isPlaying) | ||||
| @@ -69,12 +69,6 @@ class Animation extends St.Bin { | ||||
|         this._isLoaded = false; | ||||
|         this.destroy_all_children(); | ||||
|  | ||||
|         if (!validResourceScale) { | ||||
|             if (wasPlaying) | ||||
|                 this.play(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let textureCache = St.TextureCache.get_default(); | ||||
|         let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|         this._animations = textureCache.load_sliced_image(file, width, height, | ||||
|   | ||||
							
								
								
									
										1688
									
								
								js/ui/appDisplay.js
									
									
									
									
									
								
							
							
						
						
									
										1688
									
								
								js/ui/appDisplay.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -712,13 +712,18 @@ var BackgroundManager = class BackgroundManager { | ||||
|         } | ||||
|  | ||||
|         let newBackgroundActor = this._createBackgroundActor(); | ||||
|         newBackgroundActor.vignette_sharpness = this.backgroundActor.vignette_sharpness; | ||||
|         newBackgroundActor.brightness = this.backgroundActor.brightness; | ||||
|  | ||||
|         const oldContent = this.backgroundActor.content; | ||||
|         const newContent = newBackgroundActor.content; | ||||
|  | ||||
|         newContent.vignette_sharpness = oldContent.vignette_sharpness; | ||||
|         newContent.brightness = oldContent.brightness; | ||||
|  | ||||
|         newBackgroundActor.visible = this.backgroundActor.visible; | ||||
|  | ||||
|         this._newBackgroundActor = newBackgroundActor; | ||||
|  | ||||
|         let background = newBackgroundActor.background; | ||||
|         const { background } = newBackgroundActor.content; | ||||
|  | ||||
|         if (background.isLoaded) { | ||||
|             this._swapBackgroundActor(); | ||||
|   | ||||
| @@ -784,6 +784,7 @@ class DateMenuButton extends PanelMenu.Button { | ||||
|  | ||||
|         this._clockDisplay = new St.Label({ style_class: 'clock' }); | ||||
|         this._clockDisplay.clutter_text.y_align = Clutter.ActorAlign.CENTER; | ||||
|         this._clockDisplay.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|  | ||||
|         this._indicator = new MessagesIndicator(); | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported Dialog, MessageDialogContent, ListSection, ListSectionItem */ | ||||
|  | ||||
| const { Clutter, GObject, Meta, Pango, St } = imports.gi; | ||||
| const { Clutter, GLib, GObject, Meta, Pango, St } = imports.gi; | ||||
|  | ||||
| function _setLabel(label, value) { | ||||
|     label.set({ | ||||
| @@ -221,13 +221,16 @@ var MessageDialogContent = GObject.registerClass({ | ||||
|             this._updateTitleStyleLater = Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => { | ||||
|                 this._updateTitleStyleLater = 0; | ||||
|                 this._title.add_style_class_name('leightweight'); | ||||
|                 return false; | ||||
|                 return GLib.SOURCE_REMOVE; | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     set title(title) { | ||||
|         if (this._title.text === title) | ||||
|             return; | ||||
|  | ||||
|         _setLabel(this._title, title); | ||||
|  | ||||
|         this._title.remove_style_class_name('leightweight'); | ||||
| @@ -237,6 +240,9 @@ var MessageDialogContent = GObject.registerClass({ | ||||
|     } | ||||
|  | ||||
|     set description(description) { | ||||
|         if (this._description.text === description) | ||||
|             return; | ||||
|  | ||||
|         _setLabel(this._description, description); | ||||
|         this.notify('description'); | ||||
|     } | ||||
|   | ||||
							
								
								
									
										44
									
								
								js/ui/dnd.js
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								js/ui/dnd.js
									
									
									
									
									
								
							| @@ -375,19 +375,30 @@ var _Draggable = class _Draggable { | ||||
|  | ||||
|             this._dragActorSource = undefined; | ||||
|             this._dragOrigParent = this.actor.get_parent(); | ||||
|             this._dragOrigX = this._dragActor.x; | ||||
|             this._dragOrigY = this._dragActor.y; | ||||
|             this._dragActorHadFixedPos = this._dragActor.fixed_position_set; | ||||
|             this._dragOrigX = this._dragActor.allocation.x1; | ||||
|             this._dragOrigY = this._dragActor.allocation.y1; | ||||
|             this._dragOrigWidth = this._dragActor.allocation.get_width(); | ||||
|             this._dragOrigHeight = this._dragActor.allocation.get_height(); | ||||
|             this._dragOrigScale = this._dragActor.scale_x; | ||||
|  | ||||
|             // When the actor gets reparented to the uiGroup, it will be | ||||
|             // allocated its preferred size, so use that size instead of the | ||||
|             // current allocation size. | ||||
|             const [, newAllocatedWidth] = this._dragActor.get_preferred_width(-1); | ||||
|             const [, newAllocatedHeight] = this._dragActor.get_preferred_height(-1); | ||||
|  | ||||
|             const transformedAllocation = | ||||
|                 Shell.util_get_transformed_allocation(this._dragActor); | ||||
|  | ||||
|             // Set the actor's scale such that it will keep the same | ||||
|             // transformed size when it's reparented to the uiGroup | ||||
|             let [scaledWidth, scaledHeight] = this.actor.get_transformed_size(); | ||||
|             this._dragActor.set_scale(scaledWidth / this.actor.width, | ||||
|                                       scaledHeight / this.actor.height); | ||||
|             this._dragActor.set_scale( | ||||
|                 transformedAllocation.get_width() / newAllocatedWidth, | ||||
|                 transformedAllocation.get_height() / newAllocatedHeight); | ||||
|  | ||||
|             let [actorStageX, actorStageY] = this.actor.get_transformed_position(); | ||||
|             this._dragOffsetX = actorStageX - this._dragStartX; | ||||
|             this._dragOffsetY = actorStageY - this._dragStartY; | ||||
|             this._dragOffsetX = transformedAllocation.x1 - this._dragStartX; | ||||
|             this._dragOffsetY = transformedAllocation.y1 - this._dragStartY; | ||||
|  | ||||
|             this._dragOrigParent.remove_actor(this._dragActor); | ||||
|             Main.uiGroup.add_child(this._dragActor); | ||||
| @@ -417,6 +428,10 @@ var _Draggable = class _Draggable { | ||||
|         this._dragOffsetX -= transX; | ||||
|         this._dragOffsetY -= transY; | ||||
|  | ||||
|         this._dragActor.set_position( | ||||
|             this._dragX + this._dragOffsetX, | ||||
|             this._dragY + this._dragOffsetY); | ||||
|  | ||||
|         if (this._dragActorMaxSize != undefined) { | ||||
|             let [scaledWidth, scaledHeight] = this._dragActor.get_transformed_size(); | ||||
|             let currentSize = Math.max(scaledWidth, scaledHeight); | ||||
| @@ -635,9 +650,15 @@ var _Draggable = class _Draggable { | ||||
|             if (parentWidth != 0) | ||||
|                 parentScale = parentScaledWidth / parentWidth; | ||||
|  | ||||
|             // Also adjust for the difference in the original actor width | ||||
|             // and the width it is now (children of uiGroup always get | ||||
|             // allocated their preferred size) | ||||
|             const childScaleX = | ||||
|                 this._dragOrigWidth / this._dragActor.allocation.get_width(); | ||||
|  | ||||
|             x = parentX + parentScale * this._dragOrigX; | ||||
|             y = parentY + parentScale * this._dragOrigY; | ||||
|             scale = this._dragOrigScale * parentScale; | ||||
|             scale = this._dragOrigScale * parentScale * childScaleX; | ||||
|         } else { | ||||
|             // Snap back actor to its original stage position | ||||
|             x = this._snapBackX; | ||||
| @@ -718,7 +739,10 @@ var _Draggable = class _Draggable { | ||||
|             Main.uiGroup.remove_child(this._dragActor); | ||||
|             this._dragOrigParent.add_actor(this._dragActor); | ||||
|             dragActor.set_scale(this._dragOrigScale, this._dragOrigScale); | ||||
|             dragActor.set_position(this._dragOrigX, this._dragOrigY); | ||||
|             if (this._dragActorHadFixedPos) | ||||
|                 dragActor.set_position(this._dragOrigX, this._dragOrigY); | ||||
|             else | ||||
|                 dragActor.fixed_position_set = false; | ||||
|         } else { | ||||
|             dragActor.destroy(); | ||||
|         } | ||||
|   | ||||
| @@ -139,7 +139,9 @@ function checkForUpdates() { | ||||
|             return; | ||||
|         if (extension.hasUpdate) | ||||
|             return; | ||||
|         metadatas[uuid] = extension.metadata; | ||||
|         metadatas[uuid] = { | ||||
|             version: extension.metadata.version, | ||||
|         }; | ||||
|     }); | ||||
|  | ||||
|     if (Object.keys(metadatas).length === 0) | ||||
|   | ||||
							
								
								
									
										1724
									
								
								js/ui/iconGrid.js
									
									
									
									
									
								
							
							
						
						
									
										1724
									
								
								js/ui/iconGrid.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,5 +1,5 @@ | ||||
| /* exported InhibitShortcutsDialog */ | ||||
| const { Clutter, Gio, GLib, GObject, Gtk, Meta, Shell, St } = imports.gi; | ||||
| const { Clutter, Gio, GLib, GObject, Gtk, Meta, Pango, Shell, St } = imports.gi; | ||||
|  | ||||
| const Dialog = imports.ui.dialog; | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
| @@ -90,6 +90,8 @@ var InhibitShortcutsDialog = GObject.registerClass({ | ||||
|                 text: _('You can restore shortcuts by pressing %s.').format(restoreAccel), | ||||
|                 style_class: 'message-dialog-description', | ||||
|             }); | ||||
|             restoreLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|             restoreLabel.clutter_text.line_wrap = true; | ||||
|             content.add_child(restoreLabel); | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -42,7 +42,7 @@ const defaultKeysPost = [ | ||||
|     [[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }], | ||||
|      [{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }], | ||||
|      [{ label: '=/<', width: 3, level: 3, right: true }], | ||||
|      [{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key' }, { action: 'hide', extraClassName: 'hide-key' }]], | ||||
|      [{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key', icon: 'keyboard-layout-filled-symbolic' }, { action: 'hide', extraClassName: 'hide-key', icon: 'go-down-symbolic' }]], | ||||
|     [[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }], | ||||
|      [{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }], | ||||
|      [{ label: '?123', width: 3, level: 2, right: true }], | ||||
| @@ -61,6 +61,24 @@ class AspectContainer extends St.Widget { | ||||
|         this.queue_relayout(); | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_width(forHeight) { | ||||
|         let [min, nat] = super.vfunc_get_preferred_width(forHeight); | ||||
|  | ||||
|         if (forHeight > 0) | ||||
|             nat = forHeight * this._ratio; | ||||
|  | ||||
|         return [min, nat]; | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_height(forWidth) { | ||||
|         let [min, nat] = super.vfunc_get_preferred_height(forWidth); | ||||
|  | ||||
|         if (forWidth > 0) | ||||
|             nat = forWidth / this._ratio; | ||||
|  | ||||
|         return [min, nat]; | ||||
|     } | ||||
|  | ||||
|     vfunc_allocate(box) { | ||||
|         if (box.get_width() > 0 && box.get_height() > 0) { | ||||
|             let sizeRatio = box.get_width() / box.get_height(); | ||||
| @@ -1608,7 +1626,9 @@ class Keyboard extends St.BoxLayout { | ||||
|              * we allow the OSK being smaller than 1/3rd of the monitor height | ||||
|              * there. | ||||
|              */ | ||||
|             this.height = Math.min(maxHeight, this.get_preferred_height(monitor.width)); | ||||
|             const forWidth = this.get_theme_node().adjust_for_width(monitor.width); | ||||
|             const [, natHeight] = this.get_preferred_height(forWidth); | ||||
|             this.height = Math.min(maxHeight, natHeight); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,6 @@ const Signals = imports.signals; | ||||
|  | ||||
| const Background = imports.ui.background; | ||||
| const BackgroundMenu = imports.ui.backgroundMenu; | ||||
| const LoginManager = imports.misc.loginManager; | ||||
|  | ||||
| const DND = imports.ui.dnd; | ||||
| const Main = imports.ui.main; | ||||
| @@ -295,18 +294,6 @@ var LayoutManager = GObject.registerClass({ | ||||
|         monitorManager.connect('monitors-changed', | ||||
|                                this._monitorsChanged.bind(this)); | ||||
|         this._monitorsChanged(); | ||||
|  | ||||
|         // NVIDIA drivers don't preserve FBO contents across | ||||
|         // suspend/resume, see | ||||
|         // https://bugzilla.gnome.org/show_bug.cgi?id=739178 | ||||
|         if (Shell.util_need_background_refresh()) { | ||||
|             LoginManager.getLoginManager().connect('prepare-for-sleep', | ||||
|                 (lm, suspending) => { | ||||
|                     if (suspending) | ||||
|                         return; | ||||
|                     Meta.Background.refresh_all(); | ||||
|                 }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // This is called by Main after everything else is constructed | ||||
| @@ -1204,7 +1191,8 @@ class HotCorner extends Clutter.Actor { | ||||
|  | ||||
|             if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) { | ||||
|                 this._corner.set_position(this.width - this._corner.width, 0); | ||||
|                 this.set_anchor_point_from_gravity(Clutter.Gravity.NORTH_EAST); | ||||
|                 this.set_pivot_point(1.0, 0.0); | ||||
|                 this.translation_x = -this.width; | ||||
|             } else { | ||||
|                 this._corner.set_position(0, 0); | ||||
|             } | ||||
|   | ||||
| @@ -489,7 +489,10 @@ var Magnifier = class Magnifier { | ||||
|     _updateMouseSprite() { | ||||
|         this._updateSpriteTexture(); | ||||
|         let [xHot, yHot] = this._cursorTracker.get_hot(); | ||||
|         this._mouseSprite.set_anchor_point(xHot, yHot); | ||||
|         this._mouseSprite.set({ | ||||
|             translation_x: -xHot, | ||||
|             translation_y: -yHot, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _updateSpriteTexture() { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported NotificationPolicy, NotificationGenericPolicy, | ||||
|    NotificationApplicationPolicy, Source, SourceActor, SourceActorWithLabel, | ||||
|    NotificationApplicationPolicy, Source, SourceActor, | ||||
|    SystemNotificationSource, MessageTray */ | ||||
|  | ||||
| const { Clutter, Gio, GLib, GObject, Meta, Shell, St } = imports.gi; | ||||
| @@ -655,77 +655,6 @@ class SourceActor extends St.Widget { | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var SourceActorWithLabel = GObject.registerClass( | ||||
| class SourceActorWithLabel extends SourceActor { | ||||
|     _init(source, size) { | ||||
|         super._init(source, size); | ||||
|  | ||||
|         this._counterLabel = new St.Label({ x_align: Clutter.ActorAlign.CENTER, | ||||
|                                             x_expand: true, | ||||
|                                             y_align: Clutter.ActorAlign.CENTER, | ||||
|                                             y_expand: true }); | ||||
|  | ||||
|         this._counterBin = new St.Bin({ style_class: 'summary-source-counter', | ||||
|                                         child: this._counterLabel, | ||||
|                                         layout_manager: new Clutter.BinLayout() }); | ||||
|         this._counterBin.hide(); | ||||
|  | ||||
|         this._counterBin.connect('style-changed', () => { | ||||
|             let themeNode = this._counterBin.get_theme_node(); | ||||
|             this._counterBin.translation_x = themeNode.get_length('-shell-counter-overlap-x'); | ||||
|             this._counterBin.translation_y = themeNode.get_length('-shell-counter-overlap-y'); | ||||
|         }); | ||||
|  | ||||
|         this.add_actor(this._counterBin); | ||||
|  | ||||
|         this._countUpdatedId = this._source.connect('notify::count', this._updateCount.bind(this)); | ||||
|         this._updateCount(); | ||||
|  | ||||
|         this.connect('destroy', () => { | ||||
|             this._source.disconnect(this._countUpdatedId); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     vfunc_allocate(box) { | ||||
|         super.vfunc_allocate(box); | ||||
|  | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [, , naturalWidth, naturalHeight] = this._counterBin.get_preferred_size(); | ||||
|         let direction = this.get_text_direction(); | ||||
|  | ||||
|         if (direction == Clutter.TextDirection.LTR) { | ||||
|             // allocate on the right in LTR | ||||
|             childBox.x1 = box.x2 - naturalWidth; | ||||
|             childBox.x2 = box.x2; | ||||
|         } else { | ||||
|             // allocate on the left in RTL | ||||
|             childBox.x1 = 0; | ||||
|             childBox.x2 = naturalWidth; | ||||
|         } | ||||
|  | ||||
|         childBox.y1 = box.y2 - naturalHeight; | ||||
|         childBox.y2 = box.y2; | ||||
|  | ||||
|         this._counterBin.allocate(childBox); | ||||
|     } | ||||
|  | ||||
|     _updateCount() { | ||||
|         if (this._actorDestroyed) | ||||
|             return; | ||||
|  | ||||
|         this._counterBin.visible = this._source.countVisible; | ||||
|  | ||||
|         let text; | ||||
|         if (this._source.count < 100) | ||||
|             text = this._source.count.toString(); | ||||
|         else | ||||
|             text = String.fromCharCode(0x22EF); // midline horizontal ellipsis | ||||
|  | ||||
|         this._counterLabel.set_text(text); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var Source = GObject.registerClass({ | ||||
|     Properties: { | ||||
|         'count': GObject.ParamSpec.int( | ||||
|   | ||||
| @@ -4,6 +4,10 @@ | ||||
| const { Clutter, GLib, GObject, Meta, Shell, St } = imports.gi; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| // Time for initial animation going into Overview mode; | ||||
| // this is defined here to make it available in imports. | ||||
| var ANIMATION_TIME = 250; | ||||
|  | ||||
| const Background = imports.ui.background; | ||||
| const DND = imports.ui.dnd; | ||||
| const LayoutManager = imports.ui.layout; | ||||
| @@ -14,9 +18,6 @@ const OverviewControls = imports.ui.overviewControls; | ||||
| const Params = imports.misc.params; | ||||
| const WorkspaceThumbnail = imports.ui.workspaceThumbnail; | ||||
|  | ||||
| // Time for initial animation going into Overview mode | ||||
| var ANIMATION_TIME = 250; | ||||
|  | ||||
| // Must be less than ANIMATION_TIME, since we switch to | ||||
| // or from the overview completely after ANIMATION_TIME, | ||||
| // and don't want the shading animation to get cut off | ||||
| @@ -439,19 +440,19 @@ var Overview = class { | ||||
|         this.emit('windows-restacked', stackIndices); | ||||
|     } | ||||
|  | ||||
|     beginItemDrag(_source) { | ||||
|         this.emit('item-drag-begin'); | ||||
|     beginItemDrag(source) { | ||||
|         this.emit('item-drag-begin', source); | ||||
|         this._inItemDrag = true; | ||||
|     } | ||||
|  | ||||
|     cancelledItemDrag(_source) { | ||||
|         this.emit('item-drag-cancelled'); | ||||
|     cancelledItemDrag(source) { | ||||
|         this.emit('item-drag-cancelled', source); | ||||
|     } | ||||
|  | ||||
|     endItemDrag(_source) { | ||||
|     endItemDrag(source) { | ||||
|         if (!this._inItemDrag) | ||||
|             return; | ||||
|         this.emit('item-drag-end'); | ||||
|         this.emit('item-drag-end', source); | ||||
|         this._inItemDrag = false; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -8,8 +8,9 @@ const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const ViewSelector = imports.ui.viewSelector; | ||||
| const WorkspaceThumbnail = imports.ui.workspaceThumbnail; | ||||
| const Overview = imports.ui.overview; | ||||
|  | ||||
| var SIDE_CONTROLS_ANIMATION_TIME = 160; | ||||
| var SIDE_CONTROLS_ANIMATION_TIME = Overview.ANIMATION_TIME; | ||||
|  | ||||
| function getRtlSlideDirection(direction, actor) { | ||||
|     let rtl = actor.text_direction == Clutter.TextDirection.RTL; | ||||
| @@ -422,6 +423,7 @@ class ControlsManager extends St.Widget { | ||||
|         let activeWorkspaceIndex = workspaceManager.get_active_workspace_index(); | ||||
|  | ||||
|         this._workspaceAdjustment = new St.Adjustment({ | ||||
|             actor: this, | ||||
|             value: activeWorkspaceIndex, | ||||
|             lower: 0, | ||||
|             page_increment: 1, | ||||
| @@ -453,8 +455,6 @@ class ControlsManager extends St.Widget { | ||||
|         this._group.add_child(this.viewSelector); | ||||
|         this._group.add_actor(this._thumbnailsSlider); | ||||
|  | ||||
|         layout.connect('allocation-changed', this._updateWorkspacesGeometry.bind(this)); | ||||
|  | ||||
|         Main.overview.connect('showing', this._updateSpacerVisibility.bind(this)); | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
| @@ -477,26 +477,6 @@ class ControlsManager extends St.Widget { | ||||
|         this._workspaceAdjustment.value = activeIndex; | ||||
|     } | ||||
|  | ||||
|     _updateWorkspacesGeometry() { | ||||
|         let [x, y] = this.get_transformed_position(); | ||||
|         let [width, height] = this.get_transformed_size(); | ||||
|         let geometry = { x, y, width, height }; | ||||
|  | ||||
|         let spacing = this.get_theme_node().get_length('spacing'); | ||||
|         let dashWidth = this._dashSlider.getVisibleWidth() + spacing; | ||||
|         let thumbnailsWidth = this._thumbnailsSlider.getNonExpandedWidth() + spacing; | ||||
|  | ||||
|         geometry.width -= dashWidth; | ||||
|         geometry.width -= thumbnailsWidth; | ||||
|  | ||||
|         if (this.get_text_direction() == Clutter.TextDirection.LTR) | ||||
|             geometry.x += dashWidth; | ||||
|         else | ||||
|             geometry.x += thumbnailsWidth; | ||||
|  | ||||
|         this.viewSelector.setWorkspacesFullGeometry(geometry); | ||||
|     } | ||||
|  | ||||
|     _setVisibility() { | ||||
|         // Ignore the case when we're leaving the overview, since | ||||
|         // actors will be made visible again when entering the overview | ||||
|   | ||||
| @@ -120,6 +120,10 @@ var PageIndicators = GObject.registerClass({ | ||||
|         for (let i = 0; i < children.length; i++) | ||||
|             this._updateIndicator(children[i], i); | ||||
|     } | ||||
|  | ||||
|     get nPages() { | ||||
|         return this._nPages; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var AnimatedPageIndicators = GObject.registerClass( | ||||
|   | ||||
| @@ -675,7 +675,7 @@ class PanelCorner extends St.DrawingArea { | ||||
|         let borderWidth = node.get_length('-panel-corner-border-width'); | ||||
|  | ||||
|         this.set_size(cornerRadius, borderWidth + cornerRadius); | ||||
|         this.set_anchor_point(0, borderWidth); | ||||
|         this.translation_y = -borderWidth; | ||||
|     } | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -62,6 +62,7 @@ var StreamSlider = class { | ||||
|  | ||||
|         this._stream = null; | ||||
|         this._volumeCancellable = null; | ||||
|         this._icons = []; | ||||
|     } | ||||
|  | ||||
|     get stream() { | ||||
| @@ -182,24 +183,15 @@ var StreamSlider = class { | ||||
|         if (!this._stream) | ||||
|             return null; | ||||
|  | ||||
|         let icons = ["audio-volume-muted-symbolic", | ||||
|                      "audio-volume-low-symbolic", | ||||
|                      "audio-volume-medium-symbolic", | ||||
|                      "audio-volume-high-symbolic", | ||||
|                      "audio-volume-overamplified-symbolic"]; | ||||
|  | ||||
|         let volume = this._stream.volume; | ||||
|         let n; | ||||
|         if (this._stream.is_muted || volume <= 0) { | ||||
|             n = 0; | ||||
|         } else { | ||||
|             n = Math.ceil(3 * volume / this._control.get_vol_max_norm()); | ||||
|             if (n < 1) | ||||
|                 n = 1; | ||||
|             else if (n > 3) | ||||
|                 n = 4; | ||||
|             n = Math.clamp(n, 1, this._icons.length - 1); | ||||
|         } | ||||
|         return icons[n]; | ||||
|         return this._icons[n]; | ||||
|     } | ||||
|  | ||||
|     getLevel() { | ||||
| @@ -223,6 +215,13 @@ var OutputStreamSlider = class extends StreamSlider { | ||||
|     constructor(control) { | ||||
|         super(control); | ||||
|         this._slider.accessible_name = _("Volume"); | ||||
|         this._icons = [ | ||||
|             'audio-volume-muted-symbolic', | ||||
|             'audio-volume-low-symbolic', | ||||
|             'audio-volume-medium-symbolic', | ||||
|             'audio-volume-high-symbolic', | ||||
|             'audio-volume-overamplified-symbolic', | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     _connectStream(stream) { | ||||
| @@ -274,6 +273,12 @@ var InputStreamSlider = class extends StreamSlider { | ||||
|         this._control.connect('stream-added', this._maybeShowInput.bind(this)); | ||||
|         this._control.connect('stream-removed', this._maybeShowInput.bind(this)); | ||||
|         this._icon.icon_name = 'audio-input-microphone-symbolic'; | ||||
|         this._icons = [ | ||||
|             'microphone-sensitivity-muted-symbolic', | ||||
|             'microphone-sensitivity-low-symbolic', | ||||
|             'microphone-sensitivity-medium-symbolic', | ||||
|             'microphone-sensitivity-high-symbolic', | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     _connectStream(stream) { | ||||
| @@ -319,7 +324,7 @@ var VolumeMenu = class extends PopupMenu.PopupMenuSection { | ||||
|  | ||||
|         this._output = new OutputStreamSlider(this._control); | ||||
|         this._output.connect('stream-updated', () => { | ||||
|             this.emit('icon-changed'); | ||||
|             this.emit('output-icon-changed'); | ||||
|         }); | ||||
|         this.addMenuItem(this._output.item); | ||||
|  | ||||
| @@ -327,6 +332,9 @@ var VolumeMenu = class extends PopupMenu.PopupMenuSection { | ||||
|         this._input.item.connect('notify::visible', () => { | ||||
|             this.emit('input-visible-changed'); | ||||
|         }); | ||||
|         this._input.connect('stream-updated', () => { | ||||
|             this.emit('input-icon-changed'); | ||||
|         }); | ||||
|         this.addMenuItem(this._input.item); | ||||
|  | ||||
|         this.addMenuItem(new PopupMenu.PopupSeparatorMenuItem()); | ||||
| @@ -343,7 +351,7 @@ var VolumeMenu = class extends PopupMenu.PopupMenuSection { | ||||
|             this._readInput(); | ||||
|             this._readOutput(); | ||||
|         } else { | ||||
|             this.emit('icon-changed'); | ||||
|             this.emit('output-icon-changed'); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -355,10 +363,14 @@ var VolumeMenu = class extends PopupMenu.PopupMenuSection { | ||||
|         this._input.stream = this._control.get_default_source(); | ||||
|     } | ||||
|  | ||||
|     getIcon() { | ||||
|     getOutputIcon() { | ||||
|         return this._output.getIcon(); | ||||
|     } | ||||
|  | ||||
|     getInputIcon() { | ||||
|         return this._input.getIcon(); | ||||
|     } | ||||
|  | ||||
|     getLevel() { | ||||
|         return this._output.getLevel(); | ||||
|     } | ||||
| @@ -382,21 +394,24 @@ class Indicator extends PanelMenu.SystemIndicator { | ||||
|  | ||||
|         this._control = getMixerControl(); | ||||
|         this._volumeMenu = new VolumeMenu(this._control); | ||||
|         this._volumeMenu.connect('icon-changed', () => { | ||||
|             let icon = this._volumeMenu.getIcon(); | ||||
|         this._volumeMenu.connect('output-icon-changed', () => { | ||||
|             let icon = this._volumeMenu.getOutputIcon(); | ||||
|  | ||||
|             if (icon != null) | ||||
|                 this._primaryIndicator.icon_name = icon; | ||||
|             this._primaryIndicator.visible = icon !== null; | ||||
|         }); | ||||
|  | ||||
|         this._inputIndicator.set({ | ||||
|             icon_name: 'audio-input-microphone-symbolic', | ||||
|             visible: this._volumeMenu.getInputVisible(), | ||||
|         }); | ||||
|         this._inputIndicator.visible = this._volumeMenu.getInputVisible(); | ||||
|         this._volumeMenu.connect('input-visible-changed', () => { | ||||
|             this._inputIndicator.visible = this._volumeMenu.getInputVisible(); | ||||
|         }); | ||||
|         this._volumeMenu.connect('input-icon-changed', () => { | ||||
|             let icon = this._volumeMenu.getInputIcon(); | ||||
|  | ||||
|             if (icon !== null) | ||||
|                 this._inputIndicator.icon_name = icon; | ||||
|         }); | ||||
|  | ||||
|         this.menu.addMenuItem(this._volumeMenu); | ||||
|     } | ||||
| @@ -406,7 +421,7 @@ class Indicator extends PanelMenu.SystemIndicator { | ||||
|         if (result == Clutter.EVENT_PROPAGATE || this.menu.actor.mapped) | ||||
|             return result; | ||||
|  | ||||
|         let gicon = new Gio.ThemedIcon({ name: this._volumeMenu.getIcon() }); | ||||
|         let gicon = new Gio.ThemedIcon({ name: this._volumeMenu.getOutputIcon() }); | ||||
|         let level = this._volumeMenu.getLevel(); | ||||
|         let maxLevel = this._volumeMenu.getMaxLevel(); | ||||
|         Main.osdWindowManager.show(-1, gicon, null, level, maxLevel); | ||||
|   | ||||
| @@ -485,6 +485,7 @@ var UnlockDialog = GObject.registerClass({ | ||||
|         this._gdmClient = new Gdm.Client(); | ||||
|  | ||||
|         this._adjustment = new St.Adjustment({ | ||||
|             actor: this, | ||||
|             lower: 0, | ||||
|             upper: 2, | ||||
|             page_size: 1, | ||||
| @@ -524,6 +525,10 @@ var UnlockDialog = GObject.registerClass({ | ||||
|  | ||||
|         this._bgManagers = []; | ||||
|  | ||||
|         const themeContext = St.ThemeContext.get_for_stage(global.stage); | ||||
|         this._scaleChangedId = themeContext.connect('notify::scale-factor', | ||||
|             () => this._updateBackgroundEffects()); | ||||
|  | ||||
|         this._updateBackgrounds(); | ||||
|         this._monitorsChangedId = | ||||
|             Main.layoutManager.connect('monitors-changed', this._updateBackgrounds.bind(this)); | ||||
| @@ -568,11 +573,9 @@ var UnlockDialog = GObject.registerClass({ | ||||
|  | ||||
|         this._screenSaverSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.screensaver' }); | ||||
|  | ||||
|         this._userSwitchEnabledId = 0; | ||||
|         this._userSwitchEnabledId = this._screenSaverSettings.connect('changed::user-switch-enabled', | ||||
|             this._updateUserSwitchVisibility.bind(this)); | ||||
|  | ||||
|         this._userLoadedId = 0; | ||||
|         this._userLoadedId = this._user.connect('notify::is-loaded', | ||||
|             this._updateUserSwitchVisibility.bind(this)); | ||||
|  | ||||
| @@ -626,6 +629,7 @@ var UnlockDialog = GObject.registerClass({ | ||||
|             y: monitor.y, | ||||
|             width: monitor.width, | ||||
|             height: monitor.height, | ||||
|             effect: new Shell.BlurEffect({ name: 'blur' }), | ||||
|         }); | ||||
|  | ||||
|         let bgManager = new Background.BackgroundManager({ | ||||
| @@ -637,19 +641,17 @@ var UnlockDialog = GObject.registerClass({ | ||||
|         this._bgManagers.push(bgManager); | ||||
|  | ||||
|         this._backgroundGroup.add_child(widget); | ||||
|     } | ||||
|  | ||||
|     _updateBackgroundEffects() { | ||||
|         const themeContext = St.ThemeContext.get_for_stage(global.stage); | ||||
|  | ||||
|         let effect = new Shell.BlurEffect({ | ||||
|             brightness: BLUR_BRIGHTNESS, | ||||
|             sigma: BLUR_SIGMA * themeContext.scale_factor, | ||||
|         }); | ||||
|  | ||||
|         this._scaleChangedId = themeContext.connect('notify::scale-factor', () => { | ||||
|             effect.sigma = BLUR_SIGMA * themeContext.scale_factor; | ||||
|         }); | ||||
|  | ||||
|         widget.add_effect(effect); | ||||
|         for (const widget of this._backgroundGroup) { | ||||
|             widget.get_effect('blur').set({ | ||||
|                 brightness: BLUR_BRIGHTNESS, | ||||
|                 sigma: BLUR_SIGMA * themeContext.scale_factor, | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _updateBackgrounds() { | ||||
| @@ -661,6 +663,7 @@ var UnlockDialog = GObject.registerClass({ | ||||
|  | ||||
|         for (let i = 0; i < Main.layoutManager.monitors.length; i++) | ||||
|             this._createBackground(i); | ||||
|         this._updateBackgroundEffects(); | ||||
|     } | ||||
|  | ||||
|     _ensureAuthPrompt() { | ||||
|   | ||||
| @@ -299,10 +299,6 @@ var ViewSelector = GObject.registerClass({ | ||||
|             Main.overview.fadeInDesktop(); | ||||
|     } | ||||
|  | ||||
|     setWorkspacesFullGeometry(geom) { | ||||
|         this._workspacesDisplay.setWorkspacesFullGeometry(geom); | ||||
|     } | ||||
|  | ||||
|     vfunc_hide() { | ||||
|         this.reset(); | ||||
|         this._workspacesDisplay.hide(); | ||||
|   | ||||
| @@ -14,9 +14,9 @@ const WindowMenu = imports.ui.windowMenu; | ||||
| const PadOsd = imports.ui.padOsd; | ||||
| const EdgeDragAction = imports.ui.edgeDragAction; | ||||
| const CloseDialog = imports.ui.closeDialog; | ||||
| const SwipeTracker = imports.ui.swipeTracker; | ||||
| const SwitchMonitor = imports.ui.switchMonitor; | ||||
| const IBusManager = imports.misc.ibusManager; | ||||
| const WorkspaceAnimation = imports.ui.workspaceAnimation; | ||||
|  | ||||
| const { loadInterfaceXML } = imports.misc.fileUtils; | ||||
|  | ||||
| @@ -561,7 +561,6 @@ var WindowManager = class { | ||||
|         this._resizing = new Set(); | ||||
|         this._resizePending = new Set(); | ||||
|         this._destroying = new Set(); | ||||
|         this._movingWindow = null; | ||||
|  | ||||
|         this._dimmedWindows = []; | ||||
|  | ||||
| @@ -571,15 +570,6 @@ var WindowManager = class { | ||||
|  | ||||
|         this._isWorkspacePrepended = false; | ||||
|  | ||||
|         this._switchData = null; | ||||
|         this._shellwm.connect('kill-switch-workspace', shellwm => { | ||||
|             if (this._switchData) { | ||||
|                 if (this._switchData.inProgress) | ||||
|                     this._switchWorkspaceDone(shellwm); | ||||
|                 else if (!this._switchData.gestureActivated) | ||||
|                     this._finishWorkspaceSwitch(this._switchData); | ||||
|             } | ||||
|         }); | ||||
|         this._shellwm.connect('kill-window-effects', (shellwm, actor) => { | ||||
|             this._minimizeWindowDone(shellwm, actor); | ||||
|             this._mapWindowDone(shellwm, actor); | ||||
| @@ -601,7 +591,6 @@ var WindowManager = class { | ||||
|         this._shellwm.connect('confirm-display-change', this._confirmDisplayChange.bind(this)); | ||||
|         this._shellwm.connect('create-close-dialog', this._createCloseDialog.bind(this)); | ||||
|         this._shellwm.connect('create-inhibit-shortcuts-dialog', this._createInhibitShortcutsDialog.bind(this)); | ||||
|         global.display.connect('restacked', this._syncStacking.bind(this)); | ||||
|  | ||||
|         this._workspaceSwitcherPopup = null; | ||||
|         this._tilePreview = null; | ||||
| @@ -947,17 +936,10 @@ var WindowManager = class { | ||||
|         Main.overview.connect('showing', () => { | ||||
|             for (let i = 0; i < this._dimmedWindows.length; i++) | ||||
|                 this._undimWindow(this._dimmedWindows[i]); | ||||
|  | ||||
|             if (this._switchData) { | ||||
|                 if (this._switchData.gestureActivated) | ||||
|                     this._switchWorkspaceStop(); | ||||
|                 this._swipeTracker.enabled = false; | ||||
|             } | ||||
|         }); | ||||
|         Main.overview.connect('hiding', () => { | ||||
|             for (let i = 0; i < this._dimmedWindows.length; i++) | ||||
|                 this._dimWindow(this._dimmedWindows[i]); | ||||
|             this._swipeTracker.enabled = true; | ||||
|         }); | ||||
|  | ||||
|         this._windowMenuManager = new WindowMenu.WindowMenuManager(); | ||||
| @@ -968,13 +950,6 @@ var WindowManager = class { | ||||
|         global.workspace_manager.override_workspace_layout(Meta.DisplayCorner.TOPLEFT, | ||||
|                                                            false, -1, 1); | ||||
|  | ||||
|         let swipeTracker = new SwipeTracker.SwipeTracker(global.stage, | ||||
|             Shell.ActionMode.NORMAL, { allowDrag: false, allowScroll: false }); | ||||
|         swipeTracker.connect('begin', this._switchWorkspaceBegin.bind(this)); | ||||
|         swipeTracker.connect('update', this._switchWorkspaceUpdate.bind(this)); | ||||
|         swipeTracker.connect('end', this._switchWorkspaceEnd.bind(this)); | ||||
|         this._swipeTracker = swipeTracker; | ||||
|  | ||||
|         let appSwitchAction = new AppSwitchAction(); | ||||
|         appSwitchAction.connect('activated', this._switchApp.bind(this)); | ||||
|         global.stage.add_action(appSwitchAction); | ||||
| @@ -1006,6 +981,14 @@ var WindowManager = class { | ||||
|         global.display.connect('in-fullscreen-changed', updateUnfullscreenGesture); | ||||
|  | ||||
|         global.stage.add_action(topDragAction); | ||||
|  | ||||
|         this._workspaceAnimation = | ||||
|             new WorkspaceAnimation.WorkspaceAnimationController(); | ||||
|  | ||||
|         this._shellwm.connect('kill-switch-workspace', () => { | ||||
|             this._workspaceAnimation.cancelSwitchAnimation(); | ||||
|             this._switchWorkspaceDone(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _showPadOsd(display, device, settings, imagePath, editionMode, monitorIndex) { | ||||
| @@ -1128,8 +1111,7 @@ var WindowManager = class { | ||||
|     } | ||||
|  | ||||
|     _shouldAnimate() { | ||||
|         return !(Main.overview.visible || | ||||
|             (this._switchData && this._switchData.gestureActivated)); | ||||
|         return !(Main.overview.visible || this._workspaceAnimation.gestureActive); | ||||
|     } | ||||
|  | ||||
|     _shouldAnimateActor(actor, types) { | ||||
| @@ -1278,13 +1260,13 @@ var WindowManager = class { | ||||
|     } | ||||
|  | ||||
|     _sizeChangeWindow(shellwm, actor, whichChange, oldFrameRect, _oldBufferRect) { | ||||
|         let types = [Meta.WindowType.NORMAL]; | ||||
|         if (!this._shouldAnimateActor(actor, types)) { | ||||
|             shellwm.completed_size_change(actor); | ||||
|             return; | ||||
|         } | ||||
|         const types = [Meta.WindowType.NORMAL]; | ||||
|         const shouldAnimate = | ||||
|             this._shouldAnimateActor(actor, types) && | ||||
|             oldFrameRect.width > 0 && | ||||
|             oldFrameRect.height > 0; | ||||
|  | ||||
|         if (oldFrameRect.width > 0 && oldFrameRect.height > 0) | ||||
|         if (shouldAnimate) | ||||
|             this._prepareAnimationInfo(shellwm, actor, oldFrameRect, whichChange); | ||||
|         else | ||||
|             shellwm.completed_size_change(actor); | ||||
| @@ -1299,17 +1281,24 @@ var WindowManager = class { | ||||
|         actorClone.set_position(oldFrameRect.x, oldFrameRect.y); | ||||
|         actorClone.set_size(oldFrameRect.width, oldFrameRect.height); | ||||
|  | ||||
|         if (this._clearAnimationInfo(actor)) | ||||
|         actor.freeze(); | ||||
|  | ||||
|         if (this._clearAnimationInfo(actor)) { | ||||
|             log('Old animationInfo removed from actor %s'.format(actor)); | ||||
|             this._shellwm.completed_size_change(actor); | ||||
|         } | ||||
|  | ||||
|         let destroyId = actor.connect('destroy', () => { | ||||
|             this._clearAnimationInfo(actor); | ||||
|         }); | ||||
|  | ||||
|         this._resizePending.add(actor); | ||||
|         actor.__animationInfo = { clone: actorClone, | ||||
|                                   oldRect: oldFrameRect, | ||||
|                                   destroyId }; | ||||
|         actor.__animationInfo = { | ||||
|             clone: actorClone, | ||||
|             oldRect: oldFrameRect, | ||||
|             frozen: true, | ||||
|             destroyId, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     _sizeChangedWindow(shellwm, actor) { | ||||
| @@ -1362,13 +1351,17 @@ var WindowManager = class { | ||||
|         // Now unfreeze actor updates, to get it to the new size. | ||||
|         // It's important that we don't wait until the animation is completed to | ||||
|         // do this, otherwise our scale will be applied to the old texture size. | ||||
|         shellwm.completed_size_change(actor); | ||||
|         actor.thaw(); | ||||
|         actor.__animationInfo.frozen = false; | ||||
|     } | ||||
|  | ||||
|     _clearAnimationInfo(actor) { | ||||
|         if (actor.__animationInfo) { | ||||
|             actor.__animationInfo.clone.destroy(); | ||||
|             actor.disconnect(actor.__animationInfo.destroyId); | ||||
|             if (actor.__animationInfo.frozen) | ||||
|                 actor.thaw(); | ||||
|  | ||||
|             delete actor.__animationInfo; | ||||
|             return true; | ||||
|         } | ||||
| @@ -1383,10 +1376,13 @@ var WindowManager = class { | ||||
|             actor.translation_x = 0; | ||||
|             actor.translation_y = 0; | ||||
|             this._clearAnimationInfo(actor); | ||||
|             this._shellwm.completed_size_change(actor); | ||||
|         } | ||||
|  | ||||
|         if (this._resizePending.delete(actor)) | ||||
|         if (this._resizePending.delete(actor)) { | ||||
|             this._clearAnimationInfo(actor); | ||||
|             this._shellwm.completed_size_change(actor); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _hasAttachedDialogs(window, ignoreWindow) { | ||||
| @@ -1611,377 +1607,26 @@ var WindowManager = class { | ||||
|         return !(this._allowedKeybindings[binding.get_name()] & Main.actionMode); | ||||
|     } | ||||
|  | ||||
|     _syncStacking() { | ||||
|         if (this._switchData == null) | ||||
|             return; | ||||
|  | ||||
|         let windows = global.get_window_actors(); | ||||
|         let lastCurSibling = null; | ||||
|         let lastDirSibling = []; | ||||
|         for (let i = 0; i < windows.length; i++) { | ||||
|             if (windows[i].get_parent() == this._switchData.curGroup) { | ||||
|                 this._switchData.curGroup.set_child_above_sibling(windows[i], lastCurSibling); | ||||
|                 lastCurSibling = windows[i]; | ||||
|             } else { | ||||
|                 for (let dir of Object.values(Meta.MotionDirection)) { | ||||
|                     let info = this._switchData.surroundings[dir]; | ||||
|                     if (!info || windows[i].get_parent() != info.actor) | ||||
|                         continue; | ||||
|  | ||||
|                     let sibling = lastDirSibling[dir]; | ||||
|                     if (sibling == undefined) | ||||
|                         sibling = null; | ||||
|  | ||||
|                     info.actor.set_child_above_sibling(windows[i], sibling); | ||||
|                     lastDirSibling[dir] = windows[i]; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _getPositionForDirection(direction, fromWs, toWs) { | ||||
|         let xDest = 0, yDest = 0; | ||||
|  | ||||
|         let oldWsIsFullscreen = fromWs.list_windows().some(w => w.is_fullscreen()); | ||||
|         let newWsIsFullscreen = toWs.list_windows().some(w => w.is_fullscreen()); | ||||
|  | ||||
|         // We have to shift windows up or down by the height of the panel to prevent having a | ||||
|         // visible gap between the windows while switching workspaces. Since fullscreen windows | ||||
|         // hide the panel, they don't need to be shifted up or down. | ||||
|         let shiftHeight = Main.panel.height; | ||||
|  | ||||
|         if (direction == Meta.MotionDirection.UP || | ||||
|             direction == Meta.MotionDirection.UP_LEFT || | ||||
|             direction == Meta.MotionDirection.UP_RIGHT) | ||||
|             yDest = -global.screen_height + (oldWsIsFullscreen ? 0 : shiftHeight); | ||||
|         else if (direction == Meta.MotionDirection.DOWN || | ||||
|             direction == Meta.MotionDirection.DOWN_LEFT || | ||||
|             direction == Meta.MotionDirection.DOWN_RIGHT) | ||||
|             yDest = global.screen_height - (newWsIsFullscreen ? 0 : shiftHeight); | ||||
|  | ||||
|         if (direction == Meta.MotionDirection.LEFT || | ||||
|             direction == Meta.MotionDirection.UP_LEFT || | ||||
|             direction == Meta.MotionDirection.DOWN_LEFT) | ||||
|             xDest = -global.screen_width; | ||||
|         else if (direction == Meta.MotionDirection.RIGHT || | ||||
|                  direction == Meta.MotionDirection.UP_RIGHT || | ||||
|                  direction == Meta.MotionDirection.DOWN_RIGHT) | ||||
|             xDest = global.screen_width; | ||||
|  | ||||
|         return [xDest, yDest]; | ||||
|     } | ||||
|  | ||||
|     _prepareWorkspaceSwitch(from, to, direction) { | ||||
|         if (this._switchData) | ||||
|             return; | ||||
|  | ||||
|         let wgroup = global.window_group; | ||||
|         let windows = global.get_window_actors(); | ||||
|         let switchData = {}; | ||||
|  | ||||
|         this._switchData = switchData; | ||||
|         switchData.curGroup = new Clutter.Actor(); | ||||
|         switchData.movingWindowBin = new Clutter.Actor(); | ||||
|         switchData.windows = []; | ||||
|         switchData.surroundings = {}; | ||||
|         switchData.gestureActivated = false; | ||||
|         switchData.inProgress = false; | ||||
|  | ||||
|         switchData.container = new Clutter.Actor(); | ||||
|         switchData.container.add_actor(switchData.curGroup); | ||||
|  | ||||
|         wgroup.add_actor(switchData.movingWindowBin); | ||||
|         wgroup.add_actor(switchData.container); | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let curWs = workspaceManager.get_workspace_by_index(from); | ||||
|  | ||||
|         for (let dir of Object.values(Meta.MotionDirection)) { | ||||
|             let ws = null; | ||||
|  | ||||
|             if (to < 0) | ||||
|                 ws = curWs.get_neighbor(dir); | ||||
|             else if (dir == direction) | ||||
|                 ws = workspaceManager.get_workspace_by_index(to); | ||||
|  | ||||
|             if (ws == null || ws == curWs) { | ||||
|                 switchData.surroundings[dir] = null; | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             let [x, y] = this._getPositionForDirection(dir, curWs, ws); | ||||
|             let info = { | ||||
|                 index: ws.index(), | ||||
|                 actor: new Clutter.Actor(), | ||||
|                 xDest: x, | ||||
|                 yDest: y, | ||||
|             }; | ||||
|             switchData.surroundings[dir] = info; | ||||
|             switchData.container.add_actor(info.actor); | ||||
|             switchData.container.set_child_above_sibling(info.actor, null); | ||||
|  | ||||
|             info.actor.set_position(x, y); | ||||
|         } | ||||
|  | ||||
|         wgroup.set_child_above_sibling(switchData.movingWindowBin, null); | ||||
|  | ||||
|         for (let i = 0; i < windows.length; i++) { | ||||
|             let actor = windows[i]; | ||||
|             let window = actor.get_meta_window(); | ||||
|  | ||||
|             if (!window.showing_on_its_workspace()) | ||||
|                 continue; | ||||
|  | ||||
|             if (window.is_on_all_workspaces()) | ||||
|                 continue; | ||||
|  | ||||
|             let record = { window: actor, | ||||
|                            parent: actor.get_parent() }; | ||||
|  | ||||
|             if (this._movingWindow && window == this._movingWindow) { | ||||
|                 record.parent.remove_child(actor); | ||||
|                 switchData.movingWindow = record; | ||||
|                 switchData.windows.push(switchData.movingWindow); | ||||
|                 switchData.movingWindowBin.add_child(actor); | ||||
|             } else if (window.get_workspace().index() == from) { | ||||
|                 record.parent.remove_child(actor); | ||||
|                 switchData.windows.push(record); | ||||
|                 switchData.curGroup.add_child(actor); | ||||
|             } else { | ||||
|                 let visible = false; | ||||
|                 for (let dir of Object.values(Meta.MotionDirection)) { | ||||
|                     let info = switchData.surroundings[dir]; | ||||
|  | ||||
|                     if (!info || info.index != window.get_workspace().index()) | ||||
|                         continue; | ||||
|  | ||||
|                     record.parent.remove_child(actor); | ||||
|                     switchData.windows.push(record); | ||||
|                     info.actor.add_child(actor); | ||||
|                     visible = true; | ||||
|                     break; | ||||
|                 } | ||||
|  | ||||
|                 actor.visible = visible; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         for (let i = 0; i < switchData.windows.length; i++) { | ||||
|             let w = switchData.windows[i]; | ||||
|  | ||||
|             w.windowDestroyId = w.window.connect('destroy', () => { | ||||
|                 switchData.windows.splice(switchData.windows.indexOf(w), 1); | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _finishWorkspaceSwitch(switchData) { | ||||
|         this._switchData = null; | ||||
|  | ||||
|         for (let i = 0; i < switchData.windows.length; i++) { | ||||
|             let w = switchData.windows[i]; | ||||
|  | ||||
|             w.window.disconnect(w.windowDestroyId); | ||||
|             w.window.get_parent().remove_child(w.window); | ||||
|             w.parent.add_child(w.window); | ||||
|  | ||||
|             if (w.window.get_meta_window().get_workspace() != | ||||
|                 global.workspace_manager.get_active_workspace()) | ||||
|                 w.window.hide(); | ||||
|         } | ||||
|         switchData.container.destroy(); | ||||
|         switchData.movingWindowBin.destroy(); | ||||
|  | ||||
|         this._movingWindow = null; | ||||
|     } | ||||
|  | ||||
|     _switchWorkspace(shellwm, from, to, direction) { | ||||
|         if (!Main.sessionMode.hasWorkspaces || !this._shouldAnimate()) { | ||||
|             shellwm.completed_switch_workspace(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._prepareWorkspaceSwitch(from, to, direction); | ||||
|         this._switchData.inProgress = true; | ||||
|         this._switchInProgress = true; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let fromWs = workspaceManager.get_workspace_by_index(from); | ||||
|         let toWs = workspaceManager.get_workspace_by_index(to); | ||||
|  | ||||
|         let [xDest, yDest] = this._getPositionForDirection(direction, fromWs, toWs); | ||||
|  | ||||
|         /* @direction is the direction that the "camera" moves, so the | ||||
|          * screen contents have to move one screen's worth in the | ||||
|          * opposite direction. | ||||
|          */ | ||||
|         xDest = -xDest; | ||||
|         yDest = -yDest; | ||||
|  | ||||
|         this._switchData.container.ease({ | ||||
|             x: xDest, | ||||
|             y: yDest, | ||||
|             duration: WINDOW_ANIMATION_TIME, | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             onComplete: () => this._switchWorkspaceDone(shellwm), | ||||
|         this._workspaceAnimation.animateSwitch(from, to, direction, () => { | ||||
|             this._shellwm.completed_switch_workspace(); | ||||
|             this._switchInProgress = false; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceDone(shellwm) { | ||||
|         this._finishWorkspaceSwitch(this._switchData); | ||||
|         shellwm.completed_switch_workspace(); | ||||
|     } | ||||
|  | ||||
|     _directionForProgress(progress) { | ||||
|         if (global.workspace_manager.layout_rows === -1) { | ||||
|             return progress > 0 | ||||
|                 ? Meta.MotionDirection.DOWN | ||||
|                 : Meta.MotionDirection.UP; | ||||
|         } else if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) { | ||||
|             return progress > 0 | ||||
|                 ? Meta.MotionDirection.LEFT | ||||
|                 : Meta.MotionDirection.RIGHT; | ||||
|         } else { | ||||
|             return progress > 0 | ||||
|                 ? Meta.MotionDirection.RIGHT | ||||
|                 : Meta.MotionDirection.LEFT; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _getProgressRange() { | ||||
|         if (!this._switchData) | ||||
|             return [0, 0]; | ||||
|  | ||||
|         let lower = 0; | ||||
|         let upper = 0; | ||||
|  | ||||
|         let horiz = global.workspace_manager.layout_rows !== -1; | ||||
|         let baseDistance; | ||||
|         if (horiz) | ||||
|             baseDistance = global.screen_width; | ||||
|         else | ||||
|             baseDistance = global.screen_height; | ||||
|  | ||||
|         let direction = this._directionForProgress(-1); | ||||
|         let info = this._switchData.surroundings[direction]; | ||||
|         if (info !== null) { | ||||
|             let distance = horiz ? info.xDest : info.yDest; | ||||
|             lower = -Math.abs(distance) / baseDistance; | ||||
|         } | ||||
|  | ||||
|         direction = this._directionForProgress(1); | ||||
|         info = this._switchData.surroundings[direction]; | ||||
|         if (info !== null) { | ||||
|             let distance = horiz ? info.xDest : info.yDest; | ||||
|             upper = Math.abs(distance) / baseDistance; | ||||
|         } | ||||
|  | ||||
|         return [lower, upper]; | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceBegin(tracker, monitor) { | ||||
|         if (Meta.prefs_get_workspaces_only_on_primary() && | ||||
|             monitor !== Main.layoutManager.primaryIndex) | ||||
|     _switchWorkspaceDone() { | ||||
|         if (!this._switchInProgress) | ||||
|             return; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let horiz = workspaceManager.layout_rows !== -1; | ||||
|         tracker.orientation = horiz | ||||
|             ? Clutter.Orientation.HORIZONTAL | ||||
|             : Clutter.Orientation.VERTICAL; | ||||
|  | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|  | ||||
|         let baseDistance; | ||||
|         if (horiz) | ||||
|             baseDistance = global.screen_width; | ||||
|         else | ||||
|             baseDistance = global.screen_height; | ||||
|  | ||||
|         let progress; | ||||
|         if (this._switchData && this._switchData.gestureActivated) { | ||||
|             this._switchData.container.remove_all_transitions(); | ||||
|             if (!horiz) | ||||
|                 progress = -this._switchData.container.y / baseDistance; | ||||
|             else if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) | ||||
|                 progress = this._switchData.container.x / baseDistance; | ||||
|             else | ||||
|                 progress = -this._switchData.container.x / baseDistance; | ||||
|         } else { | ||||
|             this._prepareWorkspaceSwitch(activeWorkspace.index(), -1); | ||||
|             progress = 0; | ||||
|         } | ||||
|  | ||||
|         let points = []; | ||||
|         let [lower, upper] = this._getProgressRange(); | ||||
|  | ||||
|         if (lower !== 0) | ||||
|             points.push(lower); | ||||
|  | ||||
|         points.push(0); | ||||
|  | ||||
|         if (upper !== 0) | ||||
|             points.push(upper); | ||||
|  | ||||
|         tracker.confirmSwipe(baseDistance, points, progress, 0); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceUpdate(tracker, progress) { | ||||
|         if (!this._switchData) | ||||
|             return; | ||||
|  | ||||
|         let direction = this._directionForProgress(progress); | ||||
|         let info = this._switchData.surroundings[direction]; | ||||
|         let xPos = 0; | ||||
|         let yPos = 0; | ||||
|         if (info) { | ||||
|             if (global.workspace_manager.layout_rows === -1) | ||||
|                 yPos = -Math.round(progress * global.screen_height); | ||||
|             else if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) | ||||
|                 xPos = Math.round(progress * global.screen_width); | ||||
|             else | ||||
|                 xPos = -Math.round(progress * global.screen_width); | ||||
|         } | ||||
|  | ||||
|         this._switchData.container.set_position(xPos, yPos); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceEnd(tracker, duration, endProgress) { | ||||
|         if (!this._switchData) | ||||
|             return; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|         let newWs = activeWorkspace; | ||||
|         let xDest = 0; | ||||
|         let yDest = 0; | ||||
|         if (endProgress !== 0) { | ||||
|             let direction = this._directionForProgress(endProgress); | ||||
|             newWs = activeWorkspace.get_neighbor(direction); | ||||
|             xDest = -this._switchData.surroundings[direction].xDest; | ||||
|             yDest = -this._switchData.surroundings[direction].yDest; | ||||
|         } | ||||
|  | ||||
|         let switchData = this._switchData; | ||||
|         switchData.gestureActivated = true; | ||||
|  | ||||
|         this._switchData.container.ease({ | ||||
|             x: xDest, | ||||
|             y: yDest, | ||||
|             duration, | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             onComplete: () => { | ||||
|                 if (newWs !== activeWorkspace) | ||||
|                     this.actionMoveWorkspace(newWs); | ||||
|                 this._finishWorkspaceSwitch(switchData); | ||||
|             }, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceStop() { | ||||
|         this._switchData.container.x = 0; | ||||
|         this._switchData.container.y = 0; | ||||
|         this._finishWorkspaceSwitch(this._switchData); | ||||
|         this._shellwm.completed_switch_workspace(); | ||||
|         this._switchInProgress = false; | ||||
|     } | ||||
|  | ||||
|     _showTilePreview(shellwm, window, tileRect, monitorIndex) { | ||||
| @@ -2101,8 +1746,16 @@ var WindowManager = class { | ||||
|             newWs = workspaceManager.get_workspace_by_index(workspaceManager.n_workspaces - 1); | ||||
|         } else if (isNaN(target)) { | ||||
|             // Prepend a new workspace dynamically | ||||
|             if (workspaceManager.get_active_workspace_index() == 0 && | ||||
|                 action == 'move' && target == 'up' && this._isWorkspacePrepended == false) { | ||||
|             let prependTarget; | ||||
|             if (vertical) | ||||
|                 prependTarget = 'up'; | ||||
|             else if (rtl) | ||||
|                 prependTarget = 'right'; | ||||
|             else | ||||
|                 prependTarget = 'left'; | ||||
|             if (workspaceManager.get_active_workspace_index() === 0 && | ||||
|                 action === 'move' && target === prependTarget && | ||||
|                 this._isWorkspacePrepended === false) { | ||||
|                 this.insertWorkspace(0); | ||||
|                 this._isWorkspacePrepended = true; | ||||
|             } | ||||
| @@ -2163,10 +1816,7 @@ var WindowManager = class { | ||||
|         if (!Main.sessionMode.hasWorkspaces) | ||||
|             return; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|  | ||||
|         if (activeWorkspace != workspace) | ||||
|         if (!workspace.active) | ||||
|             workspace.activate(global.get_current_time()); | ||||
|     } | ||||
|  | ||||
| @@ -2174,14 +1824,11 @@ var WindowManager = class { | ||||
|         if (!Main.sessionMode.hasWorkspaces) | ||||
|             return; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|  | ||||
|         if (activeWorkspace != workspace) { | ||||
|         if (!workspace.active) { | ||||
|             // This won't have any effect for "always sticky" windows | ||||
|             // (like desktop windows or docks) | ||||
|  | ||||
|             this._movingWindow = window; | ||||
|             this._workspaceAnimation.movingWindow = window; | ||||
|             window.change_workspace(workspace); | ||||
|  | ||||
|             global.display.clear_mouse_mode(); | ||||
|   | ||||
							
								
								
									
										771
									
								
								js/ui/windowPreview.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										771
									
								
								js/ui/windowPreview.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,771 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported WindowPreview */ | ||||
|  | ||||
| const { Atk, Clutter, GLib, GObject, | ||||
|         Graphene, Meta, Pango, Shell, St } = imports.gi; | ||||
|  | ||||
| const DND = imports.ui.dnd; | ||||
|  | ||||
| var WINDOW_DND_SIZE = 256; | ||||
|  | ||||
| var WINDOW_OVERLAY_IDLE_HIDE_TIMEOUT = 750; | ||||
| var WINDOW_OVERLAY_FADE_TIME = 200; | ||||
|  | ||||
| var DRAGGING_WINDOW_OPACITY = 100; | ||||
|  | ||||
| var WindowPreviewLayout = GObject.registerClass({ | ||||
|     Properties: { | ||||
|         'bounding-box': GObject.ParamSpec.boxed( | ||||
|             'bounding-box', 'Bounding box', 'Bounding box', | ||||
|             GObject.ParamFlags.READABLE, | ||||
|             Clutter.ActorBox.$gtype), | ||||
|     }, | ||||
| }, class WindowPreviewLayout extends Clutter.LayoutManager { | ||||
|     _init() { | ||||
|         super._init(); | ||||
|  | ||||
|         this._container = null; | ||||
|         this._boundingBox = new Clutter.ActorBox(); | ||||
|         this._windows = new Map(); | ||||
|     } | ||||
|  | ||||
|     _layoutChanged() { | ||||
|         let frameRect; | ||||
|  | ||||
|         for (const windowInfo of this._windows.values()) { | ||||
|             const frame = windowInfo.metaWindow.get_frame_rect(); | ||||
|             frameRect = frameRect ? frameRect.union(frame) : frame; | ||||
|         } | ||||
|  | ||||
|         if (!frameRect) | ||||
|             frameRect = new Meta.Rectangle(); | ||||
|  | ||||
|         const oldBox = this._boundingBox.copy(); | ||||
|         this._boundingBox.set_origin(frameRect.x, frameRect.y); | ||||
|         this._boundingBox.set_size(frameRect.width, frameRect.height); | ||||
|  | ||||
|         if (!this._boundingBox.equal(oldBox)) | ||||
|             this.notify('bounding-box'); | ||||
|  | ||||
|         // Always call layout_changed(), a size or position change of an | ||||
|         // attached dialog might not affect the boundingBox | ||||
|         this.layout_changed(); | ||||
|     } | ||||
|  | ||||
|     vfunc_set_container(container) { | ||||
|         this._container = container; | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_height(_container, _forWidth) { | ||||
|         return [0, this._boundingBox.get_height()]; | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_width(_container, _forHeight) { | ||||
|         return [0, this._boundingBox.get_width()]; | ||||
|     } | ||||
|  | ||||
|     vfunc_allocate(container, box) { | ||||
|         // If the scale isn't 1, we weren't allocated our preferred size | ||||
|         // and have to scale the children allocations accordingly. | ||||
|         const scaleX = this._boundingBox.get_width() > 0 | ||||
|             ? box.get_width() / this._boundingBox.get_width() | ||||
|             : 1; | ||||
|         const scaleY = this._boundingBox.get_height() > 0 | ||||
|             ? box.get_height() / this._boundingBox.get_height() | ||||
|             : 1; | ||||
|  | ||||
|         const childBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         for (const child of container) { | ||||
|             if (!child.visible) | ||||
|                 continue; | ||||
|  | ||||
|             const windowInfo = this._windows.get(child); | ||||
|             if (windowInfo) { | ||||
|                 const bufferRect = windowInfo.metaWindow.get_buffer_rect(); | ||||
|                 childBox.set_origin( | ||||
|                     bufferRect.x - this._boundingBox.x1, | ||||
|                     bufferRect.y - this._boundingBox.y1); | ||||
|  | ||||
|                 const [, , natWidth, natHeight] = child.get_preferred_size(); | ||||
|                 childBox.set_size(natWidth, natHeight); | ||||
|  | ||||
|                 childBox.x1 *= scaleX; | ||||
|                 childBox.x2 *= scaleX; | ||||
|                 childBox.y1 *= scaleY; | ||||
|                 childBox.y2 *= scaleY; | ||||
|  | ||||
|                 child.allocate(childBox); | ||||
|             } else { | ||||
|                 child.allocate_preferred_size(0, 0); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * addWindow: | ||||
|      * @param {Meta.Window} window: the MetaWindow instance | ||||
|      * | ||||
|      * Creates a ClutterActor drawing the texture of @window and adds it | ||||
|      * to the container. If @window is already part of the preview, this | ||||
|      * function will do nothing. | ||||
|      * | ||||
|      * @returns {Clutter.Actor} The newly created actor drawing @window | ||||
|      */ | ||||
|     addWindow(window) { | ||||
|         const index = [...this._windows.values()].findIndex(info => | ||||
|             info.metaWindow === window); | ||||
|  | ||||
|         if (index !== -1) | ||||
|             return null; | ||||
|  | ||||
|         const windowActor = window.get_compositor_private(); | ||||
|         const actor = new Clutter.Clone({ source: windowActor }); | ||||
|  | ||||
|         this._windows.set(actor, { | ||||
|             metaWindow: window, | ||||
|             windowActor, | ||||
|             sizeChangedId: window.connect('size-changed', () => | ||||
|                 this._layoutChanged()), | ||||
|             positionChangedId: window.connect('position-changed', () => | ||||
|                 this._layoutChanged()), | ||||
|             windowActorDestroyId: windowActor.connect('destroy', () => | ||||
|                 actor.destroy()), | ||||
|             destroyId: actor.connect('destroy', () => | ||||
|                 this.removeWindow(window)), | ||||
|         }); | ||||
|  | ||||
|         this._container.add_child(actor); | ||||
|  | ||||
|         this._layoutChanged(); | ||||
|  | ||||
|         return actor; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * removeWindow: | ||||
|      * @param {Meta.Window} window: the window to remove from the preview | ||||
|      * | ||||
|      * Removes a MetaWindow @window from the preview which has been added | ||||
|      * previously using addWindow(). If @window is not part of preview, | ||||
|      * this function will do nothing. | ||||
|      */ | ||||
|     removeWindow(window) { | ||||
|         const entry = [...this._windows].find( | ||||
|             ([, i]) => i.metaWindow === window); | ||||
|  | ||||
|         if (!entry) | ||||
|             return; | ||||
|  | ||||
|         const [actor, windowInfo] = entry; | ||||
|  | ||||
|         windowInfo.metaWindow.disconnect(windowInfo.sizeChangedId); | ||||
|         windowInfo.metaWindow.disconnect(windowInfo.positionChangedId); | ||||
|         windowInfo.windowActor.disconnect(windowInfo.windowActorDestroyId); | ||||
|         actor.disconnect(windowInfo.destroyId); | ||||
|  | ||||
|         this._windows.delete(actor); | ||||
|         this._container.remove_child(actor); | ||||
|  | ||||
|         this._layoutChanged(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * getWindows: | ||||
|      * | ||||
|      * Gets an array of all MetaWindows that were added to the layout | ||||
|      * using addWindow(), ordered by the insertion order. | ||||
|      * | ||||
|      * @returns {Array} An array including all windows | ||||
|      */ | ||||
|     getWindows() { | ||||
|         return [...this._windows.values()].map(i => i.metaWindow); | ||||
|     } | ||||
|  | ||||
|     // eslint-disable-next-line camelcase | ||||
|     get bounding_box() { | ||||
|         return this._boundingBox; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var WindowPreview = GObject.registerClass({ | ||||
|     Properties: { | ||||
|         'overlay-enabled': GObject.ParamSpec.boolean( | ||||
|             'overlay-enabled', 'overlay-enabled', 'overlay-enabled', | ||||
|             GObject.ParamFlags.READWRITE, | ||||
|             true), | ||||
|     }, | ||||
|     Signals: { | ||||
|         'drag-begin': {}, | ||||
|         'drag-cancelled': {}, | ||||
|         'drag-end': {}, | ||||
|         'selected': { param_types: [GObject.TYPE_UINT] }, | ||||
|         'show-chrome': {}, | ||||
|         'size-changed': {}, | ||||
|     }, | ||||
| }, class WindowPreview extends St.Widget { | ||||
|     _init(metaWindow, workspace) { | ||||
|         this.metaWindow = metaWindow; | ||||
|         this.metaWindow._delegate = this; | ||||
|         this._windowActor = metaWindow.get_compositor_private(); | ||||
|         this._workspace = workspace; | ||||
|  | ||||
|         super._init({ | ||||
|             reactive: true, | ||||
|             can_focus: true, | ||||
|             accessible_role: Atk.Role.PUSH_BUTTON, | ||||
|             offscreen_redirect: Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY, | ||||
|         }); | ||||
|  | ||||
|         this._windowContainer = new Clutter.Actor(); | ||||
|         // gjs currently can't handle setting an actors layout manager during | ||||
|         // the initialization of the actor if that layout manager keeps track | ||||
|         // of its container, so set the layout manager after creating the | ||||
|         // container | ||||
|         this._windowContainer.layout_manager = new WindowPreviewLayout(); | ||||
|         this.add_child(this._windowContainer); | ||||
|  | ||||
|         this._addWindow(metaWindow); | ||||
|  | ||||
|         this._delegate = this; | ||||
|  | ||||
|         this._stackAbove = null; | ||||
|  | ||||
|         this._windowContainer.layout_manager.connect( | ||||
|             'notify::bounding-box', layout => { | ||||
|                 // A bounding box of 0x0 means all windows were removed | ||||
|                 if (layout.bounding_box.get_area() > 0) | ||||
|                     this.emit('size-changed'); | ||||
|             }); | ||||
|  | ||||
|         this._windowDestroyId = | ||||
|             this._windowActor.connect('destroy', () => this.destroy()); | ||||
|  | ||||
|         this._updateAttachedDialogs(); | ||||
|  | ||||
|         let clickAction = new Clutter.ClickAction(); | ||||
|         clickAction.connect('clicked', () => this._activate()); | ||||
|         clickAction.connect('long-press', this._onLongPress.bind(this)); | ||||
|         this.add_action(clickAction); | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|  | ||||
|         this._draggable = DND.makeDraggable(this, | ||||
|                                             { restoreOnSuccess: true, | ||||
|                                               manualMode: true, | ||||
|                                               dragActorMaxSize: WINDOW_DND_SIZE, | ||||
|                                               dragActorOpacity: DRAGGING_WINDOW_OPACITY }); | ||||
|         this._draggable.connect('drag-begin', this._onDragBegin.bind(this)); | ||||
|         this._draggable.connect('drag-cancelled', this._onDragCancelled.bind(this)); | ||||
|         this._draggable.connect('drag-end', this._onDragEnd.bind(this)); | ||||
|         this.inDrag = false; | ||||
|  | ||||
|         this._selected = false; | ||||
|         this._overlayEnabled = true; | ||||
|         this._closeRequested = false; | ||||
|         this._idleHideOverlayId = 0; | ||||
|  | ||||
|         this._border = new St.Widget({ | ||||
|             visible: false, | ||||
|             style_class: 'window-clone-border', | ||||
|         }); | ||||
|         this._borderConstraint = new Clutter.BindConstraint({ | ||||
|             source: this._windowContainer, | ||||
|             coordinate: Clutter.BindCoordinate.SIZE, | ||||
|         }); | ||||
|         this._border.add_constraint(this._borderConstraint); | ||||
|         this._border.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._windowContainer, | ||||
|             align_axis: Clutter.AlignAxis.BOTH, | ||||
|             factor: 0.5, | ||||
|         })); | ||||
|         this._borderCenter = new Clutter.Actor(); | ||||
|         this._border.bind_property('visible', this._borderCenter, 'visible', | ||||
|             GObject.BindingFlags.SYNC_CREATE); | ||||
|         this._borderCenterConstraint = new Clutter.BindConstraint({ | ||||
|             source: this._windowContainer, | ||||
|             coordinate: Clutter.BindCoordinate.SIZE, | ||||
|         }); | ||||
|         this._borderCenter.add_constraint(this._borderCenterConstraint); | ||||
|         this._borderCenter.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._windowContainer, | ||||
|             align_axis: Clutter.AlignAxis.BOTH, | ||||
|             factor: 0.5, | ||||
|         })); | ||||
|         this._border.connect('style-changed', | ||||
|             this._onBorderStyleChanged.bind(this)); | ||||
|  | ||||
|         this._title = new St.Label({ | ||||
|             visible: false, | ||||
|             style_class: 'window-caption', | ||||
|             text: this._getCaption(), | ||||
|             reactive: true, | ||||
|         }); | ||||
|         this._title.add_constraint(new Clutter.BindConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             coordinate: Clutter.BindCoordinate.POSITION, | ||||
|         })); | ||||
|         this._title.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             align_axis: Clutter.AlignAxis.X_AXIS, | ||||
|             factor: 0.5, | ||||
|         })); | ||||
|         this._title.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             align_axis: Clutter.AlignAxis.Y_AXIS, | ||||
|             pivot_point: new Graphene.Point({ x: -1, y: 0.5 }), | ||||
|             factor: 1, | ||||
|         })); | ||||
|         this._title.clutter_text.ellipsize = Pango.EllipsizeMode.END; | ||||
|         this.label_actor = this._title; | ||||
|         this._updateCaptionId = this.metaWindow.connect('notify::title', () => { | ||||
|             this._title.text = this._getCaption(); | ||||
|         }); | ||||
|  | ||||
|         const layout = Meta.prefs_get_button_layout(); | ||||
|         this._closeButtonSide = | ||||
|             layout.left_buttons.includes(Meta.ButtonFunction.CLOSE) | ||||
|                 ? St.Side.LEFT : St.Side.RIGHT; | ||||
|  | ||||
|         this._closeButton = new St.Button({ | ||||
|             visible: false, | ||||
|             style_class: 'window-close', | ||||
|             child: new St.Icon({ icon_name: 'window-close-symbolic' }), | ||||
|         }); | ||||
|         this._closeButton.add_constraint(new Clutter.BindConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             coordinate: Clutter.BindCoordinate.POSITION, | ||||
|         })); | ||||
|         this._closeButton.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             align_axis: Clutter.AlignAxis.X_AXIS, | ||||
|             pivot_point: new Graphene.Point({ x: 0.5, y: -1 }), | ||||
|             factor: this._closeButtonSide === St.Side.LEFT ? 0 : 1, | ||||
|         })); | ||||
|         this._closeButton.add_constraint(new Clutter.AlignConstraint({ | ||||
|             source: this._borderCenter, | ||||
|             align_axis: Clutter.AlignAxis.Y_AXIS, | ||||
|             pivot_point: new Graphene.Point({ x: -1, y: 0.5 }), | ||||
|             factor: 0, | ||||
|         })); | ||||
|         this._closeButton.connect('clicked', () => this._deleteAll()); | ||||
|  | ||||
|         this.add_child(this._borderCenter); | ||||
|         this.add_child(this._border); | ||||
|         this.add_child(this._title); | ||||
|         this.add_child(this._closeButton); | ||||
|  | ||||
|         this.connect('notify::realized', () => { | ||||
|             if (!this.realized) | ||||
|                 return; | ||||
|  | ||||
|             this._border.ensure_style(); | ||||
|             this._title.ensure_style(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_width(forHeight) { | ||||
|         const themeNode = this.get_theme_node(); | ||||
|  | ||||
|         // Only include window previews in size request, not chrome | ||||
|         const [minWidth, natWidth] = | ||||
|             this._windowContainer.get_preferred_width( | ||||
|                 themeNode.adjust_for_height(forHeight)); | ||||
|  | ||||
|         return themeNode.adjust_preferred_width(minWidth, natWidth); | ||||
|     } | ||||
|  | ||||
|     vfunc_get_preferred_height(forWidth) { | ||||
|         const themeNode = this.get_theme_node(); | ||||
|         const [minHeight, natHeight] = | ||||
|             this._windowContainer.get_preferred_height( | ||||
|                 themeNode.adjust_for_width(forWidth)); | ||||
|  | ||||
|         return themeNode.adjust_preferred_height(minHeight, natHeight); | ||||
|     } | ||||
|  | ||||
|     vfunc_allocate(box) { | ||||
|         this.set_allocation(box); | ||||
|  | ||||
|         for (const child of this) | ||||
|             child.allocate_available_size(0, 0, box.get_width(), box.get_height()); | ||||
|     } | ||||
|  | ||||
|     _onBorderStyleChanged() { | ||||
|         let borderNode = this._border.get_theme_node(); | ||||
|         this._borderSize = borderNode.get_border_width(St.Side.TOP); | ||||
|  | ||||
|         // Increase the size of the border actor so the border outlines | ||||
|         // the bounding box | ||||
|         this._borderConstraint.offset = this._borderSize * 2; | ||||
|         this._borderCenterConstraint.offset = this._borderSize; | ||||
|     } | ||||
|  | ||||
|     _windowCanClose() { | ||||
|         return this.metaWindow.can_close() && | ||||
|                !this._hasAttachedDialogs(); | ||||
|     } | ||||
|  | ||||
|     _getCaption() { | ||||
|         if (this.metaWindow.title) | ||||
|             return this.metaWindow.title; | ||||
|  | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         let app = tracker.get_window_app(this.metaWindow); | ||||
|         return app.get_name(); | ||||
|     } | ||||
|  | ||||
|     chromeHeights() { | ||||
|         const [, closeButtonHeight] = this._closeButton.get_preferred_height(-1); | ||||
|         const [, titleHeight] = this._title.get_preferred_height(-1); | ||||
|  | ||||
|         const topOversize = (this._borderSize / 2) + (closeButtonHeight / 2); | ||||
|         const bottomOversize = Math.max( | ||||
|             this._borderSize, | ||||
|             (titleHeight / 2) + (this._borderSize / 2)); | ||||
|  | ||||
|         return [topOversize, bottomOversize]; | ||||
|     } | ||||
|  | ||||
|     chromeWidths() { | ||||
|         const [, closeButtonWidth] = this._closeButton.get_preferred_width(-1); | ||||
|  | ||||
|         const leftOversize = this._closeButtonSide === St.Side.LEFT | ||||
|             ? (this._borderSize / 2) + (closeButtonWidth / 2) | ||||
|             : this._borderSize; | ||||
|         const rightOversize = this._closeButtonSide === St.Side.LEFT | ||||
|             ? this._borderSize | ||||
|             : (this._borderSize / 2) + (closeButtonWidth / 2); | ||||
|  | ||||
|         return [leftOversize, rightOversize]; | ||||
|     } | ||||
|  | ||||
|     showOverlay(animate) { | ||||
|         if (!this._overlayEnabled) | ||||
|             return; | ||||
|  | ||||
|         const ongoingTransition = this._border.get_transition('opacity'); | ||||
|  | ||||
|         // Don't do anything if we're fully visible already | ||||
|         if (this._border.visible && !ongoingTransition) | ||||
|             return; | ||||
|  | ||||
|         // If we're supposed to animate and an animation in our direction | ||||
|         // is already happening, let that one continue | ||||
|         if (animate && | ||||
|             ongoingTransition && | ||||
|             ongoingTransition.get_interval().peek_final_value() === 255) | ||||
|             return; | ||||
|  | ||||
|         const toShow = this._windowCanClose() | ||||
|             ? [this._border, this._title, this._closeButton] | ||||
|             : [this._border, this._title]; | ||||
|  | ||||
|         toShow.forEach(a => { | ||||
|             a.opacity = 0; | ||||
|             a.show(); | ||||
|             a.ease({ | ||||
|                 opacity: 255, | ||||
|                 duration: animate ? WINDOW_OVERLAY_FADE_TIME : 0, | ||||
|                 mode: Clutter.AnimationMode.EASE_OUT_QUAD, | ||||
|             }); | ||||
|         }); | ||||
|  | ||||
|         this.emit('show-chrome'); | ||||
|     } | ||||
|  | ||||
|     hideOverlay(animate) { | ||||
|         const ongoingTransition = this._border.get_transition('opacity'); | ||||
|  | ||||
|         // Don't do anything if we're fully hidden already | ||||
|         if (!this._border.visible && !ongoingTransition) | ||||
|             return; | ||||
|  | ||||
|         // If we're supposed to animate and an animation in our direction | ||||
|         // is already happening, let that one continue | ||||
|         if (animate && | ||||
|             ongoingTransition && | ||||
|             ongoingTransition.get_interval().peek_final_value() === 0) | ||||
|             return; | ||||
|  | ||||
|         [this._border, this._title, this._closeButton].forEach(a => { | ||||
|             a.opacity = 255; | ||||
|             a.ease({ | ||||
|                 opacity: 0, | ||||
|                 duration: animate ? WINDOW_OVERLAY_FADE_TIME : 0, | ||||
|                 mode: Clutter.AnimationMode.EASE_OUT_QUAD, | ||||
|                 onComplete: () => a.hide(), | ||||
|             }); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _addWindow(metaWindow) { | ||||
|         const clone = this._windowContainer.layout_manager.addWindow(metaWindow); | ||||
|  | ||||
|         // We expect this to be used for all interaction rather than | ||||
|         // the ClutterClone; as the former is reactive and the latter | ||||
|         // is not, this just works for most cases. However, for DND all | ||||
|         // actors are picked, so DND operations would operate on the clone. | ||||
|         // To avoid this, we hide it from pick. | ||||
|         Shell.util_set_hidden_from_pick(clone, true); | ||||
|     } | ||||
|  | ||||
|     vfunc_has_overlaps() { | ||||
|         return this._hasAttachedDialogs(); | ||||
|     } | ||||
|  | ||||
|     _deleteAll() { | ||||
|         const windows = this._windowContainer.layout_manager.getWindows(); | ||||
|  | ||||
|         // Delete all windows, starting from the bottom-most (most-modal) one | ||||
|         for (const window of windows.reverse()) | ||||
|             window.delete(global.get_current_time()); | ||||
|  | ||||
|         this._closeRequested = true; | ||||
|     } | ||||
|  | ||||
|     addDialog(win) { | ||||
|         let parent = win.get_transient_for(); | ||||
|         while (parent.is_attached_dialog()) | ||||
|             parent = parent.get_transient_for(); | ||||
|  | ||||
|         // Display dialog if it is attached to our metaWindow | ||||
|         if (win.is_attached_dialog() && parent == this.metaWindow) | ||||
|             this._addWindow(win); | ||||
|  | ||||
|         // The dialog popped up after the user tried to close the window, | ||||
|         // assume it's a close confirmation and leave the overview | ||||
|         if (this._closeRequested) | ||||
|             this._activate(); | ||||
|     } | ||||
|  | ||||
|     _hasAttachedDialogs() { | ||||
|         return this._windowContainer.layout_manager.getWindows().length > 1; | ||||
|     } | ||||
|  | ||||
|     _updateAttachedDialogs() { | ||||
|         let iter = win => { | ||||
|             let actor = win.get_compositor_private(); | ||||
|  | ||||
|             if (!actor) | ||||
|                 return false; | ||||
|             if (!win.is_attached_dialog()) | ||||
|                 return false; | ||||
|  | ||||
|             this._addWindow(win); | ||||
|             win.foreach_transient(iter); | ||||
|             return true; | ||||
|         }; | ||||
|         this.metaWindow.foreach_transient(iter); | ||||
|     } | ||||
|  | ||||
|     get boundingBox() { | ||||
|         const box = this._windowContainer.layout_manager.bounding_box; | ||||
|  | ||||
|         return { | ||||
|             x: box.x1, | ||||
|             y: box.y1, | ||||
|             width: box.get_width(), | ||||
|             height: box.get_height(), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     get windowCenter() { | ||||
|         const box = this._windowContainer.layout_manager.bounding_box; | ||||
|  | ||||
|         return new Graphene.Point({ | ||||
|             x: box.get_x() + box.get_width() / 2, | ||||
|             y: box.get_y() + box.get_height() / 2, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     // eslint-disable-next-line camelcase | ||||
|     get overlay_enabled() { | ||||
|         return this._overlayEnabled; | ||||
|     } | ||||
|  | ||||
|     // eslint-disable-next-line camelcase | ||||
|     set overlay_enabled(enabled) { | ||||
|         if (this._overlayEnabled === enabled) | ||||
|             return; | ||||
|  | ||||
|         this._overlayEnabled = enabled; | ||||
|         this.notify('overlay-enabled'); | ||||
|  | ||||
|         if (!enabled) | ||||
|             this.hideOverlay(false); | ||||
|         else if (this['has-pointer'] || global.stage.key_focus === this) | ||||
|             this.showOverlay(true); | ||||
|     } | ||||
|  | ||||
|     // Find the actor just below us, respecting reparenting done by DND code | ||||
|     _getActualStackAbove() { | ||||
|         if (this._stackAbove == null) | ||||
|             return null; | ||||
|  | ||||
|         if (this.inDrag) { | ||||
|             if (this._stackAbove._delegate) | ||||
|                 return this._stackAbove._delegate._getActualStackAbove(); | ||||
|             else | ||||
|                 return null; | ||||
|         } else { | ||||
|             return this._stackAbove; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     setStackAbove(actor) { | ||||
|         this._stackAbove = actor; | ||||
|         if (this.inDrag) | ||||
|             // We'll fix up the stack after the drag | ||||
|             return; | ||||
|  | ||||
|         let parent = this.get_parent(); | ||||
|         let actualAbove = this._getActualStackAbove(); | ||||
|         if (actualAbove == null) | ||||
|             parent.set_child_below_sibling(this, null); | ||||
|         else | ||||
|             parent.set_child_above_sibling(this, actualAbove); | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
|         this._windowActor.disconnect(this._windowDestroyId); | ||||
|  | ||||
|         this.metaWindow._delegate = null; | ||||
|         this._delegate = null; | ||||
|  | ||||
|         this.metaWindow.disconnect(this._updateCaptionId); | ||||
|  | ||||
|         if (this._longPressLater) { | ||||
|             Meta.later_remove(this._longPressLater); | ||||
|             delete this._longPressLater; | ||||
|         } | ||||
|  | ||||
|         if (this._idleHideOverlayId > 0) { | ||||
|             GLib.source_remove(this._idleHideOverlayId); | ||||
|             this._idleHideOverlayId = 0; | ||||
|         } | ||||
|  | ||||
|         if (this.inDrag) { | ||||
|             this.emit('drag-end'); | ||||
|             this.inDrag = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _activate() { | ||||
|         this._selected = true; | ||||
|         this.emit('selected', global.get_current_time()); | ||||
|     } | ||||
|  | ||||
|     vfunc_enter_event(crossingEvent) { | ||||
|         this.showOverlay(true); | ||||
|         return super.vfunc_enter_event(crossingEvent); | ||||
|     } | ||||
|  | ||||
|     vfunc_leave_event(crossingEvent) { | ||||
|         if (this._idleHideOverlayId > 0) | ||||
|             GLib.source_remove(this._idleHideOverlayId); | ||||
|  | ||||
|         this._idleHideOverlayId = GLib.timeout_add( | ||||
|             GLib.PRIORITY_DEFAULT, | ||||
|             WINDOW_OVERLAY_IDLE_HIDE_TIMEOUT, () => { | ||||
|                 if (this._closeButton['has-pointer'] || | ||||
|                     this._title['has-pointer']) | ||||
|                     return GLib.SOURCE_CONTINUE; | ||||
|  | ||||
|                 if (!this['has-pointer']) | ||||
|                     this.hideOverlay(true); | ||||
|  | ||||
|                 this._idleHideOverlayId = 0; | ||||
|                 return GLib.SOURCE_REMOVE; | ||||
|             }); | ||||
|  | ||||
|         GLib.Source.set_name_by_id(this._idleHideOverlayId, '[gnome-shell] this._idleHideOverlayId'); | ||||
|  | ||||
|         return super.vfunc_leave_event(crossingEvent); | ||||
|     } | ||||
|  | ||||
|     vfunc_key_focus_in() { | ||||
|         super.vfunc_key_focus_in(); | ||||
|         this.showOverlay(true); | ||||
|     } | ||||
|  | ||||
|     vfunc_key_focus_out() { | ||||
|         super.vfunc_key_focus_out(); | ||||
|         this.hideOverlay(true); | ||||
|     } | ||||
|  | ||||
|     vfunc_key_press_event(keyEvent) { | ||||
|         let symbol = keyEvent.keyval; | ||||
|         let isEnter = symbol == Clutter.KEY_Return || symbol == Clutter.KEY_KP_Enter; | ||||
|         if (isEnter) { | ||||
|             this._activate(); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         return super.vfunc_key_press_event(keyEvent); | ||||
|     } | ||||
|  | ||||
|     _onLongPress(action, actor, state) { | ||||
|         // Take advantage of the Clutter policy to consider | ||||
|         // a long-press canceled when the pointer movement | ||||
|         // exceeds dnd-drag-threshold to manually start the drag | ||||
|         if (state == Clutter.LongPressState.CANCEL) { | ||||
|             let event = Clutter.get_current_event(); | ||||
|             this._dragTouchSequence = event.get_event_sequence(); | ||||
|  | ||||
|             if (this._longPressLater) | ||||
|                 return true; | ||||
|  | ||||
|             // A click cancels a long-press before any click handler is | ||||
|             // run - make sure to not start a drag in that case | ||||
|             this._longPressLater = Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => { | ||||
|                 delete this._longPressLater; | ||||
|                 if (this._selected) | ||||
|                     return; | ||||
|                 let [x, y] = action.get_coords(); | ||||
|                 action.release(); | ||||
|                 this._draggable.startDrag(x, y, global.get_current_time(), this._dragTouchSequence, event.get_device()); | ||||
|             }); | ||||
|         } else { | ||||
|             this.showOverlay(true); | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     _onDragBegin(_draggable, _time) { | ||||
|         this.inDrag = true; | ||||
|         this.hideOverlay(false); | ||||
|         this.emit('drag-begin'); | ||||
|     } | ||||
|  | ||||
|     handleDragOver(source, actor, x, y, time) { | ||||
|         return this._workspace.handleDragOver(source, actor, x, y, time); | ||||
|     } | ||||
|  | ||||
|     acceptDrop(source, actor, x, y, time) { | ||||
|         return this._workspace.acceptDrop(source, actor, x, y, time); | ||||
|     } | ||||
|  | ||||
|     _onDragCancelled(_draggable, _time) { | ||||
|         this.emit('drag-cancelled'); | ||||
|     } | ||||
|  | ||||
|     _onDragEnd(_draggable, _time, _snapback) { | ||||
|         this.inDrag = false; | ||||
|  | ||||
|         // We may not have a parent if DnD completed successfully, in | ||||
|         // which case our clone will shortly be destroyed and replaced | ||||
|         // with a new one on the target workspace. | ||||
|         let parent = this.get_parent(); | ||||
|         if (parent !== null) { | ||||
|             if (this._stackAbove == null) | ||||
|                 parent.set_child_below_sibling(this, null); | ||||
|             else | ||||
|                 parent.set_child_above_sibling(this, this._stackAbove); | ||||
|         } | ||||
|  | ||||
|         if (this['has-pointer']) | ||||
|             this.showOverlay(true); | ||||
|  | ||||
|         this.emit('drag-end'); | ||||
|     } | ||||
| }); | ||||
							
								
								
									
										1931
									
								
								js/ui/workspace.js
									
									
									
									
									
								
							
							
						
						
									
										1931
									
								
								js/ui/workspace.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										475
									
								
								js/ui/workspaceAnimation.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										475
									
								
								js/ui/workspaceAnimation.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,475 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
| /* exported WorkspaceAnimationController */ | ||||
|  | ||||
| const { Clutter, GObject, Meta, Shell } = imports.gi; | ||||
|  | ||||
| const Background = imports.ui.background; | ||||
| const Layout = imports.ui.layout; | ||||
| const Main = imports.ui.main; | ||||
| const SwipeTracker = imports.ui.swipeTracker; | ||||
|  | ||||
| const WINDOW_ANIMATION_TIME = 250; | ||||
|  | ||||
| const WorkspaceGroup = GObject.registerClass( | ||||
| class WorkspaceGroup extends Clutter.Actor { | ||||
|     _init(workspace, monitor, movingWindow) { | ||||
|         super._init(); | ||||
|  | ||||
|         this._workspace = workspace; | ||||
|         this._monitor = monitor; | ||||
|         this._movingWindow = movingWindow; | ||||
|         this._windowRecords = []; | ||||
|  | ||||
|         this._createWindows(); | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|         this._restackedId = global.display.connect('restacked', | ||||
|             this._syncStacking.bind(this)); | ||||
|     } | ||||
|  | ||||
|     get workspace() { | ||||
|         return this._workspace; | ||||
|     } | ||||
|  | ||||
|     _shouldShowWindow(window) { | ||||
|         if (!window.showing_on_its_workspace()) | ||||
|             return false; | ||||
|  | ||||
|         const geometry = global.display.get_monitor_geometry(this._monitor.index); | ||||
|         const [intersects, intersection_] = window.get_frame_rect().intersect(geometry); | ||||
|         if (!intersects) | ||||
|             return false; | ||||
|  | ||||
|         const isSticky = | ||||
|             window.is_on_all_workspaces() || window === this._movingWindow; | ||||
|  | ||||
|         // No workspace means we should show windows that are on all workspaces | ||||
|         if (!this._workspace) | ||||
|             return isSticky; | ||||
|  | ||||
|         // Otherwise only show windows that are (only) on that workspace | ||||
|         return !isSticky && window.located_on_workspace(this._workspace); | ||||
|     } | ||||
|  | ||||
|     _syncStacking() { | ||||
|         const windowActors = global.get_window_actors().filter(w => | ||||
|             this._shouldShowWindow(w.meta_window)); | ||||
|  | ||||
|         let lastRecord; | ||||
|  | ||||
|         for (const windowActor of windowActors) { | ||||
|             const record = this._windowRecords.find(r => r.windowActor === windowActor); | ||||
|  | ||||
|             this.set_child_above_sibling(record.clone, lastRecord ? lastRecord.clone : null); | ||||
|             lastRecord = record; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _createWindows() { | ||||
|         const windowActors = global.get_window_actors().filter(w => | ||||
|             this._shouldShowWindow(w.meta_window)); | ||||
|  | ||||
|         for (const windowActor of windowActors) { | ||||
|             const clone = new Clutter.Clone({ | ||||
|                 source: windowActor, | ||||
|                 x: windowActor.x - this._monitor.x, | ||||
|                 y: windowActor.y - this._monitor.y, | ||||
|             }); | ||||
|  | ||||
|             this.add_child(clone); | ||||
|  | ||||
|             const record = { windowActor, clone }; | ||||
|  | ||||
|             record.windowDestroyId = windowActor.connect('destroy', () => { | ||||
|                 clone.destroy(); | ||||
|                 this._windowRecords.splice(this._windowRecords.indexOf(record), 1); | ||||
|             }); | ||||
|  | ||||
|             this._windowRecords.push(record); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _removeWindows() { | ||||
|         for (const record of this._windowRecords) { | ||||
|             record.windowActor.disconnect(record.windowDestroyId); | ||||
|             record.clone.destroy(); | ||||
|         } | ||||
|  | ||||
|         this._windowRecords = []; | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
|         global.display.disconnect(this._restackedId); | ||||
|         this._removeWindows(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const MonitorGroup = GObject.registerClass({ | ||||
|     Properties: { | ||||
|         'progress': GObject.ParamSpec.double( | ||||
|             'progress', 'progress', 'progress', | ||||
|             GObject.ParamFlags.READWRITE, | ||||
|             -Infinity, Infinity, 0), | ||||
|     }, | ||||
| }, class MonitorGroup extends Clutter.Actor { | ||||
|     _init(monitor, workspaceIndices, movingWindow) { | ||||
|         super._init({ | ||||
|             clip_to_allocation: true, | ||||
|         }); | ||||
|  | ||||
|         this._monitor = monitor; | ||||
|  | ||||
|         const constraint = new Layout.MonitorConstraint({ index: monitor.index }); | ||||
|         this.add_constraint(constraint); | ||||
|  | ||||
|         const background = new Meta.BackgroundGroup(); | ||||
|  | ||||
|         this.add_child(background); | ||||
|  | ||||
|         this._container = new Clutter.Actor(); | ||||
|         this.add_child(this._container); | ||||
|  | ||||
|         const stickyGroup = new WorkspaceGroup(null, monitor, movingWindow); | ||||
|         this.add_child(stickyGroup); | ||||
|  | ||||
|         this._workspaceGroups = []; | ||||
|  | ||||
|         const workspaceManager = global.workspace_manager; | ||||
|         const vertical = workspaceManager.layout_rows === -1; | ||||
|         const activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|  | ||||
|         let x = 0; | ||||
|         let y = 0; | ||||
|  | ||||
|         for (const i of workspaceIndices) { | ||||
|             const ws = workspaceManager.get_workspace_by_index(i); | ||||
|             const fullscreen = ws.list_windows().some(w => w.get_monitor() === monitor.index && w.is_fullscreen()); | ||||
|  | ||||
|             if (i > 0 && vertical && !fullscreen && monitor.index === Main.layoutManager.primaryIndex) { | ||||
|                 // We have to shift windows up or down by the height of the panel to prevent having a | ||||
|                 // visible gap between the windows while switching workspaces. Since fullscreen windows | ||||
|                 // hide the panel, they don't need to be shifted up or down. | ||||
|                 y -= Main.panel.height; | ||||
|             } | ||||
|  | ||||
|             const group = new WorkspaceGroup(ws, monitor, movingWindow); | ||||
|  | ||||
|             this._workspaceGroups.push(group); | ||||
|             this._container.add_child(group); | ||||
|             group.set_position(x, y); | ||||
|  | ||||
|             if (vertical) | ||||
|                 y += this.baseDistance; | ||||
|             else if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) | ||||
|                 x -= this.baseDistance; | ||||
|             else | ||||
|                 x += this.baseDistance; | ||||
|         } | ||||
|  | ||||
|         this.progress = this.getWorkspaceProgress(activeWorkspace); | ||||
|  | ||||
|         this._bgManager = new Background.BackgroundManager({ | ||||
|             container: background, | ||||
|             monitorIndex: monitor.index, | ||||
|             controlPosition: false, | ||||
|         }); | ||||
|  | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
|         this._bgManager.destroy(); | ||||
|     } | ||||
|  | ||||
|     get baseDistance() { | ||||
|         if (global.workspace_manager.layout_rows === -1) | ||||
|             return this._monitor.height; | ||||
|         else | ||||
|             return this._monitor.width; | ||||
|     } | ||||
|  | ||||
|     get progress() { | ||||
|         if (global.workspace_manager.layout_rows === -1) | ||||
|             return -this._container.y / this.baseDistance; | ||||
|         else if (this.get_text_direction() === Clutter.TextDirection.RTL) | ||||
|             return this._container.x / this.baseDistance; | ||||
|         else | ||||
|             return -this._container.x / this.baseDistance; | ||||
|     } | ||||
|  | ||||
|     set progress(p) { | ||||
|         if (global.workspace_manager.layout_rows === -1) | ||||
|             this._container.y = -Math.round(p * this.baseDistance); | ||||
|         else if (this.get_text_direction() === Clutter.TextDirection.RTL) | ||||
|             this._container.x = Math.round(p * this.baseDistance); | ||||
|         else | ||||
|             this._container.x = -Math.round(p * this.baseDistance); | ||||
|     } | ||||
|  | ||||
|     get index() { | ||||
|         return this._monitor.index; | ||||
|     } | ||||
|  | ||||
|     getWorkspaceProgress(workspace) { | ||||
|         const group = this._workspaceGroups.find(g => | ||||
|             g.workspace.index() === workspace.index()); | ||||
|         return this._getWorkspaceGroupProgress(group); | ||||
|     } | ||||
|  | ||||
|     _getWorkspaceGroupProgress(group) { | ||||
|         if (global.workspace_manager.layout_rows === -1) | ||||
|             return group.y / this.baseDistance; | ||||
|         else if (this.get_text_direction() === Clutter.TextDirection.RTL) | ||||
|             return -group.x / this.baseDistance; | ||||
|         else | ||||
|             return group.x / this.baseDistance; | ||||
|     } | ||||
|  | ||||
|     getSnapPoints() { | ||||
|         return this._workspaceGroups.map(g => | ||||
|             this._getWorkspaceGroupProgress(g)); | ||||
|     } | ||||
|  | ||||
|     findClosestWorkspace(progress) { | ||||
|         const distances = this.getSnapPoints().map(p => | ||||
|             Math.abs(p - progress)); | ||||
|         const index = distances.indexOf(Math.min(...distances)); | ||||
|         return this._workspaceGroups[index].workspace; | ||||
|     } | ||||
|  | ||||
|     _interpolateProgress(progress, monitorGroup) { | ||||
|         if (this.index === monitorGroup.index) | ||||
|             return progress; | ||||
|  | ||||
|         const points1 = monitorGroup.getSnapPoints(); | ||||
|         const points2 = this.getSnapPoints(); | ||||
|  | ||||
|         const upper = points1.indexOf(points1.find(p => p >= progress)); | ||||
|         const lower = points1.indexOf(points1.slice().reverse().find(p => p <= progress)); | ||||
|  | ||||
|         if (points1[upper] === points1[lower]) | ||||
|             return points2[upper]; | ||||
|  | ||||
|         const t = (progress - points1[lower]) / (points1[upper] - points1[lower]); | ||||
|  | ||||
|         return points2[lower] + (points2[upper] - points2[lower]) * t; | ||||
|     } | ||||
|  | ||||
|     updateSwipeForMonitor(progress, monitorGroup) { | ||||
|         this.progress = this._interpolateProgress(progress, monitorGroup); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| var WorkspaceAnimationController = class { | ||||
|     constructor() { | ||||
|         this._movingWindow = null; | ||||
|         this._switchData = null; | ||||
|  | ||||
|         Main.overview.connect('showing', () => { | ||||
|             if (this._switchData) { | ||||
|                 if (this._switchData.gestureActivated) | ||||
|                     this._finishWorkspaceSwitch(this._switchData); | ||||
|                 this._swipeTracker.enabled = false; | ||||
|             } | ||||
|         }); | ||||
|         Main.overview.connect('hiding', () => { | ||||
|             this._swipeTracker.enabled = true; | ||||
|         }); | ||||
|  | ||||
|         let swipeTracker = new SwipeTracker.SwipeTracker(global.stage, | ||||
|             Shell.ActionMode.NORMAL, { allowDrag: false, allowScroll: false }); | ||||
|         swipeTracker.connect('begin', this._switchWorkspaceBegin.bind(this)); | ||||
|         swipeTracker.connect('update', this._switchWorkspaceUpdate.bind(this)); | ||||
|         swipeTracker.connect('end', this._switchWorkspaceEnd.bind(this)); | ||||
|         this._swipeTracker = swipeTracker; | ||||
|     } | ||||
|  | ||||
|     _prepareWorkspaceSwitch(workspaceIndices) { | ||||
|         if (this._switchData) | ||||
|             return; | ||||
|  | ||||
|         const workspaceManager = global.workspace_manager; | ||||
|         const nWorkspaces = workspaceManager.get_n_workspaces(); | ||||
|  | ||||
|         const switchData = {}; | ||||
|  | ||||
|         this._switchData = switchData; | ||||
|         switchData.monitors = []; | ||||
|  | ||||
|         switchData.gestureActivated = false; | ||||
|         switchData.inProgress = false; | ||||
|  | ||||
|         if (!workspaceIndices) | ||||
|             workspaceIndices = [...Array(nWorkspaces).keys()]; | ||||
|  | ||||
|         const monitors = Meta.prefs_get_workspaces_only_on_primary() | ||||
|             ? [Main.layoutManager.primaryMonitor] : Main.layoutManager.monitors; | ||||
|  | ||||
|         for (const monitor of monitors) { | ||||
|             if (Meta.prefs_get_workspaces_only_on_primary() && | ||||
|                 monitor.index !== Main.layoutManager.primaryIndex) | ||||
|                 continue; | ||||
|  | ||||
|             const group = new MonitorGroup(monitor, workspaceIndices, this.movingWindow); | ||||
|  | ||||
|             Main.uiGroup.insert_child_above(group, global.window_group); | ||||
|  | ||||
|             switchData.monitors.push(group); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _finishWorkspaceSwitch(switchData) { | ||||
|         this._switchData = null; | ||||
|  | ||||
|         switchData.monitors.forEach(m => m.destroy()); | ||||
|  | ||||
|         this.movingWindow = null; | ||||
|     } | ||||
|  | ||||
|     animateSwitch(from, to, direction, onComplete) { | ||||
|         this._swipeTracker.enabled = false; | ||||
|  | ||||
|         let workspaceIndices = []; | ||||
|  | ||||
|         switch (direction) { | ||||
|         case Meta.MotionDirection.UP: | ||||
|         case Meta.MotionDirection.LEFT: | ||||
|         case Meta.MotionDirection.UP_LEFT: | ||||
|         case Meta.MotionDirection.UP_RIGHT: | ||||
|             workspaceIndices = [to, from]; | ||||
|             break; | ||||
|  | ||||
|         case Meta.MotionDirection.DOWN: | ||||
|         case Meta.MotionDirection.RIGHT: | ||||
|         case Meta.MotionDirection.DOWN_LEFT: | ||||
|         case Meta.MotionDirection.DOWN_RIGHT: | ||||
|             workspaceIndices = [from, to]; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL && | ||||
|             direction !== Meta.MotionDirection.UP && | ||||
|             direction !== Meta.MotionDirection.DOWN) | ||||
|             workspaceIndices.reverse(); | ||||
|  | ||||
|         this._prepareWorkspaceSwitch(workspaceIndices); | ||||
|         this._switchData.inProgress = true; | ||||
|  | ||||
|         const fromWs = global.workspace_manager.get_workspace_by_index(from); | ||||
|         const toWs = global.workspace_manager.get_workspace_by_index(to); | ||||
|  | ||||
|         for (const monitorGroup of this._switchData.monitors) { | ||||
|             monitorGroup.progress = monitorGroup.getWorkspaceProgress(fromWs); | ||||
|             const progress = monitorGroup.getWorkspaceProgress(toWs); | ||||
|  | ||||
|             const params = { | ||||
|                 duration: WINDOW_ANIMATION_TIME, | ||||
|                 mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             }; | ||||
|  | ||||
|             if (monitorGroup.index === Main.layoutManager.primaryIndex) { | ||||
|                 params.onComplete = () => { | ||||
|                     this._finishWorkspaceSwitch(this._switchData); | ||||
|                     onComplete(); | ||||
|                     this._swipeTracker.enabled = true; | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             monitorGroup.ease_property('progress', progress, params); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _findMonitorGroup(monitorIndex) { | ||||
|         return this._switchData.monitors.find(m => m.index === monitorIndex); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceBegin(tracker, monitor) { | ||||
|         if (Meta.prefs_get_workspaces_only_on_primary() && | ||||
|             monitor !== Main.layoutManager.primaryIndex) | ||||
|             return; | ||||
|  | ||||
|         const workspaceManager = global.workspace_manager; | ||||
|         const horiz = workspaceManager.layout_rows !== -1; | ||||
|         tracker.orientation = horiz | ||||
|             ? Clutter.Orientation.HORIZONTAL | ||||
|             : Clutter.Orientation.VERTICAL; | ||||
|  | ||||
|         if (this._switchData && this._switchData.gestureActivated) { | ||||
|             for (const group of this._switchData.monitors) | ||||
|                 group.remove_all_transitions(); | ||||
|         } else { | ||||
|             this._prepareWorkspaceSwitch(); | ||||
|         } | ||||
|  | ||||
|         const monitorGroup = this._findMonitorGroup(monitor); | ||||
|         const baseDistance = monitorGroup.baseDistance; | ||||
|         const progress = monitorGroup.progress; | ||||
|  | ||||
|         const closestWs = monitorGroup.findClosestWorkspace(progress); | ||||
|         const cancelProgress = monitorGroup.getWorkspaceProgress(closestWs); | ||||
|         const points = monitorGroup.getSnapPoints(); | ||||
|  | ||||
|         this._switchData.baseMonitorGroup = monitorGroup; | ||||
|  | ||||
|         tracker.confirmSwipe(baseDistance, points, progress, cancelProgress); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceUpdate(tracker, progress) { | ||||
|         if (!this._switchData) | ||||
|             return; | ||||
|  | ||||
|         for (const monitorGroup of this._switchData.monitors) | ||||
|             monitorGroup.updateSwipeForMonitor(progress, this._switchData.baseMonitorGroup); | ||||
|     } | ||||
|  | ||||
|     _switchWorkspaceEnd(tracker, duration, endProgress) { | ||||
|         if (!this._switchData) | ||||
|             return; | ||||
|  | ||||
|         const switchData = this._switchData; | ||||
|         switchData.gestureActivated = true; | ||||
|  | ||||
|         const newWs = switchData.baseMonitorGroup.findClosestWorkspace(endProgress); | ||||
|  | ||||
|         for (const monitorGroup of this._switchData.monitors) { | ||||
|             const progress = monitorGroup.getWorkspaceProgress(newWs); | ||||
|  | ||||
|             const params = { | ||||
|                 duration, | ||||
|                 mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             }; | ||||
|  | ||||
|             if (monitorGroup.index === Main.layoutManager.primaryIndex) { | ||||
|                 params.onComplete = () => { | ||||
|                     if (!newWs.active) | ||||
|                         newWs.activate(global.get_current_time()); | ||||
|                     this._finishWorkspaceSwitch(switchData); | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             monitorGroup.ease_property('progress', progress, params); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     get gestureActive() { | ||||
|         return this._switchData !== null && this._switchData.gestureActivated; | ||||
|     } | ||||
|  | ||||
|     cancelSwitchAnimation() { | ||||
|         if (!this._switchData) | ||||
|             return; | ||||
|  | ||||
|         if (this._switchData.gestureActivated) | ||||
|             return; | ||||
|  | ||||
|         this._finishWorkspaceSwitch(this._switchData); | ||||
|     } | ||||
|  | ||||
|     set movingWindow(movingWindow) { | ||||
|         this._movingWindow = movingWindow; | ||||
|     } | ||||
|  | ||||
|     get movingWindow() { | ||||
|         return this._movingWindow; | ||||
|     } | ||||
| }; | ||||
| @@ -11,7 +11,10 @@ var DISPLAY_TIMEOUT = 600; | ||||
| var WorkspaceSwitcherPopupList = GObject.registerClass( | ||||
| class WorkspaceSwitcherPopupList extends St.Widget { | ||||
|     _init() { | ||||
|         super._init({ style_class: 'workspace-switcher' }); | ||||
|         super._init({ | ||||
|             style_class: 'workspace-switcher', | ||||
|             offscreen_redirect: Clutter.OffscreenRedirect.ALWAYS, | ||||
|         }); | ||||
|  | ||||
|         this._itemSpacing = 0; | ||||
|         this._childHeight = 0; | ||||
|   | ||||
| @@ -549,9 +549,7 @@ var WorkspaceThumbnail = GObject.registerClass({ | ||||
|             return; | ||||
|  | ||||
|         // a click on the already current workspace should go back to the main view | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|         if (this.metaWorkspace == activeWorkspace) | ||||
|         if (this.metaWorkspace.active) | ||||
|             Main.overview.hide(); | ||||
|         else | ||||
|             this.metaWorkspace.activate(time); | ||||
| @@ -567,7 +565,8 @@ var WorkspaceThumbnail = GObject.registerClass({ | ||||
|         if (this.state > ThumbnailState.NORMAL) | ||||
|             return DND.DragMotionResult.CONTINUE; | ||||
|  | ||||
|         if (source.realWindow && !this._isMyWindow(source.realWindow)) | ||||
|         if (source.metaWindow && | ||||
|             !this._isMyWindow(source.metaWindow.get_compositor_private())) | ||||
|             return DND.DragMotionResult.MOVE_DROP; | ||||
|         if (source.app && source.app.can_open_new_window()) | ||||
|             return DND.DragMotionResult.COPY_DROP; | ||||
| @@ -581,8 +580,8 @@ var WorkspaceThumbnail = GObject.registerClass({ | ||||
|         if (this.state > ThumbnailState.NORMAL) | ||||
|             return false; | ||||
|  | ||||
|         if (source.realWindow) { | ||||
|             let win = source.realWindow; | ||||
|         if (source.metaWindow) { | ||||
|             let win = source.metaWindow.get_compositor_private(); | ||||
|             if (this._isMyWindow(win)) | ||||
|                 return false; | ||||
|  | ||||
| @@ -795,7 +794,7 @@ var ThumbnailsBox = GObject.registerClass({ | ||||
|  | ||||
|     // Draggable target interface | ||||
|     handleDragOver(source, actor, x, y, time) { | ||||
|         if (!source.realWindow && | ||||
|         if (!source.metaWindow && | ||||
|             (!source.app || !source.app.can_open_new_window()) && | ||||
|             (source.app || !source.shellWorkspaceLaunch) && | ||||
|             source != Main.xdndHandler) | ||||
| @@ -846,7 +845,7 @@ var ThumbnailsBox = GObject.registerClass({ | ||||
|         if (this._dropWorkspace != -1) | ||||
|             return this._thumbnails[this._dropWorkspace].handleDragOverInternal(source, actor, time); | ||||
|         else if (this._dropPlaceholderPos != -1) | ||||
|             return source.realWindow ? DND.DragMotionResult.MOVE_DROP : DND.DragMotionResult.COPY_DROP; | ||||
|             return source.metaWindow ? DND.DragMotionResult.MOVE_DROP : DND.DragMotionResult.COPY_DROP; | ||||
|         else | ||||
|             return DND.DragMotionResult.CONTINUE; | ||||
|     } | ||||
| @@ -855,12 +854,12 @@ var ThumbnailsBox = GObject.registerClass({ | ||||
|         if (this._dropWorkspace != -1) { | ||||
|             return this._thumbnails[this._dropWorkspace].acceptDropInternal(source, actor, time); | ||||
|         } else if (this._dropPlaceholderPos != -1) { | ||||
|             if (!source.realWindow && | ||||
|             if (!source.metaWindow && | ||||
|                 (!source.app || !source.app.can_open_new_window()) && | ||||
|                 (source.app || !source.shellWorkspaceLaunch)) | ||||
|                 return false; | ||||
|  | ||||
|             let isWindow = !!source.realWindow; | ||||
|             let isWindow = !!source.metaWindow; | ||||
|  | ||||
|             let newWorkspaceIndex; | ||||
|             [newWorkspaceIndex, this._dropPlaceholderPos] = [this._dropPlaceholderPos, -1]; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ const Main = imports.ui.main; | ||||
| const SwipeTracker = imports.ui.swipeTracker; | ||||
| const Workspace = imports.ui.workspace; | ||||
|  | ||||
| var { ANIMATION_TIME } = imports.ui.overview; | ||||
| var WORKSPACE_SWITCH_TIME = 250; | ||||
| var SCROLL_TIMEOUT_TIME = 150; | ||||
|  | ||||
| @@ -21,18 +22,18 @@ var WorkspacesViewBase = GObject.registerClass({ | ||||
|     GTypeFlags: GObject.TypeFlags.ABSTRACT, | ||||
| }, class WorkspacesViewBase extends St.Widget { | ||||
|     _init(monitorIndex) { | ||||
|         super._init({ style_class: 'workspaces-view', reactive: true }); | ||||
|         const { x, y, width, height } = | ||||
|             Main.layoutManager.getWorkAreaForMonitor(monitorIndex); | ||||
|  | ||||
|         super._init({ | ||||
|             style_class: 'workspaces-view', | ||||
|             x, y, width, height, | ||||
|         }); | ||||
|         this.connect('destroy', this._onDestroy.bind(this)); | ||||
|         global.focus_manager.add_group(this); | ||||
|  | ||||
|         // The actor itself isn't a drop target, so we don't want to pick on its area | ||||
|         this.set_size(0, 0); | ||||
|  | ||||
|         this._monitorIndex = monitorIndex; | ||||
|  | ||||
|         this._fullGeometry = null; | ||||
|         this._actualGeometry = null; | ||||
|  | ||||
|         this._inDrag = false; | ||||
|         this._windowDragBeginId = Main.overview.connect('window-drag-begin', this._dragBegin.bind(this)); | ||||
|         this._windowDragEndId = Main.overview.connect('window-drag-end', this._dragEnd.bind(this)); | ||||
| @@ -51,24 +52,19 @@ var WorkspacesViewBase = GObject.registerClass({ | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _dragBegin(overview, window) { | ||||
|     _dragBegin() { | ||||
|         this._inDrag = true; | ||||
|         this._setReservedSlot(window); | ||||
|     } | ||||
|  | ||||
|     _dragEnd() { | ||||
|         this._inDrag = false; | ||||
|         this._setReservedSlot(null); | ||||
|     } | ||||
|  | ||||
|     setFullGeometry(geom) { | ||||
|         this._fullGeometry = geom; | ||||
|         this._syncFullGeometry(); | ||||
|     } | ||||
|     vfunc_allocate(box) { | ||||
|         this.set_allocation(box); | ||||
|  | ||||
|     setActualGeometry(geom) { | ||||
|         this._actualGeometry = geom; | ||||
|         this._syncActualGeometry(); | ||||
|         for (const child of this) | ||||
|             child.allocate_available_size(0, 0, box.get_width(), box.get_height()); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -83,9 +79,8 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|         this._gestureActive = false; // touch(pad) gestures | ||||
|  | ||||
|         this._scrollAdjustment = scrollAdjustment; | ||||
|         this._onScrollId = | ||||
|             this._scrollAdjustment.connect('notify::value', | ||||
|                 this._onScroll.bind(this)); | ||||
|         this._onScrollId = this._scrollAdjustment.connect('notify::value', | ||||
|             this._updateScrollPosition.bind(this)); | ||||
|  | ||||
|         this._workspaces = []; | ||||
|         this._updateWorkspaces(); | ||||
| @@ -97,34 +92,42 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|                 this._workspaces.sort((a, b) => { | ||||
|                     return a.metaWorkspace.index() - b.metaWorkspace.index(); | ||||
|                 }); | ||||
|                 this._updateWorkspaceActors(false); | ||||
|                 this._workspaces.forEach( | ||||
|                     (ws, i) => this.set_child_at_index(ws, i)); | ||||
|             }); | ||||
|  | ||||
|  | ||||
|         this._overviewShownId = | ||||
|             Main.overview.connect('shown', () => { | ||||
|                 this.set_clip(this._fullGeometry.x, this._fullGeometry.y, | ||||
|                               this._fullGeometry.width, this._fullGeometry.height); | ||||
|             }); | ||||
|         this._overviewShownId = Main.overview.connect('shown', () => { | ||||
|             this.clip_to_allocation = true; | ||||
|         }); | ||||
|  | ||||
|         this._switchWorkspaceNotifyId = | ||||
|             global.window_manager.connect('switch-workspace', | ||||
|                                           this._activeWorkspaceChanged.bind(this)); | ||||
|     } | ||||
|  | ||||
|     _setReservedSlot(window) { | ||||
|         for (let i = 0; i < this._workspaces.length; i++) | ||||
|             this._workspaces[i].setReservedSlot(window); | ||||
|     } | ||||
|     vfunc_allocate(box) { | ||||
|         this.set_allocation(box); | ||||
|  | ||||
|     _syncFullGeometry() { | ||||
|         for (let i = 0; i < this._workspaces.length; i++) | ||||
|             this._workspaces[i].setFullGeometry(this._fullGeometry); | ||||
|     } | ||||
|         if (this.get_n_children() === 0) | ||||
|             return; | ||||
|  | ||||
|     _syncActualGeometry() { | ||||
|         for (let i = 0; i < this._workspaces.length; i++) | ||||
|             this._workspaces[i].setActualGeometry(this._actualGeometry); | ||||
|         const { workspaceManager } = global; | ||||
|         const { nWorkspaces } = workspaceManager; | ||||
|  | ||||
|         const vertical = workspaceManager.layout_rows === -1; | ||||
|         const rtl = this.text_direction === Clutter.TextDirection.RTL; | ||||
|  | ||||
|         this._workspaces.forEach((child, index) => { | ||||
|             if (rtl && !vertical) | ||||
|                 index = nWorkspaces - index - 1; | ||||
|  | ||||
|             const x = vertical ? 0 : index * this.width; | ||||
|             const y = vertical ? index * this.height : 0; | ||||
|  | ||||
|             child.allocate_available_size(x, y, box.get_width(), box.get_height()); | ||||
|         }); | ||||
|  | ||||
|         this._updateScrollPosition(); | ||||
|     } | ||||
|  | ||||
|     getActiveWorkspace() { | ||||
| @@ -140,11 +143,11 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|             else | ||||
|                 this._workspaces[w].fadeToOverview(); | ||||
|         } | ||||
|         this._updateWorkspaceActors(false); | ||||
|         this._updateScrollPosition(); | ||||
|     } | ||||
|  | ||||
|     animateFromOverview(animationType) { | ||||
|         this.remove_clip(); | ||||
|         this.clip_to_allocation = false; | ||||
|  | ||||
|         for (let w = 0; w < this._workspaces.length; w++) { | ||||
|             if (animationType == AnimationType.ZOOM) | ||||
| @@ -159,49 +162,22 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|             this._workspaces[i].syncStacking(stackIndices); | ||||
|     } | ||||
|  | ||||
|     // Update workspace actors parameters | ||||
|     // @showAnimation: iff %true, transition between states | ||||
|     _updateWorkspaceActors(showAnimation) { | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let active = workspaceManager.get_active_workspace_index(); | ||||
|     _scrollToActive() { | ||||
|         const { workspaceManager } = global; | ||||
|         const active = workspaceManager.get_active_workspace_index(); | ||||
|  | ||||
|         this._animating = showAnimation; | ||||
|         this._animating = true; | ||||
|         this._updateVisibility(); | ||||
|  | ||||
|         for (let w = 0; w < this._workspaces.length; w++) { | ||||
|             let workspace = this._workspaces[w]; | ||||
|  | ||||
|             workspace.remove_all_transitions(); | ||||
|  | ||||
|             let params = {}; | ||||
|             if (workspaceManager.layout_rows == -1) | ||||
|                 params.y = (w - active) * this._fullGeometry.height; | ||||
|             else if (this.text_direction == Clutter.TextDirection.RTL) | ||||
|                 params.x = (active - w) * this._fullGeometry.width; | ||||
|             else | ||||
|                 params.x = (w - active) * this._fullGeometry.width; | ||||
|  | ||||
|             if (showAnimation) { | ||||
|                 let easeParams = Object.assign(params, { | ||||
|                     duration: WORKSPACE_SWITCH_TIME, | ||||
|                     mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|                 }); | ||||
|                 // we have to call _updateVisibility() once before the | ||||
|                 // animation and once afterwards - it does not really | ||||
|                 // matter which tween we use, so we pick the first one ... | ||||
|                 if (w == 0) { | ||||
|                     this._updateVisibility(); | ||||
|                     easeParams.onComplete = () => { | ||||
|                         this._animating = false; | ||||
|                         this._updateVisibility(); | ||||
|                     }; | ||||
|                 } | ||||
|                 workspace.ease(easeParams); | ||||
|             } else { | ||||
|                 workspace.set(params); | ||||
|                 if (w == 0) | ||||
|                     this._updateVisibility(); | ||||
|             } | ||||
|         } | ||||
|         this._scrollAdjustment.remove_transition('value'); | ||||
|         this._scrollAdjustment.ease(active, { | ||||
|             duration: WORKSPACE_SWITCH_TIME, | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             onComplete: () => { | ||||
|                 this._animating = false; | ||||
|                 this._updateVisibility(); | ||||
|             }, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _updateVisibility() { | ||||
| @@ -242,19 +218,14 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (this._fullGeometry) { | ||||
|             this._updateWorkspaceActors(false); | ||||
|             this._syncFullGeometry(); | ||||
|         } | ||||
|         if (this._actualGeometry) | ||||
|             this._syncActualGeometry(); | ||||
|         this._updateScrollPosition(); | ||||
|     } | ||||
|  | ||||
|     _activeWorkspaceChanged(_wm, _from, _to, _direction) { | ||||
|         if (this._scrolling) | ||||
|             return; | ||||
|  | ||||
|         this._updateWorkspaceActors(true); | ||||
|         this._scrollToActive(); | ||||
|     } | ||||
|  | ||||
|     _onDestroy() { | ||||
| @@ -270,27 +241,33 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|  | ||||
|     startTouchGesture() { | ||||
|         this._gestureActive = true; | ||||
|  | ||||
|         this._updateVisibility(); | ||||
|     } | ||||
|  | ||||
|     endTouchGesture() { | ||||
|         this._gestureActive = false; | ||||
|  | ||||
|         // Make sure title captions etc are shown as necessary | ||||
|         this._updateWorkspaceActors(true); | ||||
|         this._scrollToActive(); | ||||
|         this._updateVisibility(); | ||||
|     } | ||||
|  | ||||
|     // sync the workspaces' positions to the value of the scroll adjustment | ||||
|     // and change the active workspace if appropriate | ||||
|     _onScroll(adj) { | ||||
|         if (adj.get_transition('value') !== null && !this._gestureActive) | ||||
|     _updateScrollPosition() { | ||||
|         if (!this.has_allocation()) | ||||
|             return; | ||||
|  | ||||
|         const adj = this._scrollAdjustment; | ||||
|         const allowSwitch = | ||||
|             adj.get_transition('value') === null && !this._gestureActive; | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let active = workspaceManager.get_active_workspace_index(); | ||||
|         let current = Math.round(adj.value); | ||||
|  | ||||
|         if (active != current && !this._gestureActive) { | ||||
|         if (allowSwitch && active !== current) { | ||||
|             if (!this._workspaces[current]) { | ||||
|                 // The current workspace was destroyed. This could happen | ||||
|                 // when you are on the last empty workspace, and consolidate | ||||
| @@ -307,36 +284,16 @@ class WorkspacesView extends WorkspacesViewBase { | ||||
|         if (adj.upper == 1) | ||||
|             return; | ||||
|  | ||||
|         let last = this._workspaces.length - 1; | ||||
|         const vertical = workspaceManager.layout_rows === -1; | ||||
|         const rtl = this.text_direction === Clutter.TextDirection.RTL; | ||||
|         const progress = vertical || !rtl | ||||
|             ? adj.value : adj.upper - adj.value; | ||||
|  | ||||
|         if (workspaceManager.layout_rows == -1) { | ||||
|             let firstWorkspaceY = this._workspaces[0].y; | ||||
|             let lastWorkspaceY = this._workspaces[last].y; | ||||
|             let workspacesHeight = lastWorkspaceY - firstWorkspaceY; | ||||
|  | ||||
|             let currentY = firstWorkspaceY; | ||||
|             let newY = -Math.round(adj.value / (adj.upper - 1) * workspacesHeight); | ||||
|  | ||||
|             let dy = newY - currentY; | ||||
|  | ||||
|             for (let i = 0; i < this._workspaces.length; i++) { | ||||
|                 this._workspaces[i].visible = Math.abs(i - adj.value) <= 1; | ||||
|                 this._workspaces[i].y += dy; | ||||
|             } | ||||
|         } else { | ||||
|             let firstWorkspaceX = this._workspaces[0].x; | ||||
|             let lastWorkspaceX = this._workspaces[last].x; | ||||
|             let workspacesWidth = lastWorkspaceX - firstWorkspaceX; | ||||
|  | ||||
|             let currentX = firstWorkspaceX; | ||||
|             let newX = -Math.round(adj.value / (adj.upper - 1) * workspacesWidth); | ||||
|  | ||||
|             let dx = newX - currentX; | ||||
|  | ||||
|             for (let i = 0; i < this._workspaces.length; i++) { | ||||
|                 this._workspaces[i].visible = Math.abs(i - adj.value) <= 1; | ||||
|                 this._workspaces[i].x += dx; | ||||
|             } | ||||
|         for (const ws of this._workspaces) { | ||||
|             if (vertical) | ||||
|                 ws.translation_y = -progress * this.height; | ||||
|             else | ||||
|                 ws.translation_x = -progress * this.width; | ||||
|         } | ||||
|     } | ||||
| }); | ||||
| @@ -349,18 +306,6 @@ class ExtraWorkspaceView extends WorkspacesViewBase { | ||||
|         this.add_actor(this._workspace); | ||||
|     } | ||||
|  | ||||
|     _setReservedSlot(window) { | ||||
|         this._workspace.setReservedSlot(window); | ||||
|     } | ||||
|  | ||||
|     _syncFullGeometry() { | ||||
|         this._workspace.setFullGeometry(this._fullGeometry); | ||||
|     } | ||||
|  | ||||
|     _syncActualGeometry() { | ||||
|         this._workspace.setActualGeometry(this._actualGeometry); | ||||
|     } | ||||
|  | ||||
|     getActiveWorkspace() { | ||||
|         return this._workspace; | ||||
|     } | ||||
| @@ -438,6 +383,10 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         this._windowDragEndId = | ||||
|             Main.overview.connect('window-drag-begin', | ||||
|                 this._windowDragEnd.bind(this)); | ||||
|         this._overviewShownId = Main.overview.connect('shown', () => { | ||||
|             this._inWindowFade = false; | ||||
|             this._syncWorkspacesActualGeometry(); | ||||
|         }); | ||||
|  | ||||
|         this._primaryIndex = Main.layoutManager.primaryIndex; | ||||
|         this._workspacesViews = []; | ||||
| @@ -452,9 +401,11 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         this._scrollEventId = 0; | ||||
|         this._keyPressEventId = 0; | ||||
|         this._scrollTimeoutId = 0; | ||||
|         this._syncActualGeometryLater = 0; | ||||
|  | ||||
|         this._fullGeometry = null; | ||||
|         this._actualGeometry = null; | ||||
|         this._inWindowDrag = false; | ||||
|         this._inWindowFade = false; | ||||
|  | ||||
|         this._gestureActive = false; // touch(pad) gestures | ||||
|         this._canScroll = true; // limiting scrolling speed | ||||
| @@ -475,6 +426,11 @@ class WorkspacesDisplay extends St.Widget { | ||||
|             this._parentSetLater = 0; | ||||
|         } | ||||
|  | ||||
|         if (this._syncActualGeometryLater) { | ||||
|             Meta.later_remove(this._syncActualGeometryLater); | ||||
|             this._syncActualGeometryLater = 0; | ||||
|         } | ||||
|  | ||||
|         if (this._scrollTimeoutId !== 0) { | ||||
|             GLib.source_remove(this._scrollTimeoutId); | ||||
|             this._scrollTimeoutId = 0; | ||||
| @@ -484,6 +440,7 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         global.workspace_manager.disconnect(this._reorderWorkspacesdId); | ||||
|         Main.overview.disconnect(this._windowDragBeginId); | ||||
|         Main.overview.disconnect(this._windowDragEndId); | ||||
|         Main.overview.disconnect(this._overviewShownId); | ||||
|     } | ||||
|  | ||||
|     _windowDragBegin() { | ||||
| @@ -507,25 +464,11 @@ class WorkspacesDisplay extends St.Widget { | ||||
|             workspaceManager.get_active_workspace_index(); | ||||
|     } | ||||
|  | ||||
|     _activeWorkspaceChanged(_wm, _from, _to, _direction) { | ||||
|     _activeWorkspaceChanged(_wm, _from, to, _direction) { | ||||
|         if (this._gestureActive) | ||||
|             return; | ||||
|  | ||||
|         this._scrollToActive(); | ||||
|     } | ||||
|  | ||||
|     _scrollToActive() { | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let active = workspaceManager.get_active_workspace_index(); | ||||
|  | ||||
|         this._updateScrollAdjustment(active); | ||||
|     } | ||||
|  | ||||
|     _updateScrollAdjustment(index) { | ||||
|         if (this._gestureActive) | ||||
|             return; | ||||
|  | ||||
|         this._scrollAdjustment.ease(index, { | ||||
|         this._scrollAdjustment.ease(to, { | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             duration: WORKSPACE_SWITCH_TIME, | ||||
|         }); | ||||
| @@ -563,11 +506,8 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         for (let i = 0; i < this._workspacesViews.length; i++) | ||||
|             this._workspacesViews[i].startTouchGesture(); | ||||
|  | ||||
|         let monitors = Main.layoutManager.monitors; | ||||
|         let geometry = monitor === this._primaryIndex | ||||
|             ? this._fullGeometry : monitors[monitor]; | ||||
|         let distance = global.workspace_manager.layout_rows === -1 | ||||
|             ? geometry.height : geometry.width; | ||||
|             ? this.height : this.width; | ||||
|  | ||||
|         let progress = adjustment.value / adjustment.page_size; | ||||
|         let points = Array.from( | ||||
| @@ -587,14 +527,13 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         this._clickAction.release(); | ||||
|  | ||||
|         let workspaceManager = global.workspace_manager; | ||||
|         let activeWorkspace = workspaceManager.get_active_workspace(); | ||||
|         let newWs = workspaceManager.get_workspace_by_index(endProgress); | ||||
|  | ||||
|         this._scrollAdjustment.ease(endProgress, { | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_CUBIC, | ||||
|             duration, | ||||
|             onComplete: () => { | ||||
|                 if (newWs !== activeWorkspace) | ||||
|                 if (!newWs.active) | ||||
|                     newWs.activate(global.get_current_time()); | ||||
|                 this._endTouchGesture(); | ||||
|             }, | ||||
| @@ -614,6 +553,7 @@ class WorkspacesDisplay extends St.Widget { | ||||
|     animateToOverview(fadeOnPrimary) { | ||||
|         this.show(); | ||||
|         this._updateWorkspacesViews(); | ||||
|  | ||||
|         for (let i = 0; i < this._workspacesViews.length; i++) { | ||||
|             let animationType; | ||||
|             if (fadeOnPrimary && i == this._primaryIndex) | ||||
| @@ -623,6 +563,11 @@ class WorkspacesDisplay extends St.Widget { | ||||
|             this._workspacesViews[i].animateToOverview(animationType); | ||||
|         } | ||||
|  | ||||
|         this._inWindowFade = fadeOnPrimary; | ||||
|  | ||||
|         if (this._actualGeometry && !fadeOnPrimary) | ||||
|             this._syncWorkspacesActualGeometry(); | ||||
|  | ||||
|         this._restackedNotifyId = | ||||
|             Main.overview.connect('windows-restacked', | ||||
|                                   this._onRestacked.bind(this)); | ||||
| @@ -642,6 +587,17 @@ class WorkspacesDisplay extends St.Widget { | ||||
|                 animationType = AnimationType.ZOOM; | ||||
|             this._workspacesViews[i].animateFromOverview(animationType); | ||||
|         } | ||||
|  | ||||
|         this._inWindowFade = fadeOnPrimary; | ||||
|  | ||||
|         const { primaryIndex } = Main.layoutManager; | ||||
|         const { x, y, width, height } = | ||||
|             Main.layoutManager.getWorkAreaForMonitor(primaryIndex); | ||||
|         this._getPrimaryView().ease({ | ||||
|             x, y, width, height, | ||||
|             duration: fadeOnPrimary ? 0 : ANIMATION_TIME, | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_QUAD, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     vfunc_hide() { | ||||
| @@ -671,6 +627,7 @@ class WorkspacesDisplay extends St.Widget { | ||||
|             return; | ||||
|  | ||||
|         this._updateWorkspacesViews(); | ||||
|         this._syncWorkspacesActualGeometry(); | ||||
|     } | ||||
|  | ||||
|     _updateWorkspacesViews() { | ||||
| @@ -687,17 +644,9 @@ class WorkspacesDisplay extends St.Widget { | ||||
|             else | ||||
|                 view = new WorkspacesView(i, this._scrollAdjustment); | ||||
|  | ||||
|             // HACK: Avoid spurious allocation changes while updating views | ||||
|             view.hide(); | ||||
|  | ||||
|             this._workspacesViews.push(view); | ||||
|             Main.layoutManager.overviewGroup.add_actor(view); | ||||
|         } | ||||
|  | ||||
|         this._workspacesViews.forEach(v => v.show()); | ||||
|  | ||||
|         this._updateWorkspacesFullGeometry(); | ||||
|         this._updateWorkspacesActualGeometry(); | ||||
|     } | ||||
|  | ||||
|     _getMonitorIndexForEvent(event) { | ||||
| @@ -744,40 +693,35 @@ class WorkspacesDisplay extends St.Widget { | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     // This geometry should always be the fullest geometry | ||||
|     // the workspaces switcher can ever be allocated, as if | ||||
|     // the sliding controls were never slid in at all. | ||||
|     setWorkspacesFullGeometry(geom) { | ||||
|         this._fullGeometry = geom; | ||||
|         this._updateWorkspacesFullGeometry(); | ||||
|     } | ||||
|  | ||||
|     _updateWorkspacesFullGeometry() { | ||||
|         if (!this._workspacesViews.length) | ||||
|             return; | ||||
|  | ||||
|         let monitors = Main.layoutManager.monitors; | ||||
|         for (let i = 0; i < monitors.length; i++) { | ||||
|             let geometry = i == this._primaryIndex ? this._fullGeometry : monitors[i]; | ||||
|             this._workspacesViews[i].setFullGeometry(geometry); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     _updateWorkspacesActualGeometry() { | ||||
|         if (!this._workspacesViews.length) | ||||
|         const [x, y] = this.get_transformed_position(); | ||||
|         const width = this.allocation.get_width(); | ||||
|         const height = this.allocation.get_height(); | ||||
|  | ||||
|         this._actualGeometry = { x, y, width, height }; | ||||
|  | ||||
|         if (this._syncActualGeometryLater > 0) | ||||
|             return; | ||||
|  | ||||
|         let [x, y] = this.get_transformed_position(); | ||||
|         let allocation = this.allocation; | ||||
|         let width = allocation.x2 - allocation.x1; | ||||
|         let height = allocation.y2 - allocation.y1; | ||||
|         let primaryGeometry = { x, y, width, height }; | ||||
|         this._syncActualGeometryLater = | ||||
|             Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => { | ||||
|                 this._syncWorkspacesActualGeometry(); | ||||
|  | ||||
|         let monitors = Main.layoutManager.monitors; | ||||
|         for (let i = 0; i < monitors.length; i++) { | ||||
|             let geometry = i == this._primaryIndex ? primaryGeometry : monitors[i]; | ||||
|             this._workspacesViews[i].setActualGeometry(geometry); | ||||
|         } | ||||
|                 this._syncActualGeometryLater = 0; | ||||
|                 return GLib.SOURCE_REMOVE; | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     _syncWorkspacesActualGeometry() { | ||||
|         const primaryView = this._getPrimaryView(); | ||||
|         if (!primaryView || this._inWindowFade) | ||||
|             return; | ||||
|  | ||||
|         primaryView.ease({ | ||||
|             ...this._actualGeometry, | ||||
|             duration: Main.overview.animationInProgress ? ANIMATION_TIME : 0, | ||||
|             mode: Clutter.AnimationMode.EASE_OUT_QUAD, | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     _onRestacked(overview, stackIndices) { | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| project('gnome-shell', 'c', | ||||
|   version: '3.37.2', | ||||
|   version: '3.37.3', | ||||
|   meson_version: '>= 0.53.0', | ||||
|   license: 'GPLv2+' | ||||
| ) | ||||
| @@ -25,7 +25,7 @@ gio_req = '>= 2.56.0' | ||||
| gi_req = '>= 1.49.1' | ||||
| gjs_req = '>= 1.65.1' | ||||
| gtk_req = '>= 3.15.0' | ||||
| mutter_req = '>= 3.37.2' | ||||
| mutter_req = '>= 3.37.3' | ||||
| polkit_req = '>= 0.100' | ||||
| schemas_req = '>= 3.33.1' | ||||
| startup_req = '>= 0.11' | ||||
|   | ||||
							
								
								
									
										896
									
								
								po/pt_BR.po
									
									
									
									
									
								
							
							
						
						
									
										896
									
								
								po/pt_BR.po
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										378
									
								
								po/ro.po
									
									
									
									
									
								
							
							
						
						
									
										378
									
								
								po/ro.po
									
									
									
									
									
								
							| @@ -10,9 +10,10 @@ msgid "" | ||||
| msgstr "" | ||||
| "Project-Id-Version: gnome-shell master\n" | ||||
| "Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n" | ||||
| "POT-Creation-Date: 2020-05-21 23:22+0000\n" | ||||
| "PO-Revision-Date: 2020-05-22 15:07+0300\n" | ||||
| "Last-Translator: Florentina Mușat <emryslokadottir [at] gmail [dot] com>\n" | ||||
| "POT-Creation-Date: 2020-07-21 05:44+0000\n" | ||||
| "PO-Revision-Date: 2020-07-21 09:07+0200\n" | ||||
| "Last-Translator: Florentina Mușat <florentina [dot] musat [dot] 28 [at] " | ||||
| "gmail [dot] com>\n" | ||||
| "Language-Team: Gnome Romanian Translation Team <gnomero-list@lists." | ||||
| "sourceforge.net>\n" | ||||
| "Language: ro\n" | ||||
| @@ -223,95 +224,112 @@ msgid "" | ||||
| msgstr "" | ||||
| "Activează un API D-Bus care permite introspecția stării aplicației shell." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:119 | ||||
| #: data/org.gnome.shell.gschema.xml.in:114 | ||||
| msgid "Layout of the app picker" | ||||
| msgstr "Aspectul selectorului de aplicații" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:115 | ||||
| msgid "" | ||||
| "Layout of the app picker. Each entry in the array is a page. Pages are " | ||||
| "stored in the order they appear in GNOME Shell. Each page contains an " | ||||
| "“application id” → 'data' pair. Currently, the following values are stored " | ||||
| "as 'data': • “position”: the position of the application icon in the page" | ||||
| msgstr "" | ||||
| "Aspectul selectorului de aplicații. Fiecare intrare din matrice este o " | ||||
| "pagină. Paginile sunt stocate în ordinea în care apar în GNOME Shell. " | ||||
| "Fiecare pagină conține o pereche de „id aplicație” → „date”. Momentan, " | ||||
| "următoarele valori sunt stocate ca „data”: • „poziție”: poziția iconiței " | ||||
| "aplicației în pagină." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:130 | ||||
| msgid "Keybinding to open the application menu" | ||||
| msgstr "Combinație de taste pentru deschiderea meniului aplicației" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:120 | ||||
| #: data/org.gnome.shell.gschema.xml.in:131 | ||||
| msgid "Keybinding to open the application menu." | ||||
| msgstr "Combinație de taste pentru deschiderea meniului aplicației." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:126 | ||||
| #: data/org.gnome.shell.gschema.xml.in:137 | ||||
| msgid "Keybinding to open the “Show Applications” view" | ||||
| msgstr "" | ||||
| "Combinație de taste pentru deschiderea modului de afișare „Arată aplicațiile”" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:127 | ||||
| #: data/org.gnome.shell.gschema.xml.in:138 | ||||
| msgid "" | ||||
| "Keybinding to open the “Show Applications” view of the Activities Overview." | ||||
| msgstr "" | ||||
| "Combinație de taste pentru deschiderea modului de afișare „Arată " | ||||
| "aplicațiile” a prezentării generale a activităților." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:134 | ||||
| #: data/org.gnome.shell.gschema.xml.in:145 | ||||
| msgid "Keybinding to open the overview" | ||||
| msgstr "Combinație de taste pentru deschiderea prezentării generale" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:135 | ||||
| #: data/org.gnome.shell.gschema.xml.in:146 | ||||
| msgid "Keybinding to open the Activities Overview." | ||||
| msgstr "" | ||||
| "Combinație de taste pentru deschiderea prezentării generale a activităților." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:141 | ||||
| #: data/org.gnome.shell.gschema.xml.in:152 | ||||
| msgid "Keybinding to toggle the visibility of the notification list" | ||||
| msgstr "" | ||||
| "Combinație de taste pentru comutarea vizibilității listei de notificare" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:142 | ||||
| #: data/org.gnome.shell.gschema.xml.in:153 | ||||
| msgid "Keybinding to toggle the visibility of the notification list." | ||||
| msgstr "" | ||||
| "Combinație de taste pentru comutarea vizibilității listei de notificare." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:148 | ||||
| #: data/org.gnome.shell.gschema.xml.in:159 | ||||
| msgid "Keybinding to focus the active notification" | ||||
| msgstr "Combinație de taste pentru focalizarea notificării active" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:149 | ||||
| #: data/org.gnome.shell.gschema.xml.in:160 | ||||
| msgid "Keybinding to focus the active notification." | ||||
| msgstr "Combinație de taste pentru focalizarea notificării active." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:155 | ||||
| #: data/org.gnome.shell.gschema.xml.in:166 | ||||
| msgid "Switch to application 1" | ||||
| msgstr "Comută la aplicația 1" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:159 | ||||
| #: data/org.gnome.shell.gschema.xml.in:170 | ||||
| msgid "Switch to application 2" | ||||
| msgstr "Comută la aplicația 2" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:163 | ||||
| #: data/org.gnome.shell.gschema.xml.in:174 | ||||
| msgid "Switch to application 3" | ||||
| msgstr "Comută la aplicația 3" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:167 | ||||
| #: data/org.gnome.shell.gschema.xml.in:178 | ||||
| msgid "Switch to application 4" | ||||
| msgstr "Comută la aplicația 4" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:171 | ||||
| #: data/org.gnome.shell.gschema.xml.in:182 | ||||
| msgid "Switch to application 5" | ||||
| msgstr "Comută la aplicația 5" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:175 | ||||
| #: data/org.gnome.shell.gschema.xml.in:186 | ||||
| msgid "Switch to application 6" | ||||
| msgstr "Comută la aplicația 6" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:179 | ||||
| #: data/org.gnome.shell.gschema.xml.in:190 | ||||
| msgid "Switch to application 7" | ||||
| msgstr "Comută la aplicația 7" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:183 | ||||
| #: data/org.gnome.shell.gschema.xml.in:194 | ||||
| msgid "Switch to application 8" | ||||
| msgstr "Comută la aplicația 8" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:187 | ||||
| #: data/org.gnome.shell.gschema.xml.in:198 | ||||
| msgid "Switch to application 9" | ||||
| msgstr "Comută la aplicația 9" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:196 | ||||
| #: data/org.gnome.shell.gschema.xml.in:223 | ||||
| #: data/org.gnome.shell.gschema.xml.in:207 | ||||
| #: data/org.gnome.shell.gschema.xml.in:234 | ||||
| msgid "Limit switcher to current workspace." | ||||
| msgstr "Limitează comutatorul la spațiul de lucru curent." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:197 | ||||
| #: data/org.gnome.shell.gschema.xml.in:208 | ||||
| msgid "" | ||||
| "If true, only applications that have windows on the current workspace are " | ||||
| "shown in the switcher. Otherwise, all applications are included." | ||||
| @@ -319,11 +337,11 @@ msgstr "" | ||||
| "Dacă este activat, doar aplicațiile care au ferestre în spațiul de lucru " | ||||
| "curent sunt arătate în comutator. Altfel, toate aplicațiile sunt incluse." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:214 | ||||
| #: data/org.gnome.shell.gschema.xml.in:225 | ||||
| msgid "The application icon mode." | ||||
| msgstr "Miniatură și pictograma aplicației." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:215 | ||||
| #: data/org.gnome.shell.gschema.xml.in:226 | ||||
| msgid "" | ||||
| "Configures how the windows are shown in the switcher. Valid possibilities " | ||||
| "are “thumbnail-only” (shows a thumbnail of the window), “app-icon-" | ||||
| @@ -333,7 +351,7 @@ msgstr "" | ||||
| "Posibilități valide sunt „mod miniatură” (arată o miniatură a ferestrei) " | ||||
| "„mod iconiță” (arată doar iconița aplicației) sau „ambele”." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:224 | ||||
| #: data/org.gnome.shell.gschema.xml.in:235 | ||||
| msgid "" | ||||
| "If true, only windows from the current workspace are shown in the switcher. " | ||||
| "Otherwise, all windows are included." | ||||
| @@ -341,59 +359,59 @@ msgstr "" | ||||
| "Dacă este activat, doar ferestrele din spațiul de lucru curent sunt arătate " | ||||
| "în comutator. Altfel, toate ferestrele sunt incluse." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:234 | ||||
| #: data/org.gnome.shell.gschema.xml.in:245 | ||||
| msgid "Locations" | ||||
| msgstr "Locații" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:235 | ||||
| #: data/org.gnome.shell.gschema.xml.in:246 | ||||
| msgid "The locations to show in world clocks" | ||||
| msgstr "Locațiile de arătat în ceasuri globale" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:245 | ||||
| #: data/org.gnome.shell.gschema.xml.in:256 | ||||
| msgid "Automatic location" | ||||
| msgstr "Locație automată" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:246 | ||||
| #: data/org.gnome.shell.gschema.xml.in:257 | ||||
| msgid "Whether to fetch the current location or not" | ||||
| msgstr "Dacă să se obțină locația curentă" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:253 | ||||
| #: data/org.gnome.shell.gschema.xml.in:264 | ||||
| msgid "Location" | ||||
| msgstr "Locație" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:254 | ||||
| #: data/org.gnome.shell.gschema.xml.in:265 | ||||
| msgid "The location for which to show a forecast" | ||||
| msgstr "Locația pentru care să se arate o prognoză" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:266 | ||||
| #: data/org.gnome.shell.gschema.xml.in:277 | ||||
| msgid "Attach modal dialog to the parent window" | ||||
| msgstr "Atașează dialogul modal la fereastra părinte" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:267 | ||||
| #: data/org.gnome.shell.gschema.xml.in:276 | ||||
| #: data/org.gnome.shell.gschema.xml.in:284 | ||||
| #: data/org.gnome.shell.gschema.xml.in:292 | ||||
| #: data/org.gnome.shell.gschema.xml.in:300 | ||||
| #: data/org.gnome.shell.gschema.xml.in:278 | ||||
| #: data/org.gnome.shell.gschema.xml.in:287 | ||||
| #: data/org.gnome.shell.gschema.xml.in:295 | ||||
| #: data/org.gnome.shell.gschema.xml.in:303 | ||||
| #: data/org.gnome.shell.gschema.xml.in:311 | ||||
| msgid "" | ||||
| "This key overrides the key in org.gnome.mutter when running GNOME Shell." | ||||
| msgstr "" | ||||
| "Această cheie suprascrie cheia corespondentă din org.gnome.mutter când " | ||||
| "Vizualizatorul activităților GNOME rulează." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:275 | ||||
| #: data/org.gnome.shell.gschema.xml.in:286 | ||||
| msgid "Enable edge tiling when dropping windows on screen edges" | ||||
| msgstr "" | ||||
| "Activează mozaic lateral la plasarea ferestrelor pe marginile ecranului" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:283 | ||||
| #: data/org.gnome.shell.gschema.xml.in:294 | ||||
| msgid "Workspaces are managed dynamically" | ||||
| msgstr "Spațiile de lucru sunt gestionate în mod dinamic" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:291 | ||||
| #: data/org.gnome.shell.gschema.xml.in:302 | ||||
| msgid "Workspaces only on primary monitor" | ||||
| msgstr "Spații de lucru doar pe monitorul principal" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:299 | ||||
| #: data/org.gnome.shell.gschema.xml.in:310 | ||||
| msgid "Delay focus changes in mouse mode until the pointer stops moving" | ||||
| msgstr "" | ||||
| "Întârzie schimbările de focalizare în maus până când cursorul încetează să " | ||||
| @@ -432,7 +450,7 @@ msgstr "Vizitează pagina principală a extensiei" | ||||
|  | ||||
| #: js/gdm/authPrompt.js:135 js/ui/audioDeviceSelection.js:57 | ||||
| #: js/ui/components/networkAgent.js:110 js/ui/components/polkitAgent.js:139 | ||||
| #: js/ui/endSessionDialog.js:369 js/ui/extensionDownloader.js:181 | ||||
| #: js/ui/endSessionDialog.js:369 js/ui/extensionDownloader.js:183 | ||||
| #: js/ui/shellMountOperation.js:376 js/ui/shellMountOperation.js:386 | ||||
| #: js/ui/status/network.js:916 subprojects/extensions-app/js/main.js:149 | ||||
| msgid "Cancel" | ||||
| @@ -474,7 +492,7 @@ msgstr "Nume de utilizator" | ||||
| msgid "Login Window" | ||||
| msgstr "Fereastră de autentificare" | ||||
|  | ||||
| #: js/gdm/util.js:345 | ||||
| #: js/gdm/util.js:355 | ||||
| msgid "Authentication error" | ||||
| msgstr "Eroare de autentificare" | ||||
|  | ||||
| @@ -483,7 +501,7 @@ msgstr "Eroare de autentificare" | ||||
| #. as a cue to display our own message. | ||||
| #. Translators: this message is shown below the password entry field | ||||
| #. to indicate the user can swipe their finger instead | ||||
| #: js/gdm/util.js:471 | ||||
| #: js/gdm/util.js:481 | ||||
| msgid "(or swipe finger)" | ||||
| msgstr "(sau treceți degetul peste)" | ||||
|  | ||||
| @@ -594,7 +612,7 @@ msgstr[0] "%d oră în urmă" | ||||
| msgstr[1] "%d ore în urmă" | ||||
| msgstr[2] "%d de ore în urmă" | ||||
|  | ||||
| #: js/misc/util.js:191 | ||||
| #: js/misc/util.js:191 js/ui/dateMenu.js:162 | ||||
| msgid "Yesterday" | ||||
| msgstr "Ieri" | ||||
|  | ||||
| @@ -718,56 +736,44 @@ msgstr "" | ||||
|  | ||||
| #. No support for non-modal system dialogs, so ignore the option | ||||
| #. let modal = options['modal'] || true; | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:374 | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:369 | ||||
| msgid "Deny Access" | ||||
| msgstr "Refuză accesul" | ||||
|  | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:377 | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:372 | ||||
| msgid "Grant Access" | ||||
| msgstr "Permite accesul" | ||||
|  | ||||
| #: js/ui/appDisplay.js:944 | ||||
| #: js/ui/appDisplay.js:1297 | ||||
| msgid "Unnamed Folder" | ||||
| msgstr "Dosar nedenumit" | ||||
|  | ||||
| #: js/ui/appDisplay.js:967 | ||||
| msgid "Frequently used applications will appear here" | ||||
| msgstr "Aplicațiile utilizate frecvent vor fi afișate aici" | ||||
|  | ||||
| #: js/ui/appDisplay.js:1102 | ||||
| msgid "Frequent" | ||||
| msgstr "Frecvente" | ||||
|  | ||||
| #: js/ui/appDisplay.js:1109 | ||||
| msgid "All" | ||||
| msgstr "Toate" | ||||
|  | ||||
| #. Translators: This is the heading of a list of open windows | ||||
| #: js/ui/appDisplay.js:2486 js/ui/panel.js:75 | ||||
| #: js/ui/appDisplay.js:2767 js/ui/panel.js:75 | ||||
| msgid "Open Windows" | ||||
| msgstr "Ferestre deschise" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2505 js/ui/panel.js:82 | ||||
| #: js/ui/appDisplay.js:2786 js/ui/panel.js:82 | ||||
| msgid "New Window" | ||||
| msgstr "Fereastră nouă" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2521 | ||||
| #: js/ui/appDisplay.js:2802 | ||||
| msgid "Launch using Integrated Graphics Card" | ||||
| msgstr "Lansează folosind placa grafică integrată" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2522 | ||||
| #: js/ui/appDisplay.js:2803 | ||||
| msgid "Launch using Discrete Graphics Card" | ||||
| msgstr "Lansează folosind placa grafică discretă" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2550 js/ui/dash.js:239 | ||||
| #: js/ui/appDisplay.js:2831 js/ui/dash.js:239 | ||||
| msgid "Remove from Favorites" | ||||
| msgstr "Elimină din favorite" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2556 | ||||
| #: js/ui/appDisplay.js:2837 | ||||
| msgid "Add to Favorites" | ||||
| msgstr "Adaugă la Favorite" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2566 js/ui/panel.js:93 | ||||
| #: js/ui/appDisplay.js:2847 js/ui/panel.js:93 | ||||
| msgid "Show Details" | ||||
| msgstr "Arată detaliile" | ||||
|  | ||||
| @@ -797,7 +803,7 @@ msgstr "Căști auriculare" | ||||
| msgid "Headset" | ||||
| msgstr "Căști cu microfon" | ||||
|  | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:273 | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:272 | ||||
| msgid "Microphone" | ||||
| msgstr "Microfon" | ||||
|  | ||||
| @@ -814,7 +820,7 @@ msgid "Settings" | ||||
| msgstr "Configurări" | ||||
|  | ||||
| #. Translators: Enter 0-6 (Sunday-Saturday) for non-work days. Examples: "0" (Sunday) "6" (Saturday) "06" (Sunday and Saturday). | ||||
| #: js/ui/calendar.js:41 | ||||
| #: js/ui/calendar.js:36 | ||||
| msgctxt "calendar-no-work" | ||||
| msgid "06" | ||||
| msgstr "06" | ||||
| @@ -824,43 +830,43 @@ msgstr "06" | ||||
| #. * NOTE: These grid abbreviations are always shown together | ||||
| #. * and in order, e.g. "S M T W T F S". | ||||
| #. | ||||
| #: js/ui/calendar.js:70 | ||||
| #: js/ui/calendar.js:65 | ||||
| msgctxt "grid sunday" | ||||
| msgid "S" | ||||
| msgstr "D" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Monday | ||||
| #: js/ui/calendar.js:72 | ||||
| #: js/ui/calendar.js:67 | ||||
| msgctxt "grid monday" | ||||
| msgid "M" | ||||
| msgstr "L" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Tuesday | ||||
| #: js/ui/calendar.js:74 | ||||
| #: js/ui/calendar.js:69 | ||||
| msgctxt "grid tuesday" | ||||
| msgid "T" | ||||
| msgstr "M" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Wednesday | ||||
| #: js/ui/calendar.js:76 | ||||
| #: js/ui/calendar.js:71 | ||||
| msgctxt "grid wednesday" | ||||
| msgid "W" | ||||
| msgstr "M" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Thursday | ||||
| #: js/ui/calendar.js:78 | ||||
| #: js/ui/calendar.js:73 | ||||
| msgctxt "grid thursday" | ||||
| msgid "T" | ||||
| msgstr "J" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Friday | ||||
| #: js/ui/calendar.js:80 | ||||
| #: js/ui/calendar.js:75 | ||||
| msgctxt "grid friday" | ||||
| msgid "F" | ||||
| msgstr "V" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Saturday | ||||
| #: js/ui/calendar.js:82 | ||||
| #: js/ui/calendar.js:77 | ||||
| msgctxt "grid saturday" | ||||
| msgid "S" | ||||
| msgstr "S" | ||||
| @@ -871,7 +877,7 @@ msgstr "S" | ||||
| #. * "%OB" is the new format specifier introduced in glibc 2.27, | ||||
| #. * in most cases you should not change it. | ||||
| #. | ||||
| #: js/ui/calendar.js:397 | ||||
| #: js/ui/calendar.js:392 | ||||
| msgid "%OB" | ||||
| msgstr "%OB" | ||||
|  | ||||
| @@ -884,61 +890,37 @@ msgstr "%OB" | ||||
| #. * in most cases you should not use the old "%B" here unless you | ||||
| #. * absolutely know what you are doing. | ||||
| #. | ||||
| #: js/ui/calendar.js:407 | ||||
| #: js/ui/calendar.js:402 | ||||
| msgid "%OB %Y" | ||||
| msgstr "%OB %Y" | ||||
|  | ||||
| #: js/ui/calendar.js:466 | ||||
| #: js/ui/calendar.js:461 | ||||
| msgid "Previous month" | ||||
| msgstr "Luna precedentă" | ||||
|  | ||||
| #: js/ui/calendar.js:481 | ||||
| #: js/ui/calendar.js:476 | ||||
| msgid "Next month" | ||||
| msgstr "Luna viitoare" | ||||
|  | ||||
| #: js/ui/calendar.js:631 | ||||
| #: js/ui/calendar.js:626 | ||||
| #, no-javascript-format | ||||
| msgctxt "date day number format" | ||||
| msgid "%d" | ||||
| msgstr "%d" | ||||
|  | ||||
| #: js/ui/calendar.js:687 | ||||
| #: js/ui/calendar.js:682 | ||||
| msgid "Week %V" | ||||
| msgstr "Săptămâna %V" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/calendar.js:762 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "Toată ziua" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/calendar.js:900 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d" | ||||
| msgstr "%A, %B %-d" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/calendar.js:903 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d, %Y" | ||||
| msgstr "%A, %B %-d, %Y" | ||||
|  | ||||
| #: js/ui/calendar.js:1133 | ||||
| #: js/ui/calendar.js:895 | ||||
| msgid "No Notifications" | ||||
| msgstr "Nu există notificări" | ||||
|  | ||||
| #: js/ui/calendar.js:1136 | ||||
| msgid "No Events" | ||||
| msgstr "Nu există evenimente" | ||||
|  | ||||
| #: js/ui/calendar.js:1190 | ||||
| #: js/ui/calendar.js:949 | ||||
| msgid "Do Not Disturb" | ||||
| msgstr "Nu deranja" | ||||
|  | ||||
| #: js/ui/calendar.js:1209 | ||||
| #: js/ui/calendar.js:968 | ||||
| msgid "Clear" | ||||
| msgstr "Curăță" | ||||
|  | ||||
| @@ -1093,7 +1075,7 @@ msgstr "Nu a funcționat. Încercați din nou." | ||||
| msgid "%s is now known as %s" | ||||
| msgstr "%s este acum cunoscut ca %s" | ||||
|  | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:177 | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:178 | ||||
| msgid "Windows" | ||||
| msgstr "Ferestre" | ||||
|  | ||||
| @@ -1112,7 +1094,7 @@ msgstr "Dash" | ||||
| #. * "Tue 9:29 AM").  The string itself should become a full date, e.g., | ||||
| #. * "February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:75 | ||||
| #: js/ui/dateMenu.js:79 | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%B %-d %Y" | ||||
|  | ||||
| @@ -1120,35 +1102,67 @@ msgstr "%B %-d %Y" | ||||
| #. * below the time in the shell; it should combine the weekday and the | ||||
| #. * date, e.g. "Tuesday February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:82 | ||||
| #: js/ui/dateMenu.js:86 | ||||
| msgid "%A %B %e %Y" | ||||
| msgstr "%A %B %e %Y" | ||||
|  | ||||
| #: js/ui/dateMenu.js:162 | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/dateMenu.js:151 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d" | ||||
| msgstr "%B %-d" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/dateMenu.js:154 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%B %-d %Y" | ||||
|  | ||||
| #: js/ui/dateMenu.js:160 | ||||
| msgid "Today" | ||||
| msgstr "Astăzi" | ||||
|  | ||||
| #: js/ui/dateMenu.js:164 | ||||
| msgid "Tomorrow" | ||||
| msgstr "Mâine" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/dateMenu.js:180 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "Toată ziua" | ||||
|  | ||||
| #: js/ui/dateMenu.js:231 | ||||
| msgid "No Events" | ||||
| msgstr "Nu există evenimente" | ||||
|  | ||||
| #: js/ui/dateMenu.js:348 | ||||
| msgid "Add world clocks…" | ||||
| msgstr "Adaugă un fus orar…" | ||||
|  | ||||
| #: js/ui/dateMenu.js:163 | ||||
| #: js/ui/dateMenu.js:349 | ||||
| msgid "World Clocks" | ||||
| msgstr "Fusuri orare" | ||||
|  | ||||
| #: js/ui/dateMenu.js:443 | ||||
| #: js/ui/dateMenu.js:629 | ||||
| msgid "Loading…" | ||||
| msgstr "Se încarcă…" | ||||
|  | ||||
| #: js/ui/dateMenu.js:453 | ||||
| #: js/ui/dateMenu.js:639 | ||||
| msgid "Go online for weather information" | ||||
| msgstr "Conectați-vă la internet pentru informații despre vreme" | ||||
|  | ||||
| #: js/ui/dateMenu.js:455 | ||||
| #: js/ui/dateMenu.js:641 | ||||
| msgid "Weather information is currently unavailable" | ||||
| msgstr "Informațiile despre vreme nu sunt disponibile momentan" | ||||
|  | ||||
| #: js/ui/dateMenu.js:465 | ||||
| #: js/ui/dateMenu.js:651 | ||||
| msgid "Weather" | ||||
| msgstr "Vreme" | ||||
|  | ||||
| #: js/ui/dateMenu.js:467 | ||||
| #: js/ui/dateMenu.js:653 | ||||
| msgid "Select weather location…" | ||||
| msgstr "Selectează locația vremii…" | ||||
|  | ||||
| @@ -1303,15 +1317,15 @@ msgstr "%s (la distanță)" | ||||
| msgid "%s (console)" | ||||
| msgstr "%s (consolă)" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:185 | ||||
| #: js/ui/extensionDownloader.js:187 | ||||
| msgid "Install" | ||||
| msgstr "Instalează" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:191 | ||||
| #: js/ui/extensionDownloader.js:193 | ||||
| msgid "Install Extension" | ||||
| msgstr "Instalează extensia" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:192 | ||||
| #: js/ui/extensionDownloader.js:194 | ||||
| #, javascript-format | ||||
| msgid "Download and install “%s” from extensions.gnome.org?" | ||||
| msgstr "Descărcați și instalați „%s” de la extensions.gnome.org?" | ||||
| @@ -1344,11 +1358,11 @@ msgstr "O aplicație vrea să inhibe scurtăturile" | ||||
| msgid "You can restore shortcuts by pressing %s." | ||||
| msgstr "Puteți restaura scurtăturile apăsând %s." | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:98 | ||||
| #: js/ui/inhibitShortcutsDialog.js:100 | ||||
| msgid "Deny" | ||||
| msgstr "Refuză" | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:105 | ||||
| #: js/ui/inhibitShortcutsDialog.js:107 | ||||
| msgid "Allow" | ||||
| msgstr "Permite" | ||||
|  | ||||
| @@ -1417,7 +1431,7 @@ msgstr "Oprește" | ||||
| msgid "Leave Off" | ||||
| msgstr "Lasă oprit" | ||||
|  | ||||
| #: js/ui/keyboard.js:207 | ||||
| #: js/ui/keyboard.js:225 | ||||
| msgid "Region & Language Settings" | ||||
| msgstr "Configurări de regiune și limbă" | ||||
|  | ||||
| @@ -1491,25 +1505,25 @@ msgstr "Blocarea ecranului este dezactivată" | ||||
| msgid "Screen Locking requires the GNOME display manager." | ||||
| msgstr "Blocarea ecranului necesită administratorul de afișaj GNOME." | ||||
|  | ||||
| #: js/ui/messageTray.js:1547 | ||||
| #: js/ui/messageTray.js:1476 | ||||
| msgid "System Information" | ||||
| msgstr "Informații despre sistem" | ||||
|  | ||||
| #: js/ui/mpris.js:204 | ||||
| #: js/ui/mpris.js:203 | ||||
| msgid "Unknown artist" | ||||
| msgstr "Artist necunoscut" | ||||
|  | ||||
| #: js/ui/mpris.js:214 | ||||
| #: js/ui/mpris.js:213 | ||||
| msgid "Unknown title" | ||||
| msgstr "Titlu necunoscut" | ||||
|  | ||||
| #: js/ui/overview.js:73 | ||||
| #: js/ui/overview.js:74 | ||||
| msgid "Undo" | ||||
| msgstr "Anulează" | ||||
|  | ||||
| #. Translators: This is the main view to select | ||||
| #. activities. See also note for "Activities" string. | ||||
| #: js/ui/overview.js:86 | ||||
| #: js/ui/overview.js:87 | ||||
| msgid "Overview" | ||||
| msgstr "Prezentare generală" | ||||
|  | ||||
| @@ -1517,7 +1531,7 @@ msgstr "Prezentare generală" | ||||
| #. in the search entry when no search is | ||||
| #. active; it should not exceed ~30 | ||||
| #. characters. | ||||
| #: js/ui/overview.js:107 | ||||
| #: js/ui/overview.js:108 | ||||
| msgid "Type to search" | ||||
| msgstr "Tastați pentru a căuta" | ||||
|  | ||||
| @@ -1545,23 +1559,23 @@ msgstr "Alochează tasta apăsată" | ||||
| msgid "Done" | ||||
| msgstr "Gata" | ||||
|  | ||||
| #: js/ui/padOsd.js:745 | ||||
| #: js/ui/padOsd.js:732 | ||||
| msgid "Edit…" | ||||
| msgstr "Editează…" | ||||
|  | ||||
| #: js/ui/padOsd.js:787 js/ui/padOsd.js:910 | ||||
| #: js/ui/padOsd.js:774 js/ui/padOsd.js:891 | ||||
| msgid "None" | ||||
| msgstr "Nespecificat" | ||||
|  | ||||
| #: js/ui/padOsd.js:863 | ||||
| #: js/ui/padOsd.js:845 | ||||
| msgid "Press a button to configure" | ||||
| msgstr "Apăsați un buton de configurat" | ||||
|  | ||||
| #: js/ui/padOsd.js:864 | ||||
| #: js/ui/padOsd.js:846 | ||||
| msgid "Press Esc to exit" | ||||
| msgstr "Apăsați Esc pentru a ieși" | ||||
|  | ||||
| #: js/ui/padOsd.js:867 | ||||
| #: js/ui/padOsd.js:849 | ||||
| msgid "Press any key to exit" | ||||
| msgstr "Apăsați orice tastă pentru a ieși" | ||||
|  | ||||
| @@ -1826,17 +1840,17 @@ msgstr "Locația dezactivată" | ||||
| msgid "Enable" | ||||
| msgstr "Activează" | ||||
|  | ||||
| #: js/ui/status/location.js:355 | ||||
| #: js/ui/status/location.js:350 | ||||
| msgid "Allow location access" | ||||
| msgstr "Permite accesul la locație" | ||||
|  | ||||
| #. Translators: %s is an application name | ||||
| #: js/ui/status/location.js:357 | ||||
| #: js/ui/status/location.js:352 | ||||
| #, javascript-format | ||||
| msgid "The app %s wants to access your location" | ||||
| msgstr "Aplicația %s dorește să vă acceseze locația" | ||||
|  | ||||
| #: js/ui/status/location.js:367 | ||||
| #: js/ui/status/location.js:362 | ||||
| msgid "Location access can be changed at any time from the privacy settings." | ||||
| msgstr "" | ||||
| "Accesul la locație poate fi schimbat oricând din configurările de " | ||||
| @@ -2175,11 +2189,11 @@ msgstr "Eroare de autorizare Thunderbolt" | ||||
| msgid "Could not authorize the Thunderbolt device: %s" | ||||
| msgstr "Nu s-a putut autoriza dispozitivul Thunderbolt: %s" | ||||
|  | ||||
| #: js/ui/status/volume.js:154 | ||||
| #: js/ui/status/volume.js:155 | ||||
| msgid "Volume changed" | ||||
| msgstr "Volumul a fost schimbat" | ||||
|  | ||||
| #: js/ui/status/volume.js:225 | ||||
| #: js/ui/status/volume.js:217 | ||||
| msgid "Volume" | ||||
| msgstr "Volum" | ||||
|  | ||||
| @@ -2225,19 +2239,19 @@ msgstr "Glisați în sus pentru a debloca" | ||||
| msgid "Click or press a key to unlock" | ||||
| msgstr "Apăsați clic sau o tastă pentru a debloca" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:550 | ||||
| #: js/ui/unlockDialog.js:555 | ||||
| msgid "Unlock Window" | ||||
| msgstr "Deblochează fereastră" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:559 | ||||
| #: js/ui/unlockDialog.js:564 | ||||
| msgid "Log in as another user" | ||||
| msgstr "Intră în sesiune ca utilizator diferit" | ||||
|  | ||||
| #: js/ui/viewSelector.js:181 | ||||
| #: js/ui/viewSelector.js:182 | ||||
| msgid "Applications" | ||||
| msgstr "Aplicații" | ||||
|  | ||||
| #: js/ui/viewSelector.js:185 | ||||
| #: js/ui/viewSelector.js:186 | ||||
| msgid "Search" | ||||
| msgstr "Caută" | ||||
|  | ||||
| @@ -2365,12 +2379,12 @@ msgstr "Utilizează un mod specific, de exemplu „gdm” pentru ecranul de loga | ||||
| msgid "List possible modes" | ||||
| msgstr "Enumeră câmpurile care pot fi afișate" | ||||
|  | ||||
| #: src/shell-app.c:286 | ||||
| #: src/shell-app.c:268 | ||||
| msgctxt "program" | ||||
| msgid "Unknown" | ||||
| msgstr "Necunoscut" | ||||
|  | ||||
| #: src/shell-app.c:537 | ||||
| #: src/shell-app.c:519 | ||||
| #, c-format | ||||
| msgid "Failed to launch “%s”" | ||||
| msgstr "Nu s-a putut lansa „%s”" | ||||
| @@ -2434,7 +2448,8 @@ msgstr "Elimină" | ||||
|  | ||||
| #: subprojects/extensions-app/js/main.js:182 | ||||
| msgid "translator-credits" | ||||
| msgstr "Florentina Mușat <emryslokadottir [at] gmail [dot] com>, 2020" | ||||
| msgstr "" | ||||
| "Florentina Mușat <florentina [dot] musat [dot] 28 [at] gmail [dot] com>, 2020" | ||||
|  | ||||
| #: subprojects/extensions-app/js/main.js:314 | ||||
| #, javascript-format | ||||
| @@ -2453,7 +2468,7 @@ msgid "The extension had an error" | ||||
| msgstr "Extensia a avut o eroare" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:109 | ||||
| #: subprojects/extensions-tool/src/command-create.c:301 | ||||
| #: subprojects/extensions-tool/src/command-create.c:325 | ||||
| #: subprojects/extensions-tool/src/main.c:241 | ||||
| msgid "Description" | ||||
| msgstr "Descriere" | ||||
| @@ -2534,12 +2549,12 @@ msgid "Log Out…" | ||||
| msgstr "Deautentificare…" | ||||
|  | ||||
| #. Translators: a file path to an extension directory | ||||
| #: subprojects/extensions-tool/src/command-create.c:202 | ||||
| #: subprojects/extensions-tool/src/command-create.c:226 | ||||
| #, c-format | ||||
| msgid "The new extension was successfully created in %s.\n" | ||||
| msgstr "Extensia nouă a fost creată cu succes la %s.\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:275 | ||||
| #: subprojects/extensions-tool/src/command-create.c:299 | ||||
| #, c-format | ||||
| msgid "" | ||||
| "Name should be a very short (ideally descriptive) string.\n" | ||||
| @@ -2548,12 +2563,12 @@ msgstr "" | ||||
| "Numele ar trebui să fie un șir foarte scurt (ideal descriptiv).\n" | ||||
| "Exemple: %s" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:281 | ||||
| #: subprojects/extensions-tool/src/command-create.c:305 | ||||
| #: subprojects/extensions-tool/src/main.c:238 | ||||
| msgid "Name" | ||||
| msgstr "Nume" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:295 | ||||
| #: subprojects/extensions-tool/src/command-create.c:319 | ||||
| #, c-format | ||||
| msgid "" | ||||
| "Description is a single-sentence explanation of what your extension does.\n" | ||||
| @@ -2563,7 +2578,7 @@ msgstr "" | ||||
| "extensia.\n" | ||||
| "Exemple: %s" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:315 | ||||
| #: subprojects/extensions-tool/src/command-create.c:339 | ||||
| msgid "" | ||||
| "UUID is a globally-unique identifier for your extension.\n" | ||||
| "This should be in the format of an email address (clicktofocus@janedoe." | ||||
| @@ -2573,56 +2588,56 @@ msgstr "" | ||||
| "Aceasta ar trebui să fie în formatul unei adrese de email " | ||||
| "(clicktofocus@janedoe.example.com)\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:342 | ||||
| #: subprojects/extensions-tool/src/command-create.c:366 | ||||
| msgid "Choose one of the available templates:\n" | ||||
| msgstr "Alege unul din șabloanele existente:\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:356 | ||||
| #: subprojects/extensions-tool/src/command-create.c:380 | ||||
| msgid "Template" | ||||
| msgstr "Șablon" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:411 | ||||
| #: subprojects/extensions-tool/src/command-create.c:435 | ||||
| msgid "The unique identifier of the new extension" | ||||
| msgstr "Identificatorul unic al noii extensii" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:414 | ||||
| #: subprojects/extensions-tool/src/command-create.c:438 | ||||
| msgid "NAME" | ||||
| msgstr "NUME" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:415 | ||||
| #: subprojects/extensions-tool/src/command-create.c:439 | ||||
| msgid "The user-visible name of the new extension" | ||||
| msgstr "Numele vizibil de utilizator pentru noua extensie" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:417 | ||||
| #: subprojects/extensions-tool/src/command-create.c:441 | ||||
| msgid "DESCRIPTION" | ||||
| msgstr "DESCRIERE" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:419 | ||||
| #: subprojects/extensions-tool/src/command-create.c:443 | ||||
| msgid "A short description of what the extension does" | ||||
| msgstr "O scurtă descriere despre ce face extensia" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:422 | ||||
| #: subprojects/extensions-tool/src/command-create.c:446 | ||||
| msgid "TEMPLATE" | ||||
| msgstr "ȘABLON" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:423 | ||||
| #: subprojects/extensions-tool/src/command-create.c:447 | ||||
| msgid "The template to use for the new extension" | ||||
| msgstr "Șablonul de utilizat pentru extensia nouă" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:429 | ||||
| #: subprojects/extensions-tool/src/command-create.c:453 | ||||
| msgid "Enter extension information interactively" | ||||
| msgstr "Introdu informațiile extensiei interactiv" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:437 | ||||
| #: subprojects/extensions-tool/src/command-create.c:461 | ||||
| msgid "Create a new extension" | ||||
| msgstr "Creează o nouă extensie" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:455 | ||||
| #: subprojects/extensions-tool/src/command-create.c:479 | ||||
| #: subprojects/extensions-tool/src/command-list.c:172 | ||||
| msgid "Unknown arguments" | ||||
| msgstr "Argumente necunoscute" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:480 | ||||
| #: subprojects/extensions-tool/src/command-create.c:504 | ||||
| msgid "UUID, name and description are required" | ||||
| msgstr "UUID-ul, numele și descrierea sunt necesare" | ||||
|  | ||||
| @@ -2946,6 +2961,23 @@ msgstr[2] "%u de intrări" | ||||
| msgid "System Sounds" | ||||
| msgstr "Sunete de sistem" | ||||
|  | ||||
| #~ msgid "Frequently used applications will appear here" | ||||
| #~ msgstr "Aplicațiile utilizate frecvent vor fi afișate aici" | ||||
|  | ||||
| #~ msgid "Frequent" | ||||
| #~ msgstr "Frecvente" | ||||
|  | ||||
| #~ msgid "All" | ||||
| #~ msgstr "Toate" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d" | ||||
| #~ msgstr "%A, %B %-d" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d, %Y" | ||||
| #~ msgstr "%A, %B %-d, %Y" | ||||
|  | ||||
| #~ msgid "Copy Error" | ||||
| #~ msgstr "Eroare la copiere" | ||||
|  | ||||
| @@ -3115,12 +3147,6 @@ msgstr "Sunete de sistem" | ||||
| #~ msgid "Nothing Scheduled" | ||||
| #~ msgstr "Nimic programat" | ||||
|  | ||||
| #~ msgid "Today" | ||||
| #~ msgstr "Astăzi" | ||||
|  | ||||
| #~ msgid "Tomorrow" | ||||
| #~ msgstr "Mâine" | ||||
|  | ||||
| #~ msgid "This week" | ||||
| #~ msgstr "Săptămâna aceasta" | ||||
|  | ||||
|   | ||||
							
								
								
									
										267
									
								
								po/tr.po
									
									
									
									
									
								
							
							
						
						
									
										267
									
								
								po/tr.po
									
									
									
									
									
								
							| @@ -17,7 +17,7 @@ msgid "" | ||||
| msgstr "" | ||||
| "Project-Id-Version: gnome-shell\n" | ||||
| "Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n" | ||||
| "POT-Creation-Date: 2020-05-15 18:43+0000\n" | ||||
| "POT-Creation-Date: 2020-06-11 10:28+0000\n" | ||||
| "PO-Revision-Date: 2020-05-16 12:12+0300\n" | ||||
| "Last-Translator: Emin Tufan Çetin <etcetin@gmail.com>\n" | ||||
| "Language-Team: Türkçe <gnome-turk@gnome.org>\n" | ||||
| @@ -586,7 +586,7 @@ msgid "%d hour ago" | ||||
| msgid_plural "%d hours ago" | ||||
| msgstr[0] "%d saat önce" | ||||
|  | ||||
| #: js/misc/util.js:191 | ||||
| #: js/misc/util.js:191 js/ui/dateMenu.js:162 | ||||
| msgid "Yesterday" | ||||
| msgstr "Dün" | ||||
|  | ||||
| @@ -702,56 +702,44 @@ msgstr "" | ||||
|  | ||||
| #. No support for non-modal system dialogs, so ignore the option | ||||
| #. let modal = options['modal'] || true; | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:374 | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:369 | ||||
| msgid "Deny Access" | ||||
| msgstr "Erişimi Reddet" | ||||
|  | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:377 | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:372 | ||||
| msgid "Grant Access" | ||||
| msgstr "Erişime İzin Ver" | ||||
|  | ||||
| #: js/ui/appDisplay.js:944 | ||||
| #: js/ui/appDisplay.js:902 | ||||
| msgid "Unnamed Folder" | ||||
| msgstr "Adsız Klasör" | ||||
|  | ||||
| #: js/ui/appDisplay.js:967 | ||||
| msgid "Frequently used applications will appear here" | ||||
| msgstr "Sık kullanılan uygulamalar burada yer alacak" | ||||
|  | ||||
| #: js/ui/appDisplay.js:1102 | ||||
| msgid "Frequent" | ||||
| msgstr "Sık Sık" | ||||
|  | ||||
| #: js/ui/appDisplay.js:1109 | ||||
| msgid "All" | ||||
| msgstr "Tümü" | ||||
|  | ||||
| #. Translators: This is the heading of a list of open windows | ||||
| #: js/ui/appDisplay.js:2486 js/ui/panel.js:75 | ||||
| #: js/ui/appDisplay.js:2241 js/ui/panel.js:75 | ||||
| msgid "Open Windows" | ||||
| msgstr "Açık Pencereler" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2505 js/ui/panel.js:82 | ||||
| #: js/ui/appDisplay.js:2260 js/ui/panel.js:82 | ||||
| msgid "New Window" | ||||
| msgstr "Yeni Pencere" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2521 | ||||
| #: js/ui/appDisplay.js:2276 | ||||
| msgid "Launch using Integrated Graphics Card" | ||||
| msgstr "Tümleşik Ekran Kartıyla Başlat" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2522 | ||||
| #: js/ui/appDisplay.js:2277 | ||||
| msgid "Launch using Discrete Graphics Card" | ||||
| msgstr "Ayrık Ekran Kartıyla Başlat" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2550 js/ui/dash.js:239 | ||||
| #: js/ui/appDisplay.js:2305 js/ui/dash.js:239 | ||||
| msgid "Remove from Favorites" | ||||
| msgstr "Sık Kullanılanlardan Çıkar" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2556 | ||||
| #: js/ui/appDisplay.js:2311 | ||||
| msgid "Add to Favorites" | ||||
| msgstr "Sık Kullanılanlara Ekle" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2566 js/ui/panel.js:93 | ||||
| #: js/ui/appDisplay.js:2321 js/ui/panel.js:93 | ||||
| msgid "Show Details" | ||||
| msgstr "Ayrıntıları Göster" | ||||
|  | ||||
| @@ -781,7 +769,7 @@ msgstr "Kulaklıklar" | ||||
| msgid "Headset" | ||||
| msgstr "Kulaklıklı Mikrofon" | ||||
|  | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:270 | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:273 | ||||
| msgid "Microphone" | ||||
| msgstr "Mikrofon" | ||||
|  | ||||
| @@ -798,7 +786,7 @@ msgid "Settings" | ||||
| msgstr "Ayarlar" | ||||
|  | ||||
| #. Translators: Enter 0-6 (Sunday-Saturday) for non-work days. Examples: "0" (Sunday) "6" (Saturday) "06" (Sunday and Saturday). | ||||
| #: js/ui/calendar.js:41 | ||||
| #: js/ui/calendar.js:36 | ||||
| msgctxt "calendar-no-work" | ||||
| msgid "06" | ||||
| msgstr "06" | ||||
| @@ -808,43 +796,43 @@ msgstr "06" | ||||
| #. * NOTE: These grid abbreviations are always shown together | ||||
| #. * and in order, e.g. "S M T W T F S". | ||||
| #. | ||||
| #: js/ui/calendar.js:70 | ||||
| #: js/ui/calendar.js:65 | ||||
| msgctxt "grid sunday" | ||||
| msgid "S" | ||||
| msgstr "P" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Monday | ||||
| #: js/ui/calendar.js:72 | ||||
| #: js/ui/calendar.js:67 | ||||
| msgctxt "grid monday" | ||||
| msgid "M" | ||||
| msgstr "P" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Tuesday | ||||
| #: js/ui/calendar.js:74 | ||||
| #: js/ui/calendar.js:69 | ||||
| msgctxt "grid tuesday" | ||||
| msgid "T" | ||||
| msgstr "S" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Wednesday | ||||
| #: js/ui/calendar.js:76 | ||||
| #: js/ui/calendar.js:71 | ||||
| msgctxt "grid wednesday" | ||||
| msgid "W" | ||||
| msgstr "Ç" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Thursday | ||||
| #: js/ui/calendar.js:78 | ||||
| #: js/ui/calendar.js:73 | ||||
| msgctxt "grid thursday" | ||||
| msgid "T" | ||||
| msgstr "P" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Friday | ||||
| #: js/ui/calendar.js:80 | ||||
| #: js/ui/calendar.js:75 | ||||
| msgctxt "grid friday" | ||||
| msgid "F" | ||||
| msgstr "C" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Saturday | ||||
| #: js/ui/calendar.js:82 | ||||
| #: js/ui/calendar.js:77 | ||||
| msgctxt "grid saturday" | ||||
| msgid "S" | ||||
| msgstr "C" | ||||
| @@ -855,7 +843,7 @@ msgstr "C" | ||||
| #. * "%OB" is the new format specifier introduced in glibc 2.27, | ||||
| #. * in most cases you should not change it. | ||||
| #. | ||||
| #: js/ui/calendar.js:397 | ||||
| #: js/ui/calendar.js:392 | ||||
| msgid "%OB" | ||||
| msgstr "%OB" | ||||
|  | ||||
| @@ -868,61 +856,37 @@ msgstr "%OB" | ||||
| #. * in most cases you should not use the old "%B" here unless you | ||||
| #. * absolutely know what you are doing. | ||||
| #. | ||||
| #: js/ui/calendar.js:407 | ||||
| #: js/ui/calendar.js:402 | ||||
| msgid "%OB %Y" | ||||
| msgstr "%OB %Y" | ||||
|  | ||||
| #: js/ui/calendar.js:466 | ||||
| #: js/ui/calendar.js:461 | ||||
| msgid "Previous month" | ||||
| msgstr "Önceki ay" | ||||
|  | ||||
| #: js/ui/calendar.js:481 | ||||
| #: js/ui/calendar.js:476 | ||||
| msgid "Next month" | ||||
| msgstr "Gelecek ay" | ||||
|  | ||||
| #: js/ui/calendar.js:631 | ||||
| #: js/ui/calendar.js:626 | ||||
| #, no-javascript-format | ||||
| msgctxt "date day number format" | ||||
| msgid "%d" | ||||
| msgstr "%d" | ||||
|  | ||||
| #: js/ui/calendar.js:687 | ||||
| #: js/ui/calendar.js:682 | ||||
| msgid "Week %V" | ||||
| msgstr "%V. Hafta" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/calendar.js:762 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "Tüm Gün" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/calendar.js:900 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d" | ||||
| msgstr "%A, %-d %B" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/calendar.js:903 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d, %Y" | ||||
| msgstr "%A, %-d %B, %Y" | ||||
|  | ||||
| #: js/ui/calendar.js:1133 | ||||
| #: js/ui/calendar.js:895 | ||||
| msgid "No Notifications" | ||||
| msgstr "Bildirim Yok" | ||||
|  | ||||
| #: js/ui/calendar.js:1136 | ||||
| msgid "No Events" | ||||
| msgstr "Olay Yok" | ||||
|  | ||||
| #: js/ui/calendar.js:1190 | ||||
| #: js/ui/calendar.js:949 | ||||
| msgid "Do Not Disturb" | ||||
| msgstr "Rahatsız Etme" | ||||
|  | ||||
| #: js/ui/calendar.js:1209 | ||||
| #: js/ui/calendar.js:968 | ||||
| msgid "Clear" | ||||
| msgstr "Temizle" | ||||
|  | ||||
| @@ -1076,7 +1040,7 @@ msgstr "Üzgünüm ama işe yaramadı. Lütfen yeniden deneyin." | ||||
| msgid "%s is now known as %s" | ||||
| msgstr "%s, şimdi %s olarak biliniyor" | ||||
|  | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:177 | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:178 | ||||
| msgid "Windows" | ||||
| msgstr "Pencereler" | ||||
|  | ||||
| @@ -1095,7 +1059,7 @@ msgstr "Konsol" | ||||
| #. * "Tue 9:29 AM").  The string itself should become a full date, e.g., | ||||
| #. * "February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:75 | ||||
| #: js/ui/dateMenu.js:79 | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%-d %B %Y" | ||||
|  | ||||
| @@ -1103,35 +1067,67 @@ msgstr "%-d %B %Y" | ||||
| #. * below the time in the shell; it should combine the weekday and the | ||||
| #. * date, e.g. "Tuesday February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:82 | ||||
| #: js/ui/dateMenu.js:86 | ||||
| msgid "%A %B %e %Y" | ||||
| msgstr "%e %B %Y %A" | ||||
|  | ||||
| #: js/ui/dateMenu.js:161 | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/dateMenu.js:151 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d" | ||||
| msgstr "%-d %B" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/dateMenu.js:154 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%-d %B %Y" | ||||
|  | ||||
| #: js/ui/dateMenu.js:160 | ||||
| msgid "Today" | ||||
| msgstr "Bugün" | ||||
|  | ||||
| #: js/ui/dateMenu.js:164 | ||||
| msgid "Tomorrow" | ||||
| msgstr "Yarın" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/dateMenu.js:180 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "Tüm Gün" | ||||
|  | ||||
| #: js/ui/dateMenu.js:231 | ||||
| msgid "No Events" | ||||
| msgstr "Olay Yok" | ||||
|  | ||||
| #: js/ui/dateMenu.js:348 | ||||
| msgid "Add world clocks…" | ||||
| msgstr "Dünya saatlerini ekle…" | ||||
|  | ||||
| #: js/ui/dateMenu.js:162 | ||||
| #: js/ui/dateMenu.js:349 | ||||
| msgid "World Clocks" | ||||
| msgstr "Dünya Saatleri" | ||||
|  | ||||
| #: js/ui/dateMenu.js:424 | ||||
| #: js/ui/dateMenu.js:629 | ||||
| msgid "Loading…" | ||||
| msgstr "Yükleniyor…" | ||||
|  | ||||
| #: js/ui/dateMenu.js:434 | ||||
| #: js/ui/dateMenu.js:639 | ||||
| msgid "Go online for weather information" | ||||
| msgstr "Hava durumu bilgisi için çevrim içi olun" | ||||
|  | ||||
| #: js/ui/dateMenu.js:436 | ||||
| #: js/ui/dateMenu.js:641 | ||||
| msgid "Weather information is currently unavailable" | ||||
| msgstr "Hava durumu bilgisi şu anda kullanılabilir değil" | ||||
|  | ||||
| #: js/ui/dateMenu.js:446 | ||||
| #: js/ui/dateMenu.js:651 | ||||
| msgid "Weather" | ||||
| msgstr "Hava Durumu" | ||||
|  | ||||
| #: js/ui/dateMenu.js:448 | ||||
| #: js/ui/dateMenu.js:653 | ||||
| msgid "Select weather location…" | ||||
| msgstr "Hava durumu konumu seç…" | ||||
|  | ||||
| @@ -1290,11 +1286,11 @@ msgstr "Uzantı Yükle" | ||||
| msgid "Download and install “%s” from extensions.gnome.org?" | ||||
| msgstr "extensions.gnome.org üstünden “%s” uzantısı indirilip kurulsun mu?" | ||||
|  | ||||
| #: js/ui/extensionSystem.js:251 | ||||
| #: js/ui/extensionSystem.js:252 | ||||
| msgid "Extension Updates Available" | ||||
| msgstr "Uzantı Güncellemeleri Var" | ||||
|  | ||||
| #: js/ui/extensionSystem.js:252 | ||||
| #: js/ui/extensionSystem.js:253 | ||||
| msgid "Extension updates are ready to be installed." | ||||
| msgstr "Uzantı güncellemeleri kuruluma hazır." | ||||
|  | ||||
| @@ -1421,6 +1417,7 @@ msgid "Disabled" | ||||
| msgstr "Devre dışı" | ||||
|  | ||||
| #: js/ui/lookingGlass.js:743 | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:188 | ||||
| msgid "Error" | ||||
| msgstr "Hata" | ||||
|  | ||||
| @@ -1464,11 +1461,11 @@ msgstr "Ekran Kilitleme, GNOME ekran yöneticisi gerektirir." | ||||
| msgid "System Information" | ||||
| msgstr "Sistem Bilgisi" | ||||
|  | ||||
| #: js/ui/mpris.js:204 | ||||
| #: js/ui/mpris.js:203 | ||||
| msgid "Unknown artist" | ||||
| msgstr "Bilinmeyen sanatçı" | ||||
|  | ||||
| #: js/ui/mpris.js:214 | ||||
| #: js/ui/mpris.js:213 | ||||
| msgid "Unknown title" | ||||
| msgstr "Bilinmeyen başlık" | ||||
|  | ||||
| @@ -1514,23 +1511,23 @@ msgstr "Tuş vuruşu ata" | ||||
| msgid "Done" | ||||
| msgstr "Bitti" | ||||
|  | ||||
| #: js/ui/padOsd.js:745 | ||||
| #: js/ui/padOsd.js:732 | ||||
| msgid "Edit…" | ||||
| msgstr "Düzenle…" | ||||
|  | ||||
| #: js/ui/padOsd.js:787 js/ui/padOsd.js:910 | ||||
| #: js/ui/padOsd.js:774 js/ui/padOsd.js:891 | ||||
| msgid "None" | ||||
| msgstr "Yok" | ||||
|  | ||||
| #: js/ui/padOsd.js:863 | ||||
| #: js/ui/padOsd.js:845 | ||||
| msgid "Press a button to configure" | ||||
| msgstr "Yapılandırmak için düğmeye bas" | ||||
|  | ||||
| #: js/ui/padOsd.js:864 | ||||
| #: js/ui/padOsd.js:846 | ||||
| msgid "Press Esc to exit" | ||||
| msgstr "Çıkmak için Esc’ye bas" | ||||
|  | ||||
| #: js/ui/padOsd.js:867 | ||||
| #: js/ui/padOsd.js:849 | ||||
| msgid "Press any key to exit" | ||||
| msgstr "Çıkmak için herhangi bir tuşa bas" | ||||
|  | ||||
| @@ -1588,15 +1585,15 @@ msgstr "Kilitlenemedi" | ||||
| msgid "Lock was blocked by an application" | ||||
| msgstr "Kilitleme bir uygulama tarafından engellendi" | ||||
|  | ||||
| #: js/ui/search.js:702 | ||||
| #: js/ui/search.js:823 | ||||
| msgid "Searching…" | ||||
| msgstr "Aranıyor…" | ||||
|  | ||||
| #: js/ui/search.js:704 | ||||
| #: js/ui/search.js:825 | ||||
| msgid "No results." | ||||
| msgstr "Sonuç yok." | ||||
|  | ||||
| #: js/ui/search.js:830 | ||||
| #: js/ui/search.js:951 | ||||
| #, javascript-format | ||||
| msgid "%d more" | ||||
| msgid_plural "%d more" | ||||
| @@ -1793,17 +1790,17 @@ msgstr "Konum Devre Dışı" | ||||
| msgid "Enable" | ||||
| msgstr "Etkinleştir" | ||||
|  | ||||
| #: js/ui/status/location.js:355 | ||||
| #: js/ui/status/location.js:350 | ||||
| msgid "Allow location access" | ||||
| msgstr "Konum erişimine izin ver" | ||||
|  | ||||
| #. Translators: %s is an application name | ||||
| #: js/ui/status/location.js:357 | ||||
| #: js/ui/status/location.js:352 | ||||
| #, javascript-format | ||||
| msgid "The app %s wants to access your location" | ||||
| msgstr "%s uygulaması konumunuza erişmek istiyor" | ||||
|  | ||||
| #: js/ui/status/location.js:367 | ||||
| #: js/ui/status/location.js:362 | ||||
| msgid "Location access can be changed at any time from the privacy settings." | ||||
| msgstr "Konum erişimi, gizlilik ayarlarından her zaman değiştirilebilir." | ||||
|  | ||||
| @@ -2134,11 +2131,11 @@ msgstr "Thunderbolt yetkilendirme hatası" | ||||
| msgid "Could not authorize the Thunderbolt device: %s" | ||||
| msgstr "Thunderbolt aygıtı yetkilendirilemedi: %s" | ||||
|  | ||||
| #: js/ui/status/volume.js:151 | ||||
| #: js/ui/status/volume.js:154 | ||||
| msgid "Volume changed" | ||||
| msgstr "Bölüm değişti" | ||||
|  | ||||
| #: js/ui/status/volume.js:222 | ||||
| #: js/ui/status/volume.js:225 | ||||
| msgid "Volume" | ||||
| msgstr "Bölüm" | ||||
|  | ||||
| @@ -2192,11 +2189,11 @@ msgstr "Kilit Açma Penceresi" | ||||
| msgid "Log in as another user" | ||||
| msgstr "Başka kullanıcı olarak oturum aç" | ||||
|  | ||||
| #: js/ui/viewSelector.js:181 | ||||
| #: js/ui/viewSelector.js:182 | ||||
| msgid "Applications" | ||||
| msgstr "Uygulamalar" | ||||
|  | ||||
| #: js/ui/viewSelector.js:185 | ||||
| #: js/ui/viewSelector.js:186 | ||||
| msgid "Search" | ||||
| msgstr "Ara" | ||||
|  | ||||
| @@ -2402,26 +2399,34 @@ msgid "%d extension will be updated on next login." | ||||
| msgid_plural "%d extensions will be updated on next login." | ||||
| msgstr[0] "%d uzantı sonraki girişte güncellenecek." | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:100 | ||||
| #: subprojects/extensions-tool/src/command-create.c:301 | ||||
| #: subprojects/extensions-app/js/main.js:461 | ||||
| msgid "The extension is incompatible with the current GNOME version" | ||||
| msgstr "Uzantı, geçerli GNOME sürümüyle uyumsuz" | ||||
|  | ||||
| #: subprojects/extensions-app/js/main.js:464 | ||||
| msgid "The extension had an error" | ||||
| msgstr "Uzantıda hata oluştu" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:109 | ||||
| #: subprojects/extensions-tool/src/command-create.c:325 | ||||
| #: subprojects/extensions-tool/src/main.c:241 | ||||
| msgid "Description" | ||||
| msgstr "Açıklama" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:123 | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:132 | ||||
| #: subprojects/extensions-tool/src/main.c:253 | ||||
| msgid "Version" | ||||
| msgstr "Sürüm" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:151 | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:160 | ||||
| msgid "Author" | ||||
| msgstr "Yazar" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:175 | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:216 | ||||
| msgid "Website" | ||||
| msgstr "Web Sitesi" | ||||
|  | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:192 | ||||
| #: subprojects/extensions-app/data/ui/extension-row.ui:233 | ||||
| msgid "Remove…" | ||||
| msgstr "Kaldır…" | ||||
|  | ||||
| @@ -2483,12 +2488,12 @@ msgid "Log Out…" | ||||
| msgstr "Oturumu Kapat…" | ||||
|  | ||||
| #. Translators: a file path to an extension directory | ||||
| #: subprojects/extensions-tool/src/command-create.c:202 | ||||
| #: subprojects/extensions-tool/src/command-create.c:226 | ||||
| #, c-format | ||||
| msgid "The new extension was successfully created in %s.\n" | ||||
| msgstr "Yeni uzantı %s içinde oluşturuldu.\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:275 | ||||
| #: subprojects/extensions-tool/src/command-create.c:299 | ||||
| #, c-format | ||||
| msgid "" | ||||
| "Name should be a very short (ideally descriptive) string.\n" | ||||
| @@ -2497,12 +2502,12 @@ msgstr "" | ||||
| "Ad çok kısa (ideal olarak açıklayıcı) dizge olmalıdır.\n" | ||||
| "Örnekler: %s" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:281 | ||||
| #: subprojects/extensions-tool/src/command-create.c:305 | ||||
| #: subprojects/extensions-tool/src/main.c:238 | ||||
| msgid "Name" | ||||
| msgstr "Ad" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:295 | ||||
| #: subprojects/extensions-tool/src/command-create.c:319 | ||||
| #, c-format | ||||
| msgid "" | ||||
| "Description is a single-sentence explanation of what your extension does.\n" | ||||
| @@ -2511,7 +2516,7 @@ msgstr "" | ||||
| "Açıklama, uzantınızın ne yaptığı anlatan tek tümcelik tanımlamadır.\n" | ||||
| "Örnekler: %s" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:315 | ||||
| #: subprojects/extensions-tool/src/command-create.c:339 | ||||
| msgid "" | ||||
| "UUID is a globally-unique identifier for your extension.\n" | ||||
| "This should be in the format of an email address (clicktofocus@janedoe." | ||||
| @@ -2520,56 +2525,56 @@ msgstr "" | ||||
| "UUID, uzantınız için genel ve benzersiz tanımlayıcıdır.\n" | ||||
| "E-posta biçiminde olmalıdır (clicktofocus@janedoe.example.com)\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:342 | ||||
| #: subprojects/extensions-tool/src/command-create.c:366 | ||||
| msgid "Choose one of the available templates:\n" | ||||
| msgstr "Uygun şablonlardan birini seçin:\n" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:356 | ||||
| #: subprojects/extensions-tool/src/command-create.c:380 | ||||
| msgid "Template" | ||||
| msgstr "Şablon" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:411 | ||||
| #: subprojects/extensions-tool/src/command-create.c:435 | ||||
| msgid "The unique identifier of the new extension" | ||||
| msgstr "Yeni uzantının eşsiz tanımlayıcısı" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:414 | ||||
| #: subprojects/extensions-tool/src/command-create.c:438 | ||||
| msgid "NAME" | ||||
| msgstr "AD" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:415 | ||||
| #: subprojects/extensions-tool/src/command-create.c:439 | ||||
| msgid "The user-visible name of the new extension" | ||||
| msgstr "Yeni uzantının kullanıcı tarafından görülebilir adı" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:417 | ||||
| #: subprojects/extensions-tool/src/command-create.c:441 | ||||
| msgid "DESCRIPTION" | ||||
| msgstr "AÇIKLAMA" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:419 | ||||
| #: subprojects/extensions-tool/src/command-create.c:443 | ||||
| msgid "A short description of what the extension does" | ||||
| msgstr "Uzantının ne yaptığına dair kısa açıklama" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:422 | ||||
| #: subprojects/extensions-tool/src/command-create.c:446 | ||||
| msgid "TEMPLATE" | ||||
| msgstr "ŞABLON" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:423 | ||||
| #: subprojects/extensions-tool/src/command-create.c:447 | ||||
| msgid "The template to use for the new extension" | ||||
| msgstr "Yeni uzantı için kullanılacak şablon" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:429 | ||||
| #: subprojects/extensions-tool/src/command-create.c:453 | ||||
| msgid "Enter extension information interactively" | ||||
| msgstr "Uzantı bilgisini etkileşimli olarak gir" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:437 | ||||
| #: subprojects/extensions-tool/src/command-create.c:461 | ||||
| msgid "Create a new extension" | ||||
| msgstr "Yeni uzantı oluştur" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:455 | ||||
| #: subprojects/extensions-tool/src/command-create.c:479 | ||||
| #: subprojects/extensions-tool/src/command-list.c:172 | ||||
| msgid "Unknown arguments" | ||||
| msgstr "Bilinmeyen argüman" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:480 | ||||
| #: subprojects/extensions-tool/src/command-create.c:504 | ||||
| msgid "UUID, name and description are required" | ||||
| msgstr "UUID, ad ve açıklama gereklidir" | ||||
|  | ||||
| @@ -2889,7 +2894,23 @@ msgstr[0] "%u Girdi" | ||||
| msgid "System Sounds" | ||||
| msgstr "Sistem Sesleri" | ||||
|  | ||||
| #, javascript-format | ||||
| #~ msgid "Frequently used applications will appear here" | ||||
| #~ msgstr "Sık kullanılan uygulamalar burada yer alacak" | ||||
|  | ||||
| #~ msgid "Frequent" | ||||
| #~ msgstr "Sık Sık" | ||||
|  | ||||
| #~ msgid "All" | ||||
| #~ msgstr "Tümü" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d" | ||||
| #~ msgstr "%A, %-d %B" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d, %Y" | ||||
| #~ msgstr "%A, %-d %B, %Y" | ||||
|  | ||||
| #~ msgid "%d Connected" | ||||
| #~ msgid_plural "%d Connected" | ||||
| #~ msgstr[0] "%d Aygıt Bağlandı" | ||||
| @@ -3217,12 +3238,6 @@ msgstr "Sistem Sesleri" | ||||
| #~ msgid "Nothing Scheduled" | ||||
| #~ msgstr "Planınız Boş" | ||||
|  | ||||
| #~ msgid "Today" | ||||
| #~ msgstr "Bugün" | ||||
|  | ||||
| #~ msgid "Tomorrow" | ||||
| #~ msgstr "Yarın" | ||||
|  | ||||
| #~ msgid "This week" | ||||
| #~ msgstr "Bu hafta" | ||||
|  | ||||
|   | ||||
							
								
								
									
										194
									
								
								po/uk.po
									
									
									
									
									
								
							
							
						
						
									
										194
									
								
								po/uk.po
									
									
									
									
									
								
							| @@ -10,8 +10,8 @@ msgid "" | ||||
| msgstr "" | ||||
| "Project-Id-Version: gnome-shell master\n" | ||||
| "Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n" | ||||
| "POT-Creation-Date: 2020-06-05 23:11+0000\n" | ||||
| "PO-Revision-Date: 2020-06-06 15:53+0300\n" | ||||
| "POT-Creation-Date: 2020-07-20 16:18+0000\n" | ||||
| "PO-Revision-Date: 2020-07-21 08:42+0300\n" | ||||
| "Last-Translator: Yuri Chornoivan <yurchor@ukr.net>\n" | ||||
| "Language-Team: Ukrainian <kde-i18n-uk@kde.org>\n" | ||||
| "Language: uk\n" | ||||
| @@ -216,89 +216,106 @@ msgid "" | ||||
| "shell." | ||||
| msgstr "Вмикає D-Bus API для самоаналізу стану оболонки." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:119 | ||||
| #: data/org.gnome.shell.gschema.xml.in:114 | ||||
| msgid "Layout of the app picker" | ||||
| msgstr "Компонування засобу вибору програм" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:115 | ||||
| msgid "" | ||||
| "Layout of the app picker. Each entry in the array is a page. Pages are " | ||||
| "stored in the order they appear in GNOME Shell. Each page contains an " | ||||
| "“application id” → 'data' pair. Currently, the following values are stored " | ||||
| "as 'data': • “position”: the position of the application icon in the page" | ||||
| msgstr "" | ||||
| "Компонування засобу вибору програм. Кожен запис у цьому масиві є сторінкою." | ||||
| " Сторінки зберігаються у порядку, у якому їх буде показано у GNOME Shell." | ||||
| " Кожна сторінка містить пану записів “ідентифікатор програми” → 'дані'. У" | ||||
| " поточній версії як 'дані' може бути збережено такі значення: • “позиція”:" | ||||
| " позиція піктограми програми на сторінці" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:130 | ||||
| msgid "Keybinding to open the application menu" | ||||
| msgstr "Комбінація клавіш, щоб відкрити меню програм" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:120 | ||||
| #: data/org.gnome.shell.gschema.xml.in:131 | ||||
| msgid "Keybinding to open the application menu." | ||||
| msgstr "Комбінація клавіш, щоб відкрити меню програм." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:126 | ||||
| #: data/org.gnome.shell.gschema.xml.in:137 | ||||
| msgid "Keybinding to open the “Show Applications” view" | ||||
| msgstr "Комбінація клавіш, щоб «Показати програми»" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:127 | ||||
| #: data/org.gnome.shell.gschema.xml.in:138 | ||||
| msgid "" | ||||
| "Keybinding to open the “Show Applications” view of the Activities Overview." | ||||
| msgstr "Комбінація клавіш, щоб «Показати програми» в огляді діяльності." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:134 | ||||
| #: data/org.gnome.shell.gschema.xml.in:145 | ||||
| msgid "Keybinding to open the overview" | ||||
| msgstr "Комбінація клавіш, щоб відкрити огляд" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:135 | ||||
| #: data/org.gnome.shell.gschema.xml.in:146 | ||||
| msgid "Keybinding to open the Activities Overview." | ||||
| msgstr "Комбінація клавіш, щоб відкрити огляд діяльності." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:141 | ||||
| #: data/org.gnome.shell.gschema.xml.in:152 | ||||
| msgid "Keybinding to toggle the visibility of the notification list" | ||||
| msgstr "Комбінація клавіш, щоб змінювати видимість переліку сповіщень" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:142 | ||||
| #: data/org.gnome.shell.gschema.xml.in:153 | ||||
| msgid "Keybinding to toggle the visibility of the notification list." | ||||
| msgstr "Комбінація клавіш, щоб змінювати видимість переліку сповіщень." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:148 | ||||
| #: data/org.gnome.shell.gschema.xml.in:159 | ||||
| msgid "Keybinding to focus the active notification" | ||||
| msgstr "Комбінація клавіш, щоб перейти до активних сповіщень" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:149 | ||||
| #: data/org.gnome.shell.gschema.xml.in:160 | ||||
| msgid "Keybinding to focus the active notification." | ||||
| msgstr "Комбінація клавіш, щоб перейти до активних сповіщень." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:155 | ||||
| #: data/org.gnome.shell.gschema.xml.in:166 | ||||
| msgid "Switch to application 1" | ||||
| msgstr "Перемкнутися на програму 1" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:159 | ||||
| #: data/org.gnome.shell.gschema.xml.in:170 | ||||
| msgid "Switch to application 2" | ||||
| msgstr "Перемкнутися на програму 2" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:163 | ||||
| #: data/org.gnome.shell.gschema.xml.in:174 | ||||
| msgid "Switch to application 3" | ||||
| msgstr "Перемкнутися на програму 3" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:167 | ||||
| #: data/org.gnome.shell.gschema.xml.in:178 | ||||
| msgid "Switch to application 4" | ||||
| msgstr "Перемкнутися на програму 4" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:171 | ||||
| #: data/org.gnome.shell.gschema.xml.in:182 | ||||
| msgid "Switch to application 5" | ||||
| msgstr "Перемкнутися на програму 5" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:175 | ||||
| #: data/org.gnome.shell.gschema.xml.in:186 | ||||
| msgid "Switch to application 6" | ||||
| msgstr "Перемкнутися на програму 6" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:179 | ||||
| #: data/org.gnome.shell.gschema.xml.in:190 | ||||
| msgid "Switch to application 7" | ||||
| msgstr "Перемкнутися на програму 7" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:183 | ||||
| #: data/org.gnome.shell.gschema.xml.in:194 | ||||
| msgid "Switch to application 8" | ||||
| msgstr "Перемкнутися на програму 8" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:187 | ||||
| #: data/org.gnome.shell.gschema.xml.in:198 | ||||
| msgid "Switch to application 9" | ||||
| msgstr "Перемкнутися на програму 9" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:196 | ||||
| #: data/org.gnome.shell.gschema.xml.in:223 | ||||
| #: data/org.gnome.shell.gschema.xml.in:207 | ||||
| #: data/org.gnome.shell.gschema.xml.in:234 | ||||
| msgid "Limit switcher to current workspace." | ||||
| msgstr "Обмежити перемикач на поточний робочий простір." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:197 | ||||
| #: data/org.gnome.shell.gschema.xml.in:208 | ||||
| msgid "" | ||||
| "If true, only applications that have windows on the current workspace are " | ||||
| "shown in the switcher. Otherwise, all applications are included." | ||||
| @@ -306,11 +323,11 @@ msgstr "" | ||||
| "Якщо це вказано, то тільки програми з поточного робочого простору будуть " | ||||
| "показані в перемикачі. Інакше — програми з усіх просторів." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:214 | ||||
| #: data/org.gnome.shell.gschema.xml.in:225 | ||||
| msgid "The application icon mode." | ||||
| msgstr "Режим піктограм для програм." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:215 | ||||
| #: data/org.gnome.shell.gschema.xml.in:226 | ||||
| msgid "" | ||||
| "Configures how the windows are shown in the switcher. Valid possibilities " | ||||
| "are “thumbnail-only” (shows a thumbnail of the window), “app-icon-" | ||||
| @@ -320,7 +337,7 @@ msgstr "" | ||||
| "«thumbnail-only» (показує мініатюру вікна), «app-icon-only» (показує тільки " | ||||
| "піктограми програм) або «both» (обидва)." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:224 | ||||
| #: data/org.gnome.shell.gschema.xml.in:235 | ||||
| msgid "" | ||||
| "If true, only windows from the current workspace are shown in the switcher. " | ||||
| "Otherwise, all windows are included." | ||||
| @@ -328,57 +345,57 @@ msgstr "" | ||||
| "Якщо це вказано, то тільки вікна з поточного робочого простору будуть " | ||||
| "показані в перемикачі. Інакше — вікна з усіх просторів." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:234 | ||||
| #: data/org.gnome.shell.gschema.xml.in:245 | ||||
| msgid "Locations" | ||||
| msgstr "Місцевості" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:235 | ||||
| #: data/org.gnome.shell.gschema.xml.in:246 | ||||
| msgid "The locations to show in world clocks" | ||||
| msgstr "Місцевість для світового годинника" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:245 | ||||
| #: data/org.gnome.shell.gschema.xml.in:256 | ||||
| msgid "Automatic location" | ||||
| msgstr "Автоматичне визначення місцевості" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:246 | ||||
| #: data/org.gnome.shell.gschema.xml.in:257 | ||||
| msgid "Whether to fetch the current location or not" | ||||
| msgstr "Чи одержувати поточну локацію" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:253 | ||||
| #: data/org.gnome.shell.gschema.xml.in:264 | ||||
| msgid "Location" | ||||
| msgstr "Місцевість" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:254 | ||||
| #: data/org.gnome.shell.gschema.xml.in:265 | ||||
| msgid "The location for which to show a forecast" | ||||
| msgstr "Місцевість для прогнозу погоди" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:266 | ||||
| #: data/org.gnome.shell.gschema.xml.in:277 | ||||
| msgid "Attach modal dialog to the parent window" | ||||
| msgstr "Приєднати модальне вікно до батьківського вікна" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:267 | ||||
| #: data/org.gnome.shell.gschema.xml.in:276 | ||||
| #: data/org.gnome.shell.gschema.xml.in:284 | ||||
| #: data/org.gnome.shell.gschema.xml.in:292 | ||||
| #: data/org.gnome.shell.gschema.xml.in:300 | ||||
| #: data/org.gnome.shell.gschema.xml.in:278 | ||||
| #: data/org.gnome.shell.gschema.xml.in:287 | ||||
| #: data/org.gnome.shell.gschema.xml.in:295 | ||||
| #: data/org.gnome.shell.gschema.xml.in:303 | ||||
| #: data/org.gnome.shell.gschema.xml.in:311 | ||||
| msgid "" | ||||
| "This key overrides the key in org.gnome.mutter when running GNOME Shell." | ||||
| msgstr "" | ||||
| "Цей ключ перевизначає ключ у org.gnome.mutter, коли запущено GNOME Shell." | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:275 | ||||
| #: data/org.gnome.shell.gschema.xml.in:286 | ||||
| msgid "Enable edge tiling when dropping windows on screen edges" | ||||
| msgstr "Увімкнути розбиття країв, коли кладуться вікна на краї екрана" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:283 | ||||
| #: data/org.gnome.shell.gschema.xml.in:294 | ||||
| msgid "Workspaces are managed dynamically" | ||||
| msgstr "Робочі простори організовуються динамічно" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:291 | ||||
| #: data/org.gnome.shell.gschema.xml.in:302 | ||||
| msgid "Workspaces only on primary monitor" | ||||
| msgstr "Робочий простір лише на основному моніторі" | ||||
|  | ||||
| #: data/org.gnome.shell.gschema.xml.in:299 | ||||
| #: data/org.gnome.shell.gschema.xml.in:310 | ||||
| msgid "Delay focus changes in mouse mode until the pointer stops moving" | ||||
| msgstr "Затримувати зміни фокусу миші, поки вказівник не перестане рухатись" | ||||
|  | ||||
| @@ -414,7 +431,7 @@ msgstr "Відвідати сторінку розширення" | ||||
|  | ||||
| #: js/gdm/authPrompt.js:135 js/ui/audioDeviceSelection.js:57 | ||||
| #: js/ui/components/networkAgent.js:110 js/ui/components/polkitAgent.js:139 | ||||
| #: js/ui/endSessionDialog.js:369 js/ui/extensionDownloader.js:181 | ||||
| #: js/ui/endSessionDialog.js:369 js/ui/extensionDownloader.js:183 | ||||
| #: js/ui/shellMountOperation.js:376 js/ui/shellMountOperation.js:386 | ||||
| #: js/ui/status/network.js:916 subprojects/extensions-app/js/main.js:149 | ||||
| msgid "Cancel" | ||||
| @@ -456,7 +473,7 @@ msgstr "Користувач" | ||||
| msgid "Login Window" | ||||
| msgstr "Вікно входу" | ||||
|  | ||||
| #: js/gdm/util.js:345 | ||||
| #: js/gdm/util.js:355 | ||||
| msgid "Authentication error" | ||||
| msgstr "Помилка розпізнавання" | ||||
|  | ||||
| @@ -465,7 +482,7 @@ msgstr "Помилка розпізнавання" | ||||
| #. as a cue to display our own message. | ||||
| #. Translators: this message is shown below the password entry field | ||||
| #. to indicate the user can swipe their finger instead | ||||
| #: js/gdm/util.js:471 | ||||
| #: js/gdm/util.js:481 | ||||
| msgid "(or swipe finger)" | ||||
| msgstr "(або проведіть пальцем)" | ||||
|  | ||||
| @@ -713,36 +730,36 @@ msgstr "Заборонити доступ" | ||||
| msgid "Grant Access" | ||||
| msgstr "Надати доступ" | ||||
|  | ||||
| #: js/ui/appDisplay.js:956 | ||||
| #: js/ui/appDisplay.js:1297 | ||||
| msgid "Unnamed Folder" | ||||
| msgstr "Неназвана тека" | ||||
|  | ||||
| #. Translators: This is the heading of a list of open windows | ||||
| #: js/ui/appDisplay.js:2215 js/ui/panel.js:75 | ||||
| #: js/ui/appDisplay.js:2767 js/ui/panel.js:75 | ||||
| msgid "Open Windows" | ||||
| msgstr "Відкрити вікна" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2234 js/ui/panel.js:82 | ||||
| #: js/ui/appDisplay.js:2786 js/ui/panel.js:82 | ||||
| msgid "New Window" | ||||
| msgstr "Нове вікно" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2250 | ||||
| #: js/ui/appDisplay.js:2802 | ||||
| msgid "Launch using Integrated Graphics Card" | ||||
| msgstr "Запустити через інтегровану графічну плату" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2251 | ||||
| #: js/ui/appDisplay.js:2803 | ||||
| msgid "Launch using Discrete Graphics Card" | ||||
| msgstr "Запустити через дискретну графічну плату" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2279 js/ui/dash.js:239 | ||||
| #: js/ui/appDisplay.js:2831 js/ui/dash.js:239 | ||||
| msgid "Remove from Favorites" | ||||
| msgstr "Вилучити з улюбленого" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2285 | ||||
| #: js/ui/appDisplay.js:2837 | ||||
| msgid "Add to Favorites" | ||||
| msgstr "Додати до улюбленого" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2295 js/ui/panel.js:93 | ||||
| #: js/ui/appDisplay.js:2847 js/ui/panel.js:93 | ||||
| msgid "Show Details" | ||||
| msgstr "Показати подробиці" | ||||
|  | ||||
| @@ -772,7 +789,7 @@ msgstr "Навушники" | ||||
| msgid "Headset" | ||||
| msgstr "Гарнітура" | ||||
|  | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:273 | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:272 | ||||
| msgid "Microphone" | ||||
| msgstr "Мікрофон" | ||||
|  | ||||
| @@ -1073,14 +1090,12 @@ msgstr "%A %-d %B %Y" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/dateMenu.js:151 | ||||
| #| msgid "%B %-d %Y" | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d" | ||||
| msgstr "%-d %B" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/dateMenu.js:154 | ||||
| #| msgid "%B %-d %Y" | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%-d %B %Y" | ||||
| @@ -1293,15 +1308,15 @@ msgstr "%s (віддалено)" | ||||
| msgid "%s (console)" | ||||
| msgstr "%s (консоль)" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:185 | ||||
| #: js/ui/extensionDownloader.js:187 | ||||
| msgid "Install" | ||||
| msgstr "Встановити" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:191 | ||||
| #: js/ui/extensionDownloader.js:193 | ||||
| msgid "Install Extension" | ||||
| msgstr "Встановити розширення" | ||||
|  | ||||
| #: js/ui/extensionDownloader.js:192 | ||||
| #: js/ui/extensionDownloader.js:194 | ||||
| #, javascript-format | ||||
| msgid "Download and install “%s” from extensions.gnome.org?" | ||||
| msgstr "Звантажити і встановити «%s» з extensions.gnome.org?" | ||||
| @@ -1334,11 +1349,11 @@ msgstr "Програма бажає успадкувати скорочення" | ||||
| msgid "You can restore shortcuts by pressing %s." | ||||
| msgstr "Можна відновити скорочення, натиснувши %s." | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:98 | ||||
| #: js/ui/inhibitShortcutsDialog.js:100 | ||||
| msgid "Deny" | ||||
| msgstr "Заборонити" | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:105 | ||||
| #: js/ui/inhibitShortcutsDialog.js:107 | ||||
| msgid "Allow" | ||||
| msgstr "Дозволити" | ||||
|  | ||||
| @@ -1404,7 +1419,7 @@ msgstr "Вимкнути" | ||||
| msgid "Leave Off" | ||||
| msgstr "Покинути" | ||||
|  | ||||
| #: js/ui/keyboard.js:207 | ||||
| #: js/ui/keyboard.js:225 | ||||
| msgid "Region & Language Settings" | ||||
| msgstr "Параметри регіону та мови" | ||||
|  | ||||
| @@ -1477,7 +1492,7 @@ msgstr "Блокування екрана вимкнено" | ||||
| msgid "Screen Locking requires the GNOME display manager." | ||||
| msgstr "Блокування екрана потребує керування входом GNOME." | ||||
|  | ||||
| #: js/ui/messageTray.js:1547 | ||||
| #: js/ui/messageTray.js:1476 | ||||
| msgid "System Information" | ||||
| msgstr "Інформація про систему" | ||||
|  | ||||
| @@ -1489,13 +1504,13 @@ msgstr "Невідомий виконавець" | ||||
| msgid "Unknown title" | ||||
| msgstr "Невідома назва" | ||||
|  | ||||
| #: js/ui/overview.js:73 | ||||
| #: js/ui/overview.js:74 | ||||
| msgid "Undo" | ||||
| msgstr "Повернути" | ||||
|  | ||||
| #. Translators: This is the main view to select | ||||
| #. activities. See also note for "Activities" string. | ||||
| #: js/ui/overview.js:86 | ||||
| #: js/ui/overview.js:87 | ||||
| msgid "Overview" | ||||
| msgstr "Огляд" | ||||
|  | ||||
| @@ -1503,7 +1518,7 @@ msgstr "Огляд" | ||||
| #. in the search entry when no search is | ||||
| #. active; it should not exceed ~30 | ||||
| #. characters. | ||||
| #: js/ui/overview.js:107 | ||||
| #: js/ui/overview.js:108 | ||||
| msgid "Type to search" | ||||
| msgstr "Введіть текст для пошуку" | ||||
|  | ||||
| @@ -2162,11 +2177,11 @@ msgstr "Помилка завірення Thunderbolt" | ||||
| msgid "Could not authorize the Thunderbolt device: %s" | ||||
| msgstr "Неможливо завірити пристрій Thunderbolt: %s" | ||||
|  | ||||
| #: js/ui/status/volume.js:154 | ||||
| #: js/ui/status/volume.js:155 | ||||
| msgid "Volume changed" | ||||
| msgstr "Гучність змінено" | ||||
|  | ||||
| #: js/ui/status/volume.js:225 | ||||
| #: js/ui/status/volume.js:217 | ||||
| msgid "Volume" | ||||
| msgstr "Гучність" | ||||
|  | ||||
| @@ -2212,11 +2227,11 @@ msgstr "Проведіть вгору для розблокування" | ||||
| msgid "Click or press a key to unlock" | ||||
| msgstr "Клацніть або натисніть клавішу для розблокування" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:550 | ||||
| #: js/ui/unlockDialog.js:555 | ||||
| msgid "Unlock Window" | ||||
| msgstr "Розблокувати вікно" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:559 | ||||
| #: js/ui/unlockDialog.js:564 | ||||
| msgid "Log in as another user" | ||||
| msgstr "Увійти як інший користувач" | ||||
|  | ||||
| @@ -2353,12 +2368,12 @@ msgstr "Використовувати вказаний режим, прикла | ||||
| msgid "List possible modes" | ||||
| msgstr "Перелік можливих режимів" | ||||
|  | ||||
| #: src/shell-app.c:286 | ||||
| #: src/shell-app.c:268 | ||||
| msgctxt "program" | ||||
| msgid "Unknown" | ||||
| msgstr "Невідомо" | ||||
|  | ||||
| #: src/shell-app.c:537 | ||||
| #: src/shell-app.c:519 | ||||
| #, c-format | ||||
| msgid "Failed to launch “%s”" | ||||
| msgstr "Не вдалося запустити «%s»" | ||||
| @@ -2938,36 +2953,3 @@ msgstr[3] "%u вхід" | ||||
| #: subprojects/gvc/gvc-mixer-control.c:2766 | ||||
| msgid "System Sounds" | ||||
| msgstr "Системні звуки" | ||||
|  | ||||
| #~ msgid "Frequently used applications will appear here" | ||||
| #~ msgstr "Часто використовувані програми будуть з'являтись тут" | ||||
|  | ||||
| #~ msgid "Frequent" | ||||
| #~ msgstr "Частовживане" | ||||
|  | ||||
| #~ msgid "All" | ||||
| #~ msgstr "Усе" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d" | ||||
| #~ msgstr "%A, %-d %B" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d, %Y" | ||||
| #~ msgstr "%A, %-d %B, %Y" | ||||
|  | ||||
| #~ msgid "Copy Error" | ||||
| #~ msgstr "Скопіювати помилку" | ||||
|  | ||||
| #~ msgid "%d Connected" | ||||
| #~ msgid_plural "%d Connected" | ||||
| #~ msgstr[0] "%d під'єднано" | ||||
| #~ msgstr[1] "%d під'єднано" | ||||
| #~ msgstr[2] "%d під'єднано" | ||||
| #~ msgstr[3] "%d під'єднано" | ||||
|  | ||||
| #~ msgid "Off" | ||||
| #~ msgstr "Вимкнено" | ||||
|  | ||||
| #~ msgid "On" | ||||
| #~ msgstr "Увімкнено" | ||||
|   | ||||
							
								
								
									
										204
									
								
								po/zh_CN.po
									
									
									
									
									
								
							
							
						
						
									
										204
									
								
								po/zh_CN.po
									
									
									
									
									
								
							| @@ -23,8 +23,8 @@ msgid "" | ||||
| msgstr "" | ||||
| "Project-Id-Version: gnome-shell master\n" | ||||
| "Report-Msgid-Bugs-To: https://gitlab.gnome.org/GNOME/gnome-shell/issues\n" | ||||
| "POT-Creation-Date: 2020-05-28 11:29+0000\n" | ||||
| "PO-Revision-Date: 2020-05-29 09:03-0400\n" | ||||
| "POT-Creation-Date: 2020-07-12 02:14+0000\n" | ||||
| "PO-Revision-Date: 2020-07-11 22:32-0400\n" | ||||
| "Last-Translator: Boyuan Yang <073plan@gmail.com>\n" | ||||
| "Language-Team: Chinese (China) <i18n-zh@googlegroups.com>\n" | ||||
| "Language: zh_CN\n" | ||||
| @@ -444,7 +444,7 @@ msgstr "用户名" | ||||
| msgid "Login Window" | ||||
| msgstr "登录窗口" | ||||
|  | ||||
| #: js/gdm/util.js:345 | ||||
| #: js/gdm/util.js:355 | ||||
| msgid "Authentication error" | ||||
| msgstr "认证出错" | ||||
|  | ||||
| @@ -453,7 +453,7 @@ msgstr "认证出错" | ||||
| #. as a cue to display our own message. | ||||
| #. Translators: this message is shown below the password entry field | ||||
| #. to indicate the user can swipe their finger instead | ||||
| #: js/gdm/util.js:471 | ||||
| #: js/gdm/util.js:481 | ||||
| msgid "(or swipe finger)" | ||||
| msgstr "(或滑动手指)" | ||||
|  | ||||
| @@ -559,7 +559,7 @@ msgid "%d hour ago" | ||||
| msgid_plural "%d hours ago" | ||||
| msgstr[0] "%d 小时前" | ||||
|  | ||||
| #: js/misc/util.js:191 | ||||
| #: js/misc/util.js:191 js/ui/dateMenu.js:162 | ||||
| msgid "Yesterday" | ||||
| msgstr "昨天" | ||||
|  | ||||
| @@ -672,44 +672,44 @@ msgstr "您到热点登录的连接不安全。您在此页面输入的密码或 | ||||
|  | ||||
| #. No support for non-modal system dialogs, so ignore the option | ||||
| #. let modal = options['modal'] || true; | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:374 | ||||
| #: js/ui/accessDialog.js:39 js/ui/status/location.js:369 | ||||
| msgid "Deny Access" | ||||
| msgstr "拒绝访问" | ||||
|  | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:377 | ||||
| #: js/ui/accessDialog.js:40 js/ui/status/location.js:372 | ||||
| msgid "Grant Access" | ||||
| msgstr "允许访问" | ||||
|  | ||||
| #: js/ui/appDisplay.js:960 | ||||
| #: js/ui/appDisplay.js:903 | ||||
| msgid "Unnamed Folder" | ||||
| msgstr "未命名文件夹" | ||||
|  | ||||
| #. Translators: This is the heading of a list of open windows | ||||
| #: js/ui/appDisplay.js:2219 js/ui/panel.js:75 | ||||
| #: js/ui/appDisplay.js:2225 js/ui/panel.js:75 | ||||
| msgid "Open Windows" | ||||
| msgstr "打开窗口" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2238 js/ui/panel.js:82 | ||||
| #: js/ui/appDisplay.js:2244 js/ui/panel.js:82 | ||||
| msgid "New Window" | ||||
| msgstr "新窗口" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2254 | ||||
| #: js/ui/appDisplay.js:2260 | ||||
| msgid "Launch using Integrated Graphics Card" | ||||
| msgstr "使用集成显卡启动" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2255 | ||||
| #: js/ui/appDisplay.js:2261 | ||||
| msgid "Launch using Discrete Graphics Card" | ||||
| msgstr "使用独立显卡启动" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2283 js/ui/dash.js:239 | ||||
| #: js/ui/appDisplay.js:2289 js/ui/dash.js:239 | ||||
| msgid "Remove from Favorites" | ||||
| msgstr "从收藏夹中移除" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2289 | ||||
| #: js/ui/appDisplay.js:2295 | ||||
| msgid "Add to Favorites" | ||||
| msgstr "添加到收藏夹" | ||||
|  | ||||
| #: js/ui/appDisplay.js:2299 js/ui/panel.js:93 | ||||
| #: js/ui/appDisplay.js:2305 js/ui/panel.js:93 | ||||
| msgid "Show Details" | ||||
| msgstr "显示细节" | ||||
|  | ||||
| @@ -739,7 +739,7 @@ msgstr "耳机" | ||||
| msgid "Headset" | ||||
| msgstr "耳麦" | ||||
|  | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:273 | ||||
| #: js/ui/audioDeviceSelection.js:68 js/ui/status/volume.js:272 | ||||
| msgid "Microphone" | ||||
| msgstr "麦克风" | ||||
|  | ||||
| @@ -756,7 +756,7 @@ msgid "Settings" | ||||
| msgstr "设置" | ||||
|  | ||||
| #. Translators: Enter 0-6 (Sunday-Saturday) for non-work days. Examples: "0" (Sunday) "6" (Saturday) "06" (Sunday and Saturday). | ||||
| #: js/ui/calendar.js:41 | ||||
| #: js/ui/calendar.js:36 | ||||
| msgctxt "calendar-no-work" | ||||
| msgid "06" | ||||
| msgstr "06" | ||||
| @@ -766,43 +766,43 @@ msgstr "06" | ||||
| #. * NOTE: These grid abbreviations are always shown together | ||||
| #. * and in order, e.g. "S M T W T F S". | ||||
| #. | ||||
| #: js/ui/calendar.js:70 | ||||
| #: js/ui/calendar.js:65 | ||||
| msgctxt "grid sunday" | ||||
| msgid "S" | ||||
| msgstr "日" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Monday | ||||
| #: js/ui/calendar.js:72 | ||||
| #: js/ui/calendar.js:67 | ||||
| msgctxt "grid monday" | ||||
| msgid "M" | ||||
| msgstr "一" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Tuesday | ||||
| #: js/ui/calendar.js:74 | ||||
| #: js/ui/calendar.js:69 | ||||
| msgctxt "grid tuesday" | ||||
| msgid "T" | ||||
| msgstr "二" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Wednesday | ||||
| #: js/ui/calendar.js:76 | ||||
| #: js/ui/calendar.js:71 | ||||
| msgctxt "grid wednesday" | ||||
| msgid "W" | ||||
| msgstr "三" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Thursday | ||||
| #: js/ui/calendar.js:78 | ||||
| #: js/ui/calendar.js:73 | ||||
| msgctxt "grid thursday" | ||||
| msgid "T" | ||||
| msgstr "四" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Friday | ||||
| #: js/ui/calendar.js:80 | ||||
| #: js/ui/calendar.js:75 | ||||
| msgctxt "grid friday" | ||||
| msgid "F" | ||||
| msgstr "五" | ||||
|  | ||||
| #. Translators: Calendar grid abbreviation for Saturday | ||||
| #: js/ui/calendar.js:82 | ||||
| #: js/ui/calendar.js:77 | ||||
| msgctxt "grid saturday" | ||||
| msgid "S" | ||||
| msgstr "六" | ||||
| @@ -813,7 +813,7 @@ msgstr "六" | ||||
| #. * "%OB" is the new format specifier introduced in glibc 2.27, | ||||
| #. * in most cases you should not change it. | ||||
| #. | ||||
| #: js/ui/calendar.js:397 | ||||
| #: js/ui/calendar.js:392 | ||||
| msgid "%OB" | ||||
| msgstr "%OB" | ||||
|  | ||||
| @@ -826,61 +826,37 @@ msgstr "%OB" | ||||
| #. * in most cases you should not use the old "%B" here unless you | ||||
| #. * absolutely know what you are doing. | ||||
| #. | ||||
| #: js/ui/calendar.js:407 | ||||
| #: js/ui/calendar.js:402 | ||||
| msgid "%OB %Y" | ||||
| msgstr "%Y %OB" | ||||
|  | ||||
| #: js/ui/calendar.js:466 | ||||
| #: js/ui/calendar.js:461 | ||||
| msgid "Previous month" | ||||
| msgstr "上个月" | ||||
|  | ||||
| #: js/ui/calendar.js:481 | ||||
| #: js/ui/calendar.js:476 | ||||
| msgid "Next month" | ||||
| msgstr "下个月" | ||||
|  | ||||
| #: js/ui/calendar.js:631 | ||||
| #: js/ui/calendar.js:626 | ||||
| #, no-javascript-format | ||||
| msgctxt "date day number format" | ||||
| msgid "%d" | ||||
| msgstr "%d" | ||||
|  | ||||
| #: js/ui/calendar.js:687 | ||||
| #: js/ui/calendar.js:682 | ||||
| msgid "Week %V" | ||||
| msgstr "第 %V 个星期" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/calendar.js:762 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "全天" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/calendar.js:900 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d" | ||||
| msgstr "%-m月%-d日 %A" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/calendar.js:903 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%A, %B %-d, %Y" | ||||
| msgstr "%Y年%-m月%-d日 %A" | ||||
|  | ||||
| #: js/ui/calendar.js:1133 | ||||
| #: js/ui/calendar.js:895 | ||||
| msgid "No Notifications" | ||||
| msgstr "无通知" | ||||
|  | ||||
| #: js/ui/calendar.js:1136 | ||||
| msgid "No Events" | ||||
| msgstr "无事件" | ||||
|  | ||||
| #: js/ui/calendar.js:1190 | ||||
| #: js/ui/calendar.js:949 | ||||
| msgid "Do Not Disturb" | ||||
| msgstr "请勿打扰" | ||||
|  | ||||
| #: js/ui/calendar.js:1209 | ||||
| #: js/ui/calendar.js:968 | ||||
| msgid "Clear" | ||||
| msgstr "清除" | ||||
|  | ||||
| @@ -1030,7 +1006,7 @@ msgstr "抱歉,认证失败。请重试。" | ||||
| msgid "%s is now known as %s" | ||||
| msgstr "%s 现在叫做 %s" | ||||
|  | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:177 | ||||
| #: js/ui/ctrlAltTab.js:21 js/ui/viewSelector.js:178 | ||||
| msgid "Windows" | ||||
| msgstr "窗口" | ||||
|  | ||||
| @@ -1049,7 +1025,7 @@ msgstr "Dash" | ||||
| #. * "Tue 9:29 AM").  The string itself should become a full date, e.g., | ||||
| #. * "February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:75 | ||||
| #: js/ui/dateMenu.js:79 | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%Y年%-m月%-d日" | ||||
|  | ||||
| @@ -1057,35 +1033,67 @@ msgstr "%Y年%-m月%-d日" | ||||
| #. * below the time in the shell; it should combine the weekday and the | ||||
| #. * date, e.g. "Tuesday February 17 2015". | ||||
| #. | ||||
| #: js/ui/dateMenu.js:82 | ||||
| #: js/ui/dateMenu.js:86 | ||||
| msgid "%A %B %e %Y" | ||||
| msgstr "%Y年%-m月%-d日 %A" | ||||
|  | ||||
| #: js/ui/dateMenu.js:162 | ||||
| #. Translators: Shown on calendar heading when selected day occurs on current year | ||||
| #: js/ui/dateMenu.js:151 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d" | ||||
| msgstr "%-m月%-d日" | ||||
|  | ||||
| #. Translators: Shown on calendar heading when selected day occurs on different year | ||||
| #: js/ui/dateMenu.js:154 | ||||
| msgctxt "calendar heading" | ||||
| msgid "%B %-d %Y" | ||||
| msgstr "%Y年%-m月%-d日" | ||||
|  | ||||
| #: js/ui/dateMenu.js:160 | ||||
| msgid "Today" | ||||
| msgstr "今天" | ||||
|  | ||||
| #: js/ui/dateMenu.js:164 | ||||
| msgid "Tomorrow" | ||||
| msgstr "明天" | ||||
|  | ||||
| #. Translators: Shown in calendar event list for all day events | ||||
| #. * Keep it short, best if you can use less then 10 characters | ||||
| #. | ||||
| #: js/ui/dateMenu.js:180 | ||||
| msgctxt "event list time" | ||||
| msgid "All Day" | ||||
| msgstr "全天" | ||||
|  | ||||
| #: js/ui/dateMenu.js:231 | ||||
| msgid "No Events" | ||||
| msgstr "无事件" | ||||
|  | ||||
| #: js/ui/dateMenu.js:348 | ||||
| msgid "Add world clocks…" | ||||
| msgstr "添加世界时钟…" | ||||
|  | ||||
| #: js/ui/dateMenu.js:163 | ||||
| #: js/ui/dateMenu.js:349 | ||||
| msgid "World Clocks" | ||||
| msgstr "世界时钟" | ||||
|  | ||||
| #: js/ui/dateMenu.js:443 | ||||
| #: js/ui/dateMenu.js:629 | ||||
| msgid "Loading…" | ||||
| msgstr "正在载入……" | ||||
|  | ||||
| #: js/ui/dateMenu.js:453 | ||||
| #: js/ui/dateMenu.js:639 | ||||
| msgid "Go online for weather information" | ||||
| msgstr "通过互联网查看天气信息" | ||||
|  | ||||
| #: js/ui/dateMenu.js:455 | ||||
| #: js/ui/dateMenu.js:641 | ||||
| msgid "Weather information is currently unavailable" | ||||
| msgstr "天气信息目前不可用" | ||||
|  | ||||
| #: js/ui/dateMenu.js:465 | ||||
| #: js/ui/dateMenu.js:651 | ||||
| msgid "Weather" | ||||
| msgstr "天气" | ||||
|  | ||||
| #: js/ui/dateMenu.js:467 | ||||
| #: js/ui/dateMenu.js:653 | ||||
| msgid "Select weather location…" | ||||
| msgstr "选择天气地点…" | ||||
|  | ||||
| @@ -1268,11 +1276,11 @@ msgstr "某应用程序希望禁用快捷键" | ||||
| msgid "You can restore shortcuts by pressing %s." | ||||
| msgstr "按 %s 以恢复快捷键。" | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:98 | ||||
| #: js/ui/inhibitShortcutsDialog.js:100 | ||||
| msgid "Deny" | ||||
| msgstr "拒绝" | ||||
|  | ||||
| #: js/ui/inhibitShortcutsDialog.js:105 | ||||
| #: js/ui/inhibitShortcutsDialog.js:107 | ||||
| msgid "Allow" | ||||
| msgstr "允许" | ||||
|  | ||||
| @@ -1337,7 +1345,7 @@ msgstr "关闭" | ||||
| msgid "Leave Off" | ||||
| msgstr "保持关闭" | ||||
|  | ||||
| #: js/ui/keyboard.js:207 | ||||
| #: js/ui/keyboard.js:225 | ||||
| msgid "Region & Language Settings" | ||||
| msgstr "区域与语言设置" | ||||
|  | ||||
| @@ -1410,25 +1418,25 @@ msgstr "屏幕锁定已禁用" | ||||
| msgid "Screen Locking requires the GNOME display manager." | ||||
| msgstr "屏幕锁定需要 GNOME 显示管理器。" | ||||
|  | ||||
| #: js/ui/messageTray.js:1547 | ||||
| #: js/ui/messageTray.js:1476 | ||||
| msgid "System Information" | ||||
| msgstr "系统信息" | ||||
|  | ||||
| #: js/ui/mpris.js:204 | ||||
| #: js/ui/mpris.js:203 | ||||
| msgid "Unknown artist" | ||||
| msgstr "未知艺人" | ||||
|  | ||||
| #: js/ui/mpris.js:214 | ||||
| #: js/ui/mpris.js:213 | ||||
| msgid "Unknown title" | ||||
| msgstr "未知标题" | ||||
|  | ||||
| #: js/ui/overview.js:73 | ||||
| #: js/ui/overview.js:74 | ||||
| msgid "Undo" | ||||
| msgstr "撤消" | ||||
|  | ||||
| #. Translators: This is the main view to select | ||||
| #. activities. See also note for "Activities" string. | ||||
| #: js/ui/overview.js:86 | ||||
| #: js/ui/overview.js:87 | ||||
| msgid "Overview" | ||||
| msgstr "概览" | ||||
|  | ||||
| @@ -1436,7 +1444,7 @@ msgstr "概览" | ||||
| #. in the search entry when no search is | ||||
| #. active; it should not exceed ~30 | ||||
| #. characters. | ||||
| #: js/ui/overview.js:107 | ||||
| #: js/ui/overview.js:108 | ||||
| msgid "Type to search" | ||||
| msgstr "输入以搜索" | ||||
|  | ||||
| @@ -1464,23 +1472,23 @@ msgstr "分配按键" | ||||
| msgid "Done" | ||||
| msgstr "完成" | ||||
|  | ||||
| #: js/ui/padOsd.js:745 | ||||
| #: js/ui/padOsd.js:732 | ||||
| msgid "Edit…" | ||||
| msgstr "编辑…" | ||||
|  | ||||
| #: js/ui/padOsd.js:787 js/ui/padOsd.js:910 | ||||
| #: js/ui/padOsd.js:774 js/ui/padOsd.js:891 | ||||
| msgid "None" | ||||
| msgstr "无" | ||||
|  | ||||
| #: js/ui/padOsd.js:863 | ||||
| #: js/ui/padOsd.js:845 | ||||
| msgid "Press a button to configure" | ||||
| msgstr "按下按键以配置" | ||||
|  | ||||
| #: js/ui/padOsd.js:864 | ||||
| #: js/ui/padOsd.js:846 | ||||
| msgid "Press Esc to exit" | ||||
| msgstr "按 Esc 以退出" | ||||
|  | ||||
| #: js/ui/padOsd.js:867 | ||||
| #: js/ui/padOsd.js:849 | ||||
| msgid "Press any key to exit" | ||||
| msgstr "按任意键退出" | ||||
|  | ||||
| @@ -1570,7 +1578,7 @@ msgstr "隐藏文本" | ||||
|  | ||||
| #: js/ui/shellEntry.js:162 | ||||
| msgid "Caps lock is on." | ||||
| msgstr "大写锁定键已开启。" | ||||
| msgstr "大写锁定已开启。" | ||||
|  | ||||
| #: js/ui/shellMountOperation.js:285 | ||||
| msgid "Hidden Volume" | ||||
| @@ -1742,17 +1750,17 @@ msgstr "定位服务已禁用" | ||||
| msgid "Enable" | ||||
| msgstr "启用" | ||||
|  | ||||
| #: js/ui/status/location.js:355 | ||||
| #: js/ui/status/location.js:350 | ||||
| msgid "Allow location access" | ||||
| msgstr "允许获取位置信息" | ||||
|  | ||||
| #. Translators: %s is an application name | ||||
| #: js/ui/status/location.js:357 | ||||
| #: js/ui/status/location.js:352 | ||||
| #, javascript-format | ||||
| msgid "The app %s wants to access your location" | ||||
| msgstr "应用 %s 想要获取您的位置信息" | ||||
|  | ||||
| #: js/ui/status/location.js:367 | ||||
| #: js/ui/status/location.js:362 | ||||
| msgid "Location access can be changed at any time from the privacy settings." | ||||
| msgstr "位置访问权限可以随时在隐私设置里更改。" | ||||
|  | ||||
| @@ -2081,11 +2089,11 @@ msgstr "Thunderbolt 授权错误" | ||||
| msgid "Could not authorize the Thunderbolt device: %s" | ||||
| msgstr "无法授权 Thunderbolt 设备:%s" | ||||
|  | ||||
| #: js/ui/status/volume.js:154 | ||||
| #: js/ui/status/volume.js:155 | ||||
| msgid "Volume changed" | ||||
| msgstr "音量已变更" | ||||
|  | ||||
| #: js/ui/status/volume.js:225 | ||||
| #: js/ui/status/volume.js:217 | ||||
| msgid "Volume" | ||||
| msgstr "音量" | ||||
|  | ||||
| @@ -2131,19 +2139,19 @@ msgstr "向上滑动解锁" | ||||
| msgid "Click or press a key to unlock" | ||||
| msgstr "单击或按键解锁" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:550 | ||||
| #: js/ui/unlockDialog.js:555 | ||||
| msgid "Unlock Window" | ||||
| msgstr "解锁窗口" | ||||
|  | ||||
| #: js/ui/unlockDialog.js:559 | ||||
| #: js/ui/unlockDialog.js:564 | ||||
| msgid "Log in as another user" | ||||
| msgstr "以另一个用户身份登录" | ||||
|  | ||||
| #: js/ui/viewSelector.js:181 | ||||
| #: js/ui/viewSelector.js:182 | ||||
| msgid "Applications" | ||||
| msgstr "应用程序" | ||||
|  | ||||
| #: js/ui/viewSelector.js:185 | ||||
| #: js/ui/viewSelector.js:186 | ||||
| msgid "Search" | ||||
| msgstr "搜索" | ||||
|  | ||||
| @@ -2269,12 +2277,12 @@ msgstr "使用指定模式,如“gdm”用于登录屏幕" | ||||
| msgid "List possible modes" | ||||
| msgstr "列出可用的模式" | ||||
|  | ||||
| #: src/shell-app.c:286 | ||||
| #: src/shell-app.c:268 | ||||
| msgctxt "program" | ||||
| msgid "Unknown" | ||||
| msgstr "未知" | ||||
|  | ||||
| #: src/shell-app.c:537 | ||||
| #: src/shell-app.c:519 | ||||
| #, c-format | ||||
| msgid "Failed to launch “%s”" | ||||
| msgstr "启动“%s”失败" | ||||
| @@ -2504,7 +2512,7 @@ msgstr "描述" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:443 | ||||
| msgid "A short description of what the extension does" | ||||
| msgstr "简短描述下扩展所做的事情" | ||||
| msgstr "扩展功能的简短描述" | ||||
|  | ||||
| #: subprojects/extensions-tool/src/command-create.c:446 | ||||
| msgid "TEMPLATE" | ||||
| @@ -2847,6 +2855,14 @@ msgstr[0] "%u 个输入" | ||||
| msgid "System Sounds" | ||||
| msgstr "系统声音" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d" | ||||
| #~ msgstr "%-m月%-d日 %A" | ||||
|  | ||||
| #~ msgctxt "calendar heading" | ||||
| #~ msgid "%A, %B %-d, %Y" | ||||
| #~ msgstr "%Y年%-m月%-d日 %A" | ||||
|  | ||||
| #~ msgid "Frequently used applications will appear here" | ||||
| #~ msgstr "常用的应用程序会出现在这里" | ||||
|  | ||||
|   | ||||
| @@ -346,8 +346,6 @@ struct _App | ||||
|  | ||||
|   GSList *notify_appointments; /* CalendarAppointment *, for EventsAdded */ | ||||
|   GSList *notify_ids; /* gchar *, for EventsRemoved */ | ||||
|   guint events_added_timeout_id; | ||||
|   guint events_removed_timeout_id; | ||||
|  | ||||
|   GSList *live_views; | ||||
| }; | ||||
| @@ -370,24 +368,19 @@ app_update_timezone (App *app) | ||||
|     } | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| on_app_schedule_events_added_cb (gpointer user_data) | ||||
| static void | ||||
| app_notify_events_added (App *app) | ||||
| { | ||||
|   App *app = user_data; | ||||
|   GVariantBuilder builder, extras_builder; | ||||
|   GSList *events, *link; | ||||
|  | ||||
|   if (g_source_is_destroyed (g_main_current_source ())) | ||||
|     return FALSE; | ||||
|  | ||||
|   events = g_slist_reverse (app->notify_appointments); | ||||
|   app->notify_appointments = NULL; | ||||
|   app->events_added_timeout_id = 0; | ||||
|  | ||||
|   print_debug ("Emitting EventsAddedOrUpdated with %d events", g_slist_length (events)); | ||||
|  | ||||
|   if (!events) | ||||
|     return FALSE; | ||||
|     return; | ||||
|  | ||||
|   /* The a{sv} is used as an escape hatch in case we want to provide more | ||||
|    * information in the future without breaking ABI | ||||
| @@ -428,41 +421,21 @@ on_app_schedule_events_added_cb (gpointer user_data) | ||||
|   g_variant_builder_clear (&builder); | ||||
|  | ||||
|   g_slist_free_full (events, calendar_appointment_free); | ||||
|  | ||||
|   return FALSE; | ||||
| } | ||||
|  | ||||
| static void | ||||
| app_schedule_events_added (App *app) | ||||
| app_notify_events_removed (App *app) | ||||
| { | ||||
|   print_debug ("Scheduling EventsAddedOrUpdated"); | ||||
|   if (app->events_added_timeout_id == 0) | ||||
|     { | ||||
|       app->events_added_timeout_id = g_timeout_add_seconds (2, | ||||
|                                                             on_app_schedule_events_added_cb, | ||||
|                                                             app); | ||||
|       g_source_set_name_by_id (app->events_added_timeout_id, "[gnome-shell] on_app_schedule_events_added_cb"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| on_app_schedule_events_removed_cb (gpointer user_data) | ||||
| { | ||||
|   App *app = user_data; | ||||
|   GVariantBuilder builder; | ||||
|   GSList *ids, *link; | ||||
|  | ||||
|   if (g_source_is_destroyed (g_main_current_source ())) | ||||
|     return FALSE; | ||||
|  | ||||
|   ids = app->notify_ids; | ||||
|   app->notify_ids = NULL; | ||||
|   app->events_removed_timeout_id = 0; | ||||
|  | ||||
|   print_debug ("Emitting EventsRemoved with %d ids", g_slist_length (ids)); | ||||
|  | ||||
|   if (!ids) | ||||
|     return FALSE; | ||||
|     return; | ||||
|  | ||||
|   g_variant_builder_init (&builder, G_VARIANT_TYPE ("as")); | ||||
|   for (link = ids; link; link = g_slist_next (link)) | ||||
| @@ -483,20 +456,7 @@ on_app_schedule_events_removed_cb (gpointer user_data) | ||||
|  | ||||
|   g_slist_free_full (ids, g_free); | ||||
|  | ||||
|   return FALSE; | ||||
| } | ||||
|  | ||||
| static void | ||||
| app_schedule_events_removed (App *app) | ||||
| { | ||||
|   print_debug ("Scheduling EventsRemoved"); | ||||
|   if (app->events_removed_timeout_id == 0) | ||||
|     { | ||||
|       app->events_removed_timeout_id = g_timeout_add_seconds (2, | ||||
|                                                               on_app_schedule_events_removed_cb, | ||||
|                                                               app); | ||||
|       g_source_set_name_by_id (app->events_removed_timeout_id, "[gnome-shell] on_app_schedule_events_removed_cb"); | ||||
|     } | ||||
|   return; | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -546,7 +506,7 @@ app_process_added_modified_objects (App *app, | ||||
|   g_clear_object (&cal_client); | ||||
|  | ||||
|   if (app->notify_appointments) | ||||
|     app_schedule_events_added (app); | ||||
|     app_notify_events_added (app); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -610,7 +570,7 @@ on_objects_removed (ECalClientView *view, | ||||
|   g_clear_object (&client); | ||||
|  | ||||
|   if (app->notify_ids) | ||||
|     app_schedule_events_removed (app); | ||||
|     app_notify_events_removed (app); | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| @@ -874,9 +834,6 @@ app_free (App *app) | ||||
| { | ||||
|   GSList *ll; | ||||
|  | ||||
|   g_clear_handle_id (&app->events_added_timeout_id, g_source_remove); | ||||
|   g_clear_handle_id (&app->events_removed_timeout_id, g_source_remove); | ||||
|  | ||||
|   for (ll = app->live_views; ll != NULL; ll = g_slist_next (ll)) | ||||
|     { | ||||
|       ECalClientView *view = E_CAL_CLIENT_VIEW (ll->data); | ||||
|   | ||||
| @@ -78,6 +78,7 @@ struct _ShellApp | ||||
|                           * want (e.g. it will be of TYPE_NORMAL from | ||||
|                           * the way shell-window-tracker.c works). | ||||
|                           */ | ||||
|   GIcon *fallback_icon; | ||||
|  | ||||
|   ShellAppRunningState *running_state; | ||||
|  | ||||
| @@ -180,66 +181,47 @@ window_backed_app_get_window (ShellApp     *app) | ||||
|     return NULL; | ||||
| } | ||||
|  | ||||
| static ClutterActor * | ||||
| window_backed_app_get_icon (ShellApp *app, | ||||
|                             int       size) | ||||
| /** | ||||
|  * shell_app_get_icon: | ||||
|  * | ||||
|  * Look up the icon for this application | ||||
|  * | ||||
|  * Return value: (transfer none): A #GIcon | ||||
|  */ | ||||
| GIcon * | ||||
| shell_app_get_icon (ShellApp *app) | ||||
| { | ||||
|   MetaWindow *window = NULL; | ||||
|   StWidget *widget; | ||||
|   int scale, scaled_size; | ||||
|   ShellGlobal *global; | ||||
|   StThemeContext *context; | ||||
|  | ||||
|   global = shell_global_get (); | ||||
|   context = st_theme_context_get_for_stage (shell_global_get_stage (global)); | ||||
|   g_object_get (context, "scale-factor", &scale, NULL); | ||||
|   g_return_val_if_fail (SHELL_IS_APP (app), NULL); | ||||
|  | ||||
|   scaled_size = size * scale; | ||||
|   if (app->info) | ||||
|     return g_app_info_get_icon (G_APP_INFO (app->info)); | ||||
|  | ||||
|   if (app->fallback_icon) | ||||
|     return app->fallback_icon; | ||||
|  | ||||
|   /* During a state transition from running to not-running for | ||||
|    * window-backend apps, it's possible we get a request for the icon. | ||||
|    * Avoid asserting here and just return an empty image. | ||||
|    * Avoid asserting here and just return a fallback icon | ||||
|    */ | ||||
|   if (app->running_state != NULL) | ||||
|     window = window_backed_app_get_window (app); | ||||
|  | ||||
|   if (window == NULL) | ||||
|   if (window && | ||||
|       meta_window_get_client_type (window) == META_WINDOW_CLIENT_TYPE_X11) | ||||
|     { | ||||
|       ClutterActor *actor; | ||||
|  | ||||
|       actor = clutter_actor_new (); | ||||
|       g_object_set (actor, | ||||
|                     "opacity", 0, | ||||
|                     "width", (float) scaled_size, | ||||
|                     "height", (float) scaled_size, | ||||
|                     NULL); | ||||
|       return actor; | ||||
|     } | ||||
|  | ||||
|   if (meta_window_get_client_type (window) == META_WINDOW_CLIENT_TYPE_X11) | ||||
|     { | ||||
|       StWidget *texture_actor; | ||||
|  | ||||
|       texture_actor = | ||||
|       app->fallback_icon = | ||||
|         st_texture_cache_bind_cairo_surface_property (st_texture_cache_get_default (), | ||||
|                                                       G_OBJECT (window), | ||||
|                                                       "icon", | ||||
|                                                       scaled_size); | ||||
|  | ||||
|       widget = g_object_new (ST_TYPE_BIN, | ||||
|                              "child", texture_actor, | ||||
|                              NULL); | ||||
|                                                       "icon"); | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       widget = g_object_new (ST_TYPE_ICON, | ||||
|                              "icon-size", size, | ||||
|                              "icon-name", "application-x-executable", | ||||
|                              NULL); | ||||
|       app->fallback_icon = g_themed_icon_new ("application-x-executable"); | ||||
|     } | ||||
|   st_widget_add_style_class_name (widget, "fallback-app-icon"); | ||||
|  | ||||
|   return CLUTTER_ACTOR (widget); | ||||
|   return app->fallback_icon; | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -257,16 +239,16 @@ shell_app_create_icon_texture (ShellApp   *app, | ||||
|   GIcon *icon; | ||||
|   ClutterActor *ret; | ||||
|  | ||||
|   if (app->info == NULL) | ||||
|     return window_backed_app_get_icon (app, size); | ||||
|  | ||||
|   ret = st_icon_new (); | ||||
|   st_icon_set_icon_size (ST_ICON (ret), size); | ||||
|   st_icon_set_fallback_icon_name (ST_ICON (ret), "application-x-executable"); | ||||
|  | ||||
|   icon = g_app_info_get_icon (G_APP_INFO (app->info)); | ||||
|   icon = shell_app_get_icon (app); | ||||
|   st_icon_set_gicon (ST_ICON (ret), icon); | ||||
|  | ||||
|   if (shell_app_is_window_backed (app)) | ||||
|     st_widget_add_style_class_name (ST_WIDGET (ret), "fallback-app-icon"); | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| @@ -1576,6 +1558,7 @@ shell_app_dispose (GObject *object) | ||||
|   ShellApp *app = SHELL_APP (object); | ||||
|  | ||||
|   g_clear_object (&app->info); | ||||
|   g_clear_object (&app->fallback_icon); | ||||
|  | ||||
|   while (app->running_state) | ||||
|     _shell_app_remove_window (app, app->running_state->windows->data); | ||||
|   | ||||
| @@ -29,6 +29,7 @@ const char *shell_app_get_id (ShellApp *app); | ||||
| GDesktopAppInfo *shell_app_get_app_info (ShellApp *app); | ||||
|  | ||||
| ClutterActor *shell_app_create_icon_texture (ShellApp *app, int size); | ||||
| GIcon *shell_app_get_icon (ShellApp *app); | ||||
| const char *shell_app_get_name (ShellApp *app); | ||||
| const char *shell_app_get_description (ShellApp *app); | ||||
| gboolean shell_app_is_window_backed (ShellApp *app); | ||||
|   | ||||
| @@ -899,8 +899,9 @@ load_gl_symbol (const char  *name, | ||||
| } | ||||
|  | ||||
| static void | ||||
| global_stage_after_paint (ClutterStage *stage, | ||||
|                           ShellGlobal  *global) | ||||
| global_stage_after_paint (ClutterStage     *stage, | ||||
|                           ClutterStageView *stage_view, | ||||
|                           ShellGlobal      *global) | ||||
| { | ||||
|   /* At this point, we've finished all layout and painting, but haven't | ||||
|    * actually flushed or swapped */ | ||||
|   | ||||
| @@ -508,9 +508,7 @@ recorder_update_size (ShellRecorder *recorder) | ||||
| } | ||||
|  | ||||
| static void | ||||
| recorder_on_stage_notify_size (GObject          *object, | ||||
|                                GParamSpec       *pspec, | ||||
|                                ShellRecorder    *recorder) | ||||
| recorder_on_stage_notify_size (ShellRecorder *recorder) | ||||
| { | ||||
|   recorder_update_size (recorder); | ||||
|  | ||||
| @@ -614,12 +612,15 @@ recorder_connect_stage_callbacks (ShellRecorder *recorder) | ||||
|                     G_CALLBACK (recorder_on_stage_destroy), recorder); | ||||
|   g_signal_connect_after (recorder->stage, "paint", | ||||
|                           G_CALLBACK (recorder_on_stage_paint), recorder); | ||||
|   g_signal_connect (recorder->stage, "notify::width", | ||||
|                     G_CALLBACK (recorder_on_stage_notify_size), recorder); | ||||
|   g_signal_connect (recorder->stage, "notify::height", | ||||
|                     G_CALLBACK (recorder_on_stage_notify_size), recorder); | ||||
|   g_signal_connect (recorder->stage, "notify::resource-scale", | ||||
|                     G_CALLBACK (recorder_on_stage_notify_size), recorder); | ||||
|   g_signal_connect_swapped (recorder->stage, "notify::width", | ||||
|                             G_CALLBACK (recorder_on_stage_notify_size), | ||||
|                             recorder); | ||||
|   g_signal_connect_swapped (recorder->stage, "notify::height", | ||||
|                             G_CALLBACK (recorder_on_stage_notify_size), | ||||
|                             recorder); | ||||
|   g_signal_connect_swapped (recorder->stage, "resource-scale-changed", | ||||
|                             G_CALLBACK (recorder_on_stage_notify_size), | ||||
|                             recorder); | ||||
| } | ||||
|  | ||||
| static void | ||||
|   | ||||
| @@ -393,8 +393,7 @@ grab_window_screenshot (ClutterActor *stage, | ||||
|       if (meta_window_get_client_type (window) == META_WINDOW_CLIENT_TYPE_WAYLAND) | ||||
|         { | ||||
|           float resource_scale; | ||||
|           if (!clutter_actor_get_resource_scale (window_actor, &resource_scale)) | ||||
|             resource_scale = 1.0f; | ||||
|           resource_scale = clutter_actor_get_resource_scale (window_actor); | ||||
|  | ||||
|           cairo_surface_set_device_scale (priv->image, resource_scale, resource_scale); | ||||
|         } | ||||
|   | ||||
| @@ -409,34 +409,6 @@ shell_util_create_pixbuf_from_data (const guchar      *data, | ||||
|  | ||||
| typedef const gchar *(*ShellGLGetString) (GLenum); | ||||
|  | ||||
| static const gchar * | ||||
| get_gl_vendor (void) | ||||
| { | ||||
|   static const gchar *vendor = NULL; | ||||
|  | ||||
|   if (!vendor) | ||||
|     { | ||||
|       ShellGLGetString gl_get_string; | ||||
|       gl_get_string = (ShellGLGetString) cogl_get_proc_address ("glGetString"); | ||||
|       if (gl_get_string) | ||||
|         vendor = gl_get_string (GL_VENDOR); | ||||
|     } | ||||
|  | ||||
|   return vendor; | ||||
| } | ||||
|  | ||||
| gboolean | ||||
| shell_util_need_background_refresh (void) | ||||
| { | ||||
|   if (!clutter_check_windowing_backend (CLUTTER_WINDOWING_X11)) | ||||
|     return FALSE; | ||||
|  | ||||
|   if (g_strcmp0 (get_gl_vendor (), "NVIDIA Corporation") == 0) | ||||
|     return TRUE; | ||||
|  | ||||
|   return FALSE; | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| canvas_draw_cb (ClutterContent *content, | ||||
|                 cairo_t        *cr, | ||||
|   | ||||
| @@ -49,8 +49,6 @@ GdkPixbuf *shell_util_create_pixbuf_from_data (const guchar      *data, | ||||
|                                                int                height, | ||||
|                                                int                rowstride); | ||||
|  | ||||
| gboolean shell_util_need_background_refresh (void); | ||||
|  | ||||
| ClutterContent * shell_util_get_content_for_window_actor (MetaWindowActor *window_actor, | ||||
|                                                           MetaRectangle   *window_rect); | ||||
|  | ||||
|   | ||||
| @@ -119,6 +119,16 @@ shell_window_tracker_class_init (ShellWindowTrackerClass *klass) | ||||
|                                                    G_TYPE_NONE, 0); | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| check_app_id_prefix (ShellApp   *app, | ||||
|                      const char *prefix) | ||||
| { | ||||
|   if (prefix == NULL) | ||||
|     return TRUE; | ||||
|  | ||||
|   return g_str_has_prefix (shell_app_get_id (app), prefix); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * get_app_from_window_wmclass: | ||||
|  * | ||||
| @@ -135,9 +145,15 @@ get_app_from_window_wmclass (MetaWindow  *window) | ||||
|   ShellAppSystem *appsys; | ||||
|   const char *wm_class; | ||||
|   const char *wm_instance; | ||||
|   const char *sandbox_id; | ||||
|   g_autofree char *app_prefix = NULL; | ||||
|  | ||||
|   appsys = shell_app_system_get_default (); | ||||
|  | ||||
|   sandbox_id = meta_window_get_sandboxed_app_id (window); | ||||
|   if (sandbox_id) | ||||
|     app_prefix = g_strdup_printf ("%s.", sandbox_id); | ||||
|  | ||||
|   /* Notes on the heuristics used here: | ||||
|      much of the complexity here comes from the desire to support | ||||
|      Chrome apps. | ||||
| @@ -176,23 +192,23 @@ get_app_from_window_wmclass (MetaWindow  *window) | ||||
|   /* first try a match from WM_CLASS (instance part) to StartupWMClass */ | ||||
|   wm_instance = meta_window_get_wm_class_instance (window); | ||||
|   app = shell_app_system_lookup_startup_wmclass (appsys, wm_instance); | ||||
|   if (app != NULL) | ||||
|   if (app != NULL && check_app_id_prefix (app, app_prefix)) | ||||
|     return g_object_ref (app); | ||||
|  | ||||
|   /* then try a match from WM_CLASS to StartupWMClass */ | ||||
|   wm_class = meta_window_get_wm_class (window); | ||||
|   app = shell_app_system_lookup_startup_wmclass (appsys, wm_class); | ||||
|   if (app != NULL) | ||||
|   if (app != NULL && check_app_id_prefix (app, app_prefix)) | ||||
|     return g_object_ref (app); | ||||
|  | ||||
|   /* then try a match from WM_CLASS (instance part) to .desktop */ | ||||
|   app = shell_app_system_lookup_desktop_wmclass (appsys, wm_instance); | ||||
|   if (app != NULL) | ||||
|   if (app != NULL && check_app_id_prefix (app, app_prefix)) | ||||
|     return g_object_ref (app); | ||||
|  | ||||
|   /* finally, try a match from WM_CLASS to .desktop */ | ||||
|   app = shell_app_system_lookup_desktop_wmclass (appsys, wm_class); | ||||
|   if (app != NULL) | ||||
|   if (app != NULL && check_app_id_prefix (app, app_prefix)) | ||||
|     return g_object_ref (app); | ||||
|  | ||||
|   return NULL; | ||||
| @@ -214,7 +230,7 @@ get_app_from_id (MetaWindow  *window, | ||||
| { | ||||
|   ShellApp *app; | ||||
|   ShellAppSystem *appsys; | ||||
|   char *desktop_file; | ||||
|   g_autofree char *desktop_file = NULL; | ||||
|  | ||||
|   g_return_val_if_fail (id != NULL, NULL); | ||||
|  | ||||
| @@ -223,10 +239,9 @@ get_app_from_id (MetaWindow  *window, | ||||
|   desktop_file = g_strconcat (id, ".desktop", NULL); | ||||
|   app = shell_app_system_lookup_app (appsys, desktop_file); | ||||
|   if (app) | ||||
|     g_object_ref (app); | ||||
|     return g_object_ref (app); | ||||
|  | ||||
|   g_free (desktop_file); | ||||
|   return app; | ||||
|   return NULL; | ||||
| } | ||||
|  | ||||
| /* | ||||
| @@ -391,6 +406,13 @@ get_app_for_window (ShellWindowTracker    *tracker, | ||||
|   if (meta_window_is_remote (window)) | ||||
|     return _shell_app_new_for_window (window); | ||||
|  | ||||
|   /* Check if the app's WM_CLASS specifies an app; this is | ||||
|    * canonical if it does. | ||||
|    */ | ||||
|   result = get_app_from_window_wmclass (window); | ||||
|   if (result != NULL) | ||||
|     return result; | ||||
|  | ||||
|   /* Check if the window was opened from within a sandbox; if this | ||||
|    * is the case, a corresponding .desktop file is guaranteed to match; | ||||
|    */ | ||||
| @@ -405,13 +427,6 @@ get_app_for_window (ShellWindowTracker    *tracker, | ||||
|   if (result != NULL) | ||||
|     return result; | ||||
|  | ||||
|   /* Check if the app's WM_CLASS specifies an app; this is | ||||
|    * canonical if it does. | ||||
|    */ | ||||
|   result = get_app_from_window_wmclass (window); | ||||
|   if (result != NULL) | ||||
|     return result; | ||||
|  | ||||
|   result = get_app_from_window_pid (tracker, window); | ||||
|   if (result != NULL) | ||||
|     return result; | ||||
|   | ||||
| @@ -41,6 +41,8 @@ typedef struct _StAdjustmentPrivate StAdjustmentPrivate; | ||||
|  | ||||
| struct _StAdjustmentPrivate | ||||
| { | ||||
|   ClutterActor *actor; | ||||
|  | ||||
|   /* Do not sanity-check values while constructing, | ||||
|    * not all properties may be set yet. */ | ||||
|   guint is_constructing : 1; | ||||
| @@ -66,6 +68,7 @@ enum | ||||
| { | ||||
|   PROP_0, | ||||
|  | ||||
|   PROP_ACTOR, | ||||
|   PROP_LOWER, | ||||
|   PROP_UPPER, | ||||
|   PROP_VALUE, | ||||
| @@ -106,9 +109,21 @@ static gboolean st_adjustment_set_page_increment (StAdjustment *adjustment, | ||||
| static gboolean st_adjustment_set_page_size      (StAdjustment *adjustment, | ||||
|                                                   gdouble       size); | ||||
|  | ||||
| static ClutterActor * | ||||
| st_adjustment_get_actor (ClutterAnimatable *animatable) | ||||
| { | ||||
|   StAdjustment *adjustment = ST_ADJUSTMENT (animatable); | ||||
|   StAdjustmentPrivate *priv = st_adjustment_get_instance_private (adjustment); | ||||
|  | ||||
|   g_warn_if_fail (priv->actor); | ||||
|  | ||||
|   return priv->actor; | ||||
| } | ||||
|  | ||||
| static void | ||||
| animatable_iface_init (ClutterAnimatableInterface *iface) | ||||
| { | ||||
|   iface->get_actor = st_adjustment_get_actor; | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -141,6 +156,10 @@ st_adjustment_get_property (GObject    *gobject, | ||||
|  | ||||
|   switch (prop_id) | ||||
|     { | ||||
|     case PROP_ACTOR: | ||||
|       g_value_set_object (value, priv->actor); | ||||
|       break; | ||||
|  | ||||
|     case PROP_LOWER: | ||||
|       g_value_set_double (value, priv->lower); | ||||
|       break; | ||||
| @@ -171,6 +190,18 @@ st_adjustment_get_property (GObject    *gobject, | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void | ||||
| actor_destroyed (gpointer  user_data, | ||||
|                  GObject  *where_the_object_was) | ||||
| { | ||||
|   StAdjustment *adj = ST_ADJUSTMENT (user_data); | ||||
|   StAdjustmentPrivate *priv = st_adjustment_get_instance_private (adj); | ||||
|  | ||||
|   priv->actor = NULL; | ||||
|  | ||||
|   g_object_notify_by_pspec (G_OBJECT (adj), props[PROP_ACTOR]); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_adjustment_set_property (GObject      *gobject, | ||||
|                             guint         prop_id, | ||||
| @@ -178,9 +209,20 @@ st_adjustment_set_property (GObject      *gobject, | ||||
|                             GParamSpec   *pspec) | ||||
| { | ||||
|   StAdjustment *adj = ST_ADJUSTMENT (gobject); | ||||
|   StAdjustmentPrivate *priv; | ||||
|  | ||||
|   priv = st_adjustment_get_instance_private (ST_ADJUSTMENT (gobject)); | ||||
|  | ||||
|   switch (prop_id) | ||||
|     { | ||||
|     case PROP_ACTOR: | ||||
|       if (priv->actor) | ||||
|         g_object_weak_unref (G_OBJECT (priv->actor), actor_destroyed, adj); | ||||
|       priv->actor = g_value_get_object (value); | ||||
|       if (priv->actor) | ||||
|         g_object_weak_ref (G_OBJECT (priv->actor), actor_destroyed, adj); | ||||
|       break; | ||||
|  | ||||
|     case PROP_LOWER: | ||||
|       st_adjustment_set_lower (adj, g_value_get_double (value)); | ||||
|       break; | ||||
| @@ -217,6 +259,11 @@ st_adjustment_dispose (GObject *object) | ||||
|   StAdjustmentPrivate *priv; | ||||
|  | ||||
|   priv = st_adjustment_get_instance_private (ST_ADJUSTMENT (object)); | ||||
|   if (priv->actor) | ||||
|     { | ||||
|       g_object_weak_unref (G_OBJECT (priv->actor), actor_destroyed, object); | ||||
|       priv->actor = NULL; | ||||
|     } | ||||
|   g_clear_pointer (&priv->transitions, g_hash_table_unref); | ||||
|  | ||||
|   G_OBJECT_CLASS (st_adjustment_parent_class)->dispose (object); | ||||
| @@ -232,6 +279,11 @@ st_adjustment_class_init (StAdjustmentClass *klass) | ||||
|   object_class->set_property = st_adjustment_set_property; | ||||
|   object_class->dispose = st_adjustment_dispose; | ||||
|  | ||||
|   props[PROP_ACTOR] = | ||||
|     g_param_spec_object ("actor", "Actor", "Actor", | ||||
|                          CLUTTER_TYPE_ACTOR, | ||||
|                          ST_PARAM_READWRITE); | ||||
|  | ||||
|   props[PROP_LOWER] = | ||||
|     g_param_spec_double ("lower", "Lower", "Lower bound", | ||||
|                          -G_MAXDOUBLE,  G_MAXDOUBLE, 0.0, | ||||
| @@ -299,7 +351,8 @@ st_adjustment_init (StAdjustment *self) | ||||
| } | ||||
|  | ||||
| StAdjustment * | ||||
| st_adjustment_new (gdouble value, | ||||
| st_adjustment_new (ClutterActor *actor, | ||||
|                    gdouble value, | ||||
|                    gdouble lower, | ||||
|                    gdouble upper, | ||||
|                    gdouble step_increment, | ||||
| @@ -307,6 +360,7 @@ st_adjustment_new (gdouble value, | ||||
|                    gdouble page_size) | ||||
| { | ||||
|   return g_object_new (ST_TYPE_ADJUSTMENT, | ||||
|                        "actor", actor, | ||||
|                        "value", value, | ||||
|                        "lower", lower, | ||||
|                        "upper", upper, | ||||
|   | ||||
| @@ -48,7 +48,8 @@ struct _StAdjustmentClass | ||||
|   void (* changed) (StAdjustment *adjustment); | ||||
| }; | ||||
|  | ||||
| StAdjustment *st_adjustment_new         (gdouble       value, | ||||
| StAdjustment *st_adjustment_new         (ClutterActor *actor, | ||||
|                                          gdouble       value, | ||||
|                                          gdouble       lower, | ||||
|                                          gdouble       upper, | ||||
|                                          gdouble       step_increment, | ||||
|   | ||||
| @@ -40,7 +40,7 @@ typedef struct _TransferData TransferData; | ||||
| struct _TransferData | ||||
| { | ||||
|   StClipboard            *clipboard; | ||||
|   StClipboardCallbackFunc callback; | ||||
|   GCallback               callback; | ||||
|   gpointer                user_data; | ||||
|   GOutputStream          *stream; | ||||
| }; | ||||
| @@ -140,12 +140,29 @@ transfer_cb (MetaSelection *selection, | ||||
|       memcpy (text, g_memory_output_stream_get_data (G_MEMORY_OUTPUT_STREAM (data->stream)), data_size); | ||||
|     } | ||||
|  | ||||
|   data->callback (data->clipboard, text, data->user_data); | ||||
|   ((StClipboardCallbackFunc) data->callback) (data->clipboard, text, | ||||
|                                               data->user_data); | ||||
|   g_object_unref (data->stream); | ||||
|   g_free (data); | ||||
|   g_free (text); | ||||
| } | ||||
|  | ||||
| static void | ||||
| transfer_bytes_cb (MetaSelection *selection, | ||||
|                    GAsyncResult  *res, | ||||
|                    TransferData  *data) | ||||
| { | ||||
|   GBytes *bytes = NULL; | ||||
|  | ||||
|   if (meta_selection_transfer_finish (selection, res, NULL)) | ||||
|     bytes = g_memory_output_stream_steal_as_bytes (G_MEMORY_OUTPUT_STREAM (data->stream)); | ||||
|  | ||||
|   ((StClipboardContentCallbackFunc) data->callback) (data->clipboard, bytes, | ||||
|                                                      data->user_data); | ||||
|   g_object_unref (data->stream); | ||||
|   g_clear_pointer (&bytes, g_bytes_unref); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_clipboard_get_mimetypes: | ||||
|  * @clipboard: a #StClipboard | ||||
| @@ -205,7 +222,7 @@ st_clipboard_get_text (StClipboard            *clipboard, | ||||
|  | ||||
|   data = g_new0 (TransferData, 1); | ||||
|   data->clipboard = clipboard; | ||||
|   data->callback = callback; | ||||
|   data->callback = G_CALLBACK (callback); | ||||
|   data->user_data = user_data; | ||||
|   data->stream = g_memory_output_stream_new_resizable (); | ||||
|  | ||||
| @@ -217,6 +234,52 @@ st_clipboard_get_text (StClipboard            *clipboard, | ||||
|                                  data); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_clipboard_get_content: | ||||
|  * @clipboard: A #StCliboard | ||||
|  * @type: The type of clipboard data you want | ||||
|  * @mimetype: The mimetype to get content for | ||||
|  * @callback: (scope async): function to be called when the type is retrieved | ||||
|  * @user_data: data to be passed to the callback | ||||
|  * | ||||
|  * Request the data from the clipboard in #GBytes form. @callback is executed | ||||
|  * when the data is retrieved. | ||||
|  * | ||||
|  */ | ||||
| void | ||||
| st_clipboard_get_content (StClipboard                    *clipboard, | ||||
|                           StClipboardType                 type, | ||||
|                           const gchar                    *mimetype, | ||||
|                           StClipboardContentCallbackFunc  callback, | ||||
|                           gpointer                        user_data) | ||||
| { | ||||
|   MetaSelectionType selection_type; | ||||
|   TransferData *data; | ||||
|  | ||||
|   g_return_if_fail (ST_IS_CLIPBOARD (clipboard)); | ||||
|   g_return_if_fail (meta_selection != NULL); | ||||
|   g_return_if_fail (callback != NULL); | ||||
|  | ||||
|   if (!mimetype || !convert_type (type, &selection_type)) | ||||
|     { | ||||
|       callback (clipboard, NULL, user_data); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|   data = g_new0 (TransferData, 1); | ||||
|   data->clipboard = clipboard; | ||||
|   data->callback = G_CALLBACK (callback); | ||||
|   data->user_data = user_data; | ||||
|   data->stream = g_memory_output_stream_new_resizable (); | ||||
|  | ||||
|   meta_selection_transfer_async (meta_selection, | ||||
|                                  selection_type, | ||||
|                                  mimetype, -1, | ||||
|                                  data->stream, NULL, | ||||
|                                  (GAsyncReadyCallback) transfer_bytes_cb, | ||||
|                                  data); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_clipboard_set_content: | ||||
|  * @clipboard: A #StClipboard | ||||
|   | ||||
| @@ -63,6 +63,18 @@ typedef void (*StClipboardCallbackFunc) (StClipboard *clipboard, | ||||
|                                          const gchar *text, | ||||
|                                          gpointer     user_data); | ||||
|  | ||||
| /** | ||||
|  * StClipboardContentCallbackFunc: | ||||
|  * @clipboard: A #StClipboard | ||||
|  * @bytes: content from the clipboard | ||||
|  * @user_data: user data | ||||
|  * | ||||
|  * Callback function called when content is retrieved from the clipboard. | ||||
|  */ | ||||
| typedef void (*StClipboardContentCallbackFunc) (StClipboard *clipboard, | ||||
|                                                 GBytes      *bytes, | ||||
|                                                 gpointer     user_data); | ||||
|  | ||||
| StClipboard* st_clipboard_get_default (void); | ||||
|  | ||||
| GList * st_clipboard_get_mimetypes (StClipboard             *clipboard, | ||||
| @@ -80,6 +92,11 @@ void st_clipboard_set_content (StClipboard          *clipboard, | ||||
|                                StClipboardType       type, | ||||
|                                const gchar          *mimetype, | ||||
|                                GBytes               *bytes); | ||||
| void st_clipboard_get_content (StClipboard                    *clipboard, | ||||
|                                StClipboardType                 type, | ||||
|                                const gchar                    *mimetype, | ||||
|                                StClipboardContentCallbackFunc  callback, | ||||
|                                gpointer                        user_data); | ||||
|  | ||||
| void st_clipboard_set_selection (MetaSelection *selection); | ||||
|  | ||||
|   | ||||
| @@ -85,12 +85,7 @@ st_drawing_area_allocate (ClutterActor          *self, | ||||
|   int width, height; | ||||
|   float resource_scale; | ||||
|  | ||||
|   if (!st_widget_get_resource_scale (ST_WIDGET (self), &resource_scale)) | ||||
|     { | ||||
|       ClutterActorBox empty = CLUTTER_ACTOR_BOX_INIT_ZERO; | ||||
|       clutter_actor_set_allocation (self, &empty); | ||||
|       return; | ||||
|     } | ||||
|   resource_scale = clutter_actor_get_resource_scale (self); | ||||
|  | ||||
|   clutter_actor_set_allocation (self, box); | ||||
|   st_theme_node_get_content_box (theme_node, box, &content_box); | ||||
| @@ -111,13 +106,16 @@ st_drawing_area_style_changed (StWidget  *self) | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_drawing_area_resource_scale_changed (StWidget *self) | ||||
| st_drawing_area_resource_scale_changed (ClutterActor *self) | ||||
| { | ||||
|   float resource_scale; | ||||
|   ClutterContent *content = clutter_actor_get_content (CLUTTER_ACTOR (self)); | ||||
|   ClutterContent *content = clutter_actor_get_content (self); | ||||
|  | ||||
|   if (st_widget_get_resource_scale (ST_WIDGET (self), &resource_scale)) | ||||
|     clutter_canvas_set_scale_factor (CLUTTER_CANVAS (content), resource_scale); | ||||
|   resource_scale = clutter_actor_get_resource_scale (self); | ||||
|   clutter_canvas_set_scale_factor (CLUTTER_CANVAS (content), resource_scale); | ||||
|  | ||||
|   if (CLUTTER_ACTOR_CLASS (st_drawing_area_parent_class)->resource_scale_changed) | ||||
|     CLUTTER_ACTOR_CLASS (st_drawing_area_parent_class)->resource_scale_changed (self); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -128,7 +126,7 @@ st_drawing_area_class_init (StDrawingAreaClass *klass) | ||||
|  | ||||
|   actor_class->allocate = st_drawing_area_allocate; | ||||
|   widget_class->style_changed = st_drawing_area_style_changed; | ||||
|   widget_class->resource_scale_changed = st_drawing_area_resource_scale_changed; | ||||
|   actor_class->resource_scale_changed = st_drawing_area_resource_scale_changed; | ||||
|  | ||||
|   st_drawing_area_signals[REPAINT] = | ||||
|     g_signal_new ("repaint", | ||||
| @@ -215,15 +213,10 @@ st_drawing_area_get_surface_size (StDrawingArea *area, | ||||
|   content = clutter_actor_get_content (CLUTTER_ACTOR (area)); | ||||
|   clutter_content_get_preferred_size (content, &w, &h); | ||||
|  | ||||
|   if (st_widget_get_resource_scale (ST_WIDGET (area), &resource_scale)) | ||||
|     { | ||||
|       w /= resource_scale; | ||||
|       h /= resource_scale; | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       w = h = 0.0f; | ||||
|     } | ||||
|   resource_scale = clutter_actor_get_resource_scale (CLUTTER_ACTOR (area)); | ||||
|  | ||||
|   w /= resource_scale; | ||||
|   h /= resource_scale; | ||||
|  | ||||
|   if (width) | ||||
|     *width = ceilf (w); | ||||
|   | ||||
| @@ -231,9 +231,12 @@ st_icon_style_changed (StWidget *widget) | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_icon_resource_scale_changed (StWidget *widget) | ||||
| st_icon_resource_scale_changed (ClutterActor *actor) | ||||
| { | ||||
|   st_icon_update (ST_ICON (widget)); | ||||
|   st_icon_update (ST_ICON (actor)); | ||||
|  | ||||
|   if (CLUTTER_ACTOR_CLASS (st_icon_parent_class)->resource_scale_changed) | ||||
|     CLUTTER_ACTOR_CLASS (st_icon_parent_class)->resource_scale_changed (actor); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -250,7 +253,7 @@ st_icon_class_init (StIconClass *klass) | ||||
|   actor_class->paint = st_icon_paint; | ||||
|  | ||||
|   widget_class->style_changed = st_icon_style_changed; | ||||
|   widget_class->resource_scale_changed = st_icon_resource_scale_changed; | ||||
|   actor_class->resource_scale_changed = st_icon_resource_scale_changed; | ||||
|  | ||||
|   props[PROP_GICON] = | ||||
|     g_param_spec_object ("gicon", | ||||
| @@ -425,8 +428,7 @@ st_icon_update (StIcon *icon) | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|   if (!st_widget_get_resource_scale (ST_WIDGET (icon), &resource_scale)) | ||||
|     return; | ||||
|   resource_scale = clutter_actor_get_resource_scale (CLUTTER_ACTOR (icon)); | ||||
|  | ||||
|   theme_node = st_widget_peek_theme_node (ST_WIDGET (icon)); | ||||
|   if (theme_node == NULL) | ||||
|   | ||||
| @@ -42,11 +42,17 @@ enum | ||||
| }; | ||||
|  | ||||
| static void clutter_content_interface_init (ClutterContentInterface *iface); | ||||
| static void g_icon_interface_init (GIconIface *iface); | ||||
| static void g_loadable_icon_interface_init (GLoadableIconIface *iface); | ||||
|  | ||||
| G_DEFINE_TYPE_WITH_CODE (StImageContent, st_image_content, CLUTTER_TYPE_IMAGE, | ||||
|                          G_ADD_PRIVATE (StImageContent) | ||||
|                          G_IMPLEMENT_INTERFACE (CLUTTER_TYPE_CONTENT, | ||||
|                                                 clutter_content_interface_init)) | ||||
|                                                 clutter_content_interface_init) | ||||
|                          G_IMPLEMENT_INTERFACE (G_TYPE_ICON, | ||||
|                                                 g_icon_interface_init) | ||||
|                          G_IMPLEMENT_INTERFACE (G_TYPE_LOADABLE_ICON, | ||||
|                                                 g_loadable_icon_interface_init)) | ||||
|  | ||||
| static void | ||||
| st_image_content_init (StImageContent *self) | ||||
| @@ -165,12 +171,157 @@ st_image_content_get_preferred_size (ClutterContent *content, | ||||
|   return TRUE; | ||||
| } | ||||
|  | ||||
| static GdkPixbuf* | ||||
| pixbuf_from_image (StImageContent *image) | ||||
| { | ||||
|   CoglTexture *texture; | ||||
|   int width, height, rowstride; | ||||
|   uint8_t *data; | ||||
|  | ||||
|   texture = clutter_image_get_texture (CLUTTER_IMAGE (image)); | ||||
|   if (!texture || !cogl_texture_is_get_data_supported (texture)) | ||||
|     return NULL; | ||||
|  | ||||
|   width = cogl_texture_get_width (texture); | ||||
|   height = cogl_texture_get_width (texture); | ||||
|   rowstride = 4 * width; | ||||
|   data = g_new (uint8_t, rowstride * height); | ||||
|  | ||||
|   cogl_texture_get_data (texture, COGL_PIXEL_FORMAT_RGBA_8888, rowstride, data); | ||||
|  | ||||
|   return gdk_pixbuf_new_from_data ((const guchar *)data, | ||||
|                                    GDK_COLORSPACE_RGB, | ||||
|                                    TRUE, 8, width, height, rowstride, | ||||
|                                    (GdkPixbufDestroyNotify)g_free, NULL); | ||||
| } | ||||
|  | ||||
| static void | ||||
| clutter_content_interface_init (ClutterContentInterface *iface) | ||||
| { | ||||
|   iface->get_preferred_size = st_image_content_get_preferred_size; | ||||
| } | ||||
|  | ||||
| static guint | ||||
| st_image_content_hash (GIcon *icon) | ||||
| { | ||||
|   return g_direct_hash (icon); | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| st_image_content_equal (GIcon *icon1, | ||||
|                         GIcon *icon2) | ||||
| { | ||||
|   return g_direct_equal (icon1, icon2); | ||||
| } | ||||
|  | ||||
| static GVariant * | ||||
| st_image_content_serialize (GIcon *icon) | ||||
| { | ||||
|   g_autoptr (GdkPixbuf) pixbuf = NULL; | ||||
|  | ||||
|   pixbuf = pixbuf_from_image (ST_IMAGE_CONTENT (icon)); | ||||
|   if (!pixbuf) | ||||
|     return NULL; | ||||
|  | ||||
|   return g_icon_serialize (G_ICON (pixbuf)); | ||||
| } | ||||
|  | ||||
| static void | ||||
| g_icon_interface_init (GIconIface *iface) | ||||
| { | ||||
|   iface->hash = st_image_content_hash; | ||||
|   iface->equal = st_image_content_equal; | ||||
|   iface->serialize = st_image_content_serialize; | ||||
| } | ||||
|  | ||||
| static GInputStream * | ||||
| st_image_load (GLoadableIcon  *icon, | ||||
|                int             size, | ||||
|                char          **type, | ||||
|                GCancellable   *cancellable, | ||||
|                GError       **error) | ||||
| { | ||||
|   g_autoptr (GdkPixbuf) pixbuf = NULL; | ||||
|  | ||||
|   pixbuf = pixbuf_from_image (ST_IMAGE_CONTENT (icon)); | ||||
|   if (!pixbuf) | ||||
|     { | ||||
|       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED, | ||||
|                            "Failed to read texture"); | ||||
|       return NULL; | ||||
|     } | ||||
|  | ||||
|   return g_loadable_icon_load (G_LOADABLE_ICON (pixbuf), | ||||
|                                size, type, cancellable, error); | ||||
| } | ||||
|  | ||||
| static void | ||||
| load_image_thread (GTask        *task, | ||||
|                    gpointer      object, | ||||
|                    gpointer      task_data, | ||||
|                    GCancellable *cancellable) | ||||
| { | ||||
|   GInputStream *stream; | ||||
|   GError *error = NULL; | ||||
|   char *type; | ||||
|  | ||||
|   stream = st_image_load (G_LOADABLE_ICON (object), | ||||
|                           GPOINTER_TO_INT (task_data), | ||||
|                           &type, | ||||
|                           cancellable, | ||||
|                           &error); | ||||
|  | ||||
|   if (error) | ||||
|     { | ||||
|       g_task_return_error (task, error); | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       g_task_set_task_data (task, type, g_free); | ||||
|       g_task_return_pointer (task, stream, g_object_unref); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_image_load_async (GLoadableIcon       *icon, | ||||
|                      int                  size, | ||||
|                      GCancellable        *cancellable, | ||||
|                      GAsyncReadyCallback  callback, | ||||
|                      gpointer             user_data) | ||||
| { | ||||
|   g_autoptr (GTask) task = NULL; | ||||
|  | ||||
|   task = g_task_new (icon, cancellable, callback, user_data); | ||||
|   g_task_set_task_data (task, GINT_TO_POINTER (size), NULL); | ||||
|   g_task_run_in_thread (task, load_image_thread); | ||||
| } | ||||
|  | ||||
| static GInputStream * | ||||
| st_image_load_finish (GLoadableIcon  *icon, | ||||
|                       GAsyncResult   *res, | ||||
|                       char          **type, | ||||
|                       GError        **error) | ||||
| { | ||||
|   GInputStream *stream; | ||||
|  | ||||
|   stream = g_task_propagate_pointer (G_TASK (res), error); | ||||
|   if (!stream) | ||||
|     return NULL; | ||||
|  | ||||
|   if (type) | ||||
|     *type = g_strdup (g_task_get_task_data (G_TASK (res))); | ||||
|  | ||||
|   return stream; | ||||
| } | ||||
|  | ||||
| static void | ||||
| g_loadable_icon_interface_init (GLoadableIconIface *iface) | ||||
| { | ||||
|   iface->load = st_image_load; | ||||
|   iface->load_async = st_image_load_async; | ||||
|   iface->load_finish = st_image_load_finish; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_image_content_new_with_preferred_size: | ||||
|  * @width: The preferred width to be used when drawing the content | ||||
|   | ||||
| @@ -201,44 +201,42 @@ st_label_paint (ClutterActor        *actor, | ||||
|  | ||||
|   if (shadow_spec) | ||||
|     { | ||||
|       ClutterActorBox allocation; | ||||
|       float width, height; | ||||
|       float resource_scale; | ||||
|  | ||||
|       if (clutter_actor_get_resource_scale (priv->label, &resource_scale)) | ||||
|       clutter_actor_get_allocation_box (priv->label, &allocation); | ||||
|       clutter_actor_box_get_size (&allocation, &width, &height); | ||||
|  | ||||
|       resource_scale = clutter_actor_get_resource_scale (priv->label); | ||||
|  | ||||
|       width *= resource_scale; | ||||
|       height *= resource_scale; | ||||
|  | ||||
|       if (priv->text_shadow_pipeline == NULL || | ||||
|           width != priv->shadow_width || | ||||
|           height != priv->shadow_height) | ||||
|         { | ||||
|           ClutterActorBox allocation; | ||||
|           float width, height; | ||||
|           g_clear_pointer (&priv->text_shadow_pipeline, cogl_object_unref); | ||||
|  | ||||
|           clutter_actor_get_allocation_box (priv->label, &allocation); | ||||
|           clutter_actor_box_get_size (&allocation, &width, &height); | ||||
|           priv->shadow_width = width; | ||||
|           priv->shadow_height = height; | ||||
|           priv->text_shadow_pipeline = | ||||
|             _st_create_shadow_pipeline_from_actor (shadow_spec, | ||||
|                                                    priv->label); | ||||
|         } | ||||
|  | ||||
|           width *= resource_scale; | ||||
|           height *= resource_scale; | ||||
|       if (priv->text_shadow_pipeline != NULL) | ||||
|         { | ||||
|           CoglFramebuffer *framebuffer; | ||||
|  | ||||
|           if (priv->text_shadow_pipeline == NULL || | ||||
|               width != priv->shadow_width || | ||||
|               height != priv->shadow_height) | ||||
|             { | ||||
|               g_clear_pointer (&priv->text_shadow_pipeline, cogl_object_unref); | ||||
|  | ||||
|               priv->shadow_width = width; | ||||
|               priv->shadow_height = height; | ||||
|               priv->text_shadow_pipeline = | ||||
|                 _st_create_shadow_pipeline_from_actor (shadow_spec, | ||||
|                                                        priv->label); | ||||
|             } | ||||
|  | ||||
|           if (priv->text_shadow_pipeline != NULL) | ||||
|             { | ||||
|               CoglFramebuffer *framebuffer; | ||||
|  | ||||
|               framebuffer = | ||||
|                 clutter_paint_context_get_framebuffer (paint_context); | ||||
|               _st_paint_shadow_with_opacity (shadow_spec, | ||||
|                                              framebuffer, | ||||
|                                              priv->text_shadow_pipeline, | ||||
|                                              &allocation, | ||||
|                                              clutter_actor_get_paint_opacity (priv->label)); | ||||
|             } | ||||
|           framebuffer = | ||||
|             clutter_paint_context_get_framebuffer (paint_context); | ||||
|           _st_paint_shadow_with_opacity (shadow_spec, | ||||
|                                          framebuffer, | ||||
|                                          priv->text_shadow_pipeline, | ||||
|                                          &allocation, | ||||
|                                          clutter_actor_get_paint_opacity (priv->label)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -246,11 +244,14 @@ st_label_paint (ClutterActor        *actor, | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_label_resource_scale_changed (StWidget *widget) | ||||
| st_label_resource_scale_changed (ClutterActor *actor) | ||||
| { | ||||
|   StLabelPrivate *priv = ST_LABEL (widget)->priv; | ||||
|   StLabelPrivate *priv = ST_LABEL (actor)->priv; | ||||
|  | ||||
|   g_clear_pointer (&priv->text_shadow_pipeline, cogl_object_unref); | ||||
|  | ||||
|   if (CLUTTER_ACTOR_CLASS (st_label_parent_class)->resource_scale_changed) | ||||
|     CLUTTER_ACTOR_CLASS (st_label_parent_class)->resource_scale_changed (actor); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -268,9 +269,9 @@ st_label_class_init (StLabelClass *klass) | ||||
|   actor_class->allocate = st_label_allocate; | ||||
|   actor_class->get_preferred_width = st_label_get_preferred_width; | ||||
|   actor_class->get_preferred_height = st_label_get_preferred_height; | ||||
|   actor_class->resource_scale_changed = st_label_resource_scale_changed; | ||||
|  | ||||
|   widget_class->style_changed = st_label_style_changed; | ||||
|   widget_class->resource_scale_changed = st_label_resource_scale_changed; | ||||
|   widget_class->get_accessible_type = st_label_accessible_get_type; | ||||
|  | ||||
|   props[PROP_CLUTTER_TEXT] = | ||||
| @@ -293,6 +294,7 @@ st_label_class_init (StLabelClass *klass) | ||||
| static void | ||||
| st_label_init (StLabel *label) | ||||
| { | ||||
|   ClutterActor *actor = CLUTTER_ACTOR (label); | ||||
|   StLabelPrivate *priv; | ||||
|  | ||||
|   label->priv = priv = st_label_get_instance_private (label); | ||||
| @@ -304,7 +306,10 @@ st_label_init (StLabel *label) | ||||
|   label->priv->shadow_width = -1.; | ||||
|   label->priv->shadow_height = -1.; | ||||
|  | ||||
|   clutter_actor_add_child (CLUTTER_ACTOR (label), priv->label); | ||||
|   clutter_actor_add_child (actor, priv->label); | ||||
|  | ||||
|   clutter_actor_set_offscreen_redirect (actor, | ||||
|                                         CLUTTER_OFFSCREEN_REDIRECT_ALWAYS); | ||||
| } | ||||
|  | ||||
| /** | ||||
|   | ||||
| @@ -461,8 +461,7 @@ _st_create_shadow_pipeline_from_actor (StShadow     *shadow_spec, | ||||
|   if (width == 0 || height == 0) | ||||
|     return NULL; | ||||
|  | ||||
|   if (!clutter_actor_get_resource_scale (actor, &resource_scale)) | ||||
|     return NULL; | ||||
|   resource_scale = clutter_actor_get_resource_scale (actor); | ||||
|  | ||||
|   width = ceilf (width * resource_scale); | ||||
|   height = ceilf (height * resource_scale); | ||||
|   | ||||
| @@ -262,14 +262,25 @@ scroll_bar_allocate_children (StScrollBar           *bar, | ||||
|         } | ||||
|       else | ||||
|         { | ||||
|           ClutterTextDirection direction; | ||||
|  | ||||
|           avail_size = content_box.x2 - content_box.x1; | ||||
|           handle_size = increment * avail_size; | ||||
|           handle_size = CLAMP (handle_size, min_size, max_size); | ||||
|  | ||||
|           handle_box.x1 = content_box.x1 + position * (avail_size - handle_size); | ||||
|           handle_box.y1 = content_box.y1; | ||||
|           direction = clutter_actor_get_text_direction (CLUTTER_ACTOR (bar)); | ||||
|           if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|             { | ||||
|               handle_box.x2 = content_box.x2 - position * (avail_size - handle_size); | ||||
|               handle_box.x1 = handle_box.x2 - handle_size; | ||||
|             } | ||||
|           else | ||||
|             { | ||||
|               handle_box.x1 = content_box.x1 + position * (avail_size - handle_size); | ||||
|               handle_box.x2 = handle_box.x1 + handle_size; | ||||
|             } | ||||
|  | ||||
|           handle_box.x2 = handle_box.x1 + handle_size; | ||||
|           handle_box.y1 = content_box.y1; | ||||
|           handle_box.y2 = content_box.y2; | ||||
|         } | ||||
|  | ||||
| @@ -464,28 +475,40 @@ st_scroll_bar_scroll_event (ClutterActor       *actor, | ||||
|                             ClutterScrollEvent *event) | ||||
| { | ||||
|   StScrollBarPrivate *priv = ST_SCROLL_BAR_PRIVATE (actor); | ||||
|   ClutterTextDirection direction; | ||||
|   ClutterScrollDirection scroll_dir; | ||||
|  | ||||
|   if (clutter_event_is_pointer_emulated ((ClutterEvent *) event)) | ||||
|     return TRUE; | ||||
|  | ||||
|   switch (event->direction) | ||||
|   direction = clutter_actor_get_text_direction (actor); | ||||
|   scroll_dir = event->direction; | ||||
|  | ||||
|   switch (scroll_dir) | ||||
|     { | ||||
|     case CLUTTER_SCROLL_SMOOTH: | ||||
|       { | ||||
|         gdouble delta_x, delta_y; | ||||
|         clutter_event_get_scroll_delta ((ClutterEvent *)event, &delta_x, &delta_y); | ||||
|  | ||||
|         if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|           delta_x *= -1; | ||||
|  | ||||
|         if (priv->vertical) | ||||
|           st_adjustment_adjust_for_scroll_event (priv->adjustment, delta_y); | ||||
|         else | ||||
|           st_adjustment_adjust_for_scroll_event (priv->adjustment, delta_x); | ||||
|       } | ||||
|       break; | ||||
|     case CLUTTER_SCROLL_UP: | ||||
|     case CLUTTER_SCROLL_DOWN: | ||||
|     case CLUTTER_SCROLL_LEFT: | ||||
|     case CLUTTER_SCROLL_RIGHT: | ||||
|       adjust_with_direction (priv->adjustment, event->direction); | ||||
|       if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|           scroll_dir = scroll_dir == CLUTTER_SCROLL_LEFT ? CLUTTER_SCROLL_RIGHT | ||||
|                                                          : CLUTTER_SCROLL_LEFT; | ||||
|     /* Fall through */ | ||||
|     case CLUTTER_SCROLL_UP: | ||||
|     case CLUTTER_SCROLL_DOWN: | ||||
|       adjust_with_direction (priv->adjustment, scroll_dir); | ||||
|       break; | ||||
|     default: | ||||
|       g_return_val_if_reached (FALSE); | ||||
| @@ -552,6 +575,7 @@ move_slider (StScrollBar *bar, | ||||
|              gfloat       y) | ||||
| { | ||||
|   StScrollBarPrivate *priv = st_scroll_bar_get_instance_private (bar); | ||||
|   ClutterTextDirection direction; | ||||
|   gdouble position, lower, upper, page_size; | ||||
|   gfloat ux, uy, pos, size; | ||||
|  | ||||
| @@ -585,6 +609,10 @@ move_slider (StScrollBar *bar, | ||||
|                             NULL, | ||||
|                             &page_size); | ||||
|  | ||||
|   direction = clutter_actor_get_text_direction (CLUTTER_ACTOR (bar)); | ||||
|   if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|     pos = size - pos; | ||||
|  | ||||
|   position = ((pos / size) | ||||
|               * (upper - lower - page_size)) | ||||
|              + lower; | ||||
| @@ -668,6 +696,7 @@ static gboolean | ||||
| trough_paging_cb (StScrollBar *self) | ||||
| { | ||||
|   StScrollBarPrivate *priv = st_scroll_bar_get_instance_private (self); | ||||
|   ClutterTextDirection direction; | ||||
|   g_autoptr (ClutterTransition) transition = NULL; | ||||
|   StSettings *settings; | ||||
|   gfloat handle_pos, event_pos, tx, ty; | ||||
| @@ -725,6 +754,10 @@ trough_paging_cb (StScrollBar *self) | ||||
|                                        priv->move_y, | ||||
|                                        &tx, &ty); | ||||
|  | ||||
|   direction = clutter_actor_get_text_direction (CLUTTER_ACTOR (self)); | ||||
|   if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|     page_increment *= -1; | ||||
|  | ||||
|   if (priv->vertical) | ||||
|     event_pos = ty; | ||||
|   else | ||||
|   | ||||
| @@ -757,6 +757,7 @@ st_scroll_view_scroll_event (ClutterActor       *self, | ||||
|                              ClutterScrollEvent *event) | ||||
| { | ||||
|   StScrollViewPrivate *priv = ST_SCROLL_VIEW (self)->priv; | ||||
|   ClutterTextDirection direction; | ||||
|  | ||||
|   /* don't handle scroll events if requested not to */ | ||||
|   if (!priv->mouse_scroll) | ||||
| @@ -765,12 +766,18 @@ st_scroll_view_scroll_event (ClutterActor       *self, | ||||
|   if (clutter_event_is_pointer_emulated ((ClutterEvent *) event)) | ||||
|     return TRUE; | ||||
|  | ||||
|   direction = clutter_actor_get_text_direction (self); | ||||
|  | ||||
|   switch (event->direction) | ||||
|     { | ||||
|     case CLUTTER_SCROLL_SMOOTH: | ||||
|       { | ||||
|         gdouble delta_x, delta_y; | ||||
|         clutter_event_get_scroll_delta ((ClutterEvent *)event, &delta_x, &delta_y); | ||||
|  | ||||
|         if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|           delta_x *= -1; | ||||
|  | ||||
|         st_adjustment_adjust_for_scroll_event (priv->hadjustment, delta_x); | ||||
|         st_adjustment_adjust_for_scroll_event (priv->vadjustment, delta_y); | ||||
|       } | ||||
| @@ -781,7 +788,18 @@ st_scroll_view_scroll_event (ClutterActor       *self, | ||||
|       break; | ||||
|     case CLUTTER_SCROLL_LEFT: | ||||
|     case CLUTTER_SCROLL_RIGHT: | ||||
|       adjust_with_direction (priv->hadjustment, event->direction); | ||||
|       if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|         { | ||||
|           ClutterScrollDirection dir; | ||||
|  | ||||
|           dir = event->direction == CLUTTER_SCROLL_LEFT ? CLUTTER_SCROLL_RIGHT | ||||
|                                                         : CLUTTER_SCROLL_LEFT; | ||||
|           adjust_with_direction (priv->hadjustment, dir); | ||||
|         } | ||||
|       else | ||||
|         { | ||||
|           adjust_with_direction (priv->hadjustment, event->direction); | ||||
|         } | ||||
|       break; | ||||
|     default: | ||||
|       g_warn_if_reached(); | ||||
| @@ -881,13 +899,17 @@ st_scroll_view_init (StScrollView *self) | ||||
|   priv->hscrollbar_policy = ST_POLICY_AUTOMATIC; | ||||
|   priv->vscrollbar_policy = ST_POLICY_AUTOMATIC; | ||||
|  | ||||
|   priv->hadjustment = g_object_new (ST_TYPE_ADJUSTMENT, NULL); | ||||
|   priv->hadjustment = g_object_new (ST_TYPE_ADJUSTMENT, | ||||
|                                     "actor", self, | ||||
|                                     NULL); | ||||
|   priv->hscroll = g_object_new (ST_TYPE_SCROLL_BAR, | ||||
|                                 "adjustment", priv->hadjustment, | ||||
|                                 "vertical", FALSE, | ||||
|                                 NULL); | ||||
|  | ||||
|   priv->vadjustment = g_object_new (ST_TYPE_ADJUSTMENT, NULL); | ||||
|   priv->vadjustment = g_object_new (ST_TYPE_ADJUSTMENT, | ||||
|                                     "actor", self, | ||||
|                                     NULL); | ||||
|   priv->vscroll = g_object_new (ST_TYPE_SCROLL_BAR, | ||||
|                                 "adjustment", priv->vadjustment, | ||||
|                                 "vertical", TRUE, | ||||
|   | ||||
| @@ -722,8 +722,7 @@ load_texture_async (StTextureCache       *cache, | ||||
|  | ||||
| typedef struct { | ||||
|   StTextureCache *cache; | ||||
|   ClutterActor *actor; | ||||
|   gint size; | ||||
|   ClutterContent *image; | ||||
|   GObject *source; | ||||
|   gulong notify_signal_id; | ||||
|   gboolean weakref_active; | ||||
| @@ -742,37 +741,33 @@ st_texture_cache_reset_texture (StTextureCachePropertyBind *bind, | ||||
|       (cairo_image_surface_get_format (surface) == CAIRO_FORMAT_ARGB32 || | ||||
|        cairo_image_surface_get_format (surface) == CAIRO_FORMAT_RGB24)) | ||||
|     { | ||||
|       g_autoptr(ClutterContent) image = NULL; | ||||
|       g_autoptr(GError) error = NULL; | ||||
|       int size = bind->size; | ||||
|       int width, height, size; | ||||
|  | ||||
|       if (size < 0) | ||||
|         clutter_actor_get_preferred_width (bind->actor, -1, NULL, (float *)&size); | ||||
|       width = cairo_image_surface_get_width (surface); | ||||
|       height = cairo_image_surface_get_width (surface); | ||||
|       size = MAX(width, height); | ||||
|  | ||||
|       image = clutter_actor_get_content (bind->actor); | ||||
|       if (!image || !CLUTTER_IS_IMAGE (image)) | ||||
|         image = st_image_content_new_with_preferred_size (size, size); | ||||
|       else | ||||
|         g_object_ref (image); | ||||
|       if (!bind->image) | ||||
|         bind->image = st_image_content_new_with_preferred_size (size, size); | ||||
|  | ||||
|       clutter_image_set_data (CLUTTER_IMAGE (image), | ||||
|       clutter_image_set_data (CLUTTER_IMAGE (bind->image), | ||||
|                               cairo_image_surface_get_data (surface), | ||||
|                               cairo_image_surface_get_format (surface) == CAIRO_FORMAT_ARGB32 ? | ||||
|                               COGL_PIXEL_FORMAT_BGRA_8888 : COGL_PIXEL_FORMAT_BGR_888, | ||||
|                               cairo_image_surface_get_width (surface), | ||||
|                               cairo_image_surface_get_height (surface), | ||||
|                               width, | ||||
|                               height, | ||||
|                               cairo_image_surface_get_stride (surface), | ||||
|                               &error); | ||||
|  | ||||
|       if (image) | ||||
|         clutter_actor_set_content (bind->actor, image); | ||||
|       else if (error) | ||||
|       if (error) | ||||
|         g_warning ("Failed to allocate texture: %s", error->message); | ||||
|  | ||||
|       clutter_actor_set_opacity (bind->actor, 255); | ||||
|     } | ||||
|   else | ||||
|     clutter_actor_set_opacity (bind->actor, 0); | ||||
|     bind->image = g_object_new (ST_TYPE_IMAGE_CONTENT, | ||||
|                                 "preferred-width", 0, /* tough luck */ | ||||
|                                 "preferred-height", 0, | ||||
|                                 NULL); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -798,57 +793,48 @@ st_texture_cache_free_bind (gpointer data) | ||||
| { | ||||
|   StTextureCachePropertyBind *bind = data; | ||||
|   if (bind->weakref_active) | ||||
|     g_object_weak_unref (G_OBJECT (bind->actor), st_texture_cache_bind_weak_notify, bind); | ||||
|     g_object_weak_unref (G_OBJECT (bind->image), st_texture_cache_bind_weak_notify, bind); | ||||
|   g_slice_free (StTextureCachePropertyBind, bind); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_texture_cache_bind_cairo_surface_property: | ||||
|  * @cache: | ||||
|  * @object: A #GObject with a property @property_name of type #GdkPixbuf | ||||
|  * @object: A #GObject with a property @property_name of type #cairo_surface_t | ||||
|  * @property_name: Name of a property | ||||
|  * | ||||
|  * Create a #ClutterActor which tracks the #cairo_surface_t value of a GObject property | ||||
|  * Create a #GIcon which tracks the #cairo_surface_t value of a GObject property | ||||
|  * named by @property_name.  Unlike other methods in StTextureCache, the underlying | ||||
|  * #CoglTexture is not shared by default with other invocations to this method. | ||||
|  * | ||||
|  * If the source object is destroyed, the texture will continue to show the last | ||||
|  * value of the property. | ||||
|  * | ||||
|  * Return value: (transfer none): A new #StWidget | ||||
|  * Return value: (transfer none): A new #GIcon | ||||
|  */ | ||||
| StWidget * | ||||
| GIcon * | ||||
| st_texture_cache_bind_cairo_surface_property (StTextureCache    *cache, | ||||
|                                               GObject           *object, | ||||
|                                               const char        *property_name, | ||||
|                                               gint               size) | ||||
|                                               const char        *property_name) | ||||
| { | ||||
|   StWidget *widget; | ||||
|   gchar *notify_key; | ||||
|   StTextureCachePropertyBind *bind; | ||||
|  | ||||
|   widget = g_object_new (ST_TYPE_WIDGET, | ||||
|                          "opacity", 0, | ||||
|                          "width", (float)size, | ||||
|                          "height", (float)size, | ||||
|                          NULL); | ||||
|  | ||||
|   bind = g_slice_new0 (StTextureCachePropertyBind); | ||||
|   bind->cache = cache; | ||||
|   bind->actor = CLUTTER_ACTOR (widget); | ||||
|   bind->size = size; | ||||
|   bind->source = object; | ||||
|   g_object_weak_ref (G_OBJECT (widget), st_texture_cache_bind_weak_notify, bind); | ||||
|   bind->weakref_active = TRUE; | ||||
|  | ||||
|   st_texture_cache_reset_texture (bind, property_name); | ||||
|  | ||||
|   g_object_weak_ref (G_OBJECT (bind->image), st_texture_cache_bind_weak_notify, bind); | ||||
|   bind->weakref_active = TRUE; | ||||
|  | ||||
|   notify_key = g_strdup_printf ("notify::%s", property_name); | ||||
|   bind->notify_signal_id = g_signal_connect_data (object, notify_key, G_CALLBACK(st_texture_cache_on_pixbuf_notify), | ||||
|                                                   bind, (GClosureNotify)st_texture_cache_free_bind, 0); | ||||
|   g_free (notify_key); | ||||
|  | ||||
|   return widget; | ||||
|   return G_ICON (bind->image); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -981,6 +967,18 @@ st_texture_cache_load_gicon (StTextureCache    *cache, | ||||
|   StIconStyle icon_style = ST_ICON_STYLE_REQUESTED; | ||||
|   GtkIconLookupFlags lookup_flags; | ||||
|  | ||||
|   actor_size = size * paint_scale; | ||||
|  | ||||
|   if (ST_IS_IMAGE_CONTENT (icon)) | ||||
|     { | ||||
|       return g_object_new (CLUTTER_TYPE_ACTOR, | ||||
|                            "request-mode", CLUTTER_REQUEST_CONTENT_SIZE, | ||||
|                            "width", actor_size, | ||||
|                            "height", actor_size, | ||||
|                            "content", CLUTTER_CONTENT (icon), | ||||
|                            NULL); | ||||
|     } | ||||
|  | ||||
|   if (theme_node) | ||||
|     { | ||||
|       colors = st_theme_node_get_icon_colors (theme_node); | ||||
| @@ -1034,7 +1032,6 @@ st_texture_cache_load_gicon (StTextureCache    *cache, | ||||
|   g_free (gicon_string); | ||||
|  | ||||
|   actor = create_invisible_actor (); | ||||
|   actor_size = size * paint_scale; | ||||
|   clutter_actor_set_size (actor, actor_size, actor_size); | ||||
|   if (ensure_request (cache, key, policy, &request, actor)) | ||||
|     { | ||||
|   | ||||
| @@ -64,10 +64,9 @@ st_texture_cache_load_sliced_image (StTextureCache *cache, | ||||
|                                     GFunc           load_callback, | ||||
|                                     gpointer        user_data); | ||||
|  | ||||
| StWidget *st_texture_cache_bind_cairo_surface_property (StTextureCache    *cache, | ||||
|                                                         GObject           *object, | ||||
|                                                         const char        *property_name, | ||||
|                                                         gint               size); | ||||
| GIcon *st_texture_cache_bind_cairo_surface_property (StTextureCache    *cache, | ||||
|                                                      GObject           *object, | ||||
|                                                      const char *property_name); | ||||
|  | ||||
| ClutterActor *st_texture_cache_load_gicon (StTextureCache *cache, | ||||
|                                            StThemeNode    *theme_node, | ||||
|   | ||||
| @@ -84,10 +84,11 @@ on_timeline_new_frame (ClutterTimeline       *timeline, | ||||
| } | ||||
|  | ||||
| StThemeNodeTransition * | ||||
| st_theme_node_transition_new (StThemeNode *from_node, | ||||
|                               StThemeNode *to_node, | ||||
| st_theme_node_transition_new (ClutterActor          *actor, | ||||
|                               StThemeNode           *from_node, | ||||
|                               StThemeNode           *to_node, | ||||
|                               StThemeNodePaintState *old_paint_state, | ||||
|                               guint        duration) | ||||
|                               unsigned int           duration) | ||||
| { | ||||
|   StThemeNodeTransition *transition; | ||||
|   g_return_val_if_fail (ST_IS_THEME_NODE (from_node), NULL); | ||||
| @@ -103,7 +104,7 @@ st_theme_node_transition_new (StThemeNode *from_node, | ||||
|   st_theme_node_paint_state_copy (&transition->priv->old_paint_state, | ||||
|                                   old_paint_state); | ||||
|  | ||||
|   transition->priv->timeline = clutter_timeline_new (duration); | ||||
|   transition->priv->timeline = clutter_timeline_new_for_actor (actor, duration); | ||||
|  | ||||
|   transition->priv->timeline_completed_id = | ||||
|     g_signal_connect (transition->priv->timeline, "completed", | ||||
|   | ||||
| @@ -32,10 +32,11 @@ G_BEGIN_DECLS | ||||
| G_DECLARE_FINAL_TYPE (StThemeNodeTransition, st_theme_node_transition, | ||||
|                       ST, THEME_NODE_TRANSITION, GObject) | ||||
|  | ||||
| StThemeNodeTransition *st_theme_node_transition_new (StThemeNode *from_node, | ||||
|                                                      StThemeNode *to_node, | ||||
| StThemeNodeTransition *st_theme_node_transition_new (ClutterActor          *actor, | ||||
|                                                      StThemeNode           *from_node, | ||||
|                                                      StThemeNode           *to_node, | ||||
|                                                      StThemeNodePaintState *old_paint_state, | ||||
|                                                      guint        duration); | ||||
|                                                      guint                  duration); | ||||
|  | ||||
| void  st_theme_node_transition_update   (StThemeNodeTransition *transition, | ||||
|                                          StThemeNode           *new_node); | ||||
|   | ||||
| @@ -78,6 +78,7 @@ adjustment_value_notify_cb (StAdjustment *adjustment, | ||||
|                             GParamSpec   *pspec, | ||||
|                             StViewport   *viewport) | ||||
| { | ||||
|   clutter_actor_invalidate_transform (CLUTTER_ACTOR (viewport)); | ||||
|   clutter_actor_queue_relayout (CLUTTER_ACTOR (viewport)); | ||||
| } | ||||
|  | ||||
| @@ -261,8 +262,10 @@ st_viewport_allocate (ClutterActor           *actor, | ||||
|   clutter_actor_set_allocation (actor, box); | ||||
|  | ||||
|   content_box = viewport_box; | ||||
|   content_box.x2 += MAX (0, min_width - avail_width); | ||||
|   content_box.y2 += MAX (0, min_height - avail_height); | ||||
|   if (priv->hadjustment) | ||||
|     content_box.x2 += MAX (0, min_width - avail_width); | ||||
|   if (priv->vadjustment) | ||||
|     content_box.y2 += MAX (0, min_height - avail_height); | ||||
|  | ||||
|   clutter_layout_manager_allocate (layout, CLUTTER_CONTAINER (actor), | ||||
|                                    &content_box); | ||||
| @@ -301,6 +304,26 @@ st_viewport_allocate (ClutterActor           *actor, | ||||
|     } | ||||
| } | ||||
|  | ||||
| static double | ||||
| get_hadjustment_value (StViewport *viewport) | ||||
| { | ||||
|   StViewportPrivate *priv = st_viewport_get_instance_private (viewport); | ||||
|   ClutterTextDirection direction; | ||||
|   double x, upper, page_size; | ||||
|  | ||||
|   if (!priv->hadjustment) | ||||
|     return 0; | ||||
|  | ||||
|   st_adjustment_get_values (priv->hadjustment, | ||||
|                             &x, NULL, &upper, NULL, NULL, &page_size); | ||||
|  | ||||
|   direction = clutter_actor_get_text_direction (CLUTTER_ACTOR (viewport)); | ||||
|   if (direction == CLUTTER_TEXT_DIRECTION_RTL) | ||||
|     return upper - page_size - x; | ||||
|  | ||||
|   return x; | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_viewport_apply_transform (ClutterActor *actor, | ||||
|                              CoglMatrix   *matrix) | ||||
| @@ -314,7 +337,7 @@ st_viewport_apply_transform (ClutterActor *actor, | ||||
|   parent_class->apply_transform (actor, matrix); | ||||
|  | ||||
|   if (priv->hadjustment) | ||||
|     x = st_adjustment_get_value (priv->hadjustment); | ||||
|     x = get_hadjustment_value (viewport); | ||||
|   else | ||||
|     x = 0; | ||||
|  | ||||
| @@ -336,7 +359,7 @@ get_border_paint_offsets (StViewport *viewport, | ||||
|   StViewportPrivate *priv = st_viewport_get_instance_private (viewport); | ||||
|  | ||||
|   if (priv->hadjustment) | ||||
|     *x = st_adjustment_get_value (priv->hadjustment); | ||||
|     *x = get_hadjustment_value (viewport); | ||||
|   else | ||||
|     *x = 0; | ||||
|  | ||||
|   | ||||
| @@ -124,7 +124,6 @@ enum | ||||
| { | ||||
|   STYLE_CHANGED, | ||||
|   POPUP_MENU, | ||||
|   RESOURCE_SCALE_CHANGED, | ||||
|  | ||||
|   LAST_SIGNAL | ||||
| }; | ||||
| @@ -412,8 +411,7 @@ st_widget_paint_background (StWidget            *widget, | ||||
|   float resource_scale; | ||||
|   guint8 opacity; | ||||
|  | ||||
|   if (!st_widget_get_resource_scale (widget, &resource_scale)) | ||||
|     return; | ||||
|   resource_scale = clutter_actor_get_resource_scale (CLUTTER_ACTOR (widget)); | ||||
|  | ||||
|   framebuffer = clutter_paint_context_get_framebuffer (paint_context); | ||||
|   theme_node = st_widget_get_theme_node (widget); | ||||
| @@ -601,8 +599,11 @@ st_widget_get_theme_node (StWidget *widget) | ||||
|  | ||||
|       if (stage == NULL) | ||||
|         { | ||||
|           g_autofree char *desc = st_describe_actor (CLUTTER_ACTOR (widget)); | ||||
|  | ||||
|           g_critical ("st_widget_get_theme_node called on the widget %s which is not in the stage.", | ||||
|                       st_describe_actor (CLUTTER_ACTOR (widget))); | ||||
|                       desc); | ||||
|  | ||||
|           return g_object_new (ST_TYPE_THEME_NODE, NULL); | ||||
|         } | ||||
|  | ||||
| @@ -817,6 +818,20 @@ st_widget_real_get_focus_chain (StWidget *widget) | ||||
|   return g_list_reverse (visible); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_widget_resource_scale_changed (ClutterActor *actor) | ||||
| { | ||||
|   StWidget *widget = ST_WIDGET (actor); | ||||
|   StWidgetPrivate *priv = st_widget_get_instance_private (widget); | ||||
|   int i; | ||||
|  | ||||
|   for (i = 0; i < G_N_ELEMENTS (priv->paint_states); i++) | ||||
|     st_theme_node_paint_state_invalidate (&priv->paint_states[i]); | ||||
|  | ||||
|   if (CLUTTER_ACTOR_CLASS (st_widget_parent_class)->resource_scale_changed) | ||||
|     CLUTTER_ACTOR_CLASS (st_widget_parent_class)->resource_scale_changed (actor); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_widget_class_init (StWidgetClass *klass) | ||||
| { | ||||
| @@ -846,6 +861,8 @@ st_widget_class_init (StWidgetClass *klass) | ||||
|   actor_class->get_accessible = st_widget_get_accessible; | ||||
|   actor_class->has_accessible = st_widget_has_accessible; | ||||
|  | ||||
|   actor_class->resource_scale_changed = st_widget_resource_scale_changed; | ||||
|  | ||||
|   klass->style_changed = st_widget_real_style_changed; | ||||
|   klass->navigate_focus = st_widget_real_navigate_focus; | ||||
|   klass->get_accessible_type = st_widget_accessible_get_type; | ||||
| @@ -999,21 +1016,6 @@ st_widget_class_init (StWidgetClass *klass) | ||||
|                   G_STRUCT_OFFSET (StWidgetClass, popup_menu), | ||||
|                   NULL, NULL, NULL, | ||||
|                   G_TYPE_NONE, 0); | ||||
|  | ||||
|   /** | ||||
|    * StWidget::resource-scale-changed: | ||||
|    * @widget: the #StWidget | ||||
|    * | ||||
|    * Emitted when the paint scale that the widget will be painted as | ||||
|    * changed. | ||||
|    */ | ||||
|   signals[RESOURCE_SCALE_CHANGED] = | ||||
|     g_signal_new ("resource-scale-changed", | ||||
|                   G_TYPE_FROM_CLASS (klass), | ||||
|                   G_SIGNAL_RUN_LAST, | ||||
|                   G_STRUCT_OFFSET (StWidgetClass, resource_scale_changed), | ||||
|                   NULL, NULL, NULL, | ||||
|                   G_TYPE_NONE, 0); | ||||
| } | ||||
|  | ||||
| static const gchar * | ||||
| @@ -1397,23 +1399,6 @@ st_widget_get_style (StWidget *actor) | ||||
|   return ST_WIDGET_PRIVATE (actor)->inline_style; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * st_widget_get_resource_scale: | ||||
|  * @widget: A #StWidget | ||||
|  * @resource_scale: (out): return location for the resource scale | ||||
|  * | ||||
|  * Retrieves the resource scale for this #StWidget, if available. | ||||
|  * | ||||
|  * The resource scale refers to the scale the actor should use for its resources. | ||||
|  */ | ||||
| gboolean | ||||
| st_widget_get_resource_scale (StWidget *widget, | ||||
|                               float    *resource_scale) | ||||
| { | ||||
|   return clutter_actor_get_resource_scale (CLUTTER_ACTOR (widget), | ||||
|                                            resource_scale); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_widget_set_first_visible_child (StWidget     *widget, | ||||
|                                    ClutterActor *actor) | ||||
| @@ -1480,23 +1465,6 @@ st_widget_name_notify (StWidget   *widget, | ||||
|   st_widget_style_changed (widget); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_widget_resource_scale_notify (StWidget   *widget, | ||||
|                                  GParamSpec *pspec, | ||||
|                                  gpointer    data) | ||||
| { | ||||
|   StWidgetPrivate *priv = st_widget_get_instance_private (widget); | ||||
|   int i; | ||||
|  | ||||
|   for (i = 0; i < G_N_ELEMENTS (priv->paint_states); i++) | ||||
|     st_theme_node_paint_state_invalidate (&priv->paint_states[i]); | ||||
|  | ||||
|   g_signal_emit (widget, signals[RESOURCE_SCALE_CHANGED], 0); | ||||
|  | ||||
|   if (clutter_actor_is_mapped (CLUTTER_ACTOR (widget))) | ||||
|     clutter_actor_queue_redraw (CLUTTER_ACTOR (widget)); | ||||
| } | ||||
|  | ||||
| static void | ||||
| st_widget_reactive_notify (StWidget   *widget, | ||||
|                            GParamSpec *pspec, | ||||
| @@ -1648,7 +1616,6 @@ st_widget_init (StWidget *actor) | ||||
|  | ||||
|   /* connect style changed */ | ||||
|   g_signal_connect (actor, "notify::name", G_CALLBACK (st_widget_name_notify), NULL); | ||||
|   g_signal_connect (actor, "notify::resource-scale", G_CALLBACK (st_widget_resource_scale_notify), NULL); | ||||
|   g_signal_connect (actor, "notify::reactive", G_CALLBACK (st_widget_reactive_notify), NULL); | ||||
|  | ||||
|   g_signal_connect (actor, "notify::visible", G_CALLBACK (st_widget_visible_notify), NULL); | ||||
| @@ -1720,7 +1687,8 @@ st_widget_recompute_style (StWidget    *widget, | ||||
|            */ | ||||
|  | ||||
|           priv->transition_animation = | ||||
|             st_theme_node_transition_new (old_theme_node, | ||||
|             st_theme_node_transition_new (CLUTTER_ACTOR (widget), | ||||
|                                           old_theme_node, | ||||
|                                           new_theme_node, | ||||
|                                           current_paint_state (widget), | ||||
|                                           transition_duration); | ||||
| @@ -1903,7 +1871,7 @@ st_widget_sync_hover (StWidget *widget) | ||||
|  | ||||
|   seat = clutter_backend_get_default_seat (clutter_get_default_backend ()); | ||||
|   pointer = clutter_seat_get_pointer (seat); | ||||
|   pointer_actor = clutter_input_device_get_pointer_actor (pointer); | ||||
|   pointer_actor = clutter_input_device_get_actor (pointer, NULL); | ||||
|   if (pointer_actor && clutter_actor_get_reactive (CLUTTER_ACTOR (widget))) | ||||
|     st_widget_set_hover (widget, clutter_actor_contains (CLUTTER_ACTOR (widget), pointer_actor)); | ||||
|   else | ||||
|   | ||||
| @@ -63,7 +63,6 @@ struct _StWidgetClass | ||||
|   /* signals */ | ||||
|   void     (* style_changed)       (StWidget         *self); | ||||
|   void     (* popup_menu)          (StWidget         *self); | ||||
|   void     (* resource_scale_changed) (StWidget         *self); | ||||
|  | ||||
|   /* vfuncs */ | ||||
|  | ||||
| @@ -135,8 +134,6 @@ StThemeNode *         st_widget_peek_theme_node           (StWidget        *widg | ||||
| GList *               st_widget_get_focus_chain           (StWidget        *widget); | ||||
| void                  st_widget_paint_background          (StWidget            *widget, | ||||
|                                                            ClutterPaintContext *paint_context); | ||||
| gboolean              st_widget_get_resource_scale        (StWidget        *widget, | ||||
|                                                            float           *resource_scale); | ||||
|  | ||||
| /* debug methods */ | ||||
| char  *st_describe_actor       (ClutterActor *actor); | ||||
|   | ||||
| @@ -566,26 +566,26 @@ main (int argc, char **argv) | ||||
|  | ||||
|   root = st_theme_context_get_root_node (context); | ||||
|   group1 = st_theme_node_new (context, root, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group1", NULL, NULL, NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group1", NULL, NULL, NULL); | ||||
|   text1 = st_theme_node_new  (context, group1, NULL, | ||||
|                               CLUTTER_TYPE_TEXT, "text1", "special-text", NULL, NULL); | ||||
|   text2 = st_theme_node_new  (context, group1, NULL, | ||||
|                               CLUTTER_TYPE_TEXT, "text2", NULL, NULL, NULL); | ||||
|   group2 = st_theme_node_new (context, root, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group2", NULL, NULL, NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group2", NULL, NULL, NULL); | ||||
|   group4 = st_theme_node_new (context, root, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group4", NULL, NULL, NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group4", NULL, NULL, NULL); | ||||
|   group5 = st_theme_node_new (context, root, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group5", NULL, NULL, NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group5", NULL, NULL, NULL); | ||||
|   group6 = st_theme_node_new (context, root, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group6", NULL, NULL, NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group6", NULL, NULL, NULL); | ||||
|   text3 = st_theme_node_new  (context, group2, NULL, | ||||
|                               CLUTTER_TYPE_TEXT, "text3", NULL, NULL, | ||||
|                               "color: #0000ff; padding-bottom: 12px;"); | ||||
|   text4 = st_theme_node_new  (context, group2, NULL, | ||||
|                               CLUTTER_TYPE_TEXT, "text4", NULL, "visited hover", NULL); | ||||
|   group3 = st_theme_node_new (context, group2, NULL, | ||||
|                               CLUTTER_TYPE_GROUP, "group3", NULL, "hover", NULL); | ||||
|                               CLUTTER_TYPE_ACTOR, "group3", NULL, "hover", NULL); | ||||
|   button = st_theme_node_new (context, root, NULL, | ||||
|                               ST_TYPE_BUTTON, "button", NULL, NULL, NULL); | ||||
|  | ||||
|   | ||||
| @@ -39,6 +39,7 @@ | ||||
|   </description> | ||||
|  | ||||
|   <releases> | ||||
|     <release version="3.37.3" date="2020-07-07"/> | ||||
|     <release version="3.37.2" date="2020-06-02"/> | ||||
|     <release version="3.37.1" date="2020-04-29"/> | ||||
|   </releases> | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| project('gnome-extensions-app', | ||||
|   version: '3.37.2', | ||||
|   version: '3.37.3', | ||||
|   meson_version: '>= 0.53.0', | ||||
|   license: 'GPLv2+' | ||||
| ) | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| project('gnome-extensions-tool', 'c', | ||||
|   version: '3.37.2', | ||||
|   version: '3.37.3', | ||||
|   meson_version: '>= 0.53.0', | ||||
|   license: 'GPLv2+' | ||||
| ) | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| project('shew', 'c', | ||||
|   version: '3.37.2', | ||||
|   version: '3.37.3', | ||||
|   meson_version: '>= 0.53.0', | ||||
|   license: 'LGPLv2+', | ||||
| ) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user