Compare commits
	
		
			3 Commits
		
	
	
		
			3.7.90
			...
			wip/screen
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | a8b2c13417 | ||
|   | bd9fe20bd1 | ||
|   | 478e546a77 | 
							
								
								
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,6 @@ config.log | ||||
| config.status | ||||
| config | ||||
| configure | ||||
| data/50-gnome-shell-*.xml | ||||
| data/gnome-shell.desktop | ||||
| data/gnome-shell.desktop.in | ||||
| data/gnome-shell-extension-prefs.desktop | ||||
| @@ -24,8 +23,6 @@ data/gnome-shell-extension-prefs.desktop.in | ||||
| data/gschemas.compiled | ||||
| data/org.gnome.shell.gschema.xml | ||||
| data/org.gnome.shell.gschema.valid | ||||
| data/org.gnome.shell.evolution.calendar.gschema.xml | ||||
| data/org.gnome.shell.evolution.calendar.gschema.valid | ||||
| docs/reference/*/*.args | ||||
| docs/reference/*/*.bak | ||||
| docs/reference/*/*.hierarchy | ||||
| @@ -38,7 +35,6 @@ docs/reference/*/*.txt | ||||
| docs/reference/*/*.types | ||||
| docs/reference/*/html/ | ||||
| docs/reference/*/xml/ | ||||
| docs/reference/shell/doc-gen-* | ||||
| gtk-doc.make | ||||
| js/misc/config.js | ||||
| intltool-extract.in | ||||
| @@ -46,14 +42,12 @@ intltool-merge.in | ||||
| intltool-update.in | ||||
| libtool | ||||
| m4/ | ||||
| man/gnome-shell.1 | ||||
| omf.make | ||||
| po/*.gmo | ||||
| po/gnome-shell.pot | ||||
| po/*.header | ||||
| po/*.sed | ||||
| po/*.sin | ||||
| po/.intltool-merge-cache | ||||
| po/Makefile.in.in | ||||
| po/Makevars.template | ||||
| po/POTFILES | ||||
| @@ -66,8 +60,6 @@ src/*-enum-types.[ch] | ||||
| src/*-marshal.[ch] | ||||
| src/Makefile | ||||
| src/Makefile.in | ||||
| src/calendar-server/evolution-calendar.desktop | ||||
| src/calendar-server/evolution-calendar.desktop.in | ||||
| src/calendar-server/org.gnome.Shell.CalendarServer.service | ||||
| src/gnome-shell | ||||
| src/gnome-shell-calendar-server | ||||
| @@ -76,7 +68,6 @@ src/gnome-shell-extension-prefs | ||||
| src/gnome-shell-hotplug-sniffer | ||||
| src/gnome-shell-jhbuild | ||||
| src/gnome-shell-perf-helper | ||||
| src/gnome-shell-perf-tool | ||||
| src/gnome-shell-real | ||||
| src/hotplug-sniffer/org.gnome.Shell.HotplugSniffer.service | ||||
| src/run-js-test | ||||
| @@ -86,8 +77,6 @@ src/test-theme | ||||
| src/st.h | ||||
| src/stamp-st.h | ||||
| src/stamp-st.h.tmp | ||||
| src/st-scroll-view-fade-generated.c | ||||
| src/stamp-st-scroll-view-fade-generated.c | ||||
| stamp-h1 | ||||
| tests/run-test.sh | ||||
| xmldocs.make | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +0,0 @@ | ||||
| [submodule "src/gvc"] | ||||
| 	path = src/gvc | ||||
| 	url = git://git.gnome.org/libgnome-volume-control | ||||
							
								
								
									
										331
									
								
								HACKING
									
									
									
									
									
								
							
							
						
						| @@ -1,331 +0,0 @@ | ||||
| Coding guide | ||||
| ============ | ||||
|  | ||||
| Our goal is to have all JavaScript code in GNOME follow a consistent style. In | ||||
| a dynamic language like JavaScript, it is essential to be rigorous about style | ||||
| (and unit tests), or you rapidly end up with a spaghetti-code mess. | ||||
|  | ||||
| A quick note | ||||
| ------------ | ||||
|  | ||||
| Life isn't fun if you can't break the rules. If a rule seems unnecessarily | ||||
| restrictive while you're coding, ignore it, and let the patch reviewer decide | ||||
| what to do. | ||||
|  | ||||
| Indentation and whitespace | ||||
| -------------------------- | ||||
|  | ||||
| Use four-space indents. Braces are on the same line as their associated | ||||
| statements.  You should only omit braces if *both* sides of the statement are | ||||
| on one line. | ||||
|  | ||||
| * One space after the `function` keyword.  No space between the function name | ||||
| * in a declaration or a call.  One space before the parens in the `if` | ||||
| * statements, or `while`, or `for` loops. | ||||
|  | ||||
|     function foo(a, b) { | ||||
|         let bar; | ||||
|  | ||||
|         if (a > b) | ||||
|             bar = do_thing(a); | ||||
|         else | ||||
|             bar = do_thing(b); | ||||
|  | ||||
|         if (var == 5) { | ||||
|             for (let i = 0; i < 10; i++) { | ||||
|                 print(i); | ||||
|             } | ||||
|         } else { | ||||
|             print(20); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| Semicolons | ||||
| ---------- | ||||
|  | ||||
| JavaScript allows omitting semicolons at the end of lines, but don't. Always | ||||
| end statements with a semicolon. | ||||
|  | ||||
| js2-mode | ||||
| -------- | ||||
|  | ||||
| If using Emacs, do not use js2-mode. It is outdated and hasn't worked for a | ||||
| while. emacs now has a built-in JavaScript mode, js-mode, based on | ||||
| espresso-mode. It is the de facto emacs mode for JavaScript. | ||||
|  | ||||
| File naming and creation | ||||
| ------------------------ | ||||
|  | ||||
| For JavaScript files, use lowerCamelCase-style names, with a `.js` extension. | ||||
|  | ||||
| We only use C where gjs/gobject-introspection is not available for the task, or | ||||
| where C would be cleaner. To work around limitations in | ||||
| gjs/gobject-introspection itself, add a new method in `shell-util.[ch]`. | ||||
|  | ||||
| Like many other GNOME projects, we prefix our C source filenames with the | ||||
| library name followed by a dash, e.g. `shell-app-system.c`. Create a | ||||
| `-private.h` header when you want to share code internally in the | ||||
| library. These headers are not installed, distributed or introspected. | ||||
|  | ||||
| Imports | ||||
| ------- | ||||
|  | ||||
| Use UpperCamelCase when importing modules to distinguish them from ordinary | ||||
| variables, e.g. | ||||
|  | ||||
|     const GLib = imports.gi.GLib; | ||||
|  | ||||
| Imports should be categorized into one of two places. The top-most import block | ||||
| should contain only "environment imports". These are either modules from | ||||
| gobject-introspection or modules added by gjs itself. | ||||
|  | ||||
| The second block of imports should contain only "application imports". These | ||||
| are the JS code that is in the gnome-shell codebase, | ||||
| e.g. `imports.ui.popupMenu`. | ||||
|  | ||||
| Each import block should be sorted alphabetically. Don't import modules you | ||||
| don't use. | ||||
|  | ||||
|     const GLib = imports.gi.GLib; | ||||
|     const Gio = imports.gi.Gio; | ||||
|     const Lang = imports.lang; | ||||
|     const St = imports.gi.St; | ||||
|  | ||||
|     const Main = imports.ui.main; | ||||
|     const Params = imports.misc.params; | ||||
|     const Tweener = imports.ui.tweener; | ||||
|     const Util = imports.misc.util; | ||||
|  | ||||
| The alphabetical ordering should be done independently of the location of the | ||||
| location. Never reference `imports` in actual code. | ||||
|  | ||||
| Constants | ||||
| --------- | ||||
|  | ||||
| We use CONSTANTS_CASE to define constants. All constants should be directly | ||||
| under the imports: | ||||
|  | ||||
|     const MY_DBUS_INTERFACE = 'org.my.Interface'; | ||||
|  | ||||
| Variable declaration | ||||
| -------------------- | ||||
|  | ||||
| Always use either `const` or `let` when defining a variable. | ||||
|  | ||||
|     // Iterating over an array | ||||
|     for (let i = 0; i < arr.length; ++i) { | ||||
|         let item = arr[i]; | ||||
|     } | ||||
|  | ||||
|     // Iterating over an object's properties | ||||
|     for (let prop in someobj) { | ||||
|         ... | ||||
|     } | ||||
|  | ||||
| If you use "var" then the variable is added to function scope, not block scope. | ||||
| See [What's new in JavaScript 1.7](https://developer.mozilla.org/en/JavaScript/New_in_JavaScript/1.7#Block_scope_with_let_%28Merge_into_let_Statement%29) | ||||
|  | ||||
| Classes | ||||
| ------- | ||||
|  | ||||
| There are many approaches to classes in JavaScript. We use our own class framework | ||||
| (sigh), which is built in gjs. The advantage is that it supports inheriting from | ||||
| GObjects, although this feature isn't used very often in the Shell itself. | ||||
|  | ||||
|     const IconLabelMenuItem = new Lang.Class({ | ||||
|         Name: 'IconLabelMenuItem', | ||||
|         Extends: PopupMenu.PopupMenuBaseItem, | ||||
|  | ||||
|         _init: function(icon, label) { | ||||
|             this.parent({ reactive: false }); | ||||
|             this.addActor(icon); | ||||
|             this.addActor(label); | ||||
|         }, | ||||
|  | ||||
|         open: function() { | ||||
|             log("menu opened!"); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| * 'Name' is required. 'Extends' is optional. If you leave it out, you will | ||||
|   automatically inherit from Object. | ||||
|  | ||||
| * Leave a blank line between the "class header" (Name, Extends, and other | ||||
|   things)  and the "class body" (methods). Leave a blank line between each | ||||
|   method. | ||||
|  | ||||
| * No space before the colon, one space after. | ||||
|  | ||||
| * No trailing comma after the last item. | ||||
|  | ||||
| * Make sure to use a semicolon after the closing paren to the class. It's | ||||
|   still a giant function call, even though it may resemble a more | ||||
|   conventional syntax. | ||||
|  | ||||
| GObject Introspection | ||||
| --------------------- | ||||
|  | ||||
| GObject Introspection is a powerful feature that allows us to have native | ||||
| bindings for almost any library built around GObject. If a library requires | ||||
| you to inherit from a type to use it, you can do so: | ||||
|  | ||||
|     const MyClutterActor = new Lang.Class({ | ||||
|         Name: 'MyClutterActor', | ||||
|         Extends: Clutter.Actor, | ||||
|  | ||||
|         vfunc_get_preferred_width: function(actor, forHeight) { | ||||
|              return [100, 100]; | ||||
|         }, | ||||
|  | ||||
|         vfunc_get_preferred_height: function(actor, forWidth) { | ||||
|              return [100, 100]; | ||||
|         }, | ||||
|  | ||||
|         vfunc_paint: function(actor) { | ||||
|              let alloc = this.get_allocation_box(); | ||||
|              Cogl.set_source_color4ub(255, 0, 0, 255); | ||||
|              Cogl.rectangle(alloc.x1, alloc.y1, | ||||
|                             alloc.x2, alloc.y2); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| Translatable strings, `environment.js` | ||||
| -------------------------------------- | ||||
|  | ||||
| We use gettext to translate the GNOME Shell into all the languages that GNOME | ||||
| supports. The `gettext` function is aliased globally as `_`, you do not need to | ||||
| explicitly import it. This is done through some magic in the | ||||
| [environment.js](http://git.gnome.org/browse/gnome-shell/tree/js/ui/environment.js) | ||||
| file. If you can't find a method that's used, it's probably either in gjs itself | ||||
| or installed on the global object from the Environment. | ||||
|  | ||||
| Use 'single quotes' for programming strings that should not be translated | ||||
| and "double quotes" for strings that the user may see. This allows us to | ||||
| quickly find untranslated or mistranslated strings by grepping through the | ||||
| sources for double quotes without a gettext call around them. | ||||
|  | ||||
| `actor` and `_delegate` | ||||
| ----------------------- | ||||
|  | ||||
| gjs allows us to set so-called "expando properties" on introspected objects, | ||||
| allowing us to treat them like any other. Because the Shell was built before | ||||
| you could inherit from GTypes natively in JS, we usually have a wrapper class | ||||
| that has a property called `actor`. We call this wrapper class the "delegate". | ||||
|  | ||||
| We sometimes use expando properties to set a property called `_delegate` on | ||||
| the actor itself: | ||||
|  | ||||
|     const MyClass = new Lang.Class({ | ||||
|         Name: 'MyClass', | ||||
|  | ||||
|         _init: function() { | ||||
|             this.actor = new St.Button({ text: "This is a button" }); | ||||
|             this.actor._delegate = this; | ||||
|  | ||||
|             this.actor.connect('clicked', Lang.bind(this, this._onClicked)); | ||||
|         }, | ||||
|  | ||||
|         _onClicked: function(actor) { | ||||
|             actor.set_label("You clicked the button!"); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| The 'delegate' property is important for anything which trying to get the | ||||
| delegate object from an associated actor. For instance, the drag and drop | ||||
| system calls the `handleDragOver` function on the delegate of a "drop target" | ||||
| when the user drags an item over it. If you do not set the `_delegate` | ||||
| property, your actor will not be able to be dropped onto. | ||||
|  | ||||
| Functional style | ||||
| ---------------- | ||||
|  | ||||
| JavaScript Array objects offer a lot of common functional programming | ||||
| capabilities such as forEach, map, filter and so on. You can use these when | ||||
| they make sense, but please don't have a spaghetti mess of function programming | ||||
| messed in a procedural style. Use your best judgment. | ||||
|  | ||||
| Closures | ||||
| -------- | ||||
|  | ||||
| `this` will not be captured in a closure, it is relative to how the closure is | ||||
| invoked, not to the value of this where the closure is created, because "this" | ||||
| is a keyword with a value passed in at function invocation time, it is not a | ||||
| variable that can be captured in closures. | ||||
|  | ||||
| All closures should be wrapped with a Lang.bind. | ||||
|  | ||||
|     const Lang = imports.lang; | ||||
|  | ||||
|     let closure = Lang.bind(this, function() { this._fnorbate(); }); | ||||
|  | ||||
| A more realistic example would be connecting to a signal on a method of a | ||||
| prototype: | ||||
|  | ||||
|     const Lang = imports.lang; | ||||
|     const FnorbLib = imports.fborbLib; | ||||
|  | ||||
|     const MyClass = new Lang.Class({ | ||||
|         _init: function() { | ||||
|             let fnorb = new FnorbLib.Fnorb(); | ||||
|             fnorb.connect('frobate', Lang.bind(this, this._onFnorbFrobate)); | ||||
|         }, | ||||
|  | ||||
|         _onFnorbFrobate: function(fnorb) { | ||||
|             this._updateFnorb(); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| Object literal syntax | ||||
| --------------------- | ||||
|  | ||||
| In JavaScript, these are equivalent: | ||||
|  | ||||
|     foo = { 'bar': 42 }; | ||||
|     foo = { bar: 42 }; | ||||
|  | ||||
| and so are these: | ||||
|  | ||||
|     var b = foo['bar']; | ||||
|     var b = foo.bar; | ||||
|  | ||||
| If your usage of an object is like an object, then you're defining "member | ||||
| variables." For member variables, use the no-quotes no-brackets syntax: `{ bar: | ||||
| 42 }` `foo.bar`. | ||||
|  | ||||
| If your usage of an object is like a hash table (and thus conceptually the keys | ||||
| can have special chars in them), don't use quotes, but use brackets: `{ bar: 42 | ||||
| }`, `foo['bar']`. | ||||
|  | ||||
| Getters, setters, and Tweener | ||||
| ----------------------------- | ||||
|  | ||||
| Getters and setters should be used when you are dealing with an API that is | ||||
| designed around setting properties, like Tweener. If you want to animate an | ||||
| arbitrary property, create a getter and setter, and use Tweener to animate the | ||||
| property. | ||||
|  | ||||
|     const ANIMATION_TIME = 2000; | ||||
|  | ||||
|     const MyClass = new Lang.Class({ | ||||
|         Name: 'MyClass', | ||||
|  | ||||
|         _init: function() { | ||||
|             this.actor = new St.BoxLayout(); | ||||
|             this._position = 0; | ||||
|         }, | ||||
|  | ||||
|         get position() { | ||||
|             return this._position; | ||||
|         }, | ||||
|  | ||||
|         set position(value) { | ||||
|             this._position = value; | ||||
|             this.actor.set_position(value, value); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     let myThing = new MyClass(); | ||||
|     Tweener.addTween(myThing, | ||||
|                      { position: 100, | ||||
|                        time: ANIMATION_TIME, | ||||
|                        transition: 'easeOutQuad' }); | ||||
							
								
								
									
										10
									
								
								Makefile.am
									
									
									
									
									
								
							
							
						
						| @@ -1,11 +1,7 @@ | ||||
| # Point to our macro directory and pick up user flags from the environment | ||||
| ACLOCAL_AMFLAGS  = -I m4 ${ACLOCAL_FLAGS} | ||||
|  | ||||
| SUBDIRS = data js src browser-plugin tests po docs | ||||
|  | ||||
| if ENABLE_MAN | ||||
| SUBDIRS += man | ||||
| endif | ||||
| SUBDIRS = data js src browser-plugin tests po man docs | ||||
|  | ||||
| EXTRA_DIST =		\ | ||||
| 	.project	\ | ||||
| @@ -16,9 +12,7 @@ EXTRA_DIST =		\ | ||||
| # These are files checked into Git that we don't want to distribute | ||||
| DIST_EXCLUDE =					\ | ||||
| 	.gitignore				\ | ||||
| 	.gitmodules				\ | ||||
| 	gnome-shell.doap			\ | ||||
| 	HACKING					\ | ||||
| 	MAINTAINERS				\ | ||||
| 	tools/build/* | ||||
|  | ||||
| @@ -26,4 +20,4 @@ distcheck-hook: | ||||
| 	@echo "Checking disted files against files in git" | ||||
| 	@$(srcdir)/tools/check-for-missing.py $(srcdir) $(distdir) $(DIST_EXCLUDE) | ||||
|  | ||||
| DISTCHECK_CONFIGURE_FLAGS = --enable-gtk-doc --enable-man | ||||
| DISTCHECK_CONFIGURE_FLAGS = --enable-gtk-doc | ||||
|   | ||||
							
								
								
									
										819
									
								
								NEWS
									
									
									
									
									
								
							
							
						
						| @@ -1,810 +1,3 @@ | ||||
| 3.7.90 | ||||
| ====== | ||||
| * Let GNOME Shell work on EGL and GLES2 [Neil; #693225, #693438, #693339] | ||||
| * Implement middle-click paste [Jasper; #645019] | ||||
| * Fix top bar transition between session modes [Rui; #692966] | ||||
| * Trigger the message tray with downward pressure [Jasper; #677215] | ||||
| * Don't ask for a password on shutdown [Adel; #693385] | ||||
| * Add a context menu to the message tray [Adel; #691035, #693887] | ||||
| * Use unicode formatting in the date menu [Matthias; #689251] | ||||
| * Use proper ellipsis instead of three dots [Jeremy; #689542] | ||||
| * Tweak screen shield animation [Giovanni; #691964] | ||||
| * Always hide the OSK when showing the message tray [Florian; #662687] | ||||
| * Support sound in notifications [Giovanni; #642831] | ||||
| * Place application popup menus above chrome [Jasper; #633620] | ||||
| * Hide overview elements while searching [Cosimo; #682050] | ||||
| * Implement updated IBus candidate popup designs [Rui; #691902] | ||||
| * Add support for enable-animations preference [Cosimo; #655746] | ||||
| * Don't always show the message tray in the overview [Cosimo; #693987] | ||||
| * Improve arrangement of window previews [Adel; #690313] | ||||
| * Remove builtin settings provider [Giovanni; #690824] | ||||
| * Minimize fullscreen windows when they end up in the background [Adel; #693991] | ||||
| * Add context menu to the background actor [Jasper; #681540] | ||||
| * Handle backgrounds in the shell, improve startup animation [Ray; #682429] | ||||
| * Hide universal access menu when not needed [Giovanni; #681528] | ||||
| * Implement updated app picker designs [Florian; #694192] | ||||
| * Improve login manager -> session transition [Ray; #694062] | ||||
| * Don't use a grid layout in window picker [Adel; #694210] | ||||
| * Use scroll wheel for workspace switching rather than zoom [Florian; #686639] | ||||
| * Misc bug fixes and cleanups: [Jasper, Florian, Debarshi, Adel, Matthias, | ||||
|   Giovanni, Daiki, Rico, Bastien, Cosimo, Ray, Allan, Antonio; #693284, | ||||
|   #692680, #691746, #693303, #693162, #693161, #693522, #693385, #691715, | ||||
|   #688915, #689106, #682429, #693570, #693737, #693458, #692845, 693836, | ||||
|   #681540, #679925, #688227, #692773, #693909, #683288, #693854, #693746, | ||||
|   #693931, #693924, #693940, #693970, #693935, #693937, #693974, #693936, | ||||
|   #693975, #693822, #694030, #685849, #694052, #694035, #694038, #694079, | ||||
|   #694064, #681735, #694100, #694057, #694070, #693572, #693896, #686984, | ||||
|   #694123, #694125, #693756, #693757, #687556, #694215, 694062, #694227, | ||||
|   #694240, #694234, #694264, 694276, 694282, #694241, #689394, #694202, | ||||
|   #694265, #694289, #691806, #694290, #694296] | ||||
|  | ||||
| Contributors: | ||||
|   Jeremy Bicha, Giovanni Campagna, Cosimo Cecchi, Matthias Clasen, Allan Day, | ||||
|   António Fernandes, Adel Gadllah, Rui Matos, Florian Müllner, Bastien Nocera, | ||||
|   Debarshi Ray, Neil Roberts, Jasper St. Pierre, Ray Strode, Rico Tzschichholz, | ||||
|   Daiki Ueno | ||||
|  | ||||
| Translations: | ||||
|   Yasumichi Akahoshi [ja], Yoji TOYODA [ja], Dušan Kazik [sk], | ||||
|   Wouter Bolsterlee [nl], Matej Urbančič [sl], Gheyret Kenji [ug], | ||||
|   Ivaylo Valkov [bg], Daniel Korostil [uk], Gheyret Kenji [ug], | ||||
|   Daniel Mustieles [es], Anish A [ml], Gil Forcada [ca], | ||||
|   Carles Ferrando [ca@valencia], Мирослав Николић [sr, sr@latin], | ||||
|   Aurimas Černius [lt], Rafael Ferreira [pt_BR], Fran Diéguez [gl], | ||||
|   Piotr Drąg [pl], Luca Ferretti [it], A S Alam [pa] | ||||
|  | ||||
| 3.7.5 | ||||
| ===== | ||||
| * MessageTray: pass keyboard events to tray icons [Giovanni; #687425] | ||||
| * network: add support for virtual devices (vlan, bond, bridge) [Dan; #677144] | ||||
| * gdm: Allow right-clicking buttons for left-handed users [Jasper; #688748] | ||||
| * Make list search results span all available horizontal space [Tanner; #691967] | ||||
| * Make Show-Applications button depress when held down [Hashem; #692319] | ||||
| * Set a max width on search results [Cosimo; #692453] | ||||
| * Reserve scrollbar allocation for automatic policy [Cosimo; #686881] | ||||
| * Improve scaling algorithm for window thumbnails [Jasper; #686944] | ||||
| * Fix launching settings panels after g-c-c changes [Jasper; #692483] | ||||
| * Stop launching applications from empty searches [Hashem; #692391] | ||||
| * Implement per-source notification filtering [Giovanni; #685926] | ||||
| * ScreenShield: Omit ActiveChanged() signal at end of fade [Giovanni; #691964] | ||||
| * ScreenShield: Lower the shield on idle before locking [Giovanni; #692560] | ||||
| * Make previews of minimized windows translucent in overview [Florian; #692999] | ||||
| * windowManager: Respect icon geometry when minimizing [Florian; #692997] | ||||
| * ScreenShield: Only show lock icon when actually locked [Giovanni; #693007] | ||||
| * general: Use & instead of 'and' for Settings panels [Jeremy; #689590] | ||||
| * network: Add support for new ModemManager1 interface [Aleksander; #687359] | ||||
| * network: Handle LTE-only modems as GSM ones [Aleksander; #688144] | ||||
| * mobile-providers: Port to libnm-gtk [Aleksander; #688943] | ||||
| * general: Consistently use Title Case in top bar [Jeremy; #689589] | ||||
| * panel: Add :overview pseudo class while in overview [Florian; #693218] | ||||
| * sessionMode: Add support for mode-specific styling [Florian; #693219] | ||||
| * loginManager: Make suspend a NOP in the ConsoleKit patch [Florian; #693162] | ||||
| * screenShield: Inhibit suspend until the screen is locked [Florian; #686482] | ||||
| * Misc bug fixes and cleanups [Jasper, Giovanni, Rui, Cosimo, Florian, Stefano, | ||||
|   Adel, Yanko; #691745, #691731, #690171, #689091, #691976, #691963, #684279, | ||||
|   #692052, #692091, #642831, #692454, #692715, #692678, #692723, #692677, | ||||
|   #683986, #692693, #692749, #692948, #692995, #692996, #692994, #677215, | ||||
|   #692586, #693067, #693031, #693049, #643111, #693161, #693220] | ||||
|  | ||||
| Contributors: | ||||
|   Jeremy Bicha, Giovanni Campagna, Cosimo Cecchi, Tanner Doshier, | ||||
|   Stefano Facchini, Adel Gadllah, Yanko Kaneti, Rui Matos, Aleksander Morgado, | ||||
|   Florian Müllner, Hashem Nasarat, Jasper St. Pierre, Dan Winship | ||||
|  | ||||
| Translations: | ||||
|   Duarte Loreto [pt], Daniel Mustieles [es], Kjartan Maraas [nb], | ||||
|   Nilamdyuti Goswami [as], Мирослав Николић [sr,sr@latin], | ||||
|   Tobias Endrigkeit [de], Fabio Tomat [fur], Matej Urbančič [sl], A S Alam [pa], | ||||
|   Inaki Larranaga Murgoitio [eu], Piotr Drąg [pl], Wouter Bolsterlee [nl], | ||||
|   Gheyret Kenji [ug], Yaron Shahrabani [he], Chao-Hsiung Liao [zh_HK,zh_TW], | ||||
|   Milo Casagrande [it], Benjamin Steinwender [de] | ||||
|  | ||||
| 3.7.4.1 | ||||
| ======= | ||||
| * userMenu: Use show-full-name-in-top-bar setting [Bastien; #689561] | ||||
| * dateMenu: Add "Open Clocks" entry [Mathieu; #644390] | ||||
| * screenshot: Immediately show the flash spot [Jasper; #691875] | ||||
| * Misc. bug fixes [Rico, Jeremy] | ||||
|  | ||||
| Contributors: | ||||
|   Jeremy Bicha, Mathieu Bridon, Bastien Nocera, Jasper St. Pierre, | ||||
|   Rico Tzschichholz | ||||
|  | ||||
| Translations: | ||||
|   Ihar Hrachyshka [be] | ||||
|  | ||||
| 3.7.4 | ||||
| ===== | ||||
| * Make menu separators crisp [Giovanni, Allan; #641745] | ||||
| * power: Update for new D-Bus name [Bastien; #690506] | ||||
| * Add smooth scrolling support [Jasper; #687573] | ||||
| * Tweak notification layout [Allan; #688506] | ||||
| * Ping the active window when using the app menu [Giovanni; #684340] | ||||
| * Make password entries insensitive after submission [Jasper; #690594, #690895] | ||||
| * Honor lock-delay GSettings key [Giovanni, Matthias; #690766, #691170] | ||||
| * Use text/calendar preferred app as the calendar app [Giovanni; #690767] | ||||
| * lookingGlass: Move to an inspect() function [Jasper; #690726] | ||||
| * Make OSK animation quicker, snappier [Rui; #688642] | ||||
| * Allow to close chat notifications with Escape [Jasper; #690897] | ||||
| * Honor org.gnome.desktop.screensaver.user-switch-enabled [Giovanni; #691042] | ||||
| * Add a SelectArea() DBus method [Cosimo; #687954] | ||||
| * Support non-absolute paths when saving screenshots [Cosimo; #688004] | ||||
| * OSK: Fix extended keys popups [Rui; #674955] | ||||
| * Don't hide or show the keyboard immediately [Rui; #688646] | ||||
| * Improve padding in power menu [Giovanni; #689297] | ||||
| * Add per-window input source switching [Rui; #691414] | ||||
| * Misc bug fixes and cleanups [Rico, Jasper, Giovanni, Rui, Florian, Dan; | ||||
|   #690608, #690589, #690539, #687081, #690667, #690665, #690666, #685856, | ||||
|   #690858, #690895, #680414, #690965, #691019, #690590, #681376, #690180, | ||||
|   #685513, #689263, #691553, #691720, #691743, #691750] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Cosimo Cecchi, Matthias Clasen, Allan Day, Rui Matos, | ||||
|   Florian Müllner, Bastien Nocera, Jasper St. Pierre, Rico Tzschichholz, | ||||
|   Dan Winship | ||||
|  | ||||
| Translations: | ||||
|   Matej Urbančič [sl], Kjartan Maraas [nb], Mattias Põldaru [et], | ||||
|   Yaron Shahrabani [he], Aurimas Černius [lt], Khaled Hosny [ar], | ||||
|   Fran Diéguez [gl], Daniel Mustieles [es], Piotr Drąg [pl], Balázs Úr [hu], | ||||
|   Baurzhan Muftakhidinov [kk], Tobias Endrigkeit [de], Dušan Kazik [sk], | ||||
|   Aron Xu [zh_CN], Gheyret Kenji [ug] | ||||
|  | ||||
| 3.7.3 | ||||
| ===== | ||||
| * Add 'No Messages' label when message tray is empty [Victoria; #686738] | ||||
| * Use better icons in Ctrl-Alt-Tab popup [Stéphane; #641303] | ||||
| * Show the OSK on the monitor where the focused window lives [Giovanni; #685856] | ||||
| * Highlight window clone and caption when hovered [Giovanni, Marc; #665310] | ||||
| * Improve login process indication [Stéphane; #687113] | ||||
| * Omit empty categories in apps view [Stéphane; #687970] | ||||
| * Style panel differently according to mode [Florian; #684573] | ||||
| * Make it possible to hide the user name [Matthias; #688577] | ||||
| * Consolidate and improve chat connection notifications [Giovanni; #687213] | ||||
| * Improve notification scrollbar appearance [Carlos; #688393] | ||||
| * Fade scroll view fade near scrolling edges [Jasper; #689249] | ||||
| * Add a read-only org.gnome.Shell.Mode property [Debarshi; #689300] | ||||
| * Don't close message tray after using context menus [Giovanni; #689296] | ||||
| * Port swipe-scrolling to ClutterPanAction [Jasper, Florian; #689062, #689552] | ||||
| * Remember state of 'Remember Password' checkbox [Ron; #688039] | ||||
| * Improve timestamp format in chat notifications [Carlos; #680989] | ||||
| * Improve style of missed-messages counter [Carlos; #686472] | ||||
| * Omit connection failure notifications if cancelled by user [Giovanni; #684823] | ||||
| * Add window-based Alt-Tab popup [Florian; #688913] | ||||
| * Support external session mode definitions [Florian; #689304] | ||||
| * Support session-mode-specific extensions [Florian; #689305] | ||||
| * Support 'parentMode' property in session modes [Florian; #689308] | ||||
| * Support a new org.gnome.ShellSearchProvider2 DBus interface | ||||
|   [Cosimo; #689735, #690009] | ||||
| * Add "windows" to Ctrl-Alt-Tab popup [Jasper; #689653] | ||||
| * Port PopupMenu to GrabHelper [Jasper; #689109, #689954] | ||||
| * Show headphone icon when headphones are plugged in [Giovanni; #675902] | ||||
| * Display (non-app) search results as list [Tanner, Cosimo; #681797] | ||||
| * Skip diacritical marks in search terms [Aleksander; #648587] | ||||
| * Expose all engine options in input sources [Giovanni, Rui; #682318] | ||||
| * Add input source switcher popup [Rui; #682315] | ||||
| * Add minimal support for InfiniBand in network menu [Dan; #677150] | ||||
| * Misc bug fixes and cleanups [Sebastian, Aleksander, Giovanni, Tim, Cosimo, | ||||
|   Florian, Matthias, Rui, Lionel, Colin, Piotr, Guillaume, Bastien, Tanner, | ||||
|   Carlos, Stéphane, Jakub; #688422, #688379, #688750, #688771, #686800, | ||||
|   #688133, #688895, #688966, #683986, #688004, #689108, #689029, #683449, | ||||
|   #688196, #689304, #689243, #689295, #689325, #689400, #679168, #689568, | ||||
|   #689537, #689528, #689749, #689789, #689353, #689820, #689868, #689778, | ||||
|   #689959, #688589, #688589, #689955, #687250, #689965, #690046, #690049, | ||||
|   #689884, #682286, #690173, #690174, #672941, #689876, #687881, #690171, | ||||
|   #690241, #690312, #690175, #687955, #650843, #688234, #690427 | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Cosimo Cecchi, Matthias Clasen, Stéphane Démurget, | ||||
|   Guillaume Desmottes, Tanner Doshier, Piotr Drąg, Sebastian Keller, | ||||
|   Lionel Landwerlin, Tim Lunn, Victoria Martínez de la Cruz, Aleksander Morgado, | ||||
|   Florian Müllner, Bastien Nocera, Marc Plano-Lesay, Carlos Soriano Sánchez, | ||||
|   Jakub Steiner, Jasper St. Pierre, Colin Walters, Dan Winship, Ron Yorston | ||||
|  | ||||
| Translations: | ||||
|   Yuri Myasoedov [ru], Wouter Bolsterlee [nl], Yaron Shahrabani [he], | ||||
|   Nilamdyuti Goswami [as], Ani Peter [ml], Kjartan Maraas [nb], | ||||
|   Dr.T.Vasudevan [ta], A S Alam [pa], Shankar Prasad [kn], Khaled Hosny [ar], | ||||
|   Daniel Mustieles [es], Dušan Kazik [sk] | ||||
|  | ||||
| 3.7.2 | ||||
| ===== | ||||
| * Enforce RTL in he for messages that might end up as LTR [Florian; #686630] | ||||
| * gdm: Move logo into the panel [Florian; #685852] | ||||
| * Hide notifications when closed button is clicked [Jasper, Florian; #682237] | ||||
| * Tweak screenShield animations [Rui; #686745] | ||||
| * Restore Fittsability of summary items in message tray [Florian; #686474] | ||||
| * Save screencasts as recent item [Ray; #680647] | ||||
| * overview: Resize window captions on content change [Giovanni, Alex; #620874] | ||||
| * App search: Match GenericName too [Matthias; #687121] | ||||
| * runDialog: Better match style of other modal dialogs [Florian, Allan; #687127] | ||||
| * Improve the button insensitive style [Stéphane; #687110] | ||||
| * network: Don't use a global switch for all VPN connections [Giovanni; #682929] | ||||
| * appMenu: Update on icon theme changes [Florian; #687224] | ||||
| * Show 'Log out' in more situations [Matthias; #686736] | ||||
| * Add a setting to force the 'Log out' menuitem [Matthias; #686057] | ||||
| * overview: Improve styling of search box [Stéphane; #686479] | ||||
| * Implement 'disable-user-list' in login screen [Ray; #660660] | ||||
| * Fix auto-scroll to bottom in chat notifications [Sjoerd; #686571] | ||||
| * Show feedback notifications when user is busy [Stéphane; #662900] | ||||
| * Disable login button when there is no input [Stéphane; #687112] | ||||
| * Use non-linear overview shade for background [Giovanni, Pierre-Eric; #669798] | ||||
| * Reduce blocking in compositor thread [Simon, Jasper; #687465] | ||||
| * network: new country-specific type to gather providers [Aleksander; #687356] | ||||
| * Update man page [Matthias; #680601] | ||||
| * st-entry: Change the pointer cursor on enter/leave events [Thomas; #687130] | ||||
| * screenShield: Blur and desaturate the background [Giovanni, Cosimo; #682536] | ||||
| * Change height of chat notifications to have more context [Carlos; #665255] | ||||
| * screenShield: Account for motion velocity when hiding [Giovanni; #682537] | ||||
| * screenShield: hide the cursor while the lock screen is on [Giovanni; #682535] | ||||
| * Support remote search provider settings [Cosimo; #687491] | ||||
| * unlockDialog: Improve label of confirmation button [Stéphane; #687656] | ||||
| * userMenu: Rename "System Settings" item to "Settings" [Elad; #687738] | ||||
| * messageTray: Add keybinding to focus current notification [Stéphane; #652082] | ||||
| * Remove shell-screen-grabber [Neil; #685915] | ||||
| * main: Stop using Metacity's keybinding files [Florian; #687672] | ||||
| * Bluetooth: Remove ObexFTP functionality [Bastien; #688160] | ||||
| * a11y: Also set WM theme when HighContrast is switched on [Cosimo; #688256] | ||||
| * network: Rework multiple NIC support [Giovanni; #677142] | ||||
| * Rework keybindings to allow selective blocking/processing [Florian; #688202] | ||||
| * recorder: Show indicator on primary monitor [Adel; #688470] | ||||
| * recorder: Set frame duration to fix broken video headers [Adel; #688487] | ||||
| * Misc. bugfixes and cleanups [Florian, Jasper, Giovanni, Matthew, Stéphane, | ||||
|   Allan, Daiki, Owen, Alejandro, Jean-François, Cosimo, Sebastian, Adel, Alban; | ||||
|   #686484, #686728, #686805, #686574, #686763, #682428, #687132, #685239, | ||||
|   #687189, #687226, #658091, #670687, #687457, #687242, #687287, #687020, | ||||
|   #686583, #661194, #687491, #657315, #687958, #683986, #688089, #687708, | ||||
|   #686530, #684810, #688181, #688475, #688557, #688507, #638351] | ||||
|  | ||||
| Contributors: | ||||
|   Elad Alfassa, Matthew Barnes, Alban Browaeys, Giovanni Campagna, | ||||
|   Cosimo Cecchi, Matthias Clasen, Allan Day, Stéphane Démurget, | ||||
|   Jean-François Fortin Tam, Adel Gadllah, Alex Hultman, Sebastian Keller, | ||||
|   Rui Matos, Simon McVittie, Aleksander Morgado, Florian Müllner, | ||||
|   Bastien Nocera, Pierre-Eric Pelloux-Prayer, Alejandro Piñeiro, Neil Roberts, | ||||
|   Sjoerd Simons, Carlos Soriano Sánchez, Jasper St. Pierre, Ray Strode, | ||||
|   Owen Taylor, Daiki Ueno, Thomas Wood | ||||
|  | ||||
| Translations: | ||||
|   Dušan Kazik [sk], Pavol Klačanský [sk], Piotr Drąg [pl], Yuri Myasoedov [ru], | ||||
|   Marek Černocký [cs], Kjartan Maraas [nb], Wolfgang Stöggl [de], | ||||
|   Yaron Shahrabani [he], Fran Diéguez [gl], Mattias Põldaru [et] | ||||
|  | ||||
| 3.7.1 | ||||
| ===== | ||||
| * Add shortcut to open application view directly [Jeremy; #685738] | ||||
| * Expose '<Super>F10' shortcut in System Settings [Florian; #672909] | ||||
| * Clean up timestamp format in chat notifications [Carlos; #680989] | ||||
| * loginScreen: Add support for 'disable-restart-buttons' [Florian; #686247] | ||||
| * Update textures automatically on file changes [Florian; #679268] | ||||
| * Implement org.gnome.ScreenSaver.GetActiveTime [Giovanni; #686064] | ||||
| * Add missing translations for GSetting schema [Giovanni; #686413] | ||||
| * Hide workspace switcher completely when it's not necessary [Seif; #686483] | ||||
| * Explicitly load gnome-screensaver when not running GDM [Tim; #683060] | ||||
| * Port to GnomeIdleMonitor [Jasper; #682224] | ||||
| * Set Empathy as preferred handler when delegating channels [Xavier; #686296] | ||||
| * Allow testing GDM login dialog from the session [Giovanni; #683725] | ||||
| * Use all available space for windows in window picker [Jasper, Pierre-Eric; | ||||
|   #582650] | ||||
| * Use logind for suspend if available [Florian; #686482] | ||||
| * Misc. fixes and cleanups [Jasper, Florian, Adel, Rui; #677426, #680426, | ||||
|   #686233, #686241, #686318, #686240, #686484, #686002, #684650, #686487] | ||||
|  | ||||
| Contributors: | ||||
|   Jeremy Bicha, Giovanni Campagna, Xavier Claessens, Adel Gadllah, Seif Lotfy, | ||||
|   Tim Lunn, Rui Matos, Florian Müllner, Pierre-Eric Pelloux-Prayer, | ||||
|   Carlos Soriano, Jasper St. Pierre | ||||
|  | ||||
| Translations: | ||||
|   Andika Triwidada [id], Matej Urbančič [sl], Ihar Hrachyshka [be], | ||||
|   Daniel Mustieles [es], Fran Diéguez [gl], Takayuki KUSANO [ja], | ||||
|   Мирослав Николић [sr, sr@latin], Dušan Kazik [sk], Tom Tryfonidis [el] | ||||
|  | ||||
| 3.6.1 | ||||
| ===== | ||||
| * dash: Make padding even on the top/bottom of the dash [Jasper; #684619] | ||||
| * Fix a crash when dragging search results [Jasper; #684888] | ||||
| * workspaceThumbnail: Fix dragging with static workspaces [Jasper; #684641] | ||||
| * Really hide 'Show Keyboard Layout' on the lock screen [Matthias] | ||||
| * Misc. improvements to jhbuild setup [Owen; #685352, #685353, #685354, #685355] | ||||
| * Show message tray in Ctrl+Alt+Tab outside of the overview [Jasper, Florian; | ||||
|   #684633, #685914] | ||||
| * Disable hotplug sniffer on remote filesystems [Jasper; #684093] | ||||
| * userMenu: Remove 'Switch Session' item [Florian; #685062] | ||||
| * unlockDialog: Make prompt entry insensitive while logging in [Jasper; #685444] | ||||
| * messageTray: Don't animate desktop clone for failed grabs [Jasper; #685342] | ||||
| * Fix crash on dragging windows between workspaces [Ryan; #681399] | ||||
| * userMenu: Ignore 'lock-enabled' setting for user switching [Florian; #685536] | ||||
| * gdm: Fix key-focus on first user [Adel; #684650] | ||||
| * Make grid button insensitive when dragging non-favorites [Jasper; #685313] | ||||
| * Calendar: hide all actions when on the login screen [Matthias; #685142] | ||||
| * Adapt unlock dialog layout for the login screen [Florian; #685201] | ||||
| * Make focus-follows-mouse work better with Shell UI [Florian; #678169] | ||||
| * Improve look of screen shield [Jasper; #685919] | ||||
| * Fix keynav in the login screen [Florian; #684730] | ||||
| * dateMenu: Hide "Open Calendar" item if calendar unavailable [Florian; #686050] | ||||
| * unlockDialog: Reset UI on verification failure [Giovanni; #685441] | ||||
| * Show unlock dialog on primary monitor when using keynav [Giovanni; #685855] | ||||
| * Fix height changes of entries when entering text [Florian; #685534] | ||||
| * Fix show-apps label after successful drags [Florian; #684627] | ||||
| * Misc. bugfixes and cleanups [Jasper, Olivier, Florian, Owen, Adel, Tanner, Tim, Matthias; #685434, #685511, #685466, #685341, #685156, #681159, #673189, #686016, 684869, #686079, #686063 | ||||
|  | ||||
| Contributors: | ||||
| Jasper St. Pierre | ||||
| Matthias Clasen | ||||
| Owen Taylor | ||||
| Olivier Blin | ||||
| Florian Müllner | ||||
| Ryan Lortie | ||||
| Adel Gadllah | ||||
| Tanner Doshier | ||||
| Tim Lunn | ||||
| Giovanni Campagna | ||||
|  | ||||
| Translations: | ||||
|   Tobias Endrigkeit [de], Rudolfs Mazurs [lv], Ask H. Larsen [da], | ||||
|   Shankar Prasad [kn], Changwoo Ryu [ko], Chris Leonard [en_GB], | ||||
|   Arash Mousavi [fa], Theppitak Karoonboonyanan [th], Seán de Búrca [ga], | ||||
|   Yaron Shahrabani [he], Alexander Shopov [bg], Žygimantas Beručka [lt], | ||||
|   Milo Casagrande [it], Kjartan Maraas [nb], Kris Thomsen [da], | ||||
|   Aurimas Černius [lt], Yuri Myasoedov [ru], Мирослав Николић [sr], | ||||
|   Marek Černocký [cs], Gabor Kelemen [hu], Ihar Hrachyshka [be], | ||||
|   Chao-Hsiung Liao [zh_HK, zh_TW], Eleanor Chen [zh_CN], | ||||
|   Carles Ferrando [ca@valencia], Vicent Cubells [ca], Daniel Korostil [uk], | ||||
|   Alexandre Franke [fr], Piotr Drąg [pl] | ||||
|  | ||||
| 3.6.0 | ||||
| ===== | ||||
| * keyboard: Make input source items accessible [Florian; #684462] | ||||
| * Don't show network dialogs in the lock screen [Giovanni; #684384] | ||||
| * popupMenu: Fix initial visibility of settings items [Florian; #684473] | ||||
| * userMenu: Close menu immediately on user/session switch [Florian; #684459] | ||||
| * Fix alignment of search section headers in RTL locales [Florian; #684379] | ||||
| * screenShield: Fix unlock animation [Florian; #684591] | ||||
| * Don't open the tray from a dwell while in a modal grab [Jasper; #684458] | ||||
| * userMenu: Fix texture updates on icon changes [Florian; #679268] | ||||
| * Fix a11y support in the login screen [Florian, Ray; #684727, #684728, #684748] | ||||
| * Make On-Screen-Keyboard usable with new message tray [Giovanni, Florian; | ||||
|   #683546] | ||||
| * Fix initial visibility of input volume in lock-screen [Florian; #684611] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Florian Müllner, Jasper St. Pierre, Ray Strode | ||||
|  | ||||
| Translations: | ||||
|   Matej Urbančič [sl], Dr.T.Vasudevan [ta], Piotr Drąg [pl], A S Alam [pa], | ||||
|   Alexander Shopov [bg], Nilamdyuti Goswami [as], Chandan Kumar [hi], | ||||
|   Khaled Hosny [ar], Ibrahim Saed [ar], Sandeep Sheshrao Shedmake [mr], | ||||
|   Tom Tryfonidis [el], Theppitak Karoonboonyanan [th], Alexandre Franke [fr], | ||||
|   Fran Diéguez [gl], Gabor Kelemen [hu], Ani Peter [ml], Daniel Mustieles [es], | ||||
|   Мирослав Николић [sr, sr@latin], Duarte Loreto [pt], ManojKumar Giri [or], | ||||
|   Ihar Hrachyshka [be], Aurimas Černius [lt], Djavan Fagundes [pt_BR], | ||||
|   Changwoo Ryu [ko], Bruce Cowan [en_GB], Kris Thomsen [da], Gil Forcada [ca], | ||||
|   Yaron Shahrabani [he], Milo Casagrande [it], Ville-Pekka Vainio [fi], | ||||
|   YunQiang Su [zh_CN], Carles Ferrando [ca@valencia], Mario Blättermann [de], | ||||
|   Rajesh Ranjan [hi], Yuri Myasoedov [ru], Rūdolfs Mazurs [lv], | ||||
|   Jiro Matsuzawa [ja], Mattias Põldaru [et], Timur Zhamakeev [ky], | ||||
|   Petr Kovar [cs], Chao-Hsiung Liao [zh_HK,zh_TW], Andika Triwidada [id] | ||||
|  | ||||
| 3.5.92 | ||||
| ====== | ||||
| * Login/UnlockDialog: Don't reset immediately if auth fails [Giovanni; #682544] | ||||
| * Allow changing session mode at runtime [Jasper, Giovanni; #683156] | ||||
| * Add zoom out animation on login [Jasper; #683170] | ||||
| * Bluetooth: don't restrict the length of non numeric PINs [Giovanni; #683356] | ||||
| * Force chat notification to stay open when focusing entry [Debarshi; #682236] | ||||
| * Make sure the screen is fully locked before suspending [Giovanni; #683448] | ||||
| * st-texture-cache: Fix a case of distorted textures [Florian; #683483] | ||||
| * popupSubMenu: Fix padding for non-scrolled submenus [Florian; #683009] | ||||
| * popupMenu: Fix width changes on submenu open/close [Florian; #683485] | ||||
| * boxpointer: Avoid malformed boxpointer arrow [Debarshi; #680077] | ||||
| * Change stage background color to grey [Adel; #683514] | ||||
| * messageTray: Update style of summary counters [Debarshi; #682891] | ||||
| * Don't fail if a legacy tray icon has no WM_CLASS [Giovanni; #683724] | ||||
| * PolkitAgent: Fix a crash if there is no avatar [Giovanni; #683707] | ||||
| * Hide the a11y menu in the lock screen, but show it in the login screen | ||||
|   [Giovanni; #682542] | ||||
| * Fix show-apps button dropping off the dash [Florian; #683340] | ||||
| * Fix committing strings to shell entries from input method [Florian; #658325] | ||||
| * Make IBus display strings consistent with control-center [Rui; #683124] | ||||
| * Fix missing short codes for some input sources [Rui; #683613] | ||||
| * Remove support for long-press from entry context menus [Jasper; #683509] | ||||
| * screenShield: Add box-shadow to the shield [Florian] | ||||
| * Don't show a right-click menu for the hotplug source [Jasper; #683438] | ||||
| * Fix extension styling [Giovanni; #682128] | ||||
| * Fix on-screen keyboard not working with system-modal dialogs | ||||
|   [Giovanni; #664309] | ||||
| * Fix insensitive styling for popup menu items [Giovanni; #683988] | ||||
| * Disable the message tray dwell when the user is interacting [Owen; #683811] | ||||
| * Animate going from the unlock dialog to the lock screen [Giovanni; #681143] | ||||
| * Autostart fprintd when necessary [Ray; #683131] | ||||
| * UnlockDialog: Allow typing before the first PAM question [Giovanni; #681576] | ||||
| * Make Return key dismiss screenshield [Ray; #683889] | ||||
| * Fix keyboard navigation in the message tray [Florian; #682243] | ||||
| * Remove the places & devices search provider [Giovanni; #683506] | ||||
| * Enable hot corner while the message tray is up [Florian; #682255] | ||||
| * Port screen recorder to new GStreamer vp8enc API [Adel; #684206] | ||||
| * Fix fish flickering [Giovanni; #684154] | ||||
| * Fix extension ordering with !important [Jasper; #684163] | ||||
| * Allow the shell to run without the screenshield [Giovanni; #683060] | ||||
| * Add menu items for IBus Anthy's InputMode, TypingMode [Rui; #682314] | ||||
| * Improve transition to the login dialog [Jasper; #682428] | ||||
| * Keep unlock dialog around until shield animation ends [Florian; #684342] | ||||
| * Expose shell keybindings in System Settings [Florian; #671010] | ||||
| * Misc. bugfixes and cleanups [Debarshi, Florian, Giovanni, Jasper, Rico, Rui; | ||||
|   #672790, #677434, #683305, #683357, #683369, #683377, #683378, #683400, | ||||
|   #683449, #683472, #683482, #683487, #683488, #683526, #683529, #683546, | ||||
|   #683583, #683628, #683705, #683982, #683989, #684035, #684036, #684040, | ||||
|   #684162, #684214, #684343] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Adel Gadllah, Rui Matos, Florian Müllner, Debarshi Ray, | ||||
|   Jasper St. Pierre, Ray Strode, Owen Taylor, Rico Tzschichholz | ||||
|  | ||||
| Translations: | ||||
|   Gabor Kelemen [hu], Piotr Drąg [pl], Khaled Hosny [ar], | ||||
|   Мирослав Николић [sr, sr@latin], Chao-Hsiung Liao [zh_HK, zh_TW], | ||||
|   Bruce Cowan [en_GB], Dirgita [id], Tom Tryfonidis [el], Timo Jyrinki [fi], | ||||
|   Adorilson Bezerra [pt_BR], Arash Mousavi [fa], Matej Urbančič [sl], | ||||
|   Christian Kirbach [de], Yaron Shahrabani [he], Ihar Hrachyshka [be], | ||||
|   Changwoo Ryu [ko], Duarte Loreto [pt], Theppitak Karoonboonyanan [th], | ||||
|   Nilamdyuti Goswami [as], Sandeep Sheshrao Shedmake [mr], | ||||
|   Alexandre Franke [fr], Ivaylo Valkov [bg], tuhaihe [zh_CN], | ||||
|   Yuri Myasoedov [ru], Aurimas Černius [lt], Andika Triwidada [id], | ||||
|   Rajesh Ranjan [hi], Sweta Kothari [gu], Daniel Mustieles [es], | ||||
|   Fran Diéguez [gl], Praveen Illa [te] | ||||
|  | ||||
| 3.5.91 | ||||
| ====== | ||||
| * Improve modal dialog styling of network secret prompts [Jasper; #682412] | ||||
| * Fix visibility of non-active workspaces during overview transition | ||||
|   [Florian; #682002] | ||||
| * Improve scrollbar theming [Cosimo; #682476] | ||||
| * Make sure the app menu remains hidden in locked state [Florian; #682475] | ||||
| * Add tooltip to show-applications icon [Jasper; #682445] | ||||
| * Do not add duplicate remote search providers [Florian; #682470] | ||||
| * Handle 'popup-menu' signal on summary items [Florian; #682486] | ||||
| * Fix dwelling during mouse-down [Owen; #682385] | ||||
| * Set label actor for endSessionDialog.ListItem [Alejandro; #677503] | ||||
| * Don't match on comments when searching applications [Florian; #682529] | ||||
| * Make workspace selector more similar to the mockup [Stefano; #662087] | ||||
| * Fix extension installation and reloading [Jasper; #682578] | ||||
| * Hide removable devices in the lock screen [Giovanni; #681143] | ||||
| * Reset cancellable after hitting Escape on login screen [Alban; #681537] | ||||
| * Fix suspend from the user menu [Giovanni; #682746] | ||||
| * Set label actor for summary items in message tray [Alejandro; #677229] | ||||
| * Set label for the "Show applications" dash button [Alejandro; #682366] | ||||
| * Load extensions as late as possible [Jasper; #682822] | ||||
| * Improve mount operation dialogs [Jon; #682645] | ||||
| * Remove "Connect to ..." item from places search [Florian; #682817] | ||||
| * Don't auto-expand notifications with actions [Giovanni; #682738] | ||||
| * Add a new lock screen menu to combine volume network and power | ||||
|   [Giovanni; #682540] | ||||
| * Add support for pre-edit to StIMText [Daiki; #664041] | ||||
| * Remove StIconType [Jasper, Florian, Rui, Giovanni, Debarshi; #682540] | ||||
| * Use monitor geometry for dwelling [Florian; #683044] | ||||
| * Add support for surrounding-text to StIMText [Daiki; #683015] | ||||
| * Improve the placement and style of the "No results" text [Jasper; #683135] | ||||
| * Remove broken network device activation policy [Giovanni; #683136] | ||||
| * Hide power status icon when no battery is present [Tim; #683080] | ||||
| * Ensure summary items are square and have spacing [Debarshi; #682248] | ||||
| * Fix close buttons overlapping screen edge [Debarshi; #682343] | ||||
| * Escape the tray when a legacy icon is clicked [Giovanni; #682244] | ||||
| * Update arrow in the screen shield to match latest mockups [Giovanni; #682285] | ||||
| * Allow lifting the screen shield with the mouse wheel [Giovanni; #683164] | ||||
| * Make sure to show the app menu after unlocking the screen [Jasper; #683154] | ||||
| * Misc bug fixes and cleanups [Debarshi, Florian, Giovanni, Jasper, Rui; | ||||
|   #582650, #667439, #682238, #682268, #682429, #682455, #682544, #682546, | ||||
|   #682683, #682710, #682998, #683073, #683137, #683156] | ||||
|  | ||||
| Contributors: | ||||
|   Alban Browaeys, Giovanni Campagna, Cosimo Cecchi, Stefano Facchini, | ||||
|   Adel Gadllah, Tim Lunn, Rui Matos, William Jon McCann, Florian Müllner, | ||||
|   Alejandro Piñeiro, Debarshi Ray, Jasper St. Pierre, Owen Taylor, Daiki Ueno | ||||
|  | ||||
| Translations: | ||||
|   Piotr Drąg [pl], Takayuki KUSANO [ja], Kjartan Maraas [nb], | ||||
|   Aurimas Černius [lt], Daniel Mustieles [es], Yuri Myasoedov [ru], | ||||
|   Khaled Hosny [ar], Yaron Shahrabani [he], Tom Tryfonidis [el], | ||||
|   Nilamdyuti Goswami [as], Fran Diéguez [gl], Nguyễn Thái Ngọc Duy [vi], | ||||
|   A S Alam [pa], Dr.T.Vasudevan [ta], Luca Ferretti [it] | ||||
|  | ||||
| 3.5.90 | ||||
| ====== | ||||
| * Use symbolic icons for workspace switch OSD [Jon; #680738] | ||||
| * Lock screen improvements: | ||||
|   - Hide user menu and a11y menu in the screen lock [Giovanni; #681143] | ||||
|   - Bump the lock screen slightly when pressing a key [Giovanni; #681143] | ||||
|   - Constrain vertical movement of the screen shield [Giovanni; #681143] | ||||
|   - Return to lock screen on idle [Giovanni; #682041] | ||||
|   - Unlock screen automatically after fast-user switching [Giovanni; #682096] | ||||
|   - Fix "other user" label [Ray; #681750] | ||||
| * Constrain content of system modals to primary monitor [#681743] | ||||
| * Respect automatic lock setting on suspend/user-switch [Giovanni; #680231] | ||||
| * Improve styling of keyring prompt [Jasper; #681821] | ||||
| * Do not hard-code <super> as overlay-key [Florian; #665547] | ||||
| * Update style of attached modal dialogs [Florian; #681601] | ||||
| * a11y: allow navigation on non reactive items [Alejandro; #667439, #667439] | ||||
| * Implement mode-less overview design [Joost, Florian; #682109] | ||||
| * Implement message-tray redesign: | ||||
|   - Restyle the message tray [Ana, Allan, Florian; #677213, #682342] | ||||
|   - Move the desktop upwards when showing the tray [Debarshi; #681392] | ||||
|   - Add a close button to notifications [Ana, Jasper; #682253] | ||||
|   - Add a keybinding to toggle the tray [Debarshi; #681392] | ||||
|   - Make the tray keyboard navigable [Debarshi; #681519] | ||||
|   - Add dwelling at the bottom of the screen to open the tray [Owen; #682310] | ||||
|   - Don't time out banners when the user is inactive [Marina, Jasper] | ||||
|   - Misc fixes and cleanups [Jasper, Marina] | ||||
| * Fix showing "Next Week" on Sundays [Sebastian; #682198] | ||||
| * Delay restoring IM presence until the network comes up [Florian; #677982] | ||||
| * Display enterprise login hint [Ray; #681975] | ||||
| * Ignore unrecognized/irrelevant network devices/connections [Dan; #682364] | ||||
| * Misc bug fixes and cleanups: [Dan, Florian, Jasper, Jiro, Piotr, Rico; | ||||
|   #643687, #682045, #682189] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Allan Day, Piotr Drąg, William Jon McCann, | ||||
|   Sebastian Keller, Jiro Matsuzawa, Florian Müllner, Alejandro Piñeiro, | ||||
|   Debarshi Ray, Ana Risteska, Jasper St. Pierre, Ray Strode, Owen Taylor, | ||||
|   Rico Tzschichholz, Joost Verdoorn, Dan Winship, Marina Zhurakhinskaya | ||||
|  | ||||
| Translations: | ||||
|   Nilamdyuti Goswami [as], Daniel Mustieles [es], Yaron Shahrabani [he], | ||||
|   Chao-Hsiung Liao [zh_HK, zh_TW], Tobias Endrigkeit [de], A S Alam [pa], | ||||
|   Sandeep Sheshrao Shedmake [mr], Fran Diéguez [gl], | ||||
|   Мирослав Николић [sr, sr@latin] | ||||
|  | ||||
| 3.5.5 | ||||
| ===== | ||||
| * Update style to match mockups [Allan] | ||||
|   - improve calendar layout and legibility | ||||
|   - update notifications and menus | ||||
|   - use a common style for entries | ||||
|   - update scrollbars to match GTK+ | ||||
|   - improve clock/unlock button in lock screen | ||||
|   - update polkit dialogs [Jasper] | ||||
| * Fix login dialog growing when selecting different users [Florian; #675076] | ||||
| * Implement screen lock in the shell [Giovanni] | ||||
|   - restructure login code to be shared with session unlock [#619955] | ||||
|   - add initial screen shield / unlock dialog implementation [#619955] | ||||
|   - implement (optional) notification list on lock shield [#619955] | ||||
|   - update login dialog style to match lock screen [#619955] | ||||
|   - filter notifications to only show new ones on the screen lock [#681143] | ||||
|   - make notifications scrollable if necessary [#681143] | ||||
|   - use correct application names in notifications [#681143] | ||||
|   - allow to return to the shield by pressing Escape [#681143] | ||||
| * Minor login dialog improvements [Florian] | ||||
|   - update style to match the overall visuals [#660913] | ||||
|   - indicate whether users are logged in [#658185] | ||||
| * Add support for background-repeat CSS property [Jasper; #680801] | ||||
| * Add :active pseudo class on scroll handles [Florian] | ||||
| * Remove markup from translated strings [Matthias; #681270] | ||||
| * Misc bug fixes and cleanups: [Alban, Florian, Giovanni, Jasper, Jeremy, | ||||
|   Matthias, Piotr; #677893, #679944, #680064, #680170, #680216, #680426, | ||||
|   #681101, #681382] | ||||
|  | ||||
| Contributors: | ||||
|   Jeremy Bicha, Alban Browaeys, Giovanni Campagna, Matthias Clasen, Allan Day, | ||||
|   Piotr Drąg , Florian Müllner, Jasper St. Pierre | ||||
|  | ||||
| Translations: | ||||
|   Matej Urbančič [sl], Tom Tryfonidis [el], Yaron Shahrabani [he], | ||||
|   Kjartan Maraas [nb], Baurzhan Muftakhidinov [kk], Praveen Illa [te], | ||||
|   Khaled Hosny [ar], Daniel Mustieles [es], Gabor Kelemen [hu], | ||||
|   Fran Diéguez [gl], Sweta Kothari [gu], Aleksej Kabanov [ru], | ||||
|   Nilamdyuti Goswami [as], Arash Mousavi [fa], Мирослав Николић [sr, sr@latin] | ||||
|  | ||||
| 3.5.4 | ||||
| ===== | ||||
| * Fix wrong result handling of remote calls [Florian; #678852] | ||||
| * dateMenu: Fix regression that caused no date to be displayed [Colin] | ||||
| * WindowTracker: Fix refcounting bug in get_app_for_window() [Giovanni; #678992] | ||||
| * Show the workspace switcher for move-to-workspace keybinding | ||||
|   [Giovanni, Jasper; #674104, #660839, #679005] | ||||
| * userMenu: Move "Power off" item to the bottom [Florian; #678887] | ||||
| * Remove contacts search provider [Florian, Rui; #677442] | ||||
| * network: don't ask for always-ask secrets when interaction isn't allowed | ||||
|   [Dan; #679091] | ||||
| * PolkitAgent: Look for the right password prompt [Matthias; #675300] | ||||
| * Implement extension updates [Jasper; #679099] | ||||
| * userMenu: Don't disconnect account signals when disabled [Guillaume; #669112] | ||||
| * Fix startup notification when opening calendar [Florian; #677907] | ||||
| * networkAgent: use absolute path if configured [Clemens; #679212] | ||||
| * recorder: Port to GStreamer-1.0 API [Florian; #679445] | ||||
| * telepathyClient: don't add log messages on presence changes [Ana; #669508] | ||||
| * lookingGlass: Don't use a signal callback on 'paint' to draw the border | ||||
|   [Jasper; #679464] | ||||
| * Add support for inhibiting automount [Hans; #678597] | ||||
| * Implemented banner support for the login screen [Matthias, Marius; #665346] | ||||
| * boxpointer: Flip side if we would end outside the monitor [Rui; #678164] | ||||
| * boxpointer: Change 'animate' parameter on show/hide to a bitmask | ||||
|   [Rui; #678337] | ||||
| * Add a grayscale effect [Matthias, Jasper, Florian: #676782, #674499] | ||||
| * UserMenu: show "Install Updates & Restart" when appropriate | ||||
|   [Giovanni; #677394, #680080] | ||||
| * messageTray: don't show the message tray when a new notification is shown | ||||
|   [Ana; #677210] | ||||
| * panel: don't break when indicator has no menu [Jean-Philippe; #678694] | ||||
| * appMenu: Disable app menu during startup animations [Florian; #672322] | ||||
| * autorun: Add a notification when unmounting drives [Cosimo; #676125] | ||||
| * st-icon: Fix potential crash involving shadows [Jasper; #679776] | ||||
| * Remove manual garbage collection on tweeners end [Cosimo; #679832] | ||||
| * dash: hide tooltips when overview begins hiding [Stefano; #674241] | ||||
| * Update modal dialog animation for new centered position [Florian; #674499] | ||||
| * calendar: Fix grid lines in RTL locales [Florian; #679879] | ||||
| * Integrate IBus with keyboard indicator [Rui; #641531] | ||||
| * Move ibus status icon under keyboard [Matthias] | ||||
| * gdm: port from libgdmgreeter to libgdm [Ray; #676401] | ||||
| * Misc bug fixes and cleanups [Antoine, Cosimo, Giovanni, Jasper, Rico; | ||||
|   #678978, #672790, #679847, #679944] | ||||
|  | ||||
| Contributors: | ||||
|  Jean-Philippe Braun, Clemens Buchacher, Giovanni Campagna, Cosimo Cecchi, | ||||
|  Matthias Clasen, Hans de Goede, Guillaume Desmottes, Stefano Facchini, | ||||
|  Antoine Jacoutot, Rui Matos, Florian Müllner, Marius Rieder, Ana Risteska, | ||||
|  Jasper St. Pierre, Rico Tzschichholz, Colin Walters, Dan Williams | ||||
|  | ||||
| Translations: | ||||
|  Matej Urbančič [sl], Khaled Hosny [ar], Nguyễn Thái Ngọc Duy [vi], | ||||
|  Nilamdyuti Goswami [as], Alexander Shopov [bg], Ivaylo Valkov [bg], | ||||
|  Daniel Mustieles [es], Kjartan Maraas [nb,nn], Yaron Shahrabani [he], | ||||
|  Nilamdyuti Goswami [as], Chao-Hsiung Liao [zh_HK, zh_TW], Ihar Hrachyshka [be], | ||||
|  Praveen Illa [te] | ||||
|  | ||||
| 3.5.3 | ||||
| ===== | ||||
| * calendar: Adapt to Evolution-Data-Server API changes [Matthew; #677402] | ||||
| * messageTray: Don't show non urgent notifications while in fullscreen | ||||
|   [Adel; #677590] | ||||
| * modalDialog: show dialogs on monitor with the mouse pointer [Tim; #642591] | ||||
| * extensionSystem: Prepare for extension updating system [Jasper; #677586] | ||||
| * appDisplay: Don't show apps in NoDisplay categories in the All view | ||||
|   [Jasper; #658176] | ||||
| * st: Trigger theme updates on resolution changes [Florian; #677975] | ||||
| * Always enable a11y [Bastien; #678095] | ||||
| * telepathyClient: ignore invalidated channels [Guillaume; #677457] | ||||
| * shell-app: Update app menu if necessary [Florian; #676238] | ||||
| * Enable the Screen Reader menu item [Matthias; #663256] | ||||
| * Disable unredirection when a modal operation is active [Giovanni] | ||||
| * Make folks optional [Colin] | ||||
| * Improve mount-operation support [Cosimo] | ||||
|   - Fix exception when showing password entry [#678428] | ||||
|   - Close the password entry on operation abort [#673787] | ||||
|   - autorun: Don't allow autorun for things we mount on startup [#660595] | ||||
|   - Turn passphrase prompt into a dialog [#674962] | ||||
|   - Implement org.Gtk.MountOperationHandler [#678516] | ||||
| * Network menu improvements | ||||
|   - Sort Wifi networks by strength [Giovanni; #658946] | ||||
|   - Prefer wifi/3g over VPN in the panel [Cosimo; #672591] | ||||
| * clock: Switch to using GnomeWallClock [Colin; #657074] | ||||
| * remoteSearch: Allow to reference .desktop file for Title/Icon | ||||
|   [Florian; #678816] | ||||
| * Fix memory leaks [Jasper, Pavel; #678079, #678406, #678737] | ||||
| * Misc fixes [Florian, Giovanni, Guillaume, Jasper, Kjartan, Piotr, Rui; | ||||
|   #658955, #677497, #678396, #678502] | ||||
| * Misc cleanups [Bastien, Florian, Jasper; #677426, #677515, #678096, #678416] | ||||
|  | ||||
| Contributors: | ||||
|  Matthew Barnes, Giovanni Campagna, Cosimo Cecchi, Matthias Clasen, | ||||
|  Guillaume Desmottes, Piotr Drąg, Adel Gadllah, Tim L, Kjartan Maraas, | ||||
|  Rui Matos, Florian Müllner, Bastien Nocera, Jasper St. Pierre, Colin Walters | ||||
|  | ||||
| Translations: | ||||
|  Matej Urbančič [sl], Yuri Kozlov [ru], Tom Tryfonidis [el], | ||||
|  Kjartan Maraas [nb], Žygimantas Beručka [lt], Luca Ferretti [it], | ||||
|  Khaled Hosny [ar], Daniel Mustieles [es], Fran Diéguez [gl], A S Alam [pa] | ||||
|  | ||||
| 3.5.2 | ||||
| ===== | ||||
| * main: Move 'toggle-recording' binding into the shell [Florian; #674377] | ||||
| * popupMenu: make sure to break the grab when the slider is not visible | ||||
|   [Stefano; #672713] | ||||
| * st-theme-node-drawing: Don't use GL types [Neil; #672711] | ||||
| * Mirror Evolution calendar settings into our own schema [Owen; #674424] | ||||
| * shell-network-agent: don't crash if a request isn't found [Dan; #674961] | ||||
| * notificationDaemon: Match app based on WM_CLASS [Jasper; #673761] | ||||
| * NetworkMenu: use network-offline while loading [Giovanni; #674426] | ||||
| * lookingGlass: Remove the Errors tab [Jasper; #675104] | ||||
| * searchDisplay: Reset keyboard focus after displaying async results | ||||
|   [Rui; #675078] | ||||
| * gdm: don't fail if fprintd is unavailable [Ray; #675006] | ||||
| * messageTray: Fix scrolling up [Jasper; #661615] | ||||
| * main: Close the recorder instead of pausing it [Rui; #675128] | ||||
| * Accessibility [Alejandro] | ||||
|   - Use the proper label_actor for date menu on top panel [#675307] | ||||
|   - Set the proper role/label_actor for SearchResult.content [#672242] | ||||
|   - do not expose a label text if 'hidden' style class is used [#675341] | ||||
| * Magnifier: Add brightness and contrast functionality [Joseph; #639851] | ||||
| * theme: use a smaller border-radius for top bar [Jakub; #672430] | ||||
| * placeDisplay: use new bookmark file location [Matthias; #675443] | ||||
| * port all synchronous search providers to the async API [Jasper, Rui; #675328] | ||||
| * NetworkAgent: disallow multiple requests for the same connection/setting | ||||
|   [Giovanni; #674961] | ||||
| * userMenu: Update to latest mockups [Florian; #675802] | ||||
| * util: Don't double-fork when spawning from Alt-F2 [Colin; #675789] | ||||
| * messageTray: Make Source usable without subclassing [Jasper; #661236] | ||||
| * panel: Check for appMenu button's reactivity before opening [Florian; #676316] | ||||
| * Fix formatting of bluetooth passkey [Florian; #651251] | ||||
| * notificationDaemon: Filter out file-transfer notifications [Jasper; #676175] | ||||
| * Don't use global.log() [Jasper; #675790] | ||||
| * Fix broken extension loading in some distributions [Owen, Alexandre; #670477] | ||||
| * shell-app: Raise windows in reverse order to preserve the stacking | ||||
|   [Rui; #676371] | ||||
| * Generalize gdm-mode [Florian; #676156] | ||||
| * Switch string formatting to the one inside gjs [Jasper; #675479] | ||||
| * extensionUtils: Support subdirectories in getCurrentExtension | ||||
|   [Jasper; #677001] | ||||
| * panel: Refuse to add (legacy) status icons not part of the session mode | ||||
|   [Florian; #677058] | ||||
| * Add an initial-setup mode [Matthias; #676697] | ||||
| * status/keyboard: Port to the new input sources settings [Rui; #641531] | ||||
| * NetworkMenu: show notifications for failed VPN connections [Giovanni; #676330] | ||||
| * userMenu: Indicate progress on status changes [Florian; #659067] | ||||
| * recorder: Honor "disable-save-to-disk" lockdown key [Rūdolfs; #673630] | ||||
| * searchDisplay: Use the rowLimit we pass to the IconGrid [Christian; #675527] | ||||
| * endSessionDialog: Factor out _updateDescription from _updateContent | ||||
|   [Alejandro; #674210] | ||||
| * Fix empathy's appMenu freezing the shell [Alban; #676447] | ||||
| * Code cleanups [Florian, Giovanni, Jasper; #672807, #672413, #676837, #676850, | ||||
|   #672272] | ||||
| * Misc bug fixes [Alban, Florian, Giovanni, Guillaume, Jasper, Piotr, Rico, | ||||
|   Ron, Rui, Stefano; #659968, #672192, #673177, #673198, #674323, #675301, | ||||
|   #675370, #676347, #676806, #677097] | ||||
|  | ||||
| Contributors: | ||||
|  Alban Browaeys, Giovanni Campagna, Matthias Clasen, Guillaume Desmottes, | ||||
|  Piotr Drąg, Stefano Facchini, Rui Matos, Rūdolfs Mazurs, Florian Müllner, | ||||
|  Alejandro Piñeiro, Neil Roberts, Alexandre Rostovtsev, Joseph Scheuhammer, | ||||
|  Jakub Steiner, Jasper St. Pierre, Ray Strode, Owen Taylor, Rico Tzschichholz, | ||||
|  Colin Walters, Dan Winship, Ron Yorston | ||||
|  | ||||
| Translations: | ||||
|  OKANO Takayoshi [ja], Daniel Mustieles [es], Changwoo Ryu [ko], | ||||
|  Yaron Shahrabani [he], Fran Diéguez [gl], Jonh Wendell [pt_BR], | ||||
|  Kjartan Maraas [nb], Luca Ferretti [it], Tom Tryfonidis [el], | ||||
|  Sandeep Sheshrao Shedmake [mr], Takanori MATSUURA [ja], Dirgita [id], | ||||
|  Mantas Kriaučiūnas [lt], Matej Urbančič [sl], Jiro Matsuzawa [ja] | ||||
|  | ||||
| 3.4.1 | ||||
| ===== | ||||
| * Fix crash that occurred when an icon theme change caused unexpected | ||||
|   reentrancy in the icon loading code [Jasper; #673512] | ||||
| * Don't show system and other disabled users in the GDM user list | ||||
|   [Adel; #673784] | ||||
| * Make gnome-shell-calendar-server initialize GTK+ so it can display | ||||
|   password prompts if needed [#673608; Owen, Rico] | ||||
| * Adapt to Mutter API change for keybinding addition [Florian; #673014] | ||||
| * Fix crash when an extension was installed as both a user extension | ||||
|   and a system extension [#673613; Jasper] | ||||
| * Fix bug where chat entry could end up partially offscreen [Joost, 661944] | ||||
| * Fix problem where icons weren't updating when theme was changed | ||||
|   [#672941; Florian] | ||||
| * Look for Evolution calendar settings in GSettings, not GConf [#673610; Owen] | ||||
| * Add <super>F10 for the application menu [#672909; Florian] | ||||
| * Fix %Id format characters to work in translations [#673106; Cosimo] | ||||
|   (were already used in fa translation) | ||||
| * Fix error when NetworkManager restarts [#673043; Giovanni] | ||||
| * Improve efficiency of overview redraws by working around Clutter issue | ||||
|   [Stefano; #670636] | ||||
| * Misc bug fixes [Florian, Giovanni, Jasper, Rui, Stefano; | ||||
|   #672592, #672641, #672719, #673187, #673233, #673656] | ||||
|  | ||||
| Contributors: | ||||
|  Giovanni Campagna, Cosimo Cecchi, Stefano Facchini, Adel Gadllah, Rui Matos, | ||||
|  Florian Müllner, Jasper St. Pierre, Owen Taylor, Rico Tzschichholz, | ||||
|  Joost Verdoorn | ||||
|  | ||||
| Translations: | ||||
|  Khaled Hosny [ar], Ihar Hrachyshka [be], Alexander Shopov [bg], Gil Forcada, | ||||
|  Jordi Serratosa [ca], Petr Kovar [cs], Bruce Cowan [en_GB], | ||||
|  Carles Ferrando [ca@valencia], Wolfgang Stöggl [de], Daniel Mustieles [es], | ||||
|  Arash Mousavi [fa], Bruno Brouard [fr], Fran Diéguez [gl], | ||||
|  Sweta Kothari [gu], Yaron Shahrabani [he], Gabor Kelemen [hu], | ||||
|  Shankar Prasad [kn], Žygimantas Beručka [lt], Rudolfs Mazurs [lv], | ||||
|  Sandeep Sheshrao Shedmake [mr], Kjartan Maraas [nb], Piotr Drąg [pl], | ||||
|  Yuri Myasoedov [ru], Daniel Nylander [se], Matej Urbančič [sl], | ||||
|  Miroslav Nikolić [sr], Sasi Bhushan, Praveen Illa [te], Yinghua Wang [zh_CN] | ||||
|  | ||||
| 3.4.0 | ||||
| ===== | ||||
| * Don't crash when taking screenshots [Jasper; #672775] | ||||
| @@ -815,10 +8,10 @@ Contributors: | ||||
|  | ||||
| Translations: | ||||
|  Khaled Hosny, Abderrahim Kitouni [ar], Ihar Hrachyshka [be], | ||||
|  Alexander Shopov [bg], Marek Černocký [cs], Jiri Grönroos, Timo Jyrinki [fi], | ||||
|  Alexander Shopov [bg], Marek Černocký [cz], Jiri Grönroos, Timo Jyrinki [fi], | ||||
|  Bruno Brouard [fr], Fran Diéguez [gl], Yaron Shahrabani [he], | ||||
|  Gabor Kelemen [hu], Jiro Matsuzawa [ja], Kenneth Nielsen [dk], | ||||
|  Mattias Põldaru [et], Changwoo Ryu [ko], Rudolfs Mazurs [lv], | ||||
|  Mattias Põldaru [et], Changwoo Ryu [kr], Rudolfs Mazurs [lv], | ||||
|  Jonh Wendell [pt_BR], Yuri Myasoedov[ru], Daniel Korostil [uk], | ||||
|  Nguyễn Thái Ngọc Duy [vi], Chao-Hsiung Liao [zh_HK, zh_TW] | ||||
|  | ||||
| @@ -887,7 +80,7 @@ Contributors: | ||||
|  | ||||
| Translations: | ||||
|  Nilamdyuti Goswami [as], Ihar Hrachyshka, Kasia Bondarava [be], | ||||
|  Alexander Shopov, Ivaylo Valkov [bg], Gil Forcada [ca], Marek Černocký [cs], | ||||
|  Alexander Shopov, Ivaylo Valkov [bg], Gil Forcada [ca], Marek Černocký [cz], | ||||
|  Mario Blättermann [de], Kris Thomsen [dk], Bruce Cowan [en_GB], | ||||
|  Kristjan Schmidt [eo], Daniel Mustieles [es], Mattias Põldaru [et], | ||||
|  Inaki Larranaga Murgoitio [eu], Arash Mousavi [fa], Timo Jyrinki [fi], | ||||
| @@ -936,7 +129,7 @@ Contributors: | ||||
|  Will Thompson, Stef Walter | ||||
|  | ||||
| Translations: | ||||
|  Ihar Hrachyshka [be], Marek Černocký, Adam Matoušek [cs], | ||||
|  Ihar Hrachyshka [be], Marek Černocký, Adam Matoušek [cz], | ||||
|  Kenneth Nielsen [dk], Daniel Mustieles [es], Mattias Põldaru [et], | ||||
|  Fran Diéguez [gl], Yaron Shahrabani [he], Luca Ferretti [it], | ||||
|  Baurzhan Muftakhidinov [kk], Aurimas Černius [lt], Kjartan Maraas [nb], | ||||
| @@ -1067,7 +260,7 @@ Contributors: | ||||
|  Marina Zhurakhinskaya | ||||
|  | ||||
| Translations: | ||||
|  Petr Kovar [cs], Kris Thomsen [dk], Daniel Mustieles [es], | ||||
|  Petr Kovar [cz], Kris Thomsen [dk], Daniel Mustieles [es], | ||||
|  Ville-Pekka Vainio [fi], Yaron Shahrabani [he], Luca Ferretti [it], | ||||
|  Hideki Yamane [ja], Žygimantas Beručka [lt], Jovan Naumovski [mk], | ||||
|  Kjartan Maraas [nb], "Andreas N" [nn], Lucian Adrian Grijincu [ro], | ||||
| @@ -1230,7 +423,7 @@ Contributors: | ||||
| Translations: | ||||
|  Friedel Wolff [af], Nilamdyuti Goswami [as], Ihar Hrachyshka [be], | ||||
|  Ivaylo Valkov [bg], Gil Forcada [ca], Carles Ferrando [ca@valencia], | ||||
|  Petr Kovar [cs], Mario Blättermann [de], Kris Thomsen [dk], | ||||
|  Petr Kovar [cz], Mario Blättermann [de], Kris Thomsen [dk], | ||||
|  Tiffany Antopolski, Kristjan Schmidt [eo], Daniel Mustieles [es], | ||||
|  Inaki Larranaga Murgoitio [eu], Tommi Vainikainen [fi], Bruno Brouard [fr], | ||||
|  Fran Dieguez [gl], Yaron Shahrabani [he], Gabor Kelemen [hu], | ||||
|   | ||||
| @@ -13,14 +13,6 @@ PKG_NAME="gnome-shell" | ||||
|     exit 1 | ||||
| } | ||||
|  | ||||
| # Fetch submodules if needed | ||||
| if test ! -f src/gvc/Makefile.am; | ||||
| then | ||||
|   echo "+ Setting up submodules" | ||||
|   git submodule init | ||||
| fi | ||||
| git submodule update | ||||
|  | ||||
| which gnome-autogen.sh || { | ||||
|     echo "You need to install gnome-common from GNOME Git (or from" | ||||
|     echo "your OS vendor's package manager)." | ||||
|   | ||||
| @@ -41,7 +41,7 @@ | ||||
|       "It can be used only by extensions.gnome.org" | ||||
| #define PLUGIN_MIME_STRING "application/x-gnome-shell-integration::Gnome Shell Integration Dummy Content-Type"; | ||||
|  | ||||
| #define PLUGIN_API_VERSION 5 | ||||
| #define PLUGIN_API_VERSION 3 | ||||
|  | ||||
| typedef struct { | ||||
|   GDBusProxy *proxy; | ||||
| @@ -153,6 +153,8 @@ NP_Initialize(NPNetscapeFuncs *pfuncs, NPPluginFuncs *plugin) | ||||
|   /* global initialization routine, called once when plugin | ||||
|      is loaded */ | ||||
|  | ||||
|   g_type_init (); | ||||
|  | ||||
|   g_debug ("plugin loaded"); | ||||
|  | ||||
|   memcpy (&funcs, pfuncs, sizeof (funcs)); | ||||
| @@ -161,7 +163,6 @@ NP_Initialize(NPNetscapeFuncs *pfuncs, NPPluginFuncs *plugin) | ||||
|   plugin->newp = NPP_New; | ||||
|   plugin->destroy = NPP_Destroy; | ||||
|   plugin->getvalue = NPP_GetValue; | ||||
|   plugin->setwindow = NPP_SetWindow; | ||||
|  | ||||
|   return NPERR_NO_ERROR; | ||||
| } | ||||
| @@ -223,7 +224,7 @@ NPP_New(NPMIMEType    mimetype, | ||||
|                                                NULL, /* interface info */ | ||||
|                                                "org.gnome.Shell", | ||||
|                                                "/org/gnome/Shell", | ||||
|                                                "org.gnome.Shell.Extensions", | ||||
|                                                "org.gnome.Shell", | ||||
|                                                NULL, /* GCancellable */ | ||||
|                                                &error); | ||||
|   if (!data->proxy) | ||||
| @@ -266,7 +267,6 @@ typedef struct { | ||||
|   NPObject     parent; | ||||
|   NPP          instance; | ||||
|   GDBusProxy  *proxy; | ||||
|   GSettings   *settings; | ||||
|   NPObject    *listener; | ||||
|   NPObject    *restart_listener; | ||||
|   gint         signal_id; | ||||
| @@ -323,9 +323,6 @@ on_shell_appeared (GDBusConnection *connection, | ||||
|     } | ||||
| } | ||||
|  | ||||
| #define SHELL_SCHEMA "org.gnome.shell" | ||||
| #define ENABLED_EXTENSIONS_KEY "enabled-extensions" | ||||
|  | ||||
| static NPObject * | ||||
| plugin_object_allocate (NPP      instance, | ||||
|                         NPClass *klass) | ||||
| @@ -335,7 +332,6 @@ plugin_object_allocate (NPP      instance, | ||||
|  | ||||
|   obj->instance = instance; | ||||
|   obj->proxy = g_object_ref (data->proxy); | ||||
|   obj->settings = g_settings_new (SHELL_SCHEMA); | ||||
|   obj->signal_id = g_signal_connect (obj->proxy, "g-signal", | ||||
|                                      G_CALLBACK (on_shell_signal), obj); | ||||
|  | ||||
| @@ -371,14 +367,39 @@ plugin_object_deallocate (NPObject *npobj) | ||||
|   g_slice_free (PluginObject, obj); | ||||
| } | ||||
|  | ||||
| static NPIdentifier api_version_id; | ||||
| static NPIdentifier shell_version_id; | ||||
| static NPIdentifier get_info_id; | ||||
| static NPIdentifier list_extensions_id; | ||||
| static NPIdentifier enable_extension_id; | ||||
| static NPIdentifier install_extension_id; | ||||
| static NPIdentifier uninstall_extension_id; | ||||
| static NPIdentifier onextension_changed_id; | ||||
| static NPIdentifier onrestart_id; | ||||
| static NPIdentifier get_errors_id; | ||||
| static NPIdentifier launch_extension_prefs_id; | ||||
|  | ||||
| static bool | ||||
| plugin_object_has_method (NPObject     *npobj, | ||||
|                           NPIdentifier  name) | ||||
| { | ||||
|   return (name == get_info_id || | ||||
|           name == list_extensions_id || | ||||
|           name == enable_extension_id || | ||||
|           name == install_extension_id || | ||||
|           name == uninstall_extension_id || | ||||
|           name == get_errors_id || | ||||
|           name == launch_extension_prefs_id); | ||||
| } | ||||
|  | ||||
| static inline gboolean | ||||
| uuid_is_valid (NPString string) | ||||
| uuid_is_valid (const gchar *uuid) | ||||
| { | ||||
|   gsize i; | ||||
|  | ||||
|   for (i = 0; i < string.UTF8Length; i++) | ||||
|   for (i = 0; uuid[i]; i ++) | ||||
|     { | ||||
|       gchar c = string.UTF8Characters[i]; | ||||
|       gchar c = uuid[i]; | ||||
|       if (c < 32 || c >= 127) | ||||
|         return FALSE; | ||||
|  | ||||
| @@ -442,73 +463,8 @@ jsonify_variant (GVariant  *variant, | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| parse_args (const gchar     *format_str, | ||||
|             uint32_t         argc, | ||||
|             const NPVariant *argv, | ||||
|             ...) | ||||
| { | ||||
|   va_list args; | ||||
|   gsize i; | ||||
|   gboolean ret = FALSE; | ||||
|  | ||||
|   if (strlen (format_str) != argc) | ||||
|     return FALSE; | ||||
|  | ||||
|   va_start (args, argv); | ||||
|  | ||||
|   for (i = 0; format_str[i]; i++) | ||||
|     { | ||||
|       gpointer arg_location; | ||||
|       const NPVariant arg = argv[i]; | ||||
|  | ||||
|       arg_location = va_arg (args, gpointer); | ||||
|  | ||||
|       switch (format_str[i]) | ||||
|         { | ||||
|         case 'u': | ||||
|           { | ||||
|             NPString string; | ||||
|  | ||||
|             if (!NPVARIANT_IS_STRING (arg)) | ||||
|               goto out; | ||||
|  | ||||
|             string = NPVARIANT_TO_STRING (arg); | ||||
|  | ||||
|             if (!uuid_is_valid (string)) | ||||
|               goto out; | ||||
|  | ||||
|             *(gchar **) arg_location = g_strndup (string.UTF8Characters, string.UTF8Length); | ||||
|           } | ||||
|           break; | ||||
|  | ||||
|         case 'b': | ||||
|           if (!NPVARIANT_IS_BOOLEAN (arg)) | ||||
|             goto out; | ||||
|  | ||||
|           *(gboolean *) arg_location = NPVARIANT_TO_BOOLEAN (arg); | ||||
|           break; | ||||
|  | ||||
|         case 'o': | ||||
|           if (!NPVARIANT_IS_OBJECT (arg)) | ||||
|             goto out; | ||||
|  | ||||
|           *(NPObject **) arg_location = NPVARIANT_TO_OBJECT (arg); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   ret = TRUE; | ||||
|  | ||||
|  out: | ||||
|   va_end (args); | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_list_extensions (PluginObject    *obj, | ||||
|                         uint32_t         argc, | ||||
|                         const NPVariant *args, | ||||
|                         NPVariant       *result) | ||||
| plugin_list_extensions (PluginObject  *obj, | ||||
|                         NPVariant     *result) | ||||
| { | ||||
|   GError *error = NULL; | ||||
|   GVariant *res; | ||||
| @@ -532,159 +488,91 @@ plugin_list_extensions (PluginObject    *obj, | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_enable_extension (PluginObject    *obj, | ||||
|                          uint32_t         argc, | ||||
|                          const NPVariant *argv, | ||||
|                          NPVariant       *result) | ||||
| plugin_enable_extension (PluginObject *obj, | ||||
|                          NPString      uuid, | ||||
|                          gboolean      enabled) | ||||
| { | ||||
|   gboolean ret; | ||||
|   gchar *uuid; | ||||
|   gboolean enabled; | ||||
|   gsize length; | ||||
|   gchar **uuids; | ||||
|   const gchar **new_uuids; | ||||
|  | ||||
|   if (!parse_args ("ub", argc, argv, &uuid, &enabled)) | ||||
|     return FALSE; | ||||
|  | ||||
|   uuids = g_settings_get_strv (obj->settings, ENABLED_EXTENSIONS_KEY); | ||||
|   length = g_strv_length (uuids); | ||||
|  | ||||
|   if (enabled) | ||||
|   gchar *uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       new_uuids = g_new (const gchar *, length + 2); /* New key, NULL */ | ||||
|       memcpy (new_uuids, uuids, length * sizeof (*new_uuids)); | ||||
|       new_uuids[length] = uuid; | ||||
|       new_uuids[length + 1] = NULL; | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       gsize i = 0, j = 0; | ||||
|       new_uuids = g_new (const gchar *, length); | ||||
|       for (i = 0; i < length; i ++) | ||||
|         { | ||||
|           if (g_str_equal (uuids[i], uuid)) | ||||
|             continue; | ||||
|  | ||||
|           new_uuids[j] = uuids[i]; | ||||
|           j++; | ||||
|         } | ||||
|  | ||||
|       new_uuids[j] = NULL; | ||||
|     } | ||||
|  | ||||
|   ret = g_settings_set_strv (obj->settings, | ||||
|                              ENABLED_EXTENSIONS_KEY, | ||||
|                              new_uuids); | ||||
|  | ||||
|   g_strfreev (uuids); | ||||
|   g_free (new_uuids); | ||||
|   g_free (uuid); | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| typedef struct _AsyncClosure AsyncClosure; | ||||
|  | ||||
| struct _AsyncClosure { | ||||
|   PluginObject *obj; | ||||
|   NPObject *callback; | ||||
|   NPObject *errback; | ||||
| }; | ||||
|  | ||||
| static void | ||||
| install_extension_cb (GObject      *proxy, | ||||
|                       GAsyncResult *async_res, | ||||
|                       gpointer      user_data) | ||||
| { | ||||
|   AsyncClosure *async_closure = (AsyncClosure *) user_data; | ||||
|   GError *error = NULL; | ||||
|   GVariant *res; | ||||
|   NPVariant args[1]; | ||||
|   NPVariant result = { NPVariantType_Void }; | ||||
|   NPObject *callback; | ||||
|  | ||||
|   res = g_dbus_proxy_call_finish (G_DBUS_PROXY (proxy), async_res, &error); | ||||
|  | ||||
|   if (res == NULL) | ||||
|     { | ||||
|       if (g_dbus_error_is_remote_error (error)) | ||||
|         g_dbus_error_strip_remote_error (error); | ||||
|       STRINGZ_TO_NPVARIANT (error->message, args[0]); | ||||
|       callback = async_closure->errback; | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       char *string_result; | ||||
|       g_variant_get (res, "(&s)", &string_result); | ||||
|       STRINGZ_TO_NPVARIANT (string_result, args[0]); | ||||
|       callback = async_closure->callback; | ||||
|     } | ||||
|  | ||||
|   funcs.invokeDefault (async_closure->obj->instance, | ||||
|                        callback, args, 1, &result); | ||||
|  | ||||
|   funcs.releasevariantvalue (&result); | ||||
|  | ||||
|   funcs.releaseobject (async_closure->callback); | ||||
|   funcs.releaseobject (async_closure->errback); | ||||
|   g_slice_free (AsyncClosure, async_closure); | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_install_extension (PluginObject    *obj, | ||||
|                           uint32_t         argc, | ||||
|                           const NPVariant *argv, | ||||
|                           NPVariant       *result) | ||||
| { | ||||
|   gchar *uuid; | ||||
|   NPObject *callback, *errback; | ||||
|   AsyncClosure *async_closure; | ||||
|  | ||||
|   if (!parse_args ("uoo", argc, argv, &uuid, &callback, &errback)) | ||||
|     return FALSE; | ||||
|  | ||||
|   async_closure = g_slice_new (AsyncClosure); | ||||
|   async_closure->obj = obj; | ||||
|   async_closure->callback = funcs.retainobject (callback); | ||||
|   async_closure->errback = funcs.retainobject (errback); | ||||
|  | ||||
|   g_dbus_proxy_call (obj->proxy, | ||||
|                      "InstallRemoteExtension", | ||||
|                      g_variant_new ("(s)", uuid), | ||||
|                      (enabled ? "EnableExtension" : "DisableExtension"), | ||||
|                      g_variant_new ("(s)", uuid_str), | ||||
|                      G_DBUS_CALL_FLAGS_NONE, | ||||
|                      -1, /* timeout */ | ||||
|                      NULL, /* cancellable */ | ||||
|                      install_extension_cb, | ||||
|                      async_closure); | ||||
|                      NULL, /* callback */ | ||||
|                      NULL /* user_data */); | ||||
|  | ||||
|   g_free (uuid); | ||||
|   g_free (uuid_str); | ||||
|  | ||||
|   return TRUE; | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_uninstall_extension (PluginObject    *obj, | ||||
|                             uint32_t         argc, | ||||
|                             const NPVariant *argv, | ||||
|                             NPVariant       *result) | ||||
| plugin_install_extension (PluginObject *obj, | ||||
|                           NPString      uuid, | ||||
|                           NPString      version_tag) | ||||
| { | ||||
|   gchar *uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   gchar *version_tag_str; | ||||
|  | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|  | ||||
|   version_tag_str = g_strndup (version_tag.UTF8Characters, | ||||
|                                version_tag.UTF8Length); | ||||
|  | ||||
|   g_dbus_proxy_call (obj->proxy, | ||||
|                      "InstallRemoteExtension", | ||||
|                      g_variant_new ("(ss)", | ||||
|                                     uuid_str, | ||||
|                                     version_tag_str), | ||||
|                      G_DBUS_CALL_FLAGS_NONE, | ||||
|                      -1, /* timeout */ | ||||
|                      NULL, /* cancellable */ | ||||
|                      NULL, /* callback */ | ||||
|                      NULL /* user_data */); | ||||
|  | ||||
|   g_free (uuid_str); | ||||
|   g_free (version_tag_str); | ||||
|  | ||||
|   return TRUE; | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_uninstall_extension (PluginObject *obj, | ||||
|                             NPString      uuid, | ||||
|                             NPVariant    *result) | ||||
| { | ||||
|   GError *error = NULL; | ||||
|   GVariant *res; | ||||
|   gchar *uuid; | ||||
|   gchar *uuid_str; | ||||
|  | ||||
|   if (!parse_args ("u", argc, argv, &uuid)) | ||||
|     return FALSE; | ||||
|   uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|  | ||||
|   res = g_dbus_proxy_call_sync (obj->proxy, | ||||
|                                 "UninstallExtension", | ||||
|                                 g_variant_new ("(s)", uuid), | ||||
|                                 g_variant_new ("(s)", | ||||
|                                                uuid_str), | ||||
|                                 G_DBUS_CALL_FLAGS_NONE, | ||||
|                                 -1, /* timeout */ | ||||
|                                 NULL, /* cancellable */ | ||||
|                                 &error); | ||||
|  | ||||
|   g_free (uuid); | ||||
|   g_free (uuid_str); | ||||
|  | ||||
|   if (!res) | ||||
|     { | ||||
| @@ -697,27 +585,30 @@ plugin_uninstall_extension (PluginObject    *obj, | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_get_info (PluginObject    *obj, | ||||
|                  uint32_t         argc, | ||||
|                  const NPVariant *argv, | ||||
|                  NPVariant       *result) | ||||
| plugin_get_info (PluginObject *obj, | ||||
|                  NPString      uuid, | ||||
|                  NPVariant    *result) | ||||
| { | ||||
|   GError *error = NULL; | ||||
|   GVariant *res; | ||||
|   gchar *uuid; | ||||
|   gchar *uuid_str; | ||||
|  | ||||
|   if (!parse_args ("u", argc, argv, &uuid)) | ||||
|     return FALSE; | ||||
|   uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|  | ||||
|   res = g_dbus_proxy_call_sync (obj->proxy, | ||||
|                                 "GetExtensionInfo", | ||||
|                                 g_variant_new ("(s)", uuid), | ||||
|                                 g_variant_new ("(s)", uuid_str), | ||||
|                                 G_DBUS_CALL_FLAGS_NONE, | ||||
|                                 -1, /* timeout */ | ||||
|                                 NULL, /* cancellable */ | ||||
|                                 &error); | ||||
|  | ||||
|   g_free (uuid); | ||||
|   g_free (uuid_str); | ||||
|  | ||||
|   if (!res) | ||||
|     { | ||||
| @@ -730,26 +621,31 @@ plugin_get_info (PluginObject    *obj, | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_get_errors (PluginObject    *obj, | ||||
|                    uint32_t         argc, | ||||
|                    const NPVariant *argv, | ||||
|                    NPVariant       *result) | ||||
| plugin_get_errors (PluginObject *obj, | ||||
|                    NPString      uuid, | ||||
|                    NPVariant    *result) | ||||
| { | ||||
|   GError *error = NULL; | ||||
|   GVariant *res; | ||||
|   gchar *uuid; | ||||
|   gchar *uuid_str; | ||||
|  | ||||
|   if (!parse_args ("u", argc, argv, &uuid)) | ||||
|     return FALSE; | ||||
|   uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|  | ||||
|   res = g_dbus_proxy_call_sync (obj->proxy, | ||||
|                                 "GetExtensionErrors", | ||||
|                                 g_variant_new ("(s)", uuid), | ||||
|                                 g_variant_new ("(s)", uuid_str), | ||||
|                                 G_DBUS_CALL_FLAGS_NONE, | ||||
|                                 -1, /* timeout */ | ||||
|                                 NULL, /* cancellable */ | ||||
|                                 &error); | ||||
|  | ||||
|   g_free (uuid_str); | ||||
|  | ||||
|   if (!res) | ||||
|     { | ||||
|       g_warning ("Failed to retrieve errors: %s", error->message); | ||||
| @@ -761,25 +657,29 @@ plugin_get_errors (PluginObject    *obj, | ||||
| } | ||||
|  | ||||
| static gboolean | ||||
| plugin_launch_extension_prefs (PluginObject    *obj, | ||||
|                                uint32_t         argc, | ||||
|                                const NPVariant *argv, | ||||
|                                NPVariant       *result) | ||||
| plugin_launch_extension_prefs (PluginObject *obj, | ||||
|                                NPString      uuid, | ||||
|                                NPVariant    *result) | ||||
| { | ||||
|   gchar *uuid; | ||||
|   gchar *uuid_str; | ||||
|  | ||||
|   if (!parse_args ("u", argc, argv, &uuid)) | ||||
|     return FALSE; | ||||
|   uuid_str = g_strndup (uuid.UTF8Characters, uuid.UTF8Length); | ||||
|   if (!uuid_is_valid (uuid_str)) | ||||
|     { | ||||
|       g_free (uuid_str); | ||||
|       return FALSE; | ||||
|     } | ||||
|  | ||||
|   g_dbus_proxy_call (obj->proxy, | ||||
|                      "LaunchExtensionPrefs", | ||||
|                      g_variant_new ("(s)", uuid), | ||||
|                      g_variant_new ("(s)", uuid_str), | ||||
|                      G_DBUS_CALL_FLAGS_NONE, | ||||
|                      -1, /* timeout */ | ||||
|                      NULL, /* cancellable */ | ||||
|                      NULL, /* callback */ | ||||
|                      NULL /* user_data */); | ||||
|  | ||||
|   g_free (uuid_str); | ||||
|   return TRUE; | ||||
| } | ||||
|  | ||||
| @@ -833,40 +733,10 @@ plugin_get_shell_version (PluginObject  *obj, | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| #define METHODS                                 \ | ||||
|   METHOD (list_extensions)                      \ | ||||
|   METHOD (get_info)                             \ | ||||
|   METHOD (enable_extension)                     \ | ||||
|   METHOD (install_extension)                    \ | ||||
|   METHOD (uninstall_extension)                  \ | ||||
|   METHOD (get_errors)                           \ | ||||
|   METHOD (launch_extension_prefs)               \ | ||||
|   /* */ | ||||
|  | ||||
| #define METHOD(x)                               \ | ||||
|   static NPIdentifier x##_id; | ||||
| METHODS | ||||
| #undef METHOD | ||||
|  | ||||
| static NPIdentifier api_version_id; | ||||
| static NPIdentifier shell_version_id; | ||||
| static NPIdentifier onextension_changed_id; | ||||
| static NPIdentifier onrestart_id; | ||||
|  | ||||
| static bool | ||||
| plugin_object_has_method (NPObject     *npobj, | ||||
|                           NPIdentifier  name) | ||||
| { | ||||
| #define METHOD(x) (name == (x##_id)) || | ||||
|   /* expands to (name == list_extensions_id) || FALSE; */ | ||||
|   return METHODS FALSE; | ||||
| #undef METHOD | ||||
| } | ||||
|  | ||||
| static bool | ||||
| plugin_object_invoke (NPObject        *npobj, | ||||
|                       NPIdentifier     name, | ||||
|                       const NPVariant *argv, | ||||
|                       const NPVariant *args, | ||||
|                       uint32_t         argc, | ||||
|                       NPVariant       *result) | ||||
| { | ||||
| @@ -878,13 +748,61 @@ plugin_object_invoke (NPObject        *npobj, | ||||
|  | ||||
|   VOID_TO_NPVARIANT (*result); | ||||
|  | ||||
| #define METHOD(x)                                       \ | ||||
|   if (name == x##_id)                                   \ | ||||
|     return plugin_##x (obj, argc, argv, result); | ||||
| METHODS | ||||
| #undef METHOD | ||||
|   if (!plugin_object_has_method (npobj, name)) | ||||
|     return FALSE; | ||||
|  | ||||
|   return FALSE; | ||||
|   if (name == list_extensions_id) | ||||
|     return plugin_list_extensions (obj, result); | ||||
|   else if (name == get_info_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|  | ||||
|       return plugin_get_info (obj, NPVARIANT_TO_STRING(args[0]), result); | ||||
|     } | ||||
|   else if (name == enable_extension_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|       if (!NPVARIANT_IS_BOOLEAN(args[1])) return FALSE; | ||||
|  | ||||
|       return plugin_enable_extension (obj, | ||||
|                                       NPVARIANT_TO_STRING(args[0]), | ||||
|                                       NPVARIANT_TO_BOOLEAN(args[1])); | ||||
|     } | ||||
|   else if (name == install_extension_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|       if (!NPVARIANT_IS_STRING(args[1])) return FALSE; | ||||
|  | ||||
|       return plugin_install_extension (obj, | ||||
|                                        NPVARIANT_TO_STRING(args[0]), | ||||
|                                        NPVARIANT_TO_STRING(args[1])); | ||||
|     } | ||||
|   else if (name == uninstall_extension_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|  | ||||
|       return plugin_uninstall_extension (obj, | ||||
|                                          NPVARIANT_TO_STRING(args[0]), | ||||
|                                          result); | ||||
|     } | ||||
|   else if (name == get_errors_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|  | ||||
|       return plugin_get_errors (obj, | ||||
|                                 NPVARIANT_TO_STRING(args[0]), | ||||
|                                 result); | ||||
|     } | ||||
|   else if (name == launch_extension_prefs_id) | ||||
|     { | ||||
|       if (!NPVARIANT_IS_STRING(args[0])) return FALSE; | ||||
|  | ||||
|       return plugin_launch_extension_prefs (obj, | ||||
|                                             NPVARIANT_TO_STRING(args[0]), | ||||
|                                             result); | ||||
|     } | ||||
|  | ||||
|   return TRUE; | ||||
| } | ||||
|  | ||||
| static bool | ||||
| @@ -1028,12 +946,3 @@ NPP_GetValue(NPP          instance, | ||||
|  | ||||
|   return NPERR_NO_ERROR; | ||||
| } | ||||
|  | ||||
| /* Opera tries to call NPP_SetWindow without checking the | ||||
|  * NULL pointer beforehand. */ | ||||
| NPError | ||||
| NPP_SetWindow(NPP          instance, | ||||
|               NPWindow    *window) | ||||
| { | ||||
|   return NPERR_NO_ERROR; | ||||
| } | ||||
|   | ||||
							
								
								
									
										181
									
								
								configure.ac
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,5 @@ | ||||
| AC_PREREQ(2.63) | ||||
| AC_INIT([gnome-shell],[3.7.90],[https://bugzilla.gnome.org/enter_bug.cgi?product=gnome-shell],[gnome-shell]) | ||||
| AC_INIT([gnome-shell],[3.4.0],[https://bugzilla.gnome.org/enter_bug.cgi?product=gnome-shell],[gnome-shell]) | ||||
|  | ||||
| AC_CONFIG_HEADERS([config.h]) | ||||
| AC_CONFIG_SRCDIR([src/shell-global.c]) | ||||
| @@ -16,6 +16,8 @@ m4_ifdef([AM_SILENT_RULES],[AM_SILENT_RULES([yes])]) | ||||
|  | ||||
| # Checks for programs. | ||||
| AC_PROG_CC | ||||
| # Needed for per-target cflags, like in gnomeshell-taskpanel | ||||
| AM_PROG_CC_C_O | ||||
|  | ||||
| # Initialize libtool | ||||
| LT_PREREQ([2.2.6]) | ||||
| @@ -34,8 +36,6 @@ AC_DEFINE_UNQUOTED(GETTEXT_PACKAGE, "$GETTEXT_PACKAGE", | ||||
|  | ||||
| PKG_PROG_PKG_CONFIG([0.22]) | ||||
|  | ||||
| AC_PATH_PROG([XSLTPROC], [xsltproc]) | ||||
|  | ||||
| GLIB_GSETTINGS | ||||
|  | ||||
| # Get a value to substitute into gnome-shell.in | ||||
| @@ -44,71 +44,86 @@ AC_SUBST(PYTHON) | ||||
|  | ||||
| # We need at least this, since gst_plugin_register_static() was added | ||||
| # in 0.10.16, but nothing older than 0.10.21 has been tested. | ||||
| GSTREAMER_MIN_VERSION=0.11.92 | ||||
| GSTREAMER_MIN_VERSION=0.10.16 | ||||
|  | ||||
| recorder_modules= | ||||
| build_recorder=false | ||||
| AC_MSG_CHECKING([for GStreamer (needed for recording functionality)]) | ||||
| if $PKG_CONFIG --exists gstreamer-1.0 '>=' $GSTREAMER_MIN_VERSION ; then | ||||
| if $PKG_CONFIG --exists gstreamer-0.10 '>=' $GSTREAMER_MIN_VERSION ; then | ||||
|    AC_MSG_RESULT(yes) | ||||
|    build_recorder=true | ||||
|    recorder_modules="gstreamer-1.0 gstreamer-base-1.0 x11 gtk+-3.0" | ||||
|    PKG_CHECK_MODULES(TEST_SHELL_RECORDER, $recorder_modules clutter-1.0 xfixes) | ||||
|    recorder_modules="gstreamer-0.10 gstreamer-base-0.10 x11" | ||||
|    PKG_CHECK_MODULES(TEST_SHELL_RECORDER, $recorder_modules clutter-1.0 xfixes gl) | ||||
| else | ||||
|    AC_MSG_RESULT(no) | ||||
| fi | ||||
|  | ||||
| AM_CONDITIONAL(BUILD_RECORDER, $build_recorder) | ||||
|  | ||||
| CLUTTER_MIN_VERSION=1.13.4 | ||||
| CLUTTER_MIN_VERSION=1.9.16 | ||||
| GOBJECT_INTROSPECTION_MIN_VERSION=0.10.1 | ||||
| GJS_MIN_VERSION=1.35.4 | ||||
| MUTTER_MIN_VERSION=3.7.90 | ||||
| GTK_MIN_VERSION=3.7.9 | ||||
| GIO_MIN_VERSION=2.35.0 | ||||
| LIBECAL_MIN_VERSION=3.5.3 | ||||
| LIBEDATASERVER_MIN_VERSION=3.5.3 | ||||
| GJS_MIN_VERSION=1.29.18 | ||||
| MUTTER_MIN_VERSION=3.3.92 | ||||
| FOLKS_MIN_VERSION=0.5.2 | ||||
| GTK_MIN_VERSION=3.3.9 | ||||
| GIO_MIN_VERSION=2.31.6 | ||||
| LIBECAL_MIN_VERSION=2.32.0 | ||||
| LIBEDATASERVER_MIN_VERSION=1.2.0 | ||||
| LIBEDATASERVERUI_MIN_VERSION=2.91.6 | ||||
| TELEPATHY_GLIB_MIN_VERSION=0.17.5 | ||||
| TELEPATHY_LOGGER_MIN_VERSION=0.2.4 | ||||
| POLKIT_MIN_VERSION=0.100 | ||||
| STARTUP_NOTIFICATION_MIN_VERSION=0.11 | ||||
| GCR_MIN_VERSION=3.3.90 | ||||
| GNOME_DESKTOP_REQUIRED_VERSION=3.7.90 | ||||
| GNOME_MENUS_REQUIRED_VERSION=3.5.3 | ||||
| NETWORKMANAGER_MIN_VERSION=0.9.6 | ||||
| PULSE_MIN_VERS=2.0 | ||||
|  | ||||
| # Collect more than 20 libraries for a prize! | ||||
| PKG_CHECK_MODULES(GNOME_SHELL, gio-unix-2.0 >= $GIO_MIN_VERSION | ||||
| 			       libxml-2.0 | ||||
|                                gtk+-3.0 >= $GTK_MIN_VERSION | ||||
|                                atk-bridge-2.0 | ||||
|                                folks >= $FOLKS_MIN_VERSION | ||||
|                                libmutter >= $MUTTER_MIN_VERSION | ||||
|                                gjs-internals-1.0 >= $GJS_MIN_VERSION | ||||
| 			       libgnome-menu-3.0 >= $GNOME_MENUS_REQUIRED_VERSION | ||||
|                                $recorder_modules | ||||
| 			       libgnome-menu-3.0 $recorder_modules | ||||
|                                gdk-x11-3.0 libsoup-2.4 | ||||
|                                gl | ||||
| 			       clutter-x11-1.0 >= $CLUTTER_MIN_VERSION | ||||
| 			       clutter-glx-1.0 >= $CLUTTER_MIN_VERSION | ||||
|                                libstartup-notification-1.0 >= $STARTUP_NOTIFICATION_MIN_VERSION | ||||
|                                gobject-introspection-1.0 >= $GOBJECT_INTROSPECTION_MIN_VERSION | ||||
| 			       libcanberra libcanberra-gtk3 | ||||
| 			       libcanberra | ||||
|                                telepathy-glib >= $TELEPATHY_GLIB_MIN_VERSION | ||||
|                                telepathy-logger-0.2 >= $TELEPATHY_LOGGER_MIN_VERSION | ||||
|                                polkit-agent-1 >= $POLKIT_MIN_VERSION xfixes | ||||
|                                libnm-glib libnm-util >= $NETWORKMANAGER_MIN_VERSION | ||||
|                                libnm-gtk >= $NETWORKMANAGER_MIN_VERSION | ||||
|                                gnome-keyring-1 gcr-3 >= $GCR_MIN_VERSION) | ||||
|                                libnm-glib libnm-util gnome-keyring-1 | ||||
|                                gcr-3 >= $GCR_MIN_VERSION) | ||||
|  | ||||
| PKG_CHECK_MODULES(SHELL_PERF_HELPER, gtk+-3.0 gio-2.0) | ||||
|  | ||||
| PKG_CHECK_MODULES(SHELL_HOTPLUG_SNIFFER, gio-2.0 gdk-pixbuf-2.0) | ||||
|  | ||||
| PKG_CHECK_MODULES(BROWSER_PLUGIN, gio-2.0 >= $GIO_MIN_VERSION json-glib-1.0 >= 0.13.2) | ||||
|  | ||||
| GJS_VERSION=`$PKG_CONFIG --modversion gjs-internals-1.0` | ||||
| AC_DEFINE_UNQUOTED([GJS_VERSION], ["$GJS_VERSION"], [The version of GJS we're linking to]) | ||||
| AC_SUBST([GJS_VERSION], ["$GJS_VERSION"]) | ||||
|  | ||||
| GOBJECT_INTROSPECTION_CHECK([$GOBJECT_INTROSPECTION_MIN_VERSION]) | ||||
| JHBUILD_TYPELIBDIR="$INTROSPECTION_TYPELIBDIR" | ||||
| AC_SUBST(JHBUILD_TYPELIBDIR) | ||||
|  | ||||
| saved_CFLAGS=$CFLAGS | ||||
| saved_LIBS=$LIBS | ||||
| CFLAGS=$GNOME_SHELL_CFLAGS | ||||
| LIBS=$GNOME_SHELL_LIBS | ||||
| AC_CHECK_FUNCS(JS_NewGlobalObject XFixesCreatePointerBarrier) | ||||
| CFLAGS=$saved_CFLAGS | ||||
| LIBS=$saved_LIBS | ||||
|  | ||||
| PKG_CHECK_MODULES(GNOME_SHELL_JS, gio-2.0 gjs-internals-1.0 >= $GJS_MIN_VERSION) | ||||
| PKG_CHECK_MODULES(ST, clutter-1.0 gtk+-3.0 libcroco-0.6 >= 0.6.8 x11) | ||||
| PKG_CHECK_MODULES(SHELL_PERF_HELPER, gtk+-3.0 gio-2.0) | ||||
| PKG_CHECK_MODULES(SHELL_HOTPLUG_SNIFFER, gio-2.0 gdk-pixbuf-2.0) | ||||
| PKG_CHECK_MODULES(BROWSER_PLUGIN, gio-2.0 >= $GIO_MIN_VERSION json-glib-1.0 >= 0.13.2) | ||||
| PKG_CHECK_MODULES(ST, clutter-1.0 gtk+-3.0 libcroco-0.6 >= 0.6.2 x11) | ||||
| PKG_CHECK_MODULES(TRAY, gtk+-3.0) | ||||
| PKG_CHECK_MODULES(GVC, libpulse >= $PULSE_MIN_VERS libpulse-mainloop-glib gobject-2.0) | ||||
| PKG_CHECK_MODULES(DESKTOP_SCHEMAS, gsettings-desktop-schemas >= 3.7.4) | ||||
| PKG_CHECK_MODULES(CARIBOU, caribou-1.0 >= 0.4.8) | ||||
| PKG_CHECK_MODULES(GVC, libpulse libpulse-mainloop-glib gobject-2.0) | ||||
| PKG_CHECK_MODULES(DESKTOP_SCHEMAS, gsettings-desktop-schemas >= 0.1.7) | ||||
|  | ||||
| AC_MSG_CHECKING([for bluetooth support]) | ||||
| PKG_CHECK_EXISTS([gnome-bluetooth-1.0 >= 3.1.0], | ||||
| @@ -124,14 +139,36 @@ PKG_CHECK_EXISTS([gnome-bluetooth-1.0 >= 3.1.0], | ||||
| 	 AC_SUBST([HAVE_BLUETOOTH],[0]) | ||||
| 	 AC_MSG_RESULT([no])]) | ||||
|  | ||||
| PKG_CHECK_MODULES(CALENDAR_SERVER, libecal-1.2 >= $LIBECAL_MIN_VERSION libedataserver-1.2 >= $LIBEDATASERVER_MIN_VERSION gio-2.0) | ||||
| PKG_CHECK_MODULES(CALENDAR_SERVER, libecal-1.2 >= $LIBECAL_MIN_VERSION libedataserver-1.2 >= $LIBEDATASERVER_MIN_VERSION libedataserverui-3.0 >= $LIBEDATASERVERUI_MIN_VERSION gio-2.0) | ||||
| AC_SUBST(CALENDAR_SERVER_CFLAGS) | ||||
| AC_SUBST(CALENDAR_SERVER_LIBS) | ||||
|  | ||||
| GNOME_KEYBINDINGS_KEYSDIR=`$PKG_CONFIG --variable keysdir gnome-keybindings` | ||||
| AC_SUBST([GNOME_KEYBINDINGS_KEYSDIR]) | ||||
| AC_ARG_WITH(systemd, | ||||
|             AS_HELP_STRING([--with-systemd], | ||||
|                            [Add systemd support]), | ||||
|             [with_systemd=$withval], [with_systemd=auto]) | ||||
|  | ||||
| GOBJECT_INTROSPECTION_CHECK([$GOBJECT_INTROSPECTION_MIN_VERSION]) | ||||
| PKG_CHECK_MODULES(SYSTEMD, | ||||
|                   [libsystemd-login libsystemd-daemon], | ||||
|                   [have_systemd=yes], [have_systemd=no]) | ||||
|  | ||||
| if test "x$with_systemd" = "xauto" ; then | ||||
|         if test x$have_systemd = xno ; then | ||||
|                 use_systemd=no | ||||
|         else | ||||
|                 use_systemd=yes | ||||
|         fi | ||||
| else | ||||
|         use_systemd=$with_systemd | ||||
| fi | ||||
|  | ||||
| if test "x$use_systemd" = "xyes"; then | ||||
|         if test "x$have_systemd" = "xno"; then | ||||
|                 AC_MSG_ERROR([Systemd support explicitly required, but systemd not found]) | ||||
|         fi | ||||
|  | ||||
|         AC_DEFINE(WITH_SYSTEMD, 1, [systemd support]) | ||||
| fi | ||||
|  | ||||
| MUTTER_GIR_DIR=`$PKG_CONFIG --variable=girdir libmutter` | ||||
| MUTTER_TYPELIB_DIR=`$PKG_CONFIG --variable=typelibdir libmutter` | ||||
| @@ -158,27 +195,75 @@ fi | ||||
|  | ||||
| # Sets GLIB_GENMARSHAL and GLIB_MKENUMS | ||||
| AM_PATH_GLIB_2_0() | ||||
| G_IR_SCANNER=`$PKG_CONFIG --variable=g_ir_scanner gobject-introspection-1.0` | ||||
| AC_SUBST(G_IR_SCANNER) | ||||
| G_IR_COMPILER=`$PKG_CONFIG --variable=g_ir_compiler gobject-introspection-1.0` | ||||
| AC_SUBST(G_IR_COMPILER) | ||||
| G_IR_GENERATE=`$PKG_CONFIG --variable=g_ir_generate gobject-introspection-1.0` | ||||
| AC_SUBST(G_IR_GENERATE) | ||||
| GIRDIR=`$PKG_CONFIG --variable=girdir gobject-introspection-1.0` | ||||
| AC_SUBST(GIRDIR) | ||||
| TYPELIBDIR="$($PKG_CONFIG --variable=typelibdir gobject-introspection-1.0)" | ||||
| AC_SUBST(TYPELIBDIR) | ||||
|  | ||||
| GTK_DOC_CHECK([1.15], [--flavour no-tmpl]) | ||||
|  | ||||
| AC_ARG_ENABLE(man, | ||||
|               [AS_HELP_STRING([--enable-man], | ||||
|                               [generate man pages [default=yes]])],, | ||||
|               enable_man=yes) | ||||
| if test "$enable_man" != no; then | ||||
|   AC_PATH_PROG([XSLTPROC], [xsltproc]) | ||||
|   if test -z "$XSLTPROC"; then | ||||
|     AC_MSG_ERROR([xsltproc is required for --enable-man]) | ||||
| # Stay command-line compatible with the gnome-common configure option. Here | ||||
| # minimum/yes/maximum are the same, however. | ||||
| AC_ARG_ENABLE(compile_warnings, | ||||
|   AS_HELP_STRING([--enable-compile-warnings=@<:@no/minimum/yes/maximum/error@:>@],[Turn on compiler warnings]),, | ||||
|   enable_compile_warnings=error) | ||||
|  | ||||
| changequote(,)dnl | ||||
| if test "$enable_compile_warnings" != no ; then | ||||
|   if test "x$GCC" = "xyes"; then | ||||
|     case " $CFLAGS " in | ||||
|     *[\ \	]-Wall[\ \	]*) ;; | ||||
|     *) CFLAGS="$CFLAGS -Wall" ;; | ||||
|     esac | ||||
|     case " $CFLAGS " in | ||||
|     *[\ \	]-Wmissing-prototypes[\ \	]*) ;; | ||||
|     *) CFLAGS="$CFLAGS -Wmissing-prototypes" ;; | ||||
|     esac | ||||
|     if test "$enable_compile_warnings" = error ; then | ||||
|       case " $CFLAGS " in | ||||
|       *[\ \	]-Werror[\ \	]*) ;; | ||||
|       *) CFLAGS="$CFLAGS -Werror -Wno-error=deprecated-declarations" ;; | ||||
|       esac | ||||
|     fi | ||||
|   fi | ||||
| fi | ||||
| AM_CONDITIONAL(ENABLE_MAN, test "$enable_man" != no) | ||||
|  | ||||
| GNOME_COMPILE_WARNINGS([error]) | ||||
| changequote([,])dnl | ||||
|  | ||||
| AC_ARG_ENABLE(jhbuild-wrapper-script, | ||||
|   AS_HELP_STRING([--enable-jhbuild-wrapper-script],[Make "gnome-shell" script work for jhbuild]),,enable_jhbuild_wrapper_script=no) | ||||
| AM_CONDITIONAL(USE_JHBUILD_WRAPPER_SCRIPT, test "x$enable_jhbuild_wrapper_script" = xyes) | ||||
|  | ||||
| AC_MSG_CHECKING([location of system Certificate Authority list]) | ||||
| AC_ARG_WITH(ca-certificates, | ||||
|             [AC_HELP_STRING([--with-ca-certificates=@<:@path@:>@], | ||||
|                             [path to system Certificate Authority list])]) | ||||
|  | ||||
| if test "$with_ca_certificates" = "no"; then | ||||
|     AC_MSG_RESULT([disabled]) | ||||
| else | ||||
|     if test -z "$with_ca_certificates"; then | ||||
|         for f in /etc/pki/tls/certs/ca-bundle.crt \ | ||||
|                  /etc/ssl/certs/ca-certificates.crt; do | ||||
|              if test -f "$f"; then | ||||
|                 with_ca_certificates="$f" | ||||
|              fi | ||||
|         done | ||||
|         if test -z "$with_ca_certificates"; then | ||||
|             AC_MSG_ERROR([could not find. Use --with-ca-certificates=path to set, or --without-ca-certificates to disable]) | ||||
|         fi | ||||
|    fi | ||||
|  | ||||
|    AC_MSG_RESULT($with_ca_certificates) | ||||
|    AC_DEFINE_UNQUOTED(SHELL_SYSTEM_CA_FILE, ["$with_ca_certificates"], [The system TLS CA list]) | ||||
| fi | ||||
| AC_SUBST(SHELL_SYSTEM_CA_FILE,["$with_ca_certificates"]) | ||||
|  | ||||
| BROWSER_PLUGIN_DIR="${BROWSER_PLUGIN_DIR:-"\${libdir}/mozilla/plugins"}" | ||||
| AC_ARG_VAR([BROWSER_PLUGIN_DIR],[Where to install the plugin to]) | ||||
|  | ||||
| @@ -192,9 +277,7 @@ AC_CONFIG_FILES([ | ||||
|   docs/reference/st/Makefile | ||||
|   docs/reference/st/st-docs.sgml | ||||
|   js/Makefile | ||||
|   src/calendar-server/evolution-calendar.desktop.in | ||||
|   src/Makefile | ||||
|   src/gvc/Makefile | ||||
|   browser-plugin/Makefile | ||||
|   tests/Makefile | ||||
|   po/Makefile.in | ||||
|   | ||||
| @@ -1,12 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" ?> | ||||
| <KeyListEntries schema="org.gnome.shell.keybindings" | ||||
|                 group="system" | ||||
|                 _name="Screenshots" | ||||
|                 wm_name="GNOME Shell" | ||||
|                 package="gnome-shell"> | ||||
|  | ||||
| 	<KeyListEntry name="toggle-recording" | ||||
|                       _description="Record a screencast"/> | ||||
|  | ||||
| </KeyListEntries> | ||||
|  | ||||
| @@ -1,21 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" ?> | ||||
| <KeyListEntries schema="org.gnome.shell.keybindings" | ||||
|                 group="system" | ||||
|                 _name="System" | ||||
|                 wm_name="GNOME Shell" | ||||
|                 package="gnome-shell"> | ||||
|  | ||||
| 	<KeyListEntry name="toggle-message-tray" | ||||
|                       _description="Show the message tray"/> | ||||
|  | ||||
| 	<KeyListEntry name="focus-active-notification" | ||||
|                       _description="Focus the active notification"/> | ||||
|  | ||||
| 	<KeyListEntry name="toggle-application-view" | ||||
|                       _description="Show all applications"/> | ||||
|  | ||||
| 	<KeyListEntry name="open-application-menu" | ||||
|                       _description="Open the application menu"/> | ||||
|  | ||||
| </KeyListEntries> | ||||
|  | ||||
| @@ -8,13 +8,17 @@ desktop_DATA = gnome-shell.desktop gnome-shell-extension-prefs.desktop | ||||
| 	    -e "s|@VERSION[@]|$(VERSION)|" \ | ||||
| 	    $< > $@ || rm $@ | ||||
|  | ||||
| @INTLTOOL_DESKTOP_RULE@ | ||||
| # Placeholder until we add intltool | ||||
| %.desktop:%.desktop.in | ||||
| 	$(AM_V_GEN) sed s/^_// < $< > $@ || rm $@ | ||||
|  | ||||
| searchprovidersdir = $(pkgdatadir)/open-search-providers | ||||
| dist_searchproviders_DATA =				\ | ||||
| 	open-search-providers/google.xml		\ | ||||
| 	open-search-providers/wikipedia.xml | ||||
|  | ||||
| introspectiondir = $(datadir)/dbus-1/interfaces | ||||
| introspection_DATA =				\ | ||||
| 	org.gnome.Shell.Screenshot.xml		\ | ||||
| 	org.gnome.ShellSearchProvider.xml	\ | ||||
| 	org.gnome.ShellSearchProvider2.xml | ||||
| introspection_DATA = org.gnome.ShellSearchProvider.xml | ||||
|  | ||||
| themedir = $(pkgdatadir)/theme | ||||
| dist_theme_DATA =				\ | ||||
| @@ -32,40 +36,26 @@ dist_theme_DATA =				\ | ||||
| 	theme/dash-placeholder.svg		\ | ||||
| 	theme/filter-selected-ltr.svg		\ | ||||
| 	theme/filter-selected-rtl.svg		\ | ||||
| 	theme/gdm.css				\ | ||||
| 	theme/gnome-shell.css			\ | ||||
| 	theme/logged-in-indicator.svg		\ | ||||
| 	theme/message-tray-background.png	\ | ||||
| 	theme/more-results.svg			\ | ||||
| 	theme/noise-texture.png			\ | ||||
| 	theme/panel-button-border.svg		\ | ||||
| 	theme/panel-button-highlight-narrow.svg	\ | ||||
| 	theme/panel-button-highlight-wide.svg	\ | ||||
| 	theme/process-working.svg		\ | ||||
| 	theme/running-indicator.svg		\ | ||||
| 	theme/scroll-hhandle.svg		\ | ||||
| 	theme/scroll-vhandle.svg		\ | ||||
| 	theme/source-button-border.svg		\ | ||||
| 	theme/summary-counter.svg		\ | ||||
| 	theme/toggle-off-us.svg			\ | ||||
| 	theme/toggle-off-intl.svg		\ | ||||
| 	theme/toggle-on-us.svg			\ | ||||
| 	theme/toggle-on-intl.svg		\ | ||||
| 	theme/ws-switch-arrow-up.png		\ | ||||
| 	theme/ws-switch-arrow-down.png | ||||
|  | ||||
| keysdir = @GNOME_KEYBINDINGS_KEYSDIR@ | ||||
| keys_in_files =					\ | ||||
| 	50-gnome-shell-screenshot.xml.in	\ | ||||
| 	50-gnome-shell-system.xml.in		\ | ||||
| 	$(NULL) | ||||
| keys_DATA = $(keys_in_files:.xml.in=.xml) | ||||
| 	theme/ws-switch-arrow-up.svg		\ | ||||
| 	theme/ws-switch-arrow-down.svg | ||||
|  | ||||
| gsettings_SCHEMAS = org.gnome.shell.gschema.xml | ||||
|  | ||||
| @INTLTOOL_XML_NOMERGE_RULE@ | ||||
|  | ||||
| %.gschema.xml.in: %.gschema.xml.in.in Makefile | ||||
| 	$(AM_V_GEN) sed -e 's|@GETTEXT_PACKAGE[@]|$(GETTEXT_PACKAGE)|g' \ | ||||
| 	$< > $@ || rm $@ | ||||
|  | ||||
| @GSETTINGS_RULES@ | ||||
|  | ||||
| # We need to compile schemas at make time | ||||
| @@ -78,21 +68,24 @@ all-local: gschemas.compiled | ||||
| convertdir = $(datadir)/GConf/gsettings | ||||
| convert_DATA = gnome-shell-overrides.convert | ||||
|  | ||||
| shadersdir = $(pkgdatadir)/shaders | ||||
| shaders_DATA = \ | ||||
| 	shaders/dim-window.glsl | ||||
|  | ||||
|  | ||||
| EXTRA_DIST =						\ | ||||
| 	gnome-shell.desktop.in.in			\ | ||||
| 	gnome-shell-extension-prefs.desktop.in.in	\ | ||||
| 	$(introspection_DATA)				\ | ||||
| 	$(menu_DATA)					\ | ||||
| 	$(shaders_DATA)					\ | ||||
| 	$(convert_DATA)					\ | ||||
| 	$(keys_in_files)				\ | ||||
| 	org.gnome.shell.gschema.xml.in.in | ||||
| 	org.gnome.shell.gschema.xml.in | ||||
|  | ||||
| CLEANFILES =						\ | ||||
| 	gnome-shell.desktop.in				\ | ||||
| 	gnome-shell-extension-prefs.in			\ | ||||
| 	$(desktop_DATA)					\ | ||||
| 	$(keys_DATA)					\ | ||||
| 	$(gsettings_SCHEMAS)				\ | ||||
| 	gschemas.compiled				\ | ||||
| 	org.gnome.shell.gschema.valid			\ | ||||
| 	org.gnome.shell.gschema.xml.in | ||||
| 	gschemas.compiled | ||||
|  | ||||
|   | ||||
							
								
								
									
										7
									
								
								data/open-search-providers/google.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/"> | ||||
| <ShortName>Google</ShortName> | ||||
| <Description>Google Search</Description> | ||||
| <InputEncoding>UTF-8</InputEncoding> | ||||
| <Image width="16" height="16">data:image/x-icon;base64,AAABAAEAEBAAAAEAGABoAwAAFgAAACgAAAAQAAAAIAAAAAEAGAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADs9Pt8xetPtu9FsfFNtu%2BTzvb2%2B%2Fne4dFJeBw0egA%2FfAJAfAA8ewBBegAAAAD%2B%2FPtft98Mp%2BwWsfAVsvEbs%2FQeqvF8xO7%2F%2F%2F63yqkxdgM7gwE%2FggM%2BfQA%2BegBDeQDe7PIbotgQufcMufEPtfIPsvAbs%2FQvq%2Bfz%2Bf%2F%2B%2B%2FZKhR05hgBBhQI8hgBAgAI9ewD0%2B%2Fg3pswAtO8Cxf4Kw%2FsJvvYAqupKsNv%2B%2Fv7%2F%2FP5VkSU0iQA7jQA9hgBDgQU%2BfQH%2F%2Ff%2FQ6fM4sM4KsN8AteMCruIqqdbZ7PH8%2Fv%2Fg6Nc%2Fhg05kAA8jAM9iQI%2BhQA%2BgQDQu6b97uv%2F%2F%2F7V8Pqw3eiWz97q8%2Ff%2F%2F%2F%2F7%2FPptpkkqjQE4kwA7kAA5iwI8iAA8hQCOSSKdXjiyflbAkG7u2s%2F%2B%2F%2F39%2F%2F7r8utrqEYtjQE8lgA7kwA7kwA9jwA9igA9hACiWSekVRyeSgiYSBHx6N%2F%2B%2Fv7k7OFRmiYtlAA5lwI7lwI4lAA7kgI9jwE9iwI4iQCoVhWcTxCmb0K%2BooT8%2Fv%2F7%2F%2F%2FJ2r8fdwI1mwA3mQA3mgA8lAE8lAE4jwA9iwE%2BhwGfXifWvqz%2B%2Ff%2F58u%2Fev6Dt4tr%2B%2F%2F2ZuIUsggA7mgM6mAM3lgA5lgA6kQE%2FkwBChwHt4dv%2F%2F%2F728ei1bCi7VAC5XQ7kz7n%2F%2F%2F6bsZkgcB03lQA9lgM7kwA2iQktZToPK4r9%2F%2F%2F9%2F%2F%2FSqYK5UwDKZAS9WALIkFn%2B%2F%2F3%2F%2BP8oKccGGcIRJrERILYFEMwAAuEAAdX%2F%2Ff7%2F%2FP%2B%2BfDvGXQLIZgLEWgLOjlf7%2F%2F%2F%2F%2F%2F9QU90EAPQAAf8DAP0AAfMAAOUDAtr%2F%2F%2F%2F7%2B%2Fu2bCTIYwDPZgDBWQDSr4P%2F%2Fv%2F%2F%2FP5GRuABAPkAA%2FwBAfkDAPAAAesAAN%2F%2F%2B%2Fz%2F%2F%2F64g1C5VwDMYwK8Yg7y5tz8%2Fv%2FV1PYKDOcAAP0DAf4AAf0AAfYEAOwAAuAAAAD%2F%2FPvi28ymXyChTATRrIb8%2F%2F3v8fk6P8MAAdUCAvoAAP0CAP0AAfYAAO4AAACAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACAAQAA</Image> | ||||
| <Url type="text/html" method="GET" template="http://www.google.com/search?q={searchTerms}"/> | ||||
| </OpenSearchDescription> | ||||
							
								
								
									
										44
									
								
								data/open-search-providers/wikipedia.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | ||||
| <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/"> | ||||
| <ShortName>Wikipedia</ShortName> | ||||
| <Description>Wikipedia, the free encyclopedia</Description> | ||||
| <InputEncoding>UTF-8</InputEncoding> | ||||
| <Image width="16" height="16">data:image/x-icon;base64,AAABAAEAEBAQAAEABAAoAQAAFgAAACgAAAAQAAAAIAAAAAEABAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAEAgQAhIOEAMjHyABIR0gA6ejpAGlqaQCpqKkAKCgoAPz9%2FAAZGBkAmJiYANjZ2ABXWFcAent6ALm6uQA8OjwAiIiIiIiIiIiIiI4oiL6IiIiIgzuIV4iIiIhndo53KIiIiB%2FWvXoYiIiIfEZfWBSIiIEGi%2FfoqoiIgzuL84i9iIjpGIoMiEHoiMkos3FojmiLlUipYliEWIF%2BiDe0GoRa7D6GPbjcu1yIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA</Image> | ||||
| <Url type="text/html" method="GET" template="http://{language}.wikipedia.org/wiki/Special:Search?search={searchTerms}"/> | ||||
| <!-- The criterion for being below is being listed with more than 100,000 | ||||
| articles on http://meta.wikimedia.org/wiki/List_of_Wikipedias --> | ||||
| <Language>ar</Language> | ||||
| <Language>bg</Language> | ||||
| <Language>ca</Language> | ||||
| <Language>cs</Language> | ||||
| <Language>da</Language> | ||||
| <Language>de</Language> | ||||
| <Language>en</Language> | ||||
| <Language>eo</Language> | ||||
| <Language>es</Language> | ||||
| <Language>fa</Language> | ||||
| <Language>fi</Language> | ||||
| <Language>fr</Language> | ||||
| <Language>he</Language> | ||||
| <Language>hu</Language> | ||||
| <Language>id</Language> | ||||
| <Language>it</Language> | ||||
| <Language>ja</Language> | ||||
| <Language>ko</Language> | ||||
| <Language>lt</Language> | ||||
| <Language>nl</Language> | ||||
| <Language>no</Language> | ||||
| <Language>pl</Language> | ||||
| <Language>pt</Language> | ||||
| <Language>ro</Language> | ||||
| <Language>ru</Language> | ||||
| <Language>sk</Language> | ||||
| <Language>sl</Language> | ||||
| <Language>sr</Language> | ||||
| <Language>sv</Language> | ||||
| <Language>tr</Language> | ||||
| <Language>uk</Language> | ||||
| <Language>vi</Language> | ||||
| <Language>vo</Language> | ||||
| <Language>war</Language> | ||||
| <Language>zh</Language> | ||||
| </OpenSearchDescription> | ||||
| @@ -1,128 +0,0 @@ | ||||
| <!DOCTYPE node PUBLIC | ||||
| '-//freedesktop//DTD D-BUS Object Introspection 1.0//EN' | ||||
| 'http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd'> | ||||
| <node> | ||||
|  | ||||
|   <!-- | ||||
|       org.gnome.Shell.Screenshot: | ||||
|       @short_description: Screenshot interface | ||||
|  | ||||
|       The interface used to capture pictures of the screen contents. | ||||
|   --> | ||||
|   <interface name="org.gnome.Shell.Screenshot"> | ||||
|  | ||||
|     <!-- | ||||
|         Screenshot: | ||||
|         @filename: The filename for the screenshot | ||||
|         @include_cursor: Whether to include the cursor image or not | ||||
|         @flash: Whether to flash the screen or not | ||||
|         @success: whether the screenshot was captured | ||||
|         @filename_used: the file where the screenshot was saved | ||||
|  | ||||
|         Takes a screenshot of the whole screen and saves it | ||||
|         in @filename as png image, it returns a boolean | ||||
|         indicating whether the operation was successful or not. | ||||
|         @filename can either be an absolute path or a basename, in | ||||
|         which case the screenshot will be saved in the $XDG_PICTURES_DIR | ||||
|         or the home directory if it doesn't exist. The filename used | ||||
|         to save the screenshot will be returned in @filename_used. | ||||
|     --> | ||||
|     <method name="Screenshot"> | ||||
|       <arg type="b" direction="in" name="include_cursor"/> | ||||
|       <arg type="b" direction="in" name="flash"/> | ||||
|       <arg type="s" direction="in" name="filename"/> | ||||
|       <arg type="b" direction="out" name="success"/> | ||||
|       <arg type="s" direction="out" name="filename_used"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         ScreenshotWindow: | ||||
|         @include_frame: Whether to include the frame or not | ||||
|         @include_cursor: Whether to include the cursor image or not | ||||
|         @flash: Whether to flash the window area or not | ||||
|         @filename: The filename for the screenshot | ||||
|         @success: whether the screenshot was captured | ||||
|         @filename_used: the file where the screenshot was saved | ||||
|  | ||||
|         Takes a screenshot of the focused window (optionally omitting the frame) | ||||
|         and saves it in @filename as png image, it returns a boolean | ||||
|         indicating whether the operation was successful or not. | ||||
|         @filename can either be an absolute path or a basename, in | ||||
|         which case the screenshot will be saved in the $XDG_PICTURES_DIR | ||||
|         or the home directory if it doesn't exist. The filename used | ||||
|         to save the screenshot will be returned in @filename_used. | ||||
|     --> | ||||
|     <method name="ScreenshotWindow"> | ||||
|       <arg type="b" direction="in" name="include_frame"/> | ||||
|       <arg type="b" direction="in" name="include_cursor"/> | ||||
|       <arg type="b" direction="in" name="flash"/> | ||||
|       <arg type="s" direction="in" name="filename"/> | ||||
|       <arg type="b" direction="out" name="success"/> | ||||
|       <arg type="s" direction="out" name="filename_used"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         ScreenshotArea: | ||||
|         @x: the X coordinate of the area to capture | ||||
|         @y: the Y coordinate of the area to capture | ||||
|         @width: the width of the area to capture | ||||
|         @height: the height of the area to capture | ||||
|         @flash: whether to flash the area or not | ||||
|         @filename: the filename for the screenshot | ||||
|         @success: whether the screenshot was captured | ||||
|         @filename_used: the file where the screenshot was saved | ||||
|  | ||||
|         Takes a screenshot of the passed in area and saves it | ||||
|         in @filename as png image, it returns a boolean | ||||
|         indicating whether the operation was successful or not. | ||||
|         @filename can either be an absolute path or a basename, in | ||||
|         which case the screenshot will be saved in the $XDG_PICTURES_DIR | ||||
|         or the home directory if it doesn't exist. The filename used | ||||
|         to save the screenshot will be returned in @filename_used. | ||||
|     --> | ||||
|     <method name="ScreenshotArea"> | ||||
|       <arg type="i" direction="in" name="x"/> | ||||
|       <arg type="i" direction="in" name="y"/> | ||||
|       <arg type="i" direction="in" name="width"/> | ||||
|       <arg type="i" direction="in" name="height"/> | ||||
|       <arg type="b" direction="in" name="flash"/> | ||||
|       <arg type="s" direction="in" name="filename"/> | ||||
|       <arg type="b" direction="out" name="success"/> | ||||
|       <arg type="s" direction="out" name="filename_used"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         FlashArea: | ||||
|         @x: the X coordinate of the area to flash | ||||
|         @y: the Y coordinate of the area to flash | ||||
|         @width: the width of the area to flash | ||||
|         @height: the height of the area to flash | ||||
|  | ||||
|         Renders a flash spot effect in the specified rectangle of the screen. | ||||
|     --> | ||||
|     <method name="FlashArea"> | ||||
|       <arg type="i" direction="in" name="x"/> | ||||
|       <arg type="i" direction="in" name="y"/> | ||||
|       <arg type="i" direction="in" name="width"/> | ||||
|       <arg type="i" direction="in" name="height"/> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         SelectArea: | ||||
|         @x: the X coordinate of the selected area | ||||
|         @y: the Y coordinate of the selected area | ||||
|         @width: the width of the selected area | ||||
|         @height: the height of the selected area | ||||
|  | ||||
|         Interactively allows the user to select a rectangular area of | ||||
|         the screen, and returns its coordinates. | ||||
|     --> | ||||
|     <method name="SelectArea"> | ||||
|       <arg type="i" direction="out" name="x"/> | ||||
|       <arg type="i" direction="out" name="y"/> | ||||
|       <arg type="i" direction="out" name="width"/> | ||||
|       <arg type="i" direction="out" name="height"/> | ||||
|     </method> | ||||
|  | ||||
|   </interface> | ||||
| </node> | ||||
| @@ -2,72 +2,146 @@ | ||||
| '-//freedesktop//DTD D-BUS Object Introspection 1.0//EN' | ||||
| 'http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd'> | ||||
| <node> | ||||
|  | ||||
|   <!-- | ||||
|       org.gnome.Shell.SearchProvider: | ||||
|       @short_description: Search provider interface | ||||
|  | ||||
|       The interface used for integrating into GNOME Shell's search | ||||
|       interface. This interface is deprecated, and org.gnome.Shell.SearchProvider2 should be used instead. | ||||
|   --> | ||||
|   <interface name="org.gnome.Shell.SearchProvider"> | ||||
|     <doc:doc> | ||||
|       <doc:description> | ||||
|         <doc:para> | ||||
|           The interface used for integrating into GNOME Shell's search | ||||
|           interface. | ||||
|         </doc:para> | ||||
|       </doc:description> | ||||
|     </doc:doc> | ||||
|  | ||||
|     <!-- | ||||
|         GetInitialResultSet: | ||||
|         @terms: Array of search terms, which the provider should treat as logical AND. | ||||
|         @results: An array of result identifier strings representing items which match the given search terms. Identifiers must be unique within the provider's domain, but other than that may be chosen freely by the provider. | ||||
|  | ||||
|         Called when the user first begins a search. | ||||
|     --> | ||||
|     <method name="GetInitialResultSet"> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="as" name="results" direction="out" /> | ||||
|       <doc:doc> | ||||
|         <doc:description> | ||||
|           <doc:para> | ||||
|             Called when the user first begins a search. | ||||
|           </doc:para> | ||||
|         </doc:description> | ||||
|       </doc:doc> | ||||
|       <arg type="as" direction="in"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               Array of search terms, which the provider should treat as | ||||
|               logical AND. | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|       <arg type="as" direction="out"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               An array of result identifier strings representing items which | ||||
|               match the given search terms. Identifiers must be unique within | ||||
|               the provider's domain, but other than that may be chosen freely | ||||
|               by the provider. | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         GetSubsearchResultSet: | ||||
|         @previous_results: Array of results previously returned by GetInitialResultSet(). | ||||
|         @terms: Array of updated search terms, which the provider should treat as logical AND. | ||||
|         @results: An array of result identifier strings representing items which match the given search terms. Identifiers must be unique within the provider's domain, but other than that may be chosen freely by the provider. | ||||
|  | ||||
|         Called when a search is performed which is a "subsearch" of | ||||
|         the previous search, e.g. the method may return less results, but | ||||
|         not more or different results. | ||||
|  | ||||
|         This allows search providers to only search through the previous | ||||
|         result set, rather than possibly performing a full re-query. | ||||
|     --> | ||||
|     <method name="GetSubsearchResultSet"> | ||||
|       <arg type="as" name="previous_results" direction="in" /> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="as" name="results" direction="out" /> | ||||
|       <doc:doc> | ||||
|         <doc:description> | ||||
|           <doc:para> | ||||
|             Called when a search is performed which is a "subsearch" of | ||||
|             the previous search, e.g. the method may return less results, but | ||||
|             not more or different results. | ||||
|  | ||||
|             This allows search providers to only search through the previous | ||||
|             result set, rather than possibly performing a full re-query. | ||||
|           </doc:para> | ||||
|         </doc:description> | ||||
|       </doc:doc> | ||||
|       <arg type="as" direction="in"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               Array of item identifiers | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|       <arg type="as" direction="in"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               Array of updated search terms, which the provider should treat as | ||||
|               logical AND. | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|       <arg type="as" direction="out"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               An array of result identifier strings representing items which | ||||
|               match the given search terms. Identifiers must be unique within | ||||
|               the provider's domain, but other than that may be chosen freely | ||||
|               by the provider. | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         GetResultMetas: | ||||
|         @identifiers: An array of result identifiers as returned by GetInitialResultSet() or GetSubsearchResultSet() | ||||
|         @metas: A dictionary describing the given search result, containing a human-readable 'name' (string), along with the result identifier this meta is for, 'id' (string). Optionally, either 'gicon' (a serialized GIcon) or 'icon-data' (raw image data as (iiibiiay) - width, height, rowstride, has-alpha, bits per sample, channels, data) can be specified if the result can be better served with a thumbnail of the content (such as with images). A 'description' field (string) may also be specified if more context would help the user find the desired result. | ||||
|  | ||||
|         Return an array of meta data used to display each given result | ||||
|     --> | ||||
|     <method name="GetResultMetas"> | ||||
|       <arg type="as" name="identifiers" direction="in" /> | ||||
|       <arg type="aa{sv}" name="metas" direction="out" /> | ||||
|       <doc:doc> | ||||
|         <doc:description> | ||||
|           <doc:para> | ||||
|             Return an array of meta data used to display each given result | ||||
|           </doc:para> | ||||
|         </doc:description> | ||||
|       </doc:doc> | ||||
|       <arg type="as" direction="in"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               An array of result identifiers as returned by | ||||
|               GetInitialResultSet() or GetSubsearchResultSet() | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|       <arg type="a{sv}" direction="out"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               A dictionary describing the given search result, containing | ||||
|               'id', 'name' (both strings) and either 'icon' (a serialized | ||||
|               GIcon) or 'icon-data' (raw image data as (iiibiiay) - width, | ||||
|               height, rowstride, has-alpha, bits per sample, channels, data) | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         ActivateResult: | ||||
|         @identifier: A result identifier as returned by GetInitialResultSet() or GetSubsearchResultSet() | ||||
|  | ||||
|         Called when the users chooses a given result. The result should | ||||
|         be displayed in the application associated with the corresponding | ||||
|         provider. | ||||
|  | ||||
|         This method is deprecated, and providers should implement ActivateResult2() | ||||
|         instead. | ||||
|     --> | ||||
|     <method name="ActivateResult"> | ||||
|       <arg type="s" name="identifier" direction="in" /> | ||||
|       <doc:doc> | ||||
|         <doc:description> | ||||
|           <doc:para> | ||||
|             Called when the users chooses a given result. The result should | ||||
|             be displayed in the application associated with the corresponding | ||||
|             provider. | ||||
|           </doc:para> | ||||
|         </doc:description> | ||||
|       </doc:doc> | ||||
|       <arg type="s" direction="in"> | ||||
|         <doc:doc> | ||||
|           <doc:summary> | ||||
|             <doc:para> | ||||
|               A result identifier as returned by GetInitialResultSet() or | ||||
|               GetSubsearchResultSet() | ||||
|             </doc:para> | ||||
|           </doc:summary> | ||||
|         </doc:doc> | ||||
|       </arg> | ||||
|     </method> | ||||
|   </interface> | ||||
| </node> | ||||
|   | ||||
| @@ -1,87 +0,0 @@ | ||||
| <!DOCTYPE node PUBLIC | ||||
| '-//freedesktop//DTD D-BUS Object Introspection 1.0//EN' | ||||
| 'http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd'> | ||||
| <node> | ||||
|  | ||||
|   <!-- | ||||
|       org.gnome.Shell.SearchProvider2: | ||||
|       @short_description: Search provider interface | ||||
|  | ||||
|       The interface used for integrating into GNOME Shell's search | ||||
|       interface (version 2). | ||||
|   --> | ||||
|   <interface name="org.gnome.Shell.SearchProvider2"> | ||||
|  | ||||
|     <!-- | ||||
|         GetInitialResultSet: | ||||
|         @terms: Array of search terms, which the provider should treat as logical AND. | ||||
|         @results: An array of result identifier strings representing items which match the given search terms. Identifiers must be unique within the provider's domain, but other than that may be chosen freely by the provider. | ||||
|  | ||||
|         Called when the user first begins a search. | ||||
|     --> | ||||
|     <method name="GetInitialResultSet"> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="as" name="results" direction="out" /> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         GetSubsearchResultSet: | ||||
|         @previous_results: Array of results previously returned by GetInitialResultSet(). | ||||
|         @terms: Array of updated search terms, which the provider should treat as logical AND. | ||||
|         @results: An array of result identifier strings representing items which match the given search terms. Identifiers must be unique within the provider's domain, but other than that may be chosen freely by the provider. | ||||
|  | ||||
|         Called when a search is performed which is a "subsearch" of | ||||
|         the previous search, e.g. the method may return less results, but | ||||
|         not more or different results. | ||||
|  | ||||
|         This allows search providers to only search through the previous | ||||
|         result set, rather than possibly performing a full re-query. | ||||
|     --> | ||||
|     <method name="GetSubsearchResultSet"> | ||||
|       <arg type="as" name="previous_results" direction="in" /> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="as" name="results" direction="out" /> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         GetResultMetas: | ||||
|         @identifiers: An array of result identifiers as returned by GetInitialResultSet() or GetSubsearchResultSet() | ||||
|         @metas: A dictionary describing the given search result, containing a human-readable 'name' (string), along with the result identifier this meta is for, 'id' (string). Optionally, either 'gicon' (a serialized GIcon) or 'icon-data' (raw image data as (iiibiiay) - width, height, rowstride, has-alpha, bits per sample, channels, data) can be specified if the result can be better served with a thumbnail of the content (such as with images). A 'description' field (string) may also be specified if more context would help the user find the desired result. | ||||
|  | ||||
|         Return an array of meta data used to display each given result | ||||
|     --> | ||||
|     <method name="GetResultMetas"> | ||||
|       <arg type="as" name="identifiers" direction="in" /> | ||||
|       <arg type="aa{sv}" name="metas" direction="out" /> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         ActivateResult: | ||||
|         @identifier: A result identifier as returned by GetInitialResultSet() or GetSubsearchResultSet() | ||||
|         @terms: Array of search terms, which the provider should treat as logical AND. | ||||
|         @timestamp: A timestamp of the user interaction that triggered this call | ||||
|  | ||||
|         Called when the users chooses a given result. The result should | ||||
|         be displayed in the application associated with the corresponding | ||||
|         provider. The provided search terms can be used to allow launching a full search in | ||||
|         the application. | ||||
|     --> | ||||
|     <method name="ActivateResult"> | ||||
|       <arg type="s" name="identifier" direction="in" /> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="u" name="timestamp" direction="in" /> | ||||
|     </method> | ||||
|  | ||||
|     <!-- | ||||
|         LaunchSearch: | ||||
|         @terms: Array of search terms, which the provider should treat as logical AND. | ||||
|         @timestamp: A timestamp of the user interaction that triggered this call | ||||
|  | ||||
|         Asks the search provider to launch a full search in the application for the provided terms. | ||||
|     --> | ||||
|     <method name="LaunchSearch"> | ||||
|       <arg type="as" name="terms" direction="in" /> | ||||
|       <arg type="u" name="timestamp" direction="in" /> | ||||
|     </method> | ||||
|   </interface> | ||||
| </node> | ||||
| @@ -39,13 +39,9 @@ | ||||
|         will be displayed in the favorites area. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="app-folder-categories" type="as"> | ||||
|       <default>[ 'Utilities', 'Sundry' ]</default> | ||||
|       <_summary>List of categories that should be displayed as folders</_summary> | ||||
|       <_description> | ||||
|         Each category name in this list will be represented as folder in the | ||||
|         application view, rather than being displayed inline in the main view. | ||||
|       </_description> | ||||
|     <key name="disabled-open-search-providers" type="as"> | ||||
|       <default>[]</default> | ||||
|       <_summary>disabled OpenSearch providers</_summary> | ||||
|     </key> | ||||
|     <key name="command-history" type="as"> | ||||
|       <default>[]</default> | ||||
| @@ -65,27 +61,9 @@ value here is from the TpConnectionPresenceType enumeration.</_summary> | ||||
|       <_summary>Internally used to store the last session presence status for the user. The | ||||
| value here is from the GsmPresenceStatus enumeration.</_summary> | ||||
|     </key> | ||||
|     <key name="always-show-log-out" type="b"> | ||||
|       <default>false</default> | ||||
|       <_summary>Always show the 'Log out' menuitem in the user menu.</_summary> | ||||
|       <_description> | ||||
|         This key overrides the automatic hiding of the 'Log out' | ||||
|         menuitem in single-user, single-session situations. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="remember-mount-password" type="b"> | ||||
|       <default>false</default> | ||||
|       <_summary>Whether to remember password for mounting encrypted or remote filesystems</_summary> | ||||
|       <_description> | ||||
|         The shell will request a password when an encrypted device or a | ||||
|         remote filesystem is mounted.  If the password can be saved for | ||||
|         future use a 'Remember Password' checkbox will be present. | ||||
|         This key sets the default state of the checkbox. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <child name="clock" schema="org.gnome.shell.clock"/> | ||||
|     <child name="calendar" schema="org.gnome.shell.calendar"/> | ||||
|     <child name="recorder" schema="org.gnome.shell.recorder"/> | ||||
|     <child name="keybindings" schema="org.gnome.shell.keybindings"/> | ||||
|     <child name="keyboard" schema="org.gnome.shell.keyboard"/> | ||||
|   </schema> | ||||
| 
 | ||||
| @@ -100,46 +78,6 @@ value here is from the GsmPresenceStatus enumeration.</_summary> | ||||
|       </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <schema id="org.gnome.shell.keybindings" path="/org/gnome/shell/keybindings/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="open-application-menu" type="as"> | ||||
|       <default>["<Super>F10"]</default> | ||||
|       <_summary>Keybinding to open the application menu</_summary> | ||||
|       <_description> | ||||
|         Keybinding to open the application menu. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="toggle-application-view" type="as"> | ||||
|       <default>["<Super>a"]</default> | ||||
|       <_summary>Keybinding to open the "Show Applications" view</_summary> | ||||
|       <_description> | ||||
|         Keybinding to open the "Show Applications" view of the Activities | ||||
|         Overview. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="toggle-message-tray" type="as"> | ||||
|       <default>["<Super>m"]</default> | ||||
|       <_summary>Keybinding to toggle the visibility of the message tray</_summary> | ||||
|       <_description> | ||||
|         Keybinding to toggle the visibility of the message tray. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="focus-active-notification" type="as"> | ||||
|       <default>["<Super>n"]</default> | ||||
|       <_summary>Keybinding to focus the active notification</_summary> | ||||
|       <_description> | ||||
|         Keybinding to focus the active notification. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="toggle-recording" type="as"> | ||||
|       <default><![CDATA[['<Control><Shift><Alt>r']]]></default> | ||||
|       <_summary>Keybinding to toggle the screen recorder</_summary> | ||||
|       <_description> | ||||
|         Keybinding to start/stop the builtin screen recorder. | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <schema id="org.gnome.shell.keyboard" path="/org/gnome/shell/keyboard/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="keyboard-type" type="s"> | ||||
| @@ -151,6 +89,24 @@ value here is from the GsmPresenceStatus enumeration.</_summary> | ||||
|     </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <schema id="org.gnome.shell.clock" path="/org/gnome/shell/clock/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="show-seconds" type="b"> | ||||
|       <default>false</default> | ||||
|       <_summary>Show time with seconds</_summary> | ||||
|       <_description> | ||||
|         If true, display seconds in time. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="show-date" type="b"> | ||||
|       <default>false</default> | ||||
|       <_summary>Show date in clock</_summary> | ||||
|       <_description> | ||||
|         If true, display date in the clock, in addition to time. | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <schema id="org.gnome.shell.recorder" path="/org/gnome/shell/recorder/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="framerate" type="i"> | ||||
| @@ -173,7 +129,7 @@ value here is from the GsmPresenceStatus enumeration.</_summary> | ||||
|         take care of its own output - this might be used to send the output | ||||
|         to an icecast server via shout2send or similar. When unset or set | ||||
|         to an empty value, the default pipeline will be used. This is currently | ||||
|         'vp8enc min_quantizer=13 max_quantizer=13 cpu-used=5 deadline=1000000 threads=%T ! queue ! webmmux' | ||||
|         'vp8enc quality=8 speed=6 threads=%T ! queue ! webmmux' | ||||
|         and records to WEBM using the VP8 codec. %T is used as a placeholder | ||||
|         for a guess at the optimal thread count on the system. | ||||
|       </_description> | ||||
| @@ -189,80 +145,44 @@ value here is from the GsmPresenceStatus enumeration.</_summary> | ||||
|     </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <enum id="org.gnome.shell.window-switcher.AppIconMode"> | ||||
|     <value value="1" nick="thumbnail-only"/> | ||||
|     <value value="2" nick="app-icon-only"/> | ||||
|     <value value="3" nick="both"/> | ||||
|   </enum> | ||||
|   <schema id="org.gnome.shell.window-switcher" | ||||
|           path="/org/gnome/shell/window-switcher/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="app-icon-mode" enum="org.gnome.shell.window-switcher.AppIconMode"> | ||||
|       <default>'both'</default> | ||||
|       <_summary>The application icon mode.</_summary> | ||||
|       <_description> | ||||
| 	Configures how the windows are shown in the switcher. Valid possibilities | ||||
| 	are 'thumbnail-only' (shows a thumbnail of the window), 'app-icon-only' | ||||
| 	(shows only the application icon) or 'both'. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key type="b" name="current-workspace-only"> | ||||
|       <default>false</default> | ||||
|       <summary>Limit switcher to current workspace.</summary> | ||||
|       <description> | ||||
| 	If true, only windows from the current workspace are shown in the switcher. | ||||
| 	Otherwise, all windows are included. | ||||
|       </description> | ||||
|     </key> | ||||
|   </schema> | ||||
| 
 | ||||
|   <schema id="org.gnome.shell.overrides" path="/org/gnome/shell/overrides/" | ||||
| 	  gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|   <schema id="org.gnome.shell.overrides" path="/org/gnome/shell/overrides/"> | ||||
|     <key name="attach-modal-dialogs" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary>Attach modal dialog to the parent window</_summary> | ||||
|       <_description> | ||||
|       <summary>Attach modal dialog to the parent window</summary> | ||||
|       <description> | ||||
|         This key overrides the key in org.gnome.mutter when running | ||||
|         GNOME Shell. | ||||
|       </_description> | ||||
|       </description> | ||||
|     </key> | ||||
| 
 | ||||
|     <key name="button-layout" type="s"> | ||||
|       <default>":close"</default> | ||||
|       <_summary>Arrangement of buttons on the titlebar</_summary> | ||||
|       <_description> | ||||
|       <summary>Arrangement of buttons on the titlebar</summary> | ||||
|       <description> | ||||
|         This key overrides the key in org.gnome.desktop.wm.preferences when | ||||
|         running GNOME Shell. | ||||
|       </_description> | ||||
|       </description> | ||||
|     </key> | ||||
| 
 | ||||
|     <key name="edge-tiling" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary>Enable edge tiling when dropping windows on screen edges</_summary> | ||||
|       <_description> | ||||
|       <summary>Enable edge tiling when dropping windows on screen edges</summary> | ||||
|       <description> | ||||
|         This key overrides the key in org.gnome.mutter when running GNOME Shell. | ||||
|       </_description> | ||||
|       </description> | ||||
|     </key> | ||||
| 
 | ||||
|     <key name="dynamic-workspaces" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary>Workspaces are managed dynamically</_summary> | ||||
|       <_description> | ||||
|       <summary>Workspaces are managed dynamically</summary> | ||||
|       <description> | ||||
|         This key overrides the key in org.gnome.mutter when running GNOME Shell. | ||||
|       </_description> | ||||
|       </description> | ||||
|     </key> | ||||
| 
 | ||||
|     <key name="workspaces-only-on-primary" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary>Workspaces only on primary monitor</_summary> | ||||
|       <_description> | ||||
|         This key overrides the key in org.gnome.mutter when running GNOME Shell. | ||||
|       </_description> | ||||
|     </key> | ||||
| 
 | ||||
|     <key name="focus-change-on-pointer-rest" type="b"> | ||||
|       <default>true</default> | ||||
|       <summary>Delay focus changes in mouse mode until the pointer stops moving</summary> | ||||
|       <summary>Workspaces only on primary monitor</summary> | ||||
|       <description> | ||||
|         This key overrides the key in org.gnome.mutter when running GNOME Shell. | ||||
|       </description> | ||||
							
								
								
									
										27
									
								
								data/shaders/dim-window.glsl
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| #version 110 | ||||
| uniform sampler2D tex; | ||||
| uniform float fraction; | ||||
| uniform float height; | ||||
| const float c = -0.2; | ||||
| const float border_max_height = 60.0; | ||||
|  | ||||
| mat4 contrast = mat4 (1.0 + c, 0.0, 0.0, 0.0, | ||||
|                       0.0, 1.0 + c, 0.0, 0.0, | ||||
|                       0.0, 0.0, 1.0 + c, 0.0, | ||||
|                       0.0, 0.0, 0.0, 1.0); | ||||
| vec4 off = vec4(0.633, 0.633, 0.633, 0); | ||||
| void main() | ||||
| { | ||||
|   vec4 color = texture2D(tex, cogl_tex_coord_in[0].xy); | ||||
|   float y = height * cogl_tex_coord_in[0].y; | ||||
|  | ||||
|   // To reduce contrast, blend with a mid gray | ||||
|   cogl_color_out = color * contrast - off * c * color.a; | ||||
|  | ||||
|   // We only fully dim at a distance of BORDER_MAX_HEIGHT from the top and | ||||
|   // when the fraction is 1.0. For other locations and fractions we linearly | ||||
|   // interpolate back to the original undimmed color, so the top of the window | ||||
|   // is at full color. | ||||
|   cogl_color_out = color + (cogl_color_out - color) * max(min(y / border_max_height, 1.0), 0.0); | ||||
|   cogl_color_out = color + (cogl_color_out - color) * fraction; | ||||
| } | ||||
| @@ -10,11 +10,11 @@ | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="29" | ||||
|    height="29" | ||||
|    width="28" | ||||
|    height="25" | ||||
|    id="svg10621" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.2 r9819" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="calendar-today.svg"> | ||||
|   <defs | ||||
|      id="defs10623"> | ||||
| @@ -118,17 +118,6 @@ | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3113" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
| @@ -138,29 +127,20 @@ | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="15.839192" | ||||
|      inkscape:cx="20.652108" | ||||
|      inkscape:cy="11.839084" | ||||
|      inkscape:cx="8.3750933" | ||||
|      inkscape:cy="8.0837211" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      showgrid="false" | ||||
|      fit-margin-top="0" | ||||
|      fit-margin-left="0" | ||||
|      fit-margin-right="0" | ||||
|      fit-margin-bottom="0" | ||||
|      inkscape:window-width="1280" | ||||
|      inkscape:window-height="741" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid3109" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10626"> | ||||
|     <rdf:RDF> | ||||
| @@ -177,28 +157,31 @@ | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-469.08263,-532.99307)"> | ||||
|     <path | ||||
|        sodipodi:type="arc" | ||||
|        style="opacity:0.4625;color:#000000;fill:url(#radialGradient3113);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:3;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="path34506-3" | ||||
|        sodipodi:cx="51" | ||||
|        sodipodi:cy="30" | ||||
|        sodipodi:rx="42" | ||||
|        sodipodi:ry="16" | ||||
|        d="M 9,29.999999 A 42,16 0 0 1 93,30 l -42,0 z" | ||||
|        sodipodi:start="3.1415927" | ||||
|        sodipodi:end="6.2831853" | ||||
|        transform="matrix(0.43692393,0,0,1.3783114,461.29951,517.6437)" | ||||
|        inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/motion/textures/panel.png" | ||||
|        inkscape:export-xdpi="90" | ||||
|        inkscape:export-ydpi="90" /> | ||||
|     <rect | ||||
|        style="fill:#ffffff;fill-opacity:0.50196078;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect2996" | ||||
|        width="31" | ||||
|        height="3" | ||||
|        x="468.08264" | ||||
|        y="558.99304" /> | ||||
|      transform="translate(-469.08263,-536.99307)"> | ||||
|     <g | ||||
|        id="g3003"> | ||||
|       <path | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/motion/textures/panel.png" | ||||
|          transform="matrix(0.43692393,0,0,1.3783114,460.60467,517.48289)" | ||||
|          sodipodi:end="6.2831853" | ||||
|          sodipodi:start="3.1415927" | ||||
|          d="M 9,29.999999 C 9.0000011,21.163443 27.804042,14 51.000002,14 74.195961,14 93,21.163444 93,30 l -42,0 z" | ||||
|          sodipodi:ry="16" | ||||
|          sodipodi:rx="42" | ||||
|          sodipodi:cy="30" | ||||
|          sodipodi:cx="51" | ||||
|          id="path34506-3" | ||||
|          style="opacity:0.4625;color:#000000;fill:url(#radialGradient2997);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:3;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          sodipodi:type="arc" /> | ||||
|       <rect | ||||
|          y="558.85046" | ||||
|          x="468.96878" | ||||
|          height="3.1425927" | ||||
|          width="28.149134" | ||||
|          id="rect2996" | ||||
|          style="fill:#ffffff;fill-opacity:0.50196078;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 5.7 KiB | 
							
								
								
									
										180
									
								
								data/theme/gdm.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,180 @@ | ||||
| /* Copyright 2011, Red Hat, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify it | ||||
|  * under the terms and conditions of the GNU Lesser General Public License, | ||||
|  * version 2.1, as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope it will be useful, but WITHOUT ANY | ||||
|  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
|  * FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for | ||||
|  * more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU Lesser General Public License | ||||
|  * along with this program; if not, write to the Free Software Foundation, | ||||
|  * Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||
|  */ | ||||
|  | ||||
| /* Login Dialog */ | ||||
|  | ||||
| .login-dialog-title { | ||||
|     font-size: 14pt; | ||||
|     font-weight: bold; | ||||
|     color: #666666; | ||||
|     padding-bottom: 2em; | ||||
| } | ||||
|  | ||||
| .login-dialog { | ||||
|     border-radius: 16px; | ||||
|     min-height: 150px; | ||||
|     max-height: 700px; | ||||
|     min-width: 350px; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-fingerprint-message { | ||||
|     font-size: 10.5pt; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-view { | ||||
|     -st-vfade-offset: 1em; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list { | ||||
|     spacing: 12px; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item { | ||||
|     color: #666666; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item:ltr { | ||||
|     padding-right: 1em; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item:rtl { | ||||
|     padding-left: 1em; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item .login-dialog-user-list-item-name { | ||||
|     font-size: 20pt; | ||||
|     padding-left: 1em; | ||||
|     color: #666666; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item:hover .login-dialog-user-list-item-name { | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item:focus .login-dialog-user-list-item-name { | ||||
|     color: white; | ||||
|     text-shadow: black 0px 2px 2px; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item-vertical-layout { | ||||
|     spacing: 2px; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item .login-dialog-user-list-item-focus-bin { | ||||
|     background-color: rgba(0,0,0,0.0); | ||||
|     height: 2px; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item:focus .login-dialog-user-list-item-focus-bin { | ||||
|     background-color: #666666; | ||||
| } | ||||
|  | ||||
| .login-dialog-user-list-item-icon { | ||||
|     border: 2px solid #8b8b8b; | ||||
|     border-radius: 8px; | ||||
|     width: 64px; | ||||
|     height: 64px; | ||||
| } | ||||
|  | ||||
| .login-dialog-not-listed-button { | ||||
|     padding-top: 2em; | ||||
| } | ||||
| .login-dialog-not-listed-label { | ||||
|     font-size: 14pt; | ||||
|     font-weight: bold; | ||||
|     color: #666666; | ||||
| } | ||||
|  | ||||
| .login-dialog-not-listed-button:hover .login-dialog-not-listed-label { | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-layout { | ||||
|     padding-bottom: 32px; | ||||
| } | ||||
| .login-dialog-prompt-label { | ||||
|     color: white; | ||||
|     font-size: 20pt; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-entry { | ||||
|     padding: 4px; | ||||
|     border-radius: 4px; | ||||
|     border: 2px solid #5656cc; | ||||
|     color: black; | ||||
|     background-color: white; | ||||
|     caret-color: black; | ||||
|     caret-size: 1px; | ||||
|     width: 15em; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-entry .capslock-warning { | ||||
|     icon-size: 16px; | ||||
|     warning-color: #999; | ||||
| } | ||||
|  | ||||
| .login-dialog-prompt-entry:insensitive { | ||||
|     color: rgba(0,0,0,0.7); | ||||
|     border: 2px solid #565656; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list { | ||||
|     color: #ffffff; | ||||
|     font-size: 10.5pt; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button { | ||||
|     padding: 4px; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button:focus { | ||||
|     background-color: #4c4c4c; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button:active { | ||||
|     background-color: #4c4c4c; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-button:hover { | ||||
|     font-weight: bold; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-scroll-view { | ||||
|     background-gradient-start: rgba(80,80,80,0.3); | ||||
|     background-gradient-end: rgba(80,80,80,0.7); | ||||
|     background-gradient-direction: vertical; | ||||
|     box-shadow: inset 0px 2px 4px rgba(0,0,0,0.9); | ||||
|     border-radius: 8px; | ||||
|     border: 1px solid rgba(80,80,80,1.0); | ||||
|     padding: .5em; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-item:focus { | ||||
|     background-color: #666666; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-triangle { | ||||
|     padding-right: .5em; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-item-box { | ||||
|     spacing: .25em; | ||||
| } | ||||
|  | ||||
| .login-dialog-session-list-item-dot { | ||||
|     width: .75em; | ||||
|     height: .75em; | ||||
| } | ||||
| @@ -1,130 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="300" | ||||
|    height="80" | ||||
|    id="svg7355" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.2 r9819" | ||||
|    sodipodi:docname="logged-in-indicator.svg"> | ||||
|   <metadata | ||||
|      id="metadata4175"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <sodipodi:namedview | ||||
|      pagecolor="#2c1cff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1" | ||||
|      objecttolerance="10" | ||||
|      gridtolerance="10" | ||||
|      guidetolerance="10" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      id="namedview4173" | ||||
|      showgrid="false" | ||||
|      inkscape:zoom="2.8760889" | ||||
|      inkscape:cx="106.00403" | ||||
|      inkscape:cy="80.68078" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:current-layer="g30864" /> | ||||
|   <defs | ||||
|      id="defs7357"> | ||||
|     <radialGradient | ||||
|        xlink:href="#linearGradient36429" | ||||
|        id="radialGradient7461" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.5919312,0,0,0.57582113,-20.687059,48.400487)" | ||||
|        cx="47.428951" | ||||
|        cy="167.16817" | ||||
|        fx="47.428951" | ||||
|        fy="167.16817" | ||||
|        r="37" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient36429"> | ||||
|       <stop | ||||
|          id="stop36431" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          id="stop36433" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        xlink:href="#linearGradient36471" | ||||
|        id="radialGradient7463" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1891549,0,0,0.55513246,-9.281289,36.12653)" | ||||
|        cx="49.067139" | ||||
|        cy="242.50381" | ||||
|        fx="49.067139" | ||||
|        fy="242.50381" | ||||
|        r="37.00671" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient36471"> | ||||
|       <stop | ||||
|          id="stop36473" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          id="stop36475" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        r="37.00671" | ||||
|        fy="242.50381" | ||||
|        fx="49.067139" | ||||
|        cy="242.50381" | ||||
|        cx="49.067139" | ||||
|        gradientTransform="matrix(3.4218418,0,0,0.03365337,-61.309005,138.5071)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="radialGradient7488" | ||||
|        xlink:href="#linearGradient36471" /> | ||||
|   </defs> | ||||
|   <g | ||||
|      id="layer1" | ||||
|      transform="matrix(1.6213276,0,0,1.6213276,-431.6347,-272.5745)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        id="g30864" | ||||
|        transform="translate(255.223,70.118091)"> | ||||
|       <rect | ||||
|          ry="3.4593496" | ||||
|          rx="8.8641119" | ||||
|          y="76.159348" | ||||
|          x="12.596948" | ||||
|          height="71.116341" | ||||
|          width="182.22595" | ||||
|          id="rect14000" | ||||
|          style="opacity:0.371875;fill:url(#radialGradient7461);fill-opacity:1;stroke:none" /> | ||||
|       <path | ||||
|          id="rect34520" | ||||
|          d="m 194.80022,146.83551 -182.559919,0" | ||||
|          style="opacity:0.35;fill:none;stroke:url(#radialGradient7488);stroke-width:0.61184424;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|          connector-curvature="0" | ||||
|          inkscape:connector-curvature="0" | ||||
|          sodipodi:nodetypes="cc" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.8 KiB | 
| Before Width: | Height: | Size: 25 KiB | 
| @@ -1,109 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="16" | ||||
|    height="16" | ||||
|    id="svg12430" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.3.1 r9886" | ||||
|    sodipodi:docname="more-results.svg"> | ||||
|   <defs | ||||
|      id="defs12432" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#7a7a7a" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="8.3155237" | ||||
|      inkscape:cy="0.89548874" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="g14642-3-0" | ||||
|      showgrid="false" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1376" | ||||
|      inkscape:window-x="1200" | ||||
|      inkscape:window-y="187" | ||||
|      inkscape:window-maximized="1"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid13002" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata12435"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(0,-1036.3622)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        transform="translate(-141.99984,638.37113)" | ||||
|        inkscape:label="zoom-in" | ||||
|        id="g14642-3-0"> | ||||
|       <path | ||||
|          sodipodi:type="inkscape:offset" | ||||
|          inkscape:radius="0" | ||||
|          inkscape:original="M 145.1875 400 C 144.5248 400 144 400.54899 144 401.21875 L 144 410.78125 C 144 411.45101 144.5248 412 145.1875 412 L 154.8125 412 C 155.4752 412 156 411.45101 156 410.78125 L 156 401.21875 C 156 400.54899 155.4752 400 154.8125 400 L 145.1875 400 z M 149 403 L 151 403 L 151 405 L 153 405 L 153 407 L 151 407 L 151 409 L 149 409 L 149 407 L 147 407 L 147 405 L 149 405 L 149 403 z " | ||||
|          xlink:href="#rect11749-5-0-1-8" | ||||
|          style="color:#bebebe;fill:#000000;fill-opacity:1;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible;opacity:0.8" | ||||
|          id="path13004" | ||||
|          inkscape:href="#rect11749-5-0-1-8" | ||||
|          d="M 145.1875,400 C 144.5248,400 144,400.54899 144,401.21875 l 0,9.5625 c 0,0.66976 0.5248,1.21875 1.1875,1.21875 l 9.625,0 c 0.6627,0 1.1875,-0.54899 1.1875,-1.21875 l 0,-9.5625 C 156,400.54899 155.4752,400 154.8125,400 L 145.1875,400 z m 3.8125,3 2,0 0,2 2,0 0,2 -2,0 0,2 -2,0 0,-2 -2,0 0,-2 2,0 L 149,403 Z" | ||||
|          transform="translate(0,1)" /> | ||||
|       <use | ||||
|          x="0" | ||||
|          y="0" | ||||
|          xlink:href="#path13004" | ||||
|          id="use11960" | ||||
|          transform="translate(1,-1)" | ||||
|          width="16" | ||||
|          height="16" /> | ||||
|       <use | ||||
|          x="0" | ||||
|          y="0" | ||||
|          xlink:href="#use11960" | ||||
|          id="use11962" | ||||
|          transform="translate(-2,0)" | ||||
|          width="16" | ||||
|          height="16" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          style="color:#bebebe;fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          d="M 145.1875,400 C 144.5248,400 144,400.54899 144,401.21875 l 0,9.5625 c 0,0.66976 0.5248,1.21875 1.1875,1.21875 l 9.625,0 c 0.6627,0 1.1875,-0.54899 1.1875,-1.21875 l 0,-9.5625 C 156,400.54899 155.4752,400 154.8125,400 L 145.1875,400 z m 3.8125,3 2,0 0,2 2,0 0,2 -2,0 0,2 -2,0 0,-2 -2,0 0,-2 2,0 L 149,403 Z" | ||||
|          id="rect11749-5-0-1-8" /> | ||||
|       <rect | ||||
|          style="fill:none;stroke:none" | ||||
|          id="rect3620-5-4" | ||||
|          width="15.981825" | ||||
|          height="16" | ||||
|          x="142" | ||||
|          y="398" | ||||
|          rx="0" | ||||
|          ry="0" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 4.1 KiB | 
| Before Width: | Height: | Size: 78 KiB | 
| @@ -9,7 +9,7 @@ | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="17" | ||||
|    width="21" | ||||
|    height="10" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
| @@ -66,9 +66,9 @@ | ||||
|     <rect | ||||
|        style="opacity:0.8;fill:#ffffff;fill-opacity:1;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect3796" | ||||
|        width="7" | ||||
|        width="3" | ||||
|        height="2" | ||||
|        x="5" | ||||
|        x="9" | ||||
|        y="8" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 2.0 KiB After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										64
									
								
								data/theme/scroll-hhandle.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,64 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="10" | ||||
|    height="4" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="scroll-hhandle.svg"> | ||||
|   <defs | ||||
|      id="defs4"> | ||||
|   </defs> | ||||
|   <metadata | ||||
|      id="metadata7"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1"> | ||||
|     <rect | ||||
|        style="fill:#323232;fill-opacity:1;fill-rule:evenodd;stroke:none" | ||||
|        id="rect3592" | ||||
|        width="2" | ||||
|        height="4" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        rx="0" | ||||
|        ry="0" /> | ||||
|     <use | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#rect3592" | ||||
|        id="use2825" | ||||
|        transform="translate(8,0)" | ||||
|        width="10" | ||||
|        height="4" /> | ||||
|     <use | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use2825" | ||||
|        id="use2827" | ||||
|        transform="translate(-4,0)" | ||||
|        width="10" | ||||
|        height="4" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										62
									
								
								data/theme/scroll-vhandle.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,62 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="4" | ||||
|    height="10" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="scroll-hhandle.svg"> | ||||
|   <metadata | ||||
|      id="metadata7"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1"> | ||||
|     <rect | ||||
|        style="fill:#323232;fill-opacity:1;fill-rule:evenodd;stroke:none" | ||||
|        id="rect3592" | ||||
|        width="2" | ||||
|        height="4" | ||||
|        x="0" | ||||
|        y="-4" | ||||
|        rx="0" | ||||
|        ry="0" | ||||
|        transform="matrix(0,1,-1,0,0,0)" /> | ||||
|     <use | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#rect3592" | ||||
|        id="use3705" | ||||
|        transform="translate(0,4)" | ||||
|        width="4" | ||||
|        height="10" /> | ||||
|     <use | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use3705" | ||||
|        id="use3707" | ||||
|        transform="translate(0,4)" | ||||
|        width="4" | ||||
|        height="10" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
| @@ -1,120 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Generator: Adobe Illustrator 13.0.2, SVG Export Plug-In . SVG Version: 6.00 Build 14948)  --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    version="1.0" | ||||
|    id="Foreground" | ||||
|    x="0px" | ||||
|    y="0px" | ||||
|    width="32" | ||||
|    height="32" | ||||
|    viewBox="0 0 23.272727 23.272727" | ||||
|    enable-background="new 0 0 16 16" | ||||
|    xml:space="preserve" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.48.2 r9819" | ||||
|    sodipodi:docname="summary-counter.svg" | ||||
|    inkscape:output_extension="org.inkscape.output.svg.inkscape"><metadata | ||||
|      id="metadata2399"><rdf:RDF><cc:Work | ||||
|          rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /><dc:title /></cc:Work></rdf:RDF></metadata><defs | ||||
|      id="defs2397"><linearGradient | ||||
|        id="linearGradient3173"><stop | ||||
|          style="stop-color:#c4c4c4;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop3175" /><stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="1" | ||||
|          id="stop3177" /></linearGradient><inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 8 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="16 : 8 : 1" | ||||
|        inkscape:persp3d-origin="8 : 5.3333333 : 1" | ||||
|        id="perspective2401" /><filter | ||||
|        color-interpolation-filters="sRGB" | ||||
|        inkscape:collect="always" | ||||
|        id="filter16494-4" | ||||
|        x="-0.20989846" | ||||
|        width="1.4197969" | ||||
|        y="-0.20903821" | ||||
|        height="1.4180764"><feGaussianBlur | ||||
|          inkscape:collect="always" | ||||
|          stdDeviation="1.3282637" | ||||
|          id="feGaussianBlur16496-8" /></filter><radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient16498-6" | ||||
|        id="radialGradient16504-1" | ||||
|        cx="7.6582627" | ||||
|        cy="5.8191104" | ||||
|        fx="7.6582627" | ||||
|        fy="5.8191104" | ||||
|        r="8.6928644" | ||||
|        gradientTransform="matrix(1.0474339,0,0,1.0517402,-0.3632615,-0.42032492)" | ||||
|        gradientUnits="userSpaceOnUse" /><linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient16498-6"><stop | ||||
|          style="stop-color:#9FD0FF;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop16500-8" /><stop | ||||
|          style="stop-color:#3465A4;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop16502-0" /></linearGradient></defs><sodipodi:namedview | ||||
|      inkscape:window-height="709" | ||||
|      inkscape:window-width="1366" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:pageopacity="0" | ||||
|      guidetolerance="10.0" | ||||
|      gridtolerance="10.0" | ||||
|      objecttolerance="10.0" | ||||
|      borderopacity="1.0" | ||||
|      bordercolor="#666666" | ||||
|      pagecolor="#000000" | ||||
|      id="base" | ||||
|      showgrid="false" | ||||
|      inkscape:zoom="11.313708" | ||||
|      inkscape:cx="15.386407" | ||||
|      inkscape:cy="13.739577" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="1179" | ||||
|      inkscape:current-layer="g16402-8" | ||||
|      showguides="true" | ||||
|      inkscape:guide-bbox="true" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:window-maximized="1"><inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid11246" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /></sodipodi:namedview><g | ||||
|      style="display:inline" | ||||
|      id="g16402-8" | ||||
|      transform="translate(4.7533483,2.8238929)"><g | ||||
|        id="g3175-4" | ||||
|        transform="translate(-0.89995416,0.94028614)"><path | ||||
|          sodipodi:type="inkscape:offset" | ||||
|          inkscape:radius="0" | ||||
|          inkscape:original="M 7.65625 0.125 C 3.2589349 0.125 -0.3125 3.7070002 -0.3125 8.125 C -0.3125 12.543001 3.2589349 16.125 7.65625 16.125 C 12.053566 16.125 15.625 12.543001 15.625 8.125 C 15.625 3.7070002 12.053566 0.125 7.65625 0.125 z " | ||||
|          xlink:href="#path2394-32" | ||||
|          style="opacity:0.52994014;color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:2.18181825;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;filter:url(#filter16494-4);enable-background:accumulate" | ||||
|          id="path16480-5" | ||||
|          inkscape:href="#path2394-32" | ||||
|          d="m 7.65625,0.125 c -4.3973151,0 -7.96875,3.5820002 -7.96875,8 0,4.418001 3.5714349,8 7.96875,8 4.397316,0 7.96875,-3.581999 7.96875,-8 0,-4.4179998 -3.571434,-8 -7.96875,-8 z" | ||||
|          transform="translate(0,1.028519)" /><path | ||||
|          clip-rule="evenodd" | ||||
|          d="m -0.30428257,8.1237596 c 0,-4.4179998 3.56522987,-7.9999996 7.96254497,-7.9999996 4.3973156,0 7.9625456,3.5819998 7.9625456,7.9999996 0,4.4180014 -3.56523,8.0000004 -7.9625456,8.0000004 -4.3973151,0 -7.96254497,-3.581999 -7.96254497,-8.0000004 z" | ||||
|          id="path2394-32" | ||||
|          style="color:#000000;fill:url(#radialGradient16504-1);fill-opacity:1;fill-rule:nonzero;stroke:#eeeeec;stroke-width:1.4545455;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          sodipodi:nodetypes="csssc" | ||||
|          inkscape:connector-curvature="0" /><g | ||||
|          id="g3172-6" /></g></g></svg> | ||||
| Before Width: | Height: | Size: 5.4 KiB | 
| @@ -9,7 +9,7 @@ | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="65" | ||||
|    width="64" | ||||
|    height="22" | ||||
|    id="svg3273" | ||||
|    version="1.1" | ||||
|   | ||||
| Before Width: | Height: | Size: 4.7 KiB After Width: | Height: | Size: 4.7 KiB | 
| @@ -9,7 +9,7 @@ | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="65" | ||||
|    width="64" | ||||
|    height="22" | ||||
|    id="svg3012" | ||||
|    version="1.1" | ||||
|   | ||||
| Before Width: | Height: | Size: 7.2 KiB After Width: | Height: | Size: 7.2 KiB | 
| Before Width: | Height: | Size: 850 B | 
							
								
								
									
										376
									
								
								data/theme/ws-switch-arrow-down.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,376 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    version="1.1" | ||||
|    width="96" | ||||
|    height="96" | ||||
|    id="svg25070" | ||||
|    inkscape:version="0.48.0 r9654" | ||||
|    sodipodi:docname="ws-switch-arrow-down.svg"> | ||||
|   <metadata | ||||
|      id="metadata3353"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <sodipodi:namedview | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1" | ||||
|      objecttolerance="10" | ||||
|      gridtolerance="10" | ||||
|      guidetolerance="10" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:window-width="718" | ||||
|      inkscape:window-height="480" | ||||
|      id="namedview3351" | ||||
|      showgrid="false" | ||||
|      inkscape:zoom="2.6979167" | ||||
|      inkscape:cx="48" | ||||
|      inkscape:cy="48" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" | ||||
|      inkscape:current-layer="svg25070" /> | ||||
|   <defs | ||||
|      id="defs25072"> | ||||
|     <linearGradient | ||||
|        x1="-86.552246" | ||||
|        y1="185.439" | ||||
|        x2="-83.37072" | ||||
|        y2="197.31261" | ||||
|        id="linearGradient24957" | ||||
|        xlink:href="#linearGradient4034-0-4" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="translate(6,0)" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4034-0-4"> | ||||
|       <stop | ||||
|          id="stop4036-5-7" | ||||
|          style="stop-color:#eeeeec;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4038-9-6" | ||||
|          style="stop-color:#babdb6;stop-opacity:1" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter24765"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix24767" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix24769" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        x1="-74.520325" | ||||
|        y1="169.06032" | ||||
|        x2="-74.520325" | ||||
|        y2="205.94189" | ||||
|        id="linearGradient24955" | ||||
|        xlink:href="#linearGradient4632-1-3-9-3-2" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="translate(-5,0)" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4632-1-3-9-3-2"> | ||||
|       <stop | ||||
|          id="stop4634-1-8-3-9-0" | ||||
|          style="stop-color:#eeeeec;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4636-1-9-9-8-8" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" | ||||
|          offset="0.0274937" /> | ||||
|       <stop | ||||
|          id="stop4638-8-3-9-6-6" | ||||
|          style="stop-color:#f2f2f2;stop-opacity:1" | ||||
|          offset="0.274937" /> | ||||
|       <stop | ||||
|          id="stop4640-8-5-7-8-9" | ||||
|          style="stop-color:#eeeeec;stop-opacity:1" | ||||
|          offset="0.38707438" /> | ||||
|       <stop | ||||
|          id="stop4642-5-41-9-6-9" | ||||
|          style="stop-color:#d9dad8;stop-opacity:1" | ||||
|          offset="0.66528589" /> | ||||
|       <stop | ||||
|          id="stop4644-5-2-7-9-2" | ||||
|          style="stop-color:#dfe0dd;stop-opacity:1" | ||||
|          offset="0.76745707" /> | ||||
|       <stop | ||||
|          id="stop4646-3-2-3-7-3" | ||||
|          style="stop-color:#f0f0f0;stop-opacity:1" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        cx="-33.412369" | ||||
|        cy="185.74171" | ||||
|        r="2.3554697" | ||||
|        fx="-33.412369" | ||||
|        fy="185.74171" | ||||
|        id="radialGradient24959" | ||||
|        xlink:href="#linearGradient4869-4-1" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0075,0,0,1.0075,-5.4544,-1.25141)" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4869-4-1"> | ||||
|       <stop | ||||
|          id="stop4871-6-2" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4879-7-4" | ||||
|          style="stop-color:#eeeeec;stop-opacity:1" | ||||
|          offset="0.31807542" /> | ||||
|       <stop | ||||
|          id="stop4877-6-1" | ||||
|          style="stop-color:#c8c9c6;stop-opacity:1" | ||||
|          offset="0.74691135" /> | ||||
|       <stop | ||||
|          id="stop4873-1-0" | ||||
|          style="stop-color:#d3d7cf;stop-opacity:1" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25011"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25013" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25015" /> | ||||
|     </filter> | ||||
|     <radialGradient | ||||
|        cx="-33.412369" | ||||
|        cy="185.74171" | ||||
|        r="2.3554697" | ||||
|        fx="-33.412369" | ||||
|        fy="185.74171" | ||||
|        id="radialGradient24961" | ||||
|        xlink:href="#linearGradient4869-4-0" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0075,0,0,1.0075,-5.4544,-1.25141)" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4869-4-0"> | ||||
|       <stop | ||||
|          id="stop4871-6-8" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4879-7-5" | ||||
|          style="stop-color:#eeeeec;stop-opacity:1" | ||||
|          offset="0.31807542" /> | ||||
|       <stop | ||||
|          id="stop4877-6-5" | ||||
|          style="stop-color:#c8c9c6;stop-opacity:1" | ||||
|          offset="0.74691135" /> | ||||
|       <stop | ||||
|          id="stop4873-1-4" | ||||
|          style="stop-color:#d3d7cf;stop-opacity:1" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25023"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25025" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25027" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        x1="-39.858727" | ||||
|        y1="184.61784" | ||||
|        x2="-38.244785" | ||||
|        y2="188.84898" | ||||
|        id="linearGradient24963" | ||||
|        xlink:href="#linearGradient4941" | ||||
|        gradientUnits="userSpaceOnUse" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4941"> | ||||
|       <stop | ||||
|          id="stop4943" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4945" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25033"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25035" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25037" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        x1="-39.858727" | ||||
|        y1="184.61784" | ||||
|        x2="-38.244785" | ||||
|        y2="188.84898" | ||||
|        id="linearGradient24965" | ||||
|        xlink:href="#linearGradient4941-7" | ||||
|        gradientUnits="userSpaceOnUse" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4941-7"> | ||||
|       <stop | ||||
|          id="stop4943-2" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" | ||||
|          offset="0" /> | ||||
|       <stop | ||||
|          id="stop4945-5" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="1" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25043"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25045" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25047" /> | ||||
|     </filter> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25049"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25051" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25053" /> | ||||
|     </filter> | ||||
|     <filter | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        color-interpolation-filters="sRGB" | ||||
|        id="filter25055"> | ||||
|       <feColorMatrix | ||||
|          result="fbSourceGraphic" | ||||
|          values="1" | ||||
|          type="saturate" | ||||
|          id="feColorMatrix25057" /> | ||||
|       <feColorMatrix | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" | ||||
|          in="fbSourceGraphic" | ||||
|          id="feColorMatrix25059" /> | ||||
|     </filter> | ||||
|   </defs> | ||||
|   <g | ||||
|      transform="matrix(0,1,-1,0,48.0003,4.1307112e-7)" | ||||
|      id="layer1"> | ||||
|     <g | ||||
|        transform="matrix(-2,0,0,2,-97.2497,-374.967)" | ||||
|        id="g4030-1-8" | ||||
|        style="stroke:#000000;stroke-opacity:1;display:inline"> | ||||
|       <path | ||||
|          d="m -72.5,173.5 -14,14 14,14" | ||||
|          id="path3165-7-3" | ||||
|          style="color:#000000;fill:none;stroke:#000000;stroke-width:7;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|     </g> | ||||
|     <path | ||||
|        d="m -36.5,186.40625 a 2.09375,2.09375 0 1 1 -4.1875,0 2.09375,2.09375 0 1 1 4.1875,0 z" | ||||
|        transform="matrix(-3.34328,0,0,3.34328,-89.2797,-623.176)" | ||||
|        id="path4050-2-7-9-4" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.52343899;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|     <path | ||||
|        d="m -36.5,186.40625 a 2.09375,2.09375 0 1 1 -4.1875,0 2.09375,2.09375 0 1 1 4.1875,0 z" | ||||
|        transform="matrix(-3.34328,0,0,3.34328,-111.2797,-623.176)" | ||||
|        id="path4050-2-7-9-4-8" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.52343899;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|     <path | ||||
|        d="m -36.5,186.40625 a 2.09375,2.09375 0 1 1 -4.1875,0 2.09375,2.09375 0 1 1 4.1875,0 z" | ||||
|        transform="matrix(-2.86565,0,0,2.86565,-70.8457,-534.143)" | ||||
|        id="path4050-2-7-9-4-0" | ||||
|        style="color:#000000;fill:none;stroke:#000000;stroke-width:0.69792098;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|     <path | ||||
|        d="m -36.5,186.40625 a 2.09375,2.09375 0 1 1 -4.1875,0 2.09375,2.09375 0 1 1 4.1875,0 z" | ||||
|        transform="matrix(-2.86565,0,0,2.86565,-92.8457,-534.143)" | ||||
|        id="path4050-2-7-9-4-0-9" | ||||
|        style="color:#000000;fill:none;stroke:#000000;stroke-width:0.69792098;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|     <path | ||||
|        d="m 47.87528,-34.0295 c 1.53896,0.0448 3.0511,0.70928 4.125,1.8125 l 32.25,32.25 -32.25,32.25 c -2.2253,2.2253 -6.2747,2.2253 -8.5,0 -2.2253,-2.22528 -2.2253,-6.2747 0,-8.5 l 23.75,-23.75 -23.75,-23.75 c -1.73168,-1.6731 -2.295,-4.44228 -1.3546,-6.65894 0.94042,-2.21668 3.32312,-3.73604 5.7296,-3.65356 z" | ||||
|        id="path3165-7-3-1" | ||||
|        style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0pt;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;text-anchor:start;opacity:0.35;color:#000000;fill:none;stroke:#000000;stroke-width:2;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;marker:none;visibility:visible;display:inline;overflow:visible;font-family:Bitstream Vera Sans" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|     <path | ||||
|        d="m 41.8316,28.09418 c -0.014,-1.58898 0.54158,-3.18406 1.66868,-4.31118 l 23.75,-23.75 m -25.1046,-30.40894 c 0.94042,-2.21668 3.32312,-3.73604 5.7296,-3.65356 1.53896,0.0448 3.0511,0.70928 4.125,1.8125 l 32.25,32.25" | ||||
|        id="path3165-7-3-1-9" | ||||
|        style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0pt;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;text-anchor:start;color:#000000;fill:none;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;marker:none;visibility:visible;display:inline;overflow:visible;font-family:Bitstream Vera Sans" | ||||
|        inkscape:connector-curvature="0" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 13 KiB | 
| Before Width: | Height: | Size: 841 B | 
							
								
								
									
										447
									
								
								data/theme/ws-switch-arrow-up.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,447 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="96" | ||||
|    height="96" | ||||
|    id="svg25070" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.0 r9654" | ||||
|    sodipodi:docname="ws-switch-arrow-up.svg"> | ||||
|   <defs | ||||
|      id="defs25072"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 24 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="48 : 24 : 1" | ||||
|        inkscape:persp3d-origin="24 : 16 : 1" | ||||
|        id="perspective25078" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective24985" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4034-0-4" | ||||
|        id="linearGradient24957" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="translate(6)" | ||||
|        x1="-86.552246" | ||||
|        y1="185.439" | ||||
|        x2="-83.37072" | ||||
|        y2="197.31261" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient4034-0-4"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(238, 238, 236); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4036-5-7" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(186, 189, 182); stop-opacity: 1;" | ||||
|          offset="1" | ||||
|          id="stop4038-9-6" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        id="filter24765" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix24767" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix24769" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4632-1-3-9-3-2" | ||||
|        id="linearGradient24955" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="translate(-5)" | ||||
|        x1="-74.520325" | ||||
|        y1="169.06032" | ||||
|        x2="-74.520325" | ||||
|        y2="205.94189" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4632-1-3-9-3-2"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(238, 238, 236); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4634-1-8-3-9-0" /> | ||||
|       <stop | ||||
|          id="stop4636-1-9-9-8-8" | ||||
|          offset="0.0274937" | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4638-8-3-9-6-6" | ||||
|          offset="0.274937" | ||||
|          style="stop-color: rgb(242, 242, 242); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4640-8-5-7-8-9" | ||||
|          offset="0.38707438" | ||||
|          style="stop-color: rgb(238, 238, 236); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4642-5-41-9-6-9" | ||||
|          offset="0.66528589" | ||||
|          style="stop-color: rgb(217, 218, 216); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4644-5-2-7-9-2" | ||||
|          offset="0.76745707" | ||||
|          style="stop-color: rgb(223, 224, 221); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(240, 240, 240); stop-opacity: 1;" | ||||
|          offset="1" | ||||
|          id="stop4646-3-2-3-7-3" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4869-4-1" | ||||
|        id="radialGradient24959" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0075, 0, 0, 1.0075, -5.4544, -1.25141)" | ||||
|        cx="-33.412369" | ||||
|        cy="185.74171" | ||||
|        fx="-33.412369" | ||||
|        fy="185.74171" | ||||
|        r="2.3554697" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4869-4-1"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4871-6-2" /> | ||||
|       <stop | ||||
|          id="stop4879-7-4" | ||||
|          offset="0.31807542" | ||||
|          style="stop-color: rgb(238, 238, 236); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4877-6-1" | ||||
|          offset="0.74691135" | ||||
|          style="stop-color: rgb(200, 201, 198); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(211, 215, 207); stop-opacity: 1;" | ||||
|          offset="1" | ||||
|          id="stop4873-1-0" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        id="filter25011" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25013" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25015" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4869-4-0" | ||||
|        id="radialGradient24961" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0075, 0, 0, 1.0075, -5.4544, -1.25141)" | ||||
|        cx="-33.412369" | ||||
|        cy="185.74171" | ||||
|        fx="-33.412369" | ||||
|        fy="185.74171" | ||||
|        r="2.3554697" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient4869-4-0"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4871-6-8" /> | ||||
|       <stop | ||||
|          id="stop4879-7-5" | ||||
|          offset="0.31807542" | ||||
|          style="stop-color: rgb(238, 238, 236); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          id="stop4877-6-5" | ||||
|          offset="0.74691135" | ||||
|          style="stop-color: rgb(200, 201, 198); stop-opacity: 1;" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(211, 215, 207); stop-opacity: 1;" | ||||
|          offset="1" | ||||
|          id="stop4873-1-4" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        id="filter25023" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25025" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25027" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4941" | ||||
|        id="linearGradient24963" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        x1="-39.858727" | ||||
|        y1="184.61784" | ||||
|        x2="-38.244785" | ||||
|        y2="188.84898" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient4941"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4943" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 0;" | ||||
|          offset="1" | ||||
|          id="stop4945" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        id="filter25033" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25035" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25037" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient4941-7" | ||||
|        id="linearGradient24965" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        x1="-39.858727" | ||||
|        y1="184.61784" | ||||
|        x2="-38.244785" | ||||
|        y2="188.84898" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient4941-7"> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 1;" | ||||
|          offset="0" | ||||
|          id="stop4943-2" /> | ||||
|       <stop | ||||
|          style="stop-color: rgb(255, 255, 255); stop-opacity: 0;" | ||||
|          offset="1" | ||||
|          id="stop4945-5" /> | ||||
|     </linearGradient> | ||||
|     <filter | ||||
|        id="filter25043" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25045" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25047" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <filter | ||||
|        id="filter25049" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25051" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25053" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|     <filter | ||||
|        id="filter25055" | ||||
|        inkscape:label="Invert" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        width="1" | ||||
|        height="1" | ||||
|        inkscape:menu="Color" | ||||
|        inkscape:menu-tooltip="Invert colors" | ||||
|        color-interpolation-filters="sRGB"> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25057" | ||||
|          type="saturate" | ||||
|          values="1" | ||||
|          result="fbSourceGraphic" /> | ||||
|       <feColorMatrix | ||||
|          id="feColorMatrix25059" | ||||
|          in="fbSourceGraphic" | ||||
|          values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /> | ||||
|     </filter> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="2.8284271" | ||||
|      inkscape:cx="-12.356322" | ||||
|      inkscape:cy="57.536221" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1200" | ||||
|      inkscape:window-height="840" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata25075"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      id="layer1" | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      transform="translate(0, 48)"> | ||||
|     <g | ||||
|        id="g3181" | ||||
|        transform="matrix(0,1,-1,0,48.0003,-48)"> | ||||
|       <g | ||||
|          style="stroke:#000000;stroke-opacity:1;display:inline" | ||||
|          transform="matrix(2,0,0,2,193.25,-374.967)" | ||||
|          id="g4030-1-8"> | ||||
|         <path | ||||
|            style="color:#000000;fill:none;stroke:#000000;stroke-width:7;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|            d="m -72.5,173.5 -14,14 14,14" | ||||
|            id="path3165-7-3" | ||||
|            sodipodi:nodetypes="ccc" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|       </g> | ||||
|       <path | ||||
|          transform="matrix(3.34328,0,0,3.34328,185.28,-623.176)" | ||||
|          d="m -36.5,186.40625 c 0,1.15635 -0.937404,2.09375 -2.09375,2.09375 -1.156346,0 -2.09375,-0.9374 -2.09375,-2.09375 0,-1.15635 0.937404,-2.09375 2.09375,-2.09375 1.156346,0 2.09375,0.9374 2.09375,2.09375 z" | ||||
|          sodipodi:ry="2.09375" | ||||
|          sodipodi:rx="2.09375" | ||||
|          sodipodi:cy="186.40625" | ||||
|          sodipodi:cx="-38.59375" | ||||
|          id="path4050-2-7-9-4" | ||||
|          style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.52343899;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          sodipodi:type="arc" /> | ||||
|       <path | ||||
|          transform="matrix(3.34328,0,0,3.34328,207.28,-623.176)" | ||||
|          d="m -36.5,186.40625 c 0,1.15635 -0.937404,2.09375 -2.09375,2.09375 -1.156346,0 -2.09375,-0.9374 -2.09375,-2.09375 0,-1.15635 0.937404,-2.09375 2.09375,-2.09375 1.156346,0 2.09375,0.9374 2.09375,2.09375 z" | ||||
|          sodipodi:ry="2.09375" | ||||
|          sodipodi:rx="2.09375" | ||||
|          sodipodi:cy="186.40625" | ||||
|          sodipodi:cx="-38.59375" | ||||
|          id="path4050-2-7-9-4-8" | ||||
|          style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.52343899;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          sodipodi:type="arc" /> | ||||
|       <path | ||||
|          transform="matrix(2.86565,0,0,2.86565,166.846,-534.143)" | ||||
|          d="m -36.5,186.40625 c 0,1.15635 -0.937404,2.09375 -2.09375,2.09375 -1.156346,0 -2.09375,-0.9374 -2.09375,-2.09375 0,-1.15635 0.937404,-2.09375 2.09375,-2.09375 1.156346,0 2.09375,0.9374 2.09375,2.09375 z" | ||||
|          sodipodi:ry="2.09375" | ||||
|          sodipodi:rx="2.09375" | ||||
|          sodipodi:cy="186.40625" | ||||
|          sodipodi:cx="-38.59375" | ||||
|          id="path4050-2-7-9-4-0" | ||||
|          style="color:#000000;fill:none;stroke:#000000;stroke-width:0.69792098;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          sodipodi:type="arc" /> | ||||
|       <path | ||||
|          transform="matrix(2.86565,0,0,2.86565,188.846,-534.143)" | ||||
|          d="m -36.5,186.40625 c 0,1.15635 -0.937404,2.09375 -2.09375,2.09375 -1.156346,0 -2.09375,-0.9374 -2.09375,-2.09375 0,-1.15635 0.937404,-2.09375 2.09375,-2.09375 1.156346,0 2.09375,0.9374 2.09375,2.09375 z" | ||||
|          sodipodi:ry="2.09375" | ||||
|          sodipodi:rx="2.09375" | ||||
|          sodipodi:cy="186.40625" | ||||
|          sodipodi:cx="-38.59375" | ||||
|          id="path4050-2-7-9-4-0-9" | ||||
|          style="color:#000000;fill:none;stroke:#000000;stroke-width:0.69792098;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible" | ||||
|          sodipodi:type="arc" /> | ||||
|       <path | ||||
|          transform="matrix(2,0,0,2,-586,-765.967)" | ||||
|          sodipodi:nodetypes="ccccscccsc" | ||||
|          id="path3165-7-3-1" | ||||
|          d="m 317.06251,365.96875 c -0.76948,0.0224 -1.52555,0.35464 -2.0625,0.90625 l -16.125,16.125 16.125,16.125 c 1.11265,1.11265 3.13735,1.11265 4.25,0 1.11265,-1.11264 1.11265,-3.13735 0,-4.25 l -11.875,-11.875 11.875,-11.875 c 0.86584,-0.83655 1.1475,-2.22114 0.6773,-3.32947 -0.47021,-1.10834 -1.66156,-1.86802 -2.8648,-1.82678 z" | ||||
|          style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0pt;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;text-anchor:start;opacity:0.35;color:#000000;fill:none;stroke:#000000;stroke-width:1;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;marker:none;visibility:visible;display:inline;overflow:visible;font-family:Bitstream Vera Sans" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|       <path | ||||
|          transform="matrix(2,0,0,2,-586,-765.967)" | ||||
|          sodipodi:nodetypes="ccccccc" | ||||
|          id="path3165-7-3-1-9" | ||||
|          d="m 320.08435,397.03059 c 0.007,-0.79449 -0.27079,-1.59203 -0.83434,-2.15559 L 307.37501,383 m 12.5523,-15.20447 c -0.47021,-1.10834 -1.66156,-1.86802 -2.8648,-1.82678 -0.76948,0.0224 -1.52555,0.35464 -2.0625,0.90625 L 298.87501,383" | ||||
|          style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0pt;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;text-anchor:start;color:#000000;fill:none;stroke:#000000;stroke-width:1;stroke-linecap:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;marker:none;visibility:visible;display:inline;overflow:visible;font-family:Bitstream Vera Sans" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 16 KiB | 
| @@ -68,38 +68,13 @@ IGNORE_HFILES=					\ | ||||
| 	gactionobserver.h			\ | ||||
| 	shell-recorder-src.h | ||||
|  | ||||
| if !BUILD_RECORDER | ||||
| IGNORE_HFILES += shell-recorder.h | ||||
| endif | ||||
|  | ||||
| # Images to copy into HTML directory. | ||||
| # e.g. HTML_IMAGES=$(top_srcdir)/gtk/stock-icons/stock_about_24.png | ||||
| HTML_IMAGES= | ||||
|  | ||||
| doc-gen-org.gnome.Shell.SearchProvider.xml: $(top_srcdir)/data/org.gnome.ShellSearchProvider.xml | ||||
| 	gdbus-codegen 						\ | ||||
| 	--interface-prefix org.gnome.ShellSearchProvider. 	\ | ||||
| 	--generate-docbook doc-gen				\ | ||||
| 	$(top_srcdir)/data/org.gnome.ShellSearchProvider.xml | ||||
|  | ||||
| doc-gen-org.gnome.Shell.SearchProvider2.xml: $(top_srcdir)/data/org.gnome.ShellSearchProvider2.xml | ||||
| 	gdbus-codegen 						\ | ||||
| 	--interface-prefix org.gnome.ShellSearchProvider2. 	\ | ||||
| 	--generate-docbook doc-gen				\ | ||||
| 	$(top_srcdir)/data/org.gnome.ShellSearchProvider2.xml | ||||
|  | ||||
| doc-gen-org.gnome.Shell.Screenshot.xml: $(top_srcdir)/data/org.gnome.Shell.Screenshot.xml | ||||
| 	gdbus-codegen 						\ | ||||
| 	--interface-prefix org.gnome.Shell.Screenshot.		\ | ||||
| 	--generate-docbook doc-gen				\ | ||||
| 	$(top_srcdir)/data/org.gnome.Shell.Screenshot.xml | ||||
|  | ||||
| # Extra SGML files that are included by $(DOC_MAIN_SGML_FILE). | ||||
| # e.g. content_files=running.sgml building.sgml changes-2.0.sgml | ||||
| content_files= \ | ||||
| 	doc-gen-org.gnome.Shell.SearchProvider.xml	\ | ||||
| 	doc-gen-org.gnome.Shell.SearchProvider2.xml	\ | ||||
| 	doc-gen-org.gnome.Shell.Screenshot.xml | ||||
| content_files= | ||||
|  | ||||
| # SGML files where gtk-doc abbrevations (#GtkWidget) are expanded | ||||
| # These files must be listed here *and* in content_files | ||||
|   | ||||
| @@ -29,6 +29,8 @@ | ||||
|   <chapter> | ||||
|     <title>Search</title> | ||||
|     <xi:include href="xml/shell-app-system.xml"/> | ||||
|     <xi:include href="xml/shell-contact-system.xml"/> | ||||
|     <xi:include href="xml/shell-doc-system.xml"/> | ||||
|   </chapter> | ||||
|   <chapter> | ||||
|     <title>Tray Icons</title> | ||||
| @@ -40,16 +42,16 @@ | ||||
|   <chapter> | ||||
|     <title>Recorder</title> | ||||
|     <xi:include href="xml/shell-recorder.xml"/> | ||||
|     <xi:include href="xml/shell-recorder-src.xml"/> | ||||
|   </chapter> | ||||
|   <chapter> | ||||
|     <title>Integration helpers and utilities</title> | ||||
|     <xi:include href="doc-gen-org.gnome.Shell.SearchProvider.xml"/> | ||||
|     <xi:include href="doc-gen-org.gnome.Shell.SearchProvider2.xml"/> | ||||
|     <xi:include href="xml/shell-global.xml"/> | ||||
|     <xi:include href="xml/shell-wm.xml"/> | ||||
|     <xi:include href="xml/shell-xfixes-cursor.xml"/> | ||||
|     <xi:include href="xml/shell-util.xml"/> | ||||
|     <xi:include href="xml/shell-mount-operation.xml"/> | ||||
|     <xi:include href="xml/shell-mobile-providers.xml"/> | ||||
|     <xi:include href="xml/shell-network-agent.xml"/> | ||||
|     <xi:include href="xml/shell-polkit-authentication-agent.xml"/> | ||||
|     <xi:include href="xml/shell-tp-client.xml"/> | ||||
|   | ||||
| @@ -52,6 +52,13 @@ its dependencies to build from tarballs.</description> | ||||
|       <gnome:userid>walters</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
|   <maintainer> | ||||
|     <foaf:Person> | ||||
|       <foaf:name>Dan Winship</foaf:name> | ||||
|       <foaf:mbox rdf:resource="mailto:danw@gnome.org" /> | ||||
|       <gnome:userid>danw</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
|   <maintainer> | ||||
|     <foaf:Person> | ||||
|       <foaf:name>Marina Zhurakhinskaya</foaf:name> | ||||
| @@ -59,18 +66,4 @@ its dependencies to build from tarballs.</description> | ||||
|       <gnome:userid>marinaz</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
|   <maintainer> | ||||
|     <foaf:Person> | ||||
|       <foaf:name>Florian Müllner</foaf:name> | ||||
|       <foaf:mbox rdf:resource="mailto:fmuellner@gnome.org" /> | ||||
|       <gnome:userid>fmuellner</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
|   <maintainer> | ||||
|     <foaf:Person> | ||||
|       <foaf:name>Ray Strode</foaf:name> | ||||
|       <foaf:mbox rdf:resource="mailto:halfline@gmail.com" /> | ||||
|       <gnome:userid>halfline</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
| </Project> | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| NULL = | ||||
|  | ||||
| EXTRA_DIST = misc/config.js.in | ||||
| CLEANFILES = misc/config.js | ||||
| @@ -7,7 +6,9 @@ misc/config.js: misc/config.js.in Makefile | ||||
| 	[ -d $(@D) ] || $(mkdir_p) $(@D) ; \ | ||||
| 	sed -e "s|[@]PACKAGE_NAME@|$(PACKAGE_NAME)|g" \ | ||||
| 	    -e "s|[@]PACKAGE_VERSION@|$(PACKAGE_VERSION)|g" \ | ||||
| 	    -e "s|[@]GJS_VERSION@|$(GJS_VERSION)|g" \ | ||||
| 	    -e "s|[@]HAVE_BLUETOOTH@|$(HAVE_BLUETOOTH)|g" \ | ||||
| 	    -e "s|[@]SHELL_SYSTEM_CA_FILE@|$(SHELL_SYSTEM_CA_FILE)|g" \ | ||||
| 	    -e "s|[@]GETTEXT_PACKAGE@|$(GETTEXT_PACKAGE)|g" \ | ||||
| 	    -e "s|[@]datadir@|$(datadir)|g" \ | ||||
| 	    -e "s|[@]libexecdir@|$(libexecdir)|g" \ | ||||
| @@ -18,83 +19,80 @@ jsdir = $(pkgdatadir)/js | ||||
|  | ||||
| nobase_dist_js_DATA = 	\ | ||||
| 	gdm/batch.js		\ | ||||
| 	gdm/consoleKit.js	\ | ||||
| 	gdm/fingerprint.js	\ | ||||
| 	gdm/loginDialog.js	\ | ||||
| 	gdm/powerMenu.js	\ | ||||
| 	gdm/realmd.js		\ | ||||
| 	gdm/util.js		\ | ||||
| 	gdm/systemd.js	 	\ | ||||
| 	extensionPrefs/main.js	\ | ||||
| 	misc/config.js		\ | ||||
| 	misc/extensionUtils.js	\ | ||||
| 	misc/fileUtils.js	\ | ||||
| 	misc/format.js		\ | ||||
| 	misc/gnomeSession.js	\ | ||||
| 	misc/hash.js		\ | ||||
| 	misc/history.js		\ | ||||
| 	misc/jsParse.js		\ | ||||
| 	misc/loginManager.js	\ | ||||
| 	misc/modemManager.js	\ | ||||
| 	misc/params.js		\ | ||||
| 	misc/screenSaver.js     \ | ||||
| 	misc/util.js		\ | ||||
| 	perf/core.js		\ | ||||
| 	ui/altTab.js		\ | ||||
| 	ui/appDisplay.js	\ | ||||
| 	ui/appFavorites.js	\ | ||||
| 	ui/backgroundMenu.js	\ | ||||
| 	ui/background.js	\ | ||||
| 	ui/automountManager.js  \ | ||||
| 	ui/autorunManager.js    \ | ||||
| 	ui/boxpointer.js	\ | ||||
| 	ui/calendar.js		\ | ||||
| 	ui/checkBox.js		\ | ||||
| 	ui/contactDisplay.js \ | ||||
| 	ui/ctrlAltTab.js	\ | ||||
| 	ui/dash.js		\ | ||||
| 	ui/dateMenu.js		\ | ||||
| 	ui/dnd.js		\ | ||||
| 	ui/endSessionDialog.js	\ | ||||
| 	ui/extensionSystem.js	\ | ||||
| 	ui/extensionDownloader.js \ | ||||
| 	ui/environment.js	\ | ||||
| 	ui/ibusCandidatePopup.js\ | ||||
| 	ui/grabHelper.js	\ | ||||
| 	ui/extensionSystem.js	\ | ||||
| 	ui/flashspot.js		\ | ||||
| 	ui/iconGrid.js		\ | ||||
| 	ui/keyboard.js		\ | ||||
| 	ui/keyringPrompt.js	\ | ||||
| 	ui/layout.js		\ | ||||
| 	ui/lightbox.js		\ | ||||
| 	ui/link.js		\ | ||||
| 	ui/lookingGlass.js	\ | ||||
| 	ui/magnifier.js		\ | ||||
| 	ui/magnifierDBus.js	\ | ||||
| 	ui/main.js		\ | ||||
| 	ui/messageTray.js	\ | ||||
| 	ui/modalDialog.js	\ | ||||
| 	ui/separator.js		\ | ||||
| 	ui/sessionMode.js	\ | ||||
| 	ui/networkAgent.js	\ | ||||
| 	ui/shellEntry.js	\ | ||||
| 	ui/shellMountOperation.js \ | ||||
| 	ui/notificationDaemon.js \ | ||||
| 	ui/overview.js		\ | ||||
| 	ui/overviewControls.js	\ | ||||
| 	ui/panel.js		\ | ||||
| 	ui/panelMenu.js		\ | ||||
| 	ui/pointerWatcher.js    \ | ||||
| 	ui/placeDisplay.js	\ | ||||
| 	ui/polkitAuthenticationAgent.js \ | ||||
| 	ui/popupMenu.js		\ | ||||
| 	ui/remoteSearch.js	\ | ||||
| 	ui/runDialog.js		\ | ||||
| 	ui/screenshot.js	\ | ||||
|         ui/screenShield.js	\ | ||||
| 	ui/scripting.js		\ | ||||
| 	ui/search.js		\ | ||||
| 	ui/searchDisplay.js	\ | ||||
| 	ui/shellDBus.js		\ | ||||
| 	ui/statusIconDispatcher.js	\ | ||||
| 	ui/status/accessibility.js	\ | ||||
| 	ui/status/keyboard.js	\ | ||||
| 	ui/status/lockScreenMenu.js	\ | ||||
| 	ui/status/network.js	\ | ||||
| 	ui/status/power.js	\ | ||||
| 	ui/status/volume.js	\ | ||||
| 	ui/status/bluetooth.js	\ | ||||
| 	ui/switcherPopup.js	\ | ||||
| 	ui/telepathyClient.js	\ | ||||
| 	ui/tweener.js		\ | ||||
| 	ui/unlockDialog.js	\ | ||||
| 	ui/userMenu.js		\ | ||||
| 	ui/userWidget.js	\ | ||||
| 	ui/viewSelector.js	\ | ||||
| 	ui/wanda.js		\ | ||||
| 	ui/windowAttentionHandler.js	\ | ||||
| @@ -103,13 +101,4 @@ nobase_dist_js_DATA = 	\ | ||||
| 	ui/workspaceThumbnail.js	\ | ||||
| 	ui/workspacesView.js	\ | ||||
| 	ui/workspaceSwitcherPopup.js    \ | ||||
| 	ui/xdndHandler.js	\ | ||||
| 	ui/components/__init__.js		\ | ||||
| 	ui/components/autorunManager.js		\ | ||||
| 	ui/components/automountManager.js	\ | ||||
| 	ui/components/networkAgent.js		\ | ||||
| 	ui/components/polkitAgent.js		\ | ||||
| 	ui/components/recorder.js		\ | ||||
| 	ui/components/telepathyClient.js	\ | ||||
| 	ui/components/keyring.js		\ | ||||
| 	$(NULL) | ||||
| 	ui/xdndHandler.js | ||||
|   | ||||
| @@ -6,14 +6,15 @@ const GObject = imports.gi.GObject; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Format = imports.format; | ||||
|  | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const Format = imports.misc.format; | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
|  | ||||
| const GnomeShellIface = <interface name="org.gnome.Shell.Extensions"> | ||||
|  | ||||
| const GnomeShellIface = <interface name="org.gnome.Shell"> | ||||
| <signal name="ExtensionStatusChanged"> | ||||
|     <arg type="s" name="uuid"/> | ||||
|     <arg type="i" name="state"/> | ||||
| @@ -161,7 +162,7 @@ const Application = new Lang.Class({ | ||||
|         vbox.add(toolbar); | ||||
|         let toolitem; | ||||
|  | ||||
|         let label = new Gtk.Label({ label: '<b>' + _("Extension") + '</b>', | ||||
|         let label = new Gtk.Label({ label: _("<b>Extension</b>"), | ||||
|                                     use_markup: true }); | ||||
|         toolitem = new Gtk.ToolItem({ child: label }); | ||||
|         toolbar.add(toolitem); | ||||
| @@ -174,7 +175,7 @@ const Application = new Lang.Class({ | ||||
|         let renderer = new Gtk.CellRendererText(); | ||||
|         this._extensionSelector.pack_start(renderer, true); | ||||
|         this._extensionSelector.add_attribute(renderer, 'text', 1); | ||||
|         this._extensionSelector.set_cell_data_func(renderer, Lang.bind(this, this._setExtensionInsensitive)); | ||||
|         this._extensionSelector.set_cell_data_func(renderer, Lang.bind(this, this._setExtensionInsensitive), null); | ||||
|         this._extensionSelector.connect('changed', Lang.bind(this, this._extensionSelected)); | ||||
|  | ||||
|         toolitem = new Gtk.ToolItem({ child: this._extensionSelector }); | ||||
| @@ -201,17 +202,23 @@ const Application = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _scanExtensions: function() { | ||||
|         let finder = new ExtensionUtils.ExtensionFinder(); | ||||
|         finder.connect('extension-found', Lang.bind(this, this._extensionFound)); | ||||
|         finder.scanExtensions(); | ||||
|     }, | ||||
|         ExtensionUtils.scanExtensions(Lang.bind(this, function(uuid, dir, type) { | ||||
|             if (ExtensionUtils.extensions[uuid] !== undefined) | ||||
|                 return; | ||||
|  | ||||
|     _extensionFound: function(signals, extension) { | ||||
|         let iter = this._model.append(); | ||||
|         this._model.set(iter, [0, 1], [extension.uuid, extension.metadata.name]); | ||||
|         this._extensionIters[extension.uuid] = iter; | ||||
|     }, | ||||
|             let extension; | ||||
|             try { | ||||
|                 extension = ExtensionUtils.createExtensionObject(uuid, dir, type); | ||||
|             } catch(e) { | ||||
|                 global.logError('' + e); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             let iter = this._model.append(); | ||||
|             this._model.set(iter, [0, 1], [uuid, extension.metadata.name]); | ||||
|             this._extensionIters[uuid] = iter; | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _onActivate: function() { | ||||
|         this._window.present(); | ||||
| @@ -250,7 +257,7 @@ function initEnvironment() { | ||||
|         }, | ||||
|  | ||||
|         logError: function(s) { | ||||
|             log('ERROR: ' + s); | ||||
|             global.log('ERROR: ' + s); | ||||
|         }, | ||||
|  | ||||
|         userdatadir: GLib.build_filenamev([GLib.get_user_data_dir(), 'gnome-shell']) | ||||
| @@ -261,6 +268,7 @@ function initEnvironment() { | ||||
|  | ||||
| function main(argv) { | ||||
|     initEnvironment(); | ||||
|     ExtensionUtils.init(); | ||||
|  | ||||
|     Gettext.bindtextdomain(Config.GETTEXT_PACKAGE, Config.LOCALEDIR); | ||||
|     Gettext.textdomain(Config.GETTEXT_PACKAGE); | ||||
|   | ||||
							
								
								
									
										22
									
								
								js/gdm/consoleKit.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
|  | ||||
| const ConsoleKitManagerIface = <interface name='org.freedesktop.ConsoleKit.Manager'> | ||||
| <method name='CanRestart'> | ||||
|     <arg type='b' direction='out'/> | ||||
| </method> | ||||
| <method name='CanStop'> | ||||
|     <arg type='b' direction='out'/> | ||||
| </method> | ||||
| <method name='Restart' /> | ||||
| <method name='Stop' /> | ||||
| </interface>; | ||||
|  | ||||
| const ConsoleKitProxy = Gio.DBusProxy.makeProxyWrapper(ConsoleKitManagerIface); | ||||
|  | ||||
| function ConsoleKitManager() { | ||||
|     return new ConsoleKitProxy(Gio.DBus.system, | ||||
|                                'org.freedesktop.ConsoleKit', | ||||
|                                '/org/freedesktop/ConsoleKit/Manager'); | ||||
| }; | ||||
| @@ -11,16 +11,10 @@ const FprintManagerIface = <interface name='net.reactivated.Fprint.Manager'> | ||||
| </method> | ||||
| </interface>; | ||||
|  | ||||
| const FprintManagerInfo = Gio.DBusInterfaceInfo.new_for_xml(FprintManagerIface); | ||||
| const FprintManagerProxy = Gio.DBusProxy.makeProxyWrapper(FprintManagerIface); | ||||
|  | ||||
| function FprintManager() { | ||||
|     var self = new Gio.DBusProxy({ g_connection: Gio.DBus.system, | ||||
|                                    g_interface_name: FprintManagerInfo.name, | ||||
|                                    g_interface_info: FprintManagerInfo, | ||||
|                                    g_name: 'net.reactivated.Fprint', | ||||
|                                    g_object_path: '/net/reactivated/Fprint/Manager', | ||||
|                                    g_flags: (Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES) }); | ||||
|  | ||||
|     self.init(null); | ||||
|     return self; | ||||
| } | ||||
|     return new FprintManagerProxy(Gio.DBus.system, | ||||
|                            'net.reactivated.Fprint', | ||||
|                            '/net/reactivated/Fprint/Manager'); | ||||
| }; | ||||
|   | ||||
| @@ -18,12 +18,12 @@ | ||||
|  * 02111-1307, USA. | ||||
|  */ | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const UPowerGlib = imports.gi.UPowerGlib; | ||||
|  | ||||
| const LoginManager = imports.misc.loginManager; | ||||
| const ConsoleKit = imports.gdm.consoleKit; | ||||
| const Systemd = imports.gdm.systemd; | ||||
|  | ||||
| const GdmUtil = imports.gdm.util; | ||||
| const PanelMenu = imports.ui.panelMenu; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
|  | ||||
| @@ -32,17 +32,20 @@ const PowerMenuButton = new Lang.Class({ | ||||
|     Extends: PanelMenu.SystemStatusButton, | ||||
|  | ||||
|     _init: function() { | ||||
|         /* Translators: accessible name of the power menu in the login screen */ | ||||
|         this.parent('system-shutdown-symbolic', _("Power")); | ||||
|         this.parent('system-shutdown', null); | ||||
|         this._upClient = new UPowerGlib.Client(); | ||||
|  | ||||
|         this._loginManager = LoginManager.getLoginManager(); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: GdmUtil.LOGIN_SCREEN_SCHEMA }); | ||||
|         this._settings.connect('changed::disable-restart-buttons', | ||||
|                                Lang.bind(this, this._updateVisibility)); | ||||
|         if (Systemd.haveSystemd()) | ||||
|             this._systemdLoginManager = new Systemd.SystemdLoginManager(); | ||||
|         else | ||||
|             this._consoleKitManager = new ConsoleKit.ConsoleKitManager(); | ||||
|  | ||||
|         this._createSubMenu(); | ||||
|  | ||||
|         this._upClient.connect('notify::can-suspend', | ||||
|                                Lang.bind(this, this._updateHaveSuspend)); | ||||
|         this._updateHaveSuspend(); | ||||
|  | ||||
|         // ConsoleKit doesn't send notifications when shutdown/reboot | ||||
|         // are disabled, so we update the menu item each time the menu opens | ||||
|         this.menu.connect('open-state-changed', Lang.bind(this, | ||||
| @@ -50,41 +53,100 @@ const PowerMenuButton = new Lang.Class({ | ||||
|                 if (open) { | ||||
|                     this._updateHaveShutdown(); | ||||
|                     this._updateHaveRestart(); | ||||
|                     this._updateHaveSuspend(); | ||||
|                 } | ||||
|             })); | ||||
|         this._updateHaveShutdown(); | ||||
|         this._updateHaveRestart(); | ||||
|         this._updateHaveSuspend(); | ||||
|     }, | ||||
|  | ||||
|     _updateVisibility: function() { | ||||
|         let shouldBeVisible = (this._haveSuspend || this._haveShutdown || this._haveRestart); | ||||
|         this.actor.visible = shouldBeVisible && !this._settings.get_boolean('disable-restart-buttons'); | ||||
|         if (!this._haveSuspend && !this._haveShutdown && !this._haveRestart) | ||||
|             this.actor.hide(); | ||||
|         else | ||||
|             this.actor.show(); | ||||
|     }, | ||||
|  | ||||
|     _updateHaveShutdown: function() { | ||||
|         this._loginManager.canPowerOff(Lang.bind(this, function(result) { | ||||
|             this._haveShutdown = result; | ||||
|             this._powerOffItem.actor.visible = this._haveShutdown; | ||||
|             this._updateVisibility(); | ||||
|         })); | ||||
|  | ||||
|         if (Systemd.haveSystemd()) { | ||||
|             this._systemdLoginManager.CanPowerOffRemote(Lang.bind(this, | ||||
|                 function(result, error) { | ||||
|                     if (!error) | ||||
|                         this._haveShutdown = result != 'no'; | ||||
|                     else | ||||
|                         this._haveShutdown = false; | ||||
|  | ||||
|                     if (this._haveShutdown) | ||||
|                         this._powerOffItem.actor.show(); | ||||
|                     else | ||||
|                         this._powerOffItem.actor.hide(); | ||||
|  | ||||
|                     this._updateVisibility(); | ||||
|                 })); | ||||
|         } else { | ||||
|             this._consoleKitManager.CanStopRemote(Lang.bind(this, | ||||
|                 function(result, error) { | ||||
|                     if (!error) | ||||
|                         this._haveShutdown = result; | ||||
|                     else | ||||
|                         this._haveShutdown = false; | ||||
|  | ||||
|                     if (this._haveShutdown) { | ||||
|                         this._powerOffItem.actor.show(); | ||||
|                     } else { | ||||
|                         this._powerOffItem.actor.hide(); | ||||
|                     } | ||||
|  | ||||
|                     this._updateVisibility(); | ||||
|                 })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateHaveRestart: function() { | ||||
|         this._loginManager.canReboot(Lang.bind(this, function(result) { | ||||
|             this._haveRestart = result; | ||||
|             this._restartItem.actor.visible = this._haveRestart; | ||||
|             this._updateVisibility(); | ||||
|         })); | ||||
|  | ||||
|         if (Systemd.haveSystemd()) { | ||||
|             this._systemdLoginManager.CanRebootRemote(Lang.bind(this, | ||||
|                 function(result, error) { | ||||
|                     if (!error) | ||||
|                         this._haveRestart = result != 'no'; | ||||
|                     else | ||||
|                         this._haveRestart = false; | ||||
|  | ||||
|                     if (this._haveRestart) | ||||
|                         this._restartItem.actor.show(); | ||||
|                     else | ||||
|                         this._restartItem.actor.hide(); | ||||
|  | ||||
|                     this._updateVisibility(); | ||||
|                 })); | ||||
|         } else { | ||||
|             this._consoleKitManager.CanRestartRemote(Lang.bind(this, | ||||
|                 function(result, error) { | ||||
|                     if (!error) | ||||
|                         this._haveRestart = result; | ||||
|                     else | ||||
|                         this._haveRestart = false; | ||||
|  | ||||
|                     if (this._haveRestart) { | ||||
|                         this._restartItem.actor.show(); | ||||
|                     } else { | ||||
|                         this._restartItem.actor.hide(); | ||||
|                     } | ||||
|  | ||||
|                     this._updateVisibility(); | ||||
|                 })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateHaveSuspend: function() { | ||||
|         this._loginManager.canSuspend(Lang.bind(this, function(result) { | ||||
|             this._haveSuspend = result; | ||||
|             this._suspendItem.actor.visible = this._haveSuspend; | ||||
|             this._updateVisibility(); | ||||
|         })); | ||||
|         this._haveSuspend = this._upClient.get_can_suspend(); | ||||
|  | ||||
|         if (this._haveSuspend) | ||||
|             this._suspendItem.actor.show(); | ||||
|         else | ||||
|             this._suspendItem.actor.hide(); | ||||
|  | ||||
|         this._updateVisibility(); | ||||
|     }, | ||||
|  | ||||
|     _createSubMenu: function() { | ||||
| @@ -107,23 +169,27 @@ const PowerMenuButton = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onActivateSuspend: function() { | ||||
|         if (!this._haveSuspend) | ||||
|             return; | ||||
|  | ||||
|         this._loginManager.suspend(); | ||||
|         if (this._haveSuspend) | ||||
|             this._upClient.suspend_sync(null); | ||||
|     }, | ||||
|  | ||||
|     _onActivateRestart: function() { | ||||
|         if (!this._haveRestart) | ||||
|             return; | ||||
|  | ||||
|         this._loginManager.reboot(); | ||||
|         if (Systemd.haveSystemd()) | ||||
|             this._systemdLoginManager.RebootRemote(true); | ||||
|         else | ||||
|             this._consoleKitManager.RestartRemote(); | ||||
|     }, | ||||
|  | ||||
|     _onActivatePowerOff: function() { | ||||
|         if (!this._haveShutdown) | ||||
|             return; | ||||
|  | ||||
|         this._loginManager.powerOff(); | ||||
|         if (Systemd.haveSystemd()) | ||||
|             this._systemdLoginManager.PowerOffRemote(true); | ||||
|         else | ||||
|             this._consoleKitManager.StopRemote(); | ||||
|     } | ||||
| }); | ||||
|   | ||||
							
								
								
									
										139
									
								
								js/gdm/realmd.js
									
									
									
									
									
								
							
							
						
						| @@ -1,139 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const ProviderIface = <interface name='org.freedesktop.realmd.Provider'> | ||||
|     <property name="Name" type="s" access="read"/> | ||||
|     <property name="Version" type="s" access="read"/> | ||||
|     <property name="Realms" type="ao" access="read"/> | ||||
|     <method name="Discover"> | ||||
|         <arg name="string" type="s" direction="in"/> | ||||
|         <arg name="options" type="a{sv}" direction="in"/> | ||||
|         <arg name="relevance" type="i" direction="out"/> | ||||
|         <arg name="realm" type="ao" direction="out"/> | ||||
|     </method> | ||||
| </interface>; | ||||
| const Provider = Gio.DBusProxy.makeProxyWrapper(ProviderIface); | ||||
|  | ||||
| const ServiceIface = <interface name="org.freedesktop.realmd.Service"> | ||||
|     <method name="Cancel"> | ||||
|         <arg name="operation" type="s" direction="in"/> | ||||
|     </method> | ||||
|     <method name="Release" /> | ||||
|     <method name="SetLocale"> | ||||
|         <arg name="locale" type="s" direction="in"/> | ||||
|     </method> | ||||
|     <signal name="Diagnostics"> | ||||
|         <arg name="data" type="s"/> | ||||
|         <arg name="operation" type="s"/> | ||||
|     </signal> | ||||
| </interface>; | ||||
| const Service = Gio.DBusProxy.makeProxyWrapper(ServiceIface); | ||||
|  | ||||
| const RealmIface = <interface name="org.freedesktop.realmd.Realm"> | ||||
|     <property name="Name" type="s" access="read"/> | ||||
|     <property name="Configured" type="s" access="read"/> | ||||
|     <property name="Details" type="a(ss)" access="read"/> | ||||
|     <property name="LoginFormats" type="as" access="read"/> | ||||
|     <property name="LoginPolicy" type="s" access="read"/> | ||||
|     <property name="PermittedLogins" type="as" access="read"/> | ||||
|     <property name="SupportedInterfaces" type="as" access="read"/> | ||||
|     <method name="ChangeLoginPolicy"> | ||||
|         <arg name="login_policy" type="s" direction="in"/> | ||||
|         <arg name="permitted_add" type="as" direction="in"/> | ||||
|         <arg name="permitted_remove" type="as" direction="in"/> | ||||
|         <arg name="options" type="a{sv}" direction="in"/> | ||||
|     </method> | ||||
|     <method name="Deconfigure"> | ||||
|         <arg name="options" type="a{sv}" direction="in"/> | ||||
|     </method> | ||||
| </interface>; | ||||
| const Realm = Gio.DBusProxy.makeProxyWrapper(RealmIface); | ||||
|  | ||||
| const Manager = new Lang.Class({ | ||||
|     Name: 'Manager', | ||||
|  | ||||
|     _init: function(parentActor) { | ||||
|         this._aggregateProvider = Provider(Gio.DBus.system, | ||||
|                                            'org.freedesktop.realmd', | ||||
|                                            '/org/freedesktop/realmd', | ||||
|                                            Lang.bind(this, this._reloadRealms)) | ||||
|         this._realms = {}; | ||||
|  | ||||
|         this._aggregateProvider.connect('g-properties-changed', | ||||
|                                         Lang.bind(this, function(proxy, properties) { | ||||
|                                             if ('Realms' in properties.deep_unpack()) | ||||
|                                                 this._reloadRealms(); | ||||
|                                         })); | ||||
|     }, | ||||
|  | ||||
|     _reloadRealms: function() { | ||||
|         let realmPaths = this._aggregateProvider.Realms; | ||||
|  | ||||
|         if (!realmPaths) | ||||
|             return; | ||||
|  | ||||
|         for (let i = 0; i < realmPaths.length; i++) { | ||||
|             let realm = Realm(Gio.DBus.system, | ||||
|                               'org.freedesktop.realmd', | ||||
|                               realmPaths[i], | ||||
|                               Lang.bind(this, this._onRealmLoaded)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _reloadRealm: function(realm) { | ||||
|         if (!realm.Configured) { | ||||
|             if (this._realms[realm.get_object_path()]) | ||||
|                 delete this._realms[realm.get_object_path()]; | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._realms[realm.get_object_path()] = realm; | ||||
|  | ||||
|         this._updateLoginFormat(); | ||||
|     }, | ||||
|  | ||||
|     _onRealmLoaded: function(realm, error) { | ||||
|         if (error) | ||||
|             return; | ||||
|  | ||||
|         this._reloadRealm(realm); | ||||
|  | ||||
|         realm.connect('g-properties-changed', | ||||
|                       Lang.bind(this, function(proxy, properties) { | ||||
|                                 if ('Configured' in properties.deep_unpack()) | ||||
|                                     this._reloadRealm(); | ||||
|                                 })); | ||||
|     }, | ||||
|  | ||||
|     _updateLoginFormat: function() { | ||||
|         let newLoginFormat; | ||||
|  | ||||
|         for (let realmPath in this._realms) { | ||||
|             let realm = this._realms[realmPath]; | ||||
|             if (realm.LoginFormats && realm.LoginFormats.length > 0) { | ||||
|                 newLoginFormat = realm.LoginFormats[0]; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (this._loginFormat != newLoginFormat) { | ||||
|             this._loginFormat = newLoginFormat; | ||||
|             this.emit('login-format-changed', newLoginFormat); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     get loginFormat() { | ||||
|         if (this._loginFormat !== undefined) | ||||
|             return this._loginFormat; | ||||
|  | ||||
|         this._updateLoginFormat(); | ||||
|  | ||||
|         return this._loginFormat; | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Manager.prototype) | ||||
							
								
								
									
										31
									
								
								js/gdm/systemd.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
|  | ||||
| const SystemdLoginManagerIface = <interface name='org.freedesktop.login1.Manager'> | ||||
| <method name='PowerOff'> | ||||
|     <arg type='b' direction='in'/> | ||||
| </method> | ||||
| <method name='Reboot'> | ||||
|     <arg type='b' direction='in'/> | ||||
| </method> | ||||
| <method name='CanPowerOff'> | ||||
|     <arg type='s' direction='out'/> | ||||
| </method> | ||||
| <method name='CanReboot'> | ||||
|     <arg type='s' direction='out'/> | ||||
| </method> | ||||
| </interface>; | ||||
|  | ||||
| const SystemdLoginManagerProxy = Gio.DBusProxy.makeProxyWrapper(SystemdLoginManagerIface); | ||||
|  | ||||
| function SystemdLoginManager() { | ||||
|     return new SystemdLoginManagerProxy(Gio.DBus.system, | ||||
|                                         'org.freedesktop.login1', | ||||
|                                         '/org/freedesktop/login1'); | ||||
| }; | ||||
|  | ||||
| function haveSystemd() { | ||||
|     return GLib.access("/sys/fs/cgroup/systemd", 0) >= 0; | ||||
| } | ||||
							
								
								
									
										402
									
								
								js/gdm/util.js
									
									
									
									
									
								
							
							
						
						| @@ -1,402 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Batch = imports.gdm.batch; | ||||
| const Fprint = imports.gdm.fingerprint; | ||||
| const Realmd = imports.gdm.realmd; | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const PASSWORD_SERVICE_NAME = 'gdm-password'; | ||||
| const FINGERPRINT_SERVICE_NAME = 'gdm-fingerprint'; | ||||
| const FADE_ANIMATION_TIME = 0.16; | ||||
| const CLONE_FADE_ANIMATION_TIME = 0.25; | ||||
|  | ||||
| const LOGIN_SCREEN_SCHEMA = 'org.gnome.login-screen'; | ||||
| const FINGERPRINT_AUTHENTICATION_KEY = 'enable-fingerprint-authentication'; | ||||
| const BANNER_MESSAGE_KEY = 'banner-message-enable'; | ||||
| const BANNER_MESSAGE_TEXT_KEY = 'banner-message-text'; | ||||
| const ALLOWED_FAILURES_KEY = 'allowed-failures'; | ||||
|  | ||||
| const LOGO_KEY = 'logo'; | ||||
| const DISABLE_USER_LIST_KEY = 'disable-user-list'; | ||||
|  | ||||
| function fadeInActor(actor) { | ||||
|     if (actor.opacity == 255 && actor.visible) | ||||
|         return null; | ||||
|  | ||||
|     let hold = new Batch.Hold(); | ||||
|     actor.show(); | ||||
|     let [minHeight, naturalHeight] = actor.get_preferred_height(-1); | ||||
|  | ||||
|     actor.opacity = 0; | ||||
|     actor.set_height(0); | ||||
|     Tweener.addTween(actor, | ||||
|                      { opacity: 255, | ||||
|                        height: naturalHeight, | ||||
|                        time: FADE_ANIMATION_TIME, | ||||
|                        transition: 'easeOutQuad', | ||||
|                        onComplete: function() { | ||||
|                            this.set_height(-1); | ||||
|                            hold.release(); | ||||
|                        }, | ||||
|                      }); | ||||
|  | ||||
|     return hold; | ||||
| } | ||||
|  | ||||
| function fadeOutActor(actor) { | ||||
|     if (!actor.visible || actor.opacity == 0) { | ||||
|         actor.opacity = 0; | ||||
|         actor.hide(); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     let hold = new Batch.Hold(); | ||||
|     Tweener.addTween(actor, | ||||
|                      { opacity: 0, | ||||
|                        height: 0, | ||||
|                        time: FADE_ANIMATION_TIME, | ||||
|                        transition: 'easeOutQuad', | ||||
|                        onComplete: function() { | ||||
|                            this.hide(); | ||||
|                            this.set_height(-1); | ||||
|                            hold.release(); | ||||
|                        }, | ||||
|                      }); | ||||
|     return hold; | ||||
| } | ||||
|  | ||||
| function cloneAndFadeOutActor(actor) { | ||||
|     // Immediately hide actor so its sibling can have its space | ||||
|     // and position, but leave a non-reactive clone on-screen, | ||||
|     // so from the user's point of view it smoothly fades away | ||||
|     // and reveals its sibling. | ||||
|     actor.hide(); | ||||
|  | ||||
|     let clone = new Clutter.Clone({ source: actor, | ||||
|                                     reactive: false }); | ||||
|  | ||||
|     Main.uiGroup.add_child(clone); | ||||
|  | ||||
|     let [x, y] = actor.get_transformed_position(); | ||||
|     clone.set_position(x, y); | ||||
|  | ||||
|     let hold = new Batch.Hold(); | ||||
|     Tweener.addTween(clone, | ||||
|                      { opacity: 0, | ||||
|                        time: CLONE_FADE_ANIMATION_TIME, | ||||
|                        transition: 'easeOutQuad', | ||||
|                        onComplete: function() { | ||||
|                            clone.destroy(); | ||||
|                            hold.release(); | ||||
|                        } | ||||
|                      }); | ||||
|     return hold; | ||||
| } | ||||
|  | ||||
| const ShellUserVerifier = new Lang.Class({ | ||||
|     Name: 'ShellUserVerifier', | ||||
|  | ||||
|     _init: function(client, params) { | ||||
|         params = Params.parse(params, { reauthenticationOnly: false }); | ||||
|         this._reauthOnly = params.reauthenticationOnly; | ||||
|  | ||||
|         this._client = client; | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: LOGIN_SCREEN_SCHEMA }); | ||||
|  | ||||
|         this._fprintManager = new Fprint.FprintManager(); | ||||
|         this._realmManager = new Realmd.Manager(); | ||||
|  | ||||
|         this._failCounter = 0; | ||||
|     }, | ||||
|  | ||||
|     begin: function(userName, hold) { | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|         this._hold = hold; | ||||
|         this._userName = userName; | ||||
|  | ||||
|         this._checkForFingerprintReader(); | ||||
|  | ||||
|         if (userName) { | ||||
|             // If possible, reauthenticate an already running session, | ||||
|             // so any session specific credentials get updated appropriately | ||||
|             this._client.open_reauthentication_channel(userName, this._cancellable, | ||||
|                                                        Lang.bind(this, this._reauthenticationChannelOpened)); | ||||
|         } else { | ||||
|             this._client.get_user_verifier(this._cancellable, Lang.bind(this, this._userVerifierGot)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|         if (this._cancellable) | ||||
|             this._cancellable.cancel(); | ||||
|  | ||||
|         if (this._userVerifier) | ||||
|             this._userVerifier.call_cancel_sync(null); | ||||
|     }, | ||||
|  | ||||
|     clear: function() { | ||||
|         if (this._cancellable) { | ||||
|             this._cancellable.cancel(); | ||||
|             this._cancellable = null; | ||||
|         } | ||||
|  | ||||
|         if (this._userVerifier) { | ||||
|             this._userVerifier.run_dispose(); | ||||
|             this._userVerifier = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     answerQuery: function(serviceName, answer) { | ||||
|         // Clear any previous message | ||||
|         this.emit('show-message', null, null); | ||||
|  | ||||
|         this._userVerifier.call_answer_query(serviceName, answer, this._cancellable, null); | ||||
|     }, | ||||
|  | ||||
|     _checkForFingerprintReader: function() { | ||||
|         this._haveFingerprintReader = false; | ||||
|  | ||||
|         if (!this._settings.get_boolean(FINGERPRINT_AUTHENTICATION_KEY)) | ||||
|             return; | ||||
|  | ||||
|         this._fprintManager.GetDefaultDeviceRemote(Gio.DBusCallFlags.NONE, this._cancellable, Lang.bind(this, | ||||
|             function(device, error) { | ||||
|                 if (!error && device) | ||||
|                     this._haveFingerprintReader = true; | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     _reportInitError: function(where, error) { | ||||
|         logError(error, where); | ||||
|         this._hold.release(); | ||||
|  | ||||
|         this.emit('show-message', _("Authentication error"), 'login-dialog-message-warning'); | ||||
|         this._verificationFailed(false); | ||||
|     }, | ||||
|  | ||||
|     _reauthenticationChannelOpened: function(client, result) { | ||||
|         try { | ||||
|             this._userVerifier = client.open_reauthentication_channel_finish(result); | ||||
|         } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|             return; | ||||
|         } catch(e if e.matches(Gio.DBusError, Gio.DBusError.ACCESS_DENIED) && | ||||
|                 !this._reauthOnly) { | ||||
|             // Gdm emits org.freedesktop.DBus.Error.AccessDenied when there is | ||||
|             // no session to reauthenticate. Fall back to performing verification | ||||
|             // from this login session | ||||
|             client.get_user_verifier(this._cancellable, Lang.bind(this, this._userVerifierGot)); | ||||
|             return; | ||||
|         } catch(e) { | ||||
|             this._reportInitError('Failed to open reauthentication channel', e); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._connectSignals(); | ||||
|         this._beginVerification(); | ||||
|         this._hold.release(); | ||||
|     }, | ||||
|  | ||||
|     _userVerifierGot: function(client, result) { | ||||
|         try { | ||||
|             this._userVerifier = client.get_user_verifier_finish(result); | ||||
|         } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|             return; | ||||
|         } catch(e) { | ||||
|             this._reportInitError('Failed to obtain user verifier', e); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._connectSignals(); | ||||
|         this._beginVerification(); | ||||
|         this._hold.release(); | ||||
|     }, | ||||
|  | ||||
|     _connectSignals: function() { | ||||
|         this._userVerifier.connect('info', Lang.bind(this, this._onInfo)); | ||||
|         this._userVerifier.connect('problem', Lang.bind(this, this._onProblem)); | ||||
|         this._userVerifier.connect('info-query', Lang.bind(this, this._onInfoQuery)); | ||||
|         this._userVerifier.connect('secret-info-query', Lang.bind(this, this._onSecretInfoQuery)); | ||||
|         this._userVerifier.connect('conversation-stopped', Lang.bind(this, this._onConversationStopped)); | ||||
|         this._userVerifier.connect('reset', Lang.bind(this, this._onReset)); | ||||
|         this._userVerifier.connect('verification-complete', Lang.bind(this, this._onVerificationComplete)); | ||||
|     }, | ||||
|  | ||||
|     _beginVerification: function() { | ||||
|         this._hold.acquire(); | ||||
|  | ||||
|         if (this._userName) { | ||||
|             this._userVerifier.call_begin_verification_for_user(PASSWORD_SERVICE_NAME, | ||||
|                                                                 this._userName, | ||||
|                                                                 this._cancellable, | ||||
|                                                                 Lang.bind(this, function(obj, result) { | ||||
|                 try { | ||||
|                     obj.call_begin_verification_for_user_finish(result); | ||||
|                 } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|                     return; | ||||
|                 } catch(e) { | ||||
|                     this._reportInitError('Failed to start verification for user', e); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 this._hold.release(); | ||||
|             })); | ||||
|  | ||||
|             if (this._haveFingerprintReader) { | ||||
|                 this._hold.acquire(); | ||||
|  | ||||
|                 this._userVerifier.call_begin_verification_for_user(FINGERPRINT_SERVICE_NAME, | ||||
|                                                                     this._userName, | ||||
|                                                                     this._cancellable, | ||||
|                                                                     Lang.bind(this, function(obj, result) { | ||||
|                     try { | ||||
|                         obj.call_begin_verification_for_user_finish(result); | ||||
|                     } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|                         return; | ||||
|                     } catch(e) { | ||||
|                         this._reportInitError('Failed to start fingerprint verification for user', e); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
|                     this._hold.release(); | ||||
|                 })); | ||||
|             } | ||||
|         } else { | ||||
|             this._userVerifier.call_begin_verification(PASSWORD_SERVICE_NAME, | ||||
|                                                        this._cancellable, | ||||
|                                                        Lang.bind(this, function(obj, result) { | ||||
|                 try { | ||||
|                     obj.call_begin_verification_finish(result); | ||||
|                 } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|                     return; | ||||
|                 } catch(e) { | ||||
|                     this._reportInitError('Failed to start verification', e); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 this._hold.release(); | ||||
|             })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onInfo: function(client, serviceName, info) { | ||||
|         // We don't display fingerprint messages, because they | ||||
|         // have words like UPEK in them. Instead we use the messages | ||||
|         // as a cue to display our own message. | ||||
|         if (serviceName == FINGERPRINT_SERVICE_NAME && | ||||
|             this._haveFingerprintReader) { | ||||
|  | ||||
|             // Translators: this message is shown below the password entry field | ||||
|             // to indicate the user can swipe their finger instead | ||||
|             this.emit('show-login-hint', _("(or swipe finger)")); | ||||
|         } else if (serviceName == PASSWORD_SERVICE_NAME) { | ||||
|             this.emit('show-message', info, 'login-dialog-message-info'); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onProblem: function(client, serviceName, problem) { | ||||
|         // we don't want to show auth failed messages to | ||||
|         // users who haven't enrolled their fingerprint. | ||||
|         if (serviceName != PASSWORD_SERVICE_NAME) | ||||
|             return; | ||||
|         this.emit('show-message', problem, 'login-dialog-message-warning'); | ||||
|     }, | ||||
|  | ||||
|     _showRealmLoginHint: function() { | ||||
|         if (this._realmManager.loginFormat) { | ||||
|             let hint = this._realmManager.loginFormat; | ||||
|  | ||||
|             hint = hint.replace(/%U/g, 'user'); | ||||
|             hint = hint.replace(/%D/g, 'DOMAIN'); | ||||
|             hint = hint.replace(/%[^UD]/g, ''); | ||||
|  | ||||
|             // Translators: this message is shown below the username entry field | ||||
|             // to clue the user in on how to login to the local network realm | ||||
|             this.emit('show-login-hint', | ||||
|                       _("(e.g., user or %s)").format(hint)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onInfoQuery: function(client, serviceName, question) { | ||||
|         // We only expect questions to come from the main auth service | ||||
|         if (serviceName != PASSWORD_SERVICE_NAME) | ||||
|             return; | ||||
|  | ||||
|         this._showRealmLoginHint(); | ||||
|         this._realmLoginHintSignalId = this._realmManager.connect('login-format-changed', | ||||
|                                                                   Lang.bind(this, this._showRealmLoginHint)); | ||||
|  | ||||
|         this.emit('ask-question', serviceName, question, ''); | ||||
|     }, | ||||
|  | ||||
|     _onSecretInfoQuery: function(client, serviceName, secretQuestion) { | ||||
|         // We only expect secret requests to come from the main auth service | ||||
|         if (serviceName != PASSWORD_SERVICE_NAME) | ||||
|             return; | ||||
|  | ||||
|         this.emit('ask-question', serviceName, secretQuestion, '\u25cf'); | ||||
|     }, | ||||
|  | ||||
|     _onReset: function() { | ||||
|         this.clear(); | ||||
|  | ||||
|         // Clear previous attempts to authenticate | ||||
|         this._failCounter = 0; | ||||
|  | ||||
|         this.emit('reset'); | ||||
|     }, | ||||
|  | ||||
|     _onVerificationComplete: function() { | ||||
|         this.emit('verification-complete'); | ||||
|     }, | ||||
|  | ||||
|     _verificationFailed: function(retry) { | ||||
|         // For Not Listed / enterprise logins, immediately reset | ||||
|         // the dialog | ||||
|         // Otherwise, we allow ALLOWED_FAILURES attempts. After that, we | ||||
|         // go back to the welcome screen. | ||||
|  | ||||
|         this._failCounter++; | ||||
|         let canRetry = retry && this._userName && | ||||
|             this._failCounter < this._settings.get_int(ALLOWED_FAILURES_KEY); | ||||
|  | ||||
|         if (canRetry) { | ||||
|             this.clear(); | ||||
|             this.begin(this._userName, new Batch.Hold()); | ||||
|         } else { | ||||
|             // Allow some time to see the message, then reset everything | ||||
|             Mainloop.timeout_add(3000, Lang.bind(this, function() { | ||||
|                 this.cancel(); | ||||
|  | ||||
|                 this._onReset(); | ||||
|             })); | ||||
|         } | ||||
|  | ||||
|         this.emit('verification-failed'); | ||||
|     }, | ||||
|  | ||||
|     _onConversationStopped: function(client, serviceName) { | ||||
|         // if the password service fails, then cancel everything. | ||||
|         // But if, e.g., fingerprint fails, still give | ||||
|         // password authentication a chance to succeed | ||||
|         if (serviceName == PASSWORD_SERVICE_NAME) { | ||||
|             this._verificationFailed(true); | ||||
|         } | ||||
|  | ||||
|         this.emit('hide-login-hint'); | ||||
|  | ||||
|         if (this._realmLoginHintSignalId) { | ||||
|             this._realmManager.disconnect(this._realmLoginHintSignalId); | ||||
|             this._realmLoginHintSignalId = 0; | ||||
|         } | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(ShellUserVerifier.prototype); | ||||
| @@ -4,8 +4,12 @@ | ||||
| const PACKAGE_NAME = '@PACKAGE_NAME@'; | ||||
| /* The version of this package */ | ||||
| const PACKAGE_VERSION = '@PACKAGE_VERSION@'; | ||||
| /* The version of GJS we're linking to */ | ||||
| const GJS_VERSION = '@GJS_VERSION@'; | ||||
| /* 1 if gnome-bluetooth is available, 0 otherwise */ | ||||
| const HAVE_BLUETOOTH = @HAVE_BLUETOOTH@; | ||||
| /* The system TLS CA list */ | ||||
| const SHELL_SYSTEM_CA_FILE = '@SHELL_SYSTEM_CA_FILE@'; | ||||
| /* gettext package */ | ||||
| const GETTEXT_PACKAGE = '@GETTEXT_PACKAGE@'; | ||||
| /* locale dir */ | ||||
|   | ||||
| @@ -3,21 +3,20 @@ | ||||
| // Common utils for the extension system and the extension | ||||
| // preferences tool | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const ShellJS = imports.gi.ShellJS; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const FileUtils = imports.misc.fileUtils; | ||||
|  | ||||
| const ExtensionType = { | ||||
|     SYSTEM: 1, | ||||
|     PER_USER: 2 | ||||
| }; | ||||
|  | ||||
| // GFile for user extensions | ||||
| var userExtensionsDir = null; | ||||
|  | ||||
| // Maps uuid -> metadata object | ||||
| const extensions = {}; | ||||
|  | ||||
| @@ -41,18 +40,13 @@ function getCurrentExtension() { | ||||
|         throw new Error('Could not find current extension'); | ||||
|  | ||||
|     let path = match[1]; | ||||
|     let file = Gio.File.new_for_path(path); | ||||
|     let uuid = GLib.path_get_basename(GLib.path_get_dirname(path)); | ||||
|  | ||||
|     // Walk up the directory tree, looking for an extesion with | ||||
|     // the same UUID as a directory name. | ||||
|     while (file != null) { | ||||
|         let extension = extensions[file.get_basename()]; | ||||
|         if (extension !== undefined) | ||||
|             return extension; | ||||
|         file = file.get_parent(); | ||||
|     } | ||||
|     let extension = extensions[uuid]; | ||||
|     if (extension === undefined) | ||||
|         throw new Error('Could not find current extension'); | ||||
|  | ||||
|     throw new Error('Could not find current extension'); | ||||
|     return extension; | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -89,6 +83,9 @@ function isOutOfDate(extension) { | ||||
|     if (!versionCheck(extension.metadata['shell-version'], Config.PACKAGE_VERSION)) | ||||
|         return true; | ||||
|  | ||||
|     if (extension.metadata['js-version'] && !versionCheck(extension.metadata['js-version'], Config.GJS_VERSION)) | ||||
|         return true; | ||||
|  | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| @@ -121,6 +118,11 @@ function createExtensionObject(uuid, dir, type) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Encourage people to add this | ||||
|     if (!meta.url) { | ||||
|         global.log('Warning: Missing "url" property in metadata.json'); | ||||
|     } | ||||
|  | ||||
|     if (uuid != meta.uuid) { | ||||
|         throw new Error('uuid "' + meta.uuid + '" from metadata.json does not match directory name "' + uuid + '"'); | ||||
|     } | ||||
| @@ -148,38 +150,45 @@ function installImporter(extension) { | ||||
|     _extension = null; | ||||
| } | ||||
|  | ||||
| const ExtensionFinder = new Lang.Class({ | ||||
|     Name: 'ExtensionFinder', | ||||
| function init() { | ||||
|     let userExtensionsPath = GLib.build_filenamev([global.userdatadir, 'extensions']); | ||||
|     userExtensionsDir = Gio.file_new_for_path(userExtensionsPath); | ||||
|     try { | ||||
|         if (!userExtensionsDir.query_exists(null)) | ||||
|             userExtensionsDir.make_directory_with_parents(null); | ||||
|     } catch (e) { | ||||
|         global.logError('' + e); | ||||
|     } | ||||
| } | ||||
|  | ||||
|     _loadExtension: function(extensionDir, info, perUserDir) { | ||||
| function scanExtensionsInDirectory(callback, dir, type) { | ||||
|     let fileEnum; | ||||
|     let file, info; | ||||
|     try { | ||||
|         fileEnum = dir.enumerate_children('standard::*', Gio.FileQueryInfoFlags.NONE, null); | ||||
|     } catch(e) { | ||||
|         global.logError('' + e); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     while ((info = fileEnum.next_file(null)) != null) { | ||||
|         let fileType = info.get_file_type(); | ||||
|         if (fileType != Gio.FileType.DIRECTORY) | ||||
|             return; | ||||
|             continue; | ||||
|         let uuid = info.get_name(); | ||||
|         let existing = extensions[uuid]; | ||||
|         if (existing) { | ||||
|             log('Extension %s already installed in %s. %s will not be loaded'.format(uuid, existing.path, extensionDir.get_path())); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let extension; | ||||
|         let type = extensionDir.has_prefix(perUserDir) ? ExtensionType.PER_USER | ||||
|                                                        : ExtensionType.SYSTEM; | ||||
|         try { | ||||
|             extension = createExtensionObject(uuid, extensionDir, type); | ||||
|         } catch(e) { | ||||
|             logError(e, 'Could not load extension %s'.format(uuid)); | ||||
|             return; | ||||
|         } | ||||
|         this.emit('extension-found', extension); | ||||
|     }, | ||||
|  | ||||
|     scanExtensions: function() { | ||||
|         let perUserDir = Gio.File.new_for_path(global.userdatadir); | ||||
|         FileUtils.collectFromDatadirsAsync('extensions', | ||||
|                                            { processFile: Lang.bind(this, this._loadExtension), | ||||
|                                              includeUserDir: true, | ||||
|                                              data: perUserDir }); | ||||
|         let extensionDir = dir.get_child(uuid); | ||||
|         callback(uuid, extensionDir, type); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(ExtensionFinder.prototype); | ||||
|     fileEnum.close(null); | ||||
| } | ||||
|  | ||||
| function scanExtensions(callback) { | ||||
|     let systemDataDirs = GLib.get_system_data_dirs(); | ||||
|     for (let i = 0; i < systemDataDirs.length; i++) { | ||||
|         let dirPath = GLib.build_filenamev([systemDataDirs[i], 'gnome-shell', 'extensions']); | ||||
|         let dir = Gio.file_new_for_path(dirPath); | ||||
|         if (dir.query_exists(null)) | ||||
|             scanExtensionsInDirectory(callback, dir, ExtensionType.SYSTEM); | ||||
|     } | ||||
|     scanExtensionsInDirectory(callback, userExtensionsDir, ExtensionType.PER_USER); | ||||
| } | ||||
|   | ||||
| @@ -2,12 +2,10 @@ | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| function listDirAsync(file, callback) { | ||||
|     let allFiles = []; | ||||
|     file.enumerate_children_async('standard::name,standard::type', | ||||
|     file.enumerate_children_async(Gio.FILE_ATTRIBUTE_STANDARD_NAME, | ||||
|                                   Gio.FileQueryInfoFlags.NONE, | ||||
|                                   GLib.PRIORITY_LOW, null, function (obj, res) { | ||||
|         let enumerator = obj.enumerate_children_finish(res); | ||||
| @@ -25,69 +23,12 @@ function listDirAsync(file, callback) { | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function _collectFromDirectoryAsync(dir, loadState) { | ||||
|     function done() { | ||||
|         loadState.numLoading--; | ||||
|         if (loadState.loadedCallback && | ||||
|             loadState.numLoading == 0) | ||||
|             loadState.loadedCallback(loadState.data); | ||||
|     } | ||||
|  | ||||
|     dir.query_info_async('standard::type', Gio.FileQueryInfoFlags.NONE, | ||||
|         GLib.PRIORITY_DEFAULT, null, function(object, res) { | ||||
|             try { | ||||
|                 object.query_info_finish(res); | ||||
|             } catch (e) { | ||||
|                 if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.NOT_FOUND)) | ||||
|                     log(e.message); | ||||
|                 done(); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             listDirAsync(dir, Lang.bind(this, function(infos) { | ||||
|                 for (let i = 0; i < infos.length; i++) | ||||
|                     loadState.processFile(dir.get_child(infos[i].get_name()), | ||||
|                                           infos[i], loadState.data); | ||||
|                 done(); | ||||
|             })); | ||||
|         }); | ||||
| } | ||||
|  | ||||
| function collectFromDatadirsAsync(subdir, params) { | ||||
|     params = Params.parse(params, { includeUserDir: false, | ||||
|                                     processFile: null, | ||||
|                                     loadedCallback: null, | ||||
|                                     data: null }); | ||||
|     let loadState = { data: params.data, | ||||
|                       numLoading: 0, | ||||
|                       loadedCallback: params.loadedCallback, | ||||
|                       processFile: params.processFile }; | ||||
|  | ||||
|     if (params.processFile == null) { | ||||
|         if (params.loadedCallback) | ||||
|             params.loadedCallback(params.data); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     let dataDirs = GLib.get_system_data_dirs(); | ||||
|     if (params.includeUserDir) | ||||
|         dataDirs.unshift(GLib.get_user_data_dir()); | ||||
|     loadState.numLoading = dataDirs.length; | ||||
|  | ||||
|     for (let i = 0; i < dataDirs.length; i++) { | ||||
|         let path = GLib.build_filenamev([dataDirs[i], 'gnome-shell', subdir]); | ||||
|         let dir = Gio.File.new_for_path(path); | ||||
|  | ||||
|         _collectFromDirectoryAsync(dir, loadState); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function deleteGFile(file) { | ||||
|     // Work around 'delete' being a keyword in JS. | ||||
|     return file['delete'](null); | ||||
| } | ||||
|  | ||||
| function recursivelyDeleteDir(dir, deleteParent) { | ||||
| function recursivelyDeleteDir(dir) { | ||||
|     let children = dir.enumerate_children('standard::name,standard::type', | ||||
|                                           Gio.FileQueryInfoFlags.NONE, null); | ||||
|  | ||||
| @@ -98,29 +39,8 @@ function recursivelyDeleteDir(dir, deleteParent) { | ||||
|         if (type == Gio.FileType.REGULAR) | ||||
|             deleteGFile(child); | ||||
|         else if (type == Gio.FileType.DIRECTORY) | ||||
|             recursivelyDeleteDir(child, true); | ||||
|             recursivelyDeleteDir(child); | ||||
|     } | ||||
|  | ||||
|     if (deleteParent) | ||||
|         deleteGFile(dir); | ||||
| } | ||||
|  | ||||
| function recursivelyMoveDir(srcDir, destDir) { | ||||
|     let children = srcDir.enumerate_children('standard::name,standard::type', | ||||
|                                              Gio.FileQueryInfoFlags.NONE, null); | ||||
|  | ||||
|     if (!destDir.query_exists(null)) | ||||
|         destDir.make_directory_with_parents(null); | ||||
|  | ||||
|     let info, child; | ||||
|     while ((info = children.next_file(null)) != null) { | ||||
|         let type = info.get_file_type(); | ||||
|         let srcChild = srcDir.get_child(info.get_name()); | ||||
|         let destChild = destDir.get_child(info.get_name()); | ||||
|         log([srcChild.get_path(), destChild.get_path()]); | ||||
|         if (type == Gio.FileType.REGULAR) | ||||
|             srcChild.move(destChild, Gio.FileCopyFlags.NONE, null, null); | ||||
|         else if (type == Gio.FileType.DIRECTORY) | ||||
|             recursivelyMoveDir(srcChild, destChild); | ||||
|     } | ||||
|     deleteGFile(dir); | ||||
| } | ||||
|   | ||||
							
								
								
									
										60
									
								
								js/misc/format.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| /* | ||||
|  * This function is intended to extend the String object and provide | ||||
|  * an String.format API for string formatting. | ||||
|  * It has to be set up using String.prototype.format = Format.format; | ||||
|  * Usage: | ||||
|  * "somestring %s %d".format('hello', 5); | ||||
|  * It supports %s, %d, %x and %f, for %f it also support precisions like | ||||
|  * "%.2f".format(1.526). All specifiers can be prefixed with a minimum | ||||
|  * field width, e.g. "%5s".format("foo"). Unless the width is prefixed | ||||
|  * with '0', the formatted string will be padded with spaces. | ||||
|  */ | ||||
|  | ||||
| function format() { | ||||
|     let str = this; | ||||
|     let i = 0; | ||||
|     let args = arguments; | ||||
|  | ||||
|     return str.replace(/%([0-9]+)?(?:\.([0-9]+))?(.)/g, function (str, widthGroup, precisionGroup, genericGroup) { | ||||
|  | ||||
|                     if (precisionGroup != '' && genericGroup != 'f') | ||||
|                         throw new Error("Precision can only be specified for 'f'"); | ||||
|  | ||||
|                     let fillChar = (widthGroup[0] == '0') ? '0' : ' '; | ||||
|                     let width = parseInt(widthGroup, 10) || 0; | ||||
|  | ||||
|                     function fillWidth(s, c, w) { | ||||
|                         let fill = ''; | ||||
|                         for (let i = 0; i < w; i++) | ||||
|                             fill += c; | ||||
|                         return fill.substr(s.length) + s; | ||||
|                     } | ||||
|  | ||||
|                     let s = ''; | ||||
|                     switch (genericGroup) { | ||||
|                         case '%': | ||||
|                             return '%'; | ||||
|                             break; | ||||
|                         case 's': | ||||
|                             s = args[i++].toString(); | ||||
|                             break; | ||||
|                         case 'd': | ||||
|                             s = parseInt(args[i++]).toString(); | ||||
|                             break; | ||||
|                         case 'x': | ||||
|                             s = parseInt(args[i++]).toString(16); | ||||
|                             break; | ||||
|                         case 'f': | ||||
|                             if (precisionGroup == '') | ||||
|                                 s = parseFloat(args[i++]).toString(); | ||||
|                             else | ||||
|                                 s = parseFloat(args[i++]).toFixed(parseInt(precisionGroup)); | ||||
|                             break; | ||||
|                         default: | ||||
|                             throw new Error('Unsupported conversion character %' + genericGroup); | ||||
|                     } | ||||
|                     return fillWidth(s, fillChar, width); | ||||
|                 }); | ||||
| } | ||||
| @@ -50,21 +50,9 @@ const SessionManagerIface = <interface name="org.gnome.SessionManager"> | ||||
|     <arg type="u" direction="in" /> | ||||
| </method> | ||||
| <method name="Shutdown" /> | ||||
| <method name="Reboot" /> | ||||
| <method name="CanShutdown"> | ||||
|     <arg type="b" direction="out" /> | ||||
| </method> | ||||
| <method name="IsInhibited"> | ||||
|     <arg type="u" direction="in" /> | ||||
|     <arg type="b" direction="out" /> | ||||
| </method> | ||||
| <property name="SessionIsActive" type="b" access="read"/> | ||||
| <signal name="InhibitorAdded"> | ||||
|     <arg type="o" direction="out"/> | ||||
| </signal> | ||||
| <signal name="InhibitorRemoved"> | ||||
|     <arg type="o" direction="out"/> | ||||
| </signal> | ||||
| </interface>; | ||||
|  | ||||
| var SessionManagerProxy = Gio.DBusProxy.makeProxyWrapper(SessionManagerIface); | ||||
|   | ||||
							
								
								
									
										141
									
								
								js/misc/hash.js
									
									
									
									
									
								
							
							
						
						| @@ -1,141 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const System = imports.system; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| // This is an implementation of EcmaScript SameValue algorithm, | ||||
| // which returns true if two values are not observably distinguishable. | ||||
| // It was taken from http://wiki.ecmascript.org/doku.php?id=harmony:egal | ||||
| // | ||||
| // In the future, we may want to use the 'is' operator instead. | ||||
| function _sameValue(x, y) { | ||||
|     if (x === y) { | ||||
|         // 0 === -0, but they are not identical | ||||
|         return x !== 0 || 1 / x === 1 / y; | ||||
|     } | ||||
|  | ||||
|     // NaN !== NaN, but they are identical. | ||||
|     // NaNs are the only non-reflexive value, i.e., if x !== x, | ||||
|     // then x is a NaN. | ||||
|     // isNaN is broken: it converts its argument to number, so | ||||
|     // isNaN("foo") => true | ||||
|     return x !== x && y !== y; | ||||
| } | ||||
|  | ||||
| const _hashers = { | ||||
|     object: function(o) { return o ? System.addressOf(o) : 'null'; }, | ||||
|     function: function(f) { return System.addressOf(f); }, | ||||
|     string: function(s) { return s; }, | ||||
|     number: function(n) { return String(n); }, | ||||
|     undefined: function() { return 'undefined'; }, | ||||
| }; | ||||
|  | ||||
| /* Map is meant to be similar in usage to ES6 Map, which is | ||||
|    described at http://wiki.ecmascript.org/doku.php?id=harmony:simple_maps_and_sets, | ||||
|    without requiring more than ES5 + Gjs extensions. | ||||
|  | ||||
|    Known differences from other implementations: | ||||
|    Polyfills around the web usually implement HashMaps for | ||||
|    primitive values and reversed WeakMaps for object keys, | ||||
|    but we want real maps with real O(1) semantics in all cases, | ||||
|    and the easiest way is to have different hashers for different | ||||
|    types. | ||||
|  | ||||
|    Known differences from the ES6 specification: | ||||
|    - Map is a Lang.Class, not a ES6 class, so inheritance, | ||||
|      prototype, sealing, etc. work differently. | ||||
|    - items(), keys() and values() don't return iterators, | ||||
|      they return actual arrays, so they incur a full copy everytime | ||||
|      they're called, and they don't see changes if you mutate | ||||
|      the table while iterating | ||||
|      (admittedly, the ES6 spec is a bit unclear on this, and | ||||
|      the reference code would just blow up) | ||||
| */ | ||||
| const Map = new Lang.Class({ | ||||
|     Name: 'Map', | ||||
|  | ||||
|     _init: function(iterable) { | ||||
|         this._pool = { }; | ||||
|  | ||||
|         if (iterable) { | ||||
|             for (let i = 0; i < iterable.length; i++) { | ||||
|                 let [key, value] = iterable[i]; | ||||
|                 this.set(key, value); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _hashKey: function(key) { | ||||
|         let type = typeof(key); | ||||
|         return type + ':' + _hashers[type](key); | ||||
|     }, | ||||
|  | ||||
|     _internalGet: function(key) { | ||||
|         let hash = this._hashKey(key); | ||||
|         let node = this._pool[hash]; | ||||
|  | ||||
|         if (node && _sameValue(node.key, key)) | ||||
|             return [true, node.value]; | ||||
|         else | ||||
|             return [false, null]; | ||||
|     }, | ||||
|  | ||||
|     get: function(key) { | ||||
|         return this._internalGet(key)[1]; | ||||
|     }, | ||||
|  | ||||
|     has: function(key) { | ||||
|         return this._internalGet(key)[0]; | ||||
|     }, | ||||
|  | ||||
|     set: function(key, value) { | ||||
|         let hash = this._hashKey(key); | ||||
|         let node = this._pool[hash]; | ||||
|  | ||||
|         if (node) { | ||||
|             node.key = key; | ||||
|             node.value = value; | ||||
|         } else { | ||||
|             this._pool[hash] = { key: key, value: value }; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     delete: function(key) { | ||||
|         let hash = this._hashKey(key); | ||||
|         let node = this._pool[hash]; | ||||
|  | ||||
|         if (node && _sameValue(node.key, key)) { | ||||
|             delete this._pool[hash]; | ||||
|             return [node.key, node.value]; | ||||
|         } else { | ||||
|             return [null, null]; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     keys: function() { | ||||
|         let pool = this._pool; | ||||
|         return Object.getOwnPropertyNames(pool).map(function(hash) { | ||||
|             return pool[hash].key; | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     values: function() { | ||||
|         let pool = this._pool; | ||||
|         return Object.getOwnPropertyNames(pool).map(function(hash) { | ||||
|             return pool[hash].value; | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     items: function() { | ||||
|         let pool = this._pool; | ||||
|         return Object.getOwnPropertyNames(pool).map(function(hash) { | ||||
|             return [pool[hash].key, pool[hash].value]; | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     size: function() { | ||||
|         return Object.getOwnPropertyNames(this._pool).length; | ||||
|     }, | ||||
| }); | ||||
| @@ -41,26 +41,24 @@ const HistoryManager = new Lang.Class({ | ||||
|         this._historyIndex = this._history.length; | ||||
|     }, | ||||
|  | ||||
|     _setPrevItem: function(text) { | ||||
|     prevItem: function(text) { | ||||
|         if (this._historyIndex <= 0) | ||||
|             return false; | ||||
|             return text; | ||||
|  | ||||
|         if (text) | ||||
|             this._history[this._historyIndex] = text; | ||||
|         this._historyIndex--; | ||||
|         this._indexChanged(); | ||||
|         return true; | ||||
|         return this._indexChanged(); | ||||
|     }, | ||||
|  | ||||
|     _setNextItem: function(text) { | ||||
|     nextItem: function(text) { | ||||
|         if (this._historyIndex >= this._history.length) | ||||
|             return false; | ||||
|             return text; | ||||
|  | ||||
|         if (text) | ||||
|             this._history[this._historyIndex] = text; | ||||
|         this._historyIndex++; | ||||
|         this._indexChanged(); | ||||
|         return true; | ||||
|         return this._indexChanged(); | ||||
|     }, | ||||
|  | ||||
|     lastItem: function() { | ||||
| @@ -85,9 +83,11 @@ const HistoryManager = new Lang.Class({ | ||||
|     _onEntryKeyPress: function(entry, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_Up) { | ||||
|             return this._setPrevItem(entry.get_text()); | ||||
|             this.prevItem(entry.get_text()); | ||||
|             return true; | ||||
|         } else if (symbol == Clutter.KEY_Down) { | ||||
|             return this._setNextItem(entry.get_text()); | ||||
|             this.nextItem(entry.get_text()); | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
| @@ -98,6 +98,8 @@ const HistoryManager = new Lang.Class({ | ||||
|  | ||||
|         if (this._entry) | ||||
|             this._entry.set_text(current); | ||||
|  | ||||
|         return current; | ||||
|     }, | ||||
|  | ||||
|     _save: function() { | ||||
|   | ||||
| @@ -1,261 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const SystemdLoginManagerIface = <interface name='org.freedesktop.login1.Manager'> | ||||
| <method name='PowerOff'> | ||||
|     <arg type='b' direction='in'/> | ||||
| </method> | ||||
| <method name='Reboot'> | ||||
|     <arg type='b' direction='in'/> | ||||
| </method> | ||||
| <method name='Suspend'> | ||||
|     <arg type='b' direction='in'/> | ||||
| </method> | ||||
| <method name='CanPowerOff'> | ||||
|     <arg type='s' direction='out'/> | ||||
| </method> | ||||
| <method name='CanReboot'> | ||||
|     <arg type='s' direction='out'/> | ||||
| </method> | ||||
| <method name='CanSuspend'> | ||||
|     <arg type='s' direction='out'/> | ||||
| </method> | ||||
| <method name='Inhibit'> | ||||
|     <arg type='s' direction='in'/> | ||||
|     <arg type='s' direction='in'/> | ||||
|     <arg type='s' direction='in'/> | ||||
|     <arg type='s' direction='in'/> | ||||
|     <arg type='h' direction='out'/> | ||||
| </method> | ||||
| <method name='ListSessions'> | ||||
|     <arg name='sessions' type='a(susso)' direction='out'/> | ||||
| </method> | ||||
| <signal name='PrepareForSleep'> | ||||
|     <arg type='b' direction='out'/> | ||||
| </signal> | ||||
| </interface>; | ||||
|  | ||||
| const SystemdLoginSessionIface = <interface name='org.freedesktop.login1.Session'> | ||||
| <signal name='Lock' /> | ||||
| <signal name='Unlock' /> | ||||
| </interface>; | ||||
|  | ||||
| const SystemdLoginManager = Gio.DBusProxy.makeProxyWrapper(SystemdLoginManagerIface); | ||||
| const SystemdLoginSession = Gio.DBusProxy.makeProxyWrapper(SystemdLoginSessionIface); | ||||
|  | ||||
| const ConsoleKitManagerIface = <interface name='org.freedesktop.ConsoleKit.Manager'> | ||||
| <method name='CanRestart'> | ||||
|     <arg type='b' direction='out'/> | ||||
| </method> | ||||
| <method name='CanStop'> | ||||
|     <arg type='b' direction='out'/> | ||||
| </method> | ||||
| <method name='Restart' /> | ||||
| <method name='Stop' /> | ||||
| <method name='GetCurrentSession'> | ||||
|     <arg type='o' direction='out' /> | ||||
| </method> | ||||
| </interface>; | ||||
|  | ||||
| const ConsoleKitSessionIface = <interface name='org.freedesktop.ConsoleKit.Session'> | ||||
| <signal name='Lock' /> | ||||
| <signal name='Unlock' /> | ||||
| </interface>; | ||||
|  | ||||
| const ConsoleKitSession = Gio.DBusProxy.makeProxyWrapper(ConsoleKitSessionIface); | ||||
| const ConsoleKitManager = Gio.DBusProxy.makeProxyWrapper(ConsoleKitManagerIface); | ||||
|  | ||||
| function haveSystemd() { | ||||
|     return GLib.access("/sys/fs/cgroup/systemd", 0) >= 0; | ||||
| } | ||||
|  | ||||
| let _loginManager = null; | ||||
|  | ||||
| /** | ||||
|  * LoginManager: | ||||
|  * An abstraction over systemd/logind and ConsoleKit. | ||||
|  * | ||||
|  */ | ||||
| function getLoginManager() { | ||||
|     if (_loginManager == null) { | ||||
|         if (haveSystemd()) | ||||
|             _loginManager = new LoginManagerSystemd(); | ||||
|         else | ||||
|             _loginManager = new LoginManagerConsoleKit(); | ||||
|     } | ||||
|  | ||||
|     return _loginManager; | ||||
| } | ||||
|  | ||||
| const LoginManagerSystemd = new Lang.Class({ | ||||
|     Name: 'LoginManagerSystemd', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._proxy = new SystemdLoginManager(Gio.DBus.system, | ||||
|                                               'org.freedesktop.login1', | ||||
|                                               '/org/freedesktop/login1'); | ||||
|         this._proxy.connectSignal('PrepareForSleep', | ||||
|                                   Lang.bind(this, this._prepareForSleep)); | ||||
|     }, | ||||
|  | ||||
|     // Having this function is a bit of a hack since the Systemd and ConsoleKit | ||||
|     // session objects have different interfaces - but in both cases there are | ||||
|     // Lock/Unlock signals, and that's all we count upon at the moment. | ||||
|     getCurrentSessionProxy: function() { | ||||
|         if (!this._currentSession) { | ||||
|             this._currentSession = new SystemdLoginSession(Gio.DBus.system, | ||||
|                                                            'org.freedesktop.login1', | ||||
|                                                            '/org/freedesktop/login1/session/' + | ||||
|                                                            GLib.getenv('XDG_SESSION_ID')); | ||||
|         } | ||||
|  | ||||
|         return this._currentSession; | ||||
|     }, | ||||
|  | ||||
|     canPowerOff: function(asyncCallback) { | ||||
|         this._proxy.CanPowerOffRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback(false); | ||||
|             else | ||||
|                 asyncCallback(result[0] != 'no'); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     canReboot: function(asyncCallback) { | ||||
|         this._proxy.CanRebootRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback(false); | ||||
|             else | ||||
|                 asyncCallback(result[0] != 'no'); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     canSuspend: function(asyncCallback) { | ||||
|         this._proxy.CanSuspendRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback(false); | ||||
|             else | ||||
|                 asyncCallback(result[0] != 'no'); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     listSessions: function(asyncCallback) { | ||||
|         this._proxy.ListSessionsRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback([]); | ||||
|             else | ||||
|                 asyncCallback(result[0]); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     powerOff: function() { | ||||
|         this._proxy.PowerOffRemote(true); | ||||
|     }, | ||||
|  | ||||
|     reboot: function() { | ||||
|         this._proxy.RebootRemote(true); | ||||
|     }, | ||||
|  | ||||
|     suspend: function() { | ||||
|         this._proxy.SuspendRemote(true); | ||||
|     }, | ||||
|  | ||||
|     inhibit: function(reason, callback) { | ||||
|         let inVariant = GLib.Variant.new('(ssss)', | ||||
|                                          ['sleep', | ||||
|                                           'GNOME Shell', | ||||
|                                           reason, | ||||
|                                           'delay']); | ||||
|         this._proxy.call_with_unix_fd_list('Inhibit', inVariant, 0, -1, null, null, | ||||
|             Lang.bind(this, function(proxy, result) { | ||||
|                 let fd = -1; | ||||
|                 try { | ||||
|                     let [outVariant, fdList] = proxy.call_with_unix_fd_list_finish(result); | ||||
|                     fd = fdList.steal_fds(outVariant.deep_unpack())[0]; | ||||
|                     callback(new Gio.UnixInputStream({ fd: fd })); | ||||
|                 } catch(e) { | ||||
|                     logError(e, "Error getting systemd inhibitor"); | ||||
|                     callback(null); | ||||
|                 } | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     _prepareForSleep: function(proxy, sender, [aboutToSuspend]) { | ||||
|         this.emit('prepare-for-sleep', aboutToSuspend); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(LoginManagerSystemd.prototype); | ||||
|  | ||||
| const LoginManagerConsoleKit = new Lang.Class({ | ||||
|     Name: 'LoginManagerConsoleKit', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._proxy = new ConsoleKitManager(Gio.DBus.system, | ||||
|                                             'org.freedesktop.ConsoleKit', | ||||
|                                             '/org/freedesktop/ConsoleKit/Manager'); | ||||
|     }, | ||||
|  | ||||
|     // Having this function is a bit of a hack since the Systemd and ConsoleKit | ||||
|     // session objects have different interfaces - but in both cases there are | ||||
|     // Lock/Unlock signals, and that's all we count upon at the moment. | ||||
|     getCurrentSessionProxy: function() { | ||||
|         if (!this._currentSession) { | ||||
|             let [currentSessionId] = this._proxy.GetCurrentSessionSync(); | ||||
|             this._currentSession = new ConsoleKitSession(Gio.DBus.system, | ||||
|                                                          'org.freedesktop.ConsoleKit', | ||||
|                                                          currentSessionId); | ||||
|         } | ||||
|  | ||||
|         return this._currentSession; | ||||
|     }, | ||||
|  | ||||
|     canPowerOff: function(asyncCallback) { | ||||
|         this._proxy.CanStopRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback(false); | ||||
|             else | ||||
|                 asyncCallback(result[0]); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     canReboot: function(asyncCallback) { | ||||
|         this._proxy.CanRestartRemote(function(result, error) { | ||||
|             if (error) | ||||
|                 asyncCallback(false); | ||||
|             else | ||||
|                 asyncCallback(result[0]); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     canSuspend: function(asyncCallback) { | ||||
|         asyncCallback(false); | ||||
|     }, | ||||
|  | ||||
|     listSessions: function(asyncCallback) { | ||||
|         asyncCallback([]); | ||||
|     }, | ||||
|  | ||||
|     powerOff: function() { | ||||
|         this._proxy.StopRemote(); | ||||
|     }, | ||||
|  | ||||
|     reboot: function() { | ||||
|         this._proxy.RestartRemote(); | ||||
|     }, | ||||
|  | ||||
|     suspend: function() { | ||||
|         this.emit('prepare-for-sleep', true); | ||||
|         this.emit('prepare-for-sleep', false); | ||||
|     }, | ||||
|  | ||||
|     inhibit: function(reason, callback) { | ||||
|         callback(null); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(LoginManagerConsoleKit.prototype); | ||||
| @@ -2,93 +2,9 @@ | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const NMGtk = imports.gi.NMGtk; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| // _getMobileProvidersDatabase: | ||||
| // | ||||
| // Gets the database of mobile providers, with references between MCCMNC/SID and | ||||
| // operator name | ||||
| // | ||||
| let _mpd; | ||||
| function _getMobileProvidersDatabase() { | ||||
|     if (_mpd == null) { | ||||
|         try { | ||||
|             _mpd = new NMGtk.MobileProvidersDatabase(); | ||||
|             _mpd.init(null); | ||||
|         } catch (e) { | ||||
|             log(e.message); | ||||
|             _mpd = null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return _mpd; | ||||
| } | ||||
|  | ||||
| // _findProviderForMccMnc: | ||||
| // @operator_name: operator name | ||||
| // @operator_code: operator code | ||||
| // | ||||
| // Given an operator name string (which may not be a real operator name) and an | ||||
| // operator code string, tries to find a proper operator name to display. | ||||
| // | ||||
| function _findProviderForMccMnc(operator_name, operator_code) { | ||||
|     if (operator_name) { | ||||
|         if (operator_name.length != 0 && | ||||
|             (operator_name.length > 6 || operator_name.length < 5)) { | ||||
|             // this looks like a valid name, i.e. not an MCCMNC (that some | ||||
|             // devices return when not yet connected | ||||
|             return operator_name; | ||||
|         } | ||||
|  | ||||
|         if (isNaN(parseInt(operator_name))) { | ||||
|             // name is definitely not a MCCMNC, so it may be a name | ||||
|             // after all; return that | ||||
|             return operator_name; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let needle; | ||||
|     if ((!operator_name || operator_name.length == 0) && operator_code) | ||||
|         needle = operator_code; | ||||
|     else if (operator_name && (operator_name.length == 6 || operator_name.length == 5)) | ||||
|         needle = operator_name; | ||||
|     else // nothing to search | ||||
|         return null; | ||||
|  | ||||
|     let mpd = _getMobileProvidersDatabase(); | ||||
|     if (mpd) { | ||||
|         let provider = mpd.lookup_3gpp_mcc_mnc(needle); | ||||
|         if (provider) | ||||
|             return provider.get_name(); | ||||
|     } | ||||
|     return null; | ||||
| } | ||||
|  | ||||
| // _findProviderForSid: | ||||
| // @sid: System Identifier of the serving CDMA network | ||||
| // | ||||
| // Tries to find the operator name corresponding to the given SID | ||||
| // | ||||
| function _findProviderForSid(sid) { | ||||
|     if (sid == 0) | ||||
|         return null; | ||||
|  | ||||
|     let mpd = _getMobileProvidersDatabase(); | ||||
|     if (mpd) { | ||||
|         let provider = mpd.lookup_cdma_sid(sid); | ||||
|         if (provider) | ||||
|             return provider.get_name(); | ||||
|     } | ||||
|     return null; | ||||
| } | ||||
|  | ||||
|  | ||||
| //------------------------------------------------------------------------------ | ||||
| // Support for the old ModemManager interface (MM < 0.7) | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
|  | ||||
| // The following are not the complete interfaces, just the methods we need | ||||
| // (or may need in the future) | ||||
|  | ||||
| @@ -126,6 +42,14 @@ const ModemCdmaInterface = <interface name="org.freedesktop.ModemManager.Modem.C | ||||
|  | ||||
| const ModemCdmaProxy = Gio.DBusProxy.makeProxyWrapper(ModemCdmaInterface); | ||||
|  | ||||
| let _providersTable; | ||||
| function _getProvidersTable() { | ||||
|     if (_providersTable) | ||||
|         return _providersTable; | ||||
|     let [providers, countryCodes] = Shell.mobile_providers_parse(); | ||||
|     return _providersTable = providers; | ||||
| } | ||||
|  | ||||
| const ModemGsm = new Lang.Class({ | ||||
|     Name: 'ModemGsm', | ||||
|  | ||||
| @@ -141,7 +65,7 @@ const ModemGsm = new Lang.Class({ | ||||
|             this.emit('notify::signal-quality'); | ||||
|         })); | ||||
|         this._proxy.connectSignal('RegistrationInfo', Lang.bind(this, function(proxy, sender, [status, code, name]) { | ||||
|             this.operator_name = _findProviderForMccMnc(name, code); | ||||
|             this.operator_name = this._findOperatorName(name, code); | ||||
|             this.emit('notify::operator-name'); | ||||
|         })); | ||||
|         this._proxy.GetRegistrationInfoRemote(Lang.bind(this, function([result], err) { | ||||
| @@ -151,7 +75,7 @@ const ModemGsm = new Lang.Class({ | ||||
|             } | ||||
|  | ||||
|             let [status, code, name] = result; | ||||
|             this.operator_name = _findProviderForMccMnc(name, code); | ||||
|             this.operator_name = this._findOperatorName(name, code); | ||||
|             this.emit('notify::operator-name'); | ||||
|         })); | ||||
|         this._proxy.GetSignalQualityRemote(Lang.bind(this, function(result, err) { | ||||
| @@ -164,6 +88,67 @@ const ModemGsm = new Lang.Class({ | ||||
|             } | ||||
|             this.emit('notify::signal-quality'); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _findOperatorName: function(name, opCode) { | ||||
|         if (name.length != 0 && (name.length > 6 || name.length < 5)) { | ||||
|             // this looks like a valid name, i.e. not an MCCMNC (that some | ||||
|             // devices return when not yet connected | ||||
|             return name; | ||||
|         } | ||||
|         if (isNaN(parseInt(name))) { | ||||
|             // name is definitely not a MCCMNC, so it may be a name | ||||
|             // after all; return that | ||||
|             return name; | ||||
|         } | ||||
|  | ||||
|         let needle; | ||||
|         if (name.length == 0 && opCode) | ||||
|             needle = opCode; | ||||
|         else if (name.length == 6 || name.length == 5) | ||||
|             needle = name; | ||||
|         else // nothing to search | ||||
|             return null; | ||||
|  | ||||
|         return this._findProviderForMCCMNC(needle); | ||||
|     }, | ||||
|  | ||||
|     _findProviderForMCCMNC: function(needle) { | ||||
|         let table = _getProvidersTable(); | ||||
|         let needlemcc = needle.substring(0, 3); | ||||
|         let needlemnc = needle.substring(3, needle.length); | ||||
|  | ||||
|         let name2, name3; | ||||
|         for (let iter in table) { | ||||
|             let providers = table[iter]; | ||||
|  | ||||
|             // Search through each country's providers | ||||
|             for (let i = 0; i < providers.length; i++) { | ||||
|                 let provider = providers[i]; | ||||
|  | ||||
|                 // Search through MCC/MNC list | ||||
|                 let list = provider.get_gsm_mcc_mnc(); | ||||
|                 for (let j = 0; j < list.length; j++) { | ||||
|                     let mccmnc = list[j]; | ||||
|  | ||||
|                     // Match both 2-digit and 3-digit MNC; prefer a | ||||
|                     // 3-digit match if found, otherwise a 2-digit one. | ||||
|                     if (mccmnc.mcc != needlemcc) | ||||
|                         continue;  // MCC was wrong | ||||
|  | ||||
|                     if (!name3 && needle.length == 6 && needlemnc == mccmnc.mnc) | ||||
|                         name3 = provider.name; | ||||
|  | ||||
|                     if (!name2 && needlemnc.substring(0, 2) == mccmnc.mnc.substring(0, 2)) | ||||
|                         name2 = provider.name; | ||||
|  | ||||
|                     if (name2 && name3) | ||||
|                         break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return name3 || name2 || null; | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(ModemGsm.prototype); | ||||
| @@ -203,99 +188,40 @@ const ModemCdma = new Lang.Class({ | ||||
|                 // it will return an error if the device is not connected | ||||
|                 this.operator_name = null; | ||||
|             } else { | ||||
|                 let [bandClass, band, sid] = result; | ||||
|  | ||||
|                 this.operator_name = _findProviderForSid(sid) | ||||
|                 let [bandClass, band, id] = result; | ||||
|                 if (name.length > 0) | ||||
|                     this.operator_name = this._findProviderForSid(id); | ||||
|                 else | ||||
|                     this.operator_name = null; | ||||
|             } | ||||
|             this.emit('notify::operator-name'); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _findProviderForSid: function(sid) { | ||||
|         if (sid == 0) | ||||
|             return null; | ||||
|  | ||||
|         let table = _getProvidersTable(); | ||||
|  | ||||
|         // Search through each country | ||||
|         for (let iter in table) { | ||||
|             let providers = table[iter]; | ||||
|  | ||||
|             // Search through each country's providers | ||||
|             for (let i = 0; i < providers.length; i++) { | ||||
|                 let provider = providers[i]; | ||||
|                 let cdma_sid = provider.get_cdma_sid(); | ||||
|  | ||||
|                 // Search through CDMA SID list | ||||
|                 for (let j = 0; j < cdma_sid.length; j++) { | ||||
|                     if (cdma_sid[j] == sid) | ||||
|                         return provider.name; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return null; | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(ModemCdma.prototype); | ||||
|  | ||||
|  | ||||
| //------------------------------------------------------------------------------ | ||||
| // Support for the new ModemManager1 interface (MM >= 0.7) | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
| const BroadbandModemInterface = <interface name="org.freedesktop.ModemManager1.Modem"> | ||||
| <property name="SignalQuality" type="(ub)" access="read" /> | ||||
| </interface>; | ||||
| const BroadbandModemProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModemInterface); | ||||
|  | ||||
| const BroadbandModem3gppInterface = <interface name="org.freedesktop.ModemManager1.Modem.Modem3gpp"> | ||||
| <property name="OperatorCode" type="s" access="read" /> | ||||
| <property name="OperatorName" type="s" access="read" /> | ||||
| </interface>; | ||||
| const BroadbandModem3gppProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModem3gppInterface); | ||||
|  | ||||
| const BroadbandModemCdmaInterface = <interface name="org.freedesktop.ModemManager1.Modem.ModemCdma"> | ||||
| <property name="Sid" type="u" access="read" /> | ||||
| </interface>; | ||||
| const BroadbandModemCdmaProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModemCdmaInterface); | ||||
|  | ||||
| const BroadbandModem = new Lang.Class({ | ||||
|     Name: 'BroadbandModem', | ||||
|  | ||||
|     _init: function(path, capabilities) { | ||||
|         this._proxy = new BroadbandModemProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|         this._proxy_3gpp = new BroadbandModem3gppProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|         this._proxy_cdma = new BroadbandModemCdmaProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); | ||||
|         this._capabilities = capabilities; | ||||
|  | ||||
|         this._proxy.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { | ||||
|             if ('SignalQuality' in properties.deep_unpack()) | ||||
|                 this._reloadSignalQuality(); | ||||
|         })); | ||||
|         this._reloadSignalQuality(); | ||||
|  | ||||
|         this._proxy_3gpp.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { | ||||
|             let unpacked = properties.deep_unpack(); | ||||
|             if ('OperatorName' in unpacked || 'OperatorCode' in unpacked) | ||||
|                 this._reload3gppOperatorName(); | ||||
|         })); | ||||
|         this._reload3gppOperatorName(); | ||||
|  | ||||
|         this._proxy_cdma.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { | ||||
|             let unpacked = properties.deep_unpack(); | ||||
|             if ('Nid' in unpacked || 'Sid' in unpacked) | ||||
|                 this._reloadCdmaOperatorName(); | ||||
|         })); | ||||
|         this._reloadCdmaOperatorName(); | ||||
|     }, | ||||
|  | ||||
|     _reloadSignalQuality: function() { | ||||
|         let [quality, recent] = this._proxy.SignalQuality; | ||||
|         this.signal_quality = quality; | ||||
|         this.emit('notify::signal-quality'); | ||||
|     }, | ||||
|  | ||||
|     _reloadOperatorName: function() { | ||||
|         let new_name = ""; | ||||
|         if (this.operator_name_3gpp && this.operator_name_3gpp.length > 0) | ||||
|             new_name += this.operator_name_3gpp; | ||||
|  | ||||
|         if (this.operator_name_cdma && this.operator_name_cdma.length > 0) { | ||||
|             if (new_name != "") | ||||
|                 new_name += ", "; | ||||
|             new_name += this.operator_name_cdma; | ||||
|         } | ||||
|  | ||||
|         this.operator_name = new_name; | ||||
|         this.emit('notify::operator-name'); | ||||
|     }, | ||||
|  | ||||
|     _reload3gppOperatorName: function() { | ||||
|         let name = this._proxy_3gpp.OperatorName; | ||||
|         let code = this._proxy_3gpp.OperatorCode; | ||||
|         this.operator_name_3gpp = _findProviderForMccMnc(name, code); | ||||
|         this._reloadOperatorName(); | ||||
|     }, | ||||
|  | ||||
|     _reloadCdmaOperatorName: function() { | ||||
|         let sid = this._proxy_cdma.Sid; | ||||
|         this.operator_name_cdma = _findProviderForSid(sid); | ||||
|         this._reloadOperatorName(); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(BroadbandModem.prototype); | ||||
|   | ||||
							
								
								
									
										48
									
								
								js/misc/screenSaver.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
|  | ||||
| const ScreenSaverIface = <interface name="org.gnome.ScreenSaver"> | ||||
| <method name="GetActive"> | ||||
|     <arg type="b" direction="out" /> | ||||
| </method> | ||||
| <method name="Lock" /> | ||||
| <method name="SetActive"> | ||||
|     <arg type="b" direction="in" /> | ||||
| </method> | ||||
| <signal name="ActiveChanged"> | ||||
|     <arg type="b" direction="out" /> | ||||
| </signal> | ||||
| </interface>; | ||||
|  | ||||
| const ScreenSaverInfo = Gio.DBusInterfaceInfo.new_for_xml(ScreenSaverIface); | ||||
|  | ||||
| function ScreenSaverProxy() { | ||||
|     var self = new Gio.DBusProxy({ g_connection: Gio.DBus.session, | ||||
| 				   g_interface_name: ScreenSaverInfo.name, | ||||
| 				   g_interface_info: ScreenSaverInfo, | ||||
| 				   g_name: 'org.gnome.ScreenSaver', | ||||
| 				   g_object_path: '/org/gnome/ScreenSaver', | ||||
|                                    g_flags: (Gio.DBusProxyFlags.DO_NOT_AUTO_START | | ||||
|                                              Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES) }); | ||||
|     self.init(null); | ||||
|     self.screenSaverActive = false; | ||||
|  | ||||
|     self.connectSignal('ActiveChanged', function(proxy, senderName, [isActive]) { | ||||
|         self.screenSaverActive = isActive; | ||||
|     }); | ||||
|     self.connect('notify::g-name-owner', function() { | ||||
|         if (self.g_name_owner) { | ||||
|             self.GetActiveRemote(function(result, excp) { | ||||
|                 if (result) { | ||||
|                     let [isActive] = result; | ||||
|                     self.screenSaverActive = isActive; | ||||
|                 } | ||||
|             }); | ||||
|         } else | ||||
|             self.screenSaverActive = false; | ||||
|     }); | ||||
|  | ||||
|     return self; | ||||
| } | ||||
							
								
								
									
										138
									
								
								js/misc/util.js
									
									
									
									
									
								
							
							
						
						| @@ -1,8 +1,9 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| @@ -82,33 +83,24 @@ function spawnCommandLine(command_line) { | ||||
| // this will throw an error. | ||||
| function trySpawn(argv) | ||||
| { | ||||
|     var success, pid; | ||||
|     try { | ||||
|         [success, pid] = GLib.spawn_async(null, argv, null, | ||||
|                                           GLib.SpawnFlags.SEARCH_PATH | GLib.SpawnFlags.DO_NOT_REAP_CHILD, | ||||
|                                           null); | ||||
|         GLib.spawn_async(null, argv, null, | ||||
|                          GLib.SpawnFlags.SEARCH_PATH, | ||||
|                          null, null); | ||||
|     } catch (err) { | ||||
|         /* Rewrite the error in case of ENOENT */ | ||||
|         if (err.matches(GLib.SpawnError, GLib.SpawnError.NOENT)) { | ||||
|             throw new GLib.SpawnError({ code: GLib.SpawnError.NOENT, | ||||
|                                         message: _("Command not found") }); | ||||
|         } else if (err instanceof GLib.Error) { | ||||
|         if (err.code == GLib.SpawnError.G_SPAWN_ERROR_NOENT) { | ||||
|             err.message = _("Command not found"); | ||||
|         } else { | ||||
|             // The exception from gjs contains an error string like: | ||||
|             //   Error invoking GLib.spawn_command_line_async: Failed to | ||||
|             //   execute child process "foo" (No such file or directory) | ||||
|             // We are only interested in the part in the parentheses. (And | ||||
|             // we can't pattern match the text, since it gets localized.) | ||||
|             let message = err.message.replace(/.*\((.+)\)/, '$1'); | ||||
|             throw new (err.constructor)({ code: err.code, | ||||
|                                           message: message }); | ||||
|         } else { | ||||
|             throw err; | ||||
|             err.message = err.message.replace(/.*\((.+)\)/, '$1'); | ||||
|         } | ||||
|  | ||||
|         throw err; | ||||
|     } | ||||
|     // Dummy child watch; we don't want to double-fork internally | ||||
|     // because then we lose the parent-child relationship, which | ||||
|     // can break polkit.  See https://bugzilla.redhat.com//show_bug.cgi?id=819275 | ||||
|     GLib.child_watch_add(GLib.PRIORITY_DEFAULT, pid, function () {}, null); | ||||
| } | ||||
|  | ||||
| // trySpawnCommandLine: | ||||
| @@ -152,7 +144,7 @@ function killall(processName) { | ||||
|         // whatever... | ||||
|  | ||||
|         let argv = ['pkill', '-f', '^([^ ]*/)?' + processName + '($| )']; | ||||
|         GLib.spawn_sync(null, argv, null, GLib.SpawnFlags.SEARCH_PATH, null); | ||||
|         GLib.spawn_sync(null, argv, null, GLib.SpawnFlags.SEARCH_PATH, null, null); | ||||
|         // It might be useful to return success/failure, but we'd need | ||||
|         // a wrapper around WIFEXITED and WEXITSTATUS. Since none of | ||||
|         // the current callers care, we don't bother. | ||||
| @@ -161,6 +153,86 @@ function killall(processName) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| // This was ported from network-manager-applet | ||||
| // Copyright 2007 - 2011 Red Hat, Inc. | ||||
| // Author: Dan Williams <dcbw@redhat.com> | ||||
|  | ||||
| const _IGNORED_WORDS = [ | ||||
|         'Semiconductor', | ||||
|         'Components', | ||||
|         'Corporation', | ||||
|         'Communications', | ||||
|         'Company', | ||||
|         'Corp.', | ||||
|         'Corp', | ||||
|         'Co.', | ||||
|         'Inc.', | ||||
|         'Inc', | ||||
|         'Incorporated', | ||||
|         'Ltd.', | ||||
|         'Limited.', | ||||
|         'Intel', | ||||
|         'chipset', | ||||
|         'adapter', | ||||
|         '[hex]', | ||||
|         'NDIS', | ||||
|         'Module' | ||||
| ]; | ||||
|  | ||||
| const _IGNORED_PHRASES = [ | ||||
|         'Multiprotocol MAC/baseband processor', | ||||
|         'Wireless LAN Controller', | ||||
|         'Wireless LAN Adapter', | ||||
|         'Wireless Adapter', | ||||
|         'Network Connection', | ||||
|         'Wireless Cardbus Adapter', | ||||
|         'Wireless CardBus Adapter', | ||||
|         '54 Mbps Wireless PC Card', | ||||
|         'Wireless PC Card', | ||||
|         'Wireless PC', | ||||
|         'PC Card with XJACK(r) Antenna', | ||||
|         'Wireless cardbus', | ||||
|         'Wireless LAN PC Card', | ||||
|         'Technology Group Ltd.', | ||||
|         'Communication S.p.A.', | ||||
|         'Business Mobile Networks BV', | ||||
|         'Mobile Broadband Minicard Composite Device', | ||||
|         'Mobile Communications AB', | ||||
|         '(PC-Suite Mode)' | ||||
| ]; | ||||
|  | ||||
| function fixupPCIDescription(desc) { | ||||
|     desc = desc.replace(/[_,]/, ' '); | ||||
|  | ||||
|     /* Attempt to shorten ID by ignoring certain phrases */ | ||||
|     for (let i = 0; i < _IGNORED_PHRASES.length; i++) { | ||||
|         let item = _IGNORED_PHRASES[i]; | ||||
|         let pos = desc.indexOf(item); | ||||
|         if (pos != -1) { | ||||
|             let before = desc.substring(0, pos); | ||||
|             let after = desc.substring(pos + item.length, desc.length); | ||||
|             desc = before + after; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /* Attmept to shorten ID by ignoring certain individual words */ | ||||
|     let words = desc.split(' '); | ||||
|     let out = [ ]; | ||||
|     for (let i = 0; i < words.length; i++) { | ||||
|         let item = words[i]; | ||||
|  | ||||
|         // skip empty items (that come out from consecutive spaces) | ||||
|         if (item.length == 0) | ||||
|             continue; | ||||
|  | ||||
|         if (_IGNORED_WORDS.indexOf(item) == -1) { | ||||
|             out.push(item); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return out.join(' '); | ||||
| } | ||||
|  | ||||
| // lowerBound: | ||||
| // @array: an array or array-like object, already sorted | ||||
| //         according to @cmp | ||||
| @@ -210,27 +282,3 @@ function insertSorted(array, val, cmp) { | ||||
|  | ||||
|     return pos; | ||||
| } | ||||
|  | ||||
| function makeCloseButton() { | ||||
|     let closeButton = new St.Button({ style_class: 'notification-close'}); | ||||
|  | ||||
|     // This is a bit tricky. St.Bin has its own x-align/y-align properties | ||||
|     // that compete with Clutter's properties. This should be fixed for | ||||
|     // Clutter 2.0. Since St.Bin doesn't define its own setters, the | ||||
|     // setters are a workaround to get Clutter's version. | ||||
|     closeButton.set_x_align(Clutter.ActorAlign.END); | ||||
|     closeButton.set_y_align(Clutter.ActorAlign.START); | ||||
|  | ||||
|     // XXX Clutter 2.0 workaround: ClutterBinLayout needs expand | ||||
|     // to respect the alignments. | ||||
|     closeButton.set_x_expand(true); | ||||
|     closeButton.set_y_expand(true); | ||||
|  | ||||
|     closeButton.connect('style-changed', function() { | ||||
|         let themeNode = closeButton.get_theme_node(); | ||||
|         closeButton.translation_x = themeNode.get_length('-shell-close-overlap-x'); | ||||
|         closeButton.translation_y = themeNode.get_length('-shell-close-overlap-y'); | ||||
|     }); | ||||
|  | ||||
|     return closeButton; | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const System = imports.system; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Scripting = imports.ui.scripting; | ||||
|  | ||||
| @@ -101,7 +99,7 @@ function run() { | ||||
|         Main.overview.hide(); | ||||
|         yield Scripting.waitLeisure(); | ||||
|  | ||||
|         System.gc(); | ||||
|         global.gc(); | ||||
|         yield Scripting.sleep(1000); | ||||
|         Scripting.collectStatistics(); | ||||
|         Scripting.scriptEvent('afterShowHide'); | ||||
| @@ -115,10 +113,10 @@ function run() { | ||||
|  | ||||
|     for (let i = 0; i < 2; i++) { | ||||
|         Scripting.scriptEvent('applicationsShowStart'); | ||||
|         Main.overview._dash.showAppsButton.checked = true; | ||||
|         Main.overview._viewSelector.switchTab('applications'); | ||||
|         yield Scripting.waitLeisure(); | ||||
|         Scripting.scriptEvent('applicationsShowDone'); | ||||
|         Main.overview._dash.showAppsButton.checked = false; | ||||
|         Main.overview._viewSelector.switchTab('windows'); | ||||
|         yield Scripting.waitLeisure(); | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										865
									
								
								js/ui/altTab.js
									
									
									
									
									
								
							
							
						
						| @@ -84,12 +84,9 @@ const AppFavorites = new Lang.Class({ | ||||
|  | ||||
|         let app = Shell.AppSystem.get_default().lookup_app(appId); | ||||
|  | ||||
|         Main.overview.setMessage(_("%s has been added to your favorites.").format(app.get_name()), | ||||
|                                  { forFeedback: true, | ||||
|                                    undoCallback: Lang.bind(this, function () { | ||||
|                                                                this._removeFavorite(appId); | ||||
|                                                            }) | ||||
|                                  }); | ||||
|         Main.overview.setMessage(_("%s has been added to your favorites.").format(app.get_name()), Lang.bind(this, function () { | ||||
|             this._removeFavorite(appId); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     addFavorite: function(appId) { | ||||
| @@ -119,11 +116,9 @@ const AppFavorites = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         Main.overview.setMessage(_("%s has been removed from your favorites.").format(app.get_name()), | ||||
|                                  { forFeedback: true, | ||||
|                                    undoCallback: Lang.bind(this, function () { | ||||
|                                                                this._addFavorite(appId, pos); | ||||
|                                                            }) | ||||
|                                  }); | ||||
|                                  Lang.bind(this, function () { | ||||
|             this._addFavorite(appId, pos); | ||||
|         })); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(AppFavorites.prototype); | ||||
|   | ||||
							
								
								
									
										290
									
								
								js/ui/automountManager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,290 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| const Shell = imports.gi.Shell; | ||||
| const Main = imports.ui.main; | ||||
| const ShellMountOperation = imports.ui.shellMountOperation; | ||||
| const ScreenSaver = imports.misc.screenSaver; | ||||
|  | ||||
| // GSettings keys | ||||
| const SETTINGS_SCHEMA = 'org.gnome.desktop.media-handling'; | ||||
| const SETTING_ENABLE_AUTOMOUNT = 'automount'; | ||||
|  | ||||
| const AUTORUN_EXPIRE_TIMEOUT_SECS = 10; | ||||
|  | ||||
| const ConsoleKitSessionIface = <interface name="org.freedesktop.ConsoleKit.Session"> | ||||
| <method name="IsActive"> | ||||
|     <arg type="b" direction="out" /> | ||||
| </method> | ||||
| <signal name="ActiveChanged"> | ||||
|     <arg type="b" direction="out" /> | ||||
| </signal> | ||||
| </interface>; | ||||
|  | ||||
| const ConsoleKitSessionProxy = Gio.DBusProxy.makeProxyWrapper(ConsoleKitSessionIface); | ||||
|  | ||||
| const ConsoleKitManagerIface = <interface name="org.freedesktop.ConsoleKit.Manager"> | ||||
| <method name="GetCurrentSession"> | ||||
|     <arg type="o" direction="out" /> | ||||
| </method> | ||||
| </interface>; | ||||
|  | ||||
| const ConsoleKitManagerInfo = Gio.DBusInterfaceInfo.new_for_xml(ConsoleKitManagerIface); | ||||
|  | ||||
| function ConsoleKitManager() { | ||||
|     var self = new Gio.DBusProxy({ g_connection: Gio.DBus.system, | ||||
| 				   g_interface_name: ConsoleKitManagerInfo.name, | ||||
| 				   g_interface_info: ConsoleKitManagerInfo, | ||||
| 				   g_name: 'org.freedesktop.ConsoleKit', | ||||
| 				   g_object_path: '/org/freedesktop/ConsoleKit/Manager', | ||||
|                                    g_flags: (Gio.DBusProxyFlags.DO_NOT_AUTO_START | | ||||
|                                              Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES) }); | ||||
|  | ||||
|     self._updateSessionActive = function() { | ||||
|         if (self.g_name_owner) { | ||||
|             self.GetCurrentSessionRemote(function([session]) { | ||||
|                 self._ckSession = new ConsoleKitSessionProxy(Gio.DBus.system, 'org.freedesktop.ConsoleKit', session); | ||||
|  | ||||
|                 self._ckSession.connectSignal('ActiveChanged', function(object, senderName, [isActive]) { | ||||
|                     self.sessionActive = isActive; | ||||
|                 }); | ||||
|                 self._ckSession.IsActiveRemote(function([isActive]) { | ||||
|                     self.sessionActive = isActive; | ||||
|                 }); | ||||
|             }); | ||||
|         } else { | ||||
|             self.sessionActive = true; | ||||
|         } | ||||
|     }; | ||||
|     self.connect('notify::g-name-owner', | ||||
|                  Lang.bind(self, self._updateSessionActive)); | ||||
|  | ||||
|     self._updateSessionActive(); | ||||
|     self.init(null); | ||||
|     return self; | ||||
| } | ||||
|  | ||||
| function haveSystemd() { | ||||
|     return GLib.access("/sys/fs/cgroup/systemd", 0) >= 0; | ||||
| } | ||||
|  | ||||
| const AutomountManager = new Lang.Class({ | ||||
|     Name: 'AutomountManager', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|         this._volumeQueue = []; | ||||
|  | ||||
|         if (!haveSystemd()) | ||||
|             this.ckListener = new ConsoleKitManager(); | ||||
|  | ||||
|         this._ssProxy = new ScreenSaver.ScreenSaverProxy(); | ||||
|         this._ssProxy.connectSignal('ActiveChanged', | ||||
|                                     Lang.bind(this, this._screenSaverActiveChanged)); | ||||
|  | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
|  | ||||
|         this._volumeMonitor.connect('volume-added', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onVolumeAdded)); | ||||
|         this._volumeMonitor.connect('volume-removed', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onVolumeRemoved)); | ||||
|         this._volumeMonitor.connect('drive-connected', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onDriveConnected)); | ||||
|         this._volumeMonitor.connect('drive-disconnected', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onDriveDisconnected)); | ||||
|         this._volumeMonitor.connect('drive-eject-button', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onDriveEjectButton)); | ||||
|  | ||||
|         Mainloop.idle_add(Lang.bind(this, this._startupMountAll)); | ||||
|     }, | ||||
|  | ||||
|     _screenSaverActiveChanged: function(object, senderName, [isActive]) { | ||||
|         if (!isActive) { | ||||
|             this._volumeQueue.forEach(Lang.bind(this, function(volume) { | ||||
|                 this._checkAndMountVolume(volume); | ||||
|             })); | ||||
|         } | ||||
|  | ||||
|         // clear the queue anyway | ||||
|         this._volumeQueue = []; | ||||
|     }, | ||||
|  | ||||
|     _startupMountAll: function() { | ||||
|         let volumes = this._volumeMonitor.get_volumes(); | ||||
|         volumes.forEach(Lang.bind(this, function(volume) { | ||||
|             this._checkAndMountVolume(volume, { checkSession: false, | ||||
|                                                 useMountOp: false }); | ||||
|         })); | ||||
|  | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     isSessionActive: function() { | ||||
|         // Return whether the current session is active, using the | ||||
|         // right mechanism: either systemd if available or ConsoleKit | ||||
|         // as fallback. | ||||
|  | ||||
|         if (haveSystemd()) | ||||
|             return Shell.session_is_active_for_systemd(); | ||||
|  | ||||
|         return this.ckListener.sessionActive; | ||||
|     }, | ||||
|  | ||||
|     _onDriveConnected: function() { | ||||
|         // if we're not in the current ConsoleKit session, | ||||
|         // or screensaver is active, don't play sounds | ||||
|         if (!this.isSessionActive()) | ||||
|             return; | ||||
|  | ||||
|         if (this._ssProxy.screenSaverActive) | ||||
|             return; | ||||
|  | ||||
|         global.play_theme_sound(0, 'device-added-media'); | ||||
|     }, | ||||
|  | ||||
|     _onDriveDisconnected: function() { | ||||
|         // if we're not in the current ConsoleKit session, | ||||
|         // or screensaver is active, don't play sounds | ||||
|         if (!this.isSessionActive()) | ||||
|             return; | ||||
|  | ||||
|         if (this._ssProxy.screenSaverActive) | ||||
|             return; | ||||
|  | ||||
|         global.play_theme_sound(0, 'device-removed-media');         | ||||
|     }, | ||||
|  | ||||
|     _onDriveEjectButton: function(monitor, drive) { | ||||
|         // TODO: this code path is not tested, as the GVfs volume monitor | ||||
|         // doesn't emit this signal just yet. | ||||
|         if (!this.isSessionActive()) | ||||
|             return; | ||||
|  | ||||
|         // we force stop/eject in this case, so we don't have to pass a | ||||
|         // mount operation object | ||||
|         if (drive.can_stop()) { | ||||
|             drive.stop | ||||
|                 (Gio.MountUnmountFlags.FORCE, null, null, | ||||
|                  Lang.bind(this, function(drive, res) { | ||||
|                      try { | ||||
|                          drive.stop_finish(res); | ||||
|                      } catch (e) { | ||||
|                          log("Unable to stop the drive after drive-eject-button " + e.toString()); | ||||
|                      } | ||||
|                  })); | ||||
|         } else if (drive.can_eject()) { | ||||
|             drive.eject_with_operation  | ||||
|                 (Gio.MountUnmountFlags.FORCE, null, null, | ||||
|                  Lang.bind(this, function(drive, res) { | ||||
|                      try { | ||||
|                          drive.eject_with_operation_finish(res); | ||||
|                      } catch (e) { | ||||
|                          log("Unable to eject the drive after drive-eject-button " + e.toString()); | ||||
|                      } | ||||
|                  })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onVolumeAdded: function(monitor, volume) { | ||||
|         this._checkAndMountVolume(volume); | ||||
|     }, | ||||
|  | ||||
|     _checkAndMountVolume: function(volume, params) { | ||||
|         params = Params.parse(params, { checkSession: true, | ||||
|                                         useMountOp: true }); | ||||
|  | ||||
|         if (params.checkSession) { | ||||
|             // if we're not in the current ConsoleKit session, | ||||
|             // don't attempt automount | ||||
|             if (!this.isSessionActive()) | ||||
|                 return; | ||||
|  | ||||
|             if (this._ssProxy.screenSaverActive) { | ||||
|                 if (this._volumeQueue.indexOf(volume) == -1) | ||||
|                     this._volumeQueue.push(volume); | ||||
|  | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Volume is already mounted, don't bother. | ||||
|         if (volume.get_mount()) | ||||
|             return; | ||||
|  | ||||
|         if (!this._settings.get_boolean(SETTING_ENABLE_AUTOMOUNT) || | ||||
|             !volume.should_automount() || | ||||
|             !volume.can_mount()) { | ||||
|             // allow the autorun to run anyway; this can happen if the | ||||
|             // mount gets added programmatically later, even if  | ||||
|             // should_automount() or can_mount() are false, like for | ||||
|             // blank optical media. | ||||
|             this._allowAutorun(volume); | ||||
|             this._allowAutorunExpire(volume); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (params.useMountOp) { | ||||
|             let operation = new ShellMountOperation.ShellMountOperation(volume); | ||||
|             this._mountVolume(volume, operation.mountOp); | ||||
|         } else { | ||||
|             this._mountVolume(volume, null); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _mountVolume: function(volume, operation) { | ||||
|         this._allowAutorun(volume); | ||||
|         volume.mount(0, operation, null, | ||||
|                      Lang.bind(this, this._onVolumeMounted)); | ||||
|     }, | ||||
|  | ||||
|     _onVolumeMounted: function(volume, res) { | ||||
|         this._allowAutorunExpire(volume); | ||||
|  | ||||
|         try { | ||||
|             volume.mount_finish(res); | ||||
|         } catch (e) { | ||||
|             let string = e.toString(); | ||||
|  | ||||
|             // FIXME: needs proper error code handling instead of this | ||||
|             // See https://bugzilla.gnome.org/show_bug.cgi?id=591480 | ||||
|             if (string.indexOf('No key available with this passphrase') != -1) | ||||
|                 this._reaskPassword(volume); | ||||
|             else | ||||
|                 log('Unable to mount volume ' + volume.get_name() + ': ' + string); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onVolumeRemoved: function(monitor, volume) { | ||||
|         this._volumeQueue =  | ||||
|             this._volumeQueue.filter(function(element) { | ||||
|                 return (element != volume); | ||||
|             }); | ||||
|     }, | ||||
|  | ||||
|     _reaskPassword: function(volume) { | ||||
|         let operation = new ShellMountOperation.ShellMountOperation(volume, { reaskPassword: true }); | ||||
|         this._mountVolume(volume, operation.mountOp);         | ||||
|     }, | ||||
|  | ||||
|     _allowAutorun: function(volume) { | ||||
|         volume.allowAutorun = true; | ||||
|     }, | ||||
|  | ||||
|     _allowAutorunExpire: function(volume) { | ||||
|         Mainloop.timeout_add_seconds(AUTORUN_EXPIRE_TIMEOUT_SECS, function() { | ||||
|             volume.allowAutorun = false; | ||||
|             return false; | ||||
|         }); | ||||
|     } | ||||
| }); | ||||
| @@ -4,7 +4,6 @@ const Lang = imports.lang; | ||||
| const Gio = imports.gi.Gio; | ||||
| const St = imports.gi.St; | ||||
| 
 | ||||
| const GnomeSession = imports.misc.gnomeSession; | ||||
| const Main = imports.ui.main; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
| const ShellMountOperation = imports.ui.shellMountOperation; | ||||
| @@ -24,14 +23,12 @@ const AutorunSetting = { | ||||
| }; | ||||
| 
 | ||||
| // misc utils
 | ||||
| function shouldAutorunMount(mount, forTransient) { | ||||
| function ignoreAutorunForMount(mount) { | ||||
|     let root = mount.get_root(); | ||||
|     let volume = mount.get_volume(); | ||||
| 
 | ||||
|     if (!volume || (!volume.allowAutorun && forTransient)) | ||||
|         return false; | ||||
| 
 | ||||
|     if (!root.is_native() || isMountRootHidden(root)) | ||||
|     if ((root.is_native() && !isMountRootHidden(root)) || | ||||
|         (volume && volume.allowAutorun && volume.should_automount())) | ||||
|         return false; | ||||
| 
 | ||||
|     return true; | ||||
| @@ -44,17 +41,6 @@ function isMountRootHidden(root) { | ||||
|     return (path.indexOf('/.') != -1); | ||||
| } | ||||
| 
 | ||||
| function isMountNonLocal(mount) { | ||||
|     // If the mount doesn't have an associated volume, that means it's
 | ||||
|     // an uninteresting filesystem. Most devices that we care about will
 | ||||
|     // have a mount, like media players and USB sticks.
 | ||||
|     let volume = mount.get_volume(); | ||||
|     if (volume == null) | ||||
|         return true; | ||||
| 
 | ||||
|     return (volume.get_identifier("class") == "network"); | ||||
| } | ||||
| 
 | ||||
| function startAppForMount(app, mount) { | ||||
|     let files = []; | ||||
|     let root = mount.get_root(); | ||||
| @@ -94,21 +80,13 @@ const ContentTypeDiscoverer = new Lang.Class({ | ||||
| 
 | ||||
|     _init: function(callback) { | ||||
|         this._callback = callback; | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|     }, | ||||
| 
 | ||||
|     guessContentTypes: function(mount) { | ||||
|         let autorunEnabled = !this._settings.get_boolean(SETTING_DISABLE_AUTORUN); | ||||
|         let shouldScan = autorunEnabled && !isMountNonLocal(mount); | ||||
| 
 | ||||
|         if (shouldScan) { | ||||
|             // guess mount's content types using GIO
 | ||||
|             mount.guess_content_type(false, null, | ||||
|                                      Lang.bind(this, | ||||
|                                                this._onContentTypeGuessed)); | ||||
|         } else { | ||||
|             this._emitCallback(mount, []); | ||||
|         } | ||||
|         // guess mount's content types using GIO
 | ||||
|         mount.guess_content_type(false, null, | ||||
|                                  Lang.bind(this, | ||||
|                                            this._onContentTypeGuessed)); | ||||
|     }, | ||||
| 
 | ||||
|     _onContentTypeGuessed: function(mount, res) { | ||||
| @@ -162,68 +140,55 @@ const AutorunManager = new Lang.Class({ | ||||
|     Name: 'AutorunManager', | ||||
| 
 | ||||
|     _init: function() { | ||||
|         this._session = new GnomeSession.SessionManager(); | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
| 
 | ||||
|         this._transDispatcher = new AutorunTransientDispatcher(this); | ||||
|     }, | ||||
|         this._volumeMonitor.connect('mount-added', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onMountAdded)); | ||||
|         this._volumeMonitor.connect('mount-removed', | ||||
|                                     Lang.bind(this, | ||||
|                                               this._onMountRemoved)); | ||||
| 
 | ||||
|     _ensureResidentSource: function() { | ||||
|         if (this._residentSource) | ||||
|             return; | ||||
|         this._transDispatcher = new AutorunTransientDispatcher(); | ||||
|         this._createResidentSource(); | ||||
| 
 | ||||
|         this._residentSource = new AutorunResidentSource(this); | ||||
|         let destroyId = this._residentSource.connect('destroy', Lang.bind(this, function() { | ||||
|             this._residentSource.disconnect(destroyId); | ||||
|             this._residentSource = null; | ||||
|         })); | ||||
|     }, | ||||
| 
 | ||||
|     enable: function() { | ||||
|         this._scanMounts(); | ||||
| 
 | ||||
|         this._mountAddedId = this._volumeMonitor.connect('mount-added', Lang.bind(this, this._onMountAdded)); | ||||
|         this._mountRemovedId = this._volumeMonitor.connect('mount-removed', Lang.bind(this, this._onMountRemoved)); | ||||
|     }, | ||||
| 
 | ||||
|     disable: function() { | ||||
|         if (this._residentSource) | ||||
|             this._residentSource.destroy(); | ||||
|         this._volumeMonitor.disconnect(this._mountAddedId); | ||||
|         this._volumeMonitor.disconnect(this._mountRemovedId); | ||||
|     }, | ||||
| 
 | ||||
|     _processMount: function(mount, hotplug) { | ||||
|         let discoverer = new ContentTypeDiscoverer(Lang.bind(this, function(mount, apps, contentTypes) { | ||||
|             this._ensureResidentSource(); | ||||
|             this._residentSource.addMount(mount, apps); | ||||
| 
 | ||||
|             if (hotplug) | ||||
|                 this._transDispatcher.addMount(mount, apps, contentTypes); | ||||
|         })); | ||||
|         discoverer.guessContentTypes(mount); | ||||
|     }, | ||||
| 
 | ||||
|     _scanMounts: function() { | ||||
|         let mounts = this._volumeMonitor.get_mounts(); | ||||
|         mounts.forEach(Lang.bind(this, function(mount) { | ||||
|             this._processMount(mount, false); | ||||
| 
 | ||||
|         mounts.forEach(Lang.bind(this, function (mount) { | ||||
|             let discoverer = new ContentTypeDiscoverer(Lang.bind (this,  | ||||
|                 function (mount, apps) { | ||||
|                     this._residentSource.addMount(mount, apps); | ||||
|                 })); | ||||
| 
 | ||||
|             discoverer.guessContentTypes(mount); | ||||
|         })); | ||||
|     }, | ||||
| 
 | ||||
|     _createResidentSource: function() { | ||||
|         this._residentSource = new AutorunResidentSource(); | ||||
|         this._residentSource.connect('destroy', | ||||
|                                      Lang.bind(this, | ||||
|                                                this._createResidentSource)); | ||||
|     }, | ||||
| 
 | ||||
|     _onMountAdded: function(monitor, mount) { | ||||
|         // don't do anything if our session is not the currently
 | ||||
|         // active one
 | ||||
|         if (!this._session.SessionIsActive) | ||||
|         if (!Main.automountManager.isSessionActive()) | ||||
|             return; | ||||
| 
 | ||||
|         this._processMount(mount, true); | ||||
|         let discoverer = new ContentTypeDiscoverer(Lang.bind (this, | ||||
|             function (mount, apps, contentTypes) { | ||||
|                 this._transDispatcher.addMount(mount, apps, contentTypes); | ||||
|                 this._residentSource.addMount(mount, apps); | ||||
|             })); | ||||
| 
 | ||||
|         discoverer.guessContentTypes(mount); | ||||
|     }, | ||||
| 
 | ||||
|     _onMountRemoved: function(monitor, mount) { | ||||
|         this._transDispatcher.removeMount(mount); | ||||
|         if (this._residentSource) | ||||
|             this._residentSource.removeMount(mount); | ||||
|         this._residentSource.removeMount(mount); | ||||
|     }, | ||||
| 
 | ||||
|     ejectMount: function(mount) { | ||||
| @@ -259,9 +224,11 @@ const AutorunManager = new Lang.Class({ | ||||
|         try { | ||||
|             mount.unmount_with_operation_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to eject the mount ' + mount.get_name()  | ||||
|                     + ': ' + e.toString()); | ||||
|             // FIXME: we need to ignore G_IO_ERROR_FAILED_HANDLED errors here
 | ||||
|             // but we can't access the error code from JS.
 | ||||
|             // See https://bugzilla.gnome.org/show_bug.cgi?id=591480
 | ||||
|             log('Unable to eject the mount ' + mount.get_name()  | ||||
|                 + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
| 
 | ||||
| @@ -269,9 +236,11 @@ const AutorunManager = new Lang.Class({ | ||||
|         try { | ||||
|             source.eject_with_operation_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to eject the drive ' + source.get_name() | ||||
|                     + ': ' + e.toString()); | ||||
|             // FIXME: we need to ignore G_IO_ERROR_FAILED_HANDLED errors here
 | ||||
|             // but we can't access the error code from JS.
 | ||||
|             // See https://bugzilla.gnome.org/show_bug.cgi?id=591480
 | ||||
|             log('Unable to eject the drive ' + source.get_name()  | ||||
|                 + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
| 
 | ||||
| @@ -279,9 +248,11 @@ const AutorunManager = new Lang.Class({ | ||||
|         try { | ||||
|             drive.stop_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to stop the drive ' + drive.get_name()  | ||||
|                     + ': ' + e.toString()); | ||||
|             // FIXME: we need to ignore G_IO_ERROR_FAILED_HANDLED errors here
 | ||||
|             // but we can't access the error code from JS.
 | ||||
|             // See https://bugzilla.gnome.org/show_bug.cgi?id=591480
 | ||||
|             log('Unable to stop the drive ' + drive.get_name()  | ||||
|                 + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
| }); | ||||
| @@ -290,25 +261,17 @@ const AutorunResidentSource = new Lang.Class({ | ||||
|     Name: 'AutorunResidentSource', | ||||
|     Extends: MessageTray.Source, | ||||
| 
 | ||||
|     _init: function(manager) { | ||||
|         this.parent(_("Removable Devices"), 'media-removable'); | ||||
|     _init: function() { | ||||
|         this.parent(_("Removable Devices")); | ||||
| 
 | ||||
|         this._mounts = []; | ||||
| 
 | ||||
|         this._manager = manager; | ||||
|         this._notification = new AutorunResidentNotification(this._manager, this); | ||||
|     }, | ||||
| 
 | ||||
|     _createPolicy: function() { | ||||
|         return new MessageTray.NotificationPolicy({ showInLockScreen: false }); | ||||
|     }, | ||||
| 
 | ||||
|     buildRightClickMenu: function() { | ||||
|         return null; | ||||
|         this._notification = new AutorunResidentNotification(this); | ||||
|         this._setSummaryIcon(this.createNotificationIcon()); | ||||
|     }, | ||||
| 
 | ||||
|     addMount: function(mount, apps) { | ||||
|         if (!shouldAutorunMount(mount, false)) | ||||
|         if (ignoreAutorunForMount(mount)) | ||||
|             return; | ||||
| 
 | ||||
|         let filtered = this._mounts.filter(function (element) { | ||||
| @@ -347,6 +310,12 @@ const AutorunResidentSource = new Lang.Class({ | ||||
|             Main.messageTray.add(this); | ||||
|             this.pushNotification(this._notification); | ||||
|         } | ||||
|     }, | ||||
| 
 | ||||
|     createNotificationIcon: function() { | ||||
|         return new St.Icon ({ icon_name: 'media-removable', | ||||
|                               icon_type: St.IconType.FULLCOLOR, | ||||
|                               icon_size: this.ICON_SIZE }); | ||||
|     } | ||||
| }); | ||||
| 
 | ||||
| @@ -354,7 +323,7 @@ const AutorunResidentNotification = new Lang.Class({ | ||||
|     Name: 'AutorunResidentNotification', | ||||
|     Extends: MessageTray.Notification, | ||||
| 
 | ||||
|     _init: function(manager, source) { | ||||
|     _init: function(source) { | ||||
|         this.parent(source, source.title, null, { customContent: true }); | ||||
| 
 | ||||
|         // set the notification as resident
 | ||||
| @@ -362,7 +331,6 @@ const AutorunResidentNotification = new Lang.Class({ | ||||
| 
 | ||||
|         this._layout = new St.BoxLayout ({ style_class: 'hotplug-resident-box', | ||||
|                                            vertical: true }); | ||||
|         this._manager = manager; | ||||
| 
 | ||||
|         this.addActor(this._layout, | ||||
|                       { x_expand: true, | ||||
| @@ -410,7 +378,7 @@ const AutorunResidentNotification = new Lang.Class({ | ||||
|                                 expand: true }); | ||||
| 
 | ||||
|         let ejectIcon =  | ||||
|             new St.Icon({ icon_name: 'media-eject-symbolic', | ||||
|             new St.Icon({ icon_name: 'media-eject', | ||||
|                           style_class: 'hotplug-resident-eject-icon' }); | ||||
| 
 | ||||
|         let ejectButton = | ||||
| @@ -425,7 +393,7 @@ const AutorunResidentNotification = new Lang.Class({ | ||||
|         })); | ||||
| 
 | ||||
|         ejectButton.connect('clicked', Lang.bind(this, function() { | ||||
|             this._manager.ejectMount(mount); | ||||
|             Main.autorunManager.ejectMount(mount); | ||||
|         })); | ||||
| 
 | ||||
|         return item; | ||||
| @@ -435,8 +403,7 @@ const AutorunResidentNotification = new Lang.Class({ | ||||
| const AutorunTransientDispatcher = new Lang.Class({ | ||||
|     Name: 'AutorunTransientDispatcher', | ||||
| 
 | ||||
|     _init: function(manager) { | ||||
|         this._manager = manager; | ||||
|     _init: function() { | ||||
|         this._sources = []; | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|     }, | ||||
| @@ -479,7 +446,7 @@ const AutorunTransientDispatcher = new Lang.Class({ | ||||
|             return; | ||||
|       | ||||
|         // add a new source
 | ||||
|         this._sources.push(new AutorunTransientSource(this._manager, mount, apps)); | ||||
|         this._sources.push(new AutorunTransientSource(mount, apps)); | ||||
|     }, | ||||
| 
 | ||||
|     addMount: function(mount, apps, contentTypes) { | ||||
| @@ -488,7 +455,7 @@ const AutorunTransientDispatcher = new Lang.Class({ | ||||
|             return; | ||||
| 
 | ||||
|         // if the mount doesn't want to be autorun, return
 | ||||
|         if (!shouldAutorunMount(mount, true)) | ||||
|         if (ignoreAutorunForMount(mount)) | ||||
|             return; | ||||
| 
 | ||||
|         let setting = this._getAutorunSettingForType(contentTypes[0]); | ||||
| @@ -532,22 +499,23 @@ const AutorunTransientSource = new Lang.Class({ | ||||
|     Name: 'AutorunTransientSource', | ||||
|     Extends: MessageTray.Source, | ||||
| 
 | ||||
|     _init: function(manager, mount, apps) { | ||||
|         this._manager = manager; | ||||
|     _init: function(mount, apps) { | ||||
|         this.parent(mount.get_name()); | ||||
| 
 | ||||
|         this.mount = mount; | ||||
|         this.apps = apps; | ||||
| 
 | ||||
|         this.parent(mount.get_name()); | ||||
| 
 | ||||
|         this._notification = new AutorunTransientNotification(this._manager, this); | ||||
|         this._notification = new AutorunTransientNotification(this); | ||||
|         this._setSummaryIcon(this.createNotificationIcon()); | ||||
| 
 | ||||
|         // add ourselves as a source, and popup the notification
 | ||||
|         Main.messageTray.add(this); | ||||
|         this.notify(this._notification); | ||||
|     }, | ||||
| 
 | ||||
|     getIcon: function() { | ||||
|         return this.mount.get_icon(); | ||||
|     createNotificationIcon: function() { | ||||
|         return new St.Icon({ gicon: this.mount.get_icon(), | ||||
|                              icon_size: this.ICON_SIZE }); | ||||
|     } | ||||
| }); | ||||
| 
 | ||||
| @@ -555,10 +523,9 @@ const AutorunTransientNotification = new Lang.Class({ | ||||
|     Name: 'AutorunTransientNotification', | ||||
|     Extends: MessageTray.Notification, | ||||
| 
 | ||||
|     _init: function(manager, source) { | ||||
|     _init: function(source) { | ||||
|         this.parent(source, source.title, null, { customContent: true }); | ||||
| 
 | ||||
|         this._manager = manager; | ||||
|         this._box = new St.BoxLayout({ style_class: 'hotplug-transient-box', | ||||
|                                        vertical: true }); | ||||
|         this.addActor(this._box); | ||||
| @@ -610,7 +577,7 @@ const AutorunTransientNotification = new Lang.Class({ | ||||
| 
 | ||||
|     _buttonForEject: function() { | ||||
|         let box = new St.BoxLayout(); | ||||
|         let icon = new St.Icon({ icon_name: 'media-eject-symbolic', | ||||
|         let icon = new St.Icon({ icon_name: 'media-eject', | ||||
|                                  style_class: 'hotplug-notification-item-icon' }); | ||||
|         box.add(icon); | ||||
| 
 | ||||
| @@ -627,11 +594,10 @@ const AutorunTransientNotification = new Lang.Class({ | ||||
|                                      style_class: 'hotplug-notification-item' }); | ||||
| 
 | ||||
|         button.connect('clicked', Lang.bind(this, function() { | ||||
|             this._manager.ejectMount(this._mount); | ||||
|             Main.autorunManager.ejectMount(this._mount); | ||||
|         })); | ||||
| 
 | ||||
|         return button; | ||||
|     } | ||||
| }); | ||||
| 
 | ||||
| const Component = AutorunManager; | ||||
| @@ -1,723 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GDesktopEnums = imports.gi.GDesktopEnums; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const GnomeDesktop = imports.gi.GnomeDesktop; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const BACKGROUND_SCHEMA = 'org.gnome.desktop.background'; | ||||
| const DRAW_BACKGROUND_KEY = 'draw-background'; | ||||
| const PRIMARY_COLOR_KEY = 'primary-color'; | ||||
| const SECONDARY_COLOR_KEY = 'secondary-color'; | ||||
| const COLOR_SHADING_TYPE_KEY = 'color-shading-type'; | ||||
| const BACKGROUND_STYLE_KEY = 'picture-options'; | ||||
| const PICTURE_OPACITY_KEY = 'picture-opacity'; | ||||
| const PICTURE_URI_KEY = 'picture-uri'; | ||||
|  | ||||
| const FADE_ANIMATION_TIME = 1.0; | ||||
|  | ||||
| // These parameters affect how often we redraw. | ||||
| // The first is how different (percent crossfaded) the slide show | ||||
| // has to look before redrawing and the second is the minimum | ||||
| // frequency (in seconds) we're willing to wake up | ||||
| const ANIMATION_OPACITY_STEP_INCREMENT = 4.0; | ||||
| const ANIMATION_MIN_WAKEUP_INTERVAL = 1.0; | ||||
|  | ||||
| let _backgroundCache = null; | ||||
|  | ||||
| const BackgroundCache = new Lang.Class({ | ||||
|     Name: 'BackgroundCache', | ||||
|  | ||||
|     _init: function() { | ||||
|        this._patterns = []; | ||||
|        this._images = []; | ||||
|        this._fileMonitors = {}; | ||||
|     }, | ||||
|  | ||||
|     getPatternContent: function(params) { | ||||
|         params = Params.parse(params, { monitorIndex: 0, | ||||
|                                         color: null, | ||||
|                                         secondColor: null, | ||||
|                                         shadingType: null, | ||||
|                                         effects: Meta.BackgroundEffects.NONE }); | ||||
|  | ||||
|         let content = null; | ||||
|         let candidateContent = null; | ||||
|         for (let i = 0; i < this._patterns.length; i++) { | ||||
|             if (!this._patterns[i]) | ||||
|                 continue; | ||||
|  | ||||
|             if (this._patterns[i].get_shading() != params.shadingType) | ||||
|                 continue; | ||||
|  | ||||
|             if (!params.color.equal(this._patterns[i].get_color())) | ||||
|                 continue; | ||||
|  | ||||
|             if (params.shadingType != GDesktopEnums.BackgroundShading.SOLID && | ||||
|                 !params.secondColor.equal(this._patterns[i].get_second_color())) | ||||
|                 continue; | ||||
|  | ||||
|             candidateContent = this._patterns[i]; | ||||
|  | ||||
|             if (params.effects != this._patterns[i].effects) | ||||
|                 continue; | ||||
|  | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         if (candidateContent) { | ||||
|             content = candidateContent.copy(params.monitorIndex, params.effects); | ||||
|         } else { | ||||
|             content = new Meta.Background({ meta_screen: global.screen, | ||||
|                                             monitor: params.monitorIndex, | ||||
|                                             effects: params.effects }); | ||||
|  | ||||
|             if (params.shadingType == GDesktopEnums.BackgroundShading.SOLID) { | ||||
|                 content.load_color(params.color); | ||||
|             } else { | ||||
|                 content.load_gradient(params.shadingType, params.color, params.secondColor); | ||||
|             } | ||||
|  | ||||
|             this._patterns.push(content); | ||||
|         } | ||||
|  | ||||
|         return content; | ||||
|     }, | ||||
|  | ||||
|     _monitorFile: function(filename) { | ||||
|         if (this._fileMonitors[filename]) | ||||
|             return; | ||||
|  | ||||
|         let file = Gio.File.new_for_path(filename); | ||||
|         let monitor = file.monitor(Gio.FileMonitorFlags.NONE, null); | ||||
|  | ||||
|         let signalId = monitor.connect('changed', | ||||
|                                        Lang.bind(this, function() { | ||||
|                                            for (let i = 0; i < this._images.length; i++) { | ||||
|                                                if (this._images[i].get_filename() == filename) | ||||
|                                                    this._images.splice(i, 1); | ||||
|                                            } | ||||
|  | ||||
|                                            monitor.disconnect(signalId); | ||||
|  | ||||
|                                            this.emit('file-changed', filename); | ||||
|                                        })); | ||||
|  | ||||
|         this._fileMonitors[filename] = monitor; | ||||
|     }, | ||||
|  | ||||
|     _removeContent: function(contentList, content) { | ||||
|         let index = contentList.indexOf(content); | ||||
|  | ||||
|         if (index >= 0) | ||||
|             contentList.splice(index, 1); | ||||
|     }, | ||||
|  | ||||
|     removePatternContent: function(content) { | ||||
|         this._removeContent(this._patterns, content); | ||||
|     }, | ||||
|  | ||||
|     removeImageContent: function(content) { | ||||
|         this._removeContent(this._images, content); | ||||
|     }, | ||||
|  | ||||
|     getImageContent: function(params) { | ||||
|         params = Params.parse(params, { monitorIndex: 0, | ||||
|                                         style: null, | ||||
|                                         filename: null, | ||||
|                                         effects: Meta.BackgroundEffects.NONE, | ||||
|                                         cancellable: null, | ||||
|                                         onFinished: null }); | ||||
|  | ||||
|         let content = null; | ||||
|         let candidateContent = null; | ||||
|         for (let i = 0; i < this._images.length; i++) { | ||||
|             if (!this._images[i]) | ||||
|                 continue; | ||||
|  | ||||
|             if (this._images[i].get_style() != params.style) | ||||
|                 continue; | ||||
|  | ||||
|             if (this._images[i].get_filename() != params.filename) | ||||
|                 continue; | ||||
|  | ||||
|             if (params.style == GDesktopEnums.BackgroundStyle.SPANNED && | ||||
|                 this._images[i].monitor_index != this._monitorIndex) | ||||
|                 continue; | ||||
|  | ||||
|             candidateContent = this._images[i]; | ||||
|  | ||||
|             if (params.effects != this._images[i].effects) | ||||
|                 continue; | ||||
|  | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         if (candidateContent) { | ||||
|             content = candidateContent.copy(params.monitorIndex, params.effects); | ||||
|  | ||||
|             if (params.cancellable && params.cancellable.is_cancelled()) | ||||
|                 content = null; | ||||
|  | ||||
|             if (params.onFinished) | ||||
|                 params.onFinished(content); | ||||
|         } else { | ||||
|             content = new Meta.Background({ meta_screen: global.screen, | ||||
|                                             monitor: params.monitorIndex, | ||||
|                                             effects: params.effects }); | ||||
|  | ||||
|             content.load_file_async(params.filename, | ||||
|                                     params.style, | ||||
|                                     params.cancellable, | ||||
|                                     Lang.bind(this, | ||||
|                                               function(object, result) { | ||||
|                                                   try { | ||||
|                                                       content.load_file_finish(result); | ||||
|  | ||||
|                                                       this._monitorFile(params.filename); | ||||
|                                                       this._images.push(content); | ||||
|                                                   } catch(e) { | ||||
|                                                        content = null; | ||||
|                                                   } | ||||
|  | ||||
|                                                   if (params.onFinished) | ||||
|                                                       params.onFinished(content); | ||||
|                                               })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     getAnimation: function(params) { | ||||
|         params = Params.parse(params, { filename: null, | ||||
|                                         onLoaded: null }); | ||||
|  | ||||
|         if (this._animationFilename == params.filename) { | ||||
|             if (params.onLoaded) { | ||||
|                 GLib.idle_add(GLib.PRIORITY_DEFAULT, Lang.bind(this, function() { | ||||
|                     params.onLoaded(this._animation); | ||||
|                 })); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let animation = new Animation({ filename: params.filename }); | ||||
|  | ||||
|         animation.load(Lang.bind(this, function() { | ||||
|                            this._monitorFile(params.filename); | ||||
|                            this._animationFilename = params.filename; | ||||
|                            this._animation = animation; | ||||
|  | ||||
|                            if (params.onLoaded) { | ||||
|                                GLib.idle_add(GLib.PRIORITY_DEFAULT, Lang.bind(this, function() { | ||||
|                                    params.onLoaded(this._animation); | ||||
|                                })); | ||||
|                            } | ||||
|                        })); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(BackgroundCache.prototype); | ||||
|  | ||||
| function getBackgroundCache() { | ||||
|     if (!_backgroundCache) | ||||
|         _backgroundCache = new BackgroundCache(); | ||||
|     return _backgroundCache; | ||||
| } | ||||
|  | ||||
| const Background = new Lang.Class({ | ||||
|     Name: 'Background', | ||||
|  | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { monitorIndex: 0, | ||||
|                                         layoutManager: Main.layoutManager, | ||||
|                                         effects: Meta.BackgroundEffects.NONE }); | ||||
|         this.actor = new Meta.BackgroundGroup(); | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         this._destroySignalId = this.actor.connect('destroy', | ||||
|                                                    Lang.bind(this, this._destroy)); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: BACKGROUND_SCHEMA }); | ||||
|         this._monitorIndex = params.monitorIndex; | ||||
|         this._layoutManager = params.layoutManager; | ||||
|         this._effects = params.effects; | ||||
|         this._fileWatches = {}; | ||||
|         this._pattern = null; | ||||
|         // contains a single image for static backgrounds and | ||||
|         // two images (from and to) for slide shows | ||||
|         this._images = {}; | ||||
|  | ||||
|         this._brightness = 1.0; | ||||
|         this._vignetteSharpness = 0.2; | ||||
|         this._saturation = 1.0; | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|         this.isLoaded = false; | ||||
|  | ||||
|         this._settings.connect('changed', Lang.bind(this, function() { | ||||
|                                    this.emit('changed'); | ||||
|                                })); | ||||
|  | ||||
|         this._load(); | ||||
|     }, | ||||
|  | ||||
|     _destroy: function() { | ||||
|         this._cancellable.cancel(); | ||||
|  | ||||
|         if (this._animationUpdateTimeoutId) { | ||||
|             GLib.source_remove (this._animationUpdateTimeoutId); | ||||
|             this._animationUpdateTimeoutId = 0 | ||||
|         } | ||||
|  | ||||
|         let i; | ||||
|         let keys = Object.keys(this._fileWatches); | ||||
|         for (i = 0; i < keys.length; i++) { | ||||
|             this._cache.disconnect(this._fileWatches[keys[i]]); | ||||
|         } | ||||
|         this._fileWatches = null; | ||||
|  | ||||
|         if (this._pattern) { | ||||
|             if (this._pattern.content) | ||||
|                 this._cache.removePatternContent(this._pattern.content); | ||||
|  | ||||
|             this._pattern.destroy(); | ||||
|             this._pattern = null; | ||||
|         } | ||||
|  | ||||
|         keys = Object.keys(this._images); | ||||
|         for (i = 0; i < keys.length; i++) { | ||||
|             let actor = this._images[keys[i]]; | ||||
|  | ||||
|             if (actor.content) | ||||
|                 this._cache.removeImageContent(actor.content); | ||||
|  | ||||
|             actor.destroy(); | ||||
|             this._images[keys[i]] = null; | ||||
|         } | ||||
|  | ||||
|         this.actor.disconnect(this._destroySignalId); | ||||
|         this._destroySignalId = 0; | ||||
|     }, | ||||
|  | ||||
|     _setLoaded: function() { | ||||
|         if (this.isLoaded) | ||||
|             return; | ||||
|  | ||||
|         this.isLoaded = true; | ||||
|  | ||||
|         GLib.idle_add(GLib.PRIORITY_DEFAULT, Lang.bind(this, function() { | ||||
|             this.emit('loaded'); | ||||
|             return false; | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _loadPattern: function() { | ||||
|         let colorString, res, color, secondColor; | ||||
|  | ||||
|         colorString = this._settings.get_string(PRIMARY_COLOR_KEY); | ||||
|         [res, color] = Clutter.Color.from_string(colorString); | ||||
|         colorString = this._settings.get_string(SECONDARY_COLOR_KEY); | ||||
|         [res, secondColor] = Clutter.Color.from_string(colorString); | ||||
|  | ||||
|         let shadingType = this._settings.get_enum(COLOR_SHADING_TYPE_KEY); | ||||
|  | ||||
|         let content = this._cache.getPatternContent({ monitorIndex: this._monitorIndex, | ||||
|                                                       effects: this._effects, | ||||
|                                                       color: color, | ||||
|                                                       secondColor: secondColor, | ||||
|                                                       shadingType: shadingType }); | ||||
|  | ||||
|         this._pattern = new Meta.BackgroundActor(); | ||||
|         this.actor.add_child(this._pattern); | ||||
|  | ||||
|         this._pattern.content = content; | ||||
|     }, | ||||
|  | ||||
|     _watchCacheFile: function(filename) { | ||||
|         if (this._fileWatches[filename]) | ||||
|             return; | ||||
|  | ||||
|         let signalId = this._cache.connect('file-changed', | ||||
|                                            Lang.bind(this, function(cache, changedFile) { | ||||
|                                                if (changedFile == filename) { | ||||
|                                                    this.emit('changed'); | ||||
|                                                } | ||||
|                                            })); | ||||
|         this._fileWatches[filename] = signalId; | ||||
|     }, | ||||
|  | ||||
|     _addImage: function(content, index, filename) { | ||||
|         content.saturation = this._saturation; | ||||
|         content.brightness = this._brightness; | ||||
|         content.vignette_sharpness = this._vignetteSharpness; | ||||
|  | ||||
|         let actor = new Meta.BackgroundActor(); | ||||
|         actor.content = content; | ||||
|         this.actor.add_child(actor); | ||||
|  | ||||
|         this._images[index] = actor; | ||||
|         this._watchCacheFile(filename); | ||||
|     }, | ||||
|  | ||||
|     _updateImage: function(content, index, filename) { | ||||
|         content.saturation = this._saturation; | ||||
|         content.brightness = this._brightness; | ||||
|         content.vignette_sharpness = this._vignetteSharpness; | ||||
|  | ||||
|         this._images[index].content = content; | ||||
|         this._watchCacheFile(filename); | ||||
|     }, | ||||
|  | ||||
|     _updateAnimationProgress: function() { | ||||
|         if (this._images[1]) { | ||||
|             this._images[1].raise_top(); | ||||
|             this._images[1].opacity = this._animation.transitionProgress * 255; | ||||
|         } | ||||
|  | ||||
|         this._queueAnimationUpdate(); | ||||
|     }, | ||||
|  | ||||
|     _updateAnimation: function() { | ||||
|         this._animationUpdateTimeoutId = 0; | ||||
|  | ||||
|         let files = this._animation.getKeyFrameFiles(this._layoutManager.monitors[this._monitorIndex]); | ||||
|  | ||||
|         if (!files) { | ||||
|             this._setLoaded(); | ||||
|             this._queueAnimationUpdate(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let numPendingImages = files.length; | ||||
|         for (let i = 0; i < files.length; i++) { | ||||
|             if (this._images[i] && this._images[i].content && | ||||
|                 this._images[i].content.get_filename() == files[i]) { | ||||
|  | ||||
|                 numPendingImages--; | ||||
|                 if (numPendingImages == 0) | ||||
|                     this._updateAnimationProgress(); | ||||
|                 continue; | ||||
|             } | ||||
|             this._cache.getImageContent({ monitorIndex: this._monitorIndex, | ||||
|                                           effects: this._effects, | ||||
|                                           style: this._style, | ||||
|                                           filename: files[i], | ||||
|                                           cancellable: this._cancellable, | ||||
|                                           onFinished: Lang.bind(this, function(content) { | ||||
|                                               numPendingImages--; | ||||
|  | ||||
|                                               if (!content) { | ||||
|                                                   this._setLoaded(); | ||||
|                                                   if (numPendingImages == 0) | ||||
|                                                       this._updateAnimationProgress(); | ||||
|                                                   return; | ||||
|                                               } | ||||
|  | ||||
|                                               if (!this._images[i]) { | ||||
|                                                   this._addImage(content, i, files[i]); | ||||
|                                               } else { | ||||
|                                                   this._updateImage(content, i, files[i]); | ||||
|                                               } | ||||
|  | ||||
|                                               if (numPendingImages == 0) { | ||||
|                                                   this._setLoaded(); | ||||
|                                                   this._updateAnimationProgress(); | ||||
|                                               } | ||||
|                                           }) | ||||
|                                         }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _queueAnimationUpdate: function() { | ||||
|         if (this._animationUpdateTimeoutId != 0) | ||||
|             return; | ||||
|  | ||||
|         if (!this._cancellable || this._cancellable.is_cancelled()) | ||||
|             return; | ||||
|  | ||||
|         if (!this._animation.duration) | ||||
|             return; | ||||
|  | ||||
|         let interval = Math.max(ANIMATION_MIN_WAKEUP_INTERVAL * 1000, | ||||
|                                 ANIMATION_OPACITY_STEP_INCREMENT / this._animation.duration); | ||||
|         this._animationUpdateTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, | ||||
|                                                       interval, | ||||
|                                                       Lang.bind(this, function() { | ||||
|                                                                     this._animationUpdateTimeoutId = 0; | ||||
|                                                                     this._updateAnimation(); | ||||
|                                                                     return false; | ||||
|                                                                 })); | ||||
|     }, | ||||
|  | ||||
|     _loadAnimation: function(filename) { | ||||
|         this._cache.getAnimation({ filename: filename, | ||||
|                                              onLoaded: Lang.bind(this, function(animation) { | ||||
|                                                  this._animation = animation; | ||||
|  | ||||
|                                                  if (!this._animation || this._cancellable.is_cancelled()) { | ||||
|                                                      this._setLoaded(); | ||||
|                                                      return; | ||||
|                                                  } | ||||
|  | ||||
|                                                  this._updateAnimation(); | ||||
|                                                  this._watchCacheFile(filename); | ||||
|                                              }) | ||||
|                                            }); | ||||
|     }, | ||||
|  | ||||
|     _loadFile: function(filename) { | ||||
|         this._cache.getImageContent({ monitorIndex: this._monitorIndex, | ||||
|                                       effects: this._effects, | ||||
|                                       style: this._style, | ||||
|                                       filename: filename, | ||||
|                                       cancellable: this._cancellable, | ||||
|                                       onFinished: Lang.bind(this, function(content) { | ||||
|                                           if (!content) { | ||||
|                                               if (!this._cancellable.is_cancelled()) | ||||
|                                                   this._loadAnimation(filename); | ||||
|                                               return; | ||||
|                                           } | ||||
|  | ||||
|                                           this._addImage(content, 0, filename); | ||||
|                                           this._setLoaded(); | ||||
|                                       }) | ||||
|                                     }); | ||||
|  | ||||
|     }, | ||||
|  | ||||
|     _load: function () { | ||||
|         if (!this._settings.get_boolean(DRAW_BACKGROUND_KEY)) { | ||||
|             this._setLoaded(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._cache = getBackgroundCache(); | ||||
|  | ||||
|         this._loadPattern(this._cache); | ||||
|  | ||||
|         this._style = this._settings.get_enum(BACKGROUND_STYLE_KEY); | ||||
|         if (this._style == GDesktopEnums.BackgroundStyle.NONE) { | ||||
|             this._setLoaded(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let uri = this._settings.get_string(PICTURE_URI_KEY); | ||||
|         let filename = Gio.File.new_for_uri(uri).get_path(); | ||||
|  | ||||
|         this._loadFile(filename); | ||||
|     }, | ||||
|  | ||||
|     get saturation() { | ||||
|         return this._saturation; | ||||
|     }, | ||||
|  | ||||
|     set saturation(saturation) { | ||||
|         this._saturation = saturation; | ||||
|  | ||||
|         if (this._pattern && this._pattern.content) | ||||
|             this._pattern.content.saturation = saturation; | ||||
|  | ||||
|         let keys = Object.keys(this._images); | ||||
|         for (let i = 0; i < keys.length; i++) { | ||||
|             let image = this._images[keys[i]]; | ||||
|             if (image && image.content) | ||||
|                 image.content.saturation = saturation; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     get brightness() { | ||||
|         return this._brightness; | ||||
|     }, | ||||
|  | ||||
|     set brightness(factor) { | ||||
|         this._brightness = factor; | ||||
|         if (this._pattern && this._pattern.content) | ||||
|             this._pattern.content.brightness = factor; | ||||
|  | ||||
|         let keys = Object.keys(this._images); | ||||
|         for (let i = 0; i < keys.length; i++) { | ||||
|             let image = this._images[keys[i]]; | ||||
|             if (image && image.content) | ||||
|                 image.content.brightness = factor; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     get vignetteSharpness() { | ||||
|         return this._vignetteSharpness; | ||||
|     }, | ||||
|  | ||||
|     set vignetteSharpness(sharpness) { | ||||
|         this._vignetteSharpness = sharpness; | ||||
|         if (this._pattern && this._pattern.content) | ||||
|             this._pattern.content.vignette_sharpness = sharpness; | ||||
|  | ||||
|         let keys = Object.keys(this._images); | ||||
|         for (let i = 0; i < keys.length; i++) { | ||||
|             let image = this._images[keys[i]]; | ||||
|             if (image && image.content) | ||||
|                 image.content.vignette_sharpness = sharpness; | ||||
|         } | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Background.prototype); | ||||
|  | ||||
| const StillFrame = new Lang.Class({ | ||||
|     Name: 'StillFrame', | ||||
|  | ||||
|     _init: function(monitorIndex) { | ||||
|         this.actor = new Meta.BackgroundActor(); | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         let content = new Meta.Background({ meta_screen: global.screen, | ||||
|                                             monitor: monitorIndex, | ||||
|                                             effects: Meta.BackgroundEffects.NONE }); | ||||
|         content.load_still_frame(); | ||||
|  | ||||
|         this.actor.content = content; | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(StillFrame.prototype); | ||||
|  | ||||
| const Animation = new Lang.Class({ | ||||
|     Name: 'Animation', | ||||
|  | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { filename: null }); | ||||
|  | ||||
|         this.filename = params.filename; | ||||
|         this._keyFrames = []; | ||||
|         this.duration = 0.0; | ||||
|         this.transitionProgress = 0.0; | ||||
|         this.loaded = false; | ||||
|     }, | ||||
|  | ||||
|     load: function(callback) { | ||||
|         let file = Gio.File.new_for_path(this.filename); | ||||
|  | ||||
|         this._show = new GnomeDesktop.BGSlideShow({ filename: this.filename }); | ||||
|  | ||||
|         this._show.load_async(null, | ||||
|                               Lang.bind(this, | ||||
|                                         function(object, result) { | ||||
|                                             this.duration = this._show.get_total_duration(); | ||||
|                                             this.loaded = true; | ||||
|                                             if (callback) | ||||
|                                                 callback(); | ||||
|                                         })); | ||||
|     }, | ||||
|  | ||||
|     getKeyFrameFiles: function(monitor) { | ||||
|         if (!this._show) | ||||
|             return null; | ||||
|  | ||||
|         if (this._show.get_num_slides() < 1) | ||||
|             return null; | ||||
|  | ||||
|         let [progress, duration, isFixed, file1, file2] = this._show.get_current_slide(monitor.width, monitor.height); | ||||
|  | ||||
|         this.transitionProgress = progress; | ||||
|  | ||||
|         let files = []; | ||||
|  | ||||
|         if (file1) | ||||
|             files.push(file1); | ||||
|  | ||||
|         if (file2) | ||||
|             files.push(file2); | ||||
|  | ||||
|         return files; | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(Animation.prototype); | ||||
|  | ||||
| const BackgroundManager = new Lang.Class({ | ||||
|     Name: 'BackgroundManager', | ||||
|  | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { container: null, | ||||
|                                         layoutManager: Main.layoutManager, | ||||
|                                         monitorIndex: null, | ||||
|                                         effects: Meta.BackgroundEffects.NONE }); | ||||
|  | ||||
|         this._container = params.container; | ||||
|         this._layoutManager = params.layoutManager; | ||||
|         this._effects = params.effects; | ||||
|         this._monitorIndex = params.monitorIndex; | ||||
|  | ||||
|         this.background = this._createBackground(); | ||||
|         this._newBackground = null; | ||||
|         this._loadedSignalId = 0; | ||||
|         this._changedSignalId = 0; | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         if (this._loadedSignalId) | ||||
|             this._newBackground.disconnect(this._loadedSignalId); | ||||
|  | ||||
|         if (this._changedSignalId) | ||||
|             this.background.disconnect(this._changedSignalId); | ||||
|  | ||||
|         if (this._newBackground) { | ||||
|             this._newBackground.actor.destroy(); | ||||
|             this._newBackground = null; | ||||
|         } | ||||
|  | ||||
|         if (this.background) { | ||||
|             this.background.actor.destroy(); | ||||
|             this.background = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateBackground: function(background, monitorIndex) { | ||||
|         let newBackground = this._createBackground(monitorIndex); | ||||
|         newBackground.vignetteSharpness = background.vignetteSharpness; | ||||
|         newBackground.brightness = background.brightness; | ||||
|         newBackground.saturation = background.saturation; | ||||
|         newBackground.visible = background.visible; | ||||
|  | ||||
|         let signalId = newBackground.connect('loaded', | ||||
|             Lang.bind(this, function() { | ||||
|                 newBackground.disconnect(signalId); | ||||
|                 Tweener.addTween(background.actor, | ||||
|                                  { opacity: 0, | ||||
|                                    time: FADE_ANIMATION_TIME, | ||||
|                                    transition: 'easeOutQuad', | ||||
|                                    onComplete: Lang.bind(this, function() { | ||||
|                                        this.background = newBackground; | ||||
|                                        this._newBackground = null; | ||||
|                                        background.actor.destroy(); | ||||
|                                        this.emit('changed'); | ||||
|                                    }) | ||||
|                                  }); | ||||
|         })); | ||||
|         this._loadedSignalId = signalId; | ||||
|  | ||||
|         this._newBackground = newBackground; | ||||
|     }, | ||||
|  | ||||
|     _createBackground: function() { | ||||
|         let background = new Background({ monitorIndex: this._monitorIndex, | ||||
|                                           layoutManager: this._layoutManager, | ||||
|                                           effects: this._effects }); | ||||
|         this._container.add_child(background.actor); | ||||
|  | ||||
|         let monitor = this._layoutManager.monitors[this._monitorIndex]; | ||||
|         background.actor.set_position(monitor.x, monitor.y); | ||||
|         background.actor.set_size(monitor.width, monitor.height); | ||||
|         background.actor.lower_bottom(); | ||||
|  | ||||
|         let signalId = background.connect('changed', Lang.bind(this, function() { | ||||
|             background.disconnect(signalId); | ||||
|             this._updateBackground(background, this._monitorIndex); | ||||
|         })); | ||||
|  | ||||
|         this._changedSignalId = signalId; | ||||
|  | ||||
|         return background; | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(BackgroundManager.prototype); | ||||
| @@ -1,58 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const BoxPointer = imports.ui.boxpointer; | ||||
| const Main = imports.ui.main; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
|  | ||||
| const BackgroundMenu = new Lang.Class({ | ||||
|     Name: 'BackgroundMenu', | ||||
|     Extends: PopupMenu.PopupMenu, | ||||
|  | ||||
|     _init: function(source) { | ||||
|         this.parent(source, 0, St.Side.TOP); | ||||
|  | ||||
|         this.addSettingsAction(_("Settings"), 'gnome-control-center.desktop'); | ||||
|         this.addMenuItem(new PopupMenu.PopupSeparatorMenuItem()); | ||||
|         this.addSettingsAction(_("Change Background…"), 'gnome-background-panel.desktop'); | ||||
|  | ||||
|         this.actor.add_style_class_name('background-menu'); | ||||
|  | ||||
|         Main.uiGroup.add_actor(this.actor); | ||||
|         this.actor.hide(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function addBackgroundMenu(actor) { | ||||
|     let cursor = new St.Bin({ opacity: 0 }); | ||||
|     Main.uiGroup.add_actor(cursor); | ||||
|  | ||||
|     actor.reactive = true; | ||||
|     actor._backgroundMenu = new BackgroundMenu(cursor); | ||||
|     actor._backgroundManager = new PopupMenu.PopupMenuManager({ actor: actor }); | ||||
|     actor._backgroundManager.addMenu(actor._backgroundMenu); | ||||
|  | ||||
|     function openMenu() { | ||||
|         let [x, y] = global.get_pointer(); | ||||
|         cursor.set_position(x, y); | ||||
|         actor._backgroundMenu.open(BoxPointer.PopupAnimation.NONE); | ||||
|     } | ||||
|  | ||||
|     let clickAction = new Clutter.ClickAction(); | ||||
|     clickAction.connect('long-press', function(action, actor, state) { | ||||
|         if (state == Clutter.LongPressState.QUERY) | ||||
|             return action.get_button() == 1 && !actor._backgroundMenu.isOpen; | ||||
|         if (state == Clutter.LongPressState.ACTIVATE) | ||||
|             openMenu(); | ||||
|         return true; | ||||
|     }); | ||||
|     clickAction.connect('clicked', function(action) { | ||||
|         if (action.get_button() == 3) | ||||
|             openMenu(); | ||||
|     }); | ||||
|     actor.add_action(clickAction); | ||||
| } | ||||
| @@ -9,13 +9,6 @@ const Shell = imports.gi.Shell; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const PopupAnimation = { | ||||
|     NONE:  0, | ||||
|     SLIDE: 1 << 0, | ||||
|     FADE:  1 << 1, | ||||
|     FULL:  ~0, | ||||
| }; | ||||
|  | ||||
| const POPUP_ANIMATION_TIME = 0.15; | ||||
|  | ||||
| /** | ||||
| @@ -25,10 +18,7 @@ const POPUP_ANIMATION_TIME = 0.15; | ||||
|  * | ||||
|  * An actor which displays a triangle "arrow" pointing to a given | ||||
|  * side.  The .bin property is a container in which content can be | ||||
|  * placed.  The arrow position may be controlled via | ||||
|  * setArrowOrigin(). The arrow side might be temporarily flipped | ||||
|  * depending on the box size and source position to keep the box | ||||
|  * totally inside the monitor if possible. | ||||
|  * placed.  The arrow position may be controlled via setArrowOrigin(). | ||||
|  * | ||||
|  */ | ||||
| const BoxPointer = new Lang.Class({ | ||||
| @@ -36,9 +26,7 @@ const BoxPointer = new Lang.Class({ | ||||
|  | ||||
|     _init: function(arrowSide, binProperties) { | ||||
|         this._arrowSide = arrowSide; | ||||
|         this._userArrowSide = arrowSide; | ||||
|         this._arrowOrigin = 0; | ||||
|         this._arrowActor = null; | ||||
|         this.actor = new St.Bin({ x_fill: true, | ||||
|                                   y_fill: true }); | ||||
|         this._container = new Shell.GenericContainer(); | ||||
| @@ -77,16 +65,11 @@ const BoxPointer = new Lang.Class({ | ||||
|     show: function(animate, onComplete) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|         let animationTime = (animate & PopupAnimation.FULL) ? POPUP_ANIMATION_TIME : 0; | ||||
|  | ||||
|         if (animate & PopupAnimation.FADE) | ||||
|             this.opacity = 0; | ||||
|         else | ||||
|             this.opacity = 255; | ||||
|  | ||||
|         this.opacity = 0; | ||||
|         this.actor.show(); | ||||
|  | ||||
|         if (animate & PopupAnimation.SLIDE) { | ||||
|         if (animate) { | ||||
|             switch (this._arrowSide) { | ||||
|                 case St.Side.TOP: | ||||
|                     this.yOffset = -rise; | ||||
| @@ -112,7 +95,7 @@ const BoxPointer = new Lang.Class({ | ||||
|                                      if (onComplete) | ||||
|                                          onComplete(); | ||||
|                                  }), | ||||
|                                  time: animationTime }); | ||||
|                                  time: POPUP_ANIMATION_TIME }); | ||||
|     }, | ||||
|  | ||||
|     hide: function(animate, onComplete) { | ||||
| @@ -120,10 +103,8 @@ const BoxPointer = new Lang.Class({ | ||||
|         let yOffset = 0; | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|         let fade = (animate & PopupAnimation.FADE); | ||||
|         let animationTime = (animate & PopupAnimation.FULL) ? POPUP_ANIMATION_TIME : 0; | ||||
|  | ||||
|         if (animate & PopupAnimation.SLIDE) { | ||||
|         if (animate) { | ||||
|             switch (this._arrowSide) { | ||||
|                 case St.Side.TOP: | ||||
|                     yOffset = rise; | ||||
| @@ -142,15 +123,13 @@ const BoxPointer = new Lang.Class({ | ||||
|  | ||||
|         this._muteInput(); | ||||
|  | ||||
|         Tweener.removeTweens(this); | ||||
|         Tweener.addTween(this, { opacity: fade ? 0 : 255, | ||||
|         Tweener.addTween(this, { opacity: 0, | ||||
|                                  xOffset: xOffset, | ||||
|                                  yOffset: yOffset, | ||||
|                                  transition: 'linear', | ||||
|                                  time: animationTime, | ||||
|                                  time: POPUP_ANIMATION_TIME, | ||||
|                                  onComplete: Lang.bind(this, function () { | ||||
|                                      this.actor.hide(); | ||||
|                                      this.opacity = 0; | ||||
|                                      this.xOffset = 0; | ||||
|                                      this.yOffset = 0; | ||||
|                                      if (onComplete) | ||||
| @@ -220,28 +199,13 @@ const BoxPointer = new Lang.Class({ | ||||
|         } | ||||
|         this.bin.allocate(childBox, flags); | ||||
|  | ||||
|         if (this._sourceActor && this._sourceActor.mapped) { | ||||
|             this._reposition(); | ||||
|             this._updateFlip(); | ||||
|         } | ||||
|         if (this._sourceActor && this._sourceActor.mapped) | ||||
|             this._reposition(this._sourceActor, this._arrowAlignment); | ||||
|     }, | ||||
|  | ||||
|     _drawBorder: function(area) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|  | ||||
|         if (this._arrowActor) { | ||||
|             let [sourceX, sourceY] = this._arrowActor.get_transformed_position(); | ||||
|             let [sourceWidth, sourceHeight] = this._arrowActor.get_transformed_size(); | ||||
|             let [absX, absY] = this.actor.get_transformed_position(); | ||||
|  | ||||
|             if (this._arrowSide == St.Side.TOP || | ||||
|                 this._arrowSide == St.Side.BOTTOM) { | ||||
|                 this._arrowOrigin = sourceX - absX + sourceWidth / 2; | ||||
|             } else { | ||||
|                 this._arrowOrigin = sourceY - absY + sourceHeight / 2; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let borderWidth = themeNode.get_length('-arrow-border-width'); | ||||
|         let base = themeNode.get_length('-arrow-base'); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
| @@ -250,6 +214,7 @@ const BoxPointer = new Lang.Class({ | ||||
|         let halfBorder = borderWidth / 2; | ||||
|         let halfBase = Math.floor(base/2); | ||||
|  | ||||
|         let borderColor = themeNode.get_color('-arrow-border-color'); | ||||
|         let backgroundColor = themeNode.get_color('-arrow-background-color'); | ||||
|  | ||||
|         let [width, height] = area.get_surface_size(); | ||||
| @@ -260,6 +225,7 @@ const BoxPointer = new Lang.Class({ | ||||
|             boxWidth -= rise; | ||||
|         } | ||||
|         let cr = area.get_context(); | ||||
|         Clutter.cairo_set_source_color(cr, borderColor); | ||||
|  | ||||
|         // Translate so that box goes from 0,0 to boxWidth,boxHeight, | ||||
|         // with the arrow poking out of that | ||||
| @@ -272,51 +238,14 @@ const BoxPointer = new Lang.Class({ | ||||
|         let [x1, y1] = [halfBorder, halfBorder]; | ||||
|         let [x2, y2] = [boxWidth - halfBorder, boxHeight - halfBorder]; | ||||
|  | ||||
|         let skipTopLeft = false; | ||||
|         let skipTopRight = false; | ||||
|         let skipBottomLeft = false; | ||||
|         let skipBottomRight = false; | ||||
|  | ||||
|         switch (this._arrowSide) { | ||||
|         case St.Side.TOP: | ||||
|             if (this._arrowOrigin == x1) | ||||
|                 skipTopLeft = true; | ||||
|             else if (this._arrowOrigin == x2) | ||||
|                 skipTopRight = true; | ||||
|             break; | ||||
|  | ||||
|         case St.Side.RIGHT: | ||||
|             if (this._arrowOrigin == y1) | ||||
|                 skipTopRight = true; | ||||
|             else if (this._arrowOrigin == y2) | ||||
|                 skipBottomRight = true; | ||||
|             break; | ||||
|  | ||||
|         case St.Side.BOTTOM: | ||||
|             if (this._arrowOrigin == x1) | ||||
|                 skipBottomLeft = true; | ||||
|             else if (this._arrowOrigin == x2) | ||||
|                 skipBottomRight = true; | ||||
|             break; | ||||
|  | ||||
|         case St.Side.LEFT: | ||||
|             if (this._arrowOrigin == y1) | ||||
|                 skipTopLeft = true; | ||||
|             else if (this._arrowOrigin == y2) | ||||
|                 skipBottomLeft = true; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         cr.moveTo(x1 + borderRadius, y1); | ||||
|         if (this._arrowSide == St.Side.TOP) { | ||||
|             if (skipTopLeft) { | ||||
|                 cr.moveTo(x1, y2 - borderRadius); | ||||
|                 cr.lineTo(x1, y1 - rise); | ||||
|                 cr.lineTo(x1 + halfBase, y1); | ||||
|             } else if (skipTopRight) { | ||||
|                 cr.lineTo(x2 - halfBase, y1); | ||||
|                 cr.lineTo(x2, y1 - rise); | ||||
|                 cr.lineTo(x2, y1 + borderRadius); | ||||
|             if (this._arrowOrigin < (x1 + (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(this._arrowOrigin, y1 - rise); | ||||
|                 cr.lineTo(Math.max(x1 + borderRadius, this._arrowOrigin) + halfBase, y1); | ||||
|             } else if (this._arrowOrigin > (x2 - (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(Math.min(x2 - borderRadius, this._arrowOrigin) - halfBase, y1); | ||||
|                 cr.lineTo(this._arrowOrigin, y1 - rise); | ||||
|             } else { | ||||
|                 cr.lineTo(this._arrowOrigin - halfBase, y1); | ||||
|                 cr.lineTo(this._arrowOrigin, y1 - rise); | ||||
| @@ -324,20 +253,19 @@ const BoxPointer = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!skipTopRight) { | ||||
|             cr.lineTo(x2 - borderRadius, y1); | ||||
|             cr.arc(x2 - borderRadius, y1 + borderRadius, borderRadius, | ||||
|                    3*Math.PI/2, Math.PI*2); | ||||
|         } | ||||
|         cr.lineTo(x2 - borderRadius, y1); | ||||
|  | ||||
|         // top-right corner | ||||
|         cr.arc(x2 - borderRadius, y1 + borderRadius, borderRadius, | ||||
|                3*Math.PI/2, Math.PI*2); | ||||
|  | ||||
|         if (this._arrowSide == St.Side.RIGHT) { | ||||
|             if (skipTopRight) { | ||||
|                 cr.lineTo(x2 + rise, y1); | ||||
|                 cr.lineTo(x2 + rise, y1 + halfBase); | ||||
|             } else if (skipBottomRight) { | ||||
|                 cr.lineTo(x2, y2 - halfBase); | ||||
|                 cr.lineTo(x2 + rise, y2); | ||||
|                 cr.lineTo(x2 - borderRadius, y2); | ||||
|             if (this._arrowOrigin < (y1 + (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(x2 + rise, this._arrowOrigin); | ||||
|                 cr.lineTo(x2, Math.max(y1 + borderRadius, this._arrowOrigin) + halfBase); | ||||
|             } else if (this._arrowOrigin > (y2 - (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(x2, Math.min(y2 - borderRadius, this._arrowOrigin) - halfBase); | ||||
|                 cr.lineTo(x2 + rise, this._arrowOrigin); | ||||
|             } else { | ||||
|                 cr.lineTo(x2, this._arrowOrigin - halfBase); | ||||
|                 cr.lineTo(x2 + rise, this._arrowOrigin); | ||||
| @@ -345,20 +273,19 @@ const BoxPointer = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!skipBottomRight) { | ||||
|             cr.lineTo(x2, y2 - borderRadius); | ||||
|             cr.arc(x2 - borderRadius, y2 - borderRadius, borderRadius, | ||||
|                    0, Math.PI/2); | ||||
|         } | ||||
|         cr.lineTo(x2, y2 - borderRadius); | ||||
|  | ||||
|         // bottom-right corner | ||||
|         cr.arc(x2 - borderRadius, y2 - borderRadius, borderRadius, | ||||
|                0, Math.PI/2); | ||||
|  | ||||
|         if (this._arrowSide == St.Side.BOTTOM) { | ||||
|             if (skipBottomLeft) { | ||||
|                 cr.lineTo(x1 + halfBase, y2); | ||||
|                 cr.lineTo(x1, y2 + rise); | ||||
|                 cr.lineTo(x1, y2 - borderRadius); | ||||
|             } else if (skipBottomRight) { | ||||
|                 cr.lineTo(x2, y2 + rise); | ||||
|                 cr.lineTo(x2 - halfBase, y2); | ||||
|             if (this._arrowOrigin < (x1 + (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(Math.max(x1 + borderRadius, this._arrowOrigin) + halfBase, y2); | ||||
|                 cr.lineTo(this._arrowOrigin, y2 + rise); | ||||
|             } else if (this._arrowOrigin > (x2 - (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(this._arrowOrigin, y2 + rise); | ||||
|                 cr.lineTo(Math.min(x2 - borderRadius, this._arrowOrigin) - halfBase, y2); | ||||
|             } else { | ||||
|                 cr.lineTo(this._arrowOrigin + halfBase, y2); | ||||
|                 cr.lineTo(this._arrowOrigin, y2 + rise); | ||||
| @@ -366,20 +293,19 @@ const BoxPointer = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!skipBottomLeft) { | ||||
|             cr.lineTo(x1 + borderRadius, y2); | ||||
|             cr.arc(x1 + borderRadius, y2 - borderRadius, borderRadius, | ||||
|                    Math.PI/2, Math.PI); | ||||
|         } | ||||
|         cr.lineTo(x1 + borderRadius, y2); | ||||
|  | ||||
|         // bottom-left corner | ||||
|         cr.arc(x1 + borderRadius, y2 - borderRadius, borderRadius, | ||||
|                Math.PI/2, Math.PI); | ||||
|  | ||||
|         if (this._arrowSide == St.Side.LEFT) { | ||||
|             if (skipTopLeft) { | ||||
|                 cr.lineTo(x1, y1 + halfBase); | ||||
|                 cr.lineTo(x1 - rise, y1); | ||||
|                 cr.lineTo(x1 + borderRadius, y1); | ||||
|             } else if (skipBottomLeft) { | ||||
|                 cr.lineTo(x1 - rise, y2) | ||||
|                 cr.lineTo(x1 - rise, y2 - halfBase); | ||||
|             if (this._arrowOrigin < (y1 + (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(x1, Math.max(y1 + borderRadius, this._arrowOrigin) + halfBase); | ||||
|                 cr.lineTo(x1 - rise, this._arrowOrigin); | ||||
|             } else if (this._arrowOrigin > (y2 - (borderRadius + halfBase))) { | ||||
|                 cr.lineTo(x1 - rise, this._arrowOrigin); | ||||
|                 cr.lineTo(x1, Math.min(y2 - borderRadius, this._arrowOrigin) - halfBase); | ||||
|             } else { | ||||
|                 cr.lineTo(x1, this._arrowOrigin + halfBase); | ||||
|                 cr.lineTo(x1 - rise, this._arrowOrigin); | ||||
| @@ -387,23 +313,17 @@ const BoxPointer = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!skipTopLeft) { | ||||
|             cr.lineTo(x1, y1 + borderRadius); | ||||
|             cr.arc(x1 + borderRadius, y1 + borderRadius, borderRadius, | ||||
|                    Math.PI, 3*Math.PI/2); | ||||
|         } | ||||
|         cr.lineTo(x1, y1 + borderRadius); | ||||
|  | ||||
|         // top-left corner | ||||
|         cr.arc(x1 + borderRadius, y1 + borderRadius, borderRadius, | ||||
|                Math.PI, 3*Math.PI/2); | ||||
|  | ||||
|         Clutter.cairo_set_source_color(cr, backgroundColor); | ||||
|         cr.fillPreserve(); | ||||
|  | ||||
|         if (borderWidth > 0) { | ||||
|             let borderColor = themeNode.get_color('-arrow-border-color'); | ||||
|             Clutter.cairo_set_source_color(cr, borderColor); | ||||
|             cr.setLineWidth(borderWidth); | ||||
|             cr.stroke(); | ||||
|         } | ||||
|  | ||||
|         cr.$dispose(); | ||||
|         Clutter.cairo_set_source_color(cr, borderColor); | ||||
|         cr.setLineWidth(borderWidth); | ||||
|         cr.stroke(); | ||||
|     }, | ||||
|  | ||||
|     setPosition: function(sourceActor, alignment) { | ||||
| @@ -414,8 +334,7 @@ const BoxPointer = new Lang.Class({ | ||||
|         this._sourceActor = sourceActor; | ||||
|         this._arrowAlignment = alignment; | ||||
|  | ||||
|         this._reposition(); | ||||
|         this._updateFlip(); | ||||
|         this._reposition(sourceActor, alignment); | ||||
|     }, | ||||
|  | ||||
|     setSourceAlignment: function(alignment) { | ||||
| @@ -424,13 +343,14 @@ const BoxPointer = new Lang.Class({ | ||||
|         if (!this._sourceActor) | ||||
|             return; | ||||
|  | ||||
|         this.setPosition(this._sourceActor, this._arrowAlignment); | ||||
|         // We need to show it now to force an allocation, | ||||
|         // so that we can query the correct size. | ||||
|         this.actor.show(); | ||||
|  | ||||
|         this._reposition(this._sourceActor, this._arrowAlignment); | ||||
|     }, | ||||
|  | ||||
|     _reposition: function() { | ||||
|         let sourceActor = this._sourceActor; | ||||
|         let alignment = this._arrowAlignment; | ||||
|  | ||||
|     _reposition: function(sourceActor, alignment) { | ||||
|         // Position correctly relative to the sourceActor | ||||
|         let sourceNode = sourceActor.get_theme_node(); | ||||
|         let sourceContentBox = sourceNode.get_content_box(sourceActor.get_allocation_box()); | ||||
| @@ -448,9 +368,10 @@ const BoxPointer = new Lang.Class({ | ||||
|         let arrowBase = themeNode.get_length('-arrow-base'); | ||||
|         let borderRadius = themeNode.get_length('-arrow-border-radius'); | ||||
|         let margin = (4 * borderRadius + borderWidth + arrowBase); | ||||
|         let halfMargin = margin / 2; | ||||
|  | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let gap = themeNode.get_length('-boxpointer-gap'); | ||||
|         let padding = themeNode.get_length('-arrow-rise'); | ||||
|  | ||||
|         let resX, resY; | ||||
|  | ||||
| @@ -469,66 +390,29 @@ const BoxPointer = new Lang.Class({ | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         // Now align and position the pointing axis, making sure it fits on | ||||
|         // screen. If the arrowOrigin is so close to the edge that the arrow | ||||
|         // will not be isosceles, we try to compensate as follows: | ||||
|         //   - We skip the rounded corner and settle for a right angled arrow | ||||
|         //     as shown below. See _drawBorder for further details. | ||||
|         //     |\_____ | ||||
|         //     | | ||||
|         //     | | ||||
|         //   - If the arrow was going to be acute angled, we move the position | ||||
|         //     of the box to maintain the arrow's accuracy. | ||||
|  | ||||
|         let arrowOrigin; | ||||
|         let halfBase = Math.floor(arrowBase/2); | ||||
|         let halfBorder = borderWidth / 2; | ||||
|         let halfMargin = margin / 2; | ||||
|         let [x1, y1] = [halfBorder, halfBorder]; | ||||
|         let [x2, y2] = [natWidth - halfBorder, natHeight - halfBorder]; | ||||
|  | ||||
|         // Now align and position the pointing axis, making sure | ||||
|         // it fits on screen | ||||
|         switch (this._arrowSide) { | ||||
|         case St.Side.TOP: | ||||
|         case St.Side.BOTTOM: | ||||
|             resX = sourceCenterX - (halfMargin + (natWidth - margin) * alignment); | ||||
|  | ||||
|             resX = Math.max(resX, monitor.x + padding); | ||||
|             resX = Math.min(resX, monitor.x + monitor.width - (padding + natWidth)); | ||||
|  | ||||
|             arrowOrigin = sourceCenterX - resX; | ||||
|             if (arrowOrigin <= (x1 + (borderRadius + halfBase))) { | ||||
|                 if (arrowOrigin > x1) | ||||
|                     resX += (arrowOrigin - x1); | ||||
|                 arrowOrigin = x1; | ||||
|             } else if (arrowOrigin >= (x2 - (borderRadius + halfBase))) { | ||||
|                 if (arrowOrigin < x2) | ||||
|                     resX -= (x2 - arrowOrigin); | ||||
|                 arrowOrigin = x2; | ||||
|             } | ||||
|             resX = Math.max(resX, monitor.x + 10); | ||||
|             resX = Math.min(resX, monitor.x + monitor.width - (10 + natWidth)); | ||||
|             this.setArrowOrigin(sourceCenterX - resX); | ||||
|             break; | ||||
|  | ||||
|         case St.Side.LEFT: | ||||
|         case St.Side.RIGHT: | ||||
|             resY = sourceCenterY - (halfMargin + (natHeight - margin) * alignment); | ||||
|  | ||||
|             resY = Math.max(resY, monitor.y + padding); | ||||
|             resY = Math.min(resY, monitor.y + monitor.height - (padding + natHeight)); | ||||
|             resY = Math.max(resY, monitor.y + 10); | ||||
|             resY = Math.min(resY, monitor.y + monitor.height - (10 + natHeight)); | ||||
|  | ||||
|             arrowOrigin = sourceCenterY - resY; | ||||
|             if (arrowOrigin <= (y1 + (borderRadius + halfBase))) { | ||||
|                 if (arrowOrigin > y1) | ||||
|                     resY += (arrowOrigin - y1); | ||||
|                 arrowOrigin = y1; | ||||
|             } else if (arrowOrigin >= (y2 - (borderRadius + halfBase))) { | ||||
|                 if (arrowOrigin < y2) | ||||
|                     resX -= (y2 - arrowOrigin); | ||||
|                 arrowOrigin = y2; | ||||
|             } | ||||
|             this.setArrowOrigin(sourceCenterY - resY); | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         this.setArrowOrigin(arrowOrigin); | ||||
|  | ||||
|         let parent = this.actor.get_parent(); | ||||
|         let success, x, y; | ||||
|         while (!success) { | ||||
| @@ -551,20 +435,10 @@ const BoxPointer = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // @actor: an actor relative to which the arrow is positioned. | ||||
|     // Differently from setPosition, this will not move the boxpointer itself, | ||||
|     // on the arrow | ||||
|     setArrowActor: function(actor) { | ||||
|         if (this._arrowActor != actor) { | ||||
|             this._arrowActor = actor; | ||||
|             this._border.queue_repaint(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _shiftActor : function() { | ||||
|         // Since the position of the BoxPointer depends on the allocated size | ||||
|         // of the BoxPointer and the position of the source actor, trying | ||||
|         // to position the BoxPointer via the x/y properties will result in | ||||
|         // to position the BoxPoiner via the x/y properties will result in | ||||
|         // allocation loops and warnings. Instead we do the positioning via | ||||
|         // the anchor point, which is independent of allocation, and leave | ||||
|         // x == y == 0. | ||||
| @@ -572,49 +446,6 @@ const BoxPointer = new Lang.Class({ | ||||
|                                     -(this._yPosition + this._yOffset)); | ||||
|     }, | ||||
|  | ||||
|     _calculateArrowSide: function(arrowSide) { | ||||
|         let sourceAllocation = Shell.util_get_transformed_allocation(this._sourceActor); | ||||
|         let [minWidth, minHeight, boxWidth, boxHeight] = this._container.get_preferred_size(); | ||||
|         let monitor = Main.layoutManager.findMonitorForActor(this.actor); | ||||
|  | ||||
|         switch (arrowSide) { | ||||
|         case St.Side.TOP: | ||||
|             if (sourceAllocation.y2 + boxHeight > monitor.y + monitor.height && | ||||
|                 boxHeight < sourceAllocation.y1 - monitor.y) | ||||
|                 return St.Side.BOTTOM; | ||||
|             break; | ||||
|         case St.Side.BOTTOM: | ||||
|             if (sourceAllocation.y1 - boxHeight < monitor.y && | ||||
|                 boxHeight < monitor.y + monitor.height - sourceAllocation.y2) | ||||
|                 return St.Side.TOP; | ||||
|             break; | ||||
|         case St.Side.LEFT: | ||||
|             if (sourceAllocation.y2 + boxWidth > monitor.x + monitor.width && | ||||
|                 boxWidth < sourceAllocation.x1 - monitor.x) | ||||
|                 return St.Side.RIGHT; | ||||
|             break; | ||||
|         case St.Side.RIGHT: | ||||
|             if (sourceAllocation.y1 - boxWidth < monitor.x && | ||||
|                 boxWidth < monitor.x + monitor.width - sourceAllocation.x2) | ||||
|                 return St.Side.LEFT; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         return arrowSide; | ||||
|     }, | ||||
|  | ||||
|     _updateFlip: function() { | ||||
|         let arrowSide = this._calculateArrowSide(this._userArrowSide); | ||||
|         if (this._arrowSide != arrowSide) { | ||||
|             this._arrowSide = arrowSide; | ||||
|             this._reposition(); | ||||
|             Meta.later_add(Meta.LaterType.BEFORE_REDRAW, Lang.bind(this, function() { | ||||
|                 this._container.queue_relayout(); | ||||
|                 return false; | ||||
|             })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     set xOffset(offset) { | ||||
|         this._xOffset = offset; | ||||
|         this._shiftActor(); | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| const St = imports.gi.St; | ||||
| const Signals = imports.signals; | ||||
| @@ -12,6 +11,7 @@ const Mainloop = imports.mainloop; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const MSECS_IN_DAY = 24 * 60 * 60 * 1000; | ||||
| const WEEKDATE_HEADER_WIDTH_DIGITS = 3; | ||||
| const SHOW_WEEKDATE_KEY = 'show-weekdate'; | ||||
|  | ||||
| // in org.gnome.desktop.interface | ||||
| @@ -63,18 +63,15 @@ function _formatEventTime(event, clockFormat) { | ||||
|     } else { | ||||
|         switch (clockFormat) { | ||||
|         case '24h': | ||||
|             /* Translators: Shown in calendar event list, if 24h format, | ||||
|                \u2236 is a ratio character, similar to : */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%H\u2236%M")); | ||||
|             /* Translators: Shown in calendar event list, if 24h format */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%H:%M")); | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
|             /* explicit fall-through */ | ||||
|         case '12h': | ||||
|             /* Transators: Shown in calendar event list, if 12h format, | ||||
|                \u2236 is a ratio character, similar to : and \u2009 is | ||||
|                a thin space */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%l\u2236%M\u2009%p")); | ||||
|             /* Transators: Shown in calendar event list, if 12h format */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%l:%M %p")); | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| @@ -98,6 +95,15 @@ function _getCalendarWeekForDate(date) { | ||||
|     return weekNumber; | ||||
| } | ||||
|  | ||||
| function _getDigitWidth(actor){ | ||||
|     let context = actor.get_pango_context(); | ||||
|     let themeNode = actor.get_theme_node(); | ||||
|     let font = themeNode.get_font(); | ||||
|     let metrics = context.get_metrics(font, context.get_language()); | ||||
|     let width = metrics.get_approximate_digit_width(); | ||||
|     return width; | ||||
| } | ||||
|  | ||||
| function _getCalendarDayAbbreviation(dayNumber) { | ||||
|     let abbreviations = [ | ||||
|         /* Translators: Calendar grid abbreviation for Sunday. | ||||
| @@ -197,12 +203,16 @@ const CalendarServerIface = <interface name="org.gnome.Shell.CalendarServer"> | ||||
| const CalendarServerInfo  = Gio.DBusInterfaceInfo.new_for_xml(CalendarServerIface); | ||||
|  | ||||
| function CalendarServer() { | ||||
|     return new Gio.DBusProxy({ g_connection: Gio.DBus.session, | ||||
|                                g_interface_name: CalendarServerInfo.name, | ||||
|                                g_interface_info: CalendarServerInfo, | ||||
|                                g_name: 'org.gnome.Shell.CalendarServer', | ||||
|                                g_object_path: '/org/gnome/Shell/CalendarServer', | ||||
|                                g_flags: Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES }); | ||||
|     var self = new Gio.DBusProxy({ g_connection: Gio.DBus.session, | ||||
| 				   g_interface_name: CalendarServerInfo.name, | ||||
| 				   g_interface_info: CalendarServerInfo, | ||||
| 				   g_name: 'org.gnome.Shell.CalendarServer', | ||||
| 				   g_object_path: '/org/gnome/Shell/CalendarServer', | ||||
|                                    g_flags: (Gio.DBusProxyFlags.DO_NOT_AUTO_START | | ||||
|                                              Gio.DBusProxyFlags.DO_NOT_LOAD_PROPERTIES) }); | ||||
|  | ||||
|     self.init(null); | ||||
|     return self; | ||||
| } | ||||
|  | ||||
| function _datesEqual(a, b) { | ||||
| @@ -230,27 +240,14 @@ const DBusEventSource = new Lang.Class({ | ||||
|     _init: function() { | ||||
|         this._resetCache(); | ||||
|  | ||||
|         this._initialized = false; | ||||
|         this._dbusProxy = new CalendarServer(); | ||||
|         this._dbusProxy.init_async(GLib.PRIORITY_DEFAULT, null, Lang.bind(this, function(object, result) { | ||||
|             try { | ||||
|                 this._dbusProxy.init_finish(result); | ||||
|             } catch(e) { | ||||
|                 log('Error loading calendars: ' + e.message); | ||||
|                 return; | ||||
|             } | ||||
|         this._dbusProxy.connectSignal('Changed', Lang.bind(this, this._onChanged)); | ||||
|  | ||||
|             this._dbusProxy.connectSignal('Changed', Lang.bind(this, this._onChanged)); | ||||
|  | ||||
|             this._dbusProxy.connect('notify::g-name-owner', Lang.bind(this, function() { | ||||
|                 if (this._dbusProxy.g_name_owner) | ||||
|                     this._onNameAppeared(); | ||||
|                 else | ||||
|                     this._onNameVanished(); | ||||
|             })); | ||||
|  | ||||
|             this._initialized = true; | ||||
|             this._onNameAppeared(); | ||||
|         this._dbusProxy.connect('notify::g-name-owner', Lang.bind(this, function() { | ||||
|             if (this._dbusProxy.g_name_owner) | ||||
|                 this._onNameAppeared(); | ||||
|             else | ||||
|                 this._onNameVanished(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
| @@ -274,9 +271,8 @@ const DBusEventSource = new Lang.Class({ | ||||
|         this._loadEvents(false); | ||||
|     }, | ||||
|  | ||||
|     _onEventsReceived: function(results, error) { | ||||
|     _onEventsReceived: function([appointments]) { | ||||
|         let newEvents = []; | ||||
|         let appointments = results ? results[0] : null; | ||||
|         if (appointments != null) { | ||||
|             for (let n = 0; n < appointments.length; n++) { | ||||
|                 let a = appointments[n]; | ||||
| @@ -297,10 +293,6 @@ const DBusEventSource = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _loadEvents: function(forceReload) { | ||||
|         // Ignore while loading | ||||
|         if (!this._initialized) | ||||
|             return; | ||||
|  | ||||
|         if (this._curRequestBegin && this._curRequestEnd){ | ||||
|             let callFlags = Gio.DBusCallFlags.NO_AUTO_START; | ||||
|             if (forceReload) | ||||
| @@ -348,11 +340,25 @@ const DBusEventSource = new Lang.Class({ | ||||
| }); | ||||
| Signals.addSignalMethods(DBusEventSource.prototype); | ||||
|  | ||||
| // Calendar: | ||||
| // @eventSource: is an object implementing the EventSource API, e.g. the | ||||
| // requestRange(), getEvents(), hasEvents() methods and the ::changed signal. | ||||
| const Calendar = new Lang.Class({ | ||||
|     Name: 'Calendar', | ||||
|  | ||||
|     _init: function() { | ||||
|     _init: function(eventSource) { | ||||
|         if (eventSource) { | ||||
|             this._eventSource = eventSource; | ||||
|  | ||||
|             this._eventSource.connect('changed', Lang.bind(this, | ||||
|                                                            function() { | ||||
|                                                                this._update(false); | ||||
|                                                            })); | ||||
|         } | ||||
|  | ||||
|         this._weekStart = Shell.util_get_week_start(); | ||||
|         this._weekdate = NaN; | ||||
|         this._digitWidth = NaN; | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.shell.calendar' }); | ||||
|  | ||||
|         this._settings.connect('changed::' + SHOW_WEEKDATE_KEY, Lang.bind(this, this._onSettingsChange)); | ||||
| @@ -386,24 +392,6 @@ const Calendar = new Lang.Class({ | ||||
|         this._buildHeader (); | ||||
|     }, | ||||
|  | ||||
|     // @eventSource: is an object implementing the EventSource API, e.g. the | ||||
|     // requestRange(), getEvents(), hasEvents() methods and the ::changed signal. | ||||
|     setEventSource: function(eventSource) { | ||||
|         if (this._eventSource) { | ||||
|             this._eventSource.disconnect(this._eventSourceChangedId); | ||||
|             this._eventSource = null; | ||||
|         } | ||||
|  | ||||
|         this._eventSource = eventSource; | ||||
|  | ||||
|         if (this._eventSource) { | ||||
|             this._eventSourceChangedId = this._eventSource.connect('changed', Lang.bind(this, function() { | ||||
|                 this._update(false); | ||||
|             })); | ||||
|             this._update(true); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Sets the calendar to show a specific date | ||||
|     setDate: function(date, forceReload) { | ||||
|         if (!_sameDay(date, this._selectedDate)) { | ||||
| @@ -425,6 +413,8 @@ const Calendar = new Lang.Class({ | ||||
|         this.actor.add(this._topBox, | ||||
|                        { row: 0, col: 0, col_span: offsetCols + 7 }); | ||||
|  | ||||
|         this.actor.connect('style-changed', Lang.bind(this, this._onStyleChange)); | ||||
|  | ||||
|         let back = new St.Button({ style_class: 'calendar-change-month-back' }); | ||||
|         this._topBox.add(back); | ||||
|         back.connect('clicked', Lang.bind(this, this._onPrevMonthButtonClicked)); | ||||
| @@ -458,7 +448,19 @@ const Calendar = new Lang.Class({ | ||||
|         } | ||||
|  | ||||
|         // All the children after this are days, and get removed when we update the calendar | ||||
|         this._firstDayIndex = this.actor.get_n_children(); | ||||
|         this._firstDayIndex = this.actor.get_children().length; | ||||
|     }, | ||||
|  | ||||
|     _onStyleChange: function(actor, event) { | ||||
|         // width of a digit in pango units | ||||
|         this._digitWidth = _getDigitWidth(this.actor) / Pango.SCALE; | ||||
|         this._setWeekdateHeaderWidth(); | ||||
|     }, | ||||
|  | ||||
|     _setWeekdateHeaderWidth: function() { | ||||
|         if (this.digitWidth != NaN && this._useWeekdate && this._weekdateHeader) { | ||||
|             this._weekdateHeader.set_width (this._digitWidth * WEEKDATE_HEADER_WIDTH_DIGITS); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onScroll : function(actor, event) { | ||||
| @@ -549,7 +551,6 @@ const Calendar = new Lang.Class({ | ||||
|         let row = 2; | ||||
|         while (true) { | ||||
|             let button = new St.Button({ label: iter.getDate().toString() }); | ||||
|             let rtl = button.get_text_direction() == Clutter.TextDirection.RTL; | ||||
|  | ||||
|             if (!this._eventSource) | ||||
|                 button.reactive = false; | ||||
| @@ -570,10 +571,7 @@ const Calendar = new Lang.Class({ | ||||
|             // Hack used in lieu of border-collapse - see gnome-shell.css | ||||
|             if (row == 2) | ||||
|                 styleClass = 'calendar-day-top ' + styleClass; | ||||
|  | ||||
|             let leftMost = rtl ? iter.getDay() == (this._weekStart + 6) % 7 | ||||
|                                : iter.getDay() == this._weekStart; | ||||
|             if (leftMost) | ||||
|             if (iter.getDay() == this._weekStart) | ||||
|                 styleClass = 'calendar-day-left ' + styleClass; | ||||
|  | ||||
|             if (_sameDay(now, iter)) | ||||
| @@ -620,25 +618,16 @@ Signals.addSignalMethods(Calendar.prototype); | ||||
| const EventsList = new Lang.Class({ | ||||
|     Name: 'EventsList', | ||||
|  | ||||
|     _init: function() { | ||||
|     _init: function(eventSource) { | ||||
|         this.actor = new St.BoxLayout({ vertical: true, style_class: 'events-header-vbox'}); | ||||
|         this._date = new Date(); | ||||
|         this._eventSource = eventSource; | ||||
|         this._eventSource.connect('changed', Lang.bind(this, this._update)); | ||||
|         this._desktopSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' }); | ||||
|         this._desktopSettings.connect('changed', Lang.bind(this, this._update)); | ||||
|         this._weekStart = Shell.util_get_week_start(); | ||||
|     }, | ||||
|  | ||||
|     setEventSource: function(eventSource) { | ||||
|         if (this._eventSource) { | ||||
|             this._eventSource.disconnect(this._eventSourceChangedId); | ||||
|             this._eventSource = null; | ||||
|         } | ||||
|  | ||||
|         this._eventSource = eventSource; | ||||
|  | ||||
|         if (this._eventSource) { | ||||
|             this._eventSourceChangedId = this._eventSource.connect('changed', Lang.bind(this, this._update)); | ||||
|         } | ||||
|         this._update(); | ||||
|     }, | ||||
|  | ||||
|     _addEvent: function(dayNameBox, timeBox, eventTitleBox, includeDayName, day, time, desc) { | ||||
| @@ -724,15 +713,13 @@ const EventsList = new Lang.Class({ | ||||
|         let tomorrowEnd = new Date(dayEnd.getTime() + 86400 * 1000); | ||||
|         this._addPeriod(_("Tomorrow"), tomorrowBegin, tomorrowEnd, false, true); | ||||
|  | ||||
|         let dayInWeek = (dayEnd.getDay() - this._weekStart + 7) % 7; | ||||
|  | ||||
|         if (dayInWeek < 5) { | ||||
|         if (dayEnd.getDay() <= 4 + this._weekStart) { | ||||
|             /* If now is within the first 5 days we show "This week" and | ||||
|              * include events up until and including Saturday/Sunday | ||||
|              * (depending on whether a week starts on Sunday/Monday). | ||||
|              */ | ||||
|             let thisWeekBegin = new Date(dayBegin.getTime() + 2 * 86400 * 1000); | ||||
|             let thisWeekEnd = new Date(dayEnd.getTime() + (6 - dayInWeek) * 86400 * 1000); | ||||
|             let thisWeekEnd = new Date(dayEnd.getTime() + (6 + this._weekStart - dayEnd.getDay()) * 86400 * 1000); | ||||
|             this._addPeriod(_("This week"), thisWeekBegin, thisWeekEnd, true, false); | ||||
|         } else { | ||||
|             /* otherwise it's one of the two last days of the week ... show | ||||
| @@ -740,7 +727,7 @@ const EventsList = new Lang.Class({ | ||||
|              * Saturday/Sunday | ||||
|              */ | ||||
|             let nextWeekBegin = new Date(dayBegin.getTime() + 2 * 86400 * 1000); | ||||
|             let nextWeekEnd = new Date(dayEnd.getTime() + (13 - dayInWeek) * 86400 * 1000); | ||||
|             let nextWeekEnd = new Date(dayEnd.getTime() + (13 + this._weekStart - dayEnd.getDay()) * 86400 * 1000); | ||||
|             this._addPeriod(_("Next week"), nextWeekBegin, nextWeekEnd, true, false); | ||||
|         } | ||||
|     }, | ||||
|   | ||||
| @@ -1,65 +0,0 @@ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const ComponentManager = new Lang.Class({ | ||||
|     Name: 'ComponentManager', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._allComponents = {}; | ||||
|         this._enabledComponents = []; | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|         this._sessionUpdated(); | ||||
|     }, | ||||
|  | ||||
|     _sessionUpdated: function() { | ||||
|         let newEnabledComponents = Main.sessionMode.components; | ||||
|  | ||||
|         newEnabledComponents.filter(Lang.bind(this, function(name) { | ||||
|             return this._enabledComponents.indexOf(name) == -1; | ||||
|         })).forEach(Lang.bind(this, function(name) { | ||||
|             this._enableComponent(name); | ||||
|         })); | ||||
|  | ||||
|         this._enabledComponents.filter(Lang.bind(this, function(name) { | ||||
|             return newEnabledComponents.indexOf(name) == -1; | ||||
|         })).forEach(Lang.bind(this, function(name) { | ||||
|             this._disableComponent(name); | ||||
|         })); | ||||
|  | ||||
|         this._enabledComponents = newEnabledComponents; | ||||
|     }, | ||||
|  | ||||
|     _importComponent: function(name) { | ||||
|         let module = imports.ui.components[name]; | ||||
|         return module.Component; | ||||
|     }, | ||||
|  | ||||
|     _ensureComponent: function(name) { | ||||
|         let component = this._allComponents[name]; | ||||
|         if (component) | ||||
|             return component; | ||||
|  | ||||
| 	if (Main.sessionMode.isLocked) | ||||
| 	    return null; | ||||
|  | ||||
|         let constructor = this._importComponent(name); | ||||
|         component = new constructor(); | ||||
|         this._allComponents[name] = component; | ||||
|         return component; | ||||
|     }, | ||||
|  | ||||
|     _enableComponent: function(name) { | ||||
|         let component = this._ensureComponent(name); | ||||
| 	if (component) | ||||
|             component.enable(); | ||||
|     }, | ||||
|  | ||||
|     _disableComponent: function(name) { | ||||
|         let component = this._allComponents[name]; | ||||
|         if (component == null) | ||||
|             return; | ||||
|         component.disable(); | ||||
|     } | ||||
| }); | ||||
| @@ -1,243 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Params = imports.misc.params; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const GnomeSession = imports.misc.gnomeSession; | ||||
| const Main = imports.ui.main; | ||||
| const ShellMountOperation = imports.ui.shellMountOperation; | ||||
|  | ||||
| const GNOME_SESSION_AUTOMOUNT_INHIBIT = 16; | ||||
|  | ||||
| // GSettings keys | ||||
| const SETTINGS_SCHEMA = 'org.gnome.desktop.media-handling'; | ||||
| const SETTING_ENABLE_AUTOMOUNT = 'automount'; | ||||
|  | ||||
| const AUTORUN_EXPIRE_TIMEOUT_SECS = 10; | ||||
|  | ||||
| const AutomountManager = new Lang.Class({ | ||||
|     Name: 'AutomountManager', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|         this._volumeQueue = []; | ||||
|         this._session = new GnomeSession.SessionManager(); | ||||
|         this._session.connectSignal('InhibitorAdded', | ||||
|                                     Lang.bind(this, this._InhibitorsChanged)); | ||||
|         this._session.connectSignal('InhibitorRemoved', | ||||
|                                     Lang.bind(this, this._InhibitorsChanged)); | ||||
|         this._inhibited = false; | ||||
|  | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
|     }, | ||||
|  | ||||
|     enable: function() { | ||||
|         this._volumeAddedId = this._volumeMonitor.connect('volume-added', Lang.bind(this, this._onVolumeAdded)); | ||||
|         this._volumeRemovedId = this._volumeMonitor.connect('volume-removed', Lang.bind(this, this._onVolumeRemoved)); | ||||
|         this._driveConnectedId = this._volumeMonitor.connect('drive-connected', Lang.bind(this, this._onDriveConnected)); | ||||
|         this._driveDisconnectedId = this._volumeMonitor.connect('drive-disconnected', Lang.bind(this, this._onDriveDisconnected)); | ||||
|         this._driveEjectButtonId = this._volumeMonitor.connect('drive-eject-button', Lang.bind(this, this._onDriveEjectButton)); | ||||
|  | ||||
|         this._mountAllId = Mainloop.idle_add(Lang.bind(this, this._startupMountAll)); | ||||
|     }, | ||||
|  | ||||
|     disable: function() { | ||||
|         this._volumeMonitor.disconnect(this._volumeAddedId); | ||||
|         this._volumeMonitor.disconnect(this._volumeRemovedId); | ||||
|         this._volumeMonitor.disconnect(this._driveConnectedId); | ||||
|         this._volumeMonitor.disconnect(this._driveDisconnectedId); | ||||
|         this._volumeMonitor.disconnect(this._driveEjectButtonId); | ||||
|  | ||||
|         if (this._mountAllId > 0) { | ||||
|             Mainloop.source_remove(this._mountAllId); | ||||
|             this._mountAllId = 0; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _InhibitorsChanged: function(object, senderName, [inhibtor]) { | ||||
|         this._session.IsInhibitedRemote(GNOME_SESSION_AUTOMOUNT_INHIBIT, | ||||
|             Lang.bind(this, | ||||
|                 function(result, error) { | ||||
|                     if (!error) { | ||||
|                         this._inhibited = result[0]; | ||||
|                     } | ||||
|                 })); | ||||
|     }, | ||||
|  | ||||
|     _startupMountAll: function() { | ||||
|         let volumes = this._volumeMonitor.get_volumes(); | ||||
|         volumes.forEach(Lang.bind(this, function(volume) { | ||||
|             this._checkAndMountVolume(volume, { checkSession: false, | ||||
|                                                 useMountOp: false, | ||||
|                                                 allowAutorun: false }); | ||||
|         })); | ||||
|  | ||||
|         this._mountAllId = 0; | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onDriveConnected: function() { | ||||
|         // if we're not in the current ConsoleKit session, | ||||
|         // or screensaver is active, don't play sounds | ||||
|         if (!this._session.SessionIsActive) | ||||
|             return; | ||||
|  | ||||
|         global.play_theme_sound(0, 'device-added-media', | ||||
|                                 _("External drive connected"), | ||||
|                                 null); | ||||
|     }, | ||||
|  | ||||
|     _onDriveDisconnected: function() { | ||||
|         // if we're not in the current ConsoleKit session, | ||||
|         // or screensaver is active, don't play sounds | ||||
|         if (!this._session.SessionIsActive) | ||||
|             return; | ||||
|  | ||||
|         global.play_theme_sound(0, 'device-removed-media', | ||||
|                                 _("External drive disconnected"), | ||||
|                                 null); | ||||
|     }, | ||||
|  | ||||
|     _onDriveEjectButton: function(monitor, drive) { | ||||
|         // TODO: this code path is not tested, as the GVfs volume monitor | ||||
|         // doesn't emit this signal just yet. | ||||
|         if (!this._session.SessionIsActive) | ||||
|             return; | ||||
|  | ||||
|         // we force stop/eject in this case, so we don't have to pass a | ||||
|         // mount operation object | ||||
|         if (drive.can_stop()) { | ||||
|             drive.stop | ||||
|                 (Gio.MountUnmountFlags.FORCE, null, null, | ||||
|                  Lang.bind(this, function(drive, res) { | ||||
|                      try { | ||||
|                          drive.stop_finish(res); | ||||
|                      } catch (e) { | ||||
|                          log("Unable to stop the drive after drive-eject-button " + e.toString()); | ||||
|                      } | ||||
|                  })); | ||||
|         } else if (drive.can_eject()) { | ||||
|             drive.eject_with_operation  | ||||
|                 (Gio.MountUnmountFlags.FORCE, null, null, | ||||
|                  Lang.bind(this, function(drive, res) { | ||||
|                      try { | ||||
|                          drive.eject_with_operation_finish(res); | ||||
|                      } catch (e) { | ||||
|                          log("Unable to eject the drive after drive-eject-button " + e.toString()); | ||||
|                      } | ||||
|                  })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onVolumeAdded: function(monitor, volume) { | ||||
|         this._checkAndMountVolume(volume); | ||||
|     }, | ||||
|  | ||||
|     _checkAndMountVolume: function(volume, params) { | ||||
|         params = Params.parse(params, { checkSession: true, | ||||
|                                         useMountOp: true, | ||||
|                                         allowAutorun: true }); | ||||
|  | ||||
|         if (params.checkSession) { | ||||
|             // if we're not in the current ConsoleKit session, | ||||
|             // don't attempt automount | ||||
|             if (!this._session.SessionIsActive) | ||||
|                 return; | ||||
|         } | ||||
|  | ||||
|         if (this._inhibited) | ||||
|             return; | ||||
|  | ||||
|         // Volume is already mounted, don't bother. | ||||
|         if (volume.get_mount()) | ||||
|             return; | ||||
|  | ||||
|         if (!this._settings.get_boolean(SETTING_ENABLE_AUTOMOUNT) || | ||||
|             !volume.should_automount() || | ||||
|             !volume.can_mount()) { | ||||
|             // allow the autorun to run anyway; this can happen if the | ||||
|             // mount gets added programmatically later, even if  | ||||
|             // should_automount() or can_mount() are false, like for | ||||
|             // blank optical media. | ||||
|             this._allowAutorun(volume); | ||||
|             this._allowAutorunExpire(volume); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (params.useMountOp) { | ||||
|             let operation = new ShellMountOperation.ShellMountOperation(volume); | ||||
|             this._mountVolume(volume, operation, params.allowAutorun); | ||||
|         } else { | ||||
|             this._mountVolume(volume, null, params.allowAutorun); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _mountVolume: function(volume, operation, allowAutorun) { | ||||
|         if (allowAutorun) | ||||
|             this._allowAutorun(volume); | ||||
|  | ||||
|         let mountOp = operation ? operation.mountOp : null; | ||||
|         volume._operation = operation; | ||||
|  | ||||
|         volume.mount(0, mountOp, null, | ||||
|                      Lang.bind(this, this._onVolumeMounted)); | ||||
|     }, | ||||
|  | ||||
|     _onVolumeMounted: function(volume, res) { | ||||
|         this._allowAutorunExpire(volume); | ||||
|  | ||||
|         try { | ||||
|             volume.mount_finish(res); | ||||
|             this._closeOperation(volume); | ||||
|         } catch (e) { | ||||
|             // FIXME: we will always get G_IO_ERROR_FAILED from the gvfs udisks | ||||
|             // backend in this case, see  | ||||
|             // https://bugs.freedesktop.org/show_bug.cgi?id=51271 | ||||
|             if (e.message.indexOf('No key available with this passphrase') != -1) { | ||||
|                 this._reaskPassword(volume); | ||||
|             } else { | ||||
|                 if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                     log('Unable to mount volume ' + volume.get_name() + ': ' + e.toString()); | ||||
|  | ||||
|                 this._closeOperation(volume); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onVolumeRemoved: function(monitor, volume) { | ||||
|         this._volumeQueue =  | ||||
|             this._volumeQueue.filter(function(element) { | ||||
|                 return (element != volume); | ||||
|             }); | ||||
|     }, | ||||
|  | ||||
|     _reaskPassword: function(volume) { | ||||
|         let existingDialog = volume._operation ? volume._operation.borrowDialog() : null; | ||||
|         let operation =  | ||||
|             new ShellMountOperation.ShellMountOperation(volume, | ||||
|                                                         { existingDialog: existingDialog }); | ||||
|         this._mountVolume(volume, operation); | ||||
|     }, | ||||
|  | ||||
|     _closeOperation: function(volume) { | ||||
|         if (volume._operation) | ||||
|             volume._operation.close(); | ||||
|     }, | ||||
|  | ||||
|     _allowAutorun: function(volume) { | ||||
|         volume.allowAutorun = true; | ||||
|     }, | ||||
|  | ||||
|     _allowAutorunExpire: function(volume) { | ||||
|         Mainloop.timeout_add_seconds(AUTORUN_EXPIRE_TIMEOUT_SECS, function() { | ||||
|             volume.allowAutorun = false; | ||||
|             return false; | ||||
|         }); | ||||
|     } | ||||
| }); | ||||
| const Component = AutomountManager; | ||||
| @@ -1,62 +0,0 @@ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Recorder = new Lang.Class({ | ||||
|     Name: 'Recorder', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._recorderSettings = new Gio.Settings({ schema: 'org.gnome.shell.recorder' }); | ||||
|         this._desktopLockdownSettings = new Gio.Settings({ schema: 'org.gnome.desktop.lockdown' }); | ||||
|         this._bindingSettings = new Gio.Settings({ schema: 'org.gnome.shell.keybindings' }); | ||||
|         this._recorder = null; | ||||
|     }, | ||||
|  | ||||
|     enable: function() { | ||||
|         Main.wm.addKeybinding('toggle-recording', | ||||
|                               this._bindingSettings, | ||||
|                               Meta.KeyBindingFlags.NONE, | ||||
|                               Shell.KeyBindingMode.NORMAL | | ||||
|                               Shell.KeyBindingMode.OVERVIEW, | ||||
|                               Lang.bind(this, this._toggleRecorder)); | ||||
|     }, | ||||
|  | ||||
|     disable: function() { | ||||
|         Main.wm.removeKeybinding('toggle-recording'); | ||||
|     }, | ||||
|  | ||||
|     _ensureRecorder: function() { | ||||
|         if (this._recorder == null) | ||||
|             this._recorder = new Shell.Recorder({ stage: global.stage }); | ||||
|         return this._recorder; | ||||
|     }, | ||||
|  | ||||
|     _toggleRecorder: function() { | ||||
|         let recorder = this._ensureRecorder(); | ||||
|         if (recorder.is_recording()) { | ||||
|             recorder.close(); | ||||
|             Meta.enable_unredirect_for_screen(global.screen); | ||||
|         } else if (!this._desktopLockdownSettings.get_boolean('disable-save-to-disk')) { | ||||
|             // read the parameters from GSettings always in case they have changed | ||||
|             recorder.set_framerate(this._recorderSettings.get_int('framerate')); | ||||
|             /* Translators: this is a filename used for screencast recording */ | ||||
|             // xgettext:no-c-format | ||||
|             recorder.set_file_template(_("Screencast from %d %t") + '.' + this._recorderSettings.get_string('file-extension')); | ||||
|             let pipeline = this._recorderSettings.get_string('pipeline'); | ||||
|  | ||||
|             if (!pipeline.match(/^\s*$/)) | ||||
|                 recorder.set_pipeline(pipeline); | ||||
|             else | ||||
|                 recorder.set_pipeline(null); | ||||
|  | ||||
|             Meta.disable_unredirect_for_screen(global.screen); | ||||
|             recorder.record(); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const Component = Recorder; | ||||
							
								
								
									
										196
									
								
								js/ui/contactDisplay.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,196 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Folks = imports.gi.Folks | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
| const Atk = imports.gi.Atk; | ||||
|  | ||||
| const Util = imports.misc.util; | ||||
| const IconGrid = imports.ui.iconGrid; | ||||
| const Search = imports.ui.search; | ||||
| const SearchDisplay = imports.ui.searchDisplay; | ||||
|  | ||||
| const MAX_SEARCH_RESULTS_ROWS = 1; | ||||
| const ICON_SIZE = 81; | ||||
|  | ||||
| function launchContact(id) { | ||||
|     Util.spawn(['gnome-contacts', '-i', id]); | ||||
| } | ||||
|  | ||||
|  | ||||
| /* This class represents a shown contact search result in the overview */ | ||||
| const Contact = new Lang.Class({ | ||||
|     Name: 'Contact', | ||||
|  | ||||
|     _init: function(id) { | ||||
|         this._contactSys = Shell.ContactSystem.get_default(); | ||||
|         this.individual = this._contactSys.get_individual(id); | ||||
|  | ||||
|         this.actor = new St.Bin({ style_class: 'contact', | ||||
|                                   reactive: true, | ||||
|                                   can_focus: true, | ||||
|                                   track_hover: true, | ||||
|                                   accessible_role: Atk.Role.PUSH_BUTTON }); | ||||
|  | ||||
|         let content = new St.BoxLayout( { style_class: 'contact-content', | ||||
|                                           vertical: false }); | ||||
|         this.actor.set_child(content); | ||||
|  | ||||
|         let icon = new St.Icon({ icon_type: St.IconType.FULLCOLOR, | ||||
|                                  icon_size: ICON_SIZE, | ||||
|                                  style_class: 'contact-icon' }); | ||||
|         if (this.individual.avatar != null) | ||||
|             icon.gicon = this.individual.avatar; | ||||
|         else | ||||
|             icon.icon_name = 'avatar-default'; | ||||
|  | ||||
|         content.add(icon, { x_fill: true, | ||||
|                             y_fill: false, | ||||
|                             x_align: St.Align.START, | ||||
|                             y_align: St.Align.MIDDLE }); | ||||
|  | ||||
|         let details = new St.BoxLayout({ style_class: 'contact-details', | ||||
|                                          vertical: true }); | ||||
|         content.add(details, { x_fill: true, | ||||
|                                y_fill: false, | ||||
|                                x_align: St.Align.START, | ||||
|                                y_align: St.Align.MIDDLE }); | ||||
|  | ||||
|         let email = this._contactSys.get_email_for_display(this.individual); | ||||
|         let aliasText = this.individual.alias     || | ||||
|                         this.individual.full_name || | ||||
|                         this.individual.nickname  || | ||||
|                         email                     || | ||||
|                         _("Unknown"); | ||||
|         let aliasLabel = new St.Label({ text: aliasText, | ||||
|                                         style_class: 'contact-details-alias' }); | ||||
|         details.add(aliasLabel, { x_fill: true, | ||||
|                                   y_fill: false, | ||||
|                                   x_align: St.Align.START, | ||||
|                                   y_align: St.Align.START }); | ||||
|  | ||||
|         this.actor.label_actor = aliasLabel; | ||||
|  | ||||
|         let presence = this._createPresence(this.individual.presence_type); | ||||
|         details.add(presence, { x_fill: false, | ||||
|                                 y_fill: true, | ||||
|                                 x_align: St.Align.START, | ||||
|                                 y_align: St.Align.END }); | ||||
|     }, | ||||
|  | ||||
|     _createPresence: function(presence) { | ||||
|         let text; | ||||
|         let iconName; | ||||
|  | ||||
|         switch(presence) { | ||||
|           case Folks.PresenceType.AVAILABLE: | ||||
|             text = _("Available"); | ||||
|             iconName = 'user-available'; | ||||
|             break; | ||||
|           case Folks.PresenceType.AWAY: | ||||
|           case Folks.PresenceType.EXTENDED_AWAY: | ||||
|             text = _("Away"); | ||||
|             iconName = 'user-away'; | ||||
|             break; | ||||
|           case Folks.PresenceType.BUSY: | ||||
|             text = _("Busy"); | ||||
|             iconName = 'user-busy'; | ||||
|             break; | ||||
|           case Folks.PresenceType.OFFLINE: | ||||
|             text = _("Offline"); | ||||
|             iconName = 'user-offline'; | ||||
|             break; | ||||
|           default: | ||||
|             text = ''; | ||||
|             iconName = null; | ||||
|           } | ||||
|  | ||||
|         let box = new St.BoxLayout({ vertical: false, | ||||
|                                      style_class: 'contact-details-status' }); | ||||
|  | ||||
|         if (iconName) { | ||||
|             let icon = new St.Icon({ icon_name: iconName, | ||||
|                                      icon_type: St.IconType.FULLCOLOR, | ||||
|                                      icon_size: 16, | ||||
|                                      style_class: 'contact-details-status-icon' }); | ||||
|             box.add(icon, { x_fill: true, | ||||
|                             y_fill: false, | ||||
|                             x_align: St.Align.START, | ||||
|                             y_align: St.Align.START }); | ||||
|         } | ||||
|  | ||||
|         let label = new St.Label({ text: text }); | ||||
|  | ||||
|         box.add(label, { x_fill: true, | ||||
|                          y_fill: false, | ||||
|                          x_align: St.Align.END, | ||||
|                          y_align: St.Align.START }); | ||||
|  | ||||
|         return box; | ||||
|     }, | ||||
|  | ||||
|     createIcon: function(size) { | ||||
|         let tc = St.TextureCache.get_default(); | ||||
|         let icon = this.individual.avatar; | ||||
|  | ||||
|         if (icon != null) { | ||||
|             return tc.load_gicon(null, icon, size); | ||||
|         } else { | ||||
|             return tc.load_icon_name(null, 'avatar-default', St.IconType.FULLCOLOR, size); | ||||
|         } | ||||
|     }, | ||||
| }); | ||||
|  | ||||
|  | ||||
| /* Searches for and returns contacts */ | ||||
| const ContactSearchProvider = new Lang.Class({ | ||||
|     Name: 'ContactSearchProvider', | ||||
|     Extends: Search.SearchProvider, | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent(_("CONTACTS")); | ||||
|         this._contactSys = Shell.ContactSystem.get_default(); | ||||
|     }, | ||||
|  | ||||
|     getResultMetas: function(ids) { | ||||
|         let metas = []; | ||||
|         for (let i = 0; i < ids.length; i++) { | ||||
|             let contact = new Contact(ids[i]); | ||||
|             metas.push({ 'id': ids[i], | ||||
|                          'name': contact.alias, | ||||
|                          'createIcon': function(size) { | ||||
|                              return contact.createIcon(size); | ||||
|                          } | ||||
|                        }); | ||||
|         } | ||||
|         return metas; | ||||
|     }, | ||||
|  | ||||
|     getInitialResultSet: function(terms) { | ||||
|         return this._contactSys.initial_search(terms); | ||||
|     }, | ||||
|  | ||||
|     getSubsearchResultSet: function(previousResults, terms) { | ||||
|         return this._contactSys.subsearch(previousResults, terms); | ||||
|     }, | ||||
|  | ||||
|     createResultActor: function(resultMeta, terms) { | ||||
|         let contact = new Contact(resultMeta.id); | ||||
|         return contact.actor; | ||||
|     }, | ||||
|  | ||||
|     createResultContainerActor: function() { | ||||
|         let grid = new IconGrid.IconGrid({ rowLimit: MAX_SEARCH_RESULTS_ROWS, | ||||
|                                              xAlign: St.Align.START }); | ||||
|         grid.actor.style_class = 'contact-grid'; | ||||
|  | ||||
|         let actor = new SearchDisplay.GridSearchResults(this, grid); | ||||
|         return actor; | ||||
|     }, | ||||
|  | ||||
|     activateResult: function(id, params) { | ||||
|         launchContact(id); | ||||
|     } | ||||
| }); | ||||
| @@ -1,14 +1,15 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const AltTab = imports.ui.altTab; | ||||
| const Main = imports.ui.main; | ||||
| const SwitcherPopup = imports.ui.switcherPopup; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| @@ -26,9 +27,7 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|  | ||||
|     _init: function() { | ||||
|         this._items = []; | ||||
|         this.addGroup(global.window_group, _("Windows"), | ||||
|                       'emblem-documents-symbolic', { sortGroup: SortGroup.TOP, | ||||
|                                                      focusCallback: Lang.bind(this, this._focusWindows) }); | ||||
|         this._focusManager = St.FocusManager.get_for_stage(global.stage); | ||||
|     }, | ||||
|  | ||||
|     addGroup: function(root, name, icon, params) { | ||||
| @@ -42,13 +41,11 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|  | ||||
|         this._items.push(item); | ||||
|         root.connect('destroy', Lang.bind(this, function() { this.removeGroup(root); })); | ||||
|         if (root instanceof St.Widget) | ||||
|             global.focus_manager.add_group(root); | ||||
|         this._focusManager.add_group(root); | ||||
|     }, | ||||
|  | ||||
|     removeGroup: function(root) { | ||||
|         if (root instanceof St.Widget) | ||||
|             global.focus_manager.remove_group(root); | ||||
|         this._focusManager.remove_group(root); | ||||
|         for (let i = 0; i < this._items.length; i++) { | ||||
|             if (this._items[i].root == root) { | ||||
|                 this._items.splice(i, 1); | ||||
| @@ -57,16 +54,17 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     focusGroup: function(item, timestamp) { | ||||
|         if (item.focusCallback) { | ||||
|             item.focusCallback(timestamp); | ||||
|         } else { | ||||
|             if (global.stage_input_mode == Shell.StageInputMode.NONREACTIVE || | ||||
|                 global.stage_input_mode == Shell.StageInputMode.NORMAL) | ||||
|                 global.set_stage_input_mode(Shell.StageInputMode.FOCUSED); | ||||
|     focusGroup: function(item) { | ||||
|         if (global.stage_input_mode == Shell.StageInputMode.NONREACTIVE || | ||||
|             global.stage_input_mode == Shell.StageInputMode.NORMAL) | ||||
|             global.set_stage_input_mode(Shell.StageInputMode.FOCUSED); | ||||
|  | ||||
|         if (item.window) | ||||
|             Main.activateWindow(item.window); | ||||
|         else if (item.focusCallback) | ||||
|             item.focusCallback(); | ||||
|         else | ||||
|             item.root.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Sort the items into a consistent order; panel first, tray last, | ||||
| @@ -77,14 +75,24 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|         if (a.sortGroup != b.sortGroup) | ||||
|             return a.sortGroup - b.sortGroup; | ||||
|  | ||||
|         let ax, bx, y; | ||||
|         [ax, y] = a.proxy.get_transformed_position(); | ||||
|         [bx, y] = b.proxy.get_transformed_position(); | ||||
|         let y; | ||||
|         if (a.x == undefined) { | ||||
|             if (a.window) | ||||
|                 a.x = a.window.get_compositor_private().x; | ||||
|             else | ||||
|                 [a.x, y] = a.proxy.get_transformed_position(); | ||||
|         } | ||||
|         if (b.x == undefined) { | ||||
|             if (b.window) | ||||
|                 b.x = b.window.get_compositor_private().x; | ||||
|             else | ||||
|                 [b.x, y] = b.proxy.get_transformed_position(); | ||||
|         } | ||||
|  | ||||
|         return ax - bx; | ||||
|         return a.x - b.x; | ||||
|     }, | ||||
|  | ||||
|     popup: function(backward, binding, mask) { | ||||
|     popup: function(backwards, mask) { | ||||
|         // Start with the set of focus groups that are currently mapped | ||||
|         let items = this._items.filter(function (item) { return item.proxy.mapped; }); | ||||
|  | ||||
| @@ -102,7 +110,8 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|                     icon = app.create_icon_texture(POPUP_APPICON_SIZE); | ||||
|                 else | ||||
|                     icon = textureCache.bind_pixbuf_property(windows[i], 'icon'); | ||||
|                 items.push({ name: windows[i].title, | ||||
|                 items.push({ window: windows[i], | ||||
|                              name: windows[i].title, | ||||
|                              iconActor: icon, | ||||
|                              sortGroup: SortGroup.MIDDLE }); | ||||
|             } | ||||
| @@ -114,63 +123,187 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|         items.sort(Lang.bind(this, this._sortItems)); | ||||
|  | ||||
|         if (!this._popup) { | ||||
|             this._popup = new CtrlAltTabPopup(items); | ||||
|             this._popup.show(backward, binding, mask); | ||||
|             this._popup = new CtrlAltTabPopup(); | ||||
|             this._popup.show(items, backwards, mask); | ||||
|  | ||||
|             this._popup.actor.connect('destroy', | ||||
|                                       Lang.bind(this, function() { | ||||
|                                           this._popup = null; | ||||
|                                       })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _focusWindows: function(timestamp) { | ||||
|         global.set_stage_input_mode(Shell.StageInputMode.NORMAL); | ||||
|         global.stage.key_focus = null; | ||||
|         global.screen.focus_default_window(timestamp); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function mod(a, b) { | ||||
|     return (a + b) % b; | ||||
| } | ||||
|  | ||||
| const CtrlAltTabPopup = new Lang.Class({ | ||||
|     Name: 'CtrlAltTabPopup', | ||||
|     Extends: SwitcherPopup.SwitcherPopup, | ||||
|  | ||||
|     _createSwitcher: function() { | ||||
|         this._switcherList = new CtrlAltTabSwitcher(this._items); | ||||
|     _init : function() { | ||||
|         this.actor = new Shell.GenericContainer({ name: 'ctrlAltTabPopup', | ||||
|                                                   reactive: true }); | ||||
|  | ||||
|         this.actor.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth)); | ||||
|         this.actor.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight)); | ||||
|         this.actor.connect('allocate', Lang.bind(this, this._allocate)); | ||||
|  | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         this._haveModal = false; | ||||
|         this._modifierMask = 0; | ||||
|         this._selection = 0; | ||||
|  | ||||
|         Main.uiGroup.add_actor(this.actor); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function (actor, forHeight, alloc) { | ||||
|         let primary = Main.layoutManager.primaryMonitor; | ||||
|  | ||||
|         alloc.min_size = primary.width; | ||||
|         alloc.natural_size = primary.width; | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function (actor, forWidth, alloc) { | ||||
|         let primary = Main.layoutManager.primaryMonitor; | ||||
|  | ||||
|         alloc.min_size = primary.height; | ||||
|         alloc.natural_size = primary.height; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function (actor, box, flags) { | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         let primary = Main.layoutManager.primaryMonitor; | ||||
|  | ||||
|         let leftPadding = this.actor.get_theme_node().get_padding(St.Side.LEFT); | ||||
|         let vPadding = this.actor.get_theme_node().get_vertical_padding(); | ||||
|         let hPadding = this.actor.get_theme_node().get_horizontal_padding(); | ||||
|  | ||||
|         let [childMinHeight, childNaturalHeight] = this._switcher.actor.get_preferred_height(primary.width - hPadding); | ||||
|         let [childMinWidth, childNaturalWidth] = this._switcher.actor.get_preferred_width(childNaturalHeight); | ||||
|         childBox.x1 = Math.max(primary.x + leftPadding, primary.x + Math.floor((primary.width - childNaturalWidth) / 2)); | ||||
|         childBox.x2 = Math.min(primary.x + primary.width - hPadding, childBox.x1 + childNaturalWidth); | ||||
|         childBox.y1 = primary.y + Math.floor((primary.height - childNaturalHeight) / 2); | ||||
|         childBox.y2 = childBox.y1 + childNaturalHeight; | ||||
|         this._switcher.actor.allocate(childBox, flags); | ||||
|     }, | ||||
|  | ||||
|     show : function(items, startBackwards, mask) { | ||||
|         if (!Main.pushModal(this.actor)) | ||||
|             return false; | ||||
|         this._haveModal = true; | ||||
|         this._modifierMask = AltTab.primaryModifier(mask); | ||||
|  | ||||
|         this._keyPressEventId = this.actor.connect('key-press-event', Lang.bind(this, this._keyPressEvent)); | ||||
|         this._keyReleaseEventId = this.actor.connect('key-release-event', Lang.bind(this, this._keyReleaseEvent)); | ||||
|  | ||||
|         this._items = items; | ||||
|         this._switcher = new CtrlAltTabSwitcher(items); | ||||
|         this.actor.add_actor(this._switcher.actor); | ||||
|  | ||||
|         if (startBackwards) | ||||
|             this._selection = this._items.length - 1; | ||||
|         this._select(this._selection); | ||||
|  | ||||
|         let [x, y, mods] = global.get_pointer(); | ||||
|         if (!(mods & this._modifierMask)) { | ||||
|             this._finish(); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         this.actor.opacity = 0; | ||||
|         this.actor.show(); | ||||
|         Tweener.addTween(this.actor, | ||||
|                          { opacity: 255, | ||||
|                            time: POPUP_FADE_TIME, | ||||
|                            transition: 'easeOutQuad' | ||||
|                          }); | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _initialSelection: function(backward, binding) { | ||||
|         if (binding == 'switch-panels') { | ||||
|             if (backward) | ||||
|                 this._selectedIndex = this._items.length - 1; | ||||
|         } else if (binding == 'switch-panels-backward') { | ||||
|             if (!backward) | ||||
|                 this._selectedIndex = this._items.length - 1; | ||||
|         } | ||||
|         this._select(this._selectedIndex); | ||||
|     _next : function() { | ||||
|         return mod(this._selection + 1, this._items.length); | ||||
|     }, | ||||
|  | ||||
|     _keyPressHandler: function(keysym, backwards, action) { | ||||
|         if (action == Meta.KeyBindingAction.SWITCH_PANELS) | ||||
|             this._select(backwards ? this._previous() : this._next()); | ||||
|         else if (action == Meta.KeyBindingAction.SWITCH_PANELS_BACKWARD) | ||||
|             this._select(backwards ? this._next() : this._previous()); | ||||
|         else if (keysym == Clutter.Left) | ||||
|             this._select(this._previous()); | ||||
|         else if (keysym == Clutter.Right) | ||||
|     _previous : function() { | ||||
|         return mod(this._selection - 1, this._items.length); | ||||
|     }, | ||||
|  | ||||
|     _keyPressEvent : function(actor, event) { | ||||
|         let keysym = event.get_key_symbol(); | ||||
|         let shift = (event.get_state() & Clutter.ModifierType.SHIFT_MASK); | ||||
|         if (shift && keysym == Clutter.KEY_Tab) | ||||
|             keysym = Clutter.ISO_Left_Tab; | ||||
|  | ||||
|         if (keysym == Clutter.KEY_Escape) | ||||
|             this.destroy(); | ||||
|         else if (keysym == Clutter.KEY_Tab) | ||||
|             this._select(this._next()); | ||||
|         else if (keysym == Clutter.KEY_ISO_Left_Tab) | ||||
|             this._select(this._previous()); | ||||
|         else if (keysym == Clutter.KEY_Left) | ||||
|             this._select(this._previous()); | ||||
|         else if (keysym == Clutter.KEY_Right) | ||||
|             this._select(this._next()); | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _finish : function(time) { | ||||
|         this.parent(time); | ||||
|         Main.ctrlAltTabManager.focusGroup(this._items[this._selectedIndex], time); | ||||
|     _keyReleaseEvent : function(actor, event) { | ||||
|         let [x, y, mods] = global.get_pointer(); | ||||
|         let state = mods & this._modifierMask; | ||||
|  | ||||
|         if (state == 0) | ||||
|             this._finish(); | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _finish : function() { | ||||
|         this.destroy(); | ||||
|  | ||||
|         Main.ctrlAltTabManager.focusGroup(this._items[this._selection]); | ||||
|     }, | ||||
|  | ||||
|     _popModal: function() { | ||||
|         if (this._haveModal) { | ||||
|             Main.popModal(this.actor); | ||||
|             this._haveModal = false; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     destroy : function() { | ||||
|         this._popModal(); | ||||
|         Tweener.addTween(this.actor, | ||||
|                          { opacity: 0, | ||||
|                            time: POPUP_FADE_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, | ||||
|                                function() { | ||||
|                                    this.actor.destroy(); | ||||
|                                }) | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy : function() { | ||||
|         this._popModal(); | ||||
|         if (this._keyPressEventId) | ||||
|             this.actor.disconnect(this._keyPressEventId); | ||||
|         if (this._keyReleaseEventId) | ||||
|             this.actor.disconnect(this._keyReleaseEventId); | ||||
|     }, | ||||
|  | ||||
|     _select : function(num) { | ||||
|         this._selection = num; | ||||
|         this._switcher.highlight(num); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const CtrlAltTabSwitcher = new Lang.Class({ | ||||
|     Name: 'CtrlAltTabSwitcher', | ||||
|     Extends: SwitcherPopup.SwitcherList, | ||||
|     Extends: AltTab.SwitcherList, | ||||
|  | ||||
|     _init : function(items) { | ||||
|         this.parent(true); | ||||
| @@ -186,6 +319,7 @@ const CtrlAltTabSwitcher = new Lang.Class({ | ||||
|         let icon = item.iconActor; | ||||
|         if (!icon) { | ||||
|             icon = new St.Icon({ icon_name: item.iconName, | ||||
|                                  icon_type: St.IconType.SYMBOLIC, | ||||
|                                  icon_size: POPUP_APPICON_SIZE }); | ||||
|         } | ||||
|         box.add(icon, { x_fill: false, y_fill: false } ); | ||||
|   | ||||
							
								
								
									
										317
									
								
								js/ui/dash.js
									
									
									
									
									
								
							
							
						
						| @@ -21,14 +21,6 @@ const DASH_ITEM_LABEL_SHOW_TIME = 0.15; | ||||
| const DASH_ITEM_LABEL_HIDE_TIME = 0.1; | ||||
| const DASH_ITEM_HOVER_TIMEOUT = 300; | ||||
|  | ||||
| function getAppFromSource(source) { | ||||
|     if (source instanceof AppDisplay.AppIcon) { | ||||
|         return source.app; | ||||
|     } else { | ||||
|         return null; | ||||
|     } | ||||
| } | ||||
|  | ||||
| // A container like StBin, but taking the child's scale into account | ||||
| // when requesting a size | ||||
| const DashItemContainer = new Lang.Class({ | ||||
| @@ -44,11 +36,7 @@ const DashItemContainer = new Lang.Class({ | ||||
|                            Lang.bind(this, this._allocate)); | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         this._labelText = ""; | ||||
|         this.label = new St.Label({ style_class: 'dash-label'}); | ||||
|         this.label.hide(); | ||||
|         Main.layoutManager.addChrome(this.label); | ||||
|         this.actor.label_actor = this.label; | ||||
|         this.label = null; | ||||
|  | ||||
|         this.child = null; | ||||
|         this._childScale = 1; | ||||
| @@ -103,10 +91,9 @@ const DashItemContainer = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     showLabel: function() { | ||||
|         if (!this._labelText) | ||||
|         if (this.label == null) | ||||
|             return; | ||||
|  | ||||
|         this.label.set_text(this._labelText); | ||||
|         this.label.opacity = 0; | ||||
|         this.label.show(); | ||||
|  | ||||
| @@ -137,11 +124,16 @@ const DashItemContainer = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     setLabelText: function(text) { | ||||
|         this._labelText = text; | ||||
|         this.child.accessible_name = text; | ||||
|         if (this.label == null) | ||||
|             this.label = new St.Label({ style_class: 'dash-label'}); | ||||
|  | ||||
|         this.label.set_text(text); | ||||
|         Main.layoutManager.addChrome(this.label); | ||||
|         this.label.hide(); | ||||
|     }, | ||||
|  | ||||
|     hideLabel: function () { | ||||
|         this.label.opacity = 255; | ||||
|         Tweener.addTween(this.label, | ||||
|                          { opacity: 0, | ||||
|                            time: DASH_ITEM_LABEL_HIDE_TIME, | ||||
| @@ -176,17 +168,7 @@ const DashItemContainer = new Lang.Class({ | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         if (this.label) | ||||
|             this.label.destroy(); | ||||
|  | ||||
|         this.actor.destroy(); | ||||
|     }, | ||||
|  | ||||
|     animateOutAndDestroy: function() { | ||||
|         if (this.label) | ||||
|             this.label.destroy(); | ||||
|  | ||||
|         if (this.child == null) { | ||||
|             this.actor.destroy(); | ||||
|             return; | ||||
| @@ -235,76 +217,52 @@ const DashItemContainer = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const ShowAppsIcon = new Lang.Class({ | ||||
|     Name: 'ShowAppsIcon', | ||||
| const RemoveFavoriteIcon = new Lang.Class({ | ||||
|     Name: 'RemoveFavoriteIcon', | ||||
|     Extends: DashItemContainer, | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent(); | ||||
|  | ||||
|         this.toggleButton = new St.Button({ style_class: 'show-apps', | ||||
|                                             track_hover: true, | ||||
|                                             can_focus: true, | ||||
|                                             toggle_mode: true }); | ||||
|         this._iconBin = new St.Bin({ style_class: 'remove-favorite' }); | ||||
|         this._iconActor = null; | ||||
|         this.icon = new IconGrid.BaseIcon(_("Show Applications"), | ||||
|         this.icon = new IconGrid.BaseIcon(_("Remove"), | ||||
|                                            { setSizeManually: true, | ||||
|                                              showLabel: false, | ||||
|                                              createIcon: Lang.bind(this, this._createIcon) }); | ||||
|         this.toggleButton.add_actor(this.icon.actor); | ||||
|         this.toggleButton._delegate = this; | ||||
|         this._iconBin.set_child(this.icon.actor); | ||||
|         this._iconBin._delegate = this; | ||||
|  | ||||
|         this.setChild(this.toggleButton); | ||||
|         this.setDragApp(null); | ||||
|         this.setChild(this._iconBin); | ||||
|     }, | ||||
|  | ||||
|     _createIcon: function(size) { | ||||
|         this._iconActor = new St.Icon({ icon_name: 'view-grid-symbolic', | ||||
|                                         icon_size: size, | ||||
|                                         style_class: 'show-apps-icon', | ||||
|                                         track_hover: true }); | ||||
|         this._iconActor = new St.Icon({ icon_name: 'user-trash', | ||||
|                                         style_class: 'remove-favorite-icon', | ||||
|                                         icon_size: size }); | ||||
|         return this._iconActor; | ||||
|     }, | ||||
|  | ||||
|     _canRemoveApp: function(app) { | ||||
|         if (app == null) | ||||
|             return false; | ||||
|  | ||||
|         let id = app.get_id(); | ||||
|         let isFavorite = AppFavorites.getAppFavorites().isFavorite(id); | ||||
|         return isFavorite; | ||||
|     }, | ||||
|  | ||||
|     setDragApp: function(app) { | ||||
|         let canRemove = this._canRemoveApp(app); | ||||
|  | ||||
|         this.toggleButton.set_hover(canRemove); | ||||
|     setHover: function(hovered) { | ||||
|         this._iconBin.set_hover(hovered); | ||||
|         if (this._iconActor) | ||||
|             this._iconActor.set_hover(canRemove); | ||||
|  | ||||
|         if (canRemove) | ||||
|             this.setLabelText(_("Remove from Favorites")); | ||||
|         else | ||||
|             this.setLabelText(_("Show Applications")); | ||||
|             this._iconActor.set_hover(hovered); | ||||
|     }, | ||||
|  | ||||
|     // Rely on the dragged item being a favorite | ||||
|     handleDragOver: function(source, actor, x, y, time) { | ||||
|         let app = getAppFromSource(source); | ||||
|         if (app == null) | ||||
|             return DND.DragMotionResult.NO_DROP; | ||||
|  | ||||
|         let id = app.get_id(); | ||||
|         let isFavorite = AppFavorites.getAppFavorites().isFavorite(id); | ||||
|         if (!isFavorite) | ||||
|             return DND.DragMotionResult.NO_DROP; | ||||
|  | ||||
|         return DND.DragMotionResult.MOVE_DROP; | ||||
|     }, | ||||
|  | ||||
|     acceptDrop: function(source, actor, x, y, time) { | ||||
|         let app = getAppFromSource(source); | ||||
|         if (app == null) | ||||
|             return false; | ||||
|         let app = null; | ||||
|         if (source instanceof AppDisplay.AppWellIcon) { | ||||
|             let appSystem = Shell.AppSystem.get_default(); | ||||
|             app = appSystem.lookup_app(source.getId()); | ||||
|         } else if (source.metaWindow) { | ||||
|             let tracker = Shell.WindowTracker.get_default(); | ||||
|             app = tracker.get_window_app(source.metaWindow); | ||||
|         } | ||||
|  | ||||
|         let id = app.get_id(); | ||||
|  | ||||
| @@ -328,39 +286,6 @@ const DragPlaceholderItem = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const DashActor = new Lang.Class({ | ||||
|     Name: 'DashActor', | ||||
|     Extends: St.Widget, | ||||
|  | ||||
|     _init: function() { | ||||
|         let layout = new Clutter.BoxLayout({ orientation: Clutter.Orientation.VERTICAL }); | ||||
|         this.parent({ name: 'dash', | ||||
|                       layout_manager: layout, | ||||
|                       clip_to_allocation: true }); | ||||
|     }, | ||||
|  | ||||
|     vfunc_allocate: function(box, flags) { | ||||
|         let contentBox = this.get_theme_node().get_content_box(box); | ||||
|         let availWidth = contentBox.x2 - contentBox.x1; | ||||
|  | ||||
|         this.set_allocation(box, flags); | ||||
|  | ||||
|         let [appIcons, showAppsButton] = this.get_children(); | ||||
|         let [showAppsMinHeight, showAppsNatHeight] = showAppsButton.get_preferred_height(availWidth); | ||||
|  | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         childBox.x1 = contentBox.x1; | ||||
|         childBox.y1 = contentBox.y1; | ||||
|         childBox.x2 = contentBox.x2; | ||||
|         childBox.y2 = contentBox.y2 - showAppsNatHeight; | ||||
|         appIcons.allocate(childBox, flags); | ||||
|  | ||||
|         childBox.y1 = contentBox.y2 - showAppsNatHeight; | ||||
|         childBox.y2 = contentBox.y2; | ||||
|         showAppsButton.allocate(childBox, flags); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const Dash = new Lang.Class({ | ||||
|     Name: 'Dash', | ||||
|  | ||||
| @@ -372,25 +297,17 @@ const Dash = new Lang.Class({ | ||||
|         this._dragPlaceholder = null; | ||||
|         this._dragPlaceholderPos = -1; | ||||
|         this._animatingPlaceholdersCount = 0; | ||||
|         this._favRemoveTarget = null; | ||||
|         this._showLabelTimeoutId = 0; | ||||
|         this._resetHoverTimeoutId = 0; | ||||
|         this._labelShowing = false; | ||||
|  | ||||
|         this._container = new DashActor(); | ||||
|         this._box = new St.BoxLayout({ vertical: true, | ||||
|         this._box = new St.BoxLayout({ name: 'dash', | ||||
|                                        vertical: true, | ||||
|                                        clip_to_allocation: true }); | ||||
|         this._box._delegate = this; | ||||
|         this._container.add_actor(this._box); | ||||
|  | ||||
|         this._showAppsIcon = new ShowAppsIcon(); | ||||
|         this._showAppsIcon.icon.setIconSize(this.iconSize); | ||||
|         this._hookUpLabel(this._showAppsIcon); | ||||
|  | ||||
|         this.showAppsButton = this._showAppsIcon.toggleButton; | ||||
|  | ||||
|         this._container.add_actor(this._showAppsIcon.actor); | ||||
|  | ||||
|         this.actor = new St.Bin({ child: this._container }); | ||||
|         this.actor = new St.Bin({ y_align: St.Align.START, child: this._box }); | ||||
|         this.actor.connect('notify::height', Lang.bind(this, | ||||
|             function() { | ||||
|                 if (this._maxHeight != this.actor.height) | ||||
| @@ -400,6 +317,7 @@ const Dash = new Lang.Class({ | ||||
|  | ||||
|         this._workId = Main.initializeDeferredWork(this._box, Lang.bind(this, this._redisplay)); | ||||
|  | ||||
|         this._tracker = Shell.WindowTracker.get_default(); | ||||
|         this._appSystem = Shell.AppSystem.get_default(); | ||||
|  | ||||
|         this._appSystem.connect('installed-changed', Lang.bind(this, this._queueRedisplay)); | ||||
| @@ -412,10 +330,12 @@ const Dash = new Lang.Class({ | ||||
|                               Lang.bind(this, this._onDragEnd)); | ||||
|         Main.overview.connect('item-drag-cancelled', | ||||
|                               Lang.bind(this, this._onDragCancelled)); | ||||
|  | ||||
|         // Translators: this is the name of the dock/favorites area on | ||||
|         // the left of the overview | ||||
|         Main.ctrlAltTabManager.addGroup(this.actor, _("Dash"), 'user-bookmarks-symbolic'); | ||||
|         Main.overview.connect('window-drag-begin', | ||||
|                               Lang.bind(this, this._onDragBegin)); | ||||
|         Main.overview.connect('window-drag-cancelled', | ||||
|                               Lang.bind(this, this._onDragCancelled)); | ||||
|         Main.overview.connect('window-drag-end', | ||||
|                               Lang.bind(this, this._onDragEnd)); | ||||
|     }, | ||||
|  | ||||
|     _onDragBegin: function() { | ||||
| @@ -440,25 +360,54 @@ const Dash = new Lang.Class({ | ||||
|  | ||||
|     _endDrag: function() { | ||||
|         this._clearDragPlaceholder(); | ||||
|         this._showAppsIcon.setDragApp(null); | ||||
|         if (this._favRemoveTarget) { | ||||
|             this._favRemoveTarget.animateOutAndDestroy(); | ||||
|             this._favRemoveTarget.actor.connect('destroy', Lang.bind(this, | ||||
|                 function() { | ||||
|                     this._favRemoveTarget = null; | ||||
|                 })); | ||||
|             this._adjustIconSize(); | ||||
|         } | ||||
|         DND.removeDragMonitor(this._dragMonitor); | ||||
|     }, | ||||
|  | ||||
|     _onDragMotion: function(dragEvent) { | ||||
|         let app = getAppFromSource(dragEvent.source); | ||||
|         if (app == null) | ||||
|         let app = null; | ||||
|         if (dragEvent.source instanceof AppDisplay.AppWellIcon) | ||||
|             app = this._appSystem.lookup_app(dragEvent.source.getId()); | ||||
|         else if (dragEvent.source.metaWindow) | ||||
|             app = this._tracker.get_window_app(dragEvent.source.metaWindow); | ||||
|         else | ||||
|             return DND.DragMotionResult.CONTINUE; | ||||
|  | ||||
|         let showAppsHovered = | ||||
|                 this._showAppsIcon.actor.contains(dragEvent.targetActor); | ||||
|         let id = app.get_id(); | ||||
|  | ||||
|         if (!this._box.contains(dragEvent.targetActor) || showAppsHovered) | ||||
|         let favorites = AppFavorites.getAppFavorites().getFavoriteMap(); | ||||
|  | ||||
|         let srcIsFavorite = (id in favorites); | ||||
|  | ||||
|         if (srcIsFavorite && | ||||
|             app.get_state() != Shell.AppState.RUNNING && | ||||
|             dragEvent.source.actor && | ||||
|             this.actor.contains (dragEvent.source.actor) && | ||||
|             this._favRemoveTarget == null) { | ||||
|                 this._favRemoveTarget = new RemoveFavoriteIcon(); | ||||
|                 this._favRemoveTarget.icon.setIconSize(this.iconSize); | ||||
|                 this._box.add(this._favRemoveTarget.actor); | ||||
|                 this._adjustIconSize(); | ||||
|                 this._favRemoveTarget.animateIn(); | ||||
|         } | ||||
|  | ||||
|         let favRemoveHovered = false; | ||||
|         if (this._favRemoveTarget) | ||||
|             favRemoveHovered = | ||||
|                 this._favRemoveTarget.actor.contains(dragEvent.targetActor); | ||||
|  | ||||
|         if (!this._box.contains(dragEvent.targetActor) || favRemoveHovered) | ||||
|             this._clearDragPlaceholder(); | ||||
|  | ||||
|         if (showAppsHovered) | ||||
|             this._showAppsIcon.setDragApp(app); | ||||
|         else | ||||
|             this._showAppsIcon.setDragApp(null); | ||||
|         if (this._favRemoveTarget) | ||||
|             this._favRemoveTarget.setHover(favRemoveHovered); | ||||
|  | ||||
|         return DND.DragMotionResult.CONTINUE; | ||||
|     }, | ||||
| @@ -474,63 +423,37 @@ const Dash = new Lang.Class({ | ||||
|         Main.queueDeferredWork(this._workId); | ||||
|     }, | ||||
|  | ||||
|     _hookUpLabel: function(item) { | ||||
|         item.child.connect('notify::hover', Lang.bind(this, function() { | ||||
|             this._onHover(item); | ||||
|         })); | ||||
|  | ||||
|         Main.overview.connect('hiding', Lang.bind(this, function() { | ||||
|             this._labelShowing = false; | ||||
|             item.hideLabel(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _createAppItem: function(app) { | ||||
|         let appIcon = new AppDisplay.AppIcon(app, | ||||
|                                              { setSizeManually: true, | ||||
|                                                showLabel: false }); | ||||
|         appIcon._draggable.connect('drag-begin', | ||||
|         let display = new AppDisplay.AppWellIcon(app, | ||||
|                                                  { setSizeManually: true, | ||||
|                                                    showLabel: false }); | ||||
|         display._draggable.connect('drag-begin', | ||||
|                                    Lang.bind(this, function() { | ||||
|                                        appIcon.actor.opacity = 50; | ||||
|                                        display.actor.opacity = 50; | ||||
|                                    })); | ||||
|         appIcon._draggable.connect('drag-end', | ||||
|         display._draggable.connect('drag-end', | ||||
|                                    Lang.bind(this, function() { | ||||
|                                        appIcon.actor.opacity = 255; | ||||
|                                        display.actor.opacity = 255; | ||||
|                                    })); | ||||
|         appIcon.connect('menu-state-changed', | ||||
|                         Lang.bind(this, function(appIcon, opened) { | ||||
|                             this._itemMenuStateChanged(item, opened); | ||||
|                         })); | ||||
|  | ||||
|         let item = new DashItemContainer(); | ||||
|         item.setChild(appIcon.actor); | ||||
|         item.setChild(display.actor); | ||||
|  | ||||
|         // Override default AppIcon label_actor, now the | ||||
|         // accessible_name is set at DashItemContainer.setLabelText | ||||
|         appIcon.actor.label_actor = null; | ||||
|         item.setLabelText(app.get_name()); | ||||
|         // Override default AppWellIcon label_actor | ||||
|         display.actor.label_actor = item.label; | ||||
|  | ||||
|         appIcon.icon.setIconSize(this.iconSize); | ||||
|         this._hookUpLabel(item); | ||||
|  | ||||
|         display.icon.setIconSize(this.iconSize); | ||||
|         display.actor.connect('notify::hover', | ||||
|                                Lang.bind(this, function() { | ||||
|                                    this._onHover(item, display) | ||||
|                                })); | ||||
|         return item; | ||||
|     }, | ||||
|  | ||||
|     _itemMenuStateChanged: function(item, opened) { | ||||
|         // When the menu closes, it calls sync_hover, which means | ||||
|         // that the notify::hover handler does everything we need to. | ||||
|         if (opened) { | ||||
|             if (this._showLabelTimeoutId > 0) { | ||||
|                 Mainloop.source_remove(this._showLabelTimeoutId); | ||||
|                 this._showLabelTimeoutId = 0; | ||||
|             } | ||||
|  | ||||
|             item.hideLabel(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onHover: function (item) { | ||||
|         if (item.child.get_hover()) { | ||||
|     _onHover: function (item, display) { | ||||
|         if (display.actor.get_hover() && !display.isMenuUp) { | ||||
|             if (this._showLabelTimeoutId == 0) { | ||||
|                 let timeout = this._labelShowing ? 0 : DASH_ITEM_HOVER_TIMEOUT; | ||||
|                 this._showLabelTimeoutId = Mainloop.timeout_add(timeout, | ||||
| @@ -541,7 +464,7 @@ const Dash = new Lang.Class({ | ||||
|                     })); | ||||
|                 if (this._resetHoverTimeoutId > 0) { | ||||
|                     Mainloop.source_remove(this._resetHoverTimeoutId); | ||||
|                     this._resetHoverTimeoutId = 0; | ||||
|                 this._resetHoverTimeoutId = 0; | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
| @@ -571,12 +494,18 @@ const Dash = new Lang.Class({ | ||||
|                    !actor._delegate.animatingOut; | ||||
|         }); | ||||
|  | ||||
|         iconChildren.push(this._showAppsIcon.actor); | ||||
|         if (iconChildren.length == 0) { | ||||
|             this._box.add_style_pseudo_class('empty'); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._box.remove_style_pseudo_class('empty'); | ||||
|  | ||||
|         if (this._maxHeight == -1) | ||||
|             return; | ||||
|  | ||||
|         let themeNode = this._container.get_theme_node(); | ||||
|  | ||||
|         let themeNode = this._box.get_theme_node(); | ||||
|         let maxAllocation = new Clutter.ActorBox({ x1: 0, y1: 0, | ||||
|                                                    x2: 42 /* whatever */, | ||||
|                                                    y2: this._maxHeight }); | ||||
| @@ -602,6 +531,7 @@ const Dash = new Lang.Class({ | ||||
|             [minHeight, natHeight] = iconChildren[0].get_preferred_height(-1); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         // Subtract icon padding and box spacing from the available height | ||||
|         availHeight -= iconChildren.length * (natHeight - this.iconSize) + | ||||
|                        (iconChildren.length - 1) * spacing; | ||||
| @@ -632,10 +562,8 @@ const Dash = new Lang.Class({ | ||||
|             icon.setIconSize(this.iconSize); | ||||
|  | ||||
|             // Don't animate the icon size change when the overview | ||||
|             // is transitioning, not visible or when initially filling | ||||
|             // the dash | ||||
|             if (!Main.overview.visible || Main.overview.animationInProgress || | ||||
|                 !this._shownInitially) | ||||
|             // is not visible or when initially filling the dash | ||||
|             if (!Main.overview.visible || !this._shownInitially) | ||||
|                 continue; | ||||
|  | ||||
|             let [targetWidth, targetHeight] = icon.icon.get_size(); | ||||
| @@ -759,12 +687,11 @@ const Dash = new Lang.Class({ | ||||
|         for (let i = 0; i < removedActors.length; i++) { | ||||
|             let item = removedActors[i]._delegate; | ||||
|  | ||||
|             // Don't animate item removal when the overview is transitioning | ||||
|             // or hidden | ||||
|             if (Main.overview.visible && !Main.overview.animationInProgress) | ||||
|             // Don't animate item removal when the overview is hidden | ||||
|             if (Main.overview.visible) | ||||
|                 item.animateOutAndDestroy(); | ||||
|             else | ||||
|                 item.destroy(); | ||||
|                 item.actor.destroy(); | ||||
|         } | ||||
|  | ||||
|         this._adjustIconSize(); | ||||
| @@ -776,9 +703,8 @@ const Dash = new Lang.Class({ | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Don't animate item addition when the overview is transitioning | ||||
|         // or hidden | ||||
|         if (!Main.overview.visible || Main.overview.animationInProgress) | ||||
|         // Don't animate item addition when the overview is hidden | ||||
|         if (!Main.overview.visible) | ||||
|             return; | ||||
|  | ||||
|         for (let i = 0; i < addedItems.length; i++) | ||||
| @@ -794,7 +720,11 @@ const Dash = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     handleDragOver : function(source, actor, x, y, time) { | ||||
|         let app = getAppFromSource(source); | ||||
|         let app = null; | ||||
|         if (source instanceof AppDisplay.AppWellIcon) | ||||
|             app = this._appSystem.lookup_app(source.getId()); | ||||
|         else if (source.metaWindow) | ||||
|             app = this._tracker.get_window_app(source.metaWindow); | ||||
|  | ||||
|         // Don't allow favoriting of transient apps | ||||
|         if (app == null || app.is_window_backed()) | ||||
| @@ -875,7 +805,12 @@ const Dash = new Lang.Class({ | ||||
|  | ||||
|     // Draggable target interface | ||||
|     acceptDrop : function(source, actor, x, y, time) { | ||||
|         let app = getAppFromSource(source); | ||||
|         let app = null; | ||||
|         if (source instanceof AppDisplay.AppWellIcon) { | ||||
|             app = this._appSystem.lookup_app(source.getId()); | ||||
|         } else if (source.metaWindow) { | ||||
|             app = this._tracker.get_window_app(source.metaWindow); | ||||
|         } | ||||
|  | ||||
|         // Don't allow favoriting of transient apps | ||||
|         if (app == null || app.is_window_backed()) { | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GnomeDesktop = imports.gi.GnomeDesktop; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Cairo = imports.cairo; | ||||
| @@ -17,6 +16,14 @@ const Main = imports.ui.main; | ||||
| const PanelMenu = imports.ui.panelMenu; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
| const Calendar = imports.ui.calendar; | ||||
| const UPowerGlib = imports.gi.UPowerGlib; | ||||
|  | ||||
| // in org.gnome.desktop.interface | ||||
| const CLOCK_FORMAT_KEY        = 'clock-format'; | ||||
|  | ||||
| // in org.gnome.shell.clock | ||||
| const CLOCK_SHOW_DATE_KEY     = 'show-date'; | ||||
| const CLOCK_SHOW_SECONDS_KEY  = 'show-seconds'; | ||||
|  | ||||
| function _onVertSepRepaint (area) | ||||
| { | ||||
| @@ -32,14 +39,15 @@ function _onVertSepRepaint (area) | ||||
|     cr.setDash([1, 3], 1); // Hard-code for now | ||||
|     cr.setLineWidth(stippleWidth); | ||||
|     cr.stroke(); | ||||
|     cr.$dispose(); | ||||
| }; | ||||
|  | ||||
| const DateMenuButton = new Lang.Class({ | ||||
|     Name: 'DateMenuButton', | ||||
|     Extends: PanelMenu.Button, | ||||
|  | ||||
|     _init: function() { | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { showEvents: true }); | ||||
|  | ||||
|         let item; | ||||
|         let hbox; | ||||
|         let vbox; | ||||
| @@ -54,8 +62,8 @@ const DateMenuButton = new Lang.Class({ | ||||
|         // role ATK_ROLE_MENU like other elements of the panel. | ||||
|         this.actor.accessible_role = Atk.Role.LABEL; | ||||
|  | ||||
|         this._clockDisplay = new St.Label(); | ||||
|         this.actor.add_actor(this._clockDisplay); | ||||
|         this._clock = new St.Label(); | ||||
|         this.actor.add_actor(this._clock); | ||||
|  | ||||
|         hbox = new St.BoxLayout({name: 'calendarArea' }); | ||||
|         this.menu.addActor(hbox); | ||||
| @@ -67,12 +75,20 @@ const DateMenuButton = new Lang.Class({ | ||||
|  | ||||
|         // Date | ||||
|         this._date = new St.Label(); | ||||
|         this.actor.label_actor = this._clockDisplay; | ||||
|         this.actor.label_actor = this._date; | ||||
|         this._date.style_class = 'datemenu-date-label'; | ||||
|         vbox.add(this._date); | ||||
|  | ||||
|         this._eventList = new Calendar.EventsList(); | ||||
|         this._calendar = new Calendar.Calendar(); | ||||
|         if (params.showEvents) { | ||||
|             this._eventSource = new Calendar.DBusEventSource(); | ||||
|             this._eventList = new Calendar.EventsList(this._eventSource); | ||||
|         } else { | ||||
|             this._eventSource = null; | ||||
|             this._eventList = null; | ||||
|         } | ||||
|  | ||||
|         // Calendar | ||||
|         this._calendar = new Calendar.Calendar(this._eventSource); | ||||
|  | ||||
|         this._calendar.connect('selected-date-changed', | ||||
|                                Lang.bind(this, function(calendar, date) { | ||||
| @@ -84,44 +100,37 @@ const DateMenuButton = new Lang.Class({ | ||||
|                                })); | ||||
|         vbox.add(this._calendar.actor); | ||||
|  | ||||
|         let separator = new PopupMenu.PopupSeparatorMenuItem(); | ||||
|         separator.setColumnWidths(1); | ||||
|         vbox.add(separator.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|  | ||||
|         this._openCalendarItem = new PopupMenu.PopupMenuItem(_("Open Calendar")); | ||||
|         this._openCalendarItem.connect('activate', Lang.bind(this, this._onOpenCalendarActivate)); | ||||
|         this._openCalendarItem.actor.can_focus = false; | ||||
|         vbox.add(this._openCalendarItem.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|  | ||||
|         this._openClocksItem = new PopupMenu.PopupMenuItem(_("Open Clocks")); | ||||
|         this._openClocksItem.connect('activate', Lang.bind(this, this._onOpenClocksActivate)); | ||||
|         this._openClocksItem.actor.can_focus = false; | ||||
|         vbox.add(this._openClocksItem.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|  | ||||
|         Shell.AppSystem.get_default().connect('installed-changed', | ||||
|                                               Lang.bind(this, this._appInstalledChanged)); | ||||
|         this._appInstalledChanged(); | ||||
|  | ||||
|         item = this.menu.addSettingsAction(_("Date & Time Settings"), 'gnome-datetime-panel.desktop'); | ||||
|         item = this.menu.addSettingsAction(_("Date and Time Settings"), 'gnome-datetime-panel.desktop'); | ||||
|         if (item) { | ||||
|             item.actor.show_on_set_parent = false; | ||||
|             let separator = new PopupMenu.PopupSeparatorMenuItem(); | ||||
|             separator.setColumnWidths(1); | ||||
|             vbox.add(separator.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|  | ||||
|             item.actor.can_focus = false; | ||||
|             item.actor.reparent(vbox); | ||||
|             this._dateAndTimeSeparator = separator; | ||||
|         } | ||||
|  | ||||
|         this._separator = new St.DrawingArea({ style_class: 'calendar-vertical-separator', | ||||
|                                                pseudo_class: 'highlighted' }); | ||||
|         this._separator.connect('repaint', Lang.bind(this, _onVertSepRepaint)); | ||||
|         hbox.add(this._separator); | ||||
|         if (params.showEvents) { | ||||
|             // Add vertical separator | ||||
|  | ||||
|         // Fill up the second column | ||||
|         vbox = new St.BoxLayout({ name: 'calendarEventsArea', | ||||
|                                   vertical: true }); | ||||
|         hbox.add(vbox, { expand: true }); | ||||
|             item = new St.DrawingArea({ style_class: 'calendar-vertical-separator', | ||||
|                                         pseudo_class: 'highlighted' }); | ||||
|             item.connect('repaint', Lang.bind(this, _onVertSepRepaint)); | ||||
|             hbox.add(item); | ||||
|  | ||||
|         // Event list | ||||
|         vbox.add(this._eventList.actor, { expand: true }); | ||||
|             // Fill up the second column | ||||
|             vbox = new St.BoxLayout({name:     'calendarEventsArea', | ||||
|                                      vertical: true}); | ||||
|             hbox.add(vbox, { expand: true }); | ||||
|  | ||||
|             // Event list | ||||
|             vbox.add(this._eventList.actor, { expand: true }); | ||||
|  | ||||
|             item = new PopupMenu.PopupMenuItem(_("Open Calendar")); | ||||
|             item.connect('activate', Lang.bind(this, this._onOpenCalendarActivate)); | ||||
|             item.actor.can_focus = false; | ||||
|             vbox.add(item.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|         } | ||||
|  | ||||
|         // Whenever the menu is opened, select today | ||||
|         this.menu.connect('open-state-changed', Lang.bind(this, function(menu, isOpen) { | ||||
| @@ -148,75 +157,90 @@ const DateMenuButton = new Lang.Class({ | ||||
|  | ||||
|         // Done with hbox for calendar and event list | ||||
|  | ||||
|         this._clock = new GnomeDesktop.WallClock(); | ||||
|         this._clock.connect('notify::clock', Lang.bind(this, this._updateClockAndDate)); | ||||
|         // Track changes to clock settings | ||||
|         this._desktopSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' }); | ||||
|         this._clockSettings = new Gio.Settings({ schema: 'org.gnome.shell.clock' }); | ||||
|         this._desktopSettings.connect('changed', Lang.bind(this, this._updateClockAndDate)); | ||||
|         this._clockSettings.connect('changed', Lang.bind(this, this._updateClockAndDate)); | ||||
|  | ||||
|         // https://bugzilla.gnome.org/show_bug.cgi?id=655129 | ||||
|         this._upClient = new UPowerGlib.Client(); | ||||
|         this._upClient.connect('notify-resume', Lang.bind(this, this._updateClockAndDate)); | ||||
|  | ||||
|         // Start the clock | ||||
|         this._updateClockAndDate(); | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|         this._sessionUpdated(); | ||||
|     }, | ||||
|  | ||||
|     _appInstalledChanged: function() { | ||||
|         let app = Shell.AppSystem.get_default().lookup_app('gnome-clocks.desktop'); | ||||
|         this._openClocksItem.actor.visible = app !== null; | ||||
|     }, | ||||
|  | ||||
|     _setEventsVisibility: function(visible) { | ||||
|         this._openCalendarItem.actor.visible = visible; | ||||
|         this._separator.visible = visible; | ||||
|         if (visible) { | ||||
|           let alignment = 0.25; | ||||
|           if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) | ||||
|             alignment = 1.0 - alignment; | ||||
|           this.menu._arrowAlignment = alignment; | ||||
|           this._eventList.actor.get_parent().show(); | ||||
|         } else { | ||||
|           this.menu._arrowAlignment = 0.5; | ||||
|           this._eventList.actor.get_parent().hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _setEventSource: function(eventSource) { | ||||
|         this._calendar.setEventSource(eventSource); | ||||
|         this._eventList.setEventSource(eventSource); | ||||
|     }, | ||||
|  | ||||
|     _sessionUpdated: function() { | ||||
|         let eventSource; | ||||
|         let showEvents = Main.sessionMode.showCalendarEvents; | ||||
|         if (showEvents) { | ||||
|             eventSource = new Calendar.DBusEventSource(); | ||||
|         } else { | ||||
|             eventSource = null; | ||||
|         } | ||||
|         this._setEventSource(eventSource); | ||||
|         this._setEventsVisibility(showEvents); | ||||
|  | ||||
|         // This needs to be handled manually, as the code to | ||||
|         // autohide separators doesn't work across the vbox | ||||
|         this._dateAndTimeSeparator.actor.visible = Main.sessionMode.allowSettings; | ||||
|     }, | ||||
|  | ||||
|     _updateClockAndDate: function() { | ||||
|         this._clockDisplay.set_text(this._clock.clock); | ||||
|         let format = this._desktopSettings.get_string(CLOCK_FORMAT_KEY); | ||||
|         let showDate = this._clockSettings.get_boolean(CLOCK_SHOW_DATE_KEY); | ||||
|         let showSeconds = this._clockSettings.get_boolean(CLOCK_SHOW_SECONDS_KEY); | ||||
|  | ||||
|         let clockFormat; | ||||
|         let dateFormat; | ||||
|  | ||||
|         switch (format) { | ||||
|             case '24h': | ||||
|                 if (showDate) | ||||
|                     /* Translators: This is the time format with date used | ||||
|                        in 24-hour mode. */ | ||||
|                     clockFormat = showSeconds ? _("%a %b %e, %R:%S") | ||||
|                                               : _("%a %b %e, %R"); | ||||
|                 else | ||||
|                     /* Translators: This is the time format without date used | ||||
|                        in 24-hour mode. */ | ||||
|                     clockFormat = showSeconds ? _("%a %R:%S") | ||||
|                                               : _("%a %R"); | ||||
|                 break; | ||||
|             case '12h': | ||||
|             default: | ||||
|                 if (showDate) | ||||
|                     /* Translators: This is a time format with date used | ||||
|                        for AM/PM. */ | ||||
|                     clockFormat = showSeconds ? _("%a %b %e, %l:%M:%S %p") | ||||
|                                               : _("%a %b %e, %l:%M %p"); | ||||
|                 else | ||||
|                     /* Translators: This is a time format without date used | ||||
|                        for AM/PM. */ | ||||
|                     clockFormat = showSeconds ? _("%a %l:%M:%S %p") | ||||
|                                               : _("%a %l:%M %p"); | ||||
|                 break; | ||||
|         } | ||||
|  | ||||
|         let displayDate = new Date(); | ||||
|  | ||||
|         this._clock.set_text(displayDate.toLocaleFormat(clockFormat)); | ||||
|  | ||||
|         /* Translators: This is the date format to use when the calendar popup is | ||||
|          * shown - it is shown just below the time in the shell (e.g. "Tue 9:29 AM"). | ||||
|          */ | ||||
|         let dateFormat = _("%A %B %e, %Y"); | ||||
|         let displayDate = new Date(); | ||||
|         dateFormat = _("%A %B %e, %Y"); | ||||
|         this._date.set_text(displayDate.toLocaleFormat(dateFormat)); | ||||
|  | ||||
|         Mainloop.timeout_add_seconds(1, Lang.bind(this, this._updateClockAndDate)); | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onOpenCalendarActivate: function() { | ||||
|         this.menu.close(); | ||||
|  | ||||
|         let app = Gio.AppInfo.get_default_for_type('text/calendar', false); | ||||
|         app.launch([], global.create_app_launch_context()); | ||||
|     }, | ||||
|  | ||||
|     _onOpenClocksActivate: function() { | ||||
|         this.menu.close(); | ||||
|         let app = Shell.AppSystem.get_default().lookup_app('gnome-clocks.desktop'); | ||||
|         app.activate(); | ||||
|         let calendarSettings = new Gio.Settings({ schema: 'org.gnome.desktop.default-applications.office.calendar' }); | ||||
|         let tool = calendarSettings.get_string('exec'); | ||||
|         if (tool.length == 0 || tool == 'evolution') { | ||||
|             // TODO: pass the selected day | ||||
|             Util.spawn(['evolution', '-c', 'calendar']); | ||||
|         } else { | ||||
|             let needTerm = calendarSettings.get_boolean('needs-term'); | ||||
|             if (needTerm) { | ||||
|                 let terminalSettings = new Gio.Settings({ schema: 'org.gnome.desktop.default-applications.terminal' }); | ||||
|                 let term = terminalSettings.get_string('exec'); | ||||
|                 let arg = terminalSettings.get_string('exec-arg'); | ||||
|                 if (arg != '') | ||||
|                     Util.spawn([term, arg, tool]); | ||||
|                 else | ||||
|                     Util.spawn([term, tool]); | ||||
|             } else { | ||||
|                 Util.spawnCommandLine(tool) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|   | ||||
							
								
								
									
										31
									
								
								js/ui/dnd.js
									
									
									
									
									
								
							
							
						
						| @@ -136,10 +136,9 @@ const _Draggable = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _ungrabActor: function() { | ||||
|         Clutter.ungrab_pointer(); | ||||
|         if (!this._onEventId) | ||||
|             return; | ||||
|  | ||||
|         Clutter.ungrab_pointer(); | ||||
|         this.actor.disconnect(this._onEventId); | ||||
|         this._onEventId = null; | ||||
|     }, | ||||
| @@ -204,19 +203,6 @@ const _Draggable = new Lang.Class({ | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * fakeRelease: | ||||
|      * | ||||
|      * Fake a release event. | ||||
|      * Must be called if you want to intercept release events on draggable | ||||
|      * actors for other purposes (for example if you're using | ||||
|      * PopupMenu.ignoreRelease()) | ||||
|      */ | ||||
|     fakeRelease: function() { | ||||
|         this._buttonDown = false; | ||||
|         this._ungrabActor(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * startDrag: | ||||
|      * @stageX: X coordinate of event | ||||
| @@ -248,11 +234,7 @@ const _Draggable = new Lang.Class({ | ||||
|         this._dragY = this._dragStartY = stageY; | ||||
|  | ||||
|         if (this.actor._delegate && this.actor._delegate.getDragActor) { | ||||
|             this._dragActor = this.actor._delegate.getDragActor(); | ||||
|             this._dragActor.reparent(Main.uiGroup); | ||||
|             this._dragActor.raise_top(); | ||||
|             Shell.util_set_hidden_from_pick(this._dragActor, true); | ||||
|  | ||||
|             this._dragActor = this.actor._delegate.getDragActor(this._dragStartX, this._dragStartY); | ||||
|             // Drag actor does not always have to be the same as actor. For example drag actor | ||||
|             // can be an image that's part of the actor. So to perform "snap back" correctly we need | ||||
|             // to know what was the drag actor source. | ||||
| @@ -281,17 +263,12 @@ const _Draggable = new Lang.Class({ | ||||
|             this._dragOffsetY = this._dragActor.y - this._dragStartY; | ||||
|         } else { | ||||
|             this._dragActor = this.actor; | ||||
|  | ||||
|             this._dragActorSource = undefined; | ||||
|             this._dragOrigParent = this.actor.get_parent(); | ||||
|             this._dragOrigX = this._dragActor.x; | ||||
|             this._dragOrigY = this._dragActor.y; | ||||
|             this._dragOrigScale = this._dragActor.scale_x; | ||||
|  | ||||
|             this._dragActor.reparent(Main.uiGroup); | ||||
|             this._dragActor.raise_top(); | ||||
|             Shell.util_set_hidden_from_pick(this._dragActor, true); | ||||
|  | ||||
|             let [actorStageX, actorStageY] = this.actor.get_transformed_position(); | ||||
|             this._dragOffsetX = actorStageX - this._dragStartX; | ||||
|             this._dragOffsetY = actorStageY - this._dragStartY; | ||||
| @@ -303,6 +280,10 @@ const _Draggable = new Lang.Class({ | ||||
|                                  scaledHeight / this.actor.height); | ||||
|         } | ||||
|  | ||||
|         this._dragActor.reparent(Main.uiGroup); | ||||
|         this._dragActor.raise_top(); | ||||
|         Shell.util_set_hidden_from_pick(this._dragActor, true); | ||||
|  | ||||
|         this._dragOrigOpacity = this._dragActor.opacity; | ||||
|         if (this._dragActorOpacity != undefined) | ||||
|             this._dragActor.opacity = this._dragActorOpacity; | ||||
|   | ||||
| @@ -19,7 +19,6 @@ | ||||
|  */ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const AccountsService = imports.gi.AccountsService; | ||||
| @@ -32,10 +31,10 @@ const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const GnomeSession = imports.misc.gnomeSession; | ||||
| const Lightbox = imports.ui.lightbox; | ||||
| const Main = imports.ui.main; | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const UserMenu = imports.ui.userMenu; | ||||
|  | ||||
| let _endSessionDialog = null; | ||||
|  | ||||
| @@ -51,7 +50,6 @@ const EndSessionDialogIface = <interface name="org.gnome.SessionManager.EndSessi | ||||
|     <arg type="u" direction="in" /> | ||||
|     <arg type="ao" direction="in" /> | ||||
| </method> | ||||
| <method name="Close" /> | ||||
| <signal name="ConfirmedLogout" /> | ||||
| <signal name="ConfirmedReboot" /> | ||||
| <signal name="ConfirmedShutdown" /> | ||||
| @@ -92,7 +90,7 @@ const shutdownDialogContent = { | ||||
|                        label:  C_("button", "Restart") }, | ||||
|                      { signal: 'ConfirmedShutdown', | ||||
|                        label:  C_("button", "Power Off") }], | ||||
|     iconName: 'system-shutdown-symbolic', | ||||
|     iconName: 'system-shutdown', | ||||
|     iconStyleClass: 'end-session-dialog-shutdown-icon' | ||||
| }; | ||||
|  | ||||
| @@ -107,7 +105,7 @@ const restartDialogContent = { | ||||
|     endDescription: _("Restarting the system."), | ||||
|     confirmButtons: [{ signal: 'ConfirmedReboot', | ||||
|                        label:  C_("button", "Restart") }], | ||||
|     iconName: 'view-refresh-symbolic', | ||||
|     iconName: 'system-shutdown', | ||||
|     iconStyleClass: 'end-session-dialog-shutdown-icon' | ||||
| }; | ||||
|  | ||||
| @@ -163,7 +161,6 @@ const ListItem = new Lang.Class({ | ||||
|  | ||||
|         this._descriptionLabel = new St.Label({ text:        this._reason, | ||||
|                                                 style_class: 'end-session-dialog-app-list-item-description' }); | ||||
|         this.actor.label_actor = this._nameLabel; | ||||
|         textLayout.add(this._descriptionLabel, | ||||
|                        { expand: true, | ||||
|                          x_fill: true }); | ||||
| @@ -283,17 +280,21 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         scrollView.hide(); | ||||
|  | ||||
|         this._applicationList = new St.BoxLayout({ vertical: true }); | ||||
|         scrollView.add_actor(this._applicationList); | ||||
|         scrollView.add_actor(this._applicationList, | ||||
|                              { x_fill:  true, | ||||
|                                y_fill:  true, | ||||
|                                x_align: St.Align.START, | ||||
|                                y_align: St.Align.MIDDLE }); | ||||
|  | ||||
|         this._applicationList.connect('actor-added', | ||||
|                                       Lang.bind(this, function() { | ||||
|                                           if (this._applicationList.get_n_children() == 1) | ||||
|                                           if (this._applicationList.get_children().length == 1) | ||||
|                                               scrollView.show(); | ||||
|                                       })); | ||||
|  | ||||
|         this._applicationList.connect('actor-removed', | ||||
|                                       Lang.bind(this, function() { | ||||
|                                           if (this._applicationList.get_n_children() == 0) | ||||
|                                           if (this._applicationList.get_children().length == 0) | ||||
|                                               scrollView.hide(); | ||||
|                                       })); | ||||
|  | ||||
| @@ -306,7 +307,42 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         this._user.disconnect(this._userChangedId); | ||||
|     }, | ||||
|  | ||||
|     _updateDescription: function() { | ||||
|     _setIconFromFile: function(iconFile, styleClass) { | ||||
|         if (styleClass) | ||||
|             this._iconBin.set_style_class_name(styleClass); | ||||
|         this._iconBin.set_style(null); | ||||
|  | ||||
|         this._iconBin.child = null; | ||||
|         if (iconFile) { | ||||
|             this._iconBin.show(); | ||||
|             this._iconBin.set_style('background-image: url("' + iconFile + '");' + | ||||
|                                     'background-size: contain;'); | ||||
|         } else { | ||||
|             this._iconBin.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _setIconFromName: function(iconName, styleClass) { | ||||
|         if (styleClass) | ||||
|             this._iconBin.set_style_class_name(styleClass); | ||||
|         this._iconBin.set_style(null); | ||||
|  | ||||
|         if (iconName != null) { | ||||
|             let textureCache = St.TextureCache.get_default(); | ||||
|             let icon = textureCache.load_icon_name(this._iconBin.get_theme_node(), | ||||
|                                                    iconName, | ||||
|                                                    St.IconType.SYMBOLIC, | ||||
|                                                    _DIALOG_ICON_SIZE); | ||||
|  | ||||
|             this._iconBin.child = icon; | ||||
|             this._iconBin.show(); | ||||
|         } else { | ||||
|             this._iconBin.child = null; | ||||
|             this._iconBin.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateContent: function() { | ||||
|         if (this.state != ModalDialog.State.OPENING && | ||||
|             this.state != ModalDialog.State.OPENED) | ||||
|             return; | ||||
| @@ -316,6 +352,17 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         let subject = dialogContent.subject; | ||||
|         let description; | ||||
|  | ||||
|         if (this._user.is_loaded && !dialogContent.iconName) { | ||||
|             let iconFile = this._user.get_icon_file(); | ||||
|             if (GLib.file_test(iconFile, GLib.FileTest.EXISTS)) | ||||
|                 this._setIconFromFile(iconFile, dialogContent.iconStyleClass); | ||||
|             else | ||||
|                 this._setIconFromName('avatar-default', dialogContent.iconStyleClass); | ||||
|         } else if (dialogContent.iconName) { | ||||
|             this._setIconFromName(dialogContent.iconName, | ||||
|                                   dialogContent.iconStyleClass); | ||||
|         } | ||||
|  | ||||
|         if (this._inhibitors.length > 0) { | ||||
|             this._stopTimer(); | ||||
|             description = dialogContent.inhibitedDescription; | ||||
| @@ -348,27 +395,6 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         _setLabelText(this._descriptionLabel, description); | ||||
|     }, | ||||
|  | ||||
|     _updateContent: function() { | ||||
|         if (this.state != ModalDialog.State.OPENING && | ||||
|             this.state != ModalDialog.State.OPENED) | ||||
|             return; | ||||
|  | ||||
|         let dialogContent = DialogContent[this._type]; | ||||
|         if (dialogContent.iconName) { | ||||
|             this._iconBin.child = new St.Icon({ icon_name: dialogContent.iconName, | ||||
|                                                 icon_size: _DIALOG_ICON_SIZE, | ||||
|                                                 style_class: dialogContent.iconStyleClass }); | ||||
|         } else { | ||||
|             let avatarWidget = new UserMenu.UserAvatarWidget(this._user, | ||||
|                                                              { iconSize: _DIALOG_ICON_SIZE, | ||||
|                                                                styleClass: dialogContent.iconStyleClass }); | ||||
|             this._iconBin.child = avatarWidget.actor; | ||||
|             avatarWidget.update(); | ||||
|         } | ||||
|  | ||||
|         this._updateDescription(); | ||||
|     }, | ||||
|  | ||||
|     _updateButtons: function() { | ||||
|         let dialogContent = DialogContent[this._type]; | ||||
|         let buttons = [{ action: Lang.bind(this, this.cancel), | ||||
| @@ -379,12 +405,7 @@ const EndSessionDialog = new Lang.Class({ | ||||
|             let signal = dialogContent.confirmButtons[i].signal; | ||||
|             let label = dialogContent.confirmButtons[i].label; | ||||
|             buttons.push({ action: Lang.bind(this, function() { | ||||
|                                        this.close(true); | ||||
|                                        let signalId = this.connect('closed', | ||||
|                                                                    Lang.bind(this, function() { | ||||
|                                                                        this.disconnect(signalId); | ||||
|                                                                        this._confirm(signal); | ||||
|                                                                    })); | ||||
|                                        this._confirm(signal); | ||||
|                                    }), | ||||
|                            label: label }); | ||||
|         } | ||||
| @@ -392,17 +413,15 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         this.setButtons(buttons); | ||||
|     }, | ||||
|  | ||||
|     close: function(skipSignal) { | ||||
|     close: function() { | ||||
|         this.parent(); | ||||
|  | ||||
|         if (!skipSignal) | ||||
|             this._dbusImpl.emit_signal('Closed', null); | ||||
|         this._dbusImpl.emit_signal('Closed', null); | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|         this._stopTimer(); | ||||
|         this._dbusImpl.emit_signal('Canceled', null); | ||||
|         this.close(); | ||||
|         this.close(global.get_current_time()); | ||||
|     }, | ||||
|  | ||||
|     _confirm: function(signal) { | ||||
| @@ -417,34 +436,22 @@ const EndSessionDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _startTimer: function() { | ||||
|         let startTime = GLib.get_monotonic_time(); | ||||
|         this._secondsLeft = this._totalSecondsToStayOpen; | ||||
|  | ||||
|         this._timerId = Mainloop.timeout_add_seconds(1, Lang.bind(this, | ||||
|             function() { | ||||
|                 let currentTime = GLib.get_monotonic_time(); | ||||
|                 let secondsElapsed = ((currentTime - startTime) / 1000000); | ||||
|  | ||||
|                 this._secondsLeft = this._totalSecondsToStayOpen - secondsElapsed; | ||||
|                 if (this._secondsLeft > 0) { | ||||
|                     this._updateDescription(); | ||||
|                     return true; | ||||
|                 } | ||||
|  | ||||
|                 let dialogContent = DialogContent[this._type]; | ||||
|                 let button = dialogContent.confirmButtons[dialogContent.confirmButtons.length - 1]; | ||||
|                 this._confirm(button.signal); | ||||
|  | ||||
|                 return false; | ||||
|             })); | ||||
|         Tweener.addTween(this, | ||||
|                          { _secondsLeft: 0, | ||||
|                            time: this._secondsLeft, | ||||
|                            transition: 'linear', | ||||
|                            onUpdate: Lang.bind(this, this._updateContent), | ||||
|                            onComplete: Lang.bind(this, function() { | ||||
|                                            let dialogContent = DialogContent[this._type]; | ||||
|                                            let button = dialogContent.confirmButtons[dialogContent.confirmButtons.length - 1]; | ||||
|                                            this._confirm(button.signal); | ||||
|                                        }), | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     _stopTimer: function() { | ||||
|         if (this._timerId != 0) { | ||||
|             Mainloop.source_remove(this._timerId); | ||||
|             this._timerId = 0; | ||||
|         } | ||||
|  | ||||
|         Tweener.removeTweens(this); | ||||
|         this._secondsLeft = 0; | ||||
|     }, | ||||
|  | ||||
| @@ -461,7 +468,7 @@ const EndSessionDialog = new Lang.Class({ | ||||
|             let item = new ListItem(app, reason); | ||||
|             item.connect('activate', | ||||
|                          Lang.bind(this, function() { | ||||
|                              this.close(); | ||||
|                              this.close(global.get_current_time()); | ||||
|                          })); | ||||
|             this._applicationList.add(item.actor, { x_fill: true }); | ||||
|             this._stopTimer(); | ||||
| @@ -509,9 +516,5 @@ const EndSessionDialog = new Lang.Class({ | ||||
|                                         invocation.return_value(null); | ||||
|                                         this.disconnect(signalId); | ||||
|                                     })); | ||||
|     }, | ||||
|  | ||||
|     Close: function(parameters, invocation) { | ||||
|         this.close(); | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -39,19 +39,11 @@ function _patchContainerClass(containerClass) { | ||||
|     }; | ||||
| } | ||||
|  | ||||
| function _makeLoggingFunc(func) { | ||||
|     return function() { | ||||
|         return func([].join.call(arguments, ', ')); | ||||
|     }; | ||||
| } | ||||
|  | ||||
| function init() { | ||||
|     // Add some bindings to the global JS namespace; (gjs keeps the web | ||||
|     // browser convention of having that namespace be called 'window'.) | ||||
|     window.global = Shell.Global.get(); | ||||
|  | ||||
|     window.log = _makeLoggingFunc(window.log); | ||||
|  | ||||
|     window._ = Gettext.gettext; | ||||
|     window.C_ = Gettext.pgettext; | ||||
|     window.ngettext = Gettext.ngettext; | ||||
| @@ -90,7 +82,7 @@ function init() { | ||||
|     } | ||||
|  | ||||
|     // OK, now things are initialized enough that we can import shell JS | ||||
|     const Format = imports.format; | ||||
|     const Format = imports.misc.format; | ||||
|     const Tweener = imports.ui.tweener; | ||||
|  | ||||
|     Tweener.init(); | ||||
|   | ||||
| @@ -1,270 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Soup = imports.gi.Soup; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
| const ExtensionSystem = imports.ui.extensionSystem; | ||||
| const FileUtils = imports.misc.fileUtils; | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
|  | ||||
| const _signals = ExtensionSystem._signals; | ||||
|  | ||||
| const REPOSITORY_URL_BASE = 'https://extensions.gnome.org'; | ||||
| const REPOSITORY_URL_DOWNLOAD = REPOSITORY_URL_BASE + '/download-extension/%s.shell-extension.zip'; | ||||
| const REPOSITORY_URL_INFO     = REPOSITORY_URL_BASE + '/extension-info/'; | ||||
| const REPOSITORY_URL_UPDATE   = REPOSITORY_URL_BASE + '/update-info/'; | ||||
|  | ||||
| let _httpSession; | ||||
|  | ||||
| function installExtension(uuid, invocation) { | ||||
|     let params = { uuid: uuid, | ||||
|                    shell_version: Config.PACKAGE_VERSION }; | ||||
|  | ||||
|     let message = Soup.form_request_new_from_hash('GET', REPOSITORY_URL_INFO, params); | ||||
|  | ||||
|     _httpSession.queue_message(message, function(session, message) { | ||||
|         if (message.status_code != Soup.KnownStatusCode.OK) { | ||||
|             ExtensionSystem.logExtensionError(uuid, 'downloading info: ' + message.status_code); | ||||
|             invocation.return_dbus_error('org.gnome.Shell.DownloadInfoError', message.status_code.toString()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let info; | ||||
|         try { | ||||
|             info = JSON.parse(message.response_body.data); | ||||
|         } catch (e) { | ||||
|             ExtensionSystem.logExtensionError(uuid, 'parsing info: ' + e); | ||||
|             invocation.return_dbus_error('org.gnome.Shell.ParseInfoError', e.toString()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let dialog = new InstallExtensionDialog(uuid, info, invocation); | ||||
|         dialog.open(global.get_current_time()); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function uninstallExtension(uuid) { | ||||
|     let extension = ExtensionUtils.extensions[uuid]; | ||||
|     if (!extension) | ||||
|         return false; | ||||
|  | ||||
|     // Don't try to uninstall system extensions | ||||
|     if (extension.type != ExtensionUtils.ExtensionType.PER_USER) | ||||
|         return false; | ||||
|  | ||||
|     if (!ExtensionSystem.unloadExtension(extension)) | ||||
|         return false; | ||||
|  | ||||
|     FileUtils.recursivelyDeleteDir(extension.dir, true); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| function gotExtensionZipFile(session, message, uuid, dir, callback, errback) { | ||||
|     if (message.status_code != Soup.KnownStatusCode.OK) { | ||||
|         errback('DownloadExtensionError', message.status_code); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|         if (!dir.query_exists(null)) | ||||
|             dir.make_directory_with_parents(null); | ||||
|     } catch (e) { | ||||
|         errback('CreateExtensionDirectoryError', e); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     let [file, stream] = Gio.File.new_tmp('XXXXXX.shell-extension.zip'); | ||||
|     let contents = message.response_body.flatten().get_as_bytes(); | ||||
|     stream.output_stream.write_bytes(contents, null); | ||||
|     stream.close(null); | ||||
|     let [success, pid] = GLib.spawn_async(null, | ||||
|                                           ['unzip', '-uod', dir.get_path(), '--', file.get_path()], | ||||
|                                           null, | ||||
|                                           GLib.SpawnFlags.SEARCH_PATH | GLib.SpawnFlags.DO_NOT_REAP_CHILD, | ||||
|                                           null); | ||||
|  | ||||
|     if (!success) { | ||||
|         errback('ExtractExtensionError'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     GLib.child_watch_add(GLib.PRIORITY_DEFAULT, pid, function(pid, status) { | ||||
|         GLib.spawn_close_pid(pid); | ||||
|  | ||||
|         if (status != 0) | ||||
|             errback('ExtractExtensionError'); | ||||
|         else | ||||
|             callback(); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function updateExtension(uuid) { | ||||
|     // This gets a bit tricky. We want the update to be seamless - | ||||
|     // if we have any error during downloading or extracting, we | ||||
|     // want to not unload the current version. | ||||
|  | ||||
|     let oldExtensionTmpDir = GLib.Dir.make_tmp('XXXXXX-shell-extension'); | ||||
|     let newExtensionTmpDir = GLib.Dir.make_tmp('XXXXXX-shell-extension'); | ||||
|  | ||||
|     let params = { shell_version: Config.PACKAGE_VERSION }; | ||||
|  | ||||
|     let url = REPOSITORY_URL_DOWNLOAD.format(uuid); | ||||
|     let message = Soup.form_request_new_from_hash('GET', url, params); | ||||
|  | ||||
|     _httpSession.queue_message(message, Lang.bind(this, function(session, message) { | ||||
|         gotExtensionZipFile(session, message, uuid, newExtensionTmpDir, function() { | ||||
|             let oldExtension = ExtensionUtils.extensions[uuid]; | ||||
|             let extensionDir = oldExtension.dir; | ||||
|  | ||||
|             if (!ExtensionSystem.unloadExtension(oldExtension)) | ||||
|                 return; | ||||
|  | ||||
|             FileUtils.recursivelyMoveDir(extensionDir, oldExtensionTmpDir); | ||||
|             FileUtils.recursivelyMoveDir(newExtensionTmpDir, extensionDir); | ||||
|  | ||||
|             let extension = ExtensionUtils.createExtensionObject(uuid, extensionDir, ExtensionUtils.ExtensionType.PER_USER); | ||||
|  | ||||
|             try { | ||||
|                 ExtensionSystem.loadExtension(extension); | ||||
|             } catch(e) { | ||||
|                 ExtensionSystem.unloadExtension(extension); | ||||
|  | ||||
|                 logError(e, 'Error loading extension %s'.format(uuid)); | ||||
|  | ||||
|                 FileUtils.recursivelyDeleteDir(extensionDir, false); | ||||
|                 FileUtils.recursivelyMoveDir(oldExtensionTmpDir, extensionDir); | ||||
|  | ||||
|                 // Restore what was there before. We can't do much if we | ||||
|                 // fail here. | ||||
|                 ExtensionSystem.loadExtension(oldExtension); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             FileUtils.recursivelyDeleteDir(oldExtensionTmpDir, true); | ||||
|         }, function(code, message) { | ||||
|             log('Error while updating extension %s: %s (%s)'.format(uuid, code, message ? message : '')); | ||||
|         }); | ||||
|     })); | ||||
| } | ||||
|  | ||||
| function checkForUpdates() { | ||||
|     let metadatas = {}; | ||||
|     for (let uuid in ExtensionUtils.extensions) { | ||||
|         metadatas[uuid] = ExtensionUtils.extensions[uuid].metadata; | ||||
|     } | ||||
|  | ||||
|     let params = { shell_version: Config.PACKAGE_VERSION, | ||||
|                    installed: JSON.stringify(metadatas) }; | ||||
|  | ||||
|     let url = REPOSITORY_URL_UPDATE; | ||||
|     let message = Soup.form_request_new_from_hash('GET', url, params); | ||||
|     _httpSession.queue_message(message, function(session, message) { | ||||
|         if (message.status_code != Soup.KnownStatusCode.OK) | ||||
|             return; | ||||
|  | ||||
|         let operations = JSON.parse(message.response_body.data); | ||||
|         for (let uuid in operations) { | ||||
|             let operation = operations[uuid]; | ||||
|             if (operation == 'blacklist') | ||||
|                 uninstallExtension(uuid); | ||||
|             else if (operation == 'upgrade' || operation == 'downgrade') | ||||
|                 updateExtension(uuid); | ||||
|         } | ||||
|     }); | ||||
| } | ||||
|  | ||||
| const InstallExtensionDialog = new Lang.Class({ | ||||
|     Name: 'InstallExtensionDialog', | ||||
|     Extends: ModalDialog.ModalDialog, | ||||
|  | ||||
|     _init: function(uuid, info, invocation) { | ||||
|         this.parent({ styleClass: 'extension-dialog' }); | ||||
|  | ||||
|         this._uuid = uuid; | ||||
|         this._info = info; | ||||
|         this._invocation = invocation; | ||||
|  | ||||
|         this.setButtons([{ label: _("Cancel"), | ||||
|                            action: Lang.bind(this, this._onCancelButtonPressed), | ||||
|                            key:    Clutter.Escape | ||||
|                          }, | ||||
|                          { label:  _("Install"), | ||||
|                            action: Lang.bind(this, this._onInstallButtonPressed), | ||||
|                            default: true | ||||
|                          }]); | ||||
|  | ||||
|         let message = _("Download and install '%s' from extensions.gnome.org?").format(info.name); | ||||
|  | ||||
|         let box = new St.BoxLayout(); | ||||
|         this.contentLayout.add(box); | ||||
|  | ||||
|         let gicon = new Gio.FileIcon({ file: Gio.File.new_for_uri(REPOSITORY_URL_BASE + info.icon) }) | ||||
|         let icon = new St.Icon({ gicon: gicon }); | ||||
|         box.add(icon); | ||||
|  | ||||
|         let label = new St.Label({ text: message }); | ||||
|         box.add(label); | ||||
|     }, | ||||
|  | ||||
|     _onCancelButtonPressed: function(button, event) { | ||||
|         this.close(); | ||||
|         this._invocation.return_value(GLib.Variant.new('(s)', ['cancelled'])); | ||||
|     }, | ||||
|  | ||||
|     _onInstallButtonPressed: function(button, event) { | ||||
|         let params = { shell_version: Config.PACKAGE_VERSION }; | ||||
|  | ||||
|         let url = REPOSITORY_URL_DOWNLOAD.format(this._uuid); | ||||
|         let message = Soup.form_request_new_from_hash('GET', url, params); | ||||
|  | ||||
|         let uuid = this._uuid; | ||||
|         let dir = Gio.File.new_for_path(GLib.build_filenamev([global.userdatadir, 'extensions', uuid])); | ||||
|         let invocation = this._invocation; | ||||
|         function errback(code, message) { | ||||
|             invocation.return_dbus_error('org.gnome.Shell.' + code, message ? message.toString() : ''); | ||||
|         } | ||||
|  | ||||
|         function callback() { | ||||
|             // Add extension to 'enabled-extensions' for the user, always... | ||||
|             let enabledExtensions = global.settings.get_strv(ExtensionSystem.ENABLED_EXTENSIONS_KEY); | ||||
|             if (enabledExtensions.indexOf(uuid) == -1) { | ||||
|                 enabledExtensions.push(uuid); | ||||
|                 global.settings.set_strv(ExtensionSystem.ENABLED_EXTENSIONS_KEY, enabledExtensions); | ||||
|             } | ||||
|  | ||||
|             let extension = ExtensionUtils.createExtensionObject(uuid, dir, ExtensionUtils.ExtensionType.PER_USER); | ||||
|  | ||||
|             try { | ||||
|                 ExtensionSystem.loadExtension(extension); | ||||
|             } catch(e) { | ||||
|                 uninstallExtension(uuid); | ||||
|                 errback('LoadExtensionError', e); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             invocation.return_value(GLib.Variant.new('(s)', 'successful')); | ||||
|         } | ||||
|  | ||||
|         _httpSession.queue_message(message, Lang.bind(this, function(session, message) { | ||||
|             gotExtensionZipFile(session, message, uuid, dir, callback, errback); | ||||
|         })); | ||||
|  | ||||
|         this.close(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function init() { | ||||
|     _httpSession = new Soup.SessionAsync({ ssl_use_system_ca_file: true }); | ||||
|  | ||||
|     // See: https://bugzilla.gnome.org/show_bug.cgi?id=655189 for context. | ||||
|     // _httpSession.add_feature(new Soup.ProxyResolverDefault()); | ||||
|     Soup.Session.prototype.add_feature.call(_httpSession, new Soup.ProxyResolverDefault()); | ||||
| } | ||||
| @@ -3,12 +3,19 @@ | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Soup = imports.gi.Soup; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
| const Main = imports.ui.main; | ||||
| const FileUtils = imports.misc.fileUtils; | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
|  | ||||
| const API_VERSION = 1; | ||||
|  | ||||
| const ExtensionState = { | ||||
|     ENABLED: 1, | ||||
| @@ -23,6 +30,29 @@ const ExtensionState = { | ||||
|     UNINSTALLED: 99 | ||||
| }; | ||||
|  | ||||
| const REPOSITORY_URL_BASE = 'https://extensions.gnome.org'; | ||||
| const REPOSITORY_URL_DOWNLOAD = REPOSITORY_URL_BASE + '/download-extension/%s.shell-extension.zip'; | ||||
| const REPOSITORY_URL_INFO =     REPOSITORY_URL_BASE + '/extension-info/'; | ||||
|  | ||||
| const _httpSession = new Soup.SessionAsync(); | ||||
|  | ||||
| // The unfortunate state of gjs, gobject-introspection and libsoup | ||||
| // means that I have to do a hack to add a feature. | ||||
| // See: https://bugzilla.gnome.org/show_bug.cgi?id=655189 for context. | ||||
|  | ||||
| if (Soup.Session.prototype.add_feature != null) | ||||
|     Soup.Session.prototype.add_feature.call(_httpSession, new Soup.ProxyResolverDefault()); | ||||
|  | ||||
| function _getCertFile() { | ||||
|     let localCert = GLib.build_filenamev([global.userdatadir, 'extensions.gnome.org.crt']); | ||||
|     if (GLib.file_test(localCert, GLib.FileTest.EXISTS)) | ||||
|         return localCert; | ||||
|     else | ||||
|         return Config.SHELL_SYSTEM_CA_FILE; | ||||
| } | ||||
|  | ||||
| _httpSession.ssl_ca_file = _getCertFile(); | ||||
|  | ||||
| // Arrays of uuids | ||||
| var enabledExtensions; | ||||
| // Contains the order that extensions were enabled in. | ||||
| @@ -39,8 +69,89 @@ const disconnect = Lang.bind(_signals, _signals.disconnect); | ||||
|  | ||||
| const ENABLED_EXTENSIONS_KEY = 'enabled-extensions'; | ||||
|  | ||||
| var initted = false; | ||||
| var enabled; | ||||
| function installExtensionFromUUID(uuid, version_tag) { | ||||
|     let params = { uuid: uuid, | ||||
|                    version_tag: version_tag, | ||||
|                    shell_version: Config.PACKAGE_VERSION, | ||||
|                    api_version: API_VERSION.toString() }; | ||||
|  | ||||
|     let message = Soup.form_request_new_from_hash('GET', REPOSITORY_URL_INFO, params); | ||||
|  | ||||
|     _httpSession.queue_message(message, | ||||
|                                function(session, message) { | ||||
|                                    let info = JSON.parse(message.response_body.data); | ||||
|                                    let dialog = new InstallExtensionDialog(uuid, version_tag, info.name); | ||||
|                                    dialog.open(global.get_current_time()); | ||||
|                                }); | ||||
| } | ||||
|  | ||||
| function uninstallExtensionFromUUID(uuid) { | ||||
|     let extension = ExtensionUtils.extensions[uuid]; | ||||
|     if (!extension) | ||||
|         return false; | ||||
|  | ||||
|     // Try to disable it -- if it's ERROR'd, we can't guarantee that, | ||||
|     // but it will be removed on next reboot, and hopefully nothing | ||||
|     // broke too much. | ||||
|     disableExtension(uuid); | ||||
|  | ||||
|     // Don't try to uninstall system extensions | ||||
|     if (extension.type != ExtensionUtils.ExtensionType.PER_USER) | ||||
|         return false; | ||||
|  | ||||
|     extension.state = ExtensionState.UNINSTALLED; | ||||
|     _signals.emit('extension-state-changed', extension); | ||||
|  | ||||
|     delete ExtensionUtils.extensions[uuid]; | ||||
|  | ||||
|     FileUtils.recursivelyDeleteDir(Gio.file_new_for_path(extension.path)); | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| function gotExtensionZipFile(session, message, uuid) { | ||||
|     if (message.status_code != Soup.KnownStatusCode.OK) { | ||||
|         logExtensionError(uuid, 'downloading extension: ' + message.status_code); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // FIXME: use a GFile mkstemp-type method once one exists | ||||
|     let fd, tmpzip; | ||||
|     try { | ||||
|         [fd, tmpzip] = GLib.file_open_tmp('XXXXXX.shell-extension.zip'); | ||||
|     } catch (e) { | ||||
|         logExtensionError(uuid, 'tempfile: ' + e.toString()); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     let stream = new Gio.UnixOutputStream({ fd: fd }); | ||||
|     let dir = ExtensionUtils.userExtensionsDir.get_child(uuid); | ||||
|     Shell.write_soup_message_to_stream(stream, message); | ||||
|     stream.close(null); | ||||
|     let [success, pid] = GLib.spawn_async(null, | ||||
|                                           ['unzip', '-uod', dir.get_path(), '--', tmpzip], | ||||
|                                           null, | ||||
|                                           GLib.SpawnFlags.SEARCH_PATH | GLib.SpawnFlags.DO_NOT_REAP_CHILD, | ||||
|                                           null); | ||||
|  | ||||
|     if (!success) { | ||||
|         logExtensionError(uuid, 'extract: could not extract'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     GLib.child_watch_add(GLib.PRIORITY_DEFAULT, pid, function(pid, status) { | ||||
|         GLib.spawn_close_pid(pid); | ||||
|  | ||||
|         // Add extension to 'enabled-extensions' for the user, always... | ||||
|         let enabledExtensions = global.settings.get_strv(ENABLED_EXTENSIONS_KEY); | ||||
|         if (enabledExtensions.indexOf(uuid) == -1) { | ||||
|             enabledExtensions.push(uuid); | ||||
|             global.settings.set_strv(ENABLED_EXTENSIONS_KEY, enabledExtensions); | ||||
|         } | ||||
|  | ||||
|         loadExtension(dir, ExtensionUtils.ExtensionType.PER_USER, true); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function disableExtension(uuid) { | ||||
|     let extension = ExtensionUtils.extensions[uuid]; | ||||
| @@ -67,23 +178,23 @@ function disableExtension(uuid) { | ||||
|         try { | ||||
|             ExtensionUtils.extensions[uuid].stateObj.disable(); | ||||
|         } catch(e) { | ||||
|             logExtensionError(uuid, e); | ||||
|             logExtensionError(uuid, e.toString()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (extension.stylesheet) { | ||||
|         let theme = St.ThemeContext.get_for_stage(global.stage).get_theme(); | ||||
|         theme.unload_stylesheet(extension.stylesheet.get_path()); | ||||
|     try { | ||||
|         extension.stateObj.disable(); | ||||
|     } catch(e) { | ||||
|         logExtensionError(uuid, e.toString()); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     extension.stateObj.disable(); | ||||
|  | ||||
|     for (let i = 0; i < order.length; i++) { | ||||
|         let uuid = order[i]; | ||||
|         try { | ||||
|             ExtensionUtils.extensions[uuid].stateObj.enable(); | ||||
|         } catch(e) { | ||||
|             logExtensionError(uuid, e); | ||||
|             logExtensionError(uuid, e.toString()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -106,85 +217,67 @@ function enableExtension(uuid) { | ||||
|  | ||||
|     extensionOrder.push(uuid); | ||||
|  | ||||
|     let stylesheetNames = [global.session_mode + '.css', 'stylesheet.css']; | ||||
|     for (let i = 0; i < stylesheetNames.length; i++) { | ||||
|         let stylesheetFile = extension.dir.get_child(stylesheetNames[i]); | ||||
|         if (stylesheetFile.query_exists(null)) { | ||||
|             let theme = St.ThemeContext.get_for_stage(global.stage).get_theme(); | ||||
|             theme.load_stylesheet(stylesheetFile.get_path()); | ||||
|             extension.stylesheet = stylesheetFile; | ||||
|             break; | ||||
|         } | ||||
|     try { | ||||
|         extension.stateObj.enable(); | ||||
|     } catch(e) { | ||||
|         logExtensionError(uuid, e.toString()); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     extension.stateObj.enable(); | ||||
|  | ||||
|     extension.state = ExtensionState.ENABLED; | ||||
|     _signals.emit('extension-state-changed', extension); | ||||
| } | ||||
|  | ||||
| function logExtensionError(uuid, error) { | ||||
| function logExtensionError(uuid, message, state) { | ||||
|     let extension = ExtensionUtils.extensions[uuid]; | ||||
|     if (!extension) | ||||
|         return; | ||||
|  | ||||
|     let message = '' + error; | ||||
|  | ||||
|     extension.state = ExtensionState.ERROR; | ||||
|     if (!extension.errors) | ||||
|         extension.errors = []; | ||||
|     extension.errors.push(message); | ||||
|  | ||||
|     log('Extension "%s" had error: %s'.format(uuid, message)); | ||||
|     extension.errors.push(message); | ||||
|     global.logError('Extension "%s" had error: %s'.format(uuid, message)); | ||||
|     state = state || ExtensionState.ERROR; | ||||
|     _signals.emit('extension-state-changed', { uuid: uuid, | ||||
|                                                error: message, | ||||
|                                                state: extension.state }); | ||||
|                                                state: state }); | ||||
| } | ||||
|  | ||||
| function loadExtension(extension) { | ||||
| function loadExtension(dir, type, enabled) { | ||||
|     let uuid = dir.get_basename(); | ||||
|     let extension; | ||||
|  | ||||
|     if (ExtensionUtils.extensions[uuid] != undefined) { | ||||
|         throw new Error('extension already loaded'); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|         extension = ExtensionUtils.createExtensionObject(uuid, dir, type); | ||||
|     } catch(e) { | ||||
|         logExtensionError(uuid, e.message); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Default to error, we set success as the last step | ||||
|     extension.state = ExtensionState.ERROR; | ||||
|  | ||||
|     if (ExtensionUtils.isOutOfDate(extension)) { | ||||
|         logExtensionError(uuid, 'extension is not compatible with current GNOME Shell and/or GJS version', ExtensionState.OUT_OF_DATE); | ||||
|         extension.state = ExtensionState.OUT_OF_DATE; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (enabled) { | ||||
|         initExtension(uuid); | ||||
|         if (extension.state == ExtensionState.DISABLED) | ||||
|             enableExtension(uuid); | ||||
|     } else { | ||||
|         let enabled = enabledExtensions.indexOf(extension.uuid) != -1; | ||||
|         if (enabled) { | ||||
|             initExtension(extension.uuid); | ||||
|             if (extension.state == ExtensionState.DISABLED) | ||||
|                 enableExtension(extension.uuid); | ||||
|         } else { | ||||
|             extension.state = ExtensionState.INITIALIZED; | ||||
|         } | ||||
|         extension.state = ExtensionState.INITIALIZED; | ||||
|     } | ||||
|  | ||||
|     _signals.emit('extension-state-changed', extension); | ||||
| } | ||||
|  | ||||
| function unloadExtension(extension) { | ||||
|     // Try to disable it -- if it's ERROR'd, we can't guarantee that, | ||||
|     // but it will be removed on next reboot, and hopefully nothing | ||||
|     // broke too much. | ||||
|     disableExtension(extension.uuid); | ||||
|  | ||||
|     extension.state = ExtensionState.UNINSTALLED; | ||||
|     _signals.emit('extension-state-changed', extension); | ||||
|  | ||||
|     delete ExtensionUtils.extensions[extension.uuid]; | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| function reloadExtension(oldExtension) { | ||||
|     // Grab the things we'll need to pass to createExtensionObject | ||||
|     // to reload it. | ||||
|     let { uuid: uuid, dir: dir, type: type } = oldExtension; | ||||
|  | ||||
|     // Then unload the old extension. | ||||
|     unloadExtension(oldExtension); | ||||
|  | ||||
|     // Now, recreate the extension and load it. | ||||
|     let newExtension = ExtensionUtils.createExtensionObject(uuid, dir, type); | ||||
|     loadExtension(newExtension); | ||||
|     global.log('Loaded extension ' + uuid); | ||||
| } | ||||
|  | ||||
| function initExtension(uuid) { | ||||
| @@ -195,51 +288,76 @@ function initExtension(uuid) { | ||||
|         throw new Error("Extension was not properly created. Call loadExtension first"); | ||||
|  | ||||
|     let extensionJs = dir.get_child('extension.js'); | ||||
|     if (!extensionJs.query_exists(null)) | ||||
|         throw new Error('Missing extension.js'); | ||||
|     if (!extensionJs.query_exists(null)) { | ||||
|         logExtensionError(uuid, 'Missing extension.js'); | ||||
|         return; | ||||
|     } | ||||
|     let stylesheetPath = null; | ||||
|     let themeContext = St.ThemeContext.get_for_stage(global.stage); | ||||
|     let theme = themeContext.get_theme(); | ||||
|     let stylesheetFile = dir.get_child('stylesheet.css'); | ||||
|     if (stylesheetFile.query_exists(null)) { | ||||
|         try { | ||||
|             theme.load_stylesheet(stylesheetFile.get_path()); | ||||
|         } catch (e) { | ||||
|             logExtensionError(uuid, 'Stylesheet parse error: ' + e); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let extensionModule; | ||||
|     let extensionState = null; | ||||
|     try { | ||||
|         ExtensionUtils.installImporter(extension); | ||||
|         extensionModule = extension.imports.extension; | ||||
|     } catch (e) { | ||||
|         if (stylesheetPath != null) | ||||
|             theme.unload_stylesheet(stylesheetPath); | ||||
|         logExtensionError(uuid, '' + e); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     ExtensionUtils.installImporter(extension); | ||||
|     extensionModule = extension.imports.extension; | ||||
|     if (!extensionModule.init) { | ||||
|         logExtensionError(uuid, 'missing \'init\' function'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (extensionModule.init) { | ||||
|     try { | ||||
|         extensionState = extensionModule.init(extension); | ||||
|     } catch (e) { | ||||
|         if (stylesheetPath != null) | ||||
|             theme.unload_stylesheet(stylesheetPath); | ||||
|         logExtensionError(uuid, 'Failed to evaluate init function:' + e); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!extensionState) | ||||
|         extensionState = extensionModule; | ||||
|     extension.stateObj = extensionState; | ||||
|  | ||||
|     if (!extensionState.enable) { | ||||
|         logExtensionError(uuid, 'missing \'enable\' function'); | ||||
|         return; | ||||
|     } | ||||
|     if (!extensionState.disable) { | ||||
|         logExtensionError(uuid, 'missing \'disable\' function'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     extension.state = ExtensionState.DISABLED; | ||||
|  | ||||
|     _signals.emit('extension-loaded', uuid); | ||||
| } | ||||
|  | ||||
| function getEnabledExtensions() { | ||||
|     let extensions = global.settings.get_strv(ENABLED_EXTENSIONS_KEY); | ||||
|     if (!Array.isArray(Main.sessionMode.enabledExtensions)) | ||||
|         return extensions; | ||||
|  | ||||
|     return Main.sessionMode.enabledExtensions.concat(extensions); | ||||
| } | ||||
|  | ||||
| function onEnabledExtensionsChanged() { | ||||
|     let newEnabledExtensions = getEnabledExtensions(); | ||||
|  | ||||
|     if (!enabled) | ||||
|         return; | ||||
|     let newEnabledExtensions = global.settings.get_strv(ENABLED_EXTENSIONS_KEY); | ||||
|  | ||||
|     // Find and enable all the newly enabled extensions: UUIDs found in the | ||||
|     // new setting, but not in the old one. | ||||
|     newEnabledExtensions.filter(function(uuid) { | ||||
|         return enabledExtensions.indexOf(uuid) == -1; | ||||
|     }).forEach(function(uuid) { | ||||
|         try { | ||||
|             enableExtension(uuid); | ||||
|         } catch(e) { | ||||
|             logExtensionError(uuid, e); | ||||
|         } | ||||
|         enableExtension(uuid); | ||||
|     }); | ||||
|  | ||||
|     // Find and disable all the newly disabled extensions: UUIDs found in the | ||||
| @@ -247,74 +365,87 @@ function onEnabledExtensionsChanged() { | ||||
|     enabledExtensions.filter(function(item) { | ||||
|         return newEnabledExtensions.indexOf(item) == -1; | ||||
|     }).forEach(function(uuid) { | ||||
|         try { | ||||
|             disableExtension(uuid); | ||||
|         } catch(e) { | ||||
|             logExtensionError(uuid, e); | ||||
|         } | ||||
|         disableExtension(uuid); | ||||
|     }); | ||||
|  | ||||
|     enabledExtensions = newEnabledExtensions; | ||||
| } | ||||
|  | ||||
| function _loadExtensions() { | ||||
|     global.settings.connect('changed::' + ENABLED_EXTENSIONS_KEY, onEnabledExtensionsChanged); | ||||
|     enabledExtensions = getEnabledExtensions(); | ||||
|  | ||||
|     let finder = new ExtensionUtils.ExtensionFinder(); | ||||
|     finder.connect('extension-found', function(signals, extension) { | ||||
|         try { | ||||
|             loadExtension(extension); | ||||
|         } catch(e) { | ||||
|             logExtensionError(extension.uuid, e); | ||||
|         } | ||||
|     }); | ||||
|     finder.scanExtensions(); | ||||
| } | ||||
|  | ||||
| function enableAllExtensions() { | ||||
|     if (enabled) | ||||
|         return; | ||||
|  | ||||
|     if (!initted) { | ||||
|         _loadExtensions(); | ||||
|         initted = true; | ||||
|     } else { | ||||
|         enabledExtensions.forEach(function(uuid) { | ||||
|             enableExtension(uuid); | ||||
|         }); | ||||
|     } | ||||
|     enabled = true; | ||||
| } | ||||
|  | ||||
| function disableAllExtensions() { | ||||
|     if (!enabled) | ||||
|         return; | ||||
|  | ||||
|     if (initted) { | ||||
|         enabledExtensions.forEach(function(uuid) { | ||||
|             disableExtension(uuid); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     enabled = false; | ||||
| } | ||||
|  | ||||
| function _sessionUpdated() { | ||||
|     // For now sessionMode.allowExtensions controls extensions from both the | ||||
|     // 'enabled-extensions' preference and the sessionMode.enabledExtensions | ||||
|     // property; it might make sense to make enabledExtensions independent | ||||
|     // from allowExtensions in the future | ||||
|     if (Main.sessionMode.allowExtensions) { | ||||
|         if (initted) | ||||
|             onEnabledExtensionsChanged(); | ||||
|         enableAllExtensions(); | ||||
|     } else { | ||||
|         disableAllExtensions(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function init() { | ||||
|     Main.sessionMode.connect('updated', _sessionUpdated); | ||||
|     _sessionUpdated(); | ||||
|     ExtensionUtils.init(); | ||||
|  | ||||
|     global.settings.connect('changed::' + ENABLED_EXTENSIONS_KEY, onEnabledExtensionsChanged); | ||||
|     enabledExtensions = global.settings.get_strv(ENABLED_EXTENSIONS_KEY); | ||||
| } | ||||
|  | ||||
| function loadExtensions() { | ||||
|     ExtensionUtils.scanExtensions(function(uuid, dir, type) { | ||||
|         let enabled = enabledExtensions.indexOf(uuid) != -1; | ||||
|         loadExtension(dir, type, enabled); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| const InstallExtensionDialog = new Lang.Class({ | ||||
|     Name: 'InstallExtensionDialog', | ||||
|     Extends: ModalDialog.ModalDialog, | ||||
|  | ||||
|     _init: function(uuid, version_tag, name) { | ||||
|         this.parent({ styleClass: 'extension-dialog' }); | ||||
|  | ||||
|         this._uuid = uuid; | ||||
|         this._version_tag = version_tag; | ||||
|         this._name = name; | ||||
|  | ||||
|         this.setButtons([{ label: _("Cancel"), | ||||
|                            action: Lang.bind(this, this._onCancelButtonPressed), | ||||
|                            key:    Clutter.Escape | ||||
|                          }, | ||||
|                          { label:  _("Install"), | ||||
|                            action: Lang.bind(this, this._onInstallButtonPressed) | ||||
|                          }]); | ||||
|  | ||||
|         let message = _("Download and install '%s' from extensions.gnome.org?").format(name); | ||||
|  | ||||
|         this._descriptionLabel = new St.Label({ text: message }); | ||||
|  | ||||
|         this.contentLayout.add(this._descriptionLabel, | ||||
|                                { y_fill:  true, | ||||
|                                  y_align: St.Align.START }); | ||||
|     }, | ||||
|  | ||||
|     _onCancelButtonPressed: function(button, event) { | ||||
|         this.close(global.get_current_time()); | ||||
|  | ||||
|         // Even though the extension is already "uninstalled", send through | ||||
|         // a state-changed signal for any users who want to know if the install | ||||
|         // went through correctly -- using proper async DBus would block more | ||||
|         // traditional clients like the plugin | ||||
|         let meta = { uuid: this._uuid, | ||||
|                      state: ExtensionState.UNINSTALLED, | ||||
|                      error: '' }; | ||||
|  | ||||
|         _signals.emit('extension-state-changed', meta); | ||||
|     }, | ||||
|  | ||||
|     _onInstallButtonPressed: function(button, event) { | ||||
|         let state = { uuid: this._uuid, | ||||
|                       state: ExtensionState.DOWNLOADING, | ||||
|                       error: '' }; | ||||
|  | ||||
|         _signals.emit('extension-state-changed', state); | ||||
|  | ||||
|         let params = { version_tag: this._version_tag, | ||||
|                        shell_version: Config.PACKAGE_VERSION, | ||||
|                        api_version: API_VERSION.toString() }; | ||||
|  | ||||
|         let url = REPOSITORY_URL_DOWNLOAD.format(this._uuid); | ||||
|         let message = Soup.form_request_new_from_hash('GET', url, params); | ||||
|  | ||||
|         _httpSession.queue_message(message, | ||||
|                                    Lang.bind(this, function(session, message) { | ||||
|                                        gotExtensionZipFile(session, message, this._uuid); | ||||
|                                    })); | ||||
|  | ||||
|         this.close(global.get_current_time()); | ||||
|     } | ||||
| }); | ||||
|   | ||||
							
								
								
									
										45
									
								
								js/ui/flashspot.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,45 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
|  | ||||
| const Lightbox = imports.ui.lightbox; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const FLASHSPOT_ANIMATION_TIME = 0.25; // seconds | ||||
|  | ||||
| const Flashspot = new Lang.Class({ | ||||
|     Name: 'Flashspot', | ||||
|     Extends: Lightbox.Lightbox, | ||||
|  | ||||
|     _init: function(area) { | ||||
|         this.parent(Main.uiGroup, { inhibitEvents: true, | ||||
|                                     width: area.width, | ||||
|                                     height: area.height }); | ||||
|  | ||||
|         this.actor.style_class = 'flashspot'; | ||||
|         this.actor.set_position(area.x, area.y); | ||||
|     }, | ||||
|  | ||||
|     fire: function() { | ||||
|         this.actor.opacity = 0; | ||||
|         Tweener.addTween(this.actor, | ||||
|                          { opacity: 255, | ||||
|                            time: FLASHSPOT_ANIMATION_TIME, | ||||
|                            transition: 'linear', | ||||
|                            onComplete: Lang.bind(this, this._onFireShowComplete) | ||||
|                          }); | ||||
|         this.actor.show(); | ||||
|     }, | ||||
|  | ||||
|     _onFireShowComplete: function() { | ||||
|         Tweener.addTween(this.actor, | ||||
|                          { opacity: 0, | ||||
|                            time: FLASHSPOT_ANIMATION_TIME, | ||||
|                            transition: 'linear', | ||||
|                            onComplete: Lang.bind(this, function() { | ||||
|                                this.destroy(); | ||||
|                            }) | ||||
|                          }); | ||||
|     } | ||||
| }); | ||||
| @@ -1,375 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| // GrabHelper: | ||||
| // @owner: the actor that owns the GrabHelper | ||||
| // @params: optional parameters to pass to Main.pushModal() | ||||
| // | ||||
| // Creates a new GrabHelper object, for dealing with keyboard and pointer grabs | ||||
| // associated with a set of actors. | ||||
| // | ||||
| // Note that the grab can be automatically dropped at any time by the user, and | ||||
| // your code just needs to deal with it; you shouldn't adjust behavior directly | ||||
| // after you call ungrab(), but instead pass an 'onUngrab' callback when you | ||||
| // call grab(). | ||||
| const GrabHelper = new Lang.Class({ | ||||
|     Name: 'GrabHelper', | ||||
|  | ||||
|     _init: function(owner, params) { | ||||
|         this._owner = owner; | ||||
|         this._modalParams = params; | ||||
|  | ||||
|         this._grabStack = []; | ||||
|  | ||||
|         this._actors = []; | ||||
|         this._capturedEventId = 0; | ||||
|         this._keyFocusNotifyId = 0; | ||||
|         this._focusWindowChangedId = 0; | ||||
|         this._ignoreRelease = false; | ||||
|         this._isUngrabbingCount = 0; | ||||
|  | ||||
|         this._modalCount = 0; | ||||
|         this._grabFocusCount = 0; | ||||
|     }, | ||||
|  | ||||
|     // addActor: | ||||
|     // @actor: an actor | ||||
|     // | ||||
|     // Adds @actor to the set of actors that are allowed to process events | ||||
|     // during a grab. | ||||
|     addActor: function(actor) { | ||||
|         actor.__grabHelperDestroyId = actor.connect('destroy', Lang.bind(this, function() { this.removeActor(actor); })); | ||||
|         this._actors.push(actor); | ||||
|     }, | ||||
|  | ||||
|     // removeActor: | ||||
|     // @actor: an actor | ||||
|     // | ||||
|     // Removes @actor from the set of actors that are allowed to | ||||
|     // process events during a grab. | ||||
|     removeActor: function(actor) { | ||||
|         let index = this._actors.indexOf(actor); | ||||
|         if (index != -1) | ||||
|             this._actors.splice(index, 1); | ||||
|         if (actor.__grabHelperDestroyId) { | ||||
|             actor.disconnect(actor.__grabHelperDestroyId); | ||||
|             delete actor.__grabHelperDestroyId; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _isWithinGrabbedActor: function(actor) { | ||||
|         let currentActor = this.currentGrab.actor; | ||||
|         while (actor) { | ||||
|             if (this._actors.indexOf(actor) != -1) | ||||
|                 return true; | ||||
|             if (actor == currentActor) | ||||
|                 return true; | ||||
|             actor = actor.get_parent(); | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     get currentGrab() { | ||||
|         return this._grabStack[this._grabStack.length - 1] || {}; | ||||
|     }, | ||||
|  | ||||
|     get grabbed() { | ||||
|         return this._grabStack.length > 0; | ||||
|     }, | ||||
|  | ||||
|     get grabStack() { | ||||
|         return this._grabStack; | ||||
|     }, | ||||
|  | ||||
|     _findStackIndex: function(actor) { | ||||
|         if (!actor) | ||||
|             return -1; | ||||
|  | ||||
|         for (let i = 0; i < this._grabStack.length; i++) { | ||||
|             if (this._grabStack[i].actor === actor) | ||||
|                 return i; | ||||
|         } | ||||
|         return -1; | ||||
|     }, | ||||
|  | ||||
|     _actorInGrabStack: function(actor) { | ||||
|         while (actor) { | ||||
|             let idx = this._findStackIndex(actor); | ||||
|             if (idx >= 0) | ||||
|                 return idx; | ||||
|             actor = actor.get_parent(); | ||||
|         } | ||||
|         return -1; | ||||
|     }, | ||||
|  | ||||
|     isActorGrabbed: function(actor) { | ||||
|         return this._findStackIndex(actor) >= 0; | ||||
|     }, | ||||
|  | ||||
|     // grab: | ||||
|     // @params: A bunch of parameters, see below | ||||
|     // | ||||
|     // Grabs the mouse and keyboard, according to the GrabHelper's | ||||
|     // parameters. If @newFocus is not %null, then the keyboard focus | ||||
|     // is moved to the first #StWidget:can-focus widget inside it. | ||||
|     // | ||||
|     // The grab will automatically be dropped if: | ||||
|     //   - The user clicks outside the grabbed actors | ||||
|     //   - The user types Escape | ||||
|     //   - The keyboard focus is moved outside the grabbed actors | ||||
|     //   - A window is focused | ||||
|     // | ||||
|     // If @params.actor is not null, then it will be focused as the | ||||
|     // new actor. If you attempt to grab an already focused actor, the | ||||
|     // request to be focused will be ignored. The actor will not be | ||||
|     // added to the grab stack, so do not call a paired ungrab(). | ||||
|     // | ||||
|     // If @params contains { modal: true }, then grab() will push a modal | ||||
|     // on the owner of the GrabHelper. As long as there is at least one | ||||
|     // { modal: true } actor on the grab stack, the grab will be kept. | ||||
|     // When the last { modal: true } actor is ungrabbed, then the modal | ||||
|     // will be dropped. A modal grab can fail if there is already a grab | ||||
|     // in effect from aother application; in this case the function returns | ||||
|     // false and nothing happens. Non-modal grabs can never fail. | ||||
|     // | ||||
|     // If @params contains { grabFocus: true }, then if you call grab() | ||||
|     // while the shell is outside the overview, it will set the stage | ||||
|     // input mode to %Shell.StageInputMode.FOCUSED, and ungrab() will | ||||
|     // revert it back, and re-focus the previously-focused window (if | ||||
|     // another window hasn't been explicitly focused before then). | ||||
|     grab: function(params) { | ||||
|         params = Params.parse(params, { actor: null, | ||||
|                                         modal: false, | ||||
|                                         grabFocus: false, | ||||
|                                         focus: null, | ||||
|                                         onUngrab: null }); | ||||
|  | ||||
|         let focus = global.stage.key_focus; | ||||
|         let hadFocus = focus && this._isWithinGrabbedActor(focus); | ||||
|         let newFocus = params.actor; | ||||
|  | ||||
|         if (this.isActorGrabbed(params.actor)) | ||||
|             return true; | ||||
|  | ||||
|         params.savedFocus = focus; | ||||
|  | ||||
|         if (params.modal && !this._takeModalGrab()) | ||||
|             return false; | ||||
|  | ||||
|         if (params.grabFocus && !this._takeFocusGrab(hadFocus)) | ||||
|             return false; | ||||
|  | ||||
|         this._grabStack.push(params); | ||||
|  | ||||
|         if (params.focus) { | ||||
|             params.focus.grab_key_focus(); | ||||
|         } else if (newFocus && (hadFocus || params.grabFocus)) { | ||||
|             if (!newFocus.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false)) | ||||
|                 newFocus.grab_key_focus(); | ||||
|         } | ||||
|  | ||||
|         if ((params.grabFocus || params.modal) && !this._capturedEventId) | ||||
|             this._capturedEventId = global.stage.connect('captured-event', Lang.bind(this, this._onCapturedEvent)); | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _takeModalGrab: function() { | ||||
|         let firstGrab = (this._modalCount == 0); | ||||
|         if (firstGrab) { | ||||
|             if (!Main.pushModal(this._owner, this._modalParams)) | ||||
|                 return false; | ||||
|         } | ||||
|  | ||||
|         this._modalCount++; | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _releaseModalGrab: function() { | ||||
|         this._modalCount--; | ||||
|         if (this._modalCount > 0) | ||||
|             return; | ||||
|  | ||||
|         Main.popModal(this._owner); | ||||
|         global.sync_pointer(); | ||||
|     }, | ||||
|  | ||||
|     _takeFocusGrab: function(hadFocus) { | ||||
|         let firstGrab = (this._grabFocusCount == 0); | ||||
|         if (firstGrab) { | ||||
|             let metaDisplay = global.screen.get_display(); | ||||
|  | ||||
|             this._grabbedFromKeynav = hadFocus; | ||||
|             this._preGrabInputMode = global.stage_input_mode; | ||||
|  | ||||
|             if (this._preGrabInputMode == Shell.StageInputMode.NONREACTIVE || | ||||
|                 this._preGrabInputMode == Shell.StageInputMode.NORMAL) { | ||||
|                 global.set_stage_input_mode(Shell.StageInputMode.FOCUSED); | ||||
|             } | ||||
|  | ||||
|             this._keyFocusNotifyId = global.stage.connect('notify::key-focus', Lang.bind(this, this._onKeyFocusChanged)); | ||||
|             this._focusWindowChangedId = metaDisplay.connect('notify::focus-window', Lang.bind(this, this._focusWindowChanged)); | ||||
|         } | ||||
|  | ||||
|         this._grabFocusCount++; | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _releaseFocusGrab: function() { | ||||
|         this._grabFocusCount--; | ||||
|         if (this._grabFocusCount > 0) | ||||
|             return; | ||||
|  | ||||
|         if (this._keyFocusNotifyId > 0) { | ||||
|             global.stage.disconnect(this._keyFocusNotifyId); | ||||
|             this._keyFocusNotifyId = 0; | ||||
|         } | ||||
|  | ||||
|         if (this._focusWindowChangedId > 0) { | ||||
|             let metaDisplay = global.screen.get_display(); | ||||
|             metaDisplay.disconnect(this._focusWindowChangedId); | ||||
|             this._focusWindowChangedId = 0; | ||||
|         } | ||||
|  | ||||
|         let prePopInputMode = global.stage_input_mode; | ||||
|  | ||||
|         if (this._grabbedFromKeynav) { | ||||
|             if (this._preGrabInputMode == Shell.StageInputMode.FOCUSED && | ||||
|                 prePopInputMode != Shell.StageInputMode.FULLSCREEN) | ||||
|                 global.set_stage_input_mode(Shell.StageInputMode.FOCUSED); | ||||
|         } | ||||
|  | ||||
|         global.screen.focus_default_window(global.get_current_time()); | ||||
|     }, | ||||
|  | ||||
|     // ignoreRelease: | ||||
|     // | ||||
|     // Make sure that the next button release event evaluated by the | ||||
|     // capture event handler returns false. This is designed for things | ||||
|     // like the ComboBoxMenu that go away on press, but need to eat | ||||
|     // the next release event. | ||||
|     ignoreRelease: function() { | ||||
|         this._ignoreRelease = true; | ||||
|     }, | ||||
|  | ||||
|     // ungrab: | ||||
|     // @params: The parameters for the grab; see below. | ||||
|     // | ||||
|     // Pops an actor from the grab stack, potentially dropping the grab. | ||||
|     // | ||||
|     // If the actor that was popped from the grab stack was not the actor | ||||
|     // That was passed in, this call is ignored. | ||||
|     ungrab: function(params) { | ||||
|         params = Params.parse(params, { actor: this.currentGrab.actor, | ||||
|                                         isUser: false }); | ||||
|  | ||||
|         let grabStackIndex = this._findStackIndex(params.actor); | ||||
|         if (grabStackIndex < 0) | ||||
|             return; | ||||
|  | ||||
|         // We may get key focus changes when calling onUngrab, which | ||||
|         // would cause an extra ungrab() on the next actor in the | ||||
|         // stack, which is wrong. Ignore key focus changes during the | ||||
|         // ungrab, and restore the saved key focus ourselves afterwards. | ||||
|         // We use a count as ungrab() may be re-entrant, as onUngrab() | ||||
|         // may ungrab additional actors. | ||||
|         this._isUngrabbingCount++; | ||||
|  | ||||
|         let focus = global.stage.key_focus; | ||||
|         let hadFocus = focus && this._isWithinGrabbedActor(focus); | ||||
|  | ||||
|         let poppedGrabs = this._grabStack.slice(grabStackIndex); | ||||
|         // "Pop" all newly ungrabbed actors off the grab stack | ||||
|         // by truncating the array. | ||||
|         this._grabStack.length = grabStackIndex; | ||||
|  | ||||
|         for (let i = poppedGrabs.length - 1; i >= 0; i--) { | ||||
|             let poppedGrab = poppedGrabs[i]; | ||||
|  | ||||
|             if (poppedGrab.onUngrab) | ||||
|                 poppedGrab.onUngrab(params.isUser); | ||||
|  | ||||
|             if (poppedGrab.modal) | ||||
|                 this._releaseModalGrab(); | ||||
|  | ||||
|             if (poppedGrab.grabFocus) | ||||
|                 this._releaseFocusGrab(); | ||||
|         } | ||||
|  | ||||
|         if (!this.grabbed && this._capturedEventId > 0) { | ||||
|             global.stage.disconnect(this._capturedEventId); | ||||
|             this._capturedEventId = 0; | ||||
|         } | ||||
|  | ||||
|         if (hadFocus) { | ||||
|             let poppedGrab = poppedGrabs[0]; | ||||
|             if (poppedGrab.savedFocus) | ||||
|                 poppedGrab.savedFocus.grab_key_focus(); | ||||
|         } | ||||
|  | ||||
|         this._isUngrabbingCount--; | ||||
|     }, | ||||
|  | ||||
|     _onCapturedEvent: function(actor, event) { | ||||
|         let type = event.type(); | ||||
|  | ||||
|         if (type == Clutter.EventType.KEY_PRESS && | ||||
|             event.get_key_symbol() == Clutter.KEY_Escape) { | ||||
|             this.ungrab({ isUser: true }); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         let press = type == Clutter.EventType.BUTTON_PRESS; | ||||
|         let release = type == Clutter.EventType.BUTTON_RELEASE; | ||||
|         let button = press || release; | ||||
|  | ||||
|         if (release && this._ignoreRelease) { | ||||
|             this._ignoreRelease = false; | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         if (!button && this._modalCount == 0) | ||||
|             return false; | ||||
|  | ||||
|         if (this._isWithinGrabbedActor(event.get_source())) | ||||
|             return false; | ||||
|  | ||||
|         if (Main.keyboard.shouldTakeEvent(event)) | ||||
|             return false; | ||||
|  | ||||
|         if (button) { | ||||
|             // If we have a press event, ignore the next event, | ||||
|             // which should be a release event. | ||||
|             if (press) | ||||
|                 this._ignoreRelease = true; | ||||
|             let i = this._actorInGrabStack(event.get_source()) + 1; | ||||
|             this.ungrab({ actor: this._grabStack[i].actor, isUser: true }); | ||||
|         } | ||||
|  | ||||
|         return this._modalCount > 0; | ||||
|     }, | ||||
|  | ||||
|     _onKeyFocusChanged: function() { | ||||
|         if (this._isUngrabbingCount > 0) | ||||
|             return; | ||||
|  | ||||
|         let focus = global.stage.key_focus; | ||||
|         if (!focus || !this._isWithinGrabbedActor(focus)) | ||||
|             this.ungrab({ isUser: true }); | ||||
|     }, | ||||
|  | ||||
|     _focusWindowChanged: function() { | ||||
|         let metaDisplay = global.screen.get_display(); | ||||
|         if (metaDisplay.focus_window != null) | ||||
|             this.ungrab({ isUser: true }); | ||||
|     } | ||||
| }); | ||||
| @@ -1,269 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const IBus = imports.gi.IBus; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const BoxPointer = imports.ui.boxpointer; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const MAX_CANDIDATES_PER_PAGE = 16; | ||||
|  | ||||
| const CandidateArea = new Lang.Class({ | ||||
|     Name: 'CandidateArea', | ||||
|  | ||||
|     _init: function() { | ||||
|         this.actor = new St.BoxLayout({ vertical: true, | ||||
|                                         visible: false }); | ||||
|         this._candidateBoxes = []; | ||||
|         for (let i = 0; i < MAX_CANDIDATES_PER_PAGE; ++i) { | ||||
|             let box = new St.BoxLayout({ style_class: 'candidate-box', | ||||
|                                          reactive: true, | ||||
|                                          track_hover: true }); | ||||
|             box._indexLabel = new St.Label({ style_class: 'candidate-index' }); | ||||
|             box._candidateLabel = new St.Label({ style_class: 'candidate-label' }); | ||||
|             box.add(box._indexLabel, { y_fill: false }); | ||||
|             box.add(box._candidateLabel, { y_fill: false }); | ||||
|             this._candidateBoxes.push(box); | ||||
|             this.actor.add(box); | ||||
|  | ||||
|             let j = i; | ||||
|             box.connect('button-release-event', Lang.bind(this, function(actor, event) { | ||||
|                 this.emit('candidate-clicked', j, event.get_button(), event.get_state()); | ||||
|             })); | ||||
|         } | ||||
|  | ||||
|         this._buttonBox = new St.BoxLayout({ style_class: 'candidate-page-button-box' }); | ||||
|  | ||||
|         this._previousButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-previous' }); | ||||
|         this._previousButton.child = new St.Icon({ style_class: 'candidate-page-button-icon' }); | ||||
|         this._buttonBox.add(this._previousButton, { expand: true }); | ||||
|  | ||||
|         this._nextButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-next' }); | ||||
|         this._nextButton.child = new St.Icon({ style_class: 'candidate-page-button-icon' }); | ||||
|         this._buttonBox.add(this._nextButton, { expand: true }); | ||||
|  | ||||
|         this.actor.add(this._buttonBox); | ||||
|  | ||||
|         this._previousButton.connect('clicked', Lang.bind(this, function() { | ||||
|             this.emit('previous-page'); | ||||
|         })); | ||||
|         this._nextButton.connect('clicked', Lang.bind(this, function() { | ||||
|             this.emit('next-page'); | ||||
|         })); | ||||
|  | ||||
|         this._orientation = -1; | ||||
|         this._cursorPosition = 0; | ||||
|     }, | ||||
|  | ||||
|     setOrientation: function(orientation) { | ||||
|         if (this._orientation == orientation) | ||||
|             return; | ||||
|  | ||||
|         this._orientation = orientation; | ||||
|  | ||||
|         if (this._orientation == IBus.Orientation.HORIZONTAL) { | ||||
|             this.actor.vertical = false; | ||||
|             this.actor.remove_style_class_name('vertical'); | ||||
|             this.actor.add_style_class_name('horizontal'); | ||||
|             this._previousButton.child.icon_name = 'go-previous-symbolic'; | ||||
|             this._nextButton.child.icon_name = 'go-next-symbolic'; | ||||
|         } else {                // VERTICAL || SYSTEM | ||||
|             this.actor.vertical = true; | ||||
|             this.actor.add_style_class_name('vertical'); | ||||
|             this.actor.remove_style_class_name('horizontal'); | ||||
|             this._previousButton.child.icon_name = 'go-up-symbolic'; | ||||
|             this._nextButton.child.icon_name = 'go-down-symbolic'; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     setCandidates: function(indexes, candidates, cursorPosition, cursorVisible) { | ||||
|         for (let i = 0; i < MAX_CANDIDATES_PER_PAGE; ++i) { | ||||
|             let visible = i < candidates.length; | ||||
|             let box = this._candidateBoxes[i]; | ||||
|             box.visible = visible; | ||||
|  | ||||
|             if (!visible) | ||||
|                 continue; | ||||
|  | ||||
|             box._indexLabel.text = ((indexes && indexes[i]) ? indexes[i] : '%x'.format(i + 1)); | ||||
|             box._candidateLabel.text = candidates[i]; | ||||
|         } | ||||
|  | ||||
|         this._candidateBoxes[this._cursorPosition].remove_style_pseudo_class('selected'); | ||||
|         this._cursorPosition = cursorPosition; | ||||
|         if (cursorVisible) | ||||
|             this._candidateBoxes[cursorPosition].add_style_pseudo_class('selected'); | ||||
|     }, | ||||
|  | ||||
|     updateButtons: function(wrapsAround, page, nPages) { | ||||
|         if (nPages < 2) { | ||||
|             this._buttonBox.hide(); | ||||
|             return; | ||||
|         } | ||||
|         this._buttonBox.show(); | ||||
|         this._previousButton.reactive = wrapsAround || page > 0; | ||||
|         this._nextButton.reactive = wrapsAround || page < nPages - 1; | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(CandidateArea.prototype); | ||||
|  | ||||
| const CandidatePopup = new Lang.Class({ | ||||
|     Name: 'CandidatePopup', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._cursor = new St.Bin({ opacity: 0 }); | ||||
|         Main.uiGroup.add_actor(this._cursor); | ||||
|  | ||||
|         this._boxPointer = new BoxPointer.BoxPointer(St.Side.TOP); | ||||
|         this._boxPointer.actor.visible = false; | ||||
|         this._boxPointer.actor.style_class = 'candidate-popup-boxpointer'; | ||||
|         Main.layoutManager.addChrome(this._boxPointer.actor); | ||||
|  | ||||
|         let box = new St.BoxLayout({ style_class: 'candidate-popup-content', | ||||
|                                      vertical: true }); | ||||
|         this._boxPointer.bin.set_child(box); | ||||
|  | ||||
|         this._preeditText = new St.Label({ style_class: 'candidate-popup-text', | ||||
|                                            visible: false }); | ||||
|         box.add(this._preeditText); | ||||
|  | ||||
|         this._auxText = new St.Label({ style_class: 'candidate-popup-text', | ||||
|                                        visible: false }); | ||||
|         box.add(this._auxText); | ||||
|  | ||||
|         this._candidateArea = new CandidateArea(); | ||||
|         box.add(this._candidateArea.actor); | ||||
|  | ||||
|         this._candidateArea.connect('previous-page', Lang.bind(this, function() { | ||||
|             this._panelService.page_up(); | ||||
|         })); | ||||
|         this._candidateArea.connect('next-page', Lang.bind(this, function() { | ||||
|             this._panelService.page_down(); | ||||
|         })); | ||||
|         this._candidateArea.connect('candidate-clicked', Lang.bind(this, function(ca, index, button, state) { | ||||
|             this._panelService.candidate_clicked(index, button, state); | ||||
|         })); | ||||
|  | ||||
|         this._panelService = null; | ||||
|     }, | ||||
|  | ||||
|     setPanelService: function(panelService) { | ||||
|         this._panelService = panelService; | ||||
|         if (!panelService) | ||||
|             return; | ||||
|  | ||||
|         panelService.connect('set-cursor-location', | ||||
|                              Lang.bind(this, function(ps, x, y, w, h) { | ||||
|                                  this._cursor.set_position(x, y); | ||||
|                                  this._cursor.set_size(w, h); | ||||
|                                  if (this._boxPointer.actor.visible) | ||||
|                                      this._boxPointer.setPosition(this._cursor, 0); | ||||
|                              })); | ||||
|         panelService.connect('update-preedit-text', | ||||
|                              Lang.bind(this, function(ps, text, cursorPosition, visible) { | ||||
|                                  this._preeditText.visible = visible; | ||||
|                                  this._updateVisibility(); | ||||
|  | ||||
|                                  this._preeditText.text = text.get_text(); | ||||
|  | ||||
|                                  let attrs = text.get_attributes(); | ||||
|                                  if (attrs) | ||||
|                                      this._setTextAttributes(this._preeditText.clutter_text, | ||||
|                                                              attrs); | ||||
|                              })); | ||||
|         panelService.connect('show-preedit-text', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._preeditText.show(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('hide-preedit-text', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._preeditText.hide(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('update-auxiliary-text', | ||||
|                              Lang.bind(this, function(ps, text, visible) { | ||||
|                                  this._auxText.visible = visible; | ||||
|                                  this._updateVisibility(); | ||||
|  | ||||
|                                  this._auxText.text = text.get_text(); | ||||
|                              })); | ||||
|         panelService.connect('show-auxiliary-text', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._auxText.show(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('hide-auxiliary-text', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._auxText.hide(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('update-lookup-table', | ||||
|                              Lang.bind(this, function(ps, lookupTable, visible) { | ||||
|                                  this._candidateArea.actor.visible = visible; | ||||
|                                  this._updateVisibility(); | ||||
|  | ||||
|                                  let nCandidates = lookupTable.get_number_of_candidates(); | ||||
|                                  let cursorPos = lookupTable.get_cursor_pos(); | ||||
|                                  let pageSize = lookupTable.get_page_size(); | ||||
|                                  let nPages = Math.ceil(nCandidates / pageSize); | ||||
|                                  let page = ((cursorPos == 0) ? 0 : Math.floor(cursorPos / pageSize)); | ||||
|                                  let startIndex = page * pageSize; | ||||
|                                  let endIndex = Math.min((page + 1) * pageSize, nCandidates); | ||||
|  | ||||
|                                  let indexes = []; | ||||
|                                  let indexLabel; | ||||
|                                  for (let i = 0; indexLabel = lookupTable.get_label(i); ++i) | ||||
|                                       indexes.push(indexLabel.get_text()); | ||||
|  | ||||
|                                  let candidates = []; | ||||
|                                  for (let i = startIndex; i < endIndex; ++i) | ||||
|                                      candidates.push(lookupTable.get_candidate(i).get_text()); | ||||
|  | ||||
|                                  this._candidateArea.setCandidates(indexes, | ||||
|                                                                    candidates, | ||||
|                                                                    cursorPos % pageSize, | ||||
|                                                                    lookupTable.is_cursor_visible()); | ||||
|                                  this._candidateArea.setOrientation(lookupTable.get_orientation()); | ||||
|                                  this._candidateArea.updateButtons(lookupTable.is_round(), page, nPages); | ||||
|                              })); | ||||
|         panelService.connect('show-lookup-table', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._candidateArea.actor.show(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('hide-lookup-table', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._candidateArea.actor.hide(); | ||||
|                                  this._updateVisibility(); | ||||
|                              })); | ||||
|         panelService.connect('focus-out', | ||||
|                              Lang.bind(this, function(ps) { | ||||
|                                  this._boxPointer.hide(BoxPointer.PopupAnimation.NONE); | ||||
|                              })); | ||||
|     }, | ||||
|  | ||||
|     _updateVisibility: function() { | ||||
|         let isVisible = (this._preeditText.visible || | ||||
|                          this._auxText.visible || | ||||
|                          this._candidateArea.actor.visible); | ||||
|  | ||||
|         if (isVisible) { | ||||
|             this._boxPointer.setPosition(this._cursor, 0); | ||||
|             this._boxPointer.show(BoxPointer.PopupAnimation.NONE); | ||||
|             this._boxPointer.actor.raise_top(); | ||||
|         } else { | ||||
|             this._boxPointer.hide(BoxPointer.PopupAnimation.NONE); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _setTextAttributes: function(clutterText, ibusAttrList) { | ||||
|         let attr; | ||||
|         for (let i = 0; attr = ibusAttrList.get(i); ++i) | ||||
|             if (attr.get_attr_type() == IBus.AttrType.BACKGROUND) | ||||
|                 clutterText.set_selection(attr.get_start_index(), attr.get_end_index()); | ||||
|     } | ||||
| }); | ||||
| @@ -23,8 +23,6 @@ const BaseIcon = new Lang.Class({ | ||||
|         this.actor._delegate = this; | ||||
|         this.actor.connect('style-changed', | ||||
|                            Lang.bind(this, this._onStyleChanged)); | ||||
|         this.actor.connect('destroy', | ||||
|                            Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         this._spacing = 0; | ||||
|  | ||||
| @@ -54,9 +52,6 @@ const BaseIcon = new Lang.Class({ | ||||
|         this._setSizeManually = params.setSizeManually; | ||||
|  | ||||
|         this.icon = null; | ||||
|  | ||||
|         let cache = St.TextureCache.get_default(); | ||||
|         this._iconThemeChangedId = cache.connect('icon-theme-changed', Lang.bind(this, this._onIconThemeChanged)); | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
| @@ -151,22 +146,12 @@ const BaseIcon = new Lang.Class({ | ||||
|             size = found ? len : ICON_SIZE; | ||||
|         } | ||||
|  | ||||
|         if (this.iconSize == size && this._iconBin.child) | ||||
|         // don't create icons unnecessarily | ||||
|         if (size == this.iconSize && | ||||
|             this._iconBin.child) | ||||
|             return; | ||||
|  | ||||
|         this._createIconTexture(size); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         if (this._iconThemeChangedId > 0) { | ||||
|             let cache = St.TextureCache.get_default(); | ||||
|             cache.disconnect(this._iconThemeChangedId); | ||||
|             this._iconThemeChangedId = 0; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onIconThemeChanged: function() { | ||||
|         this._createIconTexture(this.iconSize); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -218,11 +203,7 @@ const IconGrid = new Lang.Class({ | ||||
|  | ||||
|     _getPreferredHeight: function (grid, forWidth, alloc) { | ||||
|         let children = this._getVisibleChildren(); | ||||
|         let nColumns; | ||||
|         if (forWidth < 0) | ||||
|             nColumns = children.length; | ||||
|         else | ||||
|             nColumns = this._computeLayout(forWidth)[0]; | ||||
|         let [nColumns, usedWidth] = this._computeLayout(forWidth); | ||||
|         let nRows; | ||||
|         if (nColumns > 0) | ||||
|             nRows = Math.ceil(children.length / nColumns); | ||||
| @@ -241,7 +222,7 @@ const IconGrid = new Lang.Class({ | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|  | ||||
|         let [nColumns, usedWidth, spacing] = this._computeLayout(availWidth); | ||||
|         let [nColumns, usedWidth] = this._computeLayout(availWidth); | ||||
|  | ||||
|         let leftPadding; | ||||
|         switch(this._xAlign) { | ||||
| @@ -294,10 +275,10 @@ const IconGrid = new Lang.Class({ | ||||
|             } | ||||
|  | ||||
|             if (columnIndex == 0) { | ||||
|                 y += this._vItemSize + spacing; | ||||
|                 y += this._vItemSize + this._spacing; | ||||
|                 x = box.x1 + leftPadding; | ||||
|             } else { | ||||
|                 x += this._hItemSize + spacing; | ||||
|                 x += this._hItemSize + this._spacing; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
| @@ -306,32 +287,19 @@ const IconGrid = new Lang.Class({ | ||||
|         return this._computeLayout(rowWidth)[0]; | ||||
|     }, | ||||
|  | ||||
|     getRowLimit: function() { | ||||
|         return this._rowLimit; | ||||
|     }, | ||||
|  | ||||
|     _computeLayout: function (forWidth) { | ||||
|         let nColumns = 0; | ||||
|         let usedWidth = 0; | ||||
|         let spacing = this._spacing; | ||||
|  | ||||
|         if (this._colLimit) { | ||||
|             let itemWidth = this._hItemSize * this._colLimit; | ||||
|             let emptyArea = forWidth - itemWidth; | ||||
|             spacing = Math.max(this._spacing, emptyArea / (2 * this._colLimit)); | ||||
|             spacing = Math.round(spacing); | ||||
|         } | ||||
|  | ||||
|         while ((this._colLimit == null || nColumns < this._colLimit) && | ||||
|                (usedWidth + this._hItemSize <= forWidth)) { | ||||
|             usedWidth += this._hItemSize + spacing; | ||||
|             usedWidth += this._hItemSize + this._spacing; | ||||
|             nColumns += 1; | ||||
|         } | ||||
|  | ||||
|         if (nColumns > 0) | ||||
|             usedWidth -= spacing; | ||||
|             usedWidth -= this._spacing; | ||||
|  | ||||
|         return [nColumns, usedWidth, spacing]; | ||||
|         return [nColumns, usedWidth]; | ||||
|     }, | ||||
|  | ||||
|     _onStyleChanged: function() { | ||||
| @@ -342,22 +310,21 @@ const IconGrid = new Lang.Class({ | ||||
|         this._grid.queue_relayout(); | ||||
|     }, | ||||
|  | ||||
|     removeAll: function() { | ||||
|         this._grid.destroy_all_children(); | ||||
|     removeAll: function () { | ||||
|         this._grid.get_children().forEach(Lang.bind(this, function (child) { | ||||
|             child.destroy(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     addItem: function(actor, index) { | ||||
|         if (index !== undefined) | ||||
|             this._grid.insert_child_at_index(actor, index); | ||||
|         else | ||||
|             this._grid.add_actor(actor); | ||||
|     addItem: function(actor) { | ||||
|         this._grid.add_actor(actor); | ||||
|     }, | ||||
|  | ||||
|     getItemAtIndex: function(index) { | ||||
|         return this._grid.get_child_at_index(index); | ||||
|         return this._grid.get_children()[index]; | ||||
|     }, | ||||
|  | ||||
|     visibleItemsCount: function() { | ||||
|         return this._grid.get_n_children() - this._grid.get_n_skip_paint(); | ||||
|         return this._grid.get_children().length - this._grid.get_n_skip_paint(); | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -2,27 +2,43 @@ | ||||
|  | ||||
| const Caribou = imports.gi.Caribou; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const DBus = imports.dbus; | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const BoxPointer = imports.ui.boxpointer; | ||||
| const Layout = imports.ui.layout; | ||||
| const Main = imports.ui.main; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
|  | ||||
| const KEYBOARD_REST_TIME = Layout.KEYBOARD_ANIMATION_TIME * 2 * 1000; | ||||
|  | ||||
| const KEYBOARD_SCHEMA = 'org.gnome.shell.keyboard'; | ||||
| const KEYBOARD_TYPE = 'keyboard-type'; | ||||
|  | ||||
| const A11Y_APPLICATIONS_SCHEMA = 'org.gnome.desktop.a11y.applications'; | ||||
| const SHOW_KEYBOARD = 'screen-keyboard-enabled'; | ||||
|  | ||||
| // Key constants taken from Antler | ||||
| // FIXME: ought to be moved into libcaribou | ||||
| const PRETTY_KEYS = { | ||||
|     'BackSpace': '\u232b', | ||||
|     'space': ' ', | ||||
|     'Return': '\u23ce', | ||||
|     'Caribou_Prefs': '\u2328', | ||||
|     'Caribou_ShiftUp': '\u2b06', | ||||
|     'Caribou_ShiftDown': '\u2b07', | ||||
|     'Caribou_Emoticons': '\u263a', | ||||
|     'Caribou_Symbols': '123', | ||||
|     'Caribou_Symbols_More': '{#*', | ||||
|     'Caribou_Alpha': 'Abc', | ||||
|     'Tab': 'Tab', | ||||
|     'Escape': 'Esc', | ||||
|     'Control_L': 'Ctrl', | ||||
|     'Alt_L': 'Alt' | ||||
| }; | ||||
|  | ||||
| const CaribouKeyboardIface = <interface name='org.gnome.Caribou.Keyboard'> | ||||
| <method name='Show'> | ||||
|     <arg type='u' direction='in' /> | ||||
| @@ -59,7 +75,14 @@ const Key = new Lang.Class({ | ||||
|         if (this._key.name == 'Control_L' || this._key.name == 'Alt_L') | ||||
|             this._key.latch = true; | ||||
|  | ||||
|         this._key.connect('key-pressed', Lang.bind(this, function () | ||||
|                                                    { this.actor.checked = true })); | ||||
|         this._key.connect('key-released', Lang.bind(this, function () | ||||
|                                                     { this.actor.checked = false; })); | ||||
|  | ||||
|         if (this._extended_keys.length > 0) { | ||||
|             this._grabbed = false; | ||||
|             this._eventCaptureId = 0; | ||||
|             this._key.connect('notify::show-subkeys', Lang.bind(this, this._onShowSubkeysChanged)); | ||||
|             this._boxPointer = new BoxPointer.BoxPointer(St.Side.BOTTOM, | ||||
|                                                          { x_fill: true, | ||||
| @@ -70,12 +93,22 @@ const Key = new Lang.Class({ | ||||
|             this._getExtendedKeys(); | ||||
|             this.actor._extended_keys = this._extended_keyboard; | ||||
|             this._boxPointer.actor.hide(); | ||||
|             Main.layoutManager.addChrome(this._boxPointer.actor); | ||||
|             Main.layoutManager.addChrome(this._boxPointer.actor, { visibleInFullscreen: true }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _makeKey: function () { | ||||
|         let label = GLib.markup_escape_text(this._key.label, -1); | ||||
|         let label = this._key.name; | ||||
|  | ||||
|         if (label.length > 1) { | ||||
|             let pretty = PRETTY_KEYS[label]; | ||||
|             if (pretty) | ||||
|                 label = pretty; | ||||
|             else | ||||
|                 label = this._getUnichar(this._key); | ||||
|         } | ||||
|  | ||||
|         label = GLib.markup_escape_text(label, -1); | ||||
|         let button = new St.Button ({ label: label, | ||||
|                                       style_class: 'keyboard-key' }); | ||||
|  | ||||
| @@ -111,23 +144,52 @@ const Key = new Lang.Class({ | ||||
|         this._boxPointer.bin.add_actor(this._extended_keyboard); | ||||
|     }, | ||||
|  | ||||
|     get subkeys() { | ||||
|         return this._boxPointer; | ||||
|     _onEventCapture: function (actor, event) { | ||||
|         let source = event.get_source(); | ||||
|         let type = event.type(); | ||||
|  | ||||
|         if ((type == Clutter.EventType.BUTTON_PRESS || | ||||
|              type == Clutter.EventType.BUTTON_RELEASE) && | ||||
|             this._extended_keyboard.contains(source)) { | ||||
|             source.extended_key.press(); | ||||
|             source.extended_key.release(); | ||||
|             return false; | ||||
|         } | ||||
|         if (type == Clutter.EventType.BUTTON_PRESS) { | ||||
|             this._boxPointer.actor.hide(); | ||||
|             this._ungrab(); | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _ungrab: function () { | ||||
|         global.stage.disconnect(this._eventCaptureId); | ||||
|         this._eventCaptureId = 0; | ||||
|         this._grabbed = false; | ||||
|         Main.popModal(this.actor); | ||||
|     }, | ||||
|  | ||||
|     _onShowSubkeysChanged: function () { | ||||
|         if (this._key.show_subkeys) { | ||||
|             this.actor.fake_release(); | ||||
|             this._boxPointer.actor.raise_top(); | ||||
|             this._boxPointer.setPosition(this.actor, 0.5); | ||||
|             this.emit('show-subkeys'); | ||||
|             this.actor.fake_release(); | ||||
|             this._boxPointer.show(true); | ||||
|             this.actor.set_hover(false); | ||||
|             if (!this._grabbed) { | ||||
|                  Main.pushModal(this.actor); | ||||
|                  this._eventCaptureId = global.stage.connect('captured-event', Lang.bind(this, this._onEventCapture)); | ||||
|                  this._grabbed = true; | ||||
|             } | ||||
|             this._key.release(); | ||||
|         } else { | ||||
|             this.emit('hide-subkeys'); | ||||
|             if (this._grabbed) | ||||
|                 this._ungrab(); | ||||
|             this._boxPointer.hide(true); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Key.prototype); | ||||
|  | ||||
| const Keyboard = new Lang.Class({ | ||||
|     // HACK: we can't set Name, because it collides with Name dbus property | ||||
| @@ -138,30 +200,18 @@ const Keyboard = new Lang.Class({ | ||||
|         this._impl.export(Gio.DBus.session, '/org/gnome/Caribou/Keyboard'); | ||||
|  | ||||
|         this.actor = null; | ||||
|         this._focusInTray = false; | ||||
|         this._focusInExtendedKeys = false; | ||||
|  | ||||
|         this._timestamp = global.display.get_current_time_roundtrip(); | ||||
|         this._timestamp = global.get_current_time(); | ||||
|         Main.layoutManager.connect('monitors-changed', Lang.bind(this, this._redraw)); | ||||
|  | ||||
|         this._keyboardSettings = new Gio.Settings({ schema: KEYBOARD_SCHEMA }); | ||||
|         this._keyboardSettings.connect('changed', Lang.bind(this, this._settingsChanged)); | ||||
|         this._a11yApplicationsSettings = new Gio.Settings({ schema: A11Y_APPLICATIONS_SCHEMA }); | ||||
|         this._a11yApplicationsSettings.connect('changed', Lang.bind(this, this._settingsChanged)); | ||||
|         this._settingsChanged(); | ||||
|     }, | ||||
|  | ||||
|         this._showIdleId = 0; | ||||
|         this._subkeysBoxPointer = null; | ||||
|         this._capturedEventId = 0; | ||||
|         this._capturedPress = false; | ||||
|  | ||||
|         this._keyboardVisible = false; | ||||
|         Main.layoutManager.connect('keyboard-visible-changed', Lang.bind(this, function(o, visible) { | ||||
|             this._keyboardVisible = visible; | ||||
|         })); | ||||
|         this._keyboardRequested = false; | ||||
|         this._keyboardRestingId = 0; | ||||
|  | ||||
|         Main.layoutManager.connect('monitors-changed', Lang.bind(this, this._redraw)); | ||||
|     init: function () { | ||||
|         this._redraw(); | ||||
|     }, | ||||
|  | ||||
| @@ -176,19 +226,25 @@ const Keyboard = new Lang.Class({ | ||||
|         if (this._keyboard) | ||||
|             this._destroyKeyboard(); | ||||
|  | ||||
|         if (this._enableKeyboard) | ||||
|             this._setupKeyboard(); | ||||
|         else | ||||
|         if (this._enableKeyboard) { | ||||
|             // If we've been called because the setting actually just | ||||
|             // changed to true (as opposed to being called from | ||||
|             // this._init()), then we want to pop up the keyboard. | ||||
|             let showKeyboard = (settings != null); | ||||
|  | ||||
|             // However, caribou-gtk-module or this._onKeyFocusChanged | ||||
|             // will probably immediately tell us to hide it, so we | ||||
|             // have to fake things out so we'll ignore that request. | ||||
|             if (showKeyboard) | ||||
|                 this._timestamp = global.display.get_current_time_roundtrip() + 1; | ||||
|             this._setupKeyboard(showKeyboard); | ||||
|         } else | ||||
|             Main.layoutManager.hideKeyboard(true); | ||||
|     }, | ||||
|  | ||||
|     _destroyKeyboard: function() { | ||||
|         if (this._keyboardNotifyId) | ||||
|             this._keyboard.disconnect(this._keyboardNotifyId); | ||||
|         if (this._keyboardGroupAddedId) | ||||
|             this._keyboard.disconnect(this._keyboardGroupAddedId); | ||||
|         if (this._keyboardGroupRemovedId) | ||||
|             this._keyboard.disconnect(this._keyboardGroupRemovedId); | ||||
|         if (this._focusNotifyId) | ||||
|             global.stage.disconnect(this._focusNotifyId); | ||||
|         this._keyboard = null; | ||||
| @@ -198,7 +254,7 @@ const Keyboard = new Lang.Class({ | ||||
|         this._destroySource(); | ||||
|     }, | ||||
|  | ||||
|     _setupKeyboard: function() { | ||||
|     _setupKeyboard: function(show) { | ||||
|         this.actor = new St.BoxLayout({ name: 'keyboard', vertical: true, reactive: true }); | ||||
|         Main.layoutManager.keyboardBox.add_actor(this.actor); | ||||
|         Main.layoutManager.trackChrome(this.actor); | ||||
| @@ -219,11 +275,12 @@ const Keyboard = new Lang.Class({ | ||||
|         this.actor.text_direction = Clutter.TextDirection.LTR; | ||||
|  | ||||
|         this._keyboardNotifyId = this._keyboard.connect('notify::active-group', Lang.bind(this, this._onGroupChanged)); | ||||
|         this._keyboardGroupAddedId = this._keyboard.connect('group-added', Lang.bind(this, this._onGroupAdded)); | ||||
|         this._keyboardGroupRemovedId = this._keyboard.connect('group-removed', Lang.bind(this, this._onGroupRemoved)); | ||||
|         this._focusNotifyId = global.stage.connect('notify::key-focus', Lang.bind(this, this._onKeyFocusChanged)); | ||||
|  | ||||
|         this._createSource(); | ||||
|         if (show) | ||||
|             this.show(); | ||||
|         else | ||||
|             this._createSource(); | ||||
|     }, | ||||
|  | ||||
|     _onKeyFocusChanged: function () { | ||||
| @@ -231,114 +288,85 @@ const Keyboard = new Lang.Class({ | ||||
|  | ||||
|         // Showing an extended key popup and clicking a key from the extended keys | ||||
|         // will grab focus, but ignore that | ||||
|         let extendedKeysWereFocused = this._focusInExtendedKeys; | ||||
|         this._focusInExtendedKeys = focus && (focus._extended_keys || focus.extended_key); | ||||
|         if (this._focusInExtendedKeys || extendedKeysWereFocused) | ||||
|             return; | ||||
|  | ||||
|         // Ignore focus changes caused by message tray showing/hiding | ||||
|         let trayWasFocused = this._focusInTray; | ||||
|         this._focusInTray = (focus && Main.messageTray.actor.contains(focus)); | ||||
|         if (this._focusInTray || trayWasFocused) | ||||
|         if (focus && (focus._extended_keys || (focus._key && focus._key.extended_key))) | ||||
|             return; | ||||
|  | ||||
|         let time = global.get_current_time(); | ||||
|         if (!(focus instanceof Clutter.Text)) { | ||||
|         if (focus instanceof Clutter.Text) | ||||
|             this.Show(time); | ||||
|         else | ||||
|             this.Hide(time); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!this._showIdleId) | ||||
|             this._showIdleId = GLib.idle_add(GLib.PRIORITY_DEFAULT_IDLE, | ||||
|                                              Lang.bind(this, function() { this.Show(time); })); | ||||
|     }, | ||||
|  | ||||
|     _createLayersForGroup: function (gname) { | ||||
|         let group = this._keyboard.get_group(gname); | ||||
|         group.connect('notify::active-level', Lang.bind(this, this._onLevelChanged)); | ||||
|         let layers = {}; | ||||
|         let levels = group.get_levels(); | ||||
|         for (let j = 0; j < levels.length; ++j) { | ||||
|             let lname = levels[j]; | ||||
|             let level = group.get_level(lname); | ||||
|             let layout = new St.BoxLayout({ style_class: 'keyboard-layout', | ||||
|                                                  vertical: true }); | ||||
|             this._loadRows(level, layout); | ||||
|             layers[lname] = layout; | ||||
|             this.actor.add(layout, { x_fill: false }); | ||||
|  | ||||
|             layout.hide(); | ||||
|         } | ||||
|         return layers; | ||||
|     }, | ||||
|  | ||||
|     _addKeys: function () { | ||||
|         let groups = this._keyboard.get_groups(); | ||||
|         for (let i = 0; i < groups.length; ++i) { | ||||
|              let gname = groups[i]; | ||||
|              this._groups[gname] = this._createLayersForGroup(gname); | ||||
|              let group = this._keyboard.get_group(gname); | ||||
|              group.connect('notify::active-level', Lang.bind(this, this._onLevelChanged)); | ||||
|              let layers = {}; | ||||
|              let levels = group.get_levels(); | ||||
|              for (let j = 0; j < levels.length; ++j) { | ||||
|                  let lname = levels[j]; | ||||
|                  let level = group.get_level(lname); | ||||
|                  let layout = new St.BoxLayout({ style_class: 'keyboard-layout', | ||||
|                                                  vertical: true }); | ||||
|                  this._loadRows(level, layout); | ||||
|                  layers[lname] = layout; | ||||
|                  this.actor.add(layout, { x_fill: false }); | ||||
|  | ||||
|                  layout.hide(); | ||||
|              } | ||||
|              this._groups[gname] = layers; | ||||
|         } | ||||
|  | ||||
|         this._setActiveLayer(); | ||||
|     }, | ||||
|  | ||||
|     _onCapturedEvent: function(actor, event) { | ||||
|         let type = event.type(); | ||||
|         let press = type == Clutter.EventType.BUTTON_PRESS; | ||||
|         let release = type == Clutter.EventType.BUTTON_RELEASE; | ||||
|     _getTrayIcon: function () { | ||||
|         let trayButton = new St.Button ({ label: _("tray"), | ||||
|                                           style_class: 'keyboard-key' }); | ||||
|         trayButton.key_width = 1; | ||||
|         trayButton.connect('button-press-event', Lang.bind(this, function () { | ||||
|             Main.messageTray.toggle(); | ||||
|         })); | ||||
|  | ||||
|         if (press) | ||||
|             this._capturedPress = true; | ||||
|         else if (release && this._capturedPress) | ||||
|             this._hideSubkeys(); | ||||
|         Main.overview.connect('showing', Lang.bind(this, function () { | ||||
|             trayButton.reactive = false; | ||||
|             trayButton.add_style_pseudo_class('grayed'); | ||||
|         })); | ||||
|         Main.overview.connect('hiding', Lang.bind(this, function () { | ||||
|             trayButton.reactive = true; | ||||
|             trayButton.remove_style_pseudo_class('grayed'); | ||||
|         })); | ||||
|  | ||||
|         return true; | ||||
|         return trayButton; | ||||
|     }, | ||||
|  | ||||
|     _addRows : function (keys, layout) { | ||||
|         let keyboard_row = new St.BoxLayout(); | ||||
|         for (let i = 0; i < keys.length; ++i) { | ||||
|             let children = keys[i].get_children(); | ||||
|             let left_box = new St.BoxLayout({ style_class: 'keyboard-row' }); | ||||
|             let center_box = new St.BoxLayout({ style_class: 'keyboard-row' }); | ||||
|             let right_box = new St.BoxLayout({ style_class: 'keyboard-row' }); | ||||
|             let left_box = new St.BoxLayout({ style_class: 'keyboard-row' }); | ||||
|             for (let j = 0; j < children.length; ++j) { | ||||
|                 if (this._numOfHorizKeys == 0) | ||||
|                     this._numOfHorizKeys = children.length; | ||||
|                 let key = children[j]; | ||||
|                 let button = new Key(key); | ||||
|  | ||||
|                 switch (key.align) { | ||||
|                 case 'right': | ||||
|                 if (key.align == 'right') | ||||
|                     right_box.add(button.actor); | ||||
|                     break; | ||||
|                 case 'center': | ||||
|                     center_box.add(button.actor); | ||||
|                     break; | ||||
|                 case 'left': | ||||
|                 default: | ||||
|                 else | ||||
|                     left_box.add(button.actor); | ||||
|                     break; | ||||
|                 } | ||||
|                 if (key.name == 'Caribou_Prefs') { | ||||
|                     key.connect('key-released', Lang.bind(this, this.hide)); | ||||
|                 } | ||||
|  | ||||
|                 button.connect('show-subkeys', Lang.bind(this, function() { | ||||
|                     if (this._subkeysBoxPointer) | ||||
|                         this._subkeysBoxPointer.hide(BoxPointer.PopupAnimation.FULL); | ||||
|                     this._subkeysBoxPointer = button.subkeys; | ||||
|                     this._subkeysBoxPointer.show(BoxPointer.PopupAnimation.FULL); | ||||
|                     if (!this._capturedEventId) | ||||
|                         this._capturedEventId = this.actor.connect('captured-event', | ||||
|                                                                    Lang.bind(this, this._onCapturedEvent)); | ||||
|                 })); | ||||
|                 button.connect('hide-subkeys', Lang.bind(this, function() { | ||||
|                     this._hideSubkeys(); | ||||
|                 })); | ||||
|                     // Add new key for hiding message tray | ||||
|                     right_box.add(this._getTrayIcon()); | ||||
|                 } | ||||
|             } | ||||
|             keyboard_row.add(left_box, { expand: true, x_fill: false, x_align: St.Align.START }); | ||||
|             keyboard_row.add(center_box, { expand: true, x_fill: false, x_align: St.Align.MIDDLE }); | ||||
|             keyboard_row.add(right_box, { expand: true, x_fill: false, x_align: St.Align.END }); | ||||
|         } | ||||
|         layout.add(keyboard_row); | ||||
| @@ -359,7 +387,7 @@ const Keyboard = new Lang.Class({ | ||||
|         if (!this._enableKeyboard) | ||||
|             return; | ||||
|  | ||||
|         let monitor = Main.layoutManager.keyboardMonitor; | ||||
|         let monitor = Main.layoutManager.bottomMonitor; | ||||
|         let maxHeight = monitor.height / 3; | ||||
|         this.actor.width = monitor.width; | ||||
|  | ||||
| @@ -411,14 +439,6 @@ const Keyboard = new Lang.Class({ | ||||
|         this._redraw(); | ||||
|     }, | ||||
|  | ||||
|     _onGroupAdded: function (keyboard, gname) { | ||||
|         this._groups[gname] = this._createLayersForGroup(gname); | ||||
|     }, | ||||
|  | ||||
|     _onGroupRemoved: function (keyboard, gname) { | ||||
|         delete this._groups[gname]; | ||||
|     }, | ||||
|  | ||||
|     _setActiveLayer: function () { | ||||
|         let active_group_name = this._keyboard.active_group; | ||||
|         let active_group = this._keyboard.get_group(active_group_name); | ||||
| @@ -448,85 +468,18 @@ const Keyboard = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     shouldTakeEvent: function(event) { | ||||
|         let actor = event.get_source(); | ||||
|         return Main.layoutManager.keyboardBox.contains(actor) || | ||||
|                actor._extended_keys || actor.extended_key; | ||||
|     }, | ||||
|  | ||||
|     _clearKeyboardRestTimer: function() { | ||||
|         if (!this._keyboardRestingId) | ||||
|             return; | ||||
|         GLib.source_remove(this._keyboardRestingId); | ||||
|         this._keyboardRestingId = 0; | ||||
|     }, | ||||
|  | ||||
|     show: function (monitor) { | ||||
|         this._keyboardRequested = true; | ||||
|  | ||||
|         if (this._keyboardVisible) { | ||||
|             if (monitor != Main.layoutManager.keyboardIndex) { | ||||
|                 Main.layoutManager.keyboardIndex = monitor; | ||||
|                 this._redraw(); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._clearKeyboardRestTimer(); | ||||
|         this._keyboardRestingId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, | ||||
|                                                    KEYBOARD_REST_TIME, | ||||
|                                                    Lang.bind(this, function() { | ||||
|                                                        this._clearKeyboardRestTimer(); | ||||
|                                                        this._show(monitor); | ||||
|                                                    })); | ||||
|     }, | ||||
|  | ||||
|     _show: function(monitor) { | ||||
|         if (!this._keyboardRequested) | ||||
|             return; | ||||
|  | ||||
|         Main.layoutManager.keyboardIndex = monitor; | ||||
|     show: function () { | ||||
|         this._redraw(); | ||||
|  | ||||
|         Main.layoutManager.showKeyboard(); | ||||
|         this._destroySource(); | ||||
|     }, | ||||
|  | ||||
|     hide: function () { | ||||
|         this._keyboardRequested = false; | ||||
|  | ||||
|         if (!this._keyboardVisible) | ||||
|             return; | ||||
|  | ||||
|         this._clearKeyboardRestTimer(); | ||||
|         this._keyboardRestingId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, | ||||
|                                                    KEYBOARD_REST_TIME, | ||||
|                                                    Lang.bind(this, function() { | ||||
|                                                        this._clearKeyboardRestTimer(); | ||||
|                                                        this._hide(); | ||||
|                                                    })); | ||||
|     }, | ||||
|  | ||||
|     _hide: function() { | ||||
|         if (this._keyboardRequested) | ||||
|             return; | ||||
|  | ||||
|         this._hideSubkeys(); | ||||
|         Main.layoutManager.hideKeyboard(); | ||||
|         this._createSource(); | ||||
|     }, | ||||
|  | ||||
|     _hideSubkeys: function() { | ||||
|         if (this._subkeysBoxPointer) { | ||||
|             this._subkeysBoxPointer.hide(BoxPointer.PopupAnimation.FULL); | ||||
|             this._subkeysBoxPointer = null; | ||||
|         } | ||||
|         if (this._capturedEventId) { | ||||
|             this.actor.disconnect(this._capturedEventId); | ||||
|             this._capturedEventId = 0; | ||||
|         } | ||||
|         this._capturedPress = false; | ||||
|     }, | ||||
|  | ||||
|     _moveTemporarily: function () { | ||||
|         let currentWindow = global.screen.get_display().focus_window; | ||||
|         let rect = currentWindow.get_outer_rect(); | ||||
| @@ -541,53 +494,26 @@ const Keyboard = new Lang.Class({ | ||||
|             this._moveTemporarily(); | ||||
|     }, | ||||
|  | ||||
|     // _compareTimestamp: | ||||
|     // | ||||
|     // Compare two timestamps taking into account | ||||
|     // CURRENT_TIME (0) | ||||
|     _compareTimestamp: function(one, two) { | ||||
|         if (one == two) | ||||
|             return 0; | ||||
|         if (one == Clutter.CURRENT_TIME) | ||||
|             return 1; | ||||
|         if (two == Clutter.CURRENT_TIME) | ||||
|             return -1; | ||||
|         return one - two; | ||||
|     }, | ||||
|  | ||||
|     _clearShowIdle: function() { | ||||
|         if (!this._showIdleId) | ||||
|             return; | ||||
|         GLib.source_remove(this._showIdleId); | ||||
|         this._showIdleId = 0; | ||||
|     }, | ||||
|  | ||||
|     // D-Bus methods | ||||
|     Show: function(timestamp) { | ||||
|         if (!this._enableKeyboard) | ||||
|             return; | ||||
|  | ||||
|         if (this._compareTimestamp(timestamp, this._timestamp) < 0) | ||||
|         if (timestamp - this._timestamp < 0) | ||||
|             return; | ||||
|  | ||||
|         this._clearShowIdle(); | ||||
|  | ||||
|         if (timestamp != Clutter.CURRENT_TIME) | ||||
|             this._timestamp = timestamp; | ||||
|         this.show(Main.layoutManager.focusIndex); | ||||
|         this._timestamp = timestamp; | ||||
|         this.show(); | ||||
|     }, | ||||
|  | ||||
|     Hide: function(timestamp) { | ||||
|         if (!this._enableKeyboard) | ||||
|             return; | ||||
|  | ||||
|         if (this._compareTimestamp(timestamp, this._timestamp) < 0) | ||||
|         if (timestamp - this._timestamp < 0) | ||||
|             return; | ||||
|  | ||||
|         this._clearShowIdle(); | ||||
|  | ||||
|         if (timestamp != Clutter.CURRENT_TIME) | ||||
|             this._timestamp = timestamp; | ||||
|         this._timestamp = timestamp; | ||||
|         this.hide(); | ||||
|     }, | ||||
|  | ||||
| @@ -615,18 +541,28 @@ const KeyboardSource = new Lang.Class({ | ||||
|     Extends: MessageTray.Source, | ||||
|  | ||||
|     _init: function(keyboard) { | ||||
|         this.parent(_("Keyboard")); | ||||
|         this._keyboard = keyboard; | ||||
|         this.parent(_("Keyboard"), 'input-keyboard-symbolic'); | ||||
|         this.keepTrayOnSummaryClick = true; | ||||
|  | ||||
|         this._setSummaryIcon(this.createNotificationIcon()); | ||||
|     }, | ||||
|  | ||||
|     handleSummaryClick: function(button) { | ||||
|     createNotificationIcon: function() { | ||||
|         return new St.Icon({ icon_name: 'input-keyboard', | ||||
|                              icon_type: St.IconType.SYMBOLIC, | ||||
|                              icon_size: this.ICON_SIZE }); | ||||
|     }, | ||||
|  | ||||
|     handleSummaryClick: function() { | ||||
|         let event = Clutter.get_current_event(); | ||||
|         if (event.type() != Clutter.EventType.BUTTON_RELEASE) | ||||
|             return false; | ||||
|  | ||||
|         this.open(); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     open: function() { | ||||
|         // Show the OSK below the message tray | ||||
|         this._keyboard.show(Main.layoutManager.bottomIndex); | ||||
|         this._keyboard.show(); | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -60,14 +60,17 @@ const KeyringDialog = new Lang.Class({ | ||||
| 
 | ||||
|         this._controlTable = null; | ||||
| 
 | ||||
|         let buttons = [{ label: '', | ||||
|                          action: Lang.bind(this, this._onCancelButton), | ||||
|                          key:    Clutter.Escape | ||||
|                        }, | ||||
|                        { label: '', | ||||
|                          action: Lang.bind(this, this._onContinueButton) | ||||
|                        }] | ||||
| 
 | ||||
|         this._cancelButton = this.addButton({ label: '', | ||||
|                                               action: Lang.bind(this, this._onCancelButton), | ||||
|                                               key: Clutter.Escape }); | ||||
|         this._continueButton = this.addButton({ label: '', | ||||
|                                                 action: Lang.bind(this, this._onContinueButton), | ||||
|                                                 default: true }, | ||||
|                                               { expand: true, x_fill: false, x_align: St.Align.END }); | ||||
|         this.setButtons(buttons); | ||||
|         this._cancelButton = buttons[0].button; | ||||
|         this._continueButton = buttons[1].button; | ||||
| 
 | ||||
|         this.prompt.bind_property('cancel-label', this._cancelButton, 'label', GObject.BindingFlags.SYNC_CREATE); | ||||
|         this.prompt.bind_property('continue-label', this._continueButton, 'label', GObject.BindingFlags.SYNC_CREATE); | ||||
| @@ -80,10 +83,7 @@ const KeyringDialog = new Lang.Class({ | ||||
|         if (this.prompt.password_visible) { | ||||
|             let label = new St.Label(({ style_class: 'prompt-dialog-password-label' })); | ||||
|             label.set_text(_("Password:")); | ||||
|             table.add(label, { row: row, col: 0, | ||||
|                                x_expand: false, x_fill: true, | ||||
|                                x_align: St.Align.START, | ||||
|                                y_fill: false, y_align: St.Align.MIDDLE }); | ||||
|             table.add(label, { row: row, col: 0, x_expand: false, x_fill: true, x_align: St.Align.START }); | ||||
|             this._passwordEntry = new St.Entry({ style_class: 'prompt-dialog-password-entry', | ||||
|                                                  text: '', | ||||
|                                                  can_focus: true}); | ||||
| @@ -99,10 +99,7 @@ const KeyringDialog = new Lang.Class({ | ||||
|         if (this.prompt.confirm_visible) { | ||||
|             var label = new St.Label(({ style_class: 'prompt-dialog-password-label' })); | ||||
|             label.set_text(_("Type again:")); | ||||
|             table.add(label, { row: row, col: 0, | ||||
|                                x_expand: false, x_fill: true, | ||||
|                                x_align: St.Align.START, | ||||
|                                y_fill: false, y_align: St.Align.MIDDLE }); | ||||
|             table.add(label, { row: row, col: 0, x_expand: false, x_fill: true, x_align: St.Align.START }); | ||||
|             this._confirmEntry = new St.Entry({ style_class: 'prompt-dialog-password-entry', | ||||
|                                                 text: '', | ||||
|                                                 can_focus: true}); | ||||
| @@ -142,14 +139,6 @@ const KeyringDialog = new Lang.Class({ | ||||
|         this._messageBox.add(table, { x_fill: true, y_fill: true }); | ||||
|     }, | ||||
| 
 | ||||
|     _updateSensitivity: function(sensitive) { | ||||
|         this._passwordEntry.reactive = sensitive; | ||||
|         this._passwordEntry.clutter_text.editable = sensitive; | ||||
| 
 | ||||
|         this._continueButton.can_focus = sensitive; | ||||
|         this._continueButton.reactive = sensitive; | ||||
|     }, | ||||
| 
 | ||||
|     _ensureOpen: function() { | ||||
|         // NOTE: ModalDialog.open() is safe to call if the dialog is
 | ||||
|         // already open - it just returns true without side-effects
 | ||||
| @@ -171,14 +160,12 @@ const KeyringDialog = new Lang.Class({ | ||||
|     _onShowPassword: function(prompt) { | ||||
|         this._buildControlTable(); | ||||
|         this._ensureOpen(); | ||||
|         this._updateSensitivity(true); | ||||
|         this._passwordEntry.grab_key_focus(); | ||||
|     }, | ||||
| 
 | ||||
|     _onShowConfirm: function(prompt) { | ||||
|         this._buildControlTable(); | ||||
|         this._ensureOpen(); | ||||
|         this._updateSensitivity(true); | ||||
|         this._continueButton.grab_key_focus(); | ||||
|     }, | ||||
| 
 | ||||
| @@ -198,37 +185,23 @@ const KeyringDialog = new Lang.Class({ | ||||
|     }, | ||||
| 
 | ||||
|     _onContinueButton: function() { | ||||
|         this._updateSensitivity(false); | ||||
|         this.prompt.complete(); | ||||
|         this.prompt.complete() | ||||
|     }, | ||||
| 
 | ||||
|     _onCancelButton: function() { | ||||
|         this.prompt.cancel(); | ||||
|         this.prompt.cancel() | ||||
|     }, | ||||
| }); | ||||
| 
 | ||||
| const KeyringPrompter = new Lang.Class({ | ||||
|     Name: 'KeyringPrompter', | ||||
| function init() { | ||||
|     prompter = new Gcr.SystemPrompter(); | ||||
|     prompter.connect('new-prompt', function(prompter) { | ||||
|         let dialog = new KeyringDialog(); | ||||
|         return dialog.prompt; | ||||
|     }); | ||||
| 
 | ||||
|     _init: function() { | ||||
|         this._prompter = new Gcr.SystemPrompter(); | ||||
|         this._prompter.connect('new-prompt', function(prompter) { | ||||
|             let dialog = new KeyringDialog(); | ||||
|             return dialog.prompt; | ||||
|         }); | ||||
|         this._dbusId = null; | ||||
|     }, | ||||
| 
 | ||||
|     enable: function() { | ||||
|         this._prompter.register(Gio.DBus.session); | ||||
|         this._dbusId = Gio.DBus.session.own_name('org.gnome.keyring.SystemPrompter', | ||||
|                                                  Gio.BusNameOwnerFlags.REPLACE, null, null); | ||||
|     }, | ||||
| 
 | ||||
|     disable: function() { | ||||
|         this._prompter.unregister(false); | ||||
|         Gio.DBus.session.unown_name(this._dbusId); | ||||
|     } | ||||
| }); | ||||
| 
 | ||||
| const Component = KeyringPrompter; | ||||
|     let connection = Gio.DBus.session; | ||||
|     prompter.register(connection); | ||||
|     Gio.bus_own_name_on_connection (connection, 'org.gnome.keyring.SystemPrompter', | ||||
|                                     Gio.BusNameOwnerFlags.REPLACE, null, null); | ||||
| } | ||||
							
								
								
									
										1377
									
								
								js/ui/layout.js
									
									
									
									
									
								
							
							
						
						| @@ -3,7 +3,6 @@ | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
| @@ -102,7 +101,6 @@ const Lightbox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     show: function() { | ||||
|         Tweener.removeTweens(this.actor); | ||||
|         if (this._fadeInTime) { | ||||
|             this.shown = false; | ||||
|             this.actor.opacity = 0; | ||||
| @@ -112,20 +110,17 @@ const Lightbox = new Lang.Class({ | ||||
|                                transition: 'easeOutQuad', | ||||
|                                onComplete: Lang.bind(this, function() { | ||||
|                                    this.shown = true; | ||||
|                                    this.emit('shown'); | ||||
|                                }) | ||||
|                              }); | ||||
|         } else { | ||||
|             this.actor.opacity = 255 * this._fadeFactor; | ||||
|             this.shown = true; | ||||
|             this.emit('shown'); | ||||
|         } | ||||
|         this.actor.show(); | ||||
|     }, | ||||
|  | ||||
|     hide: function() { | ||||
|         this.shown = false; | ||||
|         Tweener.removeTweens(this.actor); | ||||
|         if (this._fadeOutTime) { | ||||
|             Tweener.addTween(this.actor, | ||||
|                              { opacity: 0, | ||||
| @@ -202,4 +197,3 @@ const Lightbox = new Lang.Class({ | ||||
|         this.highlight(null); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Lightbox.prototype); | ||||
|   | ||||
							
								
								
									
										21
									
								
								js/ui/link.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Link = new Lang.Class({ | ||||
|     Name: 'Link', | ||||
|  | ||||
|     _init : function(props) { | ||||
|         let realProps = { reactive: true, | ||||
|                           track_hover: true, | ||||
|                           style_class: 'shell-link' }; | ||||
|         // The user can pass in reactive: false to override the above and get | ||||
|         // a non-reactive link (a link to the current page, perhaps) | ||||
|         Lang.copyProperties(props, realProps); | ||||
|  | ||||
|         this.actor = new St.Button(realProps); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Link.prototype); | ||||
| @@ -12,18 +12,16 @@ const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const System = imports.system; | ||||
|  | ||||
| const History = imports.misc.history; | ||||
| const ExtensionSystem = imports.ui.extensionSystem; | ||||
| const ExtensionUtils = imports.misc.extensionUtils; | ||||
| const Link = imports.ui.link; | ||||
| const ShellEntry = imports.ui.shellEntry; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const Main = imports.ui.main; | ||||
| const JsParse = imports.misc.jsParse; | ||||
|  | ||||
| const CHEVRON = '>>> '; | ||||
|  | ||||
| /* Imports...feel free to add here as needed */ | ||||
| var commandHeader = 'const Clutter = imports.gi.Clutter; ' + | ||||
|                     'const GLib = imports.gi.GLib; ' + | ||||
| @@ -38,9 +36,9 @@ var commandHeader = 'const Clutter = imports.gi.Clutter; ' + | ||||
|                     /* Utility functions...we should probably be able to use these | ||||
|                      * in the shell core code too. */ | ||||
|                     'const stage = global.stage; ' + | ||||
|                     'const color = function(pixel) { let c= new Clutter.Color(); c.from_pixel(pixel); return c; }; ' + | ||||
|                     /* Special lookingGlass functions */ | ||||
|                     'const inspect = Lang.bind(Main.lookingGlass, Main.lookingGlass.inspect); ' + | ||||
|                     'const it = Main.lookingGlass.getIt(); ' + | ||||
|                        'const it = Main.lookingGlass.getIt(); ' + | ||||
|                     'const r = Lang.bind(Main.lookingGlass, Main.lookingGlass.getResult); '; | ||||
|  | ||||
| const HISTORY_KEY = 'looking-glass-history'; | ||||
| @@ -263,8 +261,9 @@ function objectToString(o) { | ||||
|  | ||||
| const ObjLink = new Lang.Class({ | ||||
|     Name: 'ObjLink', | ||||
|     Extends: Link.Link, | ||||
|  | ||||
|     _init: function(lookingGlass, o, title) { | ||||
|     _init: function(o, title) { | ||||
|         let text; | ||||
|         if (title) | ||||
|             text = title; | ||||
| @@ -273,30 +272,24 @@ const ObjLink = new Lang.Class({ | ||||
|         text = GLib.markup_escape_text(text, -1); | ||||
|         this._obj = o; | ||||
|  | ||||
|         this.actor = new St.Button({ reactive: true, | ||||
|                                      track_hover: true, | ||||
|                                      style_class: 'shell-link', | ||||
|                                      label: text }); | ||||
|         this.parent({ label: text }); | ||||
|         this.actor.get_child().single_line_mode = true; | ||||
|         this.actor.connect('clicked', Lang.bind(this, this._onClicked)); | ||||
|  | ||||
|         this._lookingGlass = lookingGlass; | ||||
|     }, | ||||
|  | ||||
|     _onClicked: function (link) { | ||||
|         this._lookingGlass.inspectObject(this._obj, this.actor); | ||||
|         Main.lookingGlass.inspectObject(this._obj, this.actor); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const Result = new Lang.Class({ | ||||
|     Name: 'Result', | ||||
|  | ||||
|     _init: function(lookingGlass, command, o, index) { | ||||
|     _init : function(command, o, index) { | ||||
|         this.index = index; | ||||
|         this.o = o; | ||||
|  | ||||
|         this.actor = new St.BoxLayout({ vertical: true }); | ||||
|         this._lookingGlass = lookingGlass; | ||||
|  | ||||
|         let cmdTxt = new St.Label({ text: command }); | ||||
|         cmdTxt.clutter_text.ellipsize = Pango.EllipsizeMode.END; | ||||
| @@ -306,7 +299,7 @@ const Result = new Lang.Class({ | ||||
|         let resultTxt = new St.Label({ text: 'r(' + index + ') = ' }); | ||||
|         resultTxt.clutter_text.ellipsize = Pango.EllipsizeMode.END; | ||||
|         box.add(resultTxt); | ||||
|         let objLink = new ObjLink(this._lookingGlass, o); | ||||
|         let objLink = new ObjLink(o); | ||||
|         box.add(objLink.actor); | ||||
|         let line = new Clutter.Rectangle({ name: 'Separator' }); | ||||
|         let padBin = new St.Bin({ name: 'Separator', x_fill: true, y_fill: true }); | ||||
| @@ -318,18 +311,16 @@ const Result = new Lang.Class({ | ||||
| const WindowList = new Lang.Class({ | ||||
|     Name: 'WindowList', | ||||
|  | ||||
|     _init: function(lookingGlass) { | ||||
|     _init : function () { | ||||
|         this.actor = new St.BoxLayout({ name: 'Windows', vertical: true, style: 'spacing: 8px' }); | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         this._updateId = Main.initializeDeferredWork(this.actor, Lang.bind(this, this._updateWindowList)); | ||||
|         global.display.connect('window-created', Lang.bind(this, this._updateWindowList)); | ||||
|         tracker.connect('tracked-windows-changed', Lang.bind(this, this._updateWindowList)); | ||||
|  | ||||
|         this._lookingGlass = lookingGlass; | ||||
|     }, | ||||
|  | ||||
|     _updateWindowList: function() { | ||||
|         this.actor.destroy_all_children(); | ||||
|         this.actor.get_children().forEach(function (actor) { actor.destroy(); }); | ||||
|         let windows = global.get_window_actors(); | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         for (let i = 0; i < windows.length; i++) { | ||||
| @@ -341,7 +332,7 @@ const WindowList = new Lang.Class({ | ||||
|             } | ||||
|             let box = new St.BoxLayout({ vertical: true }); | ||||
|             this.actor.add(box); | ||||
|             let windowLink = new ObjLink(this._lookingGlass, metaWindow, metaWindow.title); | ||||
|             let windowLink = new ObjLink(metaWindow, metaWindow.title); | ||||
|             box.add(windowLink.actor, { x_align: St.Align.START, x_fill: false }); | ||||
|             let propsBox = new St.BoxLayout({ vertical: true, style: 'padding-left: 6px;' }); | ||||
|             box.add(propsBox); | ||||
| @@ -352,7 +343,7 @@ const WindowList = new Lang.Class({ | ||||
|                 let propBox = new St.BoxLayout({ style: 'spacing: 6px; ' }); | ||||
|                 propsBox.add(propBox); | ||||
|                 propBox.add(new St.Label({ text: 'app: ' }), { y_fill: false }); | ||||
|                 let appLink = new ObjLink(this._lookingGlass, app, app.get_id()); | ||||
|                 let appLink = new ObjLink(app, app.get_id()); | ||||
|                 propBox.add(appLink.actor, { y_fill: false }); | ||||
|                 propBox.add(icon, { y_fill: false }); | ||||
|             } else { | ||||
| @@ -366,21 +357,18 @@ Signals.addSignalMethods(WindowList.prototype); | ||||
| const ObjInspector = new Lang.Class({ | ||||
|     Name: 'ObjInspector', | ||||
|  | ||||
|     _init: function(lookingGlass) { | ||||
|     _init : function () { | ||||
|         this._obj = null; | ||||
|         this._previousObj = null; | ||||
|  | ||||
|         this._parentList = []; | ||||
|  | ||||
|         this.actor = new St.ScrollView({ pivot_point: new Clutter.Point({ x: 0.5, y: 0.5 }), | ||||
|                                          x_fill: true, y_fill: true }); | ||||
|         this.actor = new St.ScrollView({ x_fill: true, y_fill: true }); | ||||
|         this.actor.get_hscroll_bar().hide(); | ||||
|         this._container = new St.BoxLayout({ name: 'LookingGlassPropertyInspector', | ||||
|                                              style_class: 'lg-dialog', | ||||
|                                              vertical: true }); | ||||
|         this.actor.add_actor(this._container); | ||||
|  | ||||
|         this._lookingGlass = lookingGlass; | ||||
|     }, | ||||
|  | ||||
|     selectObject: function(obj, skipPrevious) { | ||||
| @@ -390,7 +378,7 @@ const ObjInspector = new Lang.Class({ | ||||
|             this._previousObj = null; | ||||
|         this._obj = obj; | ||||
|  | ||||
|         this._container.destroy_all_children(); | ||||
|         this._container.get_children().forEach(function (child) { child.destroy(); }); | ||||
|  | ||||
|         let hbox = new St.BoxLayout({ style_class: 'lg-obj-inspector-title' }); | ||||
|         this._container.add_actor(hbox); | ||||
| @@ -412,19 +400,12 @@ const ObjInspector = new Lang.Class({ | ||||
|         button.connect('clicked', Lang.bind(this, this.close)); | ||||
|         hbox.add(button); | ||||
|         if (typeof(obj) == typeof({})) { | ||||
|             let properties = []; | ||||
|             for (let propName in obj) { | ||||
|                 properties.push(propName); | ||||
|             } | ||||
|             properties.sort(); | ||||
|  | ||||
|             for (let i = 0; i < properties.length; i++) { | ||||
|                 let propName = properties[i]; | ||||
|                 let valueStr; | ||||
|                 let link; | ||||
|                 try { | ||||
|                     let prop = obj[propName]; | ||||
|                     link = new ObjLink(this._lookingGlass, prop).actor; | ||||
|                     link = new ObjLink(prop).actor; | ||||
|                 } catch (e) { | ||||
|                     link = new St.Label({ text: '<error>' }); | ||||
|                 } | ||||
| @@ -445,6 +426,10 @@ const ObjInspector = new Lang.Class({ | ||||
|         this.actor.show(); | ||||
|         if (sourceActor) { | ||||
|             this.actor.set_scale(0, 0); | ||||
|             let [sourceX, sourceY] = sourceActor.get_transformed_position(); | ||||
|             let [sourceWidth, sourceHeight] = sourceActor.get_transformed_size(); | ||||
|             this.actor.move_anchor_point(Math.floor(sourceX + sourceWidth / 2), | ||||
|                                          Math.floor(sourceY + sourceHeight / 2)); | ||||
|             Tweener.addTween(this.actor, { scale_x: 1, scale_y: 1, | ||||
|                                            transition: 'easeOutQuad', | ||||
|                                            time: 0.2 }); | ||||
| @@ -465,7 +450,7 @@ const ObjInspector = new Lang.Class({ | ||||
|     _onInsert: function() { | ||||
|         let obj = this._obj; | ||||
|         this.close(); | ||||
|         this._lookingGlass.insertObject(obj); | ||||
|         Main.lookingGlass.insertObject(obj); | ||||
|     }, | ||||
|  | ||||
|     _onBack: function() { | ||||
| @@ -473,36 +458,34 @@ const ObjInspector = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const RedBorderEffect = new Lang.Class({ | ||||
|     Name: 'RedBorderEffect', | ||||
|     Extends: Clutter.Effect, | ||||
| function addBorderPaintHook(actor) { | ||||
|     let signalId = actor.connect_after('paint', | ||||
|         function () { | ||||
|             let color = new Cogl.Color(); | ||||
|             color.init_from_4ub(0xff, 0, 0, 0xc4); | ||||
|             Cogl.set_source_color(color); | ||||
|  | ||||
|     vfunc_paint: function() { | ||||
|         let actor = this.get_actor(); | ||||
|         actor.continue_paint(); | ||||
|             let geom = actor.get_allocation_geometry(); | ||||
|             let width = 2; | ||||
|  | ||||
|         let color = new Cogl.Color(); | ||||
|         color.init_from_4ub(0xff, 0, 0, 0xc4); | ||||
|         Cogl.set_source_color(color); | ||||
|             // clockwise order | ||||
|             Cogl.rectangle(0, 0, geom.width, width); | ||||
|             Cogl.rectangle(geom.width - width, width, | ||||
|                            geom.width, geom.height); | ||||
|             Cogl.rectangle(0, geom.height, | ||||
|                            geom.width - width, geom.height - width); | ||||
|             Cogl.rectangle(0, geom.height - width, | ||||
|                            width, width); | ||||
|         }); | ||||
|  | ||||
|         let geom = actor.get_allocation_geometry(); | ||||
|         let width = 2; | ||||
|  | ||||
|         // clockwise order | ||||
|         Cogl.rectangle(0, 0, geom.width, width); | ||||
|         Cogl.rectangle(geom.width - width, width, | ||||
|                        geom.width, geom.height); | ||||
|         Cogl.rectangle(0, geom.height, | ||||
|                        geom.width - width, geom.height - width); | ||||
|         Cogl.rectangle(0, geom.height - width, | ||||
|                        width, width); | ||||
|     }, | ||||
| }); | ||||
|     actor.queue_redraw(); | ||||
|     return signalId; | ||||
| } | ||||
|  | ||||
| const Inspector = new Lang.Class({ | ||||
|     Name: 'Inspector', | ||||
|  | ||||
|     _init: function(lookingGlass) { | ||||
|     _init: function() { | ||||
|         let container = new Shell.GenericContainer({ width: 0, | ||||
|                                                      height: 0 }); | ||||
|         container.connect('allocate', Lang.bind(this, this._allocate)); | ||||
| @@ -516,6 +499,9 @@ const Inspector = new Lang.Class({ | ||||
|         this._displayText = new St.Label(); | ||||
|         eventHandler.add(this._displayText, { expand: true }); | ||||
|  | ||||
|         this._borderPaintTarget = null; | ||||
|         this._borderPaintId = null; | ||||
|         eventHandler.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|         eventHandler.connect('key-press-event', Lang.bind(this, this._onKeyPressEvent)); | ||||
|         eventHandler.connect('button-press-event', Lang.bind(this, this._onButtonPressEvent)); | ||||
|         eventHandler.connect('scroll-event', Lang.bind(this, this._onScrollEvent)); | ||||
| @@ -530,8 +516,6 @@ const Inspector = new Lang.Class({ | ||||
|         // out, or move the pointer outside of _pointerTarget. | ||||
|         this._target = null; | ||||
|         this._pointerTarget = null; | ||||
|  | ||||
|         this._lookingGlass = lookingGlass; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
| @@ -552,13 +536,18 @@ const Inspector = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _close: function() { | ||||
|         Clutter.ungrab_pointer(); | ||||
|         Clutter.ungrab_keyboard(); | ||||
|         Clutter.ungrab_pointer(this._eventHandler); | ||||
|         Clutter.ungrab_keyboard(this._eventHandler); | ||||
|         this._eventHandler.destroy(); | ||||
|         this._eventHandler = null; | ||||
|         this.emit('closed'); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         if (this._borderPaintTarget != null) | ||||
|             this._borderPaintTarget.disconnect(this._borderPaintId); | ||||
|     }, | ||||
|  | ||||
|     _onKeyPressEvent: function (actor, event) { | ||||
|         if (event.get_key_symbol() == Clutter.Escape) | ||||
|             this._close(); | ||||
| @@ -627,12 +616,56 @@ const Inspector = new Lang.Class({ | ||||
|         this._displayText.text = ''; | ||||
|         this._displayText.text = position + ' ' + this._target; | ||||
|  | ||||
|         this._lookingGlass.setBorderPaintTarget(this._target); | ||||
|         if (this._borderPaintTarget != this._target) { | ||||
|             if (this._borderPaintTarget != null) | ||||
|                 this._borderPaintTarget.disconnect(this._borderPaintId); | ||||
|             this._borderPaintTarget = this._target; | ||||
|             this._borderPaintId = addBorderPaintHook(this._target); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|  | ||||
| Signals.addSignalMethods(Inspector.prototype); | ||||
|  | ||||
| const ErrorLog = new Lang.Class({ | ||||
|     Name: 'ErrorLog', | ||||
|  | ||||
|     _init: function() { | ||||
|         this.actor = new St.BoxLayout(); | ||||
|         this.text = new St.Label(); | ||||
|         this.actor.add(this.text); | ||||
|         // We need to override StLabel's default ellipsization when | ||||
|         // using line_wrap; otherwise ClutterText's layout is going | ||||
|         // to constrain both the width and height, which prevents | ||||
|         // scrolling. | ||||
|         this.text.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this.text.clutter_text.line_wrap = true; | ||||
|         this.actor.connect('notify::mapped', Lang.bind(this, this._renderText)); | ||||
|     }, | ||||
|  | ||||
|     _formatTime: function(d){ | ||||
|         function pad(n) { return n < 10 ? '0' + n : n; } | ||||
|         return d.getUTCFullYear()+'-' | ||||
|             + pad(d.getUTCMonth()+1)+'-' | ||||
|             + pad(d.getUTCDate())+'T' | ||||
|             + pad(d.getUTCHours())+':' | ||||
|             + pad(d.getUTCMinutes())+':' | ||||
|             + pad(d.getUTCSeconds())+'Z'; | ||||
|     }, | ||||
|  | ||||
|     _renderText: function() { | ||||
|         if (!this.actor.mapped) | ||||
|             return; | ||||
|         let text = this.text.text; | ||||
|         let stack = Main._getAndClearErrorStack(); | ||||
|         for (let i = 0; i < stack.length; i++) { | ||||
|             let logItem = stack[i]; | ||||
|             text += logItem.category + ' t=' + this._formatTime(new Date(logItem.timestamp)) + ' ' + logItem.message + '\n'; | ||||
|         } | ||||
|         this.text.text = text; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const Memory = new Lang.Class({ | ||||
|     Name: 'Memory', | ||||
|  | ||||
| @@ -661,7 +694,7 @@ const Memory = new Lang.Class({ | ||||
|  | ||||
|         this._gcbutton = new St.Button({ label: 'Full GC', | ||||
|                                          style_class: 'lg-obj-inspector-button' }); | ||||
|         this._gcbutton.connect('clicked', Lang.bind(this, function () { System.gc(); this._renderText(); })); | ||||
|         this._gcbutton.connect('clicked', Lang.bind(this, function () { global.gc(); this._renderText(); })); | ||||
|         this.actor.add(this._gcbutton, { x_align: St.Align.START, | ||||
|                                          x_fill: false }); | ||||
|  | ||||
| @@ -722,13 +755,13 @@ const Extensions = new Lang.Class({ | ||||
|         let extension = actor._extension; | ||||
|         let uri = extension.dir.get_uri(); | ||||
|         Gio.app_info_launch_default_for_uri(uri, global.create_app_launch_context()); | ||||
|         this._lookingGlass.close(); | ||||
|         Main.lookingGlass.close(); | ||||
|     }, | ||||
|  | ||||
|     _onWebPage: function (actor) { | ||||
|         let extension = actor._extension; | ||||
|         Gio.app_info_launch_default_for_uri(extension.metadata.url, global.create_app_launch_context()); | ||||
|         this._lookingGlass.close(); | ||||
|         Main.lookingGlass.close(); | ||||
|     }, | ||||
|  | ||||
|     _onViewErrors: function (actor) { | ||||
| @@ -792,33 +825,24 @@ const Extensions = new Lang.Class({ | ||||
|                                    text: this._stateToString(extension.state) }); | ||||
|         metaBox.add(state); | ||||
|  | ||||
|         let viewsource = new St.Button({ reactive: true, | ||||
|                                          track_hover: true, | ||||
|                                          style_class: 'shell-link', | ||||
|                                          label: _("View Source") }); | ||||
|         viewsource._extension = extension; | ||||
|         viewsource.connect('clicked', Lang.bind(this, this._onViewSource)); | ||||
|         metaBox.add(viewsource); | ||||
|         let viewsource = new Link.Link({ label: _("View Source") }); | ||||
|         viewsource.actor._extension = extension; | ||||
|         viewsource.actor.connect('clicked', Lang.bind(this, this._onViewSource)); | ||||
|         metaBox.add(viewsource.actor); | ||||
|  | ||||
|         if (extension.metadata.url) { | ||||
|             let webpage = new St.Button({ reactive: true, | ||||
|                                           track_hover: true, | ||||
|                                           style_class: 'shell-link', | ||||
|                                           label: _("Web Page") }); | ||||
|             webpage._extension = extension; | ||||
|             webpage.connect('clicked', Lang.bind(this, this._onWebPage)); | ||||
|             metaBox.add(webpage); | ||||
|             let webpage = new Link.Link({ label: _("Web Page") }); | ||||
|             webpage.actor._extension = extension; | ||||
|             webpage.actor.connect('clicked', Lang.bind(this, this._onWebPage)); | ||||
|             metaBox.add(webpage.actor); | ||||
|         } | ||||
|  | ||||
|         let viewerrors = new St.Button({ reactive: true, | ||||
|                                          track_hover: true, | ||||
|                                          style_class: 'shell-link', | ||||
|                                          label: _("Show Errors") }); | ||||
|         viewerrors._extension = extension; | ||||
|         viewerrors._parentBox = box; | ||||
|         viewerrors._isShowing = false; | ||||
|         viewerrors.connect('clicked', Lang.bind(this, this._onViewErrors)); | ||||
|         metaBox.add(viewerrors); | ||||
|         let viewerrors = new Link.Link({ label: _("Show Errors") }); | ||||
|         viewerrors.actor._extension = extension; | ||||
|         viewerrors.actor._parentBox = box; | ||||
|         viewerrors.actor._isShowing = false; | ||||
|         viewerrors.actor.connect('clicked', Lang.bind(this, this._onViewErrors)); | ||||
|         metaBox.add(viewerrors.actor); | ||||
|  | ||||
|         return box; | ||||
|     } | ||||
| @@ -829,7 +853,8 @@ const LookingGlass = new Lang.Class({ | ||||
|  | ||||
|     _init : function() { | ||||
|         this._borderPaintTarget = null; | ||||
|         this._redBorderEffect = new RedBorderEffect(); | ||||
|         this._borderPaintId = 0; | ||||
|         this._borderDestroyId = 0; | ||||
|  | ||||
|         this._open = false; | ||||
|  | ||||
| @@ -859,20 +884,22 @@ const LookingGlass = new Lang.Class({ | ||||
|         Main.layoutManager.keyboardBox.connect('allocation-changed', | ||||
|                                                Lang.bind(this, this._queueResize)); | ||||
|  | ||||
|         this._objInspector = new ObjInspector(this); | ||||
|         this._objInspector = new ObjInspector(); | ||||
|         Main.uiGroup.add_actor(this._objInspector.actor); | ||||
|         this._objInspector.actor.hide(); | ||||
|  | ||||
|         let toolbar = new St.BoxLayout({ name: 'Toolbar' }); | ||||
|         this.actor.add_actor(toolbar); | ||||
|         let inspectIcon = new St.Icon({ icon_name: 'gtk-color-picker', | ||||
|                                         icon_type: St.IconType.FULLCOLOR, | ||||
|                                         icon_size: 24 }); | ||||
|         toolbar.add_actor(inspectIcon); | ||||
|         inspectIcon.reactive = true; | ||||
|         inspectIcon.connect('button-press-event', Lang.bind(this, function () { | ||||
|             let inspector = new Inspector(this); | ||||
|             let inspector = new Inspector(); | ||||
|             inspector.connect('target', Lang.bind(this, function(i, target, stageX, stageY) { | ||||
|                 this._pushResult('inspect(' + Math.round(stageX) + ', ' + Math.round(stageY) + ')', target); | ||||
|                 this._pushResult('<inspect x:' + stageX + ' y:' + stageY + '>', | ||||
|                                  target); | ||||
|             })); | ||||
|             inspector.connect('closed', Lang.bind(this, function() { | ||||
|                 this.actor.show(); | ||||
| @@ -899,16 +926,23 @@ const LookingGlass = new Lang.Class({ | ||||
|         this._entryArea = new St.BoxLayout({ name: 'EntryArea' }); | ||||
|         this._evalBox.add_actor(this._entryArea); | ||||
|  | ||||
|         let label = new St.Label({ text: CHEVRON }); | ||||
|         let label = new St.Label({ text: 'js>>> ' }); | ||||
|         this._entryArea.add(label); | ||||
|  | ||||
|         this._entry = new St.Entry({ can_focus: true }); | ||||
|         ShellEntry.addContextMenu(this._entry); | ||||
|         this._entryArea.add(this._entry, { expand: true }); | ||||
|  | ||||
|         this._windowList = new WindowList(this); | ||||
|         this._windowList = new WindowList(); | ||||
|         this._windowList.connect('selected', Lang.bind(this, function(list, window) { | ||||
|             notebook.selectIndex(0); | ||||
|             this._pushResult('<window selection>', window); | ||||
|         })); | ||||
|         notebook.appendPage('Windows', this._windowList.actor); | ||||
|  | ||||
|         this._errorLog = new ErrorLog(); | ||||
|         notebook.appendPage('Errors', this._errorLog.actor); | ||||
|  | ||||
|         this._memory = new Memory(); | ||||
|         notebook.appendPage('Memory', this._memory.actor); | ||||
|  | ||||
| @@ -950,7 +984,9 @@ const LookingGlass = new Lang.Class({ | ||||
|  | ||||
|     _updateFont: function() { | ||||
|         let fontName = this._interfaceSettings.get_string('monospace-font-name'); | ||||
|         let fontDesc = Pango.FontDescription.from_string(fontName); | ||||
|         // This is mishandled by the scanner - should by Pango.FontDescription_from_string(fontName); | ||||
|         // https://bugzilla.gnome.org/show_bug.cgi?id=595889 | ||||
|         let fontDesc = Pango.font_description_from_string(fontName); | ||||
|         // We ignore everything but size and style; you'd be crazy to set your system-wide | ||||
|         // monospace font to be bold/oblique/etc. Could easily be added here. | ||||
|         this.actor.style = | ||||
| @@ -958,22 +994,23 @@ const LookingGlass = new Lang.Class({ | ||||
|             + 'font-family: "' + fontDesc.get_family() + '";'; | ||||
|     }, | ||||
|  | ||||
|     setBorderPaintTarget: function(obj) { | ||||
|         if (this._borderPaintTarget != null) | ||||
|             this._borderPaintTarget.remove_effect(this._redBorderEffect); | ||||
|         this._borderPaintTarget = obj; | ||||
|         if (this._borderPaintTarget != null) | ||||
|             this._borderPaintTarget.add_effect(this._redBorderEffect); | ||||
|     }, | ||||
|  | ||||
|     _pushResult: function(command, obj) { | ||||
|         let index = this._results.length + this._offset; | ||||
|         let result = new Result(this, CHEVRON + command, obj, index); | ||||
|         let result = new Result('>>> ' + command, obj, index); | ||||
|         this._results.push(result); | ||||
|         this._resultsArea.add(result.actor); | ||||
|         if (obj instanceof Clutter.Actor) | ||||
|             this.setBorderPaintTarget(obj); | ||||
|  | ||||
|         if (this._borderPaintTarget != null) { | ||||
|             this._borderPaintTarget.disconnect(this._borderPaintId); | ||||
|             this._borderPaintTarget = null; | ||||
|         } | ||||
|         if (obj instanceof Clutter.Actor) { | ||||
|             this._borderPaintTarget = obj; | ||||
|             this._borderPaintId = addBorderPaintHook(obj); | ||||
|             this._borderDestroyId = obj.connect('destroy', Lang.bind(this, function () { | ||||
|                 this._borderDestroyId = 0; | ||||
|                 this._borderPaintTarget = null; | ||||
|             })); | ||||
|         } | ||||
|         let children = this._resultsArea.get_children(); | ||||
|         if (children.length > this._maxItems) { | ||||
|             this._results.shift(); | ||||
| @@ -1055,10 +1092,6 @@ const LookingGlass = new Lang.Class({ | ||||
|         this._entry.text = ''; | ||||
|     }, | ||||
|  | ||||
|     inspect: function(x, y) { | ||||
|         return global.stage.get_actor_at_pos(Clutter.PickMode.REACTIVE, x, y); | ||||
|     }, | ||||
|  | ||||
|     getIt: function () { | ||||
|         return this._it; | ||||
|     }, | ||||
| @@ -1091,8 +1124,8 @@ const LookingGlass = new Lang.Class({ | ||||
|         this.actor.width = myWidth; | ||||
|         this.actor.height = myHeight; | ||||
|         this._objInspector.actor.set_size(Math.floor(myWidth * 0.8), Math.floor(myHeight * 0.8)); | ||||
|         this._objInspector.actor.set_position(primary.x + this.actor.x + Math.floor(myWidth * 0.1), | ||||
|                                               primary.y + this._targetY + Math.floor(myHeight * 0.1)); | ||||
|         this._objInspector.actor.set_position(this.actor.x + Math.floor(myWidth * 0.1), | ||||
|                                               this._targetY + Math.floor(myHeight * 0.1)); | ||||
|     }, | ||||
|  | ||||
|     insertObject: function(obj) { | ||||
| @@ -1131,7 +1164,7 @@ const LookingGlass = new Lang.Class({ | ||||
|         if (this._open) | ||||
|             return; | ||||
|  | ||||
|         if (!Main.pushModal(this._entry, { keybindingMode: Shell.KeyBindingMode.LOOKING_GLASS })) | ||||
|         if (!Main.pushModal(this._entry)) | ||||
|             return; | ||||
|  | ||||
|         this._notebook.selectIndex(0); | ||||
| @@ -1158,7 +1191,11 @@ const LookingGlass = new Lang.Class({ | ||||
|         this._open = false; | ||||
|         Tweener.removeTweens(this.actor); | ||||
|  | ||||
|         this.setBorderPaintTarget(null); | ||||
|         if (this._borderPaintTarget != null) { | ||||
|             this._borderPaintTarget.disconnect(this._borderPaintId); | ||||
|             this._borderPaintTarget.disconnect(this._borderDestroyId); | ||||
|             this._borderPaintTarget = null; | ||||
|         } | ||||
|  | ||||
|         Main.popModal(this._entry); | ||||
|  | ||||
|   | ||||
| @@ -12,11 +12,10 @@ const Signals = imports.signals; | ||||
| const Main = imports.ui.main; | ||||
| const MagnifierDBus = imports.ui.magnifierDBus; | ||||
| const Params = imports.misc.params; | ||||
| const PointerWatcher = imports.ui.pointerWatcher; | ||||
|  | ||||
|  | ||||
| const MOUSE_POLL_FREQUENCY = 50; | ||||
| const CROSSHAIRS_CLIP_SIZE = [100, 100]; | ||||
| const NO_CHANGE = 0.0; | ||||
|  | ||||
| // Settings | ||||
| const APPLICATIONS_SCHEMA       = 'org.gnome.desktop.a11y.applications'; | ||||
| @@ -25,14 +24,6 @@ const SHOW_KEY                  = 'screen-magnifier-enabled'; | ||||
| const MAGNIFIER_SCHEMA          = 'org.gnome.desktop.a11y.magnifier'; | ||||
| const SCREEN_POSITION_KEY       = 'screen-position'; | ||||
| const MAG_FACTOR_KEY            = 'mag-factor'; | ||||
| const INVERT_LIGHTNESS_KEY      = 'invert-lightness'; | ||||
| const COLOR_SATURATION_KEY      = 'color-saturation'; | ||||
| const BRIGHT_RED_KEY            = 'brightness-red'; | ||||
| const BRIGHT_GREEN_KEY          = 'brightness-green'; | ||||
| const BRIGHT_BLUE_KEY           = 'brightness-blue'; | ||||
| const CONTRAST_RED_KEY          = 'contrast-red'; | ||||
| const CONTRAST_GREEN_KEY        = 'contrast-green'; | ||||
| const CONTRAST_BLUE_KEY         = 'contrast-blue'; | ||||
| const LENS_MODE_KEY             = 'lens-mode'; | ||||
| const CLAMP_MODE_KEY            = 'scroll-at-edges'; | ||||
| const MOUSE_TRACKING_KEY        = 'mouse-tracking'; | ||||
| @@ -56,7 +47,7 @@ const Magnifier = new Lang.Class({ | ||||
|         let xfixesCursor = Shell.XFixesCursor.get_for_stage(global.stage); | ||||
|         this._mouseSprite = new Clutter.Texture(); | ||||
|         xfixesCursor.update_texture_image(this._mouseSprite); | ||||
|         this._cursorRoot = new Clutter.Actor(); | ||||
|         this._cursorRoot = new Clutter.Group(); | ||||
|         this._cursorRoot.add_actor(this._mouseSprite); | ||||
|  | ||||
|         // Create the first ZoomRegion and initialize it according to the | ||||
| @@ -136,8 +127,11 @@ const Magnifier = new Lang.Class({ | ||||
|      * Turn on mouse tracking, if not already doing so. | ||||
|      */ | ||||
|     startTrackingMouse: function() { | ||||
|         if (!this._pointerWatch) | ||||
|             this._pointerWatch = PointerWatcher.getPointerWatcher().addWatch(MOUSE_POLL_FREQUENCY, Lang.bind(this, this.scrollToMousePos)); | ||||
|         if (!this._mouseTrackingId) | ||||
|             this._mouseTrackingId = Mainloop.timeout_add( | ||||
|                 MOUSE_POLL_FREQUENCY, | ||||
|                 Lang.bind(this, this.scrollToMousePos) | ||||
|             ); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -145,10 +139,10 @@ const Magnifier = new Lang.Class({ | ||||
|      * Turn off mouse tracking, if not already doing so. | ||||
|      */ | ||||
|     stopTrackingMouse: function() { | ||||
|         if (this._pointerWatch) | ||||
|             this._pointerWatch.remove(); | ||||
|         if (this._mouseTrackingId) | ||||
|             Mainloop.source_remove(this._mouseTrackingId); | ||||
|  | ||||
|         this._pointerWatch = null; | ||||
|         this._mouseTrackingId = null; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -300,7 +294,8 @@ const Magnifier = new Lang.Class({ | ||||
|      */ | ||||
|     setCrosshairsColor: function(color) { | ||||
|         if (this._crossHairs) { | ||||
|             let [res, clutterColor] = Clutter.Color.from_string(color); | ||||
|             let clutterColor = new Clutter.Color(); | ||||
|             clutterColor.from_string(color); | ||||
|             this._crossHairs.setColor(clutterColor); | ||||
|         } | ||||
|     }, | ||||
| @@ -448,25 +443,6 @@ const Magnifier = new Lang.Class({ | ||||
|             aPref = this._settings.get_enum(MOUSE_TRACKING_KEY); | ||||
|             if (aPref) | ||||
|                 zoomRegion.setMouseTrackingMode(aPref); | ||||
|  | ||||
|             aPref = this._settings.get_boolean(INVERT_LIGHTNESS_KEY); | ||||
|             if (aPref) | ||||
|                 zoomRegion.setInvertLightness(aPref); | ||||
|  | ||||
|             aPref = this._settings.get_double(COLOR_SATURATION_KEY); | ||||
|             if (aPref) | ||||
|                 zoomRegion.setColorSaturation(aPref); | ||||
|  | ||||
|             let bc = {}; | ||||
|             bc.r = this._settings.get_double(BRIGHT_RED_KEY); | ||||
|             bc.g = this._settings.get_double(BRIGHT_GREEN_KEY); | ||||
|             bc.b = this._settings.get_double(BRIGHT_BLUE_KEY); | ||||
|             zoomRegion.setBrightness(bc); | ||||
|  | ||||
|             bc.r = this._settings.get_double(CONTRAST_RED_KEY); | ||||
|             bc.g = this._settings.get_double(CONTRAST_GREEN_KEY); | ||||
|             bc.b = this._settings.get_double(CONTRAST_BLUE_KEY); | ||||
|             zoomRegion.setContrast(bc); | ||||
|         } | ||||
|  | ||||
|         let showCrosshairs = this._settings.get_boolean(SHOW_CROSS_HAIRS_KEY); | ||||
| @@ -489,25 +465,6 @@ const Magnifier = new Lang.Class({ | ||||
|         this._settings.connect('changed::' + MOUSE_TRACKING_KEY, | ||||
|                                Lang.bind(this, this._updateMouseTrackingMode)); | ||||
|  | ||||
|         this._settings.connect('changed::' + INVERT_LIGHTNESS_KEY, | ||||
|                                Lang.bind(this, this._updateInvertLightness)); | ||||
|         this._settings.connect('changed::' + COLOR_SATURATION_KEY, | ||||
|                                Lang.bind(this, this._updateColorSaturation)); | ||||
|  | ||||
|         this._settings.connect('changed::' + BRIGHT_RED_KEY, | ||||
|                                Lang.bind(this, this._updateBrightness)); | ||||
|         this._settings.connect('changed::' + BRIGHT_GREEN_KEY, | ||||
|                                Lang.bind(this, this._updateBrightness)); | ||||
|         this._settings.connect('changed::' + BRIGHT_BLUE_KEY, | ||||
|                                Lang.bind(this, this._updateBrightness)); | ||||
|  | ||||
|         this._settings.connect('changed::' + CONTRAST_RED_KEY, | ||||
|                                Lang.bind(this, this._updateContrast)); | ||||
|         this._settings.connect('changed::' + CONTRAST_GREEN_KEY, | ||||
|                                Lang.bind(this, this._updateContrast)); | ||||
|         this._settings.connect('changed::' + CONTRAST_BLUE_KEY, | ||||
|                                Lang.bind(this, this._updateContrast)); | ||||
|  | ||||
|         this._settings.connect('changed::' + SHOW_CROSS_HAIRS_KEY, | ||||
|                                Lang.bind(this, function() { | ||||
|             this.setCrosshairsVisible(this._settings.get_boolean(SHOW_CROSS_HAIRS_KEY)); | ||||
| @@ -583,47 +540,7 @@ const Magnifier = new Lang.Class({ | ||||
|                 this._settings.get_enum(MOUSE_TRACKING_KEY) | ||||
|             ); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateInvertLightness: function() { | ||||
|         // Applies only to the first zoom region. | ||||
|         if (this._zoomRegions.length) { | ||||
|             this._zoomRegions[0].setInvertLightness( | ||||
|                 this._settings.get_boolean(INVERT_LIGHTNESS_KEY) | ||||
|             ); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateColorSaturation: function() { | ||||
|         // Applies only to the first zoom region. | ||||
|         if (this._zoomRegions.length) { | ||||
|             this._zoomRegions[0].setColorSaturation( | ||||
|                 this._settings.get_double(COLOR_SATURATION_KEY) | ||||
|             ); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateBrightness: function() { | ||||
|         // Applies only to the first zoom region. | ||||
|         if (this._zoomRegions.length) { | ||||
|             let brightness = {}; | ||||
|             brightness.r = this._settings.get_double(BRIGHT_RED_KEY); | ||||
|             brightness.g = this._settings.get_double(BRIGHT_GREEN_KEY); | ||||
|             brightness.b = this._settings.get_double(BRIGHT_BLUE_KEY); | ||||
|             this._zoomRegions[0].setBrightness(brightness); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateContrast: function() { | ||||
|         // Applies only to the first zoom region. | ||||
|         if (this._zoomRegions.length) { | ||||
|             let contrast = {}; | ||||
|             contrast.r = this._settings.get_double(CONTRAST_RED_KEY); | ||||
|             contrast.g = this._settings.get_double(CONTRAST_GREEN_KEY); | ||||
|             contrast.b = this._settings.get_double(CONTRAST_BLUE_KEY); | ||||
|             this._zoomRegions[0].setContrast(contrast); | ||||
|         } | ||||
|     }, | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Magnifier.prototype); | ||||
|  | ||||
| @@ -637,10 +554,6 @@ const ZoomRegion = new Lang.Class({ | ||||
|         this._clampScrollingAtEdges = false; | ||||
|         this._lensMode = false; | ||||
|         this._screenPosition = GDesktopEnums.MagnifierScreenPosition.FULL_SCREEN; | ||||
|         this._invertLightness = false; | ||||
|         this._colorSaturation = 1.0; | ||||
|         this._brightness = { r: NO_CHANGE, g: NO_CHANGE, b: NO_CHANGE }; | ||||
|         this._contrast = { r: NO_CHANGE, g: NO_CHANGE, b: NO_CHANGE }; | ||||
|  | ||||
|         this._magView = null; | ||||
|         this._background = null; | ||||
| @@ -966,107 +879,6 @@ const ZoomRegion = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setInvertLightness: | ||||
|      * Set whether to invert the lightness of the magnified view. | ||||
|      * @flag    Boolean to either invert brightness (true), or not (false). | ||||
|      */ | ||||
|     setInvertLightness: function(flag) { | ||||
|         this._invertLightness = flag; | ||||
|         if (this._magShaderEffects) | ||||
|             this._magShaderEffects.setInvertLightness(this._invertLightness); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getInvertLightness: | ||||
|      * Retrieve whether the lightness is inverted. | ||||
|      * @return    Boolean indicating inversion (true), or not (false). | ||||
|      */ | ||||
|     getInvertLightness: function() { | ||||
|         return this._invertLightness; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setColorSaturation: | ||||
|      * Set the color saturation of the magnified view. | ||||
|      * @sauration  A value from 0.0 to 1.0 that defines the color | ||||
|      *             saturation, with 0.0 defining no color (grayscale), | ||||
|      *             and 1.0 defining full color. | ||||
|      */ | ||||
|     setColorSaturation: function(saturation) { | ||||
|         this._colorSaturation = saturation; | ||||
|         if (this._magShaderEffects) | ||||
|             this._magShaderEffects.setColorSaturation(this._colorSaturation); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getColorSaturation: | ||||
|      * Retrieve the color saturation of the magnified view. | ||||
|      */ | ||||
|     getColorSaturation: function() { | ||||
|         return this._colorSaturation; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setBrightness: | ||||
|      * Alter the brightness of the magnified view. | ||||
|      * @brightness  Object containing the contrast for the red, green, | ||||
|      *              and blue channels.  Values of 0.0 represent "standard" | ||||
|      *              brightness (no change), whereas values less or greater than | ||||
|      *              0.0 indicate decreased or incresaed brightness, respectively. | ||||
|      */ | ||||
|     setBrightness: function(brightness) { | ||||
|         this._brightness.r = brightness.r; | ||||
|         this._brightness.g = brightness.g; | ||||
|         this._brightness.b = brightness.b; | ||||
|         if (this._magShaderEffects) | ||||
|             this._magShaderEffects.setBrightness(this._brightness); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getBrightness: | ||||
|      * Retrive the current brightness of the Zoom Region. | ||||
|      * @return  Object containing the brightness change for the red, green, | ||||
|      *          and blue channels. | ||||
|      */ | ||||
|     getBrightness: function() { | ||||
|         let brightness = {}; | ||||
|         brightness.r = this._brightness.r; | ||||
|         brightness.g = this._brightness.g; | ||||
|         brightness.b = this._brightness.b; | ||||
|         return brightness; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setContrast: | ||||
|      * Alter the contrast of the magnified view. | ||||
|      * @contrast    Object containing the contrast for the red, green, | ||||
|      *              and blue channels.  Values of 0.0 represent "standard" | ||||
|      *              contrast (no change), whereas values less or greater than | ||||
|      *              0.0 indicate decreased or incresaed contrast, respectively. | ||||
|      */ | ||||
|     setContrast: function(contrast) { | ||||
|         this._contrast.r = contrast.r; | ||||
|         this._contrast.g = contrast.g; | ||||
|         this._contrast.b = contrast.b; | ||||
|         if (this._magShaderEffects) | ||||
|             this._magShaderEffects.setContrast(this._contrast); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getContrast: | ||||
|      * Retreive the contrast of the magnified view. | ||||
|      * @return  Object containing the contrast for the red, green, | ||||
|      *          and blue channels. | ||||
|      */ | ||||
|     getContrast: function() { | ||||
|         let contrast = {}; | ||||
|         contrast.r = this._contrast.r; | ||||
|         contrast.g = this._contrast.g; | ||||
|         contrast.b = this._contrast.b; | ||||
|         return contrast; | ||||
|     }, | ||||
|  | ||||
|     //// Private methods //// | ||||
|  | ||||
|     _createActors: function() { | ||||
| @@ -1077,21 +889,21 @@ const ZoomRegion = new Lang.Class({ | ||||
|         // hide the magnified region from CLUTTER_PICK_ALL | ||||
|         Shell.util_set_hidden_from_pick (this._magView, true); | ||||
|  | ||||
|         // Add a group to clip the contents of the magnified view. | ||||
|         let mainGroup = new Clutter.Actor({ clip_to_allocation: true }); | ||||
|         // Append a Clutter.Group to clip the contents of the magnified view. | ||||
|         let mainGroup = new Clutter.Group({ clip_to_allocation: true }); | ||||
|         this._magView.set_child(mainGroup); | ||||
|  | ||||
|         // Add a background for when the magnified uiGroup is scrolled | ||||
|         // out of view (don't want to see desktop showing through). | ||||
|         this._background = new Clutter.Actor({ background_color: Main.DEFAULT_BACKGROUND_COLOR, | ||||
|                                                width: global.screen_width, | ||||
|                                                height: global.screen_height }); | ||||
|         this._background = new Clutter.Rectangle({ color: Main.DEFAULT_BACKGROUND_COLOR }); | ||||
|         mainGroup.add_actor(this._background); | ||||
|  | ||||
|         // Clone the group that contains all of UI on the screen.  This is the | ||||
|         // chrome, the windows, etc. | ||||
|         this._uiGroupClone = new Clutter.Clone({ source: Main.uiGroup }); | ||||
|         mainGroup.add_actor(this._uiGroupClone); | ||||
|         Main.uiGroup.set_size(global.screen_width, global.screen_height); | ||||
|         this._background.set_size(global.screen_width, global.screen_height); | ||||
|  | ||||
|         // Add either the given mouseSourceActor to the ZoomRegion, or a clone of | ||||
|         // it. | ||||
| @@ -1105,13 +917,6 @@ const ZoomRegion = new Lang.Class({ | ||||
|             this._crossHairsActor = this._crossHairs.addToZoomRegion(this, this._mouseActor); | ||||
|         else | ||||
|             this._crossHairsActor = null; | ||||
|  | ||||
|         // Contrast and brightness effects. | ||||
|         this._magShaderEffects = new MagShaderEffects(this._uiGroupClone); | ||||
|         this._magShaderEffects.setColorSaturation(this._colorSaturation); | ||||
|         this._magShaderEffects.setInvertLightness(this._invertLightness); | ||||
|         this._magShaderEffects.setBrightness(this._brightness); | ||||
|         this._magShaderEffects.setContrast(this._contrast); | ||||
|     }, | ||||
|  | ||||
|     _destroyActors: function() { | ||||
| @@ -1120,8 +925,6 @@ const ZoomRegion = new Lang.Class({ | ||||
|         if (this._crossHairs) | ||||
|             this._crossHairs.removeFromParent(this._crossHairsActor); | ||||
|  | ||||
|         this._magShaderEffects.destroyEffects(); | ||||
|         this._magShaderEffects = null; | ||||
|         this._magView.destroy(); | ||||
|         this._magView = null; | ||||
|         this._background = null; | ||||
| @@ -1325,7 +1128,7 @@ const ZoomRegion = new Lang.Class({ | ||||
|         this._mouseActor.set_scale(this._xMagFactor, this._yMagFactor); | ||||
|  | ||||
|         let [x, y] = this._screenToViewPort(0, 0); | ||||
|         this._uiGroupClone.set_position(Math.round(x), Math.round(y)); | ||||
|         this._uiGroupClone.set_position(x, y); | ||||
|  | ||||
|         this._updateMousePosition(); | ||||
|     }, | ||||
| @@ -1353,6 +1156,7 @@ const ZoomRegion = new Lang.Class({ | ||||
|         if (!this.isActive()) | ||||
|             return; | ||||
|  | ||||
|         Main.uiGroup.set_size(global.screen_width, global.screen_height); | ||||
|         this._background.set_size(global.screen_width, global.screen_height); | ||||
|  | ||||
|         if (this._screenPosition == GDesktopEnums.MagnifierScreenPosition.NONE) | ||||
| @@ -1376,15 +1180,15 @@ const Crosshairs = new Lang.Class({ | ||||
|         let groupWidth = global.screen_width * 3; | ||||
|         let groupHeight = global.screen_height * 3; | ||||
|  | ||||
|         this._actor = new Clutter.Actor({ | ||||
|         this._actor = new Clutter.Group({ | ||||
|             clip_to_allocation: false, | ||||
|             width: groupWidth, | ||||
|             height: groupHeight | ||||
|         }); | ||||
|         this._horizLeftHair = new Clutter.Actor(); | ||||
|         this._horizRightHair = new Clutter.Actor(); | ||||
|         this._vertTopHair = new Clutter.Actor(); | ||||
|         this._vertBottomHair = new Clutter.Actor(); | ||||
|         this._horizLeftHair = new Clutter.Rectangle(); | ||||
|         this._horizRightHair = new Clutter.Rectangle(); | ||||
|         this._vertTopHair = new Clutter.Rectangle(); | ||||
|         this._vertBottomHair = new Clutter.Rectangle(); | ||||
|         this._actor.add_actor(this._horizLeftHair); | ||||
|         this._actor.add_actor(this._horizRightHair); | ||||
|         this._actor.add_actor(this._vertTopHair); | ||||
| @@ -1424,7 +1228,10 @@ const Crosshairs = new Lang.Class({ | ||||
|                     crosshairsActor = new Clutter.Clone({ source: this._actor }); | ||||
|                     this._clones.push(crosshairsActor); | ||||
|                 } | ||||
|                 crosshairsActor.visible = this._actor.visible; | ||||
|                 if (this._actor.visible) | ||||
|                     crosshairsActor.show(); | ||||
|                 else | ||||
|                     crosshairsActor.hide(); | ||||
|  | ||||
|                 container.add_actor(crosshairsActor); | ||||
|                 container.raise_child(magnifiedMouse, crosshairsActor); | ||||
| @@ -1455,10 +1262,10 @@ const Crosshairs = new Lang.Class({ | ||||
|      * @clutterColor:   The color as a Clutter.Color. | ||||
|      */ | ||||
|     setColor: function(clutterColor) { | ||||
|         this._horizLeftHair.background_color = clutterColor; | ||||
|         this._horizRightHair.background_color = clutterColor; | ||||
|         this._vertTopHair.background_color = clutterColor; | ||||
|         this._vertBottomHair.background_color = clutterColor; | ||||
|         this._horizLeftHair.set_color(clutterColor); | ||||
|         this._horizRightHair.set_color(clutterColor); | ||||
|         this._vertTopHair.set_color(clutterColor); | ||||
|         this._vertBottomHair.set_color(clutterColor); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -1467,7 +1274,9 @@ const Crosshairs = new Lang.Class({ | ||||
|      * @color:  The color as a Clutter.Color. | ||||
|      */ | ||||
|     getColor: function() { | ||||
|         return this._horizLeftHair.get_color(); | ||||
|         let clutterColor = new Clutter.Color(); | ||||
|         this._horizLeftHair.get_color(clutterColor); | ||||
|         return clutterColor; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -1627,144 +1436,3 @@ const Crosshairs = new Lang.Class({ | ||||
|         this._vertBottomHair.set_position((groupWidth - thickness) / 2, bottom); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const MagShaderEffects = new Lang.Class({ | ||||
|     Name: 'MagShaderEffects', | ||||
|  | ||||
|     _init: function(uiGroupClone) { | ||||
|         this._inverse = new Shell.InvertLightnessEffect(); | ||||
|         this._brightnessContrast = new Clutter.BrightnessContrastEffect(); | ||||
|         this._colorDesaturation = new Clutter.DesaturateEffect(); | ||||
|         this._inverse.set_enabled(false); | ||||
|         this._brightnessContrast.set_enabled(false); | ||||
|  | ||||
|         this._magView = uiGroupClone; | ||||
|         this._magView.add_effect(this._inverse); | ||||
|         this._magView.add_effect(this._brightnessContrast); | ||||
|         this._magView.add_effect(this._colorDesaturation); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * destroyEffects: | ||||
|      * Remove contrast and brightness effects from the magnified view, and | ||||
|      * lose the reference to the actor they were applied to.  Don't use this | ||||
|      * object after calling this. | ||||
|      */ | ||||
|     destroyEffects: function() { | ||||
|         this._magView.clear_effects(); | ||||
|         this._colorDesaturation = null; | ||||
|         this._brightnessContrast = null; | ||||
|         this._inverse = null; | ||||
|         this._magView = null; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setInvertLightness: | ||||
|      * Enable/disable invert lightness effect. | ||||
|      * @invertFlag:     Enabled flag. | ||||
|      */ | ||||
|     setInvertLightness: function(invertFlag) { | ||||
|         this._inverse.set_enabled(invertFlag); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getInvertLightness: | ||||
|      * Report whether the inversion effect is enabled. | ||||
|      * @return:     Boolean. | ||||
|      */ | ||||
|     getInvertLightness: function() { | ||||
|         return this._inverse.get_enabled(); | ||||
|     }, | ||||
|  | ||||
|     setColorSaturation: function(factor) { | ||||
|         this._colorDesaturation.set_factor(1.0 - factor); | ||||
|     }, | ||||
|  | ||||
|     getColorSaturation: function() { | ||||
|         return 1.0 - this._colorDesaturation.get_factor(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setBrightness: | ||||
|      * Set the brightness of the magnified view. | ||||
|      * @brightness: Object containing the brightness for the red, green, | ||||
|      *              and blue channels.  Values of 0.0 represent "standard" | ||||
|      *              brightness (no change), whereas values less or greater than | ||||
|      *              0.0 indicate decreased or incresaed brightness, | ||||
|      *              respectively. | ||||
|      */ | ||||
|     setBrightness: function(brightness) { | ||||
|         let bRed = brightness.r; | ||||
|         let bGreen = brightness.g; | ||||
|         let bBlue = brightness.b; | ||||
|         this._brightnessContrast.set_brightness_full(bRed, bGreen, bBlue); | ||||
|  | ||||
|         // Enable the effect if the brightness OR contrast change are such that | ||||
|         // it modifies the brightness and/or contrast. | ||||
|         let [cRed, cGreen, cBlue] = this._brightnessContrast.get_contrast(); | ||||
|         this._brightnessContrast.set_enabled( | ||||
|             (bRed != NO_CHANGE || bGreen != NO_CHANGE || bBlue != NO_CHANGE || | ||||
|              cRed != NO_CHANGE || cGreen != NO_CHANGE || cBlue != NO_CHANGE) | ||||
|         ); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getBrightness: | ||||
|      * Retrieve current brightness of the magnified view. | ||||
|      * @return: Object containing the brightness for the red, green, | ||||
|      *          and blue channels.  Values of 0.0 represent "standard"  | ||||
|      *          brightness (no change), whereas values less or greater than | ||||
|      *          0.0 indicate decreased or incresaed brightness, respectively. | ||||
|      */ | ||||
|     getBrightness: function() { | ||||
|         let result = {}; | ||||
|         let [bRed, bGreen, bBlue] = this._brightnessContrast.get_brightness(); | ||||
|         result.r = bRed; | ||||
|         result.g = bGreen; | ||||
|         result.b = bBlue; | ||||
|  | ||||
|         return result; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * Set the contrast of the magnified view. | ||||
|      * @contrast:   Object containing the contrast for the red, green, | ||||
|      *              and blue channels.  Values of 0.0 represent "standard" | ||||
|      *              contrast (no change), whereas values less or greater than | ||||
|      *              0.0 indicate decreased or incresaed contrast, respectively. | ||||
|      */ | ||||
|     setContrast: function(contrast) { | ||||
|         let cRed = contrast.r; | ||||
|         let cGreen = contrast.g; | ||||
|         let cBlue = contrast.b; | ||||
|  | ||||
|         this._brightnessContrast.set_contrast_full(cRed, cGreen, cBlue); | ||||
|  | ||||
|         // Enable the effect if the contrast OR brightness change are such that | ||||
|         // it modifies the brightness and/or contrast. | ||||
|         // should be able to use Clutter.color_equal(), but that complains of | ||||
|         // a null first argument. | ||||
|         let [bRed, bGreen, bBlue] = this._brightnessContrast.get_brightness(); | ||||
|         this._brightnessContrast.set_enabled( | ||||
|              cRed != NO_CHANGE || cGreen != NO_CHANGE || cBlue != NO_CHANGE || | ||||
|              bRed != NO_CHANGE || bGreen != NO_CHANGE || bBlue != NO_CHANGE | ||||
|         ); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * Retrieve current contrast of the magnified view. | ||||
|      * @return: Object containing the contrast for the red, green, | ||||
|      *          and blue channels.  Values of 0.0 represent "standard" | ||||
|      *          contrast (no change), whereas values less or greater than | ||||
|      *          0.0 indicate decreased or incresaed contrast, respectively. | ||||
|      */ | ||||
|     getContrast: function() { | ||||
|         let resutl = {}; | ||||
|         let [cRed, cGreen, cBlue] = this._brightnessContrast.get_contrast(); | ||||
|         result.r = cRed; | ||||
|         result.g = cGreen; | ||||
|         result.b = cBlue; | ||||
|  | ||||
|         return result; | ||||
|     } | ||||
| }); | ||||
|   | ||||
							
								
								
									
										460
									
								
								js/ui/main.js
									
									
									
									
									
								
							
							
						
						| @@ -10,38 +10,44 @@ const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Components = imports.ui.components; | ||||
| const AutomountManager = imports.ui.automountManager; | ||||
| const AutorunManager = imports.ui.autorunManager; | ||||
| const CtrlAltTab = imports.ui.ctrlAltTab; | ||||
| const EndSessionDialog = imports.ui.endSessionDialog; | ||||
| const PolkitAuthenticationAgent = imports.ui.polkitAuthenticationAgent; | ||||
| const KeyringPrompt = imports.ui.keyringPrompt; | ||||
| const Environment = imports.ui.environment; | ||||
| const ExtensionSystem = imports.ui.extensionSystem; | ||||
| const ExtensionDownloader = imports.ui.extensionDownloader; | ||||
| const Keyboard = imports.ui.keyboard; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
| const Overview = imports.ui.overview; | ||||
| const Panel = imports.ui.panel; | ||||
| const Params = imports.misc.params; | ||||
| const PlaceDisplay = imports.ui.placeDisplay; | ||||
| const RunDialog = imports.ui.runDialog; | ||||
| const Layout = imports.ui.layout; | ||||
| const LookingGlass = imports.ui.lookingGlass; | ||||
| const NetworkAgent = imports.ui.networkAgent; | ||||
| const NotificationDaemon = imports.ui.notificationDaemon; | ||||
| const WindowAttentionHandler = imports.ui.windowAttentionHandler; | ||||
| const ScreenShield = imports.ui.screenShield; | ||||
| const Scripting = imports.ui.scripting; | ||||
| const SessionMode = imports.ui.sessionMode; | ||||
| const ShellDBus = imports.ui.shellDBus; | ||||
| const ShellMountOperation = imports.ui.shellMountOperation; | ||||
| const UnlockDialog = imports.ui.unlockDialog; | ||||
| const TelepathyClient = imports.ui.telepathyClient; | ||||
| const WindowManager = imports.ui.windowManager; | ||||
| const Magnifier = imports.ui.magnifier; | ||||
| const XdndHandler = imports.ui.xdndHandler; | ||||
| const StatusIconDispatcher = imports.ui.statusIconDispatcher; | ||||
| const Util = imports.misc.util; | ||||
|  | ||||
| const OVERRIDES_SCHEMA = 'org.gnome.shell.overrides'; | ||||
| const DEFAULT_BACKGROUND_COLOR = Clutter.Color.from_pixel(0x2e3436ff); | ||||
| const DEFAULT_BACKGROUND_COLOR = new Clutter.Color(); | ||||
| DEFAULT_BACKGROUND_COLOR.from_pixel(0x2266bbff); | ||||
|  | ||||
| let componentManager = null; | ||||
| let automountManager = null; | ||||
| let autorunManager = null; | ||||
| let panel = null; | ||||
| let hotCorners = []; | ||||
| let placesManager = null; | ||||
| let overview = null; | ||||
| let runDialog = null; | ||||
| let lookingGlass = null; | ||||
| @@ -50,64 +56,114 @@ let messageTray = null; | ||||
| let screenShield = null; | ||||
| let notificationDaemon = null; | ||||
| let windowAttentionHandler = null; | ||||
| let telepathyClient = null; | ||||
| let ctrlAltTabManager = null; | ||||
| let sessionMode = null; | ||||
| let recorder = null; | ||||
| let shellDBusService = null; | ||||
| let shellMountOpDBusService = null; | ||||
| let screenSaverDBus = null; | ||||
| let modalCount = 0; | ||||
| let keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
| let modalActorFocusStack = []; | ||||
| let uiGroup = null; | ||||
| let magnifier = null; | ||||
| let xdndHandler = null; | ||||
| let statusIconDispatcher = null; | ||||
| let keyboard = null; | ||||
| let layoutManager = null; | ||||
| let networkAgent = null; | ||||
| let _errorLogStack = []; | ||||
| let _startDate; | ||||
| let _defaultCssStylesheet = null; | ||||
| let _cssStylesheet = null; | ||||
| let _gdmCssStylesheet = null; | ||||
| let _overridesSettings = null; | ||||
|  | ||||
| function _sessionUpdated() { | ||||
|     wm.setCustomKeybindingHandler('panel-main-menu', | ||||
|                                   Shell.KeyBindingMode.NORMAL | | ||||
|                                   Shell.KeyBindingMode.OVERVIEW, | ||||
|                                   sessionMode.hasOverview ? Lang.bind(overview, overview.toggle) : null); | ||||
|     wm.allowKeybinding('overlay-key', Shell.KeyBindingMode.NORMAL | | ||||
|                                       Shell.KeyBindingMode.OVERVIEW); | ||||
| let background = null; | ||||
|  | ||||
| function _createUserSession() { | ||||
|     // Load the calendar server. Note that we are careful about | ||||
|     // not loading any events until the user presses the clock | ||||
|     global.launch_calendar_server(); | ||||
|  | ||||
|     placesManager = new PlaceDisplay.PlacesManager(); | ||||
|     telepathyClient = new TelepathyClient.Client(); | ||||
|     automountManager = new AutomountManager.AutomountManager(); | ||||
|     autorunManager = new AutorunManager.AutorunManager(); | ||||
|     networkAgent = new NetworkAgent.NetworkAgent(); | ||||
| } | ||||
|  | ||||
| function _createGDMSession() { | ||||
|     // We do this this here instead of at the top to prevent GDM | ||||
|     // related code from getting loaded in normal user sessions | ||||
|     const LoginDialog = imports.gdm.loginDialog; | ||||
|  | ||||
|     let loginDialog = new LoginDialog.LoginDialog(); | ||||
|     loginDialog.connect('loaded', function() { | ||||
|                             loginDialog.open(); | ||||
|                         }); | ||||
| } | ||||
|  | ||||
| function _initRecorder() { | ||||
|     let recorderSettings = new Gio.Settings({ schema: 'org.gnome.shell.recorder' }); | ||||
|  | ||||
|     global.screen.connect('toggle-recording', function() { | ||||
|         if (recorder == null) { | ||||
|             recorder = new Shell.Recorder({ stage: global.stage }); | ||||
|         } | ||||
|  | ||||
|         if (recorder.is_recording()) { | ||||
|             recorder.pause(); | ||||
|             Meta.enable_unredirect_for_screen(global.screen); | ||||
|         } else { | ||||
|             // read the parameters from GSettings always in case they have changed | ||||
|             recorder.set_framerate(recorderSettings.get_int('framerate')); | ||||
|             /* Translators: this is a filename used for screencast recording */ | ||||
|             // xgettext:no-c-format | ||||
|             recorder.set_filename(_("Screencast from %d %t") + '.' + recorderSettings.get_string('file-extension')); | ||||
|             let pipeline = recorderSettings.get_string('pipeline'); | ||||
|  | ||||
|             if (!pipeline.match(/^\s*$/)) | ||||
|                 recorder.set_pipeline(pipeline); | ||||
|             else | ||||
|                 recorder.set_pipeline(null); | ||||
|  | ||||
|             Meta.disable_unredirect_for_screen(global.screen); | ||||
|             recorder.record(); | ||||
|         } | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function _initUserSession() { | ||||
|     _initRecorder(); | ||||
|  | ||||
|     global.screen.override_workspace_layout(Meta.ScreenCorner.TOPLEFT, false, -1, 1); | ||||
|  | ||||
|     ExtensionSystem.init(); | ||||
|     ExtensionSystem.loadExtensions(); | ||||
|  | ||||
|     Meta.keybindings_set_custom_handler('panel-run-dialog', function() { | ||||
|        getRunDialog().open(); | ||||
|     }); | ||||
|  | ||||
|     Meta.keybindings_set_custom_handler('panel-main-menu', function () { | ||||
|         overview.toggle(); | ||||
|     }); | ||||
|  | ||||
|     global.display.connect('overlay-key', Lang.bind(overview, overview.toggle)); | ||||
|  | ||||
|     wm.setCustomKeybindingHandler('panel-run-dialog', | ||||
|                                   Shell.KeyBindingMode.NORMAL | | ||||
|                                   Shell.KeyBindingMode.OVERVIEW, | ||||
|                                   sessionMode.hasRunDialog ? openRunDialog : null); | ||||
|     if (sessionMode.isGreeter) | ||||
|         screenShield.showDialog(); | ||||
| } | ||||
|  | ||||
| function start() { | ||||
|     // These are here so we don't break compatibility. | ||||
|     global.logError = window.log; | ||||
|     global.log = window.log; | ||||
|     // Monkey patch utility functions into the global proxy; | ||||
|     // This is easier and faster than indirecting down into global | ||||
|     // if we want to call back up into JS. | ||||
|     global.logError = _logError; | ||||
|     global.log = _logDebug; | ||||
|  | ||||
|     // Chain up async errors reported from C | ||||
|     global.connect('notify-error', function (global, msg, detail) { notifyError(msg, detail); }); | ||||
|  | ||||
|     Gio.DesktopAppInfo.set_desktop_env('GNOME'); | ||||
|  | ||||
|     sessionMode = new SessionMode.SessionMode(); | ||||
|  | ||||
|     // start session after we know what mode we're running in | ||||
|     let signalId = sessionMode.connect('updated', function() { | ||||
|                                            sessionMode.disconnect(signalId); | ||||
|                                            startSession(); | ||||
|                                        }); | ||||
| } | ||||
|  | ||||
| function startSession() { | ||||
|     sessionMode.connect('updated', _loadDefaultStylesheet); | ||||
|  | ||||
|     shellDBusService = new ShellDBus.GnomeShell(); | ||||
|     shellMountOpDBusService = new ShellMountOperation.GnomeShellMountOpHandler(); | ||||
|  | ||||
|     // Ensure ShellWindowTracker and ShellAppUsage are initialized; this will | ||||
|     // also initialize ShellAppSystem first.  ShellAppSystem | ||||
| @@ -122,51 +178,76 @@ function startSession() { | ||||
|  | ||||
|     tracker.connect('startup-sequence-changed', _queueCheckWorkspaces); | ||||
|  | ||||
|     _loadDefaultStylesheet(); | ||||
|     // The stage is always covered so Clutter doesn't need to clear it; however | ||||
|     // the color is used as the default contents for the Mutter root background | ||||
|     // actor so set it anyways. | ||||
|     global.stage.color = DEFAULT_BACKGROUND_COLOR; | ||||
|     global.stage.no_clear_hint = true; | ||||
|  | ||||
|     _defaultCssStylesheet = global.datadir + '/theme/gnome-shell.css'; | ||||
|     _gdmCssStylesheet = global.datadir + '/theme/gdm.css'; | ||||
|     loadTheme(); | ||||
|  | ||||
|     // Set up stage hierarchy to group all UI actors under one container. | ||||
|     uiGroup = new Shell.GenericContainer({ name: 'uiGroup' }); | ||||
|     uiGroup.connect('allocate', | ||||
|                     function (actor, box, flags) { | ||||
|                         let children = uiGroup.get_children(); | ||||
|                         for (let i = 0; i < children.length; i++) | ||||
|                             children[i].allocate_preferred_size(flags); | ||||
|                     }); | ||||
|     let constraint = new Clutter.BindConstraint({ source: global.stage, | ||||
|                                                   coordinate: Clutter.BindCoordinate.SIZE }); | ||||
|     uiGroup.add_constraint(constraint); | ||||
|     global.window_group.reparent(uiGroup); | ||||
|     global.overlay_group.reparent(uiGroup); | ||||
|     global.stage.add_actor(uiGroup); | ||||
|  | ||||
|     // Setup the stage hierarchy early | ||||
|     layoutManager = new Layout.LayoutManager(); | ||||
|  | ||||
|     // Various parts of the codebase still refers to Main.uiGroup | ||||
|     // instead using the layoutManager.  This keeps that code | ||||
|     // working until it's updated. | ||||
|     uiGroup = layoutManager.uiGroup; | ||||
|  | ||||
|     xdndHandler = new XdndHandler.XdndHandler(); | ||||
|     ctrlAltTabManager = new CtrlAltTab.CtrlAltTabManager(); | ||||
|     overview = new Overview.Overview(); | ||||
|     wm = new WindowManager.WindowManager(); | ||||
|     // This overview object is just a stub for non-user sessions | ||||
|     overview = new Overview.Overview({ isDummy: global.session_type != Shell.SessionType.USER }); | ||||
|     magnifier = new Magnifier.Magnifier(); | ||||
|     if (UnlockDialog.isSupported()) | ||||
|         screenShield = new ScreenShield.ScreenShield(); | ||||
|     else | ||||
|         screenShield = new ScreenShield.ScreenShieldFallback(); | ||||
|  | ||||
|     // The message tray relies on being constructed | ||||
|     // after the panel. | ||||
|     statusIconDispatcher = new StatusIconDispatcher.StatusIconDispatcher(); | ||||
|     panel = new Panel.Panel(); | ||||
|     wm = new WindowManager.WindowManager(); | ||||
|     messageTray = new MessageTray.MessageTray(); | ||||
|     screenShield = new ScreenShield.ScreenShield(); | ||||
|     keyboard = new Keyboard.Keyboard(); | ||||
|     notificationDaemon = new NotificationDaemon.NotificationDaemon(); | ||||
|     windowAttentionHandler = new WindowAttentionHandler.WindowAttentionHandler(); | ||||
|     componentManager = new Components.ComponentManager(); | ||||
|  | ||||
|     if (global.session_type == Shell.SessionType.USER) | ||||
|         _createUserSession(); | ||||
|     else if (global.session_type == Shell.SessionType.GDM) | ||||
|         _createGDMSession(); | ||||
|  | ||||
|     panel.startStatusArea(); | ||||
|  | ||||
|     layoutManager.init(); | ||||
|     layoutManager.prepareStartupAnimation(); | ||||
|     keyboard.init(); | ||||
|     overview.init(); | ||||
|  | ||||
|     global.screen.override_workspace_layout(Meta.ScreenCorner.TOPLEFT, | ||||
|                                             false, -1, 1); | ||||
|     global.display.connect('overlay-key', Lang.bind(overview, overview.toggle)); | ||||
|     sessionMode.connect('updated', _sessionUpdated); | ||||
|     _sessionUpdated(); | ||||
|     if (global.session_type == Shell.SessionType.USER) | ||||
|         _initUserSession(); | ||||
|     statusIconDispatcher.start(messageTray.actor); | ||||
|  | ||||
|     // Provide the bus object for gnome-session to | ||||
|     // initiate logouts. | ||||
|     EndSessionDialog.init(); | ||||
|  | ||||
|     // Attempt to become a PolicyKit authentication agent | ||||
|     PolkitAuthenticationAgent.init() | ||||
|  | ||||
|     // Become a prompter for gnome keyring | ||||
|     KeyringPrompt.init(); | ||||
|  | ||||
|     _startDate = new Date(); | ||||
|  | ||||
|     global.stage.connect('captured-event', _globalKeyPressHandler); | ||||
|  | ||||
|     _log('info', 'loaded at ' + _startDate); | ||||
|     log('GNOME Shell started at ' + _startDate); | ||||
|  | ||||
|     let perfModuleName = GLib.getenv("SHELL_PERF_MODULE"); | ||||
| @@ -186,17 +267,6 @@ function startSession() { | ||||
|     global.screen.connect('restacked', _windowsRestacked); | ||||
|  | ||||
|     _nWorkspacesChanged(); | ||||
|  | ||||
|     ExtensionDownloader.init(); | ||||
|     ExtensionSystem.init(); | ||||
|  | ||||
|     // Run the startup animation as soon as the mainloop is idle enough. | ||||
|     // This is necessary to have it smooth and without interruptions from | ||||
|     // completed IO tasks | ||||
|     GLib.idle_add(GLib.PRIORITY_LOW, function() { | ||||
|         layoutManager.startupAnimation(); | ||||
|         return false; | ||||
|     }); | ||||
| } | ||||
|  | ||||
| let _workspaces = []; | ||||
| @@ -380,18 +450,6 @@ function _nWorkspacesChanged() { | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| function _loadDefaultStylesheet() { | ||||
|     if (!sessionMode.isPrimary) | ||||
|         return; | ||||
|  | ||||
|     let stylesheet = global.datadir + '/theme/' + sessionMode.stylesheetName; | ||||
|     if (_defaultCssStylesheet == stylesheet) | ||||
|         return; | ||||
|  | ||||
|     _defaultCssStylesheet = stylesheet; | ||||
|     loadTheme(); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * getThemeStylesheet: | ||||
|  * | ||||
| @@ -432,6 +490,9 @@ function loadTheme() { | ||||
|  | ||||
|     let theme = new St.Theme ({ application_stylesheet: cssStylesheet }); | ||||
|  | ||||
|     if (global.session_type == Shell.SessionType.GDM) | ||||
|         theme.load_stylesheet(_gdmCssStylesheet); | ||||
|  | ||||
|     if (previousTheme) { | ||||
|         let customStylesheets = previousTheme.get_custom_stylesheets(); | ||||
|  | ||||
| @@ -472,6 +533,138 @@ function notifyError(msg, details) { | ||||
|     notify(msg, details); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * _log: | ||||
|  * @category: string message type ('info', 'error') | ||||
|  * @msg: A message string | ||||
|  * ...: Any further arguments are converted into JSON notation, | ||||
|  *      and appended to the log message, separated by spaces. | ||||
|  * | ||||
|  * Log a message into the LookingGlass error | ||||
|  * stream.  This is primarily intended for use by the | ||||
|  * extension system as well as debugging. | ||||
|  */ | ||||
| function _log(category, msg) { | ||||
|     let text = msg; | ||||
|     if (arguments.length > 2) { | ||||
|         text += ': '; | ||||
|         for (let i = 2; i < arguments.length; i++) { | ||||
|             text += JSON.stringify(arguments[i]); | ||||
|             if (i < arguments.length - 1) | ||||
|                 text += ' '; | ||||
|         } | ||||
|     } | ||||
|     _errorLogStack.push({timestamp: new Date().getTime(), | ||||
|                          category: category, | ||||
|                          message: text }); | ||||
| } | ||||
|  | ||||
| function _logError(msg) { | ||||
|     return _log('error', msg); | ||||
| } | ||||
|  | ||||
| function _logDebug(msg) { | ||||
|     return _log('debug', msg); | ||||
| } | ||||
|  | ||||
| // Used by the error display in lookingGlass.js | ||||
| function _getAndClearErrorStack() { | ||||
|     let errors = _errorLogStack; | ||||
|     _errorLogStack = []; | ||||
|     return errors; | ||||
| } | ||||
|  | ||||
| function logStackTrace(msg) { | ||||
|     try { | ||||
|         throw new Error(); | ||||
|     } catch (e) { | ||||
|         // e.stack must have at least two lines, with the first being | ||||
|         // logStackTrace() (which we strip off), and the second being | ||||
|         // our caller. | ||||
|         let trace = e.stack.substr(e.stack.indexOf('\n') + 1); | ||||
|         log(msg ? (msg + '\n' + trace) : trace); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function isWindowActorDisplayedOnWorkspace(win, workspaceIndex) { | ||||
|     return win.get_workspace() == workspaceIndex || | ||||
|         (win.get_meta_window() && win.get_meta_window().is_on_all_workspaces()); | ||||
| } | ||||
|  | ||||
| function getWindowActorsForWorkspace(workspaceIndex) { | ||||
|     return global.get_window_actors().filter(function (win) { | ||||
|         return isWindowActorDisplayedOnWorkspace(win, workspaceIndex); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| // This function encapsulates hacks to make certain global keybindings | ||||
| // work even when we are in one of our modes where global keybindings | ||||
| // are disabled with a global grab. (When there is a global grab, then | ||||
| // all key events will be delivered to the stage, so ::captured-event | ||||
| // on the stage can be used for global keybindings.) | ||||
| function _globalKeyPressHandler(actor, event) { | ||||
|     if (modalCount == 0) | ||||
|         return false; | ||||
|     if (event.type() != Clutter.EventType.KEY_PRESS) | ||||
|         return false; | ||||
|  | ||||
|     let symbol = event.get_key_symbol(); | ||||
|     let keyCode = event.get_key_code(); | ||||
|     let ignoredModifiers = global.display.get_ignored_modifier_mask(); | ||||
|     let modifierState = event.get_state() & ~ignoredModifiers; | ||||
|  | ||||
|     // This relies on the fact that Clutter.ModifierType is the same as Gdk.ModifierType | ||||
|     let action = global.display.get_keybinding_action(keyCode, modifierState); | ||||
|  | ||||
|     // Other bindings are only available to the user session when the overview is up and | ||||
|     // no modal dialog is present. | ||||
|     if (global.session_type == Shell.SessionType.USER && (!overview.visible || modalCount > 1)) | ||||
|         return false; | ||||
|  | ||||
|     // This isn't a Meta.KeyBindingAction yet | ||||
|     if (symbol == Clutter.Super_L || symbol == Clutter.Super_R) { | ||||
|         overview.hide(); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (action == Meta.KeyBindingAction.SWITCH_PANELS) { | ||||
|         ctrlAltTabManager.popup(modifierState & Clutter.ModifierType.SHIFT_MASK, | ||||
|                                 modifierState); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     // None of the other bindings are relevant outside of the user's session | ||||
|     if (global.session_type != Shell.SessionType.USER) | ||||
|         return false; | ||||
|  | ||||
|     switch (action) { | ||||
|         // left/right would effectively act as synonyms for up/down if we enabled them; | ||||
|         // but that could be considered confusing; we also disable them in the main view. | ||||
|         // | ||||
|         // case Meta.KeyBindingAction.WORKSPACE_LEFT: | ||||
|         //     wm.actionMoveWorkspaceLeft(); | ||||
|         //     return true; | ||||
|         // case Meta.KeyBindingAction.WORKSPACE_RIGHT: | ||||
|         //     wm.actionMoveWorkspaceRight(); | ||||
|         //     return true; | ||||
|         case Meta.KeyBindingAction.WORKSPACE_UP: | ||||
|             wm.actionMoveWorkspaceUp(); | ||||
|             return true; | ||||
|         case Meta.KeyBindingAction.WORKSPACE_DOWN: | ||||
|             wm.actionMoveWorkspaceDown(); | ||||
|             return true; | ||||
|         case Meta.KeyBindingAction.PANEL_RUN_DIALOG: | ||||
|         case Meta.KeyBindingAction.COMMAND_2: | ||||
|             getRunDialog().open(); | ||||
|             return true; | ||||
|         case Meta.KeyBindingAction.PANEL_MAIN_MENU: | ||||
|             overview.hide(); | ||||
|             return true; | ||||
|     } | ||||
|  | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| function _findModal(actor) { | ||||
|     for (let i = 0; i < modalActorFocusStack.length; i++) { | ||||
|         if (modalActorFocusStack[i].actor == actor) | ||||
| @@ -480,10 +673,14 @@ function _findModal(actor) { | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| function isInModalStack(actor) { | ||||
|     return _findModal(actor) != -1; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * pushModal: | ||||
|  * @actor: #ClutterActor which will be given keyboard focus | ||||
|  * @params: optional parameters | ||||
|  * @timestamp: optional timestamp | ||||
|  * | ||||
|  * Ensure we are in a mode where all keyboard and mouse input goes to | ||||
|  * the stage, and focus @actor. Multiple calls to this function act in | ||||
| @@ -494,31 +691,24 @@ function _findModal(actor) { | ||||
|  * modal stack returns to this actor, reset the focus to the actor | ||||
|  * which was focused at the time pushModal() was invoked. | ||||
|  * | ||||
|  * @params may be used to provide the following parameters: | ||||
|  *  - timestamp: used to associate the call with a specific user initiated | ||||
|  *               event.  If not provided then the value of | ||||
|  *               global.get_current_time() is assumed. | ||||
|  * @timestamp is optionally used to associate the call with a specific user | ||||
|  * initiated event.  If not provided then the value of | ||||
|  * global.get_current_time() is assumed. | ||||
|  * | ||||
|  *  - options: Meta.ModalOptions flags to indicate that the pointer is | ||||
|  *             already grabbed | ||||
|  * | ||||
|  *  - keybindingMode: used to set the current Shell.KeyBindingMode to filter | ||||
|  *                    global keybindings; the default of NONE will filter | ||||
|  *                    out all keybindings | ||||
|  * @options: optional Meta.ModalOptions flags to indicate that the | ||||
|  *           pointer is alrady grabbed | ||||
|  * | ||||
|  * Returns: true iff we successfully acquired a grab or already had one | ||||
|  */ | ||||
| function pushModal(actor, params) { | ||||
|     params = Params.parse(params, { timestamp: global.get_current_time(), | ||||
|                                     options: 0, | ||||
|                                     keybindingMode: Shell.KeyBindingMode.NONE }); | ||||
| function pushModal(actor, timestamp, options) { | ||||
|     if (timestamp == undefined) | ||||
|         timestamp = global.get_current_time(); | ||||
|  | ||||
|     if (modalCount == 0) { | ||||
|         if (!global.begin_modal(params.timestamp, params.options)) { | ||||
|         if (!global.begin_modal(timestamp, options ? options : 0)) { | ||||
|             log('pushModal: invocation of begin_modal failed'); | ||||
|             return false; | ||||
|         } | ||||
|         Meta.disable_unredirect_for_screen(global.screen); | ||||
|     } | ||||
|  | ||||
|     global.set_stage_input_mode(Shell.StageInputMode.FULLSCREEN); | ||||
| @@ -527,25 +717,22 @@ function pushModal(actor, params) { | ||||
|     let actorDestroyId = actor.connect('destroy', function() { | ||||
|         let index = _findModal(actor); | ||||
|         if (index >= 0) | ||||
|             popModal(actor); | ||||
|             modalActorFocusStack.splice(index, 1); | ||||
|     }); | ||||
|  | ||||
|     let prevFocus = global.stage.get_key_focus(); | ||||
|     let prevFocusDestroyId; | ||||
|     if (prevFocus != null) { | ||||
|         prevFocusDestroyId = prevFocus.connect('destroy', function() { | ||||
|     let curFocus = global.stage.get_key_focus(); | ||||
|     let curFocusDestroyId; | ||||
|     if (curFocus != null) { | ||||
|         curFocusDestroyId = curFocus.connect('destroy', function() { | ||||
|             let index = _findModal(actor); | ||||
|             if (index >= 0) | ||||
|                 modalActorFocusStack[index].prevFocus = null; | ||||
|                 modalActorFocusStack[index].actor = null; | ||||
|         }); | ||||
|     } | ||||
|     modalActorFocusStack.push({ actor: actor, | ||||
|                                 focus: curFocus, | ||||
|                                 destroyId: actorDestroyId, | ||||
|                                 prevFocus: prevFocus, | ||||
|                                 prevFocusDestroyId: prevFocusDestroyId, | ||||
|                                 keybindingMode: keybindingMode }); | ||||
|                                 focusDestroyId: curFocusDestroyId }); | ||||
|  | ||||
|     keybindingMode = params.keybindingMode; | ||||
|     global.stage.set_key_focus(actor); | ||||
|     return true; | ||||
| } | ||||
| @@ -572,7 +759,6 @@ function popModal(actor, timestamp) { | ||||
|         global.stage.set_key_focus(null); | ||||
|         global.end_modal(timestamp); | ||||
|         global.set_stage_input_mode(Shell.StageInputMode.NORMAL); | ||||
|         keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
|  | ||||
|         throw new Error('incorrect pop'); | ||||
|     } | ||||
| @@ -583,34 +769,17 @@ function popModal(actor, timestamp) { | ||||
|     record.actor.disconnect(record.destroyId); | ||||
|  | ||||
|     if (focusIndex == modalActorFocusStack.length - 1) { | ||||
|         if (record.prevFocus) | ||||
|             record.prevFocus.disconnect(record.prevFocusDestroyId); | ||||
|         keybindingMode = record.keybindingMode; | ||||
|         global.stage.set_key_focus(record.prevFocus); | ||||
|         if (record.focus) | ||||
|             record.focus.disconnect(record.focusDestroyId); | ||||
|         global.stage.set_key_focus(record.focus); | ||||
|     } else { | ||||
|         // If we have: | ||||
|         //     global.stage.set_focus(a); | ||||
|         //     Main.pushModal(b); | ||||
|         //     Main.pushModal(c); | ||||
|         //     Main.pushModal(d); | ||||
|         // | ||||
|         // then we have the stack: | ||||
|         //     [{ prevFocus: a, actor: b }, | ||||
|         //      { prevFocus: b, actor: c }, | ||||
|         //      { prevFocus: c, actor: d }] | ||||
|         // | ||||
|         // When actor c is destroyed/popped, if we only simply remove the | ||||
|         // record, then the focus stack will be [a, c], rather than the correct | ||||
|         // [a, b]. Shift the focus stack up before removing the record to ensure | ||||
|         // that we get the correct result. | ||||
|         let t = modalActorFocusStack[modalActorFocusStack.length - 1]; | ||||
|         if (t.prevFocus) | ||||
|             t.prevFocus.disconnect(t.prevFocusDestroyId); | ||||
|         if (t.focus) | ||||
|             t.focus.disconnect(t.focusDestroyId); | ||||
|         // Remove from the middle, shift the focus chain up | ||||
|         for (let i = modalActorFocusStack.length - 1; i > focusIndex; i--) { | ||||
|             modalActorFocusStack[i].prevFocus = modalActorFocusStack[i - 1].prevFocus; | ||||
|             modalActorFocusStack[i].prevFocusDestroyId = modalActorFocusStack[i - 1].prevFocusDestroyId; | ||||
|             modalActorFocusStack[i].keybindingMode = modalActorFocusStack[i - 1].keybindingMode; | ||||
|             modalActorFocusStack[i].focus = modalActorFocusStack[i - 1].focus; | ||||
|             modalActorFocusStack[i].focusDestroyId = modalActorFocusStack[i - 1].focusDestroyId; | ||||
|         } | ||||
|     } | ||||
|     modalActorFocusStack.splice(focusIndex, 1); | ||||
| @@ -620,8 +789,6 @@ function popModal(actor, timestamp) { | ||||
|  | ||||
|     global.end_modal(timestamp); | ||||
|     global.set_stage_input_mode(Shell.StageInputMode.NORMAL); | ||||
|     Meta.enable_unredirect_for_screen(global.screen); | ||||
|     keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
| } | ||||
|  | ||||
| function createLookingGlass() { | ||||
| @@ -631,11 +798,11 @@ function createLookingGlass() { | ||||
|     return lookingGlass; | ||||
| } | ||||
|  | ||||
| function openRunDialog() { | ||||
| function getRunDialog() { | ||||
|     if (runDialog == null) { | ||||
|         runDialog = new RunDialog.RunDialog(); | ||||
|     } | ||||
|     runDialog.open(); | ||||
|     return runDialog; | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -764,8 +931,7 @@ function initializeDeferredWork(actor, callback, props) { | ||||
| function queueDeferredWork(workId) { | ||||
|     let data = _deferredWorkData[workId]; | ||||
|     if (!data) { | ||||
|         let message = 'Invalid work id %d'.format(workId); | ||||
|         logError(new Error(message), message); | ||||
|         global.logError('invalid work id ', workId); | ||||
|         return; | ||||
|     } | ||||
|     if (_deferredWorkQueue.indexOf(workId) < 0) | ||||
|   | ||||
							
								
								
									
										2044
									
								
								js/ui/messageTray.js
									
									
									
									
									
								
							
							
						
						| @@ -14,12 +14,12 @@ const Atk = imports.gi.Atk; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| const Layout = imports.ui.layout; | ||||
| const Lightbox = imports.ui.lightbox; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const OPEN_AND_CLOSE_TIME = 0.1; | ||||
| const FADE_IN_BUTTONS_TIME = 0.33; | ||||
| const FADE_OUT_DIALOG_TIME = 1.0; | ||||
|  | ||||
| const State = { | ||||
| @@ -35,22 +35,17 @@ const ModalDialog = new Lang.Class({ | ||||
|  | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { shellReactive: false, | ||||
|                                         styleClass: null, | ||||
|                                         parentActor: Main.uiGroup, | ||||
|                                         keybindingMode: Shell.KeyBindingMode.SYSTEM_MODAL, | ||||
|                                         shouldFadeIn: true }); | ||||
|                                         styleClass: null }); | ||||
|  | ||||
|         this.state = State.CLOSED; | ||||
|         this._hasModal = false; | ||||
|         this._keybindingMode = params.keybindingMode; | ||||
|         this._shellReactive = params.shellReactive; | ||||
|         this._shouldFadeIn = params.shouldFadeIn; | ||||
|  | ||||
|         this._group = new St.Widget({ visible: false, | ||||
|                                       x: 0, | ||||
|                                       y: 0, | ||||
|                                       accessible_role: Atk.Role.DIALOG }); | ||||
|         params.parentActor.add_actor(this._group); | ||||
|         Main.uiGroup.add_actor(this._group); | ||||
|  | ||||
|         let constraint = new Clutter.BindConstraint({ source: global.stage, | ||||
|                                                       coordinate: Clutter.BindCoordinate.ALL }); | ||||
| @@ -58,18 +53,16 @@ const ModalDialog = new Lang.Class({ | ||||
|  | ||||
|         this._group.connect('destroy', Lang.bind(this, this._onGroupDestroy)); | ||||
|  | ||||
|         this._buttonKeys = {}; | ||||
|         this._group.connect('key-release-event', Lang.bind(this, this._onKeyReleaseEvent)); | ||||
|         this._actionKeys = {}; | ||||
|         this._group.connect('key-press-event', Lang.bind(this, this._onKeyPressEvent)); | ||||
|  | ||||
|         this._backgroundBin = new St.Bin(); | ||||
|         this._monitorConstraint = new Layout.MonitorConstraint(); | ||||
|         this._backgroundBin.add_constraint(this._monitorConstraint); | ||||
|         this._group.add_actor(this._backgroundBin); | ||||
|  | ||||
|         this.dialogLayout = new St.BoxLayout({ style_class: 'modal-dialog', | ||||
|                                                vertical:    true }); | ||||
|         this._dialogLayout = new St.BoxLayout({ style_class: 'modal-dialog', | ||||
|                                                 vertical:    true }); | ||||
|         if (params.styleClass != null) { | ||||
|             this.dialogLayout.add_style_class_name(params.styleClass); | ||||
|             this._dialogLayout.add_style_class_name(params.styleClass); | ||||
|         } | ||||
|  | ||||
|         if (!this._shellReactive) { | ||||
| @@ -82,28 +75,29 @@ const ModalDialog = new Lang.Class({ | ||||
|  | ||||
|             this._eventBlocker = new Clutter.Group({ reactive: true }); | ||||
|             stack.add_actor(this._eventBlocker); | ||||
|             stack.add_actor(this.dialogLayout); | ||||
|             stack.add_actor(this._dialogLayout); | ||||
|         } else { | ||||
|             this._backgroundBin.child = this.dialogLayout; | ||||
|             this._backgroundBin.child = this._dialogLayout; | ||||
|         } | ||||
|  | ||||
|  | ||||
|         this.contentLayout = new St.BoxLayout({ vertical: true }); | ||||
|         this.dialogLayout.add(this.contentLayout, | ||||
|                               { x_fill:  true, | ||||
|                                 y_fill:  true, | ||||
|                                 x_align: St.Align.MIDDLE, | ||||
|                                 y_align: St.Align.START }); | ||||
|         this._dialogLayout.add(this.contentLayout, | ||||
|                                { x_fill:  true, | ||||
|                                  y_fill:  true, | ||||
|                                  x_align: St.Align.MIDDLE, | ||||
|                                  y_align: St.Align.START }); | ||||
|  | ||||
|         this.buttonLayout = new St.BoxLayout({ style_class: 'modal-dialog-button-box', | ||||
|                                                vertical: false }); | ||||
|         this.dialogLayout.add(this.buttonLayout, | ||||
|                               { expand:  true, | ||||
|                                 x_align: St.Align.MIDDLE, | ||||
|                                 y_align: St.Align.END }); | ||||
|         this._buttonLayout = new St.BoxLayout({ style_class: 'modal-dialog-button-box', | ||||
|                                                 visible:     false, | ||||
|                                                 vertical:    false }); | ||||
|         this._dialogLayout.add(this._buttonLayout, | ||||
|                                { expand:  true, | ||||
|                                  x_align: St.Align.MIDDLE, | ||||
|                                  y_align: St.Align.END }); | ||||
|  | ||||
|         global.focus_manager.add_group(this.dialogLayout); | ||||
|         this._initialKeyFocus = this.dialogLayout; | ||||
|         global.focus_manager.add_group(this._dialogLayout); | ||||
|         this._initialKeyFocus = this._dialogLayout; | ||||
|         this._initialKeyFocusDestroyId = 0; | ||||
|         this._savedKeyFocus = null; | ||||
|     }, | ||||
| @@ -112,16 +106,24 @@ const ModalDialog = new Lang.Class({ | ||||
|         this._group.destroy(); | ||||
|     }, | ||||
|  | ||||
|     clearButtons: function() { | ||||
|         this.buttonLayout.destroy_all_children(); | ||||
|         this._buttonKeys = {}; | ||||
|     }, | ||||
|  | ||||
|     setButtons: function(buttons) { | ||||
|         this.clearButtons(); | ||||
|         let hadChildren = this._buttonLayout.get_children() > 0; | ||||
|  | ||||
|         this._buttonLayout.destroy_all_children(); | ||||
|         this._actionKeys = {}; | ||||
|  | ||||
|         this._buttonLayout.visible = (buttons.length > 0); | ||||
|  | ||||
|         for (let i = 0; i < buttons.length; i++) { | ||||
|             let buttonInfo = buttons[i]; | ||||
|             let label = buttonInfo['label']; | ||||
|             let action = buttonInfo['action']; | ||||
|             let key = buttonInfo['key']; | ||||
|  | ||||
|             buttonInfo.button = new St.Button({ style_class: 'modal-dialog-button', | ||||
|                                                 reactive:    true, | ||||
|                                                 can_focus:   true, | ||||
|                                                 label:       label }); | ||||
|  | ||||
|             let x_alignment; | ||||
|             if (buttons.length == 1) | ||||
| @@ -133,90 +135,66 @@ const ModalDialog = new Lang.Class({ | ||||
|             else | ||||
|                 x_alignment = St.Align.MIDDLE; | ||||
|  | ||||
|             this.addButton(buttonInfo, { expand: true, | ||||
|                                          x_fill: false, | ||||
|                                          y_fill: false, | ||||
|                                          x_align: x_alignment, | ||||
|                                          y_align: St.Align.MIDDLE }); | ||||
|             if (!this._initialKeyFocusDestroyId) | ||||
|                 this._initialKeyFocus = buttonInfo.button; | ||||
|             this._buttonLayout.add(buttonInfo.button, | ||||
|                                    { expand: true, | ||||
|                                      x_fill: false, | ||||
|                                      y_fill: false, | ||||
|                                      x_align: x_alignment, | ||||
|                                      y_align: St.Align.MIDDLE }); | ||||
|  | ||||
|             buttonInfo.button.connect('clicked', action); | ||||
|  | ||||
|             if (key) | ||||
|                 this._actionKeys[key] = action; | ||||
|         } | ||||
|  | ||||
|         // Fade in buttons if there weren't any before | ||||
|         if (!hadChildren && buttons.length > 0) { | ||||
|             this._buttonLayout.opacity = 0; | ||||
|             Tweener.addTween(this._buttonLayout, | ||||
|                              { opacity: 255, | ||||
|                                time: FADE_IN_BUTTONS_TIME, | ||||
|                                transition: 'easeOutQuad', | ||||
|                                onComplete: Lang.bind(this, function() { | ||||
|                                    this.emit('buttons-set'); | ||||
|                                }) | ||||
|                              }); | ||||
|         } else { | ||||
|             this.emit('buttons-set'); | ||||
|         } | ||||
|  | ||||
|     }, | ||||
|  | ||||
|     addButton: function(buttonInfo, layoutInfo) { | ||||
|         let label = buttonInfo['label']; | ||||
|         let action = buttonInfo['action']; | ||||
|         let key = buttonInfo['key']; | ||||
|         let isDefault = buttonInfo['default']; | ||||
|     _onKeyPressEvent: function(object, keyPressEvent) { | ||||
|         let symbol = keyPressEvent.get_key_symbol(); | ||||
|         let action = this._actionKeys[symbol]; | ||||
|  | ||||
|         let keys; | ||||
|  | ||||
|         if (key) | ||||
|             keys = [key]; | ||||
|         else if (isDefault) | ||||
|             keys = [Clutter.KEY_Return, Clutter.KEY_KP_Enter, Clutter.KEY_ISO_Enter]; | ||||
|         else | ||||
|             keys = []; | ||||
|  | ||||
|         let button = new St.Button({ style_class: 'modal-dialog-button', | ||||
|                                      button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|                                      reactive:    true, | ||||
|                                      can_focus:   true, | ||||
|                                      label:       label }); | ||||
|         button.connect('clicked', action); | ||||
|  | ||||
|         buttonInfo['button'] = button; | ||||
|  | ||||
|         if (isDefault) | ||||
|             button.add_style_pseudo_class('default'); | ||||
|  | ||||
|         if (!this._initialKeyFocusDestroyId) | ||||
|             this._initialKeyFocus = button; | ||||
|  | ||||
|         for (let i in keys) | ||||
|             this._buttonKeys[keys[i]] = buttonInfo; | ||||
|  | ||||
|         this.buttonLayout.add(button, layoutInfo); | ||||
|  | ||||
|         return button; | ||||
|     }, | ||||
|  | ||||
|     _onKeyReleaseEvent: function(object, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         let buttonInfo = this._buttonKeys[symbol]; | ||||
|  | ||||
|         if (!buttonInfo) | ||||
|             return false; | ||||
|  | ||||
|         let button = buttonInfo['button']; | ||||
|         let action = buttonInfo['action']; | ||||
|  | ||||
|         if (action && button.reactive) { | ||||
|         if (action) | ||||
|             action(); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onGroupDestroy: function() { | ||||
|         this.emit('destroy'); | ||||
|     }, | ||||
|  | ||||
|     _fadeOpen: function(onPrimary) { | ||||
|         if (onPrimary) | ||||
|             this._monitorConstraint.primary = true; | ||||
|         else | ||||
|             this._monitorConstraint.index = global.screen.get_current_monitor(); | ||||
|     _fadeOpen: function() { | ||||
|         let monitor = Main.layoutManager.focusMonitor; | ||||
|  | ||||
|         this._backgroundBin.set_position(monitor.x, monitor.y); | ||||
|         this._backgroundBin.set_size(monitor.width, monitor.height); | ||||
|  | ||||
|         this.state = State.OPENING; | ||||
|  | ||||
|         this.dialogLayout.opacity = 255; | ||||
|         this._dialogLayout.opacity = 255; | ||||
|         if (this._lightbox) | ||||
|             this._lightbox.show(); | ||||
|         this._group.opacity = 0; | ||||
|         this._group.show(); | ||||
|         Tweener.addTween(this._group, | ||||
|                          { opacity: 255, | ||||
|                            time: this._shouldFadeIn ? OPEN_AND_CLOSE_TIME : 0, | ||||
|                            time: OPEN_AND_CLOSE_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, | ||||
|                                function() { | ||||
| @@ -233,19 +211,19 @@ const ModalDialog = new Lang.Class({ | ||||
|         this._initialKeyFocus = actor; | ||||
|  | ||||
|         this._initialKeyFocusDestroyId = actor.connect('destroy', Lang.bind(this, function() { | ||||
|             this._initialKeyFocus = this.dialogLayout; | ||||
|             this._initialKeyFocus = this._dialogLayout; | ||||
|             this._initialKeyFocusDestroyId = 0; | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     open: function(timestamp, onPrimary) { | ||||
|     open: function(timestamp) { | ||||
|         if (this.state == State.OPENED || this.state == State.OPENING) | ||||
|             return true; | ||||
|  | ||||
|         if (!this.pushModal({ timestamp: timestamp })) | ||||
|         if (!this.pushModal(timestamp)) | ||||
|             return false; | ||||
|  | ||||
|         this._fadeOpen(onPrimary); | ||||
|         this._fadeOpen(); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
| @@ -265,7 +243,6 @@ const ModalDialog = new Lang.Class({ | ||||
|                                function() { | ||||
|                                    this.state = State.CLOSED; | ||||
|                                    this._group.hide(); | ||||
|                                    this.emit('closed'); | ||||
|                                }) | ||||
|                          }); | ||||
|     }, | ||||
| @@ -293,8 +270,7 @@ const ModalDialog = new Lang.Class({ | ||||
|     pushModal: function (timestamp) { | ||||
|         if (this._hasModal) | ||||
|             return true; | ||||
|         if (!Main.pushModal(this._group, { timestamp: timestamp, | ||||
|                                            keybindingMode: this._keybindingMode })) | ||||
|         if (!Main.pushModal(this._group, timestamp)) | ||||
|             return false; | ||||
|  | ||||
|         this._hasModal = true; | ||||
| @@ -328,7 +304,7 @@ const ModalDialog = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         this.popModal(timestamp); | ||||
|         Tweener.addTween(this.dialogLayout, | ||||
|         Tweener.addTween(this._dialogLayout, | ||||
|                          { opacity: 0, | ||||
|                            time:    FADE_OUT_DIALOG_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|   | ||||
| @@ -1,4 +1,23 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
 | ||||
| /* | ||||
|  * Copyright 2011 Giovanni Campagna <scampa.giovanni@gmail.com> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2, or (at your option) | ||||
|  * any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA | ||||
|  * 02111-1307, USA. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| @@ -116,10 +135,7 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
|             } else | ||||
|                 secret.valid = true; | ||||
| 
 | ||||
|             secretTable.add(label, { row: pos, col: 0, | ||||
|                                      x_expand: false, x_fill: true, | ||||
|                                      x_align: St.Align.START, | ||||
|                                      y_fill: false, y_align: St.Align.MIDDLE }); | ||||
|             secretTable.add(label, { row: pos, col: 0, x_expand: false, x_fill: true, x_align: St.Align.START, y_align: St.Align.START }); | ||||
|             secretTable.add(secret.entry, { row: pos, col: 1, x_expand: true, x_fill: true, y_align: St.Align.END }); | ||||
|             pos++; | ||||
| 
 | ||||
| @@ -131,7 +147,7 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
| 
 | ||||
|         this._okButton = { label:  _("Connect"), | ||||
|                            action: Lang.bind(this, this._onOk), | ||||
|                            default: true | ||||
|                            key:    Clutter.KEY_Return, | ||||
|                          }; | ||||
| 
 | ||||
|         this.setButtons([{ label: _("Cancel"), | ||||
| @@ -150,6 +166,10 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
| 
 | ||||
|         this._okButton.button.reactive = valid; | ||||
|         this._okButton.button.can_focus = valid; | ||||
|         if (valid) | ||||
|             this._okButton.button.remove_style_pseudo_class('disabled'); | ||||
|         else | ||||
|             this._okButton.button.add_style_pseudo_class('disabled'); | ||||
|     }, | ||||
| 
 | ||||
|     _onOk: function() { | ||||
| @@ -408,10 +428,7 @@ const VPNRequestHandler = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
| 
 | ||||
|     cancel: function(respond) { | ||||
|         if (respond) | ||||
|             this._agent.respond(this._requestId, Shell.NetworkAgentResponse.USER_CANCELED); | ||||
| 
 | ||||
|     cancel: function() { | ||||
|         if (this._newStylePlugin && this._shellDialog) { | ||||
|             this._shellDialog.close(global.get_current_time()); | ||||
|             this._shellDialog.destroy(); | ||||
| @@ -587,41 +604,17 @@ const NetworkAgent = new Lang.Class({ | ||||
|     Name: 'NetworkAgent', | ||||
| 
 | ||||
|     _init: function() { | ||||
|         this._native = new Shell.NetworkAgent({ identifier: 'org.gnome.Shell.NetworkAgent' }); | ||||
|         this._native = new Shell.NetworkAgent({ auto_register: true, | ||||
|                                                 identifier: 'org.gnome.Shell.NetworkAgent' }); | ||||
| 
 | ||||
|         this._dialogs = { }; | ||||
|         this._vpnRequests = { }; | ||||
| 
 | ||||
|         this._native.connect('new-request', Lang.bind(this, this._newRequest)); | ||||
|         this._native.connect('cancel-request', Lang.bind(this, this._cancelRequest)); | ||||
| 
 | ||||
|         this._enabled = false; | ||||
|     }, | ||||
| 
 | ||||
|     enable: function() { | ||||
|         this._enabled = true; | ||||
|     }, | ||||
| 
 | ||||
|     disable: function() { | ||||
|         let requestId; | ||||
| 
 | ||||
|         for (requestId in this._dialogs) | ||||
|             this._dialogs[requestId].cancel(); | ||||
|         this._dialogs = { }; | ||||
| 
 | ||||
|         for (requestId in this._vpnRequests) | ||||
|             this._vpnRequests[requestId].cancel(true); | ||||
|         this._vpnRequests = { }; | ||||
| 
 | ||||
|         this._enabled = false; | ||||
|     }, | ||||
| 
 | ||||
|     _newRequest:  function(agent, requestId, connection, settingName, hints, flags) { | ||||
|         if (!this._enabled) { | ||||
|             agent.respond(requestId, Shell.NetworkAgentResponse.USER_CANCELED); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         if (settingName == 'vpn') { | ||||
|             this._vpnRequest(requestId, connection, hints, flags); | ||||
|             return; | ||||
| @@ -641,7 +634,7 @@ const NetworkAgent = new Lang.Class({ | ||||
|             this._dialogs[requestId].destroy(); | ||||
|             delete this._dialogs[requestId]; | ||||
|         } else if (this._vpnRequests[requestId]) { | ||||
|             this._vpnRequests[requestId].cancel(false); | ||||
|             this._vpnRequests[requestId].cancel(); | ||||
|             delete this._vpnRequests[requestId]; | ||||
|         } | ||||
|     }, | ||||
| @@ -690,10 +683,7 @@ const NetworkAgent = new Lang.Class({ | ||||
|                     try { | ||||
|                         externalUIMode = keyfile.get_boolean('GNOME', 'supports-external-ui-mode'); | ||||
|                     } catch(e) { } // ignore errors if key does not exist
 | ||||
|                     let path = binary; | ||||
|                     if (!GLib.path_is_absolute(path)) { | ||||
|                         path = GLib.build_filenamev([Config.LIBEXECDIR, path]); | ||||
|                     } | ||||
|                     let path = GLib.build_filenamev([Config.LIBEXECDIR, binary]); | ||||
| 
 | ||||
|                     if (GLib.file_test(path, GLib.FileTest.IS_EXECUTABLE)) | ||||
|                         this._vpnBinaries[service] = { fileName: path, externalUIMode: externalUIMode }; | ||||
| @@ -710,4 +700,3 @@ const NetworkAgent = new Lang.Class({ | ||||
|         } | ||||
|     } | ||||
| }); | ||||
| const Component = NetworkAgent; | ||||
| @@ -1,7 +1,6 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GdkPixbuf = imports.gi.GdkPixbuf; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| @@ -88,141 +87,6 @@ const rewriteRules = { | ||||
|     ] | ||||
| }; | ||||
|  | ||||
| const STANDARD_TRAY_ICON_IMPLEMENTATIONS = { | ||||
|     'bluetooth-applet': 'bluetooth', | ||||
|     'gnome-volume-control-applet': 'volume', // renamed to gnome-sound-applet | ||||
|                                              // when moved to control center | ||||
|     'gnome-sound-applet': 'volume', | ||||
|     'nm-applet': 'network', | ||||
|     'gnome-power-manager': 'battery', | ||||
|     'keyboard': 'keyboard', | ||||
|     'a11y-keyboard': 'a11y', | ||||
|     'kbd-scrolllock': 'keyboard', | ||||
|     'kbd-numlock': 'keyboard', | ||||
|     'kbd-capslock': 'keyboard', | ||||
|     'ibus-ui-gtk': 'keyboard' | ||||
| }; | ||||
|  | ||||
| const NotificationGenericPolicy = new Lang.Class({ | ||||
|     Name: 'NotificationGenericPolicy', | ||||
|     Extends: MessageTray.NotificationPolicy, | ||||
|  | ||||
|     _init: function() { | ||||
|         // Don't chain to parent, it would try setting | ||||
|         // our properties to the defaults | ||||
|  | ||||
|         this.id = 'generic'; | ||||
|  | ||||
|         this._masterSettings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications' }); | ||||
|         this._masterSettings.connect('changed', Lang.bind(this, this._changed)); | ||||
|     }, | ||||
|  | ||||
|     store: function() { }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         this._masterSettings.run_dispose(); | ||||
|     }, | ||||
|  | ||||
|     _changed: function(settings, key) { | ||||
|         this.emit('policy-changed', key); | ||||
|     }, | ||||
|  | ||||
|     get enable() { | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     get enableSound() { | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     get showBanners() { | ||||
|         return this._masterSettings.get_boolean('show-banners'); | ||||
|     }, | ||||
|  | ||||
|     get forceExpanded() { | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     get showInLockScreen() { | ||||
|         return this._masterSettings.get_boolean('show-in-lock-screen'); | ||||
|     }, | ||||
|  | ||||
|     get detailsInLockScreen() { | ||||
|         return false; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const NotificationApplicationPolicy = new Lang.Class({ | ||||
|     Name: 'NotificationApplicationPolicy', | ||||
|     Extends: MessageTray.NotificationPolicy, | ||||
|  | ||||
|     _init: function(id) { | ||||
|         // Don't chain to parent, it would try setting | ||||
|         // our properties to the defaults | ||||
|  | ||||
|         this.id = id; | ||||
|         this._canonicalId = this._canonicalizeId(id) | ||||
|  | ||||
|         this._masterSettings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications' }); | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications.application', | ||||
|                                             path: '/org/gnome/desktop/notifications/application/' + this._canonicalId + '/' }); | ||||
|  | ||||
|         this._masterSettings.connect('changed', Lang.bind(this, this._changed)); | ||||
|         this._settings.connect('changed', Lang.bind(this, this._changed)); | ||||
|     }, | ||||
|  | ||||
|     store: function() { | ||||
|         this._settings.set_string('application-id', this.id + '.desktop'); | ||||
|  | ||||
|         let apps = this._masterSettings.get_strv('application-children'); | ||||
|         if (apps.indexOf(this._canonicalId) < 0) { | ||||
|             apps.push(this._canonicalId); | ||||
|             this._masterSettings.set_strv('application-children', apps); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         this._masterSettings.run_dispose(); | ||||
|         this._settings.run_dispose(); | ||||
|     }, | ||||
|  | ||||
|     _changed: function(settings, key) { | ||||
|         this.emit('policy-changed', key); | ||||
|     }, | ||||
|  | ||||
|     _canonicalizeId: function(id) { | ||||
|         // Keys are restricted to lowercase alphanumeric characters and dash, | ||||
|         // and two dashes cannot be in succession | ||||
|         return id.toLowerCase().replace(/[^a-z0-9\-]/g, '-').replace(/--+/g, '-'); | ||||
|     }, | ||||
|  | ||||
|     get enable() { | ||||
|         return this._settings.get_boolean('enable'); | ||||
|     }, | ||||
|  | ||||
|     get enableSound() { | ||||
|         return this._settings.get_boolean('enable-sound-alerts'); | ||||
|     }, | ||||
|  | ||||
|     get showBanners() { | ||||
|         return this._masterSettings.get_boolean('show-banners') && | ||||
|             this._settings.get_boolean('show-banners'); | ||||
|     }, | ||||
|  | ||||
|     get forceExpanded() { | ||||
|         return this._settings.get_boolean('force-expanded'); | ||||
|     }, | ||||
|  | ||||
|     get showInLockScreen() { | ||||
|         return this._masterSettings.get_boolean('show-in-lock-screen') && | ||||
|             this._settings.get_boolean('show-in-lock-screen'); | ||||
|     }, | ||||
|  | ||||
|     get detailsInLockScreen() { | ||||
|         return this._settings.get_boolean('details-in-lock-screen'); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const NotificationDaemon = new Lang.Class({ | ||||
|     Name: 'NotificationDaemon', | ||||
|  | ||||
| @@ -235,54 +99,57 @@ const NotificationDaemon = new Lang.Class({ | ||||
|         this._notifications = {}; | ||||
|         this._busProxy = new Bus(); | ||||
|  | ||||
|         this._trayManager = new Shell.TrayManager(); | ||||
|         this._trayIconAddedId = this._trayManager.connect('tray-icon-added', Lang.bind(this, this._onTrayIconAdded)); | ||||
|         this._trayIconRemovedId = this._trayManager.connect('tray-icon-removed', Lang.bind(this, this._onTrayIconRemoved)); | ||||
|         Main.statusIconDispatcher.connect('message-icon-added', Lang.bind(this, this._onTrayIconAdded)); | ||||
|         Main.statusIconDispatcher.connect('message-icon-removed', Lang.bind(this, this._onTrayIconRemoved)); | ||||
|  | ||||
|         Shell.WindowTracker.get_default().connect('notify::focus-app', | ||||
|             Lang.bind(this, this._onFocusAppChanged)); | ||||
|         Main.overview.connect('hidden', | ||||
|             Lang.bind(this, this._onFocusAppChanged)); | ||||
|  | ||||
|         this._trayManager.manage_stage(global.stage, Main.messageTray.actor); | ||||
|     }, | ||||
|  | ||||
|     _imageForNotificationData: function(hints) { | ||||
|         if (hints['image-data']) { | ||||
|             let [width, height, rowStride, hasAlpha, | ||||
|                  bitsPerSample, nChannels, data] = hints['image-data']; | ||||
|             return Shell.util_create_pixbuf_from_data(data, GdkPixbuf.Colorspace.RGB, hasAlpha, | ||||
|                                                       bitsPerSample, width, height, rowStride); | ||||
|         } else if (hints['image-path']) { | ||||
|             return new Gio.FileIcon({ file: Gio.File.new_for_path(hints['image-path']) }); | ||||
|         } | ||||
|         return null; | ||||
|     }, | ||||
|     _iconForNotificationData: function(icon, hints, size) { | ||||
|         let textureCache = St.TextureCache.get_default(); | ||||
|  | ||||
|     _fallbackIconForNotificationData: function(hints) { | ||||
|         let stockIcon; | ||||
|         switch (hints.urgency) { | ||||
|             case Urgency.LOW: | ||||
|             case Urgency.NORMAL: | ||||
|                 stockIcon = 'gtk-dialog-info'; | ||||
|                 break; | ||||
|             case Urgency.CRITICAL: | ||||
|                 stockIcon = 'gtk-dialog-error'; | ||||
|                 break; | ||||
|         } | ||||
|         return new Gio.ThemedIcon({ name: stockIcon }); | ||||
|     }, | ||||
|  | ||||
|     _iconForNotificationData: function(icon) { | ||||
|         // If an icon is not specified, we use 'image-data' or 'image-path' hint for an icon | ||||
|         // and don't show a large image. There are currently many applications that use | ||||
|         // notify_notification_set_icon_from_pixbuf() from libnotify, which in turn sets | ||||
|         // the 'image-data' hint. These applications don't typically pass in 'app_icon' | ||||
|         // argument to Notify() and actually expect the pixbuf to be shown as an icon. | ||||
|         // So the logic here does the right thing for this case. If both an icon and either | ||||
|         // one of 'image-data' or 'image-path' are specified, we show both an icon and | ||||
|         // a large image. | ||||
|         if (icon) { | ||||
|             if (icon.substr(0, 7) == 'file://') | ||||
|                 return new Gio.FileIcon({ file: Gio.File.new_for_uri(icon) }); | ||||
|             else if (icon[0] == '/') | ||||
|                 return new Gio.FileIcon({ file: Gio.File.new_for_path(icon) }); | ||||
|             else | ||||
|                 return new Gio.ThemedIcon({ name: icon }); | ||||
|                 return textureCache.load_uri_async(icon, size, size); | ||||
|             else if (icon[0] == '/') { | ||||
|                 let uri = GLib.filename_to_uri(icon, null); | ||||
|                 return textureCache.load_uri_async(uri, size, size); | ||||
|             } else | ||||
|                 return new St.Icon({ icon_name: icon, | ||||
|                                      icon_type: St.IconType.FULLCOLOR, | ||||
|                                      icon_size: size }); | ||||
|         } else if (hints['image-data']) { | ||||
|             let [width, height, rowStride, hasAlpha, | ||||
|                  bitsPerSample, nChannels, data] = hints['image-data']; | ||||
|             return textureCache.load_from_raw(data, hasAlpha, width, height, rowStride, size); | ||||
|         } else if (hints['image-path']) { | ||||
|             return textureCache.load_uri_async(GLib.filename_to_uri(hints['image-path'], null), size, size); | ||||
|         } else { | ||||
|             let stockIcon; | ||||
|             switch (hints.urgency) { | ||||
|                 case Urgency.LOW: | ||||
|                 case Urgency.NORMAL: | ||||
|                     stockIcon = 'gtk-dialog-info'; | ||||
|                     break; | ||||
|                 case Urgency.CRITICAL: | ||||
|                     stockIcon = 'gtk-dialog-error'; | ||||
|                     break; | ||||
|             } | ||||
|             return new St.Icon({ icon_name: stockIcon, | ||||
|                                  icon_type: St.IconType.FULLCOLOR, | ||||
|                                  icon_size: size }); | ||||
|         } | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     _lookupSource: function(title, pid, trayIcon) { | ||||
| @@ -333,7 +200,7 @@ const NotificationDaemon = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let source = new Source(title, pid, sender, trayIcon, ndata ? ndata.hints['desktop-entry'] : null); | ||||
|         let source = new Source(title, pid, sender, trayIcon); | ||||
|         source.setTransient(isForTransientNotification); | ||||
|  | ||||
|         if (!isForTransientNotification) { | ||||
| @@ -354,19 +221,12 @@ const NotificationDaemon = new Lang.Class({ | ||||
|         let [appName, replacesId, icon, summary, body, actions, hints, timeout] = params; | ||||
|         let id; | ||||
|  | ||||
|         for (let hint in hints) { | ||||
|             // unpack the variants | ||||
|             hints[hint] = hints[hint].deep_unpack(); | ||||
|         } | ||||
|  | ||||
|         hints = Params.parse(hints, { urgency: Urgency.NORMAL }, true); | ||||
|  | ||||
|         // Filter out chat, presence, calls and invitation notifications from | ||||
|         // Empathy, since we handle that information from telepathyClient.js | ||||
|         if (appName == 'Empathy' && (hints['category'] == 'im.received' || | ||||
|               hints['category'] == 'x-empathy.im.room-invitation' || | ||||
|               hints['category'] == 'x-empathy.call.incoming' || | ||||
|               hints['category'] == 'x-empathy.transfer.incoming' || | ||||
|               hints['category'] == 'x-empathy.call.incoming"' || | ||||
|               hints['category'] == 'x-empathy.im.subscription-request' || | ||||
|               hints['category'] == 'presence.online' || | ||||
|               hints['category'] == 'presence.offline')) { | ||||
| @@ -376,7 +236,6 @@ const NotificationDaemon = new Lang.Class({ | ||||
|             Mainloop.idle_add(Lang.bind(this, | ||||
|                                         function () { | ||||
|                                             this._emitNotificationClosed(id, NotificationClosedReason.DISMISSED); | ||||
|                                             return false; | ||||
|                                         })); | ||||
|             return invocation.return_value(GLib.Variant.new('(u)', [id])); | ||||
|         } | ||||
| @@ -390,6 +249,13 @@ const NotificationDaemon = new Lang.Class({ | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         for (let hint in hints) { | ||||
|             // unpack the variants | ||||
|             hints[hint] = hints[hint].deep_unpack(); | ||||
|         } | ||||
|  | ||||
|         hints = Params.parse(hints, { urgency: Urgency.NORMAL }, true); | ||||
|  | ||||
|         // Be compatible with the various hints for image data and image path | ||||
|         // 'image-data' and 'image-path' are the latest name of these hints, introduced in 1.2 | ||||
|  | ||||
| @@ -475,8 +341,12 @@ const NotificationDaemon = new Lang.Class({ | ||||
|             [ndata.id, ndata.icon, ndata.summary, ndata.body, | ||||
|              ndata.actions, ndata.hints, ndata.notification]; | ||||
|  | ||||
|         let iconActor = this._iconForNotificationData(icon, hints, source.ICON_SIZE); | ||||
|  | ||||
|         if (notification == null) { | ||||
|             notification = new MessageTray.Notification(source); | ||||
|             notification = new MessageTray.Notification(source, summary, body, | ||||
|                                                         { icon: iconActor, | ||||
|                                                           bannerMarkup: true }); | ||||
|             ndata.notification = notification; | ||||
|             notification.connect('destroy', Lang.bind(this, | ||||
|                 function(n, reason) { | ||||
| @@ -499,38 +369,29 @@ const NotificationDaemon = new Lang.Class({ | ||||
|                 function(n, actionId) { | ||||
|                     this._emitActionInvoked(ndata.id, actionId); | ||||
|                 })); | ||||
|         } else { | ||||
|             notification.update(summary, body, { icon: iconActor, | ||||
|                                                  bannerMarkup: true, | ||||
|                                                  clear: true }); | ||||
|         } | ||||
|  | ||||
|         // Mark music notifications so they can be shown in the screen shield | ||||
|         notification.isMusic = (ndata.hints['category'] == 'x-gnome.music'); | ||||
|  | ||||
|         let gicon = this._iconForNotificationData(icon, hints); | ||||
|         let gimage = this._imageForNotificationData(hints); | ||||
|  | ||||
|         let image = null; | ||||
|  | ||||
|         // If an icon is not specified, we use 'image-data' or 'image-path' hint for an icon | ||||
|         // and don't show a large image. There are currently many applications that use | ||||
|         // notify_notification_set_icon_from_pixbuf() from libnotify, which in turn sets | ||||
|         // the 'image-data' hint. These applications don't typically pass in 'app_icon' | ||||
|         // argument to Notify() and actually expect the pixbuf to be shown as an icon. | ||||
|         // So the logic here does the right thing for this case. If both an icon and either | ||||
|         // one of 'image-data' or 'image-path' are specified, we show both an icon and | ||||
|         // a large image. | ||||
|         if (gicon && gimage) | ||||
|             image = new St.Icon({ gicon: gimage, | ||||
|                                   icon_size: notification.IMAGE_SIZE }); | ||||
|         else if (!gicon && gimage) | ||||
|             gicon = gimage; | ||||
|         else if (!gicon) | ||||
|             gicon = this._fallbackIconForNotificationData(hints); | ||||
|  | ||||
|         notification.update(summary, body, { gicon: gicon, | ||||
|                                              bannerMarkup: true, | ||||
|                                              clear: true, | ||||
|                                              soundFile: hints['sound-file'], | ||||
|                                              soundName: hints['sound-name'] }); | ||||
|         notification.setImage(image); | ||||
|         // We only display a large image if an icon is also specified. | ||||
|         if (icon && (hints['image-data'] || hints['image-path'])) { | ||||
|             let image = null; | ||||
|             if (hints['image-data']) { | ||||
|                 let [width, height, rowStride, hasAlpha, | ||||
|                  bitsPerSample, nChannels, data] = hints['image-data']; | ||||
|                 image = St.TextureCache.get_default().load_from_raw(data, hasAlpha, | ||||
|                                                                     width, height, rowStride, notification.IMAGE_SIZE); | ||||
|             } else if (hints['image-path']) { | ||||
|                 image = St.TextureCache.get_default().load_uri_async(GLib.filename_to_uri(hints['image-path'], null), | ||||
|                                                                      notification.IMAGE_SIZE, | ||||
|                                                                      notification.IMAGE_SIZE); | ||||
|             } | ||||
|             notification.setImage(image); | ||||
|         } else { | ||||
|             notification.unsetImage(); | ||||
|         } | ||||
|  | ||||
|         if (actions.length) { | ||||
|             notification.setUseActionIcons(hints['action-icons'] == true); | ||||
| @@ -560,8 +421,8 @@ const NotificationDaemon = new Lang.Class({ | ||||
|         // of the 'transient' hint with hints['transient'] rather than hints.transient | ||||
|         notification.setTransient(hints['transient'] == true); | ||||
|  | ||||
|         let sourceGIcon = source.useNotificationIcon ? gicon : null; | ||||
|         source.processNotification(notification, sourceGIcon); | ||||
|         let sourceIconActor = source.useNotificationIcon ? this._iconForNotificationData(icon, hints, source.ICON_SIZE) : null; | ||||
|         source.processNotification(notification, sourceIconActor); | ||||
|     }, | ||||
|  | ||||
|     CloseNotification: function(id) { | ||||
| @@ -584,7 +445,7 @@ const NotificationDaemon = new Lang.Class({ | ||||
|             // 'icon-multi', | ||||
|             'icon-static', | ||||
|             'persistence', | ||||
|             'sound', | ||||
|             // 'sound', | ||||
|         ]; | ||||
|     }, | ||||
|  | ||||
| @@ -622,11 +483,7 @@ const NotificationDaemon = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconAdded: function(o, icon) { | ||||
|         let wmClass = icon.wm_class ? icon.wm_class.toLowerCase() : ''; | ||||
|         if (STANDARD_TRAY_ICON_IMPLEMENTATIONS[wmClass] !== undefined) | ||||
|             return; | ||||
|  | ||||
|         let source = this._getSource(icon.title || icon.wm_class || C_("program", "Unknown"), icon.pid, null, null, icon); | ||||
|         let source = this._getSource(icon.title || icon.wm_class || _("Unknown"), icon.pid, null, null, icon); | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconRemoved: function(o, icon) { | ||||
| @@ -640,22 +497,12 @@ const Source = new Lang.Class({ | ||||
|     Name: 'NotificationDaemonSource', | ||||
|     Extends: MessageTray.Source, | ||||
|  | ||||
|     _init: function(title, pid, sender, trayIcon, appId) { | ||||
|         // Need to set the app before chaining up, so | ||||
|         // methods called from the parent constructor can find it | ||||
|         this.trayIcon = trayIcon; | ||||
|         this.pid = pid; | ||||
|         this.app = this._getApp(appId); | ||||
|  | ||||
|     _init: function(title, pid, sender, trayIcon) { | ||||
|         this.parent(title); | ||||
|  | ||||
|         this.initialTitle = title; | ||||
|  | ||||
|         if (this.app) | ||||
|             this.title = this.app.get_name(); | ||||
|         else | ||||
|             this.useNotificationIcon = true; | ||||
|  | ||||
|         this.pid = pid; | ||||
|         if (sender) | ||||
|             this._nameWatcherId = Gio.DBus.session.watch_name(sender, | ||||
|                                                               Gio.BusNameWatcherFlags.NONE, | ||||
| @@ -664,19 +511,16 @@ const Source = new Lang.Class({ | ||||
|         else | ||||
|             this._nameWatcherId = 0; | ||||
|  | ||||
|         if (this.trayIcon) { | ||||
|             // Try again finding the app, using the WM_CLASS from the tray icon | ||||
|             this._setSummaryIcon(this.trayIcon); | ||||
|             this.useNotificationIcon = false; | ||||
|         } | ||||
|     }, | ||||
|         this._setApp(); | ||||
|         if (this.app) | ||||
|             this.title = this.app.get_name(); | ||||
|         else | ||||
|             this.useNotificationIcon = true; | ||||
|  | ||||
|     _createPolicy: function() { | ||||
|         if (this.app) { | ||||
|             let id = this.app.get_id().replace(/\.desktop$/,''); | ||||
|             return new NotificationApplicationPolicy(id); | ||||
|         } else { | ||||
|             return new NotificationGenericPolicy(); | ||||
|         this.trayIcon = trayIcon; | ||||
|         if (this.trayIcon) { | ||||
|            this._setSummaryIcon(this.trayIcon); | ||||
|            this.useNotificationIcon = false; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
| @@ -690,11 +534,11 @@ const Source = new Lang.Class({ | ||||
|             this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     processNotification: function(notification, gicon) { | ||||
|         if (gicon) | ||||
|             this._gicon = gicon; | ||||
|         if (!this.trayIcon) | ||||
|             this.iconUpdated(); | ||||
|     processNotification: function(notification, icon) { | ||||
|         if (!this.app) | ||||
|             this._setApp(); | ||||
|         if (!this.app && icon) | ||||
|             this._setSummaryIcon(icon); | ||||
|  | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         if (notification.resident && this.app && tracker.focus_app == this.app) | ||||
| @@ -703,56 +547,42 @@ const Source = new Lang.Class({ | ||||
|             this.notify(notification); | ||||
|     }, | ||||
|  | ||||
|     handleSummaryClick: function(button) { | ||||
|     handleSummaryClick: function() { | ||||
|         if (!this.trayIcon) | ||||
|             return false; | ||||
|  | ||||
|         let event = Clutter.get_current_event(); | ||||
|         if (event.type() != Clutter.EventType.BUTTON_RELEASE) | ||||
|             return false; | ||||
|  | ||||
|         // Left clicks are passed through only where there aren't unacknowledged | ||||
|         // notifications, so it possible to open them in summary mode; right | ||||
|         // clicks are always forwarded, as the right click menu is not useful for | ||||
|         // tray icons | ||||
|         if (button == 1 && | ||||
|         if (event.get_button() == 1 && | ||||
|             this.notifications.length > 0) | ||||
|             return false; | ||||
|  | ||||
|         let id = global.connect('notify::stage-input-mode', Lang.bind(this, function () { | ||||
|             global.disconnect(id); | ||||
|         if (Main.overview.visible) { | ||||
|             // We can't just connect to Main.overview's 'hidden' signal, | ||||
|             // because it's emitted *before* it calls popModal()... | ||||
|             let id = global.connect('notify::stage-input-mode', Lang.bind(this, | ||||
|                 function () { | ||||
|                     global.disconnect(id); | ||||
|                     this.trayIcon.click(event); | ||||
|                 })); | ||||
|             Main.overview.hide(); | ||||
|         } else { | ||||
|             this.trayIcon.click(event); | ||||
|         })); | ||||
|  | ||||
|         Main.overview.hide(); | ||||
|         } | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _getApp: function(appId) { | ||||
|         let app; | ||||
|  | ||||
|         app = Shell.WindowTracker.get_default().get_app_from_pid(this.pid); | ||||
|         if (app != null) | ||||
|             return app; | ||||
|  | ||||
|         if (this.trayIcon) { | ||||
|             app = Shell.AppSystem.get_default().lookup_wmclass(this.trayIcon.wm_class); | ||||
|             if (app != null) | ||||
|                 return app; | ||||
|         } | ||||
|  | ||||
|         if (appId) { | ||||
|             app = Shell.AppSystem.get_default().lookup_app(appId + '.desktop'); | ||||
|             if (app != null) | ||||
|                 return app; | ||||
|         } | ||||
|  | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     _setApp: function(appId) { | ||||
|     _setApp: function() { | ||||
|         if (this.app) | ||||
|             return; | ||||
|  | ||||
|         this.app = this._getApp(appId); | ||||
|         this.app = Shell.WindowTracker.get_default().get_app_from_pid(this.pid); | ||||
|         if (!this.app) | ||||
|             return; | ||||
|  | ||||
| @@ -760,20 +590,10 @@ const Source = new Lang.Class({ | ||||
|         // notification-based icons (ie, not a trayicon) or if it was unset before | ||||
|         if (!this.trayIcon) { | ||||
|             this.useNotificationIcon = false; | ||||
|             this.iconUpdated(); | ||||
|             this._setSummaryIcon(this.app.create_icon_texture (this.ICON_SIZE)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     setTitle: function(title) { | ||||
|         // Do nothing if .app is set, we don't want to override the | ||||
|         // app name with whatever is provided through libnotify (usually | ||||
|         // garbage) | ||||
|         if (this.app) | ||||
|             return; | ||||
|  | ||||
|         this.parent(title); | ||||
|     }, | ||||
|  | ||||
|     open: function(notification) { | ||||
|         this.destroyNonResidentNotifications(); | ||||
|         this.openApp(); | ||||
| @@ -802,20 +622,5 @@ const Source = new Lang.Class({ | ||||
|         } | ||||
|  | ||||
|         this.parent(); | ||||
|     }, | ||||
|  | ||||
|     createIcon: function(size) { | ||||
|         if (this.trayIcon) { | ||||
|             return new Clutter.Clone({ width: size, | ||||
|                                        height: size, | ||||
|                                        source: this.trayIcon }); | ||||
|         } else if (this.app) { | ||||
|             return this.app.create_icon_texture(size); | ||||
|         } else if (this._gicon) { | ||||
|             return new St.Icon({ gicon: this._gicon, | ||||
|                                  icon_size: size }); | ||||
|         } else { | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -10,28 +10,43 @@ const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Gdk = imports.gi.Gdk; | ||||
|  | ||||
| const Background = imports.ui.background; | ||||
| const AppDisplay = imports.ui.appDisplay; | ||||
| const ContactDisplay = imports.ui.contactDisplay; | ||||
| const Dash = imports.ui.dash; | ||||
| const DND = imports.ui.dnd; | ||||
| const Lightbox = imports.ui.lightbox; | ||||
| const Main = imports.ui.main; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
| const OverviewControls = imports.ui.overviewControls; | ||||
| const Panel = imports.ui.panel; | ||||
| const Params = imports.misc.params; | ||||
| const PlaceDisplay = imports.ui.placeDisplay; | ||||
| const RemoteSearch = imports.ui.remoteSearch; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const ViewSelector = imports.ui.viewSelector; | ||||
| const Wanda = imports.ui.wanda; | ||||
| const WorkspacesView = imports.ui.workspacesView; | ||||
| const WorkspaceThumbnail = imports.ui.workspaceThumbnail; | ||||
|  | ||||
| // Time for initial animation going into Overview mode | ||||
| const ANIMATION_TIME = 0.25; | ||||
|  | ||||
| // 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 | ||||
| const SHADE_ANIMATION_TIME = .20; | ||||
| // We split the screen vertically between the dash and the view selector. | ||||
| const DASH_SPLIT_FRACTION = 0.1; | ||||
|  | ||||
| const DND_WINDOW_SWITCH_TIMEOUT = 1250; | ||||
|  | ||||
| const SwipeScrollDirection = { | ||||
|     NONE: 0, | ||||
|     HORIZONTAL: 1, | ||||
|     VERTICAL: 2 | ||||
| }; | ||||
|  | ||||
| const SwipeScrollResult = { | ||||
|     CANCEL: 0, | ||||
|     SWIPE: 1, | ||||
|     CLICK: 2 | ||||
| }; | ||||
|  | ||||
| const ShellInfo = new Lang.Class({ | ||||
|     Name: 'ShellInfo', | ||||
|  | ||||
| @@ -49,14 +64,7 @@ const ShellInfo = new Lang.Class({ | ||||
|             this._source.destroy(); | ||||
|     }, | ||||
|  | ||||
|     setMessage: function(text, options) { | ||||
|         options = Params.parse(options, { undoCallback: null, | ||||
|                                           forFeedback: false | ||||
|                                         }); | ||||
|  | ||||
|         let undoCallback = options.undoCallback; | ||||
|         let forFeedback = options.forFeedback; | ||||
|  | ||||
|     setMessage: function(text, undoCallback, undoLabel) { | ||||
|         if (this._source == null) { | ||||
|             this._source = new MessageTray.SystemNotificationSource(); | ||||
|             this._source.connect('destroy', Lang.bind(this, | ||||
| @@ -69,17 +77,19 @@ const ShellInfo = new Lang.Class({ | ||||
|         let notification = null; | ||||
|         if (this._source.notifications.length == 0) { | ||||
|             notification = new MessageTray.Notification(this._source, text, null); | ||||
|             notification.setTransient(true); | ||||
|             notification.setForFeedback(forFeedback); | ||||
|         } else { | ||||
|             notification = this._source.notifications[0]; | ||||
|             notification.update(text, null, { clear: true }); | ||||
|         } | ||||
|  | ||||
|         notification.setTransient(true); | ||||
|  | ||||
|         this._undoCallback = undoCallback; | ||||
|         if (undoCallback) { | ||||
|             notification.addButton('system-undo', _("Undo")); | ||||
|             notification.connect('action-invoked', Lang.bind(this, this._onUndoClicked)); | ||||
|             notification.addButton('system-undo', | ||||
|                                    undoLabel ? undoLabel : _("Undo")); | ||||
|             notification.connect('action-invoked', | ||||
|                                  Lang.bind(this, this._onUndoClicked)); | ||||
|         } | ||||
|  | ||||
|         this._source.notify(notification); | ||||
| @@ -89,67 +99,74 @@ const ShellInfo = new Lang.Class({ | ||||
| const Overview = new Lang.Class({ | ||||
|     Name: 'Overview', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._overviewCreated = false; | ||||
|         this._initCalled = false; | ||||
|     _init : function(params) { | ||||
|         params = Params.parse(params, { isDummy: false }); | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|         this._sessionUpdated(); | ||||
|     }, | ||||
|         this.isDummy = params.isDummy; | ||||
|  | ||||
|     _createOverview: function() { | ||||
|         if (this._overviewCreated) | ||||
|         // We only have an overview in user sessions, so | ||||
|         // create a dummy overview in other cases | ||||
|         if (this.isDummy) { | ||||
|             this.animationInProgress = false; | ||||
|             this.visible = false; | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         this._overviewCreated = true; | ||||
|  | ||||
|         // The main Background actors are inside global.window_group which are | ||||
|         // The main BackgroundActor is inside global.window_group which is | ||||
|         // hidden when displaying the overview, so we create a new | ||||
|         // one. Instances of this class share a single CoglTexture behind the | ||||
|         // scenes which allows us to show the background with different | ||||
|         // rendering options without duplicating the texture data. | ||||
|         let monitor = Main.layoutManager.primaryMonitor; | ||||
|         this._background = Meta.BackgroundActor.new_for_screen(global.screen); | ||||
|         this._background.hide(); | ||||
|         global.overlay_group.add_actor(this._background); | ||||
|  | ||||
|         this._desktopFade = new St.Bin(); | ||||
|         global.overlay_group.add_actor(this._desktopFade); | ||||
|  | ||||
|         this._spacing = 0; | ||||
|  | ||||
|         /* Translators: This is the main view to select | ||||
|            activities. See also note for "Activities" string. */ | ||||
|         this._overview = new St.BoxLayout({ name: 'overview', | ||||
|                                             accessible_name: _("Overview"), | ||||
|                                             reactive: true, | ||||
|                                             vertical: true }); | ||||
|         this._overview._delegate = this; | ||||
|         this._group = new St.Widget({ name: 'overview', | ||||
|                                       accessible_name: _("Overview"), | ||||
|                                       reactive: true }); | ||||
|         this._group._delegate = this; | ||||
|         this._group.connect('style-changed', | ||||
|             Lang.bind(this, function() { | ||||
|                 let node = this._group.get_theme_node(); | ||||
|                 let spacing = node.get_length('spacing'); | ||||
|                 if (spacing != this._spacing) { | ||||
|                     this._spacing = spacing; | ||||
|                     this._relayout(); | ||||
|                 } | ||||
|             })); | ||||
|  | ||||
|         this._group = new St.BoxLayout({ name: 'overview-group', | ||||
|                                          reactive: true, | ||||
|                                          clip_to_allocation: true }); | ||||
|         this._scrollDirection = SwipeScrollDirection.NONE; | ||||
|         this._scrollAdjustment = null; | ||||
|         this._capturedEventId = 0; | ||||
|         this._buttonPressId = 0; | ||||
|  | ||||
|         this._backgroundGroup = new Meta.BackgroundGroup(); | ||||
|         global.overlay_group.add_child(this._backgroundGroup); | ||||
|         this._backgroundGroup.hide(); | ||||
|         this._bgManagers = []; | ||||
|         this._workspacesDisplay = null; | ||||
|  | ||||
|         this.visible = false;           // animating to overview, in overview, animating out | ||||
|         this._shown = false;            // show() and not hide() | ||||
|         this._shownTemporarily = false; // showTemporarily() and not hideTemporarily() | ||||
|         this._modal = false;            // have a modal grab | ||||
|         this.animationInProgress = false; | ||||
|         this.visibleTarget = false; | ||||
|         this._hideInProgress = false; | ||||
|  | ||||
|         // During transitions, we raise this to the top to avoid having the overview | ||||
|         // area be reactive; it causes too many issues such as double clicks on | ||||
|         // Dash elements, or mouseover handlers in the workspaces. | ||||
|         this._coverPane = new Clutter.Rectangle({ opacity: 0, | ||||
|                                                   reactive: true }); | ||||
|         this._overview.add_actor(this._coverPane); | ||||
|         this._group.add_actor(this._coverPane); | ||||
|         this._coverPane.connect('event', Lang.bind(this, function (actor, event) { return true; })); | ||||
|  | ||||
|         this._overview.hide(); | ||||
|         global.overlay_group.add_actor(this._overview); | ||||
|  | ||||
|         this._group.hide(); | ||||
|         global.overlay_group.add_actor(this._group); | ||||
|  | ||||
|         this._coverPane.hide(); | ||||
|  | ||||
| @@ -161,72 +178,11 @@ const Overview = new Lang.Class({ | ||||
|         Main.xdndHandler.connect('drag-begin', Lang.bind(this, this._onDragBegin)); | ||||
|         Main.xdndHandler.connect('drag-end', Lang.bind(this, this._onDragEnd)); | ||||
|  | ||||
|         global.screen.connect('restacked', Lang.bind(this, this._onRestacked)); | ||||
|         this._group.connect('scroll-event', Lang.bind(this, this._onScrollEvent)); | ||||
|  | ||||
|         this._windowSwitchTimeoutId = 0; | ||||
|         this._windowSwitchTimestamp = 0; | ||||
|         this._lastActiveWorkspaceIndex = -1; | ||||
|         this._lastHoveredWindow = null; | ||||
|         this._needsFakePointerEvent = false; | ||||
|  | ||||
|         if (this._initCalled) | ||||
|             this.init(); | ||||
|     }, | ||||
|  | ||||
|     _updateBackgrounds: function() { | ||||
|         for (let i = 0; i < this._bgManagers.length; i++) | ||||
|             this._bgManagers[i].destroy(); | ||||
|  | ||||
|         this._bgManagers = []; | ||||
|  | ||||
|         for (let i = 0; i < Main.layoutManager.monitors.length; i++) { | ||||
|             let bgManager = new Background.BackgroundManager({ container: this._backgroundGroup, | ||||
|                                                                monitorIndex: i, | ||||
|                                                                effects: Meta.BackgroundEffects.VIGNETTE }); | ||||
|             this._bgManagers.push(bgManager); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _unshadeBackgrounds: function() { | ||||
|         let backgrounds = this._backgroundGroup.get_children(); | ||||
|         for (let i = 0; i < backgrounds.length; i++) { | ||||
|             let background = backgrounds[i]._delegate; | ||||
|  | ||||
|             Tweener.addTween(background, | ||||
|                              { brightness: 1.0, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|             Tweener.addTween(background, | ||||
|                              { vignetteSharpness: 0.0, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _shadeBackgrounds: function() { | ||||
|         let backgrounds = this._backgroundGroup.get_children(); | ||||
|         for (let i = 0; i < backgrounds.length; i++) { | ||||
|             let background = backgrounds[i]._delegate; | ||||
|  | ||||
|             Tweener.addTween(background, | ||||
|                              { brightness: 0.8, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|             Tweener.addTween(background, | ||||
|                              { vignetteSharpness: 0.7, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _sessionUpdated: function() { | ||||
|         this.isDummy = !Main.sessionMode.hasOverview; | ||||
|         this._createOverview(); | ||||
|     }, | ||||
|  | ||||
|     // The members we construct that are implemented in JS might | ||||
| @@ -234,58 +190,46 @@ const Overview = new Lang.Class({ | ||||
|     // signal handlers and so forth. So we create them after | ||||
|     // construction in this init() method. | ||||
|     init: function() { | ||||
|         this._initCalled = true; | ||||
|  | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         this._shellInfo = new ShellInfo(); | ||||
|  | ||||
|         // Add a clone of the panel to the overview so spacing and such is | ||||
|         // automatic | ||||
|         this._panelGhost = new St.Bin({ child: new Clutter.Clone({ source: Main.panel.actor }), | ||||
|                                         reactive: false, | ||||
|                                         opacity: 0 }); | ||||
|         this._overview.add_actor(this._panelGhost); | ||||
|         this._viewSelector = new ViewSelector.ViewSelector(); | ||||
|         this._group.add_actor(this._viewSelector.actor); | ||||
|  | ||||
|         this._searchEntry = new St.Entry({ name: 'searchEntry', | ||||
|                                            /* Translators: this is the text displayed | ||||
|                                               in the search entry when no search is | ||||
|                                               active; it should not exceed ~30 | ||||
|                                               characters. */ | ||||
|                                            hint_text: _("Type to search…"), | ||||
|                                            track_hover: true, | ||||
|                                            can_focus: true }); | ||||
|         this._searchEntryBin = new St.Bin({ child: this._searchEntry, | ||||
|                                             x_align: St.Align.MIDDLE }); | ||||
|         this._overview.add_actor(this._searchEntryBin); | ||||
|         this._workspacesDisplay = new WorkspacesView.WorkspacesDisplay(); | ||||
|         this._viewSelector.addViewTab('windows', _("Windows"), this._workspacesDisplay.actor, 'text-x-generic'); | ||||
|  | ||||
|         // Create controls | ||||
|         this._dash = new Dash.Dash(); | ||||
|         this._viewSelector = new ViewSelector.ViewSelector(this._searchEntry, | ||||
|                                                            this._dash.showAppsButton); | ||||
|         this._thumbnailsBox = new WorkspaceThumbnail.ThumbnailsBox(); | ||||
|         this._controls = new OverviewControls.ControlsManager(this._dash, | ||||
|                                                               this._thumbnailsBox, | ||||
|                                                               this._viewSelector); | ||||
|         let appView = new AppDisplay.AllAppDisplay(); | ||||
|         this._viewSelector.addViewTab('applications', _("Applications"), appView.actor, 'system-run'); | ||||
|  | ||||
|         // Pack all the actors into the group | ||||
|         this._group.add_actor(this._controls.dashActor); | ||||
|         this._group.add(this._viewSelector.actor, { x_fill: true, | ||||
|                                                     expand: true }); | ||||
|         this._group.add_actor(this._controls.thumbnailsActor); | ||||
|         // Default search providers | ||||
|         // Wanda comes obviously first | ||||
|         this.addSearchProvider(new Wanda.WandaSearchProvider()); | ||||
|         this.addSearchProvider(new AppDisplay.AppSearchProvider()); | ||||
|         this.addSearchProvider(new AppDisplay.SettingsSearchProvider()); | ||||
|         this.addSearchProvider(new PlaceDisplay.PlaceSearchProvider()); | ||||
|         this.addSearchProvider(new ContactDisplay.ContactSearchProvider()); | ||||
|  | ||||
|         // Add our same-line elements after the search entry | ||||
|         this._overview.add(this._group, { y_fill: true, | ||||
|                                           expand: true }); | ||||
|         // Load remote search providers provided by applications | ||||
|         RemoteSearch.loadRemoteSearchProviders(Lang.bind(this, this.addSearchProvider)); | ||||
|  | ||||
|         // TODO - recalculate everything when desktop size changes | ||||
|         this._dash = new Dash.Dash(); | ||||
|         this._group.add_actor(this._dash.actor); | ||||
|         this._dash.actor.add_constraint(this._viewSelector.constrainY); | ||||
|         this._dash.actor.add_constraint(this._viewSelector.constrainHeight); | ||||
|         this.dashIconSize = this._dash.iconSize; | ||||
|         this._dash.connect('icon-size-changed', | ||||
|                            Lang.bind(this, function() { | ||||
|                                this.dashIconSize = this._dash.iconSize; | ||||
|                            })); | ||||
|  | ||||
|         // Translators: this is the name of the dock/favorites area on | ||||
|         // the left of the overview | ||||
|         Main.ctrlAltTabManager.addGroup(this._dash.actor, _("Dash"), 'user-bookmarks'); | ||||
|  | ||||
|         Main.layoutManager.connect('monitors-changed', Lang.bind(this, this._relayout)); | ||||
|         this._relayout(); | ||||
|     }, | ||||
| @@ -298,16 +242,11 @@ const Overview = new Lang.Class({ | ||||
|         this._viewSelector.removeSearchProvider(provider); | ||||
|     }, | ||||
|  | ||||
|     // | ||||
|     // options: | ||||
|     //  - undoCallback (function): the callback to be called if undo support is needed | ||||
|     //  - forFeedback (boolean): whether the message is for direct feedback of a user action | ||||
|     // | ||||
|     setMessage: function(text, options) { | ||||
|     setMessage: function(text, undoCallback, undoLabel) { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         this._shellInfo.setMessage(text, options); | ||||
|         this._shellInfo.setMessage(text, undoCallback, undoLabel); | ||||
|     }, | ||||
|  | ||||
|     _onDragBegin: function() { | ||||
| @@ -378,15 +317,159 @@ const Overview = new Lang.Class({ | ||||
|         return DND.DragMotionResult.CONTINUE; | ||||
|     }, | ||||
|  | ||||
|     _onScrollEvent: function(actor, event) { | ||||
|         this.emit('scroll-event', event); | ||||
|     }, | ||||
|  | ||||
|     addAction: function(action) { | ||||
|     setScrollAdjustment: function(adjustment, direction) { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         this._overview.add_action(action); | ||||
|         this._scrollAdjustment = adjustment; | ||||
|         if (this._scrollAdjustment == null) | ||||
|             this._scrollDirection = SwipeScrollDirection.NONE; | ||||
|         else | ||||
|             this._scrollDirection = direction; | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress: function(actor, event) { | ||||
|         if (this._scrollDirection == SwipeScrollDirection.NONE | ||||
|             || event.get_button() != 1) | ||||
|             return; | ||||
|  | ||||
|         let [stageX, stageY] = event.get_coords(); | ||||
|         this._dragStartX = this._dragX = stageX; | ||||
|         this._dragStartY = this._dragY = stageY; | ||||
|         this._dragStartValue = this._scrollAdjustment.value; | ||||
|         this._lastMotionTime = -1; // used to track "stopping" while swipe-scrolling | ||||
|         this._capturedEventId = global.stage.connect('captured-event', | ||||
|             Lang.bind(this, this._onCapturedEvent)); | ||||
|         this.emit('swipe-scroll-begin'); | ||||
|     }, | ||||
|  | ||||
|     _onCapturedEvent: function(actor, event) { | ||||
|         let stageX, stageY; | ||||
|         let threshold = Gtk.Settings.get_default().gtk_dnd_drag_threshold; | ||||
|  | ||||
|         switch(event.type()) { | ||||
|             case Clutter.EventType.BUTTON_RELEASE: | ||||
|                 [stageX, stageY] = event.get_coords(); | ||||
|  | ||||
|                 // default to snapping back to the original value | ||||
|                 let newValue = this._dragStartValue; | ||||
|  | ||||
|                 let minValue = this._scrollAdjustment.lower; | ||||
|                 let maxValue = this._scrollAdjustment.upper - this._scrollAdjustment.page_size; | ||||
|  | ||||
|                 let direction; | ||||
|                 if (this._scrollDirection == SwipeScrollDirection.HORIZONTAL) { | ||||
|                     direction = stageX > this._dragStartX ? -1 : 1; | ||||
|                     if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) | ||||
|                         direction *= -1; | ||||
|                 } else { | ||||
|                     direction = stageY > this._dragStartY ? -1 : 1; | ||||
|                 } | ||||
|  | ||||
|                 // We default to scroll a full page size; both the first | ||||
|                 // and the last page may be smaller though, so we need to | ||||
|                 // adjust difference in those cases. | ||||
|                 let difference = direction * this._scrollAdjustment.page_size; | ||||
|                 if (this._dragStartValue + difference > maxValue) | ||||
|                     difference = maxValue - this._dragStartValue; | ||||
|                 else if (this._dragStartValue + difference < minValue) | ||||
|                     difference = minValue - this._dragStartValue; | ||||
|  | ||||
|                 // If the user has moved more than half the scroll | ||||
|                 // difference, we want to "settle" to the new value | ||||
|                 // even if the user stops dragging rather "throws" by | ||||
|                 // releasing during the drag. | ||||
|                 let distance = this._dragStartValue - this._scrollAdjustment.value; | ||||
|                 let noStop = Math.abs(distance / difference) > 0.5; | ||||
|  | ||||
|                 // We detect if the user is stopped by comparing the | ||||
|                 // timestamp of the button release with the timestamp of | ||||
|                 // the last motion. Experimentally, a difference of 0 or 1 | ||||
|                 // millisecond indicates that the mouse is in motion, a | ||||
|                 // larger difference indicates that the mouse is stopped. | ||||
|                 if ((this._lastMotionTime > 0 && | ||||
|                      this._lastMotionTime > event.get_time() - 2) || | ||||
|                     noStop) { | ||||
|                     if (this._dragStartValue + difference >= minValue && | ||||
|                         this._dragStartValue + difference <= maxValue) | ||||
|                         newValue += difference; | ||||
|                 } | ||||
|  | ||||
|                 let result; | ||||
|  | ||||
|                 // See if the user has moved the mouse enough to trigger | ||||
|                 // a drag | ||||
|                 if (Math.abs(stageX - this._dragStartX) < threshold && | ||||
|                     Math.abs(stageY - this._dragStartY) < threshold) { | ||||
|                     // no motion? It's a click! | ||||
|                     result = SwipeScrollResult.CLICK; | ||||
|                     this.emit('swipe-scroll-end', result); | ||||
|                 } else { | ||||
|                     if (newValue == this._dragStartValue) | ||||
|                         result = SwipeScrollResult.CANCEL; | ||||
|                     else | ||||
|                         result = SwipeScrollResult.SWIPE; | ||||
|  | ||||
|                     // The event capture handler is disconnected | ||||
|                     // while scrolling to the final position, so | ||||
|                     // to avoid undesired prelights we raise | ||||
|                     // the cover pane. | ||||
|                     this._coverPane.raise_top(); | ||||
|                     this._coverPane.show(); | ||||
|  | ||||
|                     Tweener.addTween(this._scrollAdjustment, | ||||
|                                      { value: newValue, | ||||
|                                        time: ANIMATION_TIME, | ||||
|                                        transition: 'easeOutQuad', | ||||
|                                        onCompleteScope: this, | ||||
|                                        onComplete: function() { | ||||
|                                           this._coverPane.hide(); | ||||
|                                           this.emit('swipe-scroll-end', | ||||
|                                                     result); | ||||
|                                        } | ||||
|                                      }); | ||||
|                 } | ||||
|  | ||||
|                 global.stage.disconnect(this._capturedEventId); | ||||
|                 this._capturedEventId = 0; | ||||
|  | ||||
|                 return result != SwipeScrollResult.CLICK; | ||||
|  | ||||
|             case Clutter.EventType.MOTION: | ||||
|                 [stageX, stageY] = event.get_coords(); | ||||
|                 let dx = this._dragX - stageX; | ||||
|                 let dy = this._dragY - stageY; | ||||
|                 let primary = Main.layoutManager.primaryMonitor; | ||||
|  | ||||
|                 this._dragX = stageX; | ||||
|                 this._dragY = stageY; | ||||
|                 this._lastMotionTime = event.get_time(); | ||||
|  | ||||
|                 // See if the user has moved the mouse enough to trigger | ||||
|                 // a drag | ||||
|                 if (Math.abs(stageX - this._dragStartX) < threshold && | ||||
|                     Math.abs(stageY - this._dragStartY) < threshold) | ||||
|                     return true; | ||||
|  | ||||
|                 if (this._scrollDirection == SwipeScrollDirection.HORIZONTAL) { | ||||
|                     if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) | ||||
|                         this._scrollAdjustment.value -= (dx / primary.width) * this._scrollAdjustment.page_size; | ||||
|                     else | ||||
|                         this._scrollAdjustment.value += (dx / primary.width) * this._scrollAdjustment.page_size; | ||||
|                 } else { | ||||
|                     this._scrollAdjustment.value += (dy / primary.height) * this._scrollAdjustment.page_size; | ||||
|                 } | ||||
|  | ||||
|                 return true; | ||||
|  | ||||
|             // Block enter/leave events to avoid prelights | ||||
|             // during swipe-scroll | ||||
|             case Clutter.EventType.ENTER: | ||||
|             case Clutter.EventType.LEAVE: | ||||
|                 return true; | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _getDesktopClone: function() { | ||||
| @@ -396,9 +479,7 @@ const Overview = new Lang.Class({ | ||||
|         if (windows.length == 0) | ||||
|             return null; | ||||
|  | ||||
|         let window = windows[0]; | ||||
|         let clone = new Clutter.Clone({ source: window.get_texture(), | ||||
|                                         x: window.x, y: window.y }); | ||||
|         let clone = new Clutter.Clone({ source: windows[0].get_texture() }); | ||||
|         clone.source.connect('destroy', Lang.bind(this, function() { | ||||
|             clone.destroy(); | ||||
|         })); | ||||
| @@ -412,27 +493,35 @@ const Overview = new Lang.Class({ | ||||
|         this.hide(); | ||||
|  | ||||
|         let primary = Main.layoutManager.primaryMonitor; | ||||
|         let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex); | ||||
|         let rtl = (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL); | ||||
|  | ||||
|         this._overview.set_position(primary.x, primary.y); | ||||
|         this._overview.set_size(primary.width, primary.height); | ||||
|         let contentY = Main.panel.actor.height; | ||||
|         let contentHeight = primary.height - contentY - Main.messageTray.actor.height; | ||||
|  | ||||
|         this._coverPane.set_position(0, workArea.y); | ||||
|         this._coverPane.set_size(workArea.width, workArea.height); | ||||
|         this._group.set_position(primary.x, primary.y); | ||||
|         this._group.set_size(primary.width, primary.height); | ||||
|  | ||||
|         this._updateBackgrounds(); | ||||
|     }, | ||||
|         this._coverPane.set_position(0, contentY); | ||||
|         this._coverPane.set_size(primary.width, contentHeight); | ||||
|  | ||||
|     _onRestacked: function() { | ||||
|         let stack = global.get_window_actors(); | ||||
|         let stackIndices = {}; | ||||
|         let dashWidth = Math.round(DASH_SPLIT_FRACTION * primary.width); | ||||
|         let viewWidth = primary.width - dashWidth - this._spacing; | ||||
|         let viewHeight = contentHeight - 2 * this._spacing; | ||||
|         let viewY = contentY + this._spacing; | ||||
|         let viewX = rtl ? 0 : dashWidth + this._spacing; | ||||
|  | ||||
|         for (let i = 0; i < stack.length; i++) { | ||||
|             // Use the stable sequence for an integer to use as a hash key | ||||
|             stackIndices[stack[i].get_meta_window().get_stable_sequence()] = i; | ||||
|         // Set the dash's x position - y is handled by a constraint | ||||
|         let dashX; | ||||
|         if (rtl) { | ||||
|             this._dash.actor.set_anchor_point_from_gravity(Clutter.Gravity.NORTH_EAST); | ||||
|             dashX = primary.width; | ||||
|         } else { | ||||
|             dashX = 0; | ||||
|         } | ||||
|         this._dash.actor.set_x(dashX); | ||||
|  | ||||
|         this.emit('windows-restacked', stackIndices); | ||||
|         this._viewSelector.actor.set_position(viewX, viewY); | ||||
|         this._viewSelector.actor.set_size(viewWidth, viewHeight); | ||||
|     }, | ||||
|  | ||||
|     //// Public methods //// | ||||
| @@ -469,33 +558,15 @@ const Overview = new Lang.Class({ | ||||
|             return; | ||||
|         if (this._shown) | ||||
|             return; | ||||
|         this._shown = true; | ||||
|         this._syncInputMode(); | ||||
|         if (!this._modal) | ||||
|         // Do this manually instead of using _syncInputMode, to handle failure | ||||
|         if (!Main.pushModal(this._group)) | ||||
|             return; | ||||
|         this._modal = true; | ||||
|         this._animateVisible(); | ||||
|     }, | ||||
|         this._shown = true; | ||||
|  | ||||
|     fadeInDesktop: function() { | ||||
|             this._desktopFade.opacity = 0; | ||||
|             this._desktopFade.show(); | ||||
|             Tweener.addTween(this._desktopFade, | ||||
|                              { opacity: 255, | ||||
|                                time: ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' }); | ||||
|     }, | ||||
|  | ||||
|     fadeOutDesktop: function() { | ||||
|         if (!this._desktopFade.child) | ||||
|             this._desktopFade.child = this._getDesktopClone(); | ||||
|  | ||||
|         this._desktopFade.opacity = 255; | ||||
|         this._desktopFade.show(); | ||||
|         Tweener.addTween(this._desktopFade, | ||||
|                          { opacity: 0, | ||||
|                            time: ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad' | ||||
|                          }); | ||||
|         this._buttonPressId = this._group.connect('button-press-event', | ||||
|             Lang.bind(this, this._onButtonPress)); | ||||
|     }, | ||||
|  | ||||
|     _animateVisible: function() { | ||||
| @@ -504,7 +575,6 @@ const Overview = new Lang.Class({ | ||||
|  | ||||
|         this.visible = true; | ||||
|         this.animationInProgress = true; | ||||
|         this.visibleTarget = true; | ||||
|  | ||||
|         // All the the actors in the window group are completely obscured, | ||||
|         // hiding the group holding them while the Overview is displayed greatly | ||||
| @@ -517,20 +587,38 @@ const Overview = new Lang.Class({ | ||||
|         // Disable unredirection while in the overview | ||||
|         Meta.disable_unredirect_for_screen(global.screen); | ||||
|         global.window_group.hide(); | ||||
|         global.top_window_group.hide(); | ||||
|         this._overview.show(); | ||||
|         this._backgroundGroup.show(); | ||||
|         this._viewSelector.show(); | ||||
|         this._group.show(); | ||||
|         this._background.show(); | ||||
|  | ||||
|         this._overview.opacity = 0; | ||||
|         Tweener.addTween(this._overview, | ||||
|         this._workspacesDisplay.show(); | ||||
|  | ||||
|         if (!this._desktopFade.child) | ||||
|             this._desktopFade.child = this._getDesktopClone(); | ||||
|  | ||||
|         if (!this._workspacesDisplay.activeWorkspaceHasMaximizedWindows()) { | ||||
|             this._desktopFade.opacity = 255; | ||||
|             this._desktopFade.show(); | ||||
|             Tweener.addTween(this._desktopFade, | ||||
|                              { opacity: 0, | ||||
|                                time: ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|         } | ||||
|  | ||||
|         this._group.opacity = 0; | ||||
|         Tweener.addTween(this._group, | ||||
|                          { opacity: 255, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            time: ANIMATION_TIME, | ||||
|                            onComplete: this._showDone, | ||||
|                            onCompleteScope: this | ||||
|                          }); | ||||
|         this._shadeBackgrounds(); | ||||
|  | ||||
|         Tweener.addTween(this._background, | ||||
|                          { dim_factor: 0.4, | ||||
|                            time: ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad' | ||||
|                          }); | ||||
|  | ||||
|         this._coverPane.raise_top(); | ||||
|         this._coverPane.show(); | ||||
| @@ -570,6 +658,10 @@ const Overview = new Lang.Class({ | ||||
|  | ||||
|         this._shown = false; | ||||
|         this._syncInputMode(); | ||||
|  | ||||
|         if (this._buttonPressId > 0) | ||||
|             this._group.disconnect(this._buttonPressId); | ||||
|         this._buttonPressId = 0; | ||||
|     }, | ||||
|  | ||||
|     // hideTemporarily: | ||||
| @@ -593,7 +685,7 @@ const Overview = new Lang.Class({ | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         if (this.visible) | ||||
|         if (this._shown) | ||||
|             this.hide(); | ||||
|         else | ||||
|             this.show(); | ||||
| @@ -610,21 +702,20 @@ const Overview = new Lang.Class({ | ||||
|  | ||||
|         if (this._shown) { | ||||
|             if (!this._modal) { | ||||
|                 if (Main.pushModal(this._overview, | ||||
|                                    { keybindingMode: Shell.KeyBindingMode.OVERVIEW })) | ||||
|                 if (Main.pushModal(this._group)) | ||||
|                     this._modal = true; | ||||
|                 else | ||||
|                     this.hide(); | ||||
|             } | ||||
|         } else if (this._shownTemporarily) { | ||||
|             if (this._modal) { | ||||
|                 Main.popModal(this._overview); | ||||
|                 Main.popModal(this._group); | ||||
|                 this._modal = false; | ||||
|             } | ||||
|             global.stage_input_mode = Shell.StageInputMode.FULLSCREEN; | ||||
|         } else { | ||||
|             if (this._modal) { | ||||
|                 Main.popModal(this._overview); | ||||
|                 Main.popModal(this._group); | ||||
|                 this._modal = false; | ||||
|             } | ||||
|             else if (global.stage_input_mode == Shell.StageInputMode.FULLSCREEN) | ||||
| @@ -637,19 +728,33 @@ const Overview = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         this.animationInProgress = true; | ||||
|         this.visibleTarget = false; | ||||
|         this._hideInProgress = true; | ||||
|  | ||||
|         this._viewSelector.zoomFromOverview(); | ||||
|         if (!this._workspacesDisplay.activeWorkspaceHasMaximizedWindows()) { | ||||
|             this._desktopFade.opacity = 0; | ||||
|             this._desktopFade.show(); | ||||
|             Tweener.addTween(this._desktopFade, | ||||
|                              { opacity: 255, | ||||
|                                time: ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' }); | ||||
|         } | ||||
|  | ||||
|         this._workspacesDisplay.zoomFromOverview(); | ||||
|  | ||||
|         // Make other elements fade out. | ||||
|         Tweener.addTween(this._overview, | ||||
|         Tweener.addTween(this._group, | ||||
|                          { opacity: 0, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            time: ANIMATION_TIME, | ||||
|                            onComplete: this._hideDone, | ||||
|                            onCompleteScope: this | ||||
|                          }); | ||||
|         this._unshadeBackgrounds(); | ||||
|  | ||||
|         Tweener.addTween(this._background, | ||||
|                          { dim_factor: 1.0, | ||||
|                            time: ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad' | ||||
|                          }); | ||||
|  | ||||
|         this._coverPane.raise_top(); | ||||
|         this._coverPane.show(); | ||||
| @@ -675,15 +780,16 @@ const Overview = new Lang.Class({ | ||||
|         Meta.enable_unredirect_for_screen(global.screen); | ||||
|  | ||||
|         global.window_group.show(); | ||||
|         global.top_window_group.show(); | ||||
|  | ||||
|         this._viewSelector.hide(); | ||||
|         this._workspacesDisplay.hide(); | ||||
|  | ||||
|         this._desktopFade.hide(); | ||||
|         this._backgroundGroup.hide(); | ||||
|         this._overview.hide(); | ||||
|         this._background.hide(); | ||||
|         this._group.hide(); | ||||
|  | ||||
|         this.visible = false; | ||||
|         this.animationInProgress = false; | ||||
|         this._hideInProgress = false; | ||||
|  | ||||
|         this._coverPane.hide(); | ||||
|  | ||||
|   | ||||
| @@ -1,375 +0,0 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const ViewSelector = imports.ui.viewSelector; | ||||
|  | ||||
| const SIDE_CONTROLS_ANIMATION_TIME = 0.16; | ||||
|  | ||||
| function getRtlSlideDirection(direction, actor) { | ||||
|     let rtl = (actor.text_direction == Clutter.TextDirection.RTL); | ||||
|     if (rtl) | ||||
|         direction = (direction == SlideDirection.LEFT) ? | ||||
|             SlideDirection.RIGHT : SlideDirection.LEFT; | ||||
|  | ||||
|     return direction; | ||||
| }; | ||||
|  | ||||
| const SlideDirection = { | ||||
|     LEFT: 0, | ||||
|     RIGHT: 1 | ||||
| }; | ||||
|  | ||||
| const SlideLayout = new Lang.Class({ | ||||
|     Name: 'SlideLayout', | ||||
|     Extends: Clutter.FixedLayout, | ||||
|  | ||||
|     _init: function(params) { | ||||
|         this._slideX = 1; | ||||
|         this._direction = SlideDirection.LEFT; | ||||
|  | ||||
|         this.parent(params); | ||||
|     }, | ||||
|  | ||||
|     vfunc_get_preferred_width: function(container, forHeight) { | ||||
|         let child = container.get_first_child(); | ||||
|  | ||||
|         let [minWidth, natWidth] = child.get_preferred_width(forHeight); | ||||
|  | ||||
|         minWidth *= this._slideX; | ||||
|         natWidth *= this._slideX; | ||||
|  | ||||
|         return [minWidth, natWidth]; | ||||
|     }, | ||||
|  | ||||
|     vfunc_allocate: function(container, box, flags) { | ||||
|         let child = container.get_first_child(); | ||||
|  | ||||
|         let [, , natWidth, natHeight] = child.get_preferred_size(); | ||||
|         let availWidth = Math.round(box.x2 - box.x1); | ||||
|         let availHeight = Math.round(box.y2 - box.y1); | ||||
|  | ||||
|         let realDirection = getRtlSlideDirection(this._direction, child); | ||||
|         let translationX = (realDirection == SlideDirection.LEFT) ? | ||||
|             (availWidth - natWidth) : (natWidth - availWidth); | ||||
|  | ||||
|         let actorBox = new Clutter.ActorBox({ x1: translationX, | ||||
|                                               y1: 0, | ||||
|                                               x2: child.x_expand ? availWidth : natWidth, | ||||
|                                               y2: child.y_expand ? availHeight : natHeight }); | ||||
|  | ||||
|         child.allocate(actorBox, flags); | ||||
|     }, | ||||
|  | ||||
|     set slideX(value) { | ||||
|         this._slideX = value; | ||||
|         this.layout_changed(); | ||||
|     }, | ||||
|  | ||||
|     get slideX() { | ||||
|         return this._slideX; | ||||
|     }, | ||||
|  | ||||
|     set slideDirection(direction) { | ||||
|         this._direction = direction; | ||||
|         this.layout_changed(); | ||||
|     }, | ||||
|  | ||||
|     get slideDirection() { | ||||
|         return this._direction; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const SlidingControl = new Lang.Class({ | ||||
|     Name: 'SlidingControl', | ||||
|  | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { slideDirection: SlideDirection.LEFT }); | ||||
|  | ||||
|         this.visible = true; | ||||
|         this.inDrag = false; | ||||
|  | ||||
|         this.layout = new SlideLayout(); | ||||
|         this.layout.slideDirection = params.slideDirection; | ||||
|         this.actor = new St.Widget({ layout_manager: this.layout, | ||||
|                                      clip_to_allocation: true }); | ||||
|  | ||||
|         Main.overview.connect('showing', Lang.bind(this, this._onOverviewShowing)); | ||||
|  | ||||
|         Main.overview.connect('item-drag-begin', Lang.bind(this, this._onDragBegin)); | ||||
|         Main.overview.connect('item-drag-end', Lang.bind(this, this._onDragEnd)); | ||||
|         Main.overview.connect('item-drag-cancelled', Lang.bind(this, this._onDragEnd)); | ||||
|  | ||||
|         Main.overview.connect('window-drag-begin', Lang.bind(this, this._onWindowDragBegin)); | ||||
|         Main.overview.connect('window-drag-cancelled', Lang.bind(this, this._onWindowDragEnd)); | ||||
|         Main.overview.connect('window-drag-end', Lang.bind(this, this._onWindowDragEnd)); | ||||
|     }, | ||||
|  | ||||
|     getSlide: function() { | ||||
|         throw new Error('getSlide() must be overridden'); | ||||
|     }, | ||||
|  | ||||
|     updateSlide: function() { | ||||
|         Tweener.addTween(this.layout, { slideX: this.getSlide(), | ||||
|                                         time: SIDE_CONTROLS_ANIMATION_TIME, | ||||
|                                         transition: 'easeOutQuad' }); | ||||
|     }, | ||||
|  | ||||
|     getVisibleWidth: function() { | ||||
|         let child = this.actor.get_first_child(); | ||||
|         let [, , natWidth, ] = child.get_preferred_size(); | ||||
|         return natWidth; | ||||
|     }, | ||||
|  | ||||
|     _getTranslation: function() { | ||||
|         let child = this.actor.get_first_child(); | ||||
|         let direction = getRtlSlideDirection(this.layout.slideDirection, child); | ||||
|         let visibleWidth = this.getVisibleWidth(); | ||||
|  | ||||
|         if (direction == SlideDirection.LEFT) | ||||
|             return - visibleWidth; | ||||
|         else | ||||
|             return visibleWidth; | ||||
|     }, | ||||
|  | ||||
|     _updateTranslation: function() { | ||||
|         let translationStart = 0; | ||||
|         let translationEnd = 0; | ||||
|         let translation = this._getTranslation(); | ||||
|  | ||||
|         if (this.visible) { | ||||
|             translationStart = translation; | ||||
|         } else { | ||||
|             translationEnd = translation; | ||||
|         } | ||||
|  | ||||
|         if (this.actor.translation_x == translationEnd) | ||||
|             return; | ||||
|  | ||||
|         this.actor.translation_x = translationStart; | ||||
|         Tweener.addTween(this.actor, { translation_x: translationEnd, | ||||
|                                        time: SIDE_CONTROLS_ANIMATION_TIME, | ||||
|                                        transition: 'easeOutQuad' | ||||
|                                      }); | ||||
|     }, | ||||
|  | ||||
|     _onOverviewShowing: function() { | ||||
|         // reset any translation and make sure the actor is visible when | ||||
|         // entering the overview | ||||
|         this.visible = true; | ||||
|         this.layout.slideX = this.getSlide(); | ||||
|         this.actor.translation_x = 0; | ||||
|     }, | ||||
|  | ||||
|     _onWindowDragBegin: function() { | ||||
|         this._onDragBegin(); | ||||
|     }, | ||||
|  | ||||
|     _onWindowDragEnd: function() { | ||||
|         this._onDragEnd(); | ||||
|     }, | ||||
|  | ||||
|     _onDragBegin: function() { | ||||
|         this.inDrag = true; | ||||
|         this.actor.translation_x = 0; | ||||
|         this.updateSlide(); | ||||
|     }, | ||||
|  | ||||
|     _onDragEnd: function() { | ||||
|         this.inDrag = false; | ||||
|         this.updateSlide(); | ||||
|     }, | ||||
|  | ||||
|     fadeIn: function() { | ||||
|         Tweener.addTween(this.actor, { opacity: 255, | ||||
|                                        time: SIDE_CONTROLS_ANIMATION_TIME / 2, | ||||
|                                        transition: 'easeInQuad' | ||||
|                                      }); | ||||
|     }, | ||||
|  | ||||
|     fadeHalf: function() { | ||||
|         Tweener.addTween(this.actor, { opacity: 128, | ||||
|                                        time: SIDE_CONTROLS_ANIMATION_TIME / 2, | ||||
|                                        transition: 'easeOutQuad' | ||||
|                                      }); | ||||
|     }, | ||||
|  | ||||
|     slideIn: function() { | ||||
|         this.visible = true; | ||||
|         // we will update slideX and the translation from pageEmpty | ||||
|     }, | ||||
|  | ||||
|     slideOut: function() { | ||||
|         this.visible = false; | ||||
|         this._updateTranslation(); | ||||
|         // we will update slideX from pageEmpty | ||||
|     }, | ||||
|  | ||||
|     pageEmpty: function() { | ||||
|         // When pageEmpty is received, there's no visible view in the | ||||
|         // selector; this means we can now safely set the full slide for | ||||
|         // the next page, since slideIn or slideOut might have been called, | ||||
|         // changing the visiblity | ||||
|         this.layout.slideX = this.getSlide(); | ||||
|         this._updateTranslation(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const ThumbnailsSlider = new Lang.Class({ | ||||
|     Name: 'ThumbnailsSlider', | ||||
|     Extends: SlidingControl, | ||||
|  | ||||
|     _init: function(thumbnailsBox) { | ||||
|         this.parent({ slideDirection: SlideDirection.RIGHT }); | ||||
|  | ||||
|         this._thumbnailsBox = thumbnailsBox; | ||||
|  | ||||
|         // SlideLayout reads the actor's expand flags to decide | ||||
|         // whether to allocate the natural size to its child, or the whole | ||||
|         // available allocation | ||||
|         this._thumbnailsBox.actor.y_expand = true; | ||||
|  | ||||
|         this.actor.request_mode = Clutter.RequestMode.WIDTH_FOR_HEIGHT; | ||||
|         this.actor.reactive = true; | ||||
|         this.actor.track_hover = true; | ||||
|         this.actor.add_actor(this._thumbnailsBox.actor); | ||||
|  | ||||
|         Main.layoutManager.connect('monitors-changed', Lang.bind(this, this.updateSlide)); | ||||
|         this.actor.connect('notify::hover', Lang.bind(this, this.updateSlide)); | ||||
|     }, | ||||
|  | ||||
|     _getAlwaysZoomOut: function() { | ||||
|         // Always show the pager when hover, during a drag, or if workspaces are | ||||
|         // actually used, e.g. there are windows on more than one | ||||
|         let alwaysZoomOut = this.actor.hover || this.inDrag || global.screen.n_workspaces > 2; | ||||
|  | ||||
|         if (!alwaysZoomOut) { | ||||
|             let monitors = Main.layoutManager.monitors; | ||||
|             let primary = Main.layoutManager.primaryMonitor; | ||||
|  | ||||
|             /* Look for any monitor to the right of the primary, if there is | ||||
|              * one, we always keep zoom out, otherwise its hard to reach | ||||
|              * the thumbnail area without passing into the next monitor. */ | ||||
|             for (let i = 0; i < monitors.length; i++) { | ||||
|                 if (monitors[i].x >= primary.x + primary.width) { | ||||
|                     alwaysZoomOut = true; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return alwaysZoomOut; | ||||
|     }, | ||||
|  | ||||
|     getSlide: function() { | ||||
|         if (!this.visible) | ||||
|             return 0; | ||||
|  | ||||
|         let alwaysZoomOut = this._getAlwaysZoomOut(); | ||||
|         if (alwaysZoomOut) | ||||
|             return 1; | ||||
|  | ||||
|         let child = this.actor.get_first_child(); | ||||
|         let preferredHeight = child.get_preferred_height(-1)[1]; | ||||
|         let expandedWidth = child.get_preferred_width(preferredHeight)[1]; | ||||
|         let visibleWidth = child.get_theme_node().get_length('visible-width'); | ||||
|  | ||||
|         return visibleWidth / expandedWidth; | ||||
|     }, | ||||
|  | ||||
|     getVisibleWidth: function() { | ||||
|         let alwaysZoomOut = this._getAlwaysZoomOut(); | ||||
|         if (alwaysZoomOut) | ||||
|             return this.parent(); | ||||
|  | ||||
|         let child = this.actor.get_first_child(); | ||||
|         return child.get_theme_node().get_length('visible-width'); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const DashSlider = new Lang.Class({ | ||||
|     Name: 'DashSlider', | ||||
|     Extends: SlidingControl, | ||||
|  | ||||
|     _init: function(dash) { | ||||
|         this.parent({ slideDirection: SlideDirection.LEFT }); | ||||
|  | ||||
|         this._dash = dash; | ||||
|  | ||||
|         // SlideLayout reads the actor's expand flags to decide | ||||
|         // whether to allocate the natural size to its child, or the whole | ||||
|         // available allocation | ||||
|         this._dash.actor.x_expand = true; | ||||
|         this._dash.actor.y_expand = true; | ||||
|         this.actor.add_actor(this._dash.actor); | ||||
|  | ||||
|         this._dash.connect('icon-size-changed', Lang.bind(this, this.updateSlide)); | ||||
|     }, | ||||
|  | ||||
|     getSlide: function() { | ||||
|         if (this.visible || this.inDrag) | ||||
|             return 1; | ||||
|         else | ||||
|             return 0; | ||||
|     }, | ||||
|  | ||||
|     _onWindowDragBegin: function() { | ||||
|         this.fadeHalf(); | ||||
|     }, | ||||
|  | ||||
|     _onWindowDragEnd: function() { | ||||
|         this.fadeIn(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const ControlsManager = new Lang.Class({ | ||||
|     Name: 'ControlsManager', | ||||
|  | ||||
|     _init: function(dash, thumbnails, viewSelector) { | ||||
|         this._dashSlider = new DashSlider(dash); | ||||
|         this.dashActor = this._dashSlider.actor; | ||||
|  | ||||
|         this._thumbnailsSlider = new ThumbnailsSlider(thumbnails); | ||||
|         this.thumbnailsActor = this._thumbnailsSlider.actor; | ||||
|  | ||||
|         this._viewSelector = viewSelector; | ||||
|         this._viewSelector.connect('page-changed', Lang.bind(this, this._setVisibility)); | ||||
|         this._viewSelector.connect('page-empty', Lang.bind(this, this._onPageEmpty)); | ||||
|     }, | ||||
|  | ||||
|     _setVisibility: function() { | ||||
|         // Ignore the case when we're leaving the overview, since | ||||
|         // actors will be made visible again when entering the overview | ||||
|         // next time, and animating them while doing so is just | ||||
|         // unnecessary noise | ||||
|         if (!Main.overview.visible || | ||||
|             (Main.overview.animationInProgress && !Main.overview.visibleTarget)) | ||||
|             return; | ||||
|  | ||||
|         let activePage = this._viewSelector.getActivePage(); | ||||
|         let dashVisible = (activePage == ViewSelector.ViewPage.WINDOWS || | ||||
|                            activePage == ViewSelector.ViewPage.APPS); | ||||
|         let thumbnailsVisible = (activePage == ViewSelector.ViewPage.WINDOWS); | ||||
|  | ||||
|         if (dashVisible) | ||||
|             this._dashSlider.slideIn(); | ||||
|         else | ||||
|             this._dashSlider.slideOut(); | ||||
|  | ||||
|         if (thumbnailsVisible) | ||||
|             this._thumbnailsSlider.slideIn(); | ||||
|         else | ||||
|             this._thumbnailsSlider.slideOut(); | ||||
|     }, | ||||
|  | ||||
|     _onPageEmpty: function() { | ||||
|         this._dashSlider.pageEmpty(); | ||||
|         this._thumbnailsSlider.pageEmpty(); | ||||
|     } | ||||
| }); | ||||
							
								
								
									
										510
									
								
								js/ui/panel.js
									
									
									
									
									
								
							
							
						
						| @@ -4,7 +4,6 @@ const Cairo = imports.cairo; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Meta = imports.gi.Meta; | ||||
| @@ -14,7 +13,6 @@ const St = imports.gi.St; | ||||
| const Signals = imports.signals; | ||||
| const Atk = imports.gi.Atk; | ||||
|  | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const CtrlAltTab = imports.ui.ctrlAltTab; | ||||
| const DND = imports.ui.dnd; | ||||
| @@ -22,6 +20,7 @@ const Layout = imports.ui.layout; | ||||
| const Overview = imports.ui.overview; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
| const PanelMenu = imports.ui.panelMenu; | ||||
| const DateMenu = imports.ui.dateMenu; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| @@ -32,6 +31,33 @@ const BUTTON_DND_ACTIVATION_TIMEOUT = 250; | ||||
| const ANIMATED_ICON_UPDATE_TIMEOUT = 100; | ||||
| const SPINNER_ANIMATION_TIME = 0.2; | ||||
|  | ||||
| const STANDARD_STATUS_AREA_ORDER = ['a11y', 'keyboard', 'volume', 'bluetooth', 'network', 'battery', 'userMenu']; | ||||
| const STANDARD_STATUS_AREA_SHELL_IMPLEMENTATION = { | ||||
|     'a11y': imports.ui.status.accessibility.ATIndicator, | ||||
|     'volume': imports.ui.status.volume.Indicator, | ||||
|     'battery': imports.ui.status.power.Indicator, | ||||
|     'keyboard': imports.ui.status.keyboard.XKBIndicator, | ||||
|     'userMenu': imports.ui.userMenu.UserMenuButton | ||||
| }; | ||||
|  | ||||
| if (Config.HAVE_BLUETOOTH) | ||||
|     STANDARD_STATUS_AREA_SHELL_IMPLEMENTATION['bluetooth'] = imports.ui.status.bluetooth.Indicator; | ||||
|  | ||||
| try { | ||||
|     STANDARD_STATUS_AREA_SHELL_IMPLEMENTATION['network'] = imports.ui.status.network.NMApplet; | ||||
| } catch(e) { | ||||
|     log('NMApplet is not supported. It is possible that your NetworkManager version is too old'); | ||||
| } | ||||
|  | ||||
| const GDM_STATUS_AREA_ORDER = ['a11y', 'display', 'keyboard', 'volume', 'battery', 'powerMenu']; | ||||
| const GDM_STATUS_AREA_SHELL_IMPLEMENTATION = { | ||||
|     'a11y': imports.ui.status.accessibility.ATIndicator, | ||||
|     'volume': imports.ui.status.volume.Indicator, | ||||
|     'battery': imports.ui.status.power.Indicator, | ||||
|     'keyboard': imports.ui.status.keyboard.XKBIndicator, | ||||
|     'powerMenu': imports.gdm.powerMenu.PowerMenuButton | ||||
| }; | ||||
|  | ||||
| // To make sure the panel corners blend nicely with the panel, | ||||
| // we draw background and borders the same way, e.g. drawing | ||||
| // them as filled shapes from the outside inwards instead of | ||||
| @@ -75,41 +101,33 @@ function _unpremultiply(color) { | ||||
|                                blue: blue, alpha: color.alpha }); | ||||
| }; | ||||
|  | ||||
| const Animation = new Lang.Class({ | ||||
|     Name: 'Animation', | ||||
|  | ||||
|     _init: function(filename, width, height, speed) { | ||||
|         this.actor = new St.Bin(); | ||||
| const AnimatedIcon = new Lang.Class({ | ||||
|     Name: 'AnimatedIcon', | ||||
|  | ||||
|     _init: function(name, size) { | ||||
|         this.actor = new St.Bin({ visible: false }); | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|         this._speed = speed; | ||||
|         this.actor.connect('notify::visible', Lang.bind(this, this._onVisibleNotify)); | ||||
|  | ||||
|         this._isLoaded = false; | ||||
|         this._isPlaying = false; | ||||
|         this._timeoutId = 0; | ||||
|         this._frame = 0; | ||||
|         this._animations = St.TextureCache.get_default().load_sliced_image (filename, width, height, | ||||
|                                                                             Lang.bind(this, this._animationsLoaded)); | ||||
|         this._animations = St.TextureCache.get_default().load_sliced_image (global.datadir + '/theme/' + name, size, size); | ||||
|         this.actor.set_child(this._animations); | ||||
|     }, | ||||
|  | ||||
|     play: function() { | ||||
|         if (this._isLoaded && this._timeoutId == 0) { | ||||
|             if (this._frame == 0) | ||||
|                 this._showFrame(0); | ||||
|  | ||||
|             this._timeoutId = Mainloop.timeout_add(this._speed, Lang.bind(this, this._update)); | ||||
|         } | ||||
|  | ||||
|         this._isPlaying = true; | ||||
|     }, | ||||
|  | ||||
|     stop: function() { | ||||
|     _disconnectTimeout: function() { | ||||
|         if (this._timeoutId > 0) { | ||||
|             Mainloop.source_remove(this._timeoutId); | ||||
|             this._timeoutId = 0; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|         this._isPlaying = false; | ||||
|     _onVisibleNotify: function() { | ||||
|         if (this.actor.visible) | ||||
|             this._timeoutId = Mainloop.timeout_add(ANIMATED_ICON_UPDATE_TIMEOUT, Lang.bind(this, this._update)); | ||||
|         else | ||||
|             this._disconnectTimeout(); | ||||
|     }, | ||||
|  | ||||
|     _showFrame: function(frame) { | ||||
| @@ -129,24 +147,8 @@ const Animation = new Lang.Class({ | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _animationsLoaded: function() { | ||||
|         this._isLoaded = true; | ||||
|  | ||||
|         if (this._isPlaying) | ||||
|             this.play(); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         this.stop(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const AnimatedIcon = new Lang.Class({ | ||||
|     Name: 'AnimatedIcon', | ||||
|     Extends: Animation, | ||||
|  | ||||
|     _init: function(name, size) { | ||||
|         this.parent(global.datadir + '/theme/' + name, size, size, ANIMATED_ICON_UPDATE_TIMEOUT); | ||||
|         this._disconnectTimeout(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -246,14 +248,14 @@ const AppMenuButton = new Lang.Class({ | ||||
|     Name: 'AppMenuButton', | ||||
|     Extends: PanelMenu.Button, | ||||
|  | ||||
|     _init: function(panel) { | ||||
|     _init: function(menuManager) { | ||||
|         this.parent(0.0, null, true); | ||||
|  | ||||
|         this.actor.accessible_role = Atk.Role.MENU; | ||||
|  | ||||
|         this._startingApps = []; | ||||
|  | ||||
|         this._menuManager = panel.menuManager; | ||||
|         this._menuManager = menuManager; | ||||
|         this._targetApp = null; | ||||
|         this._appMenuNotifyId = 0; | ||||
|         this._actionGroupNotifyId = 0; | ||||
| @@ -271,10 +273,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._container.connect('get-preferred-height', Lang.bind(this, this._getContentPreferredHeight)); | ||||
|         this._container.connect('allocate', Lang.bind(this, this._contentAllocate)); | ||||
|  | ||||
|         let textureCache = St.TextureCache.get_default(); | ||||
|         textureCache.connect('icon-theme-changed', | ||||
|                              Lang.bind(this, this._onIconThemeChanged)); | ||||
|  | ||||
|         this._iconBox = new Shell.Slicer({ name: 'appMenuIcon' }); | ||||
|         this._iconBox.connect('style-changed', | ||||
|                               Lang.bind(this, this._onIconBoxStyleChanged)); | ||||
| @@ -301,7 +299,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._spinner = new AnimatedIcon('process-working.svg', | ||||
|                                          PANEL_ICON_SIZE); | ||||
|         this._container.add_actor(this._spinner.actor); | ||||
|         this._spinner.actor.hide(); | ||||
|         this._spinner.actor.lower_bottom(); | ||||
|  | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
| @@ -361,15 +358,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._updateIconBoxClip(); | ||||
|     }, | ||||
|  | ||||
|     _onIconThemeChanged: function() { | ||||
|         if (this._iconBox.child == null) | ||||
|             return; | ||||
|  | ||||
|         this._iconBox.child.destroy(); | ||||
|         let icon = this._targetApp.get_faded_icon(2 * PANEL_ICON_SIZE); | ||||
|         this._iconBox.set_child(icon); | ||||
|     }, | ||||
|  | ||||
|     _updateIconBoxClip: function() { | ||||
|         let allocation = this._iconBox.allocation; | ||||
|         if (this._iconBottomClip > 0) | ||||
| @@ -385,14 +373,12 @@ const AppMenuButton = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         this._stop = true; | ||||
|         this.actor.reactive = true; | ||||
|         Tweener.addTween(this._spinner.actor, | ||||
|                          { opacity: 0, | ||||
|                            time: SPINNER_ANIMATION_TIME, | ||||
|                            transition: "easeOutQuad", | ||||
|                            onCompleteScope: this, | ||||
|                            onComplete: function() { | ||||
|                                this._spinner.stop(); | ||||
|                                this._spinner.actor.opacity = 255; | ||||
|                                this._spinner.actor.hide(); | ||||
|                            } | ||||
| @@ -401,8 +387,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|  | ||||
|     startAnimation: function() { | ||||
|         this._stop = false; | ||||
|         this.actor.reactive = false; | ||||
|         this._spinner.play(); | ||||
|         this._spinner.actor.show(); | ||||
|     }, | ||||
|  | ||||
| @@ -598,18 +582,12 @@ const AppMenuButton = new Lang.Class({ | ||||
|                 return; | ||||
|  | ||||
|             menu = new PopupMenu.RemoteMenu(this.actor, this._targetApp.menu, this._targetApp.action_group); | ||||
|             menu.connect('activate', Lang.bind(this, function() { | ||||
|                 let win = this._targetApp.get_windows()[0]; | ||||
|                 win.check_alive(global.get_current_time()); | ||||
|             })); | ||||
|  | ||||
|         } else { | ||||
|             if (this.menu.isDummyQuitMenu) | ||||
|             if (this.menu && !(this.menu instanceof PopupMenu.RemoteMenu)) | ||||
|                 return; | ||||
|  | ||||
|             // fallback to older menu | ||||
|             menu = new PopupMenu.PopupMenu(this.actor, 0.0, St.Side.TOP, 0); | ||||
|             menu.isDummyQuitMenu = true; | ||||
|             menu.addAction(_("Quit"), Lang.bind(this, function() { | ||||
|                 this._targetApp.request_quit(); | ||||
|             })); | ||||
| @@ -622,12 +600,15 @@ const AppMenuButton = new Lang.Class({ | ||||
|  | ||||
| Signals.addSignalMethods(AppMenuButton.prototype); | ||||
|  | ||||
| // Activities button. Because everything else in the top bar is a | ||||
| // PanelMenu.Button, it simplifies some things to make this be one too. | ||||
| // We just hack it up to not actually have a menu attached to it. | ||||
| const ActivitiesButton = new Lang.Class({ | ||||
|     Name: 'ActivitiesButton', | ||||
|     Extends: PanelMenu.Button, | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent(0.0, null, true); | ||||
|         this.parent(0.0); | ||||
|         this.actor.accessible_role = Atk.Role.TOGGLE_BUTTON; | ||||
|  | ||||
|         let container = new Shell.GenericContainer(); | ||||
| @@ -644,8 +625,13 @@ const ActivitiesButton = new Lang.Class({ | ||||
|  | ||||
|         this.actor.label_actor = this._label; | ||||
|  | ||||
|         this.hotCorner = new Layout.HotCorner(Main.layoutManager); | ||||
|         container.add_actor(this.hotCorner.actor); | ||||
|         this._hotCorner = new Layout.HotCorner(); | ||||
|         container.add_actor(this._hotCorner.actor); | ||||
|  | ||||
|         // Hack up our menu... | ||||
|         this.menu.open = Lang.bind(this, this._onMenuOpenRequest); | ||||
|         this.menu.close = Lang.bind(this, this._onMenuCloseRequest); | ||||
|         this.menu.toggle = Lang.bind(this, this._onMenuToggleRequest); | ||||
|  | ||||
|         this.actor.connect('captured-event', Lang.bind(this, this._onCapturedEvent)); | ||||
|         this.actor.connect_after('button-release-event', Lang.bind(this, this._onButtonRelease)); | ||||
| @@ -653,21 +639,16 @@ const ActivitiesButton = new Lang.Class({ | ||||
|  | ||||
|         Main.overview.connect('showing', Lang.bind(this, function() { | ||||
|             this.actor.add_style_pseudo_class('overview'); | ||||
|             this._escapeMenuGrab(); | ||||
|             this.actor.add_accessible_state (Atk.StateType.CHECKED); | ||||
|         })); | ||||
|         Main.overview.connect('hiding', Lang.bind(this, function() { | ||||
|             this.actor.remove_style_pseudo_class('overview'); | ||||
|             this._escapeMenuGrab(); | ||||
|             this.actor.remove_accessible_state (Atk.StateType.CHECKED); | ||||
|         })); | ||||
|  | ||||
|         this._xdndTimeOut = 0; | ||||
|  | ||||
|         // Since the hot corner uses stage coordinates, Clutter won't | ||||
|         // queue relayouts for us when the panel moves. Queue a relayout | ||||
|         // when that happens. | ||||
|         Main.layoutManager.connect('panel-box-changed', Lang.bind(this, function() { | ||||
|             container.queue_relayout(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _containerGetPreferredWidth: function(actor, forHeight, alloc) { | ||||
| @@ -695,10 +676,10 @@ const ActivitiesButton = new Lang.Class({ | ||||
|         } | ||||
|  | ||||
|         hotBox.x1 = Math.round(x); | ||||
|         hotBox.x2 = hotBox.x1 + this.hotCorner.actor.width; | ||||
|         hotBox.x2 = hotBox.x1 + this._hotCorner.actor.width; | ||||
|         hotBox.y1 = Math.round(y); | ||||
|         hotBox.y2 = hotBox.y1 + this.hotCorner.actor.height; | ||||
|         this.hotCorner.actor.allocate(hotBox, flags); | ||||
|         hotBox.y2 = hotBox.y1 + this._hotCorner.actor.height; | ||||
|         this._hotCorner.actor.allocate(hotBox, flags); | ||||
|     }, | ||||
|  | ||||
|     handleDragOver: function(source, actor, x, y, time) { | ||||
| @@ -713,21 +694,46 @@ const ActivitiesButton = new Lang.Class({ | ||||
|         return DND.DragMotionResult.CONTINUE; | ||||
|     }, | ||||
|  | ||||
|     _escapeMenuGrab: function() { | ||||
|         if (this.menu.isOpen) | ||||
|             this.menu.close(); | ||||
|     }, | ||||
|  | ||||
|     _onCapturedEvent: function(actor, event) { | ||||
|         if (event.type() == Clutter.EventType.BUTTON_PRESS) { | ||||
|             if (!this.hotCorner.shouldToggleOverviewOnClick()) | ||||
|             if (!this._hotCorner.shouldToggleOverviewOnClick()) | ||||
|                 return true; | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onMenuOpenRequest: function() { | ||||
|         this.menu.isOpen = true; | ||||
|         this.menu.emit('open-state-changed', true); | ||||
|     }, | ||||
|  | ||||
|     _onMenuCloseRequest: function() { | ||||
|         this.menu.isOpen = false; | ||||
|         this.menu.emit('open-state-changed', false); | ||||
|     }, | ||||
|  | ||||
|     _onMenuToggleRequest: function() { | ||||
|         this.menu.isOpen = !this.menu.isOpen; | ||||
|         this.menu.emit('open-state-changed', this.menu.isOpen); | ||||
|     }, | ||||
|  | ||||
|     _onButtonRelease: function() { | ||||
|         Main.overview.toggle(); | ||||
|         if (this.menu.isOpen) { | ||||
|             this.menu.close(); | ||||
|             Main.overview.toggle(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onKeyRelease: function(actor, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_Return || symbol == Clutter.KEY_space) { | ||||
|             if (this.menu.isOpen) | ||||
|                 this.menu.close(); | ||||
|             Main.overview.toggle(); | ||||
|         } | ||||
|     }, | ||||
| @@ -739,6 +745,7 @@ const ActivitiesButton = new Lang.Class({ | ||||
|         if (pickedActor == this.actor) { | ||||
|             if (!Main.overview.visible && !Main.overview.animationInProgress) { | ||||
|                 Main.overview.showTemporarily(); | ||||
|                 Main.overview.beginItemDrag(actor); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -750,9 +757,12 @@ const ActivitiesButton = new Lang.Class({ | ||||
| const PanelCorner = new Lang.Class({ | ||||
|     Name: 'PanelCorner', | ||||
|  | ||||
|     _init: function(side) { | ||||
|     _init: function(box, side) { | ||||
|         this._side = side; | ||||
|  | ||||
|         this._box = box; | ||||
|         this._box.connect('style-changed', Lang.bind(this, this._boxStyleChanged)); | ||||
|  | ||||
|         this.actor = new St.DrawingArea({ style_class: 'panel-corner' }); | ||||
|         this.actor.connect('style-changed', Lang.bind(this, this._styleChanged)); | ||||
|         this.actor.connect('repaint', Lang.bind(this, this._repaint)); | ||||
| @@ -768,11 +778,10 @@ const PanelCorner = new Lang.Class({ | ||||
|             return null; | ||||
|  | ||||
|         // Start at the back and work backward | ||||
|         let index; | ||||
|         for (index = children.length - 1; index >= 0; index--) { | ||||
|             if (children[index].visible) | ||||
|                 break; | ||||
|         } | ||||
|         let index = children.length - 1; | ||||
|         while (!children[index].visible && index >= 0) | ||||
|             index--; | ||||
|  | ||||
|         if (index < 0) | ||||
|             return null; | ||||
|  | ||||
| @@ -793,11 +802,10 @@ const PanelCorner = new Lang.Class({ | ||||
|             return null; | ||||
|  | ||||
|         // Start at the front and work forward | ||||
|         let index; | ||||
|         for (index = 0; index < children.length; index++) { | ||||
|             if (children[index].visible) | ||||
|                 break; | ||||
|         } | ||||
|         let index = 0; | ||||
|         while (!children[index].visible && index < children.length) | ||||
|             index++; | ||||
|  | ||||
|         if (index == children.length) | ||||
|             return null; | ||||
|  | ||||
| @@ -808,12 +816,12 @@ const PanelCorner = new Lang.Class({ | ||||
|         return children[index]; | ||||
|     }, | ||||
|  | ||||
|     setStyleParent: function(box) { | ||||
|     _boxStyleChanged: function() { | ||||
|         let side = this._side; | ||||
|  | ||||
|         let rtlAwareContainer = box instanceof St.BoxLayout; | ||||
|         let rtlAwareContainer = this._box instanceof St.BoxLayout; | ||||
|         if (rtlAwareContainer && | ||||
|             box.get_text_direction() == Clutter.TextDirection.RTL) { | ||||
|             this._box.get_text_direction() == Clutter.TextDirection.RTL) { | ||||
|             if (this._side == St.Side.LEFT) | ||||
|                 side = St.Side.RIGHT; | ||||
|             else if (this._side == St.Side.RIGHT) | ||||
| @@ -822,9 +830,9 @@ const PanelCorner = new Lang.Class({ | ||||
|  | ||||
|         let button; | ||||
|         if (side == St.Side.LEFT) | ||||
|             button = this._findLeftmostButton(box); | ||||
|             button = this._findLeftmostButton(this._box); | ||||
|         else if (side == St.Side.RIGHT) | ||||
|             button = this._findRightmostButton(box); | ||||
|             button = this._findRightmostButton(this._box); | ||||
|  | ||||
|         if (button) { | ||||
|             if (this._button && this._buttonStyleChangedSignalId) { | ||||
| @@ -851,7 +859,7 @@ const PanelCorner = new Lang.Class({ | ||||
|  | ||||
|             // The corner doesn't support theme transitions, so override | ||||
|             // the .panel-button default | ||||
|             button.style = 'transition-duration: 0ms'; | ||||
|             button.style = 'transition-duration: 0'; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
| @@ -864,13 +872,10 @@ const PanelCorner = new Lang.Class({ | ||||
|         let backgroundColor = node.get_color('-panel-corner-background-color'); | ||||
|         let borderColor = node.get_color('-panel-corner-border-color'); | ||||
|  | ||||
|         let overlap = borderColor.alpha != 0; | ||||
|         let offsetY = overlap ? 0 : borderWidth; | ||||
|  | ||||
|         let cr = this.actor.get_context(); | ||||
|         cr.setOperator(Cairo.Operator.SOURCE); | ||||
|  | ||||
|         cr.moveTo(0, offsetY); | ||||
|         cr.moveTo(0, 0); | ||||
|         if (this._side == St.Side.LEFT) | ||||
|             cr.arc(cornerRadius, | ||||
|                    borderWidth + cornerRadius, | ||||
| @@ -879,7 +884,7 @@ const PanelCorner = new Lang.Class({ | ||||
|             cr.arc(0, | ||||
|                    borderWidth + cornerRadius, | ||||
|                    cornerRadius, 3 * Math.PI / 2, 2 * Math.PI); | ||||
|         cr.lineTo(cornerRadius, offsetY); | ||||
|         cr.lineTo(cornerRadius, 0); | ||||
|         cr.closePath(); | ||||
|  | ||||
|         let savedPath = cr.copyPath(); | ||||
| @@ -889,18 +894,14 @@ const PanelCorner = new Lang.Class({ | ||||
|         Clutter.cairo_set_source_color(cr, over); | ||||
|         cr.fill(); | ||||
|  | ||||
|         if (overlap) { | ||||
|             let offset = borderWidth; | ||||
|             Clutter.cairo_set_source_color(cr, backgroundColor); | ||||
|         let offset = borderWidth; | ||||
|         Clutter.cairo_set_source_color(cr, backgroundColor); | ||||
|  | ||||
|             cr.save(); | ||||
|             cr.translate(xOffsetDirection * offset, - offset); | ||||
|             cr.appendPath(savedPath); | ||||
|             cr.fill(); | ||||
|             cr.restore(); | ||||
|         } | ||||
|  | ||||
|         cr.$dispose(); | ||||
|         cr.save(); | ||||
|         cr.translate(xOffsetDirection * offset, - offset); | ||||
|         cr.appendPath(savedPath); | ||||
|         cr.fill(); | ||||
|         cr.restore(); | ||||
|     }, | ||||
|  | ||||
|     _styleChanged: function() { | ||||
| @@ -914,31 +915,6 @@ const PanelCorner = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const PANEL_ITEM_IMPLEMENTATIONS = { | ||||
|     'activities': ActivitiesButton, | ||||
|     'appMenu': AppMenuButton, | ||||
|     'dateMenu': imports.ui.dateMenu.DateMenuButton, | ||||
|     'a11y': imports.ui.status.accessibility.ATIndicator, | ||||
|     'a11yGreeter': imports.ui.status.accessibility.ATGreeterIndicator, | ||||
|     'volume': imports.ui.status.volume.Indicator, | ||||
|     'battery': imports.ui.status.power.Indicator, | ||||
|     'lockScreen': imports.ui.status.lockScreenMenu.Indicator, | ||||
|     'logo': imports.gdm.loginDialog.LogoMenuButton, | ||||
|     'keyboard': imports.ui.status.keyboard.InputSourceIndicator, | ||||
|     'powerMenu': imports.gdm.powerMenu.PowerMenuButton, | ||||
|     'userMenu': imports.ui.userMenu.UserMenuButton | ||||
| }; | ||||
|  | ||||
| if (Config.HAVE_BLUETOOTH) | ||||
|     PANEL_ITEM_IMPLEMENTATIONS['bluetooth'] = | ||||
|         imports.ui.status.bluetooth.Indicator; | ||||
|  | ||||
| try { | ||||
|     PANEL_ITEM_IMPLEMENTATIONS['network'] = | ||||
|         imports.ui.status.network.NMApplet; | ||||
| } catch(e) { | ||||
|     log('NMApplet is not supported. It is possible that your NetworkManager version is too old'); | ||||
| } | ||||
|  | ||||
| const Panel = new Lang.Class({ | ||||
|     Name: 'Panel', | ||||
| @@ -948,11 +924,16 @@ const Panel = new Lang.Class({ | ||||
|                                                   reactive: true }); | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         this._sessionStyle = null; | ||||
|         this._statusArea = {}; | ||||
|  | ||||
|         this.statusArea = {}; | ||||
|         Main.overview.connect('shown', Lang.bind(this, function () { | ||||
|             this.actor.add_style_class_name('in-overview'); | ||||
|         })); | ||||
|         Main.overview.connect('hiding', Lang.bind(this, function () { | ||||
|             this.actor.remove_style_class_name('in-overview'); | ||||
|         })); | ||||
|  | ||||
|         this.menuManager = new PopupMenu.PopupMenuManager(this); | ||||
|         this._menus = new PopupMenu.PopupMenuManager(this); | ||||
|  | ||||
|         this._leftBox = new St.BoxLayout({ name: 'panelLeft' }); | ||||
|         this.actor.add_actor(this._leftBox); | ||||
| @@ -961,10 +942,17 @@ const Panel = new Lang.Class({ | ||||
|         this._rightBox = new St.BoxLayout({ name: 'panelRight' }); | ||||
|         this.actor.add_actor(this._rightBox); | ||||
|  | ||||
|         this._leftCorner = new PanelCorner(St.Side.LEFT); | ||||
|         if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) | ||||
|             this._leftCorner = new PanelCorner(this._rightBox, St.Side.LEFT); | ||||
|         else | ||||
|             this._leftCorner = new PanelCorner(this._leftBox, St.Side.LEFT); | ||||
|  | ||||
|         this.actor.add_actor(this._leftCorner.actor); | ||||
|  | ||||
|         this._rightCorner = new PanelCorner(St.Side.RIGHT); | ||||
|         if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) | ||||
|             this._rightCorner = new PanelCorner(this._leftBox, St.Side.RIGHT); | ||||
|         else | ||||
|             this._rightCorner = new PanelCorner(this._rightBox, St.Side.RIGHT); | ||||
|         this.actor.add_actor(this._rightCorner.actor); | ||||
|  | ||||
|         this.actor.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth)); | ||||
| @@ -972,19 +960,43 @@ const Panel = new Lang.Class({ | ||||
|         this.actor.connect('allocate', Lang.bind(this, this._allocate)); | ||||
|         this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress)); | ||||
|  | ||||
|         Main.overview.connect('showing', Lang.bind(this, function () { | ||||
|             this.actor.add_style_pseudo_class('overview'); | ||||
|         })); | ||||
|         Main.overview.connect('hiding', Lang.bind(this, function () { | ||||
|             this.actor.remove_style_pseudo_class('overview'); | ||||
|         })); | ||||
|         /* Button on the left side of the panel. */ | ||||
|         if (global.session_type == Shell.SessionType.USER) { | ||||
|             this._activitiesButton = new ActivitiesButton(); | ||||
|             this._activities = this._activitiesButton.actor; | ||||
|             this._leftBox.add(this._activities); | ||||
|  | ||||
|             // The activities button has a pretend menu, so as to integrate | ||||
|             // more cleanly with the rest of the panel | ||||
|             this._menus.addMenu(this._activitiesButton.menu); | ||||
|  | ||||
|             this._appMenu = new AppMenuButton(this._menus); | ||||
|             this._leftBox.add(this._appMenu.actor); | ||||
|         } | ||||
|  | ||||
|         /* center */ | ||||
|         if (global.session_type == Shell.SessionType.USER) | ||||
|             this._dateMenu = new DateMenu.DateMenuButton({ showEvents: true }); | ||||
|         else | ||||
|             this._dateMenu = new DateMenu.DateMenuButton({ showEvents: false }); | ||||
|         this._centerBox.add(this._dateMenu.actor, { y_fill: true }); | ||||
|         this._menus.addMenu(this._dateMenu.menu); | ||||
|  | ||||
|         /* right */ | ||||
|         if (global.session_type == Shell.SessionType.GDM) { | ||||
|             this._status_area_order = GDM_STATUS_AREA_ORDER; | ||||
|             this._status_area_shell_implementation = GDM_STATUS_AREA_SHELL_IMPLEMENTATION; | ||||
|         } else { | ||||
|             this._status_area_order = STANDARD_STATUS_AREA_ORDER; | ||||
|             this._status_area_shell_implementation = STANDARD_STATUS_AREA_SHELL_IMPLEMENTATION; | ||||
|         } | ||||
|  | ||||
|         Main.statusIconDispatcher.connect('status-icon-added', Lang.bind(this, this._onTrayIconAdded)); | ||||
|         Main.statusIconDispatcher.connect('status-icon-removed', Lang.bind(this, this._onTrayIconRemoved)); | ||||
|  | ||||
|         Main.layoutManager.panelBox.add(this.actor); | ||||
|         Main.ctrlAltTabManager.addGroup(this.actor, _("Top Bar"), 'emblem-system-symbolic', | ||||
|         Main.ctrlAltTabManager.addGroup(this.actor, _("Top Bar"), 'start-here', | ||||
|                                         { sortGroup: CtrlAltTab.SortGroup.TOP }); | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._updatePanel)); | ||||
|         this._updatePanel(); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
| @@ -1044,19 +1056,16 @@ const Panel = new Lang.Class({ | ||||
|         } | ||||
|         this._rightBox.allocate(childBox, flags); | ||||
|  | ||||
|         let cornerMinWidth, cornerMinHeight; | ||||
|         let cornerWidth, cornerHeight; | ||||
|  | ||||
|         [cornerMinWidth, cornerWidth] = this._leftCorner.actor.get_preferred_width(-1); | ||||
|         [cornerMinHeight, cornerHeight] = this._leftCorner.actor.get_preferred_height(-1); | ||||
|         let [cornerMinWidth, cornerWidth] = this._leftCorner.actor.get_preferred_width(-1); | ||||
|         let [cornerMinHeight, cornerHeight] = this._leftCorner.actor.get_preferred_width(-1); | ||||
|         childBox.x1 = 0; | ||||
|         childBox.x2 = cornerWidth; | ||||
|         childBox.y1 = allocHeight; | ||||
|         childBox.y2 = allocHeight + cornerHeight; | ||||
|         this._leftCorner.actor.allocate(childBox, flags); | ||||
|  | ||||
|         [cornerMinWidth, cornerWidth] = this._rightCorner.actor.get_preferred_width(-1); | ||||
|         [cornerMinHeight, cornerHeight] = this._rightCorner.actor.get_preferred_height(-1); | ||||
|         let [cornerMinWidth, cornerWidth] = this._rightCorner.actor.get_preferred_width(-1); | ||||
|         let [cornerMinHeight, cornerHeight] = this._rightCorner.actor.get_preferred_width(-1); | ||||
|         childBox.x1 = allocWidth - cornerWidth; | ||||
|         childBox.x2 = allocWidth; | ||||
|         childBox.y1 = allocHeight; | ||||
| @@ -1065,9 +1074,6 @@ const Panel = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress: function(actor, event) { | ||||
|         if (Main.modalCount > 0) | ||||
|             return false; | ||||
|  | ||||
|         if (event.get_source() != actor) | ||||
|             return false; | ||||
|  | ||||
| @@ -1106,142 +1112,76 @@ const Panel = new Lang.Class({ | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     openAppMenu: function() { | ||||
|         let indicator = this.statusArea.appMenu; | ||||
|         if (!indicator) // appMenu not supported by current session mode | ||||
|             return; | ||||
|  | ||||
|         let menu = indicator.menu; | ||||
|         if (!indicator.actor.reactive || menu.isOpen) | ||||
|             return; | ||||
|  | ||||
|         menu.open(); | ||||
|         menu.actor.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false); | ||||
|     }, | ||||
|  | ||||
|     set boxOpacity(value) { | ||||
|         let isReactive = value > 0; | ||||
|  | ||||
|         this._leftBox.opacity = value; | ||||
|         this._leftBox.reactive = isReactive; | ||||
|         this._centerBox.opacity = value; | ||||
|         this._centerBox.reactive = isReactive; | ||||
|         this._rightBox.opacity = value; | ||||
|         this._rightBox.reactive = isReactive; | ||||
|     }, | ||||
|  | ||||
|     get boxOpacity() { | ||||
|         return this._leftBox.opacity; | ||||
|     }, | ||||
|  | ||||
|     _updatePanel: function() { | ||||
|         let panel = Main.sessionMode.panel; | ||||
|         this._hideIndicators(); | ||||
|         this._updateBox(panel.left, this._leftBox); | ||||
|         this._updateBox(panel.center, this._centerBox); | ||||
|         this._updateBox(panel.right, this._rightBox); | ||||
|  | ||||
|         if (this._sessionStyle) | ||||
|             this._removeStyleClassName(this._sessionStyle); | ||||
|  | ||||
|         this._sessionStyle = Main.sessionMode.panelStyle; | ||||
|         if (this._sessionStyle) | ||||
|             this._addStyleClassName(this._sessionStyle); | ||||
|  | ||||
|         if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) { | ||||
|             this._leftCorner.setStyleParent(this._rightBox); | ||||
|             this._rightCorner.setStyleParent(this._leftBox); | ||||
|         } else { | ||||
|             this._leftCorner.setStyleParent(this._leftBox); | ||||
|             this._rightCorner.setStyleParent(this._rightBox); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _hideIndicators: function() { | ||||
|         for (let role in PANEL_ITEM_IMPLEMENTATIONS) { | ||||
|             let indicator = this.statusArea[role]; | ||||
|             if (!indicator) | ||||
|                 continue; | ||||
|             if (indicator.menu) | ||||
|                 indicator.menu.close(); | ||||
|             indicator.container.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _ensureIndicator: function(role) { | ||||
|         let indicator = this.statusArea[role]; | ||||
|         if (!indicator) { | ||||
|             let constructor = PANEL_ITEM_IMPLEMENTATIONS[role]; | ||||
|     startStatusArea: function() { | ||||
|         for (let i = 0; i < this._status_area_order.length; i++) { | ||||
|             let role = this._status_area_order[i]; | ||||
|             let constructor = this._status_area_shell_implementation[role]; | ||||
|             if (!constructor) { | ||||
|                 // This icon is not implemented (this is a bug) | ||||
|                 return null; | ||||
|             } | ||||
|             indicator = new constructor(this); | ||||
|             this.statusArea[role] = indicator; | ||||
|         } | ||||
|         return indicator; | ||||
|     }, | ||||
|  | ||||
|     _updateBox: function(elements, box) { | ||||
|         let nChildren = box.get_n_children(); | ||||
|  | ||||
|         for (let i = 0; i < elements.length; i++) { | ||||
|             let role = elements[i]; | ||||
|             let indicator = this._ensureIndicator(role); | ||||
|             if (indicator == null) | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             this._addToPanelBox(role, indicator, i + nChildren, box); | ||||
|             let indicator = new constructor(); | ||||
|             this.addToStatusArea(role, indicator, i); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _addToPanelBox: function(role, indicator, position, box) { | ||||
|         let container = indicator.container; | ||||
|         container.show(); | ||||
|  | ||||
|         let parent = container.get_parent(); | ||||
|         if (parent) | ||||
|             parent.remove_actor(container); | ||||
|  | ||||
|         box.insert_child_at_index(container, position); | ||||
|         if (indicator.menu) | ||||
|             this.menuManager.addMenu(indicator.menu); | ||||
|         this.statusArea[role] = indicator; | ||||
|         let destroyId = indicator.connect('destroy', Lang.bind(this, function(emitter) { | ||||
|             delete this.statusArea[role]; | ||||
|             emitter.disconnect(destroyId); | ||||
|             container.destroy(); | ||||
|         })); | ||||
|     _insertStatusItem: function(actor, position) { | ||||
|         let children = this._rightBox.get_children(); | ||||
|         let i; | ||||
|         for (i = children.length - 1; i >= 0; i--) { | ||||
|             let rolePosition = children[i]._rolePosition; | ||||
|             if (position > rolePosition) { | ||||
|                 this._rightBox.insert_child_at_index(actor, i + 1); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         if (i == -1) { | ||||
|             // If we didn't find a position, we must be first | ||||
|             this._rightBox.insert_child_at_index(actor, 0); | ||||
|         } | ||||
|         actor._rolePosition = position; | ||||
|     }, | ||||
|  | ||||
|     addToStatusArea: function(role, indicator, position, box) { | ||||
|         if (this.statusArea[role]) | ||||
|     addToStatusArea: function(role, indicator, position) { | ||||
|         if (this._statusArea[role]) | ||||
|             throw new Error('Extension point conflict: there is already a status indicator for role ' + role); | ||||
|  | ||||
|         if (!(indicator instanceof PanelMenu.Button)) | ||||
|             throw new TypeError('Status indicator must be an instance of PanelMenu.Button'); | ||||
|  | ||||
|         position = position || 0; | ||||
|         let boxes = { | ||||
|             left: this._leftBox, | ||||
|             center: this._centerBox, | ||||
|             right: this._rightBox | ||||
|         }; | ||||
|         let boxContainer = boxes[box] || this._rightBox; | ||||
|         this.statusArea[role] = indicator; | ||||
|         this._addToPanelBox(role, indicator, position, boxContainer); | ||||
|         if (!position) | ||||
|             position = 0; | ||||
|         this._insertStatusItem(indicator.actor, position); | ||||
|         this._menus.addMenu(indicator.menu); | ||||
|  | ||||
|         this._statusArea[role] = indicator; | ||||
|         let destroyId = indicator.connect('destroy', Lang.bind(this, function(emitter) { | ||||
|             this._statusArea[role] = null; | ||||
|             emitter.disconnect(destroyId); | ||||
|         })); | ||||
|  | ||||
|         return indicator; | ||||
|     }, | ||||
|  | ||||
|     _addStyleClassName: function(className) { | ||||
|         this.actor.add_style_class_name(className); | ||||
|         this._rightCorner.actor.add_style_class_name(className); | ||||
|         this._leftCorner.actor.add_style_class_name(className); | ||||
|     _onTrayIconAdded: function(o, icon, role) { | ||||
|         if (this._status_area_shell_implementation[role]) { | ||||
|             // This icon is legacy, and replaced by a Shell version | ||||
|             // Hide it | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         icon.height = PANEL_ICON_SIZE; | ||||
|         let buttonBox = new PanelMenu.ButtonBox(); | ||||
|         let box = buttonBox.actor; | ||||
|         box.add_actor(icon); | ||||
|  | ||||
|         this._insertStatusItem(box, this._status_area_order.indexOf(role)); | ||||
|     }, | ||||
|  | ||||
|     _removeStyleClassName: function(className) { | ||||
|         this.actor.remove_style_class_name(className); | ||||
|         this._rightCorner.actor.remove_style_class_name(className); | ||||
|         this._leftCorner.actor.remove_style_class_name(className); | ||||
|     } | ||||
|     _onTrayIconRemoved: function(o, icon) { | ||||
|         let box = icon.get_parent(); | ||||
|         if (box && box._delegate instanceof PanelMenu.ButtonBox) | ||||
|             box.destroy(); | ||||
|     }, | ||||
| }); | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| @@ -21,10 +20,6 @@ const ButtonBox = new Lang.Class({ | ||||
|         this.actor = new Shell.GenericContainer(params); | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         this.container = new St.Bin({ y_fill: true, | ||||
|                                       x_fill: true, | ||||
|                                       child: this.actor }); | ||||
|  | ||||
|         this.actor.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth)); | ||||
|         this.actor.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight)); | ||||
|         this.actor.connect('allocate', Lang.bind(this, this._allocate)); | ||||
| @@ -112,19 +107,13 @@ const Button = new Lang.Class({ | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this._onSourceKeyPress)); | ||||
|  | ||||
|         if (dontCreateMenu) | ||||
|             this.menu = new PopupMenu.PopupDummyMenu(this.actor); | ||||
|             this.menu = null; | ||||
|         else | ||||
|             this.setMenu(new PopupMenu.PopupMenu(this.actor, menuAlignment, St.Side.TOP, 0)); | ||||
|  | ||||
|         this.setName(nameText); | ||||
|     }, | ||||
|  | ||||
|     setSensitive: function(sensitive) { | ||||
|         this.actor.reactive = sensitive; | ||||
|         this.actor.can_focus = sensitive; | ||||
|         this.actor.track_hover = sensitive; | ||||
|     }, | ||||
|  | ||||
|     setName: function(text) { | ||||
|         if (text != null) { | ||||
|             // This is the easiest way to provide a accessible name to | ||||
| @@ -186,7 +175,8 @@ const Button = new Lang.Class({ | ||||
|     _onMenuKeyPress: function(actor, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_Left || symbol == Clutter.KEY_Right) { | ||||
|             let group = global.focus_manager.get_group(this.actor); | ||||
|             let focusManager = St.FocusManager.get_for_stage(global.stage); | ||||
|             let group = focusManager.get_group(this.actor); | ||||
|             if (group) { | ||||
|                 let direction = (symbol == Clutter.KEY_Left) ? Gtk.DirectionType.LEFT : Gtk.DirectionType.RIGHT; | ||||
|                 group.navigate_focus(this.actor, direction, false); | ||||
| @@ -205,15 +195,16 @@ const Button = new Lang.Class({ | ||||
|         // Setting the max-height won't do any good if the minimum height of the | ||||
|         // menu is higher then the screen; it's useful if part of the menu is | ||||
|         // scrollable so the minimum height is smaller than the natural height | ||||
|         let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex); | ||||
|         this.menu.actor.style = ('max-height: ' + Math.round(workArea.height) + 'px;'); | ||||
|         let monitor = Main.layoutManager.primaryMonitor; | ||||
|         this.menu.actor.style = ('max-height: ' + | ||||
|                                  Math.round(monitor.height - Main.panel.actor.height) + | ||||
|                                  'px;'); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         this.actor._delegate = null; | ||||
|  | ||||
|         if (this.menu) | ||||
|             this.menu.destroy(); | ||||
|         this.menu.destroy(); | ||||
|         this.actor.destroy(); | ||||
|  | ||||
|         this.emit('destroy'); | ||||
| @@ -233,39 +224,19 @@ const SystemStatusButton = new Lang.Class({ | ||||
|  | ||||
|     _init: function(iconName, nameText) { | ||||
|         this.parent(0.0, nameText); | ||||
|  | ||||
|         this._iconActor = new St.Icon({ icon_name: iconName, | ||||
|                                         icon_type: St.IconType.SYMBOLIC, | ||||
|                                         style_class: 'system-status-icon' }); | ||||
|         this.actor.add_actor(this._iconActor); | ||||
|         this.actor.add_style_class_name('panel-status-button'); | ||||
|  | ||||
|         this._box = new St.BoxLayout({ style_class: 'panel-status-button-box' }); | ||||
|         this.actor.add_actor(this._box); | ||||
|  | ||||
|         if (iconName) | ||||
|             this.setIcon(iconName); | ||||
|     }, | ||||
|  | ||||
|     get icons() { | ||||
|         return this._box.get_children(); | ||||
|     }, | ||||
|  | ||||
|     addIcon: function(gicon) { | ||||
|         let icon = new St.Icon({ gicon: gicon, | ||||
|                                  style_class: 'system-status-icon' }); | ||||
|         this._box.add_actor(icon); | ||||
|  | ||||
|         this.emit('icons-changed'); | ||||
|  | ||||
|         return icon; | ||||
|     }, | ||||
|  | ||||
|     setIcon: function(iconName) { | ||||
|         if (!this.mainIcon) | ||||
|             this.mainIcon = this.addIcon(null); | ||||
|         this.mainIcon.icon_name = iconName; | ||||
|         this._iconActor.icon_name = iconName; | ||||
|     }, | ||||
|  | ||||
|     setGIcon: function(gicon) { | ||||
|         if (this.mainIcon) | ||||
|             this.mainIcon.gicon = gicon; | ||||
|         else | ||||
|             this.mainIcon = this.addIcon(gicon); | ||||
|         this._iconActor.gicon = gicon; | ||||
|     } | ||||
| }); | ||||
|   | ||||