Compare commits
	
		
			33 Commits
		
	
	
		
			2.91.93
			...
			nbtk-intro
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 4f070317d2 | ||
|   | 661100abe8 | ||
|   | 3e90b11cfb | ||
|   | 4b4a1be420 | ||
|   | 1a4861ec4f | ||
|   | 2ac82da232 | ||
|   | 8f7e6f8117 | ||
|   | d2b98701be | ||
|   | faeda5dc8b | ||
|   | 70cb8e180b | ||
|   | 2232a92ba8 | ||
|   | 4743a8e750 | ||
|   | e1390c7dd5 | ||
|   | 49ba54820c | ||
|   | 8ffa161a7f | ||
|   | 0eca3efcb0 | ||
|   | 9c3af62dc4 | ||
|   | db0c2b5959 | ||
|   | 98215f497d | ||
|   | 9dbafe156e | ||
|   | 277dd7106a | ||
|   | 431f299756 | ||
|   | d8d7f5f711 | ||
|   | a07af9fb14 | ||
|   | f1f11f1e76 | ||
|   | c60a6a49de | ||
|   | 6687054474 | ||
|   | 60819b3a79 | ||
|   | 3e265b4bc6 | ||
|   | e2aa2a00f0 | ||
|   | 1edc88a2bd | ||
|   | 271e4ca07e | ||
|   | 9f79296276 | 
							
								
								
									
										20
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -18,18 +18,12 @@ config | ||||
| configure | ||||
| data/gnome-shell.desktop | ||||
| data/gnome-shell.desktop.in | ||||
| data/gschemas.compiled | ||||
| data/org.gnome.shell.gschema.xml | ||||
| data/org.gnome.shell.gschema.valid | ||||
| js/misc/config.js | ||||
| intltool-extract.in | ||||
| intltool-merge.in | ||||
| intltool-update.in | ||||
| libtool | ||||
| m4/ | ||||
| omf.make | ||||
| po/*.gmo | ||||
| po/gnome-shell.pot | ||||
| po/Makefile.in.in | ||||
| po/POTFILES | ||||
| po/stamp-it | ||||
| @@ -40,23 +34,11 @@ src/*-enum-types.[ch] | ||||
| src/*-marshal.[ch] | ||||
| src/Makefile | ||||
| src/Makefile.in | ||||
| src/calendar-server/org.gnome.Shell.CalendarServer.service | ||||
| src/gnomeshell-taskpanel | ||||
| src/gnome-shell | ||||
| src/gnome-shell-calendar-server | ||||
| src/gnome-shell-extension-tool | ||||
| src/gnome-shell-jhbuild | ||||
| src/gnome-shell-perf-helper | ||||
| src/gnome-shell-real | ||||
| src/run-js-test | ||||
| src/test-recorder | ||||
| src/test-recorder.ogg | ||||
| src/test-theme | ||||
| src/st.h | ||||
| src/stamp-st.h | ||||
| src/stamp-st.h.tmp | ||||
| stamp-h1 | ||||
| tests/run-test.sh | ||||
| xmldocs.make | ||||
| *~ | ||||
| *.patch | ||||
| *.sw? | ||||
|   | ||||
							
								
								
									
										20
									
								
								Makefile.am
									
									
									
									
									
								
							
							
						
						| @@ -1,13 +1,9 @@ | ||||
| # Point to our macro directory and pick up user flags from the environment | ||||
| ACLOCAL_AMFLAGS  = -I m4 ${ACLOCAL_FLAGS} | ||||
|  | ||||
| SUBDIRS = data js src tests po man | ||||
| SUBDIRS = data js src tests po | ||||
|  | ||||
| EXTRA_DIST =		\ | ||||
| 	.project	\ | ||||
| 	.settings	\ | ||||
| 	autogen.sh	\ | ||||
| 	tools/check-for-missing.py | ||||
| 	autogen.sh | ||||
|  | ||||
| # These are files checked into Git that we don't want to distribute | ||||
| DIST_EXCLUDE =					\ | ||||
| @@ -18,4 +14,14 @@ DIST_EXCLUDE =					\ | ||||
|  | ||||
| distcheck-hook: | ||||
| 	@echo "Checking disted files against files in git" | ||||
| 	@$(srcdir)/tools/check-for-missing.py $(srcdir) $(distdir) $(DIST_EXCLUDE) | ||||
| 	@failed=false; \ | ||||
| 	exclude=`(for p in $(DIST_EXCLUDE) ; do echo --exclude=$$p ; done)`; \ | ||||
| 	for f in `cd $(srcdir) && git ls-files $$exclude` ; do \ | ||||
| 		if ! test -e $(distdir)/$$f ; then \ | ||||
| 			echo File missing from distribution: $$f ; \ | ||||
| 			failed=true ; \ | ||||
| 		fi \ | ||||
| 	done ; \ | ||||
| 	if $$failed ; then \ | ||||
| 		exit 1 ; \ | ||||
| 	fi | ||||
|   | ||||
| @@ -5,6 +5,7 @@ srcdir=`dirname $0` | ||||
| test -z "$srcdir" && srcdir=. | ||||
|  | ||||
| PKG_NAME="gnome-shell" | ||||
| REQUIRED_AUTOMAKE_VERSION=1.10 | ||||
|  | ||||
| (test -f $srcdir/configure.ac \ | ||||
|   && test -d $srcdir/src) || { | ||||
| @@ -14,7 +15,7 @@ PKG_NAME="gnome-shell" | ||||
| } | ||||
|  | ||||
| which gnome-autogen.sh || { | ||||
|     echo "You need to install gnome-common from GNOME Git (or from" | ||||
|     echo "You need to install gnome-common from GNOME Subversion (or from" | ||||
|     echo "your OS vendor's package manager)." | ||||
|     exit 1 | ||||
| } | ||||
|   | ||||
							
								
								
									
										164
									
								
								configure.ac
									
									
									
									
									
								
							
							
						
						| @@ -1,43 +1,33 @@ | ||||
| AC_PREREQ(2.63) | ||||
| AC_INIT([gnome-shell],[2.91.93],[https://bugzilla.gnome.org/enter_bug.cgi?product=gnome-shell],[gnome-shell]) | ||||
| AC_INIT(gnome-shell, 2.27.3) | ||||
|  | ||||
| AC_CONFIG_HEADERS([config.h]) | ||||
| AC_CONFIG_SRCDIR([src/shell-global.c]) | ||||
| AC_CONFIG_MACRO_DIR([m4]) | ||||
| AC_CONFIG_AUX_DIR([config]) | ||||
| AC_CONFIG_AUX_DIR(config) | ||||
|  | ||||
| AC_SUBST([PACKAGE_NAME], ["$PACKAGE_NAME"]) | ||||
| AC_SUBST([PACKAGE_VERSION], ["$PACKAGE_VERSION"]) | ||||
|  | ||||
| AM_INIT_AUTOMAKE([1.10 dist-bzip2 no-dist-gzip foreign]) | ||||
| AM_INIT_AUTOMAKE([dist-bzip2 no-dist-gzip foreign]) | ||||
| AM_MAINTAINER_MODE | ||||
|  | ||||
| m4_ifdef([AM_SILENT_RULES],[AM_SILENT_RULES([yes])]) | ||||
| m4_ifdef([AM_SILENT_RULES],[AM_SILENT_RULES([yes])],) | ||||
|  | ||||
| # Checks for programs. | ||||
| AC_CONFIG_HEADERS(config.h) | ||||
|  | ||||
| AC_DISABLE_STATIC | ||||
| AC_PROG_CC | ||||
| # Needed for per-target cflags, like in gnomeshell-taskpanel | ||||
| AM_PROG_CC_C_O | ||||
|  | ||||
| # Initialize libtool | ||||
| LT_PREREQ([2.2.6]) | ||||
| LT_INIT([disable-static]) | ||||
| AM_PROG_LIBTOOL | ||||
|  | ||||
| GETTEXT_PACKAGE=gnome-shell | ||||
| AC_SUBST(GETTEXT_PACKAGE) | ||||
| AC_DEFINE_UNQUOTED(GETTEXT_PACKAGE, "$GETTEXT_PACKAGE", | ||||
|                    [The prefix for our gettext translation domains.]) | ||||
|  | ||||
| PKG_PROG_PKG_CONFIG(0.16) | ||||
|  | ||||
| IT_PROG_INTLTOOL(0.26) | ||||
| AM_GLIB_GNU_GETTEXT | ||||
|  | ||||
| PKG_PROG_PKG_CONFIG([0.22]) | ||||
|  | ||||
| # GConf stuff | ||||
| AC_PATH_PROG(GCONFTOOL, gconftool-2, no) | ||||
| AM_GCONF_SOURCE_2 | ||||
|  | ||||
| GLIB_GSETTINGS | ||||
|  | ||||
| # Get a value to substitute into gnome-shell.in | ||||
| AM_PATH_PYTHON([2.5]) | ||||
| AC_SUBST(PYTHON) | ||||
| @@ -52,110 +42,41 @@ AC_MSG_CHECKING([for GStreamer (needed for recording functionality)]) | ||||
| if $PKG_CONFIG --exists gstreamer-0.10 '>=' $GSTREAMER_MIN_VERSION ; then | ||||
|    AC_MSG_RESULT(yes) | ||||
|    build_recorder=true | ||||
|    recorder_modules="gstreamer-0.10 gstreamer-base-0.10 x11" | ||||
|    PKG_CHECK_MODULES(TEST_SHELL_RECORDER, $recorder_modules clutter-1.0 xfixes) | ||||
|    recorder_modules="gstreamer-0.10 gstreamer-base-0.10 xfixes" | ||||
|    PKG_CHECK_MODULES(TEST_SHELL_RECORDER, $recorder_modules clutter-1.0) | ||||
| else | ||||
|    AC_MSG_RESULT(no) | ||||
| fi | ||||
|  | ||||
| AM_CONDITIONAL(BUILD_RECORDER, $build_recorder) | ||||
|  | ||||
| CLUTTER_MIN_VERSION=1.5.15 | ||||
| GOBJECT_INTROSPECTION_MIN_VERSION=0.10.1 | ||||
| GJS_MIN_VERSION=0.7.11 | ||||
| MUTTER_MIN_VERSION=2.91.93 | ||||
| GTK_MIN_VERSION=3.0.0 | ||||
| GIO_MIN_VERSION=2.25.9 | ||||
| LIBECAL_MIN_VERSION=2.32.0 | ||||
| LIBEDATASERVER_MIN_VERSION=1.2.0 | ||||
| LIBEDATASERVERUI2_MIN_VERSION=1.2.0 | ||||
| LIBEDATASERVERUI3_MIN_VERSION=2.91.6 | ||||
| TELEPATHY_GLIB_MIN_VERSION=0.13.12 | ||||
| TELEPATHY_LOGGER_MIN_VERSION=0.2.4 | ||||
| POLKIT_MIN_VERSION=0.100 | ||||
|  | ||||
| # Collect more than 20 libraries for a prize! | ||||
| PKG_CHECK_MODULES(GNOME_SHELL, gio-2.0 >= $GIO_MIN_VERSION | ||||
|                                gio-unix-2.0 dbus-glib-1 libxml-2.0 | ||||
|                                gtk+-3.0 >= $GTK_MIN_VERSION | ||||
|                                libmutter >= $MUTTER_MIN_VERSION | ||||
|                                gjs-internals-1.0 >= $GJS_MIN_VERSION | ||||
| 			       libgnome-menu $recorder_modules gconf-2.0 | ||||
|                                gdk-x11-3.0 | ||||
| 			       clutter-x11-1.0 >= $CLUTTER_MIN_VERSION | ||||
| 			       clutter-glx-1.0 >= $CLUTTER_MIN_VERSION | ||||
|                                libstartup-notification-1.0 | ||||
|                                gobject-introspection-1.0 >= $GOBJECT_INTROSPECTION_MIN_VERSION | ||||
| 			       libcanberra | ||||
|                                telepathy-glib >= $TELEPATHY_GLIB_MIN_VERSION | ||||
|                                telepathy-logger-0.2 >= $TELEPATHY_LOGGER_MIN_VERSION | ||||
|                                polkit-agent-1 >= $POLKIT_MIN_VERSION xfixes) | ||||
| PKG_CHECK_MODULES(MUTTER_PLUGIN, gio-unix-2.0 gtk+-2.0 dbus-glib-1 mutter-plugins | ||||
|                                  gjs-gi-1.0 libgnome-menu $recorder_modules gconf-2.0 | ||||
|                                  gdk-x11-2.0 clutter-x11-1.0 clutter-glx-1.0 | ||||
|                                  gnome-desktop-2.0 >= 2.26 libstartup-notification-1.0 | ||||
|                                  gobject-introspection-1.0 >= 0.6.5) | ||||
| PKG_CHECK_MODULES(TIDY, clutter-1.0) | ||||
| PKG_CHECK_MODULES(NBTK, clutter-1.0 gtk+-2.0 clutter-imcontext-0.1) | ||||
| PKG_CHECK_MODULES(BIG, clutter-1.0 gtk+-2.0 librsvg-2.0) | ||||
| PKG_CHECK_MODULES(GDMUSER, dbus-glib-1 gtk+-2.0) | ||||
| PKG_CHECK_MODULES(TOOLKIT, clutter-1.0 libcroco-0.6) | ||||
| PKG_CHECK_MODULES(TRAY, gtk+-2.0) | ||||
|  | ||||
| PKG_CHECK_MODULES(SHELL_PERF_HELPER, gtk+-3.0 gio-2.0) | ||||
| MUTTER_BIN_DIR=`$PKG_CONFIG --variable=exec_prefix mutter-plugins`/bin | ||||
| # FIXME: metacity-plugins.pc should point directly to its .gir file | ||||
| MUTTER_LIB_DIR=`$PKG_CONFIG --variable=libdir mutter-plugins` | ||||
| MUTTER_PLUGIN_DIR=`$PKG_CONFIG --variable=plugindir mutter-plugins` | ||||
| AC_SUBST(MUTTER_BIN_DIR) | ||||
| AC_SUBST(MUTTER_LIB_DIR) | ||||
| AC_SUBST(MUTTER_PLUGIN_DIR) | ||||
|  | ||||
| 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" | ||||
| # NM is the only typelib we use that we don't jhbuild | ||||
| PKG_CHECK_EXISTS([libnm-glib >= 0.8.995], | ||||
|         [NM_TYPELIBDIR=`$PKG_CONFIG --variable=libdir libnm-glib`/girepository-1.0 | ||||
| 	 if test "$INTROSPECTION_TYPELIBDIR" != "$NM_TYPELIBDIR"; then | ||||
| 	     JHBUILD_TYPELIBDIR="$JHBUILD_TYPELIBDIR:$NM_TYPELIBDIR" | ||||
| 	 fi]) | ||||
| AC_SUBST(JHBUILD_TYPELIBDIR) | ||||
|  | ||||
| saved_CFLAGS=$CFLAGS | ||||
| saved_LIBS=$LIBS | ||||
| CFLAGS=$GNOME_SHELL_CFLAGS | ||||
| LIBS=$GNOME_SHELL_LIBS | ||||
| # sn_startup_sequence_get_application_id, we can replace with a version check later | ||||
| AC_CHECK_FUNCS(JS_NewGlobalObject sn_startup_sequence_get_application_id XFixesCreatePointerBarrier) | ||||
| CFLAGS=$saved_CFLAGS | ||||
| LIBS=$saved_LIBS | ||||
|  | ||||
| PKG_CHECK_MODULES(ST, clutter-1.0 gtk+-3.0 libcroco-0.6 gnome-desktop-3.0 >= 2.90.0 x11) | ||||
| PKG_CHECK_MODULES(GDMUSER, dbus-glib-1 gtk+-3.0) | ||||
| PKG_CHECK_MODULES(TRAY, gtk+-3.0) | ||||
| PKG_CHECK_MODULES(GVC, libpulse libpulse-mainloop-glib gobject-2.0) | ||||
| PKG_CHECK_MODULES(JS_TEST, clutter-x11-1.0 gjs-1.0 gobject-introspection-1.0 gtk+-3.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 >= 2.90.0], | ||||
|         [BLUETOOTH_DIR=`$PKG_CONFIG --variable=libdir gnome-bluetooth-1.0`/gnome-bluetooth | ||||
| 	 BLUETOOTH_LIBS="-L'$BLUETOOTH_DIR' -lgnome-bluetooth-applet" | ||||
| 	 AC_SUBST([BLUETOOTH_LIBS],["$BLUETOOTH_LIBS"]) | ||||
| 	 AC_DEFINE_UNQUOTED([BLUETOOTH_DIR],["$BLUETOOTH_DIR"],[Path to installed GnomeBluetooth typelib and library]) | ||||
| 	 AC_DEFINE([HAVE_BLUETOOTH],[1],[Define if you have libgnome-bluetooth-applet]) | ||||
| 	 AC_SUBST([HAVE_BLUETOOTH],[1]) | ||||
| 	 AC_MSG_RESULT([yes])], | ||||
| 	[AC_DEFINE([HAVE_BLUETOOTH],[0]) | ||||
| 	 AC_SUBST([HAVE_BLUETOOTH],[0]) | ||||
| 	 AC_MSG_RESULT([no])]) | ||||
|  | ||||
| # Default to libedataserverui-3.0, but allow falling back to 1.2 | ||||
| PKG_CHECK_EXISTS(libedataserverui-3.0, | ||||
|                  [EDS_API=3.0 | ||||
|                   LIBEDATASERVERUI_MIN_VERSION=$LIBEDATASERVERUI3_MIN_VERSION], | ||||
|                  [EDS_API=1.2 | ||||
|                   LIBEDATASERVERUI_MIN_VERSION=$LIBEDATASERVERUI2_MIN_VERSION]) | ||||
| PKG_CHECK_MODULES(CALENDAR_SERVER, libecal-1.2 >= $LIBECAL_MIN_VERSION libedataserver-1.2 >= $LIBEDATASERVER_MIN_VERSION libedataserverui-$EDS_API >= $LIBEDATASERVERUI_MIN_VERSION gio-2.0) | ||||
| AC_SUBST(CALENDAR_SERVER_CFLAGS) | ||||
| AC_SUBST(CALENDAR_SERVER_LIBS) | ||||
|  | ||||
| MUTTER_GIR_DIR=`$PKG_CONFIG --variable=girdir libmutter` | ||||
| MUTTER_TYPELIB_DIR=`$PKG_CONFIG --variable=typelibdir libmutter` | ||||
| AC_SUBST(MUTTER_GIR_DIR) | ||||
| AC_SUBST(MUTTER_TYPELIB_DIR) | ||||
|  | ||||
| GJS_CONSOLE=`$PKG_CONFIG --variable=gjs_console gjs-1.0` | ||||
| AC_SUBST(GJS_CONSOLE) | ||||
| GJS_JS_DIR=`$PKG_CONFIG --variable=jsdir gjs-1.0` | ||||
| GJS_JS_NATIVE_DIR=`$PKG_CONFIG --variable=jsnativedir gjs-1.0` | ||||
| AC_SUBST(GJS_JS_DIR) | ||||
| AC_SUBST(GJS_JS_NATIVE_DIR) | ||||
|  | ||||
| AC_CHECK_FUNCS(fdwalk) | ||||
| AC_CHECK_FUNCS(mallinfo) | ||||
| AC_CHECK_HEADERS([sys/resource.h]) | ||||
|  | ||||
| # Sets GLIB_GENMARSHAL and GLIB_MKENUMS | ||||
| @@ -174,7 +95,8 @@ AC_SUBST(TYPELIBDIR) | ||||
| # 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]),, | ||||
|   AC_HELP_STRING([--enable-compile-warnings=@<:@no/minimum/yes/maximum/error@:>@], | ||||
|                  [Turn on compiler warnings]),, | ||||
|   enable_compile_warnings=error) | ||||
|  | ||||
| changequote(,)dnl | ||||
| @@ -198,18 +120,16 @@ if test "$enable_compile_warnings" != no ; then | ||||
| fi | ||||
| changequote([,])dnl | ||||
|  | ||||
| AC_ARG_ENABLE(jhbuild-wrapper-script, | ||||
|   AS_HELP_STRING([--jhbuild-wrapper-script=yes],[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_PATH_PROG(mutter, [mutter]) | ||||
| AC_SUBST(mutter) | ||||
|  | ||||
| AC_CONFIG_FILES([ | ||||
| AC_OUTPUT([ | ||||
|   Makefile | ||||
|   data/Makefile | ||||
|   js/Makefile | ||||
|   js/misc/config.js | ||||
|   js/misc/Makefile | ||||
|   js/ui/Makefile | ||||
|   src/Makefile | ||||
|   tests/Makefile | ||||
|   po/Makefile.in | ||||
|   man/Makefile | ||||
| ]) | ||||
| AC_OUTPUT | ||||
|   | ||||
| @@ -3,97 +3,46 @@ desktop_DATA = gnome-shell.desktop | ||||
|  | ||||
| # We substitute in bindir so it works as an autostart | ||||
| # file when built in a non-system prefix | ||||
| %.desktop.in:%.desktop.in.in | ||||
| gnome-shell.desktop.in: gnome-shell.desktop.in.in | ||||
| 	$(AM_V_GEN) sed -e "s|@bindir[@]|$(bindir)|" \ | ||||
| 	    -e "s|@VERSION[@]|$(VERSION)|" \ | ||||
| 	    $< > $@ || rm $@ | ||||
|  | ||||
| # Placeholder until we add intltool | ||||
| %.desktop:%.desktop.in | ||||
| gnome-shell.desktop: gnome-shell.desktop.in | ||||
| 	$(AM_V_GEN) sed s/^_// < $< > $@ || rm $@ | ||||
|  | ||||
| searchprovidersdir = $(pkgdatadir)/search_providers | ||||
| dist_searchproviders_DATA =				\ | ||||
| 	search_providers/google.xml				\ | ||||
| 	search_providers/wikipedia.xml | ||||
| imagesdir = $(pkgdatadir)/images | ||||
| dist_images_DATA =				\ | ||||
| 	add-workspace.svg			\ | ||||
| 	app-well-glow.png			\ | ||||
| 	back.svg				\ | ||||
| 	close.svg				\ | ||||
| 	close-black.svg				\ | ||||
| 	info.svg				\ | ||||
| 	magnifier.svg				\ | ||||
| 	remove-workspace.svg | ||||
|  | ||||
| themedir = $(pkgdatadir)/theme | ||||
| dist_theme_DATA =				\ | ||||
| 	theme/calendar-arrow-left.svg		\ | ||||
| 	theme/calendar-arrow-right.svg		\ | ||||
| 	theme/calendar-today.svg		\ | ||||
| 	theme/close-window.svg			\ | ||||
| 	theme/close.svg				\ | ||||
| 	theme/corner-ripple-ltr.png		\ | ||||
| 	theme/corner-ripple-rtl.png		\ | ||||
| 	theme/dash-placeholder.svg		\ | ||||
| 	theme/filter-selected-ltr.svg		\ | ||||
| 	theme/filter-selected-rtl.svg		\ | ||||
| 	theme/gnome-shell.css			\ | ||||
| 	theme/mosaic-view-active.svg		\ | ||||
| 	theme/mosaic-view.svg			\ | ||||
| 	theme/move-window-on-new.svg		\ | ||||
| 	theme/panel-border.svg			\ | ||||
| 	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-button-down-hover.png	\ | ||||
| 	theme/scroll-button-down.png		\ | ||||
| 	theme/scroll-button-up-hover.png	\ | ||||
| 	theme/scroll-button-down-hover.png	\ | ||||
| 	theme/scroll-button-up.png		\ | ||||
| 	theme/scroll-hhandle.svg		\ | ||||
| 	theme/scroll-vhandle.svg		\ | ||||
| 	theme/section-more.svg			\ | ||||
| 	theme/section-more-open.svg		\ | ||||
| 	theme/separator-white.png		\ | ||||
| 	theme/single-view-active.svg		\ | ||||
| 	theme/single-view.svg			\ | ||||
| 	theme/source-button-border.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.svg		\ | ||||
| 	theme/ws-switch-arrow-down.svg | ||||
| 	theme/scroll-button-up-hover.png	\ | ||||
| 	theme/scroll-vhandle.png | ||||
|  | ||||
| gsettings_SCHEMAS = org.gnome.shell.gschema.xml | ||||
|  | ||||
| @INTLTOOL_XML_NOMERGE_RULE@ | ||||
| @GSETTINGS_RULES@ | ||||
|  | ||||
| # We need to compile schemas at make time | ||||
| # to run from source tree | ||||
| gschemas.compiled: $(gsettings_SCHEMAS:.xml=.valid) | ||||
| 	$(AM_V_GEN) $(GLIB_COMPILE_SCHEMAS) --targetdir=. . | ||||
|  | ||||
| all-local: gschemas.compiled | ||||
|  | ||||
|  | ||||
| # GConf schemas: provide defaults for keys from Metacity we are overriding | ||||
| gconfschemadir  = @GCONF_SCHEMA_FILE_DIR@ | ||||
| gconfschema_DATA = gnome-shell.schemas | ||||
|  | ||||
| shadersdir = $(pkgdatadir)/shaders | ||||
| shaders_DATA = \ | ||||
| 	shaders/dim-window.glsl | ||||
| schemadir  = @GCONF_SCHEMA_FILE_DIR@ | ||||
| schema_DATA = gnome-shell.schemas | ||||
|  | ||||
| install-data-local: | ||||
| 	GCONF_CONFIG_SOURCE=$(GCONF_SCHEMA_CONFIG_SOURCE) $(GCONFTOOL) --makefile-install-rule $(srcdir)/$(gconfschema_DATA) | ||||
| 	GCONF_CONFIG_SOURCE=$(GCONF_SCHEMA_CONFIG_SOURCE) $(GCONFTOOL) --makefile-install-rule $(srcdir)/$(schema_DATA) | ||||
|  | ||||
| EXTRA_DIST =					\ | ||||
| 	gnome-shell.desktop.in.in		\ | ||||
| 	$(schema_DATA) | ||||
|  | ||||
|  | ||||
| EXTRA_DIST =						\ | ||||
| 	gnome-shell.desktop.in.in			\ | ||||
| 	$(menu_DATA)					\ | ||||
| 	$(gconfschema_DATA)				\ | ||||
| 	$(shaders_DATA)					\ | ||||
| 	org.gnome.shell.gschema.xml.in | ||||
|  | ||||
| CLEANFILES =						\ | ||||
| 	gnome-shell.desktop.in				\ | ||||
| 	$(desktop_DATA)					\ | ||||
| 	$(gsettings_SCHEMAS)				\ | ||||
| 	gschemas.compiled | ||||
| CLEANFILES =					\ | ||||
| 	gnome-shell.desktop.in			\ | ||||
| 	$(desktop_DATA) | ||||
|  | ||||
|   | ||||
							
								
								
									
										70
									
								
								data/add-workspace.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,70 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="59.995201" | ||||
|    height="59.995102" | ||||
|    id="svg3113" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.46" | ||||
|    version="1.0" | ||||
|    sodipodi:docname="add-workspace.svg" | ||||
|    inkscape:output_extension="org.inkscape.output.svg.inkscape"> | ||||
|   <defs | ||||
|      id="defs3115"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3121" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      gridtolerance="10000" | ||||
|      guidetolerance="10" | ||||
|      objecttolerance="10" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="375" | ||||
|      inkscape:cy="520" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="641" | ||||
|      inkscape:window-height="683" | ||||
|      inkscape:window-x="4" | ||||
|      inkscape:window-y="54" /> | ||||
|   <metadata | ||||
|      id="metadata3118"> | ||||
|     <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> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-498.57383,-439.50749)"> | ||||
|     <path | ||||
|        id="path3269" | ||||
|        d="M 528.57143,439.91129 C 512.23433,439.91129 498.97763,453.16795 498.97763,469.50504 C 498.97763,485.84214 512.23433,499.09881 528.57143,499.09879 C 544.90853,499.09879 558.16513,485.84215 558.16523,469.50504 C 558.16523,453.16794 544.90853,439.9113 528.57143,439.91129 z M 525.29023,451.16129 L 531.88393,451.16129 C 533.75363,451.16129 535.25893,452.66659 535.25893,454.53629 L 535.25893,462.84879 L 543.54023,462.84879 C 545.40973,462.84879 546.91523,464.35409 546.91523,466.22379 L 546.91523,472.81754 C 546.91523,474.68728 545.40993,476.19255 543.54023,476.19254 L 535.25893,476.19254 L 535.25893,484.47379 C 535.25893,486.34353 533.75363,487.8488 531.88393,487.84879 L 525.29023,487.84879 C 523.42053,487.84881 521.91523,486.34351 521.91523,484.47379 L 521.91523,476.19254 L 513.60263,476.19254 C 511.73313,476.19257 510.22773,474.68726 510.22763,472.81754 L 510.22763,466.22379 C 510.22763,464.35407 511.73303,462.8488 513.60263,462.84879 L 521.91523,462.84879 L 521.91523,454.53629 C 521.91523,452.66657 523.42043,451.1613 525.29023,451.16129 z" | ||||
|        style="opacity:0.30701785;fill:#ffffff;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:0.807603px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1;display:inline" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 3.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								data/app-well-glow.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.6 KiB | 
							
								
								
									
										7
									
								
								data/back.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| <?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: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="12" height="16" viewBox="0 0 12 16" enable-background="new 0 0 29 18" xml:space="preserve" sodipodi:version="0.32" inkscape:version="0.46+devel" sodipodi:docname="back.svg" inkscape:output_extension="org.inkscape.output.svg.inkscape"><metadata id="metadata16"><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="defs14"><inkscape:perspective sodipodi:type="inkscape:persp3d" inkscape:vp_x="0 : 9 : 1" inkscape:vp_y="0 : 1000 : 0" inkscape:vp_z="29 : 9 : 1" inkscape:persp3d-origin="14.5 : 6 : 1" id="perspective18"/></defs><sodipodi:namedview inkscape:window-height="728" inkscape:window-width="1103" inkscape:pageshadow="2" inkscape:pageopacity="1" guidetolerance="10.0" gridtolerance="10.0" objecttolerance="10.0" borderopacity="1.0" bordercolor="#666666" pagecolor="#000000" id="base" showgrid="true" inkscape:zoom="27.260185" inkscape:cx="12.592456" inkscape:cy="8.2696842" inkscape:window-x="145" inkscape:window-y="38" inkscape:current-layer="Foreground" inkscape:snap-global="true" showguides="false"><inkscape:grid type="xygrid" id="grid2391" empspacing="5" visible="true" enabled="true" snapvisiblegridlinesonly="true"/></sodipodi:namedview> | ||||
|  | ||||
|  | ||||
|  | ||||
| <path style="fill: rgb(255, 255, 255); fill-opacity: 1; stroke: none;" d="M 10,2 10,14 2,8 10,2 z" id="path43"/></svg> | ||||
| After Width: | Height: | Size: 1.9 KiB | 
							
								
								
									
										66
									
								
								data/close-black.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,66 @@ | ||||
| <?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:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    version="1.1" | ||||
|    id="Foreground" | ||||
|    x="0px" | ||||
|    y="0px" | ||||
|    width="16px" | ||||
|    height="16px" | ||||
|    viewBox="0 0 16 16" | ||||
|    enable-background="new 0 0 16 16" | ||||
|    xml:space="preserve" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.46+devel" | ||||
|    sodipodi:docname="close-black.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></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" /></defs><sodipodi:namedview | ||||
|    inkscape:window-height="811" | ||||
|    inkscape:window-width="1272" | ||||
|    inkscape:pageshadow="2" | ||||
|    inkscape:pageopacity="0.0" | ||||
|    guidetolerance="10.0" | ||||
|    gridtolerance="10.0" | ||||
|    objecttolerance="10.0" | ||||
|    borderopacity="1.0" | ||||
|    bordercolor="#666666" | ||||
|    pagecolor="#ffffff" | ||||
|    id="base" | ||||
|    showgrid="false" | ||||
|    inkscape:zoom="32.125" | ||||
|    inkscape:cx="8" | ||||
|    inkscape:cy="10.440056" | ||||
|    inkscape:window-x="40" | ||||
|    inkscape:window-y="40" | ||||
|    inkscape:current-layer="Foreground" /> | ||||
| <path | ||||
|    fill-rule="evenodd" | ||||
|    clip-rule="evenodd" | ||||
|    d="M10.5,3.5l2,2L10,8l2.5,2.5l-2,2L8,10l-2.5,2.5l-2-2L6,8L3.5,5.5l2-2L8,6L10.5,3.5  z M0,8c0-4.418,3.582-8,8-8s8,3.582,8,8s-3.582,8-8,8S0,12.418,0,8z" | ||||
|    id="path2394" | ||||
|    style="fill-opacity:1;fill:#000000" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.3 KiB | 
| Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 2.6 KiB | 
| @@ -7,10 +7,9 @@ X-GNOME-Bugzilla-Bugzilla=GNOME | ||||
| X-GNOME-Bugzilla-Product=gnome-shell | ||||
| X-GNOME-Bugzilla-Component=general | ||||
| X-GNOME-Bugzilla-Version=@VERSION@ | ||||
| Categories=GNOME;GTK;Core; | ||||
| Categories=GNOME;GTK;Utility;Core; | ||||
| OnlyShowIn=GNOME; | ||||
| NoDisplay=true | ||||
| X-GNOME-Autostart-Phase=WindowManager | ||||
| X-GNOME-Provides=panel;windowmanager; | ||||
| X-GNOME-Autostart-Notify=true | ||||
| X-GNOME-AutoRestart=true | ||||
|   | ||||
| @@ -1,100 +1,93 @@ | ||||
| <gconfschemafile> | ||||
|     <schemalist> | ||||
|  | ||||
|       <!-- Metacity overrides --> | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/windows/attach_modal_dialogs</key> | ||||
|         <applyto>/desktop/gnome/shell/windows/attach_modal_dialogs</applyto> | ||||
|         <key>/schemas/desktop/gnome/shell/development_tools</key> | ||||
|         <applyto>/desktop/gnome/shell/development_tools</applyto> | ||||
|         <owner>gnome-shell</owner> | ||||
|         <type>bool</type> | ||||
|         <default>true</default> | ||||
|         <locale name="C"> | ||||
|           <short>Attach modal dialog to the parent window</short> | ||||
|           <long> | ||||
|              This key overrides /apps/mutter/general/attach_modal_dialogs when | ||||
|              running GNOME Shell. | ||||
|           </long> | ||||
|             <short>Enable internal tools useful for developers and testers from Alt-F2</short> | ||||
|             <long> | ||||
|                 Allows access to internal debugging and monitoring tools using | ||||
|                 the Alt-F2 dialog. | ||||
|             </long> | ||||
|         </locale> | ||||
|       </schema> | ||||
|  | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/windows/button_layout</key> | ||||
|         <applyto>/desktop/gnome/shell/windows/button_layout</applyto> | ||||
|         <owner>gnome-shell</owner> | ||||
|         <type>string</type> | ||||
|         <default>:close</default> | ||||
|         <locale name="C"> | ||||
|            <short>Arrangement of buttons on the titlebar</short> | ||||
|            <long> | ||||
|              Arrangement of buttons on the titlebar. The | ||||
|              value should be a string, such as | ||||
|              "menu:minimize,maximize,spacer,close"; the colon separates the | ||||
|              left corner of the window from the right corner, and | ||||
|              the button names are comma-separated. Duplicate buttons | ||||
|              are not allowed. Unknown button names are silently ignored | ||||
|              so that buttons can be added in future gnome-shell versions | ||||
|              without breaking older versions. | ||||
|              A special spacer tag can be used to insert some space between | ||||
|              two adjacent buttons. | ||||
|  | ||||
|              This key overrides /apps/metacity/general/button_layout when | ||||
|              running GNOME Shell. | ||||
|            </long> | ||||
|         </locale> | ||||
|         <key>/schemas/desktop/gnome/shell/app_monitor/enable_monitoring</key> | ||||
| 	<applyto>/desktop/gnome/shell/app_monitor/enable_monitoring</applyto> | ||||
| 	<owner>gnome-shell</owner> | ||||
| 	<type>bool</type> | ||||
| 	<default>true</default> | ||||
| 	<locale name="C"> | ||||
| 	  <short>Whether to collect stats about applications usage</short> | ||||
| 	  <long> | ||||
| 	    The shell normally monitors active applications in order to present the most used ones (e.g. in launchers). While this data will be kept private, you may want to disable this for privacy reasons. Please note that doing so won't remove already saved data. | ||||
| 	  </long> | ||||
| 	</locale> | ||||
|       </schema> | ||||
|  | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/windows/edge_tiling</key> | ||||
|         <applyto>/desktop/gnome/shell/windows/edge_tiling</applyto> | ||||
|         <owner>gnome-shell</owner> | ||||
|         <type>bool</type> | ||||
|         <default>true</default> | ||||
|         <locale name="C"> | ||||
|           <short>enable edge tiling when dropping windows on screen edges</short> | ||||
|           <long> | ||||
|              If enabled, dropping windows on vertical screen edges maximizes them | ||||
|              vertically and resizes them horizontally to cover half of the | ||||
|              available area. Dropping windows on the top screen edge maximizes them | ||||
|              completely. | ||||
|  | ||||
|              This key overrides /apps/metacity/general/edge_tiling when | ||||
|              running GNOME Shell. | ||||
|           </long> | ||||
|         </locale> | ||||
|         <key>/schemas/desktop/gnome/shell/favorite_apps</key> | ||||
| 	<applyto>/desktop/gnome/shell/favorite_apps</applyto> | ||||
| 	<owner>gnome-shell</owner> | ||||
| 	<type>list</type> | ||||
| 	<list_type>string</list_type> | ||||
| 	<default>[mozilla-firefox.desktop,evolution.desktop,openoffice.org-writer.desktop]</default> | ||||
| 	<locale name="C"> | ||||
| 	  <short>List of desktop file IDs for favorite applications</short> | ||||
| 	  <long> | ||||
|         The applications corresponding to these identifiers will be displayed in the favorites area. | ||||
| 	  </long> | ||||
| 	</locale> | ||||
|       </schema> | ||||
|  | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/windows/theme</key> | ||||
|         <applyto>/desktop/gnome/shell/windows/theme</applyto> | ||||
|         <owner>gnome-shell</owner> | ||||
|         <type>string</type> | ||||
|         <default>Adwaita</default> | ||||
|         <locale name="C"> | ||||
|           <short>Current theme</short> | ||||
|           <long> | ||||
|             The theme determines the appearance of window borders, | ||||
|             titlebar, and so forth. | ||||
|  | ||||
|             This key overrides /apps/metacity/general/theme when | ||||
|             running GNOME Shell. | ||||
|           </long> | ||||
|         </locale> | ||||
|         <key>/schemas/desktop/gnome/shell/sidebar/visible</key> | ||||
| 	<applyto>/desktop/gnome/shell/sidebar/visible</applyto> | ||||
| 	<owner>gnome-shell</owner> | ||||
| 	<type>bool</type> | ||||
| 	<default>false</default> | ||||
| 	<locale name="C"> | ||||
| 	  <short>Whether or not to display the sidebar</short> | ||||
| 	  <long> | ||||
| 	    Determines whether or not the sidebar is visible. | ||||
| 	  </long> | ||||
| 	</locale> | ||||
|       </schema> | ||||
|  | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/windows/workspaces_only_on_primary</key> | ||||
|         <applyto>/desktop/gnome/shell/windows/workspaces_only_on_primary</applyto> | ||||
|         <owner>gnome-shell</owner> | ||||
|         <type>bool</type> | ||||
|         <default>true</default> | ||||
|         <locale name="C"> | ||||
|           <short>Workspaces only on primary monitor</short> | ||||
|           <long> | ||||
|              This key overrides /apps/mutter/general/workspaces_only_on_primary when | ||||
|              running GNOME Shell. | ||||
|           </long> | ||||
|         </locale> | ||||
|         <key>/schemas/desktop/gnome/shell/sidebar/expanded</key> | ||||
| 	<applyto>/desktop/gnome/shell/sidebar/expanded</applyto> | ||||
| 	<owner>gnome-shell</owner> | ||||
| 	<type>bool</type> | ||||
| 	<default>true</default> | ||||
| 	<locale name="C"> | ||||
| 	  <short>Whether the sidebar should be in the expanded (wide) mode</short> | ||||
| 	  <long> | ||||
| 	    Controls the expanded/collapsed state of the sidebar. | ||||
| 	  </long> | ||||
| 	</locale> | ||||
|       </schema> | ||||
|  | ||||
|       <schema> | ||||
|         <key>/schemas/desktop/gnome/shell/sidebar/widgets</key> | ||||
| 	<applyto>/desktop/gnome/shell/sidebar/widgets</applyto> | ||||
| 	<owner>gnome-shell</owner> | ||||
| 	<type>list</type> | ||||
| 	<list_type>string</list_type> | ||||
| 	<default>[imports.ui.widget.ClockWidget,imports.ui.widget.AppsWidget,imports.ui.widget.RecentDocsWidget]</default> | ||||
| 	<locale name="C"> | ||||
| 	  <short>The widgets to display in the sidebar</short> | ||||
| 	  <long> | ||||
| 	    The widgets to display in the sidebar, in order from top to bottom. Each widget "name" is actually a JavaScript expression referring to a widget constructor object. | ||||
| 	  </long> | ||||
| 	</locale> | ||||
|       </schema> | ||||
|  | ||||
|   </schemalist> | ||||
|  | ||||
| </gconfschemafile> | ||||
|   | ||||
							
								
								
									
										74
									
								
								data/info.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,74 @@ | ||||
| <?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.1" | ||||
|    id="Foreground" | ||||
|    x="0px" | ||||
|    y="0px" | ||||
|    width="16px" | ||||
|    height="16px" | ||||
|    viewBox="0 0 16 16" | ||||
|    enable-background="new 0 0 16 16" | ||||
|    xml:space="preserve" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.46" | ||||
|    sodipodi:docname="info_16.svg" | ||||
|    inkscape:output_extension="org.inkscape.output.svg.inkscape"><metadata | ||||
|    id="metadata2389"><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><defs | ||||
|    id="defs2387"><linearGradient | ||||
|      id="linearGradient3710"><stop | ||||
|        style="stop-color:#c4c4c4;stop-opacity:1;" | ||||
|        offset="0" | ||||
|        id="stop3712" /><stop | ||||
|        style="stop-color:#ffffff;stop-opacity:1;" | ||||
|        offset="1" | ||||
|        id="stop3714" /></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="perspective2391" /><linearGradient | ||||
|      inkscape:collect="always" | ||||
|      xlink:href="#linearGradient3710" | ||||
|      id="linearGradient3716" | ||||
|      x1="7.9066148" | ||||
|      y1="15.937743" | ||||
|      x2="7.9377432" | ||||
|      y2="0.031128405" | ||||
|      gradientUnits="userSpaceOnUse" /></defs><sodipodi:namedview | ||||
|    inkscape:window-height="713" | ||||
|    inkscape:window-width="722" | ||||
|    inkscape:pageshadow="2" | ||||
|    inkscape:pageopacity="0.0" | ||||
|    guidetolerance="10.0" | ||||
|    gridtolerance="10.0" | ||||
|    objecttolerance="10.0" | ||||
|    borderopacity="1.0" | ||||
|    bordercolor="#666666" | ||||
|    pagecolor="#ffffff" | ||||
|    id="base" | ||||
|    showgrid="false" | ||||
|    inkscape:zoom="32.125" | ||||
|    inkscape:cx="8" | ||||
|    inkscape:cy="8.154146" | ||||
|    inkscape:window-x="20" | ||||
|    inkscape:window-y="20" | ||||
|    inkscape:current-layer="Foreground" /> | ||||
| <path | ||||
|    fill-rule="evenodd" | ||||
|    clip-rule="evenodd" | ||||
|    d="M7,3h2v2H7V3z M5.5,12H7V8H5.5V7H9v5h1.5v1h-5V12z M0,8c0-4.418,3.582-8,8-8  s8,3.582,8,8s-3.582,8-8,8S0,12.418,0,8z" | ||||
|    id="path2384" | ||||
|    style="fill-opacity:1;fill:url(#linearGradient3716)" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.5 KiB | 
							
								
								
									
										80
									
								
								data/magnifier.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,80 @@ | ||||
| <?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: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="18" | ||||
|    height="18" | ||||
|    viewBox="0 0 18 18" | ||||
|    enable-background="new 0 0 29 18" | ||||
|    xml:space="preserve" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.46+devel" | ||||
|    sodipodi:docname="magnifier.svg" | ||||
|    inkscape:output_extension="org.inkscape.output.svg.inkscape"><metadata | ||||
|    id="metadata16"><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="defs14"><inkscape:perspective | ||||
|      sodipodi:type="inkscape:persp3d" | ||||
|      inkscape:vp_x="0 : 9 : 1" | ||||
|      inkscape:vp_y="0 : 1000 : 0" | ||||
|      inkscape:vp_z="29 : 9 : 1" | ||||
|      inkscape:persp3d-origin="14.5 : 6 : 1" | ||||
|      id="perspective18" /></defs><sodipodi:namedview | ||||
|    inkscape:window-height="728" | ||||
|    inkscape:window-width="1103" | ||||
|    inkscape:pageshadow="2" | ||||
|    inkscape:pageopacity="1" | ||||
|    guidetolerance="10.0" | ||||
|    gridtolerance="10.0" | ||||
|    objecttolerance="10.0" | ||||
|    borderopacity="1.0" | ||||
|    bordercolor="#666666" | ||||
|    pagecolor="#000000" | ||||
|    id="base" | ||||
|    showgrid="true" | ||||
|    inkscape:zoom="27.260185" | ||||
|    inkscape:cx="9.5844061" | ||||
|    inkscape:cy="9.4435574" | ||||
|    inkscape:window-x="142" | ||||
|    inkscape:window-y="26" | ||||
|    inkscape:current-layer="Foreground" | ||||
|    inkscape:snap-global="true" | ||||
|    showguides="false"><inkscape:grid | ||||
|      type="xygrid" | ||||
|      id="grid2391" | ||||
|      empspacing="5" | ||||
|      visible="true" | ||||
|      enabled="true" | ||||
|      snapvisiblegridlinesonly="true" /></sodipodi:namedview> | ||||
|  | ||||
| <g | ||||
|    id="g5" | ||||
|    style="fill:#ffffff;fill-opacity:1" | ||||
|    transform="translate(-4,-0.023114)"> | ||||
| 	<path | ||||
|    d="m 6.246,13.98 c -0.319,-0.319 -0.319,-0.837 0,-1.157 L 9.963,9.106 c 0.319,-0.319 0.837,-0.319 1.157,0 l 0.786,0.787 c 0.32,0.319 0.32,0.837 0,1.157 l -3.717,3.717 c -0.32,0.319 -0.838,0.319 -1.157,0 l -0.786,-0.787 0,0 z" | ||||
|    id="path7" | ||||
|    style="fill:#ffffff;fill-opacity:1" /> | ||||
| 	<path | ||||
|    d="M 9.076,11.937" | ||||
|    id="path9" | ||||
|    style="fill:#ffffff;fill-opacity:1" /> | ||||
| </g> | ||||
| <path | ||||
|    clip-rule="evenodd" | ||||
|    d="m 7.25,7.476886 c 0,-1.243 1.007,-2.25 2.2499998,-2.25 1.2430002,0 2.2500002,1.007 2.2500002,2.25 0,1.243 -1.007,2.25 -2.2500002,2.25 C 8.257,9.726886 7.25,8.719886 7.25,7.476886 z m -2.25,0 c 0,-2.485 2.015,-4.5 4.4999998,-4.5 2.4850002,0 4.5000002,2.015 4.5000002,4.5 0,2.4849998 -2.015,4.5 -4.5000002,4.5 C 7.015,11.976886 5,9.9618858 5,7.476886 z" | ||||
|    id="path11" | ||||
|    style="fill:#ffffff;fill-opacity:1;fill-rule:evenodd" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.9 KiB | 
| @@ -1,123 +0,0 @@ | ||||
| <schemalist> | ||||
|   <schema id="org.gnome.shell" path="/org/gnome/shell/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="development-tools" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary> | ||||
|         Enable internal tools useful for developers and testers from Alt-F2 | ||||
|       </_summary> | ||||
|       <_description> | ||||
|         Allows access to internal debugging and monitoring tools | ||||
|         using the Alt-F2 dialog. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="disabled-extensions" type="as"> | ||||
|       <default>[]</default> | ||||
|       <_summary>Uuids of extensions to disable</_summary> | ||||
|       <_description> | ||||
|         GNOME Shell extensions have a uuid property; | ||||
|         this key lists extensions which should not be loaded. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="enable-app-monitoring" type="b"> | ||||
|       <default>true</default> | ||||
|       <_summary>Whether to collect stats about applications usage</_summary> | ||||
|       <_description> | ||||
|         The shell normally monitors active applications in order to present | ||||
|         the most used ones (e.g. in launchers). While this data will be | ||||
|         kept private, you may want to disable this for privacy reasons. | ||||
|         Please note that doing so won't remove already saved data. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="favorite-apps" type="as"> | ||||
|       <default>[ 'mozilla-firefox.desktop', 'evolution.desktop', 'empathy.desktop', 'rhythmbox.desktop', 'shotwell.desktop', 'openoffice.org-writer.desktop', 'nautilus.desktop' ]</default> | ||||
|       <_summary>List of desktop file IDs for favorite applications</_summary> | ||||
|       <_description> | ||||
|         The applications corresponding to these identifiers | ||||
|         will be displayed in the favorites area. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="disabled-open-search-providers" type="as"> | ||||
|       <default>[]</default> | ||||
|       <_summary>disabled OpenSearch providers</_summary> | ||||
|     </key> | ||||
|     <key name="command-history" type="as"> | ||||
|       <default>[]</default> | ||||
|       <_summary>History for command (Alt-F2) dialog</_summary> | ||||
|     </key> | ||||
|     <key name="looking-glass-history" type="as"> | ||||
|       <default>[]</default> | ||||
|       <_summary>History for the looking glass dialog</_summary> | ||||
|     </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"/> | ||||
|   </schema> | ||||
|  | ||||
|   <schema id="org.gnome.shell.calendar" path="/org/gnome/shell/calendar/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="show-weekdate" type="b"> | ||||
|       <default>false</default> | ||||
|       <_summary>Show the week date in the calendar</_summary> | ||||
|       <_description> | ||||
|         If true, display the ISO week date in the calendar. | ||||
|       </_description> | ||||
|       </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"> | ||||
|       <default>15</default> | ||||
|       <_summary>Framerate used for recording screencasts.</_summary> | ||||
|       <_description> | ||||
|         The framerate of the resulting screencast recordered | ||||
|         by GNOME Shell's screencast recorder in frames-per-second. | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="pipeline" type="s"> | ||||
|       <default>''</default> | ||||
|       <_summary>The gstreamer pipeline used to encode the screencast</_summary> | ||||
|       <_description> | ||||
|         Sets the GStreamer pipeline used to encode recordings. | ||||
|         It follows the syntax used for gst-launch. The pipeline should have | ||||
|         an unconnected sink pad where the recorded video is recorded. It will | ||||
|         normally have a unconnected source pad; output from that pad | ||||
|         will be written into the output file. However the pipeline can also | ||||
|         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 | ||||
|         'videorate ! vp8enc quality=10 speed=2 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> | ||||
|     </key> | ||||
|     <key name="file-extension" type="s"> | ||||
|       <default>'webm'</default> | ||||
|       <_summary>File extension used for storing the screencast</_summary> | ||||
|       <_description> | ||||
|         The filename for recorded screencasts will be a unique filename | ||||
|         based on the current date, and use this extension. It should be | ||||
|         changed when recording to a different container format. | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
| </schemalist> | ||||
							
								
								
									
										71
									
								
								data/remove-workspace.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,71 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="59.995201" | ||||
|    height="59.995102" | ||||
|    id="svg3113" | ||||
|    sodipodi:version="0.32" | ||||
|    inkscape:version="0.46" | ||||
|    version="1.0" | ||||
|    sodipodi:docname="remove-workspace.svg" | ||||
|    inkscape:output_extension="org.inkscape.output.svg.inkscape"> | ||||
|   <defs | ||||
|      id="defs3115"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3121" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      gridtolerance="10000" | ||||
|      guidetolerance="10" | ||||
|      objecttolerance="10" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="4.5" | ||||
|      inkscape:cx="-8.1974244" | ||||
|      inkscape:cy="38.948933" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1400" | ||||
|      inkscape:window-height="971" | ||||
|      inkscape:window-x="454" | ||||
|      inkscape:window-y="105" /> | ||||
|   <metadata | ||||
|      id="metadata3118"> | ||||
|     <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> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-498.57383,-439.50749)"> | ||||
|     <path | ||||
|        style="opacity:0.30701785;fill:#ffffff;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:0.807603px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1;display:inline" | ||||
|        d="M 30 0.40625 C 13.662899 0.40624999 0.40625 13.66291 0.40625 30 C 0.40624999 46.337101 13.6629 59.59377 30 59.59375 C 46.337099 59.593749 59.59365 46.33711 59.59375 30 C 59.59375 13.662901 46.3371 0.40626 30 0.40625 z M 15.03125 23.34375 L 44.96875 23.34375 C 46.83825 23.343751 48.34375 24.84905 48.34375 26.71875 L 48.34375 33.3125 C 48.34375 35.182239 46.83845 36.68751 44.96875 36.6875 L 15.03125 36.6875 C 13.16175 36.687529 11.65635 35.18222 11.65625 33.3125 L 11.65625 26.71875 C 11.65625 24.849031 13.16165 23.34376 15.03125 23.34375 z " | ||||
|        transform="translate(498.57383,439.50749)" | ||||
|        id="path2382" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 2.8 KiB | 
| @@ -1,7 +0,0 @@ | ||||
| <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> | ||||
| @@ -1,44 +0,0 @@ | ||||
| <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,26 +0,0 @@ | ||||
| #version 110 | ||||
| uniform sampler2D sampler0; | ||||
| 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(sampler0, gl_TexCoord[0].st); | ||||
|   float y = height * gl_TexCoord[0][1]; | ||||
|  | ||||
|   // To reduce contrast, blend with a mid gray | ||||
|   gl_FragColor = color * contrast - off * c; | ||||
|  | ||||
|   // We only fully dim at a distance of BORDER_MAX_HEIGHT from the edge and | ||||
|   // when the fraction is 1.0. For other locations and fractions we linearly | ||||
|   // interpolate back to the original undimmed color. | ||||
|   gl_FragColor = color + (gl_FragColor - color) * min(y / border_max_height, 1.0); | ||||
|   gl_FragColor = color + (gl_FragColor - color) * fraction; | ||||
| } | ||||
| @@ -1,82 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="16" | ||||
|    height="16" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48+devel r9942 custom" | ||||
|    sodipodi:docname="New document 4"> | ||||
|   <defs | ||||
|      id="defs4" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="8.984481" | ||||
|      inkscape:cy="5.6224906" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:window-width="930" | ||||
|      inkscape:window-height="681" | ||||
|      inkscape:window-x="1892" | ||||
|      inkscape:window-y="272" | ||||
|      inkscape:window-maximized="0"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid17403" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <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" | ||||
|      transform="translate(0,-1036.3622)"> | ||||
|     <path | ||||
|        sodipodi:type="star" | ||||
|        style="fill:#5f5f5f;fill-opacity:1;stroke:#5f5f5f;stroke-width:0.43015847;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" | ||||
|        id="path18028" | ||||
|        sodipodi:sides="3" | ||||
|        sodipodi:cx="84.5" | ||||
|        sodipodi:cy="337.5" | ||||
|        sodipodi:r1="5" | ||||
|        sodipodi:r2="2.5" | ||||
|        sodipodi:arg1="0.52359878" | ||||
|        sodipodi:arg2="1.5707963" | ||||
|        inkscape:flatsided="true" | ||||
|        inkscape:rounded="0" | ||||
|        inkscape:randomized="0" | ||||
|        d="M 88.830127,340 80.169873,340 84.5,332.5 z" | ||||
|        transform="matrix(0,1.3621708,0.99186247,0,-325.48222,929.32667)" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.5 KiB | 
| @@ -1,82 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="16" | ||||
|    height="16" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48+devel r9942 custom" | ||||
|    sodipodi:docname="arrow-left.svg"> | ||||
|   <defs | ||||
|      id="defs4" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="7.7366092" | ||||
|      inkscape:cy="6.4536271" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:window-width="930" | ||||
|      inkscape:window-height="681" | ||||
|      inkscape:window-x="1892" | ||||
|      inkscape:window-y="272" | ||||
|      inkscape:window-maximized="0"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid17403" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <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" | ||||
|      transform="translate(0,-1036.3622)"> | ||||
|     <path | ||||
|        sodipodi:type="star" | ||||
|        style="fill:#5f5f5f;fill-opacity:1;stroke:#5f5f5f;stroke-width:0.43015847;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" | ||||
|        id="path18028" | ||||
|        sodipodi:sides="3" | ||||
|        sodipodi:cx="84.5" | ||||
|        sodipodi:cy="337.5" | ||||
|        sodipodi:r1="5" | ||||
|        sodipodi:r2="2.5" | ||||
|        sodipodi:arg1="0.52359878" | ||||
|        sodipodi:arg2="1.5707963" | ||||
|        inkscape:flatsided="true" | ||||
|        inkscape:rounded="0" | ||||
|        inkscape:randomized="0" | ||||
|        d="M 88.830127,340 80.169873,340 84.5,332.5 z" | ||||
|        transform="matrix(0,1.3621708,-0.99186247,0,342.48324,929.32667)" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.5 KiB | 
| @@ -1,187 +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="28" | ||||
|    height="25" | ||||
|    id="svg10621" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="calendar-today.svg"> | ||||
|   <defs | ||||
|      id="defs10623"> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient99561-1" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient34508-1-3"> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop34510-1-9" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" | ||||
|          offset="1" | ||||
|          id="stop34512-4-5" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        r="42" | ||||
|        fy="30" | ||||
|        fx="51" | ||||
|        cy="30" | ||||
|        cx="51" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="radialGradient10592" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        inkscape:collect="always" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3770" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3001" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3007" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3067" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient3072" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient2997" | ||||
|        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" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="15.839192" | ||||
|      inkscape:cx="8.3750933" | ||||
|      inkscape:cy="8.0837211" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      fit-margin-top="0" | ||||
|      fit-margin-left="0" | ||||
|      fit-margin-right="0" | ||||
|      fit-margin-bottom="0" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10626"> | ||||
|     <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(-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: 5.7 KiB | 
| @@ -1,152 +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+devel r10081 custom" | ||||
|    sodipodi:docname="close-window.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:#7b7b7b;stop-opacity:1" | ||||
|        offset="0" | ||||
|        id="stop16500-8" /><stop | ||||
|        style="stop-color:#101010;stop-opacity:1" | ||||
|        offset="1" | ||||
|        id="stop16502-0" /></linearGradient><filter | ||||
|      color-interpolation-filters="sRGB" | ||||
|      inkscape:collect="always" | ||||
|      id="filter16524-9" | ||||
|      x="-0.212979" | ||||
|      width="1.425958" | ||||
|      y="-0.21305652" | ||||
|      height="1.426113"><feGaussianBlur | ||||
|        inkscape:collect="always" | ||||
|        stdDeviation="0.71020915" | ||||
|        id="feGaussianBlur16526-0" /></filter></defs><sodipodi:namedview | ||||
|    inkscape:window-height="1114" | ||||
|    inkscape:window-width="1463" | ||||
|    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="1" | ||||
|    inkscape:cx="10.720189" | ||||
|    inkscape:cy="13.739577" | ||||
|    inkscape:window-x="0" | ||||
|    inkscape:window-y="26" | ||||
|    inkscape:current-layer="Foreground" | ||||
|    showguides="true" | ||||
|    inkscape:guide-bbox="true" | ||||
|    borderlayer="true" | ||||
|    inkscape:showpageshadow="false" | ||||
|    inkscape:window-maximized="0"><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"><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 | ||||
|      transform="matrix(0.72727273,0,0,0.72727273,2.368236,2.1803254)" | ||||
|      style="fill:#ffffff;fill-opacity:1;display:inline" | ||||
|      id="g27275-6-6" | ||||
|      inkscape:label="window-close"><g | ||||
|        style="fill:#ffffff;fill-opacity:1;display:inline" | ||||
|        id="g27277-1-1" | ||||
|        transform="translate(-41,-760)"><path | ||||
|          sodipodi:type="inkscape:offset" | ||||
|          inkscape:radius="0" | ||||
|          inkscape:original="M 44.21875 764.1875 L 44.21875 765.1875 C 44.19684 765.46825 44.289258 765.74287 44.5 765.9375 L 46.78125 768.21875 L 44.5 770.46875 C 44.31181 770.65692 44.218747 770.92221 44.21875 771.1875 L 44.21875 772.1875 L 45.21875 772.1875 C 45.48404 772.1875 45.749336 772.09444 45.9375 771.90625 L 48.21875 769.625 L 50.5 771.90625 C 50.688164 772.0944 50.953449 772.18749 51.21875 772.1875 L 52.21875 772.1875 L 52.21875 771.1875 C 52.218742 770.9222 52.125688 770.65692 51.9375 770.46875 L 49.6875 768.21875 L 51.96875 765.9375 C 52.18441 765.73815 52.21875 765.47397 52.21875 765.1875 L 52.21875 764.1875 L 51.21875 764.1875 C 50.977922 764.1945 50.796875 764.2695 50.53125 764.5 L 48.21875 766.78125 L 45.9375 764.5 C 45.75987 764.31608 45.504951 764.1987 45.25 764.1875 C 45.23954 764.18704 45.22912 764.18738 45.21875 764.1875 L 44.21875 764.1875 z " | ||||
|          xlink:href="#path27279-0-5" | ||||
|          style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;block-progression:tb;writing-mode:lr-tb;text-anchor:start;color:#bebebe;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.78124988;marker:none;visibility:visible;display:inline;overflow:visible;filter:url(#filter16524-9);enable-background:new;font-family:Andale Mono;-inkscape-font-specification:Andale Mono" | ||||
|          id="path16506-5" | ||||
|          inkscape:href="#path27279-0-5" | ||||
|          d="m 44.21875,764.1875 0,1 c -0.02191,0.28075 0.07051,0.55537 0.28125,0.75 l 2.28125,2.28125 -2.28125,2.25 c -0.18819,0.18817 -0.281253,0.45346 -0.28125,0.71875 l 0,1 1,0 c 0.26529,0 0.530586,-0.0931 0.71875,-0.28125 L 48.21875,769.625 50.5,771.90625 c 0.188164,0.18815 0.453449,0.28124 0.71875,0.28125 l 1,0 0,-1 c -8e-6,-0.2653 -0.09306,-0.53058 -0.28125,-0.71875 l -2.25,-2.25 2.28125,-2.28125 c 0.21566,-0.19935 0.25,-0.46353 0.25,-0.75 l 0,-1 -1,0 c -0.240828,0.007 -0.421875,0.082 -0.6875,0.3125 l -2.3125,2.28125 L 45.9375,764.5 c -0.17763,-0.18392 -0.432549,-0.3013 -0.6875,-0.3125 -0.01046,-4.6e-4 -0.02088,-1.2e-4 -0.03125,0 l -1,0 z" | ||||
|          transform="translate(0,1.3535534)" /><path | ||||
|          sodipodi:nodetypes="ccsccccccccccccccccccccccc" | ||||
|          style="font-size:medium;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;text-indent:0;text-align:start;text-decoration:none;line-height:normal;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;block-progression:tb;writing-mode:lr-tb;text-anchor:start;color:#bebebe;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.78124988;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:new;font-family:Andale Mono;-inkscape-font-specification:Andale Mono" | ||||
|          id="path27279-0-5" | ||||
|          inkscape:connector-curvature="0" | ||||
|          d="m 44.226475,764.17222 1,0 c 0.01037,-1.2e-4 0.02079,-4.6e-4 0.03125,0 0.254951,0.0112 0.50987,0.12858 0.6875,0.3125 l 2.28125,2.28125 2.3125,-2.28125 c 0.265625,-0.2305 0.446672,-0.3055 0.6875,-0.3125 l 1,0 0,1 c 0,0.28647 -0.03434,0.55065 -0.25,0.75 l -2.28125,2.28125 2.25,2.25 c 0.188188,0.18817 0.281242,0.45345 0.28125,0.71875 l 0,1 -1,0 c -0.265301,-1e-5 -0.530586,-0.0931 -0.71875,-0.28125 l -2.28125,-2.28125 -2.28125,2.28125 c -0.188164,0.18819 -0.45346,0.28125 -0.71875,0.28125 l -1,0 0,-1 c -3e-6,-0.26529 0.09306,-0.53058 0.28125,-0.71875 l 2.28125,-2.25 -2.28125,-2.28125 c -0.210742,-0.19463 -0.30316,-0.46925 -0.28125,-0.75 l 0,-1 z" /></g></g></g></svg> | ||||
| Before Width: | Height: | Size: 9.4 KiB | 
| Before Width: | Height: | Size: 2.4 KiB | 
| Before Width: | Height: | Size: 2.3 KiB | 
| @@ -1,84 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    width="76" | ||||
|    height="27" | ||||
|    id="svg11252" | ||||
|    version="1.1"> | ||||
|   <defs | ||||
|      id="defs11254"> | ||||
|     <radialGradient | ||||
|        xlink:href="#linearGradient39563-4-2" | ||||
|        id="radialGradient68155-2-3" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1,0,0,0.3486842,0,317.8421)" | ||||
|        cx="49" | ||||
|        cy="488" | ||||
|        fx="49" | ||||
|        fy="488" | ||||
|        r="38" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient39563-4-2"> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop39565-1-4" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" | ||||
|          offset="1" | ||||
|          id="stop39567-7-9" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        xlink:href="#linearGradient39573-6-1" | ||||
|        id="radialGradient68157-0-8" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        cx="50.5" | ||||
|        cy="487.5" | ||||
|        fx="50.5" | ||||
|        fy="487.5" | ||||
|        r="10.5" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient39573-6-1"> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop39575-5-6" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" | ||||
|          offset="1" | ||||
|          id="stop39577-1-2" /> | ||||
|     </linearGradient> | ||||
|   </defs> | ||||
|   <g | ||||
|      id="layer1" | ||||
|      transform="translate(-337,-518.86218)"> | ||||
|     <g | ||||
|        id="g99967" | ||||
|        style="display:inline" | ||||
|        transform="translate(326,44.862171)"> | ||||
|       <rect | ||||
|          style="opacity:0.49375;color:#000000;fill:url(#radialGradient68155-2-3);fill-opacity:1;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          id="rect99969" | ||||
|          width="76" | ||||
|          height="2" | ||||
|          x="11" | ||||
|          y="487" | ||||
|          rx="0" | ||||
|          ry="0" /> | ||||
|       <path | ||||
|          style="opacity:0.43125;color:#000000;fill:url(#radialGradient68157-0-8);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          id="path99971" | ||||
|          d="M 61,487.5 C 61,493.29899 56.29899,498 50.5,498 44.70101,498 40,493.29899 40,487.5 40,481.70101 44.70101,477 50.5,477 c 5.79899,0 10.5,4.70101 10.5,10.5 z" | ||||
|          transform="matrix(1.2857143,0,0,1.2857143,-14.428572,-139.28571)" /> | ||||
|       <path | ||||
|          transform="matrix(0.43589747,0,0,0.43589747,28.487179,275)" | ||||
|          d="M 61,487.5 C 61,493.29899 56.29899,498 50.5,498 44.70101,498 40,493.29899 40,487.5 40,481.70101 44.70101,477 50.5,477 c 5.79899,0 10.5,4.70101 10.5,10.5 z" | ||||
|          id="path99973" | ||||
|          style="color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.9 KiB | 
| @@ -1,81 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="10" | ||||
|    height="20" | ||||
|    id="svg10003" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="filter-selected.svg"> | ||||
|   <defs | ||||
|      id="defs10005"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 32 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="64 : 32 : 1" | ||||
|        inkscape:persp3d-origin="32 : 21.333333 : 1" | ||||
|        id="perspective10011" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective9998" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="5.5" | ||||
|      inkscape:cx="32" | ||||
|      inkscape:cy="10.181818" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="994" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10008"> | ||||
|     <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,-44)"> | ||||
|     <path | ||||
|        inkscape:export-ydpi="90" | ||||
|        inkscape:export-xdpi="90" | ||||
|        inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/app-picker.png" | ||||
|        sodipodi:nodetypes="cccc" | ||||
|        inkscape:connector-curvature="0" | ||||
|        id="rect34320" | ||||
|        d="m -0.18726572,54.181804 10.55634072,10.55636 10e-6,-21.11269 z" | ||||
|        style="opacity:0.21000001;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.99999988;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.6 KiB | 
| @@ -1,81 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="10" | ||||
|    height="20" | ||||
|    id="svg10003" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="filter-selected-ltr.svg"> | ||||
|   <defs | ||||
|      id="defs10005"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 32 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="64 : 32 : 1" | ||||
|        inkscape:persp3d-origin="32 : 21.333333 : 1" | ||||
|        id="perspective10011" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective9998" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="5.5" | ||||
|      inkscape:cx="32.363636" | ||||
|      inkscape:cy="10.181818" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="839" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10008"> | ||||
|     <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 | ||||
|      id="layer1" | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      transform="translate(0,-44)"> | ||||
|     <path | ||||
|        inkscape:export-ydpi="90" | ||||
|        inkscape:export-xdpi="90" | ||||
|        inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/app-picker.png" | ||||
|        sodipodi:nodetypes="cccc" | ||||
|        inkscape:connector-curvature="0" | ||||
|        id="rect34320" | ||||
|        d="m 10.369085,54.181804 -10.55634072,10.55636 -1e-5,-21.11269 z" | ||||
|        style="opacity:0.21000001;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.99999988;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.7 KiB | 
| @@ -1,113 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="24" | ||||
|    height="16" | ||||
|    id="svg6503" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47pre4 r22446" | ||||
|    sodipodi:docname="mosaic-view-active.svg"> | ||||
|   <defs | ||||
|      id="defs6505"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 16 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="32 : 16 : 1" | ||||
|        inkscape:persp3d-origin="16 : 10.666667 : 1" | ||||
|        id="perspective6511" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective6494" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="11.197802" | ||||
|      inkscape:cx="-15.97056" | ||||
|      inkscape:cy="16" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata6508"> | ||||
|     <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,-16)"> | ||||
|     <g | ||||
|        style="display:inline;fill:#cbcbcb;fill-opacity:1" | ||||
|        transform="translate(-449.85476,-685.85869)" | ||||
|        id="g5306"> | ||||
|       <rect | ||||
|          style="fill:#cbcbcb;fill-opacity:1;stroke:#000000;stroke-width:0.99999970000000005;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.44262299999999999;stroke-dasharray:none" | ||||
|          id="rect5308" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="450.5" | ||||
|          y="710.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#cbcbcb;fill-opacity:1;stroke:#000000;stroke-width:0.99999970000000005;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.44262299999999999;stroke-dasharray:none;display:inline" | ||||
|          id="rect5310" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="462.5" | ||||
|          y="702.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#cbcbcb;fill-opacity:1;stroke:#000000;stroke-width:0.99999976000000002;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.44262299999999999;stroke-dasharray:none;display:inline" | ||||
|          id="rect5312" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="450.5" | ||||
|          y="702.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#cbcbcb;fill-opacity:1;stroke:#000000;stroke-width:0.99999970000000005;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.44262299999999999;stroke-dasharray:none;display:inline" | ||||
|          id="rect5314" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="462.5" | ||||
|          y="710.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.7 KiB | 
| @@ -1,113 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="24" | ||||
|    height="16" | ||||
|    id="svg6503" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47pre4 r22446" | ||||
|    sodipodi:docname="New document 19"> | ||||
|   <defs | ||||
|      id="defs6505"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 16 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="32 : 16 : 1" | ||||
|        inkscape:persp3d-origin="16 : 10.666667 : 1" | ||||
|        id="perspective6511" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective6494" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="11.197802" | ||||
|      inkscape:cx="16" | ||||
|      inkscape:cy="16" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata6508"> | ||||
|     <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 | ||||
|      id="layer1" | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      transform="translate(0,-16)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        transform="translate(-449.85476,-685.85869)" | ||||
|        id="g5306"> | ||||
|       <rect | ||||
|          style="fill:#666666;fill-opacity:1;stroke:#000000;stroke-width:0.9999997;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.442623;stroke-dasharray:none" | ||||
|          id="rect5308" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="450.5" | ||||
|          y="710.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#666666;fill-opacity:1;stroke:#000000;stroke-width:0.9999997;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.442623;stroke-dasharray:none;display:inline" | ||||
|          id="rect5310" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="462.5" | ||||
|          y="702.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#666666;fill-opacity:1;stroke:#000000;stroke-width:0.99999976;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.442623;stroke-dasharray:none;display:inline" | ||||
|          id="rect5312" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="450.5" | ||||
|          y="702.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|       <rect | ||||
|          style="fill:#666666;fill-opacity:1;stroke:#000000;stroke-width:0.9999997;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:0.442623;stroke-dasharray:none;display:inline" | ||||
|          id="rect5314" | ||||
|          width="11" | ||||
|          height="7" | ||||
|          x="462.5" | ||||
|          y="710.5" | ||||
|          rx="0.99999958" | ||||
|          ry="1" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.6 KiB | 
| @@ -1,89 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="98" | ||||
|    height="98" | ||||
|    id="svg6375" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="add-workspace.svg"> | ||||
|   <defs | ||||
|      id="defs6377"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 16 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="32 : 16 : 1" | ||||
|        inkscape:persp3d-origin="16 : 10.666667 : 1" | ||||
|        id="perspective6383" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective6366" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="3.9590209" | ||||
|      inkscape:cx="56.650687" | ||||
|      inkscape:cy="20.635343" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata6380"> | ||||
|     <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 | ||||
|      id="layer1" | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      transform="translate(0,66)"> | ||||
|     <g | ||||
|        id="g2824" | ||||
|        transform="matrix(11.568551,0,0,11.698271,-78.828159,-304.81518)"> | ||||
|       <path | ||||
|          style="fill:none;stroke:#666666;stroke-width:1.99999952;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|          d="m 11.07363,21.36834 0,6.43903" | ||||
|          id="path5322" /> | ||||
|       <path | ||||
|          style="fill:none;stroke:#666666;stroke-width:1.99999952;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" | ||||
|          d="m 14.29314,24.58786 -6.43902,0" | ||||
|          id="path5324" /> | ||||
|     </g> | ||||
|     <path | ||||
|        style="fill:#000000;fill-opacity:0.98823529" | ||||
|        d="m 48.239516,97.908047 c -0.41677,-0.05102 -1.269253,-0.222408 -1.894408,-0.380859 -4.088493,-1.036262 -7.520781,-4.753234 -8.330163,-9.021094 -0.154947,-0.817026 -0.257819,-6.68112 -0.257819,-14.696556 l 0,-13.337088 -13.829177,-0.08909 C 10.802042,60.298796 10.026884,60.268266 8.6851548,59.783022 3.6288503,57.954375 0.62673331,53.828648 0.62673331,48.708554 c 0,-5.625522 4.25936019,-10.425065 9.97721469,-11.242548 0.987903,-0.141242 7.368912,-0.254994 14.460646,-0.257791 l 12.692532,-0.005 0,-13.586668 c 0,-14.6441583 0.03287,-15.0698926 1.364686,-17.6753047 2.185477,-4.2754229 6.938193,-6.75739913 11.687647,-6.10355607 3.382776,0.46569661 6.737962,2.72496967 8.414081,5.66577137 1.480816,2.5981315 1.519067,3.0522448 1.519067,18.0333334 l 0,13.666424 12.692533,0.005 c 7.091733,0.0028 13.472742,0.116549 14.460646,0.257791 6.395303,0.914337 10.804785,6.623716 9.941157,12.871766 -0.698243,5.051565 -4.792685,9.104635 -9.941157,9.840713 -0.987904,0.141242 -7.368913,0.254995 -14.460646,0.257791 l -12.692533,0.005 0,13.801945 c 0,13.031417 -0.02798,13.895893 -0.501177,15.484801 -1.526902,5.127058 -6.919246,8.802262 -12.001914,8.18002 z" | ||||
|        id="path2828" | ||||
|        transform="translate(0,-66)" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 4.0 KiB | 
| @@ -1,33 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    width="3" | ||||
|    height="10" | ||||
|    id="svg2" | ||||
|    version="1.1"> | ||||
|   <defs | ||||
|      id="defs4" /> | ||||
|   <metadata | ||||
|      id="metadata7"> | ||||
|   </metadata> | ||||
|   <g | ||||
|      id="layer1"> | ||||
|     <rect | ||||
|        style="fill:#000000;fill-opacity:1;stroke-width:0.43599999000000000;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect3779" | ||||
|        width="3" | ||||
|        height="10" | ||||
|        x="0" | ||||
|        y="0" /> | ||||
|     <rect | ||||
|        style="fill:#536272;fill-opacity:1;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect3796" | ||||
|        width="3" | ||||
|        height="1" | ||||
|        x="0" | ||||
|        y="9" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 787 B | 
| @@ -1,74 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="21" | ||||
|    height="10" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="panel-button-border.svg"> | ||||
|   <defs | ||||
|      id="defs4" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="44.8" | ||||
|      inkscape:cx="8.6594891" | ||||
|      inkscape:cy="5.7029946" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      showguides="true" | ||||
|      inkscape:guide-bbox="true" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" | ||||
|      guidetolerance="10000" | ||||
|      objecttolerance="10000"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid3792" | ||||
|        empspacing="10" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <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="opacity:0.8;fill:#ffffff;fill-opacity:1;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect3796" | ||||
|        width="3" | ||||
|        height="2" | ||||
|        x="9" | ||||
|        y="8" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.0 KiB | 
| @@ -1,111 +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="30" | ||||
|    height="25" | ||||
|    id="svg10621" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="panel-button-highlight-narrow.svg"> | ||||
|   <defs | ||||
|      id="defs10623"> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient99561-1" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient34508-1-3"> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop34510-1-9" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" | ||||
|          offset="1" | ||||
|          id="stop34512-4-5" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        r="42" | ||||
|        fy="30" | ||||
|        fx="51" | ||||
|        cy="30" | ||||
|        cx="51" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="radialGradient10592" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        inkscape:collect="always" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1.979899" | ||||
|      inkscape:cx="-171.36384" | ||||
|      inkscape:cy="-53.255157" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      fit-margin-top="0" | ||||
|      fit-margin-left="0" | ||||
|      fit-margin-right="0" | ||||
|      fit-margin-bottom="0" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10626"> | ||||
|     <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(-468.08632,-537.03477)"> | ||||
|     <path | ||||
|        sodipodi:type="arc" | ||||
|        style="opacity:0.4625;color:#000000;fill:url(#radialGradient10592);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 C 9.0000011,21.163443 27.804042,14 51.000002,14 74.195961,14 93,21.163444 93,30 l -42,0 z" | ||||
|        sodipodi:start="3.1415927" | ||||
|        sodipodi:end="6.2831853" | ||||
|        transform="matrix(0.35714286,0,0,1.5625,464.87203,515.15977)" | ||||
|        inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/motion/textures/panel.png" | ||||
|        inkscape:export-xdpi="90" | ||||
|        inkscape:export-ydpi="90" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.5 KiB | 
| @@ -1,111 +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="84" | ||||
|    height="25" | ||||
|    id="svg10621" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.0 r9654" | ||||
|    sodipodi:docname="panel-button-highlight-wide.svg"> | ||||
|   <defs | ||||
|      id="defs10623"> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        id="radialGradient99561-1" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        cx="51" | ||||
|        cy="30" | ||||
|        fx="51" | ||||
|        fy="30" | ||||
|        r="42" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient34508-1-3"> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:1;" | ||||
|          offset="0" | ||||
|          id="stop34510-1-9" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0;" | ||||
|          offset="1" | ||||
|          id="stop34512-4-5" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        r="42" | ||||
|        fy="30" | ||||
|        fx="51" | ||||
|        cy="30" | ||||
|        cx="51" | ||||
|        gradientTransform="matrix(0.72146227,0,0,0.27484277,14.205424,21.754717)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="radialGradient10592" | ||||
|        xlink:href="#linearGradient34508-1-3" | ||||
|        inkscape:collect="always" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1.979899" | ||||
|      inkscape:cx="-118.50071" | ||||
|      inkscape:cy="27.304508" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      fit-margin-top="0" | ||||
|      fit-margin-left="0" | ||||
|      fit-margin-right="0" | ||||
|      fit-margin-bottom="0" | ||||
|      inkscape:window-width="1440" | ||||
|      inkscape:window-height="843" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata10626"> | ||||
|     <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(-441.08632,-537.03477)"> | ||||
|     <path | ||||
|        sodipodi:type="arc" | ||||
|        style="opacity:0.4625;color:#000000;fill:url(#radialGradient10592);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 C 9.0000011,21.163443 27.804042,14 51.000002,14 74.195961,14 93,21.163444 93,30 l -42,0 z" | ||||
|        sodipodi:start="3.1415927" | ||||
|        sodipodi:end="6.2831853" | ||||
|        transform="matrix(1,0,0,1.5625,432.08632,515.15977)" | ||||
|        inkscape:export-filename="/home/jimmac/src/cvs/gnome/gnome-shell-design/mockups/motion/textures/panel.png" | ||||
|        inkscape:export-xdpi="90" | ||||
|        inkscape:export-ydpi="90" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.5 KiB | 
| @@ -1,261 +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" | ||||
|    id="svg5369" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48+devel r10053 custom" | ||||
|    width="96" | ||||
|    height="48" | ||||
|    sodipodi:docname="process-working.svg" | ||||
|    style="display:inline"> | ||||
|   <metadata | ||||
|      id="metadata5375"> | ||||
|     <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> | ||||
|   <defs | ||||
|      id="defs5373" /> | ||||
|   <sodipodi:namedview | ||||
|      pagecolor="#808080" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1" | ||||
|      objecttolerance="10" | ||||
|      gridtolerance="10" | ||||
|      guidetolerance="10" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:window-width="1975" | ||||
|      inkscape:window-height="1098" | ||||
|      id="namedview5371" | ||||
|      showgrid="true" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:zoom="16" | ||||
|      inkscape:cx="53.997662" | ||||
|      inkscape:cy="22.367695" | ||||
|      inkscape:window-x="1600" | ||||
|      inkscape:window-y="33" | ||||
|      inkscape:window-maximized="0" | ||||
|      inkscape:current-layer="layer2"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid11933" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <g | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      inkscape:label="tiles" | ||||
|      style="display:none"> | ||||
|     <rect | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect12451" | ||||
|        width="24" | ||||
|        height="24" | ||||
|        x="0" | ||||
|        y="0" /> | ||||
|     <rect | ||||
|        y="24" | ||||
|        x="0" | ||||
|        height="24" | ||||
|        width="24" | ||||
|        id="rect12453" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|     <rect | ||||
|        y="0" | ||||
|        x="24" | ||||
|        height="24" | ||||
|        width="24" | ||||
|        id="rect12455" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|     <rect | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect12457" | ||||
|        width="24" | ||||
|        height="24" | ||||
|        x="24" | ||||
|        y="24" /> | ||||
|     <rect | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect12459" | ||||
|        width="24" | ||||
|        height="24" | ||||
|        x="48" | ||||
|        y="0" /> | ||||
|     <rect | ||||
|        y="24" | ||||
|        x="48" | ||||
|        height="24" | ||||
|        width="24" | ||||
|        id="rect12461" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|     <rect | ||||
|        y="0" | ||||
|        x="72" | ||||
|        height="24" | ||||
|        width="24" | ||||
|        id="rect12463" | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" /> | ||||
|     <rect | ||||
|        style="color:#000000;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect12465" | ||||
|        width="24" | ||||
|        height="24" | ||||
|        x="72" | ||||
|        y="24" /> | ||||
|   </g> | ||||
|   <g | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer2" | ||||
|      inkscape:label="spinner"> | ||||
|     <g | ||||
|        transform="matrix(0.28240106,0,0,0.28240106,146.92015,-382.52444)" | ||||
|        id="g10450-5" | ||||
|        style="display:inline"> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          style="opacity:0.6;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          d="m -477.76072,1373.3569 0,9.4717" | ||||
|          id="path18768" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          inkscape:transform-center-y="-4.6808838" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          inkscape:transform-center-y="-3.3099227" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          id="path18770" | ||||
|          d="m -461.0171,1380.2922 -7.23427,7.3824" | ||||
|          style="opacity:0.7;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          inkscape:transform-center-x="-3.3098966" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          inkscape:transform-center-x="-4.6808962" | ||||
|          style="opacity:0.8;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          d="m -454.08163,1397.0359 -9.47165,0" | ||||
|          id="path18772" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          inkscape:transform-center-y="-2.6596956e-05" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          id="path18774" | ||||
|          d="m -461.01709,1413.7796 -6.93831,-7.0864" | ||||
|          style="opacity:0.9;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          inkscape:transform-center-x="-3.3098966" | ||||
|          inkscape:transform-center-y="3.3098652" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          inkscape:transform-center-y="4.6808757" | ||||
|          style="color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          d="m -477.76074,1420.715 9e-5,-9.4716" | ||||
|          id="path18776" | ||||
|          sodipodi:nodetypes="cc" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          id="path18778" | ||||
|          d="m -494.50442,1413.7796 6.79048,-6.9384" | ||||
|          style="opacity:0.3;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          inkscape:transform-center-y="3.3098769" | ||||
|          inkscape:transform-center-x="3.3098883" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          inkscape:transform-center-x="4.6808941" | ||||
|          style="opacity:0.4;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          d="m -501.43987,1397.0359 9.47174,0" | ||||
|          id="path18780" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          inkscape:transform-center-y="-2.6596956e-05" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          sodipodi:nodetypes="cc" | ||||
|          id="path18782" | ||||
|          d="m -494.5044,1380.2922 6.64243,6.9384" | ||||
|          style="opacity:0.5;color:#000000;fill:none;stroke:#ffffff;stroke-width:7.08212566;stroke-linecap:round;stroke-opacity:1;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          inkscape:transform-center-x="3.3098902" | ||||
|          inkscape:transform-center-y="-3.3099302" /> | ||||
|     </g> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#g10450-5" | ||||
|        id="use4981" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,36,-4.9705636)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4981" | ||||
|        id="use4983" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,43.032478,-21.909695)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4983" | ||||
|        id="use4985" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,50.081986,-38.904617)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4985" | ||||
|        id="use4987" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,-38.919996,-31.872139)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4987" | ||||
|        id="use4989" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,52.986628,2.0890543)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4989" | ||||
|        id="use4991" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,60.013026,-14.912936)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|     <use | ||||
|        style="display:inline" | ||||
|        x="0" | ||||
|        y="0" | ||||
|        xlink:href="#use4991" | ||||
|        id="use4993" | ||||
|        transform="matrix(0.70710678,0.70710678,-0.70710678,0.70710678,67.022396,-31.859127)" | ||||
|        width="400" | ||||
|        height="400" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 9.8 KiB | 
| @@ -1,90 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|    width="119.97824" | ||||
|    height="119.97824" | ||||
|    id="svg7355" | ||||
|    version="1.1"> | ||||
|   <defs | ||||
|      id="defs7357"> | ||||
|     <radialGradient | ||||
|        xlink:href="#linearGradient36429" | ||||
|        id="radialGradient7461" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0525552,0,0,1.0525552,-2.5162753,-9.0000838)" | ||||
|        cx="47.878681" | ||||
|        cy="171.25" | ||||
|        fx="47.878681" | ||||
|        fy="171.25" | ||||
|        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(1.1891549,0,0,0.55513246,-9.281289,36.12653)" | ||||
|        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.5996203" | ||||
|          rx="3.5996203" | ||||
|          y="98" | ||||
|          x="11" | ||||
|          height="74" | ||||
|          width="74" | ||||
|          id="rect14000" | ||||
|          style="opacity:0.371875;fill:url(#radialGradient7461);fill-opacity:1;stroke:none" /> | ||||
|       <path | ||||
|          id="rect34520" | ||||
|          d="m 84.506708,167.95508 c 6e-6,1.96759 -1.584022,3.55162 -3.551629,3.55163 l -65.910146,0 c -1.967608,-1e-5 -3.551648,-1.58402 -3.551643,-3.55164" | ||||
|          style="opacity:0.2;fill:none;stroke:url(#radialGradient7488);stroke-width:1;stroke-opacity:1" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.7 KiB | 
| @@ -1,64 +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="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> | ||||
| Before Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								data/theme/scroll-vhandle.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 323 B | 
| @@ -1,62 +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="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> | ||||
| Before Width: | Height: | Size: 1.6 KiB | 
| @@ -1,87 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="5.8600588" | ||||
|    height="9" | ||||
|    id="svg3647" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="section-more.svg"> | ||||
|   <defs | ||||
|      id="defs3649"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3655" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3603" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="82.777778" | ||||
|      inkscape:cx="2.9300294" | ||||
|      inkscape:cy="5.466443" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata3652"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-262.78425,-490.71933)"> | ||||
|     <path | ||||
|        transform="matrix(0,-0.98149546,0.71467449,0,25.404986,578.15569)" | ||||
|        d="M 88.830127,340 80.169873,340 84.5,332.5 88.830127,340 z" | ||||
|        inkscape:randomized="0" | ||||
|        inkscape:rounded="0" | ||||
|        inkscape:flatsided="true" | ||||
|        sodipodi:arg2="1.5707963" | ||||
|        sodipodi:arg1="0.52359878" | ||||
|        sodipodi:r2="2.5" | ||||
|        sodipodi:r1="5" | ||||
|        sodipodi:cy="337.5" | ||||
|        sodipodi:cx="84.5" | ||||
|        sodipodi:sides="3" | ||||
|        id="path5497-5" | ||||
|        style="fill:#ffffff;fill-opacity:1;stroke:#ffffff;stroke-width:0.59699643;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" | ||||
|        sodipodi:type="star" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.8 KiB | 
| @@ -1,87 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="5.8600588" | ||||
|    height="9" | ||||
|    id="svg3647" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.46+devel" | ||||
|    sodipodi:docname="New document 6"> | ||||
|   <defs | ||||
|      id="defs3649"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3655" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3603" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="112.21575" | ||||
|      inkscape:cy="-32.642856" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-x="164" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata3652"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-262.78425,-490.71933)"> | ||||
|     <path | ||||
|        transform="matrix(0,0.98149546,-0.71467449,0,506.02358,412.28296)" | ||||
|        d="M 88.830127,340 80.169873,340 84.5,332.5 88.830127,340 z" | ||||
|        inkscape:randomized="0" | ||||
|        inkscape:rounded="0" | ||||
|        inkscape:flatsided="true" | ||||
|        sodipodi:arg2="1.5707963" | ||||
|        sodipodi:arg1="0.52359878" | ||||
|        sodipodi:r2="2.5" | ||||
|        sodipodi:r1="5" | ||||
|        sodipodi:cy="337.5" | ||||
|        sodipodi:cx="84.5" | ||||
|        sodipodi:sides="3" | ||||
|        id="path5497-5" | ||||
|        style="fill:#5f5f5f;fill-opacity:1;stroke:#5f5f5f;stroke-width:0.59699643;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" | ||||
|        sodipodi:type="star" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.8 KiB | 
| Before Width: | Height: | Size: 531 B | 
| @@ -1,81 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="24" | ||||
|    height="16" | ||||
|    id="svg6446" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47pre4 r22446" | ||||
|    sodipodi:docname="single-view-active.svg"> | ||||
|   <defs | ||||
|      id="defs6448"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 16 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="32 : 16 : 1" | ||||
|        inkscape:persp3d-origin="16 : 10.666667 : 1" | ||||
|        id="perspective6454" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective6441" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="11.197802" | ||||
|      inkscape:cx="0.014720032" | ||||
|      inkscape:cy="16" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata6451"> | ||||
|     <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,-17)"> | ||||
|     <rect | ||||
|        ry="0.5" | ||||
|        rx="0.49999979" | ||||
|        y="17.483809" | ||||
|        x="0.53483802" | ||||
|        height="15" | ||||
|        width="23" | ||||
|        id="rect5304" | ||||
|        style="fill:#cccccc;fill-opacity:1;stroke:#cccccc;stroke-width:1;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.4 KiB | 
| @@ -1,81 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="24" | ||||
|    height="16" | ||||
|    id="svg6446" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47pre4 r22446" | ||||
|    sodipodi:docname="single-view.svg"> | ||||
|   <defs | ||||
|      id="defs6448"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 16 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="32 : 16 : 1" | ||||
|        inkscape:persp3d-origin="16 : 10.666667 : 1" | ||||
|        id="perspective6454" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective6441" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="11.197802" | ||||
|      inkscape:cx="0.014720032" | ||||
|      inkscape:cy="16" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:window-width="1680" | ||||
|      inkscape:window-height="997" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" /> | ||||
|   <metadata | ||||
|      id="metadata6451"> | ||||
|     <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,-17)"> | ||||
|     <rect | ||||
|        ry="0.5" | ||||
|        rx="0.49999979" | ||||
|        y="17.483809" | ||||
|        x="0.53483802" | ||||
|        height="15" | ||||
|        width="23" | ||||
|        id="rect5304" | ||||
|        style="fill:#626262;fill-opacity:1;stroke:#cccccc;stroke-width:1;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;display:inline" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.4 KiB | 
| @@ -1,74 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="21" | ||||
|    height="10" | ||||
|    id="svg2" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.0 r9654" | ||||
|    sodipodi:docname="source-button-border.svg"> | ||||
|   <defs | ||||
|      id="defs4" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="44.8" | ||||
|      inkscape:cx="8.704132" | ||||
|      inkscape:cy="5.7029946" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      showguides="true" | ||||
|      inkscape:guide-bbox="true" | ||||
|      inkscape:window-width="1600" | ||||
|      inkscape:window-height="1145" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="1" | ||||
|      guidetolerance="10000" | ||||
|      objecttolerance="10000"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid3792" | ||||
|        empspacing="10" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <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="opacity:0.8;fill:#ffffff;fill-opacity:1;stroke-width:0.43599999;stroke-miterlimit:4;stroke-dasharray:none" | ||||
|        id="rect3796" | ||||
|        width="19" | ||||
|        height="2" | ||||
|        x="1" | ||||
|        y="8" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.0 KiB | 
| @@ -1,126 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="64" | ||||
|    height="22" | ||||
|    id="svg3273" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 14"> | ||||
|   <defs | ||||
|      id="defs3275"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3281" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3261" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="32.000004" | ||||
|      inkscape:cy="10.999997" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata3278"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-343,-521.36218)"> | ||||
|     <g | ||||
|        id="g17454" | ||||
|        transform="translate(-453,448.36218)" | ||||
|        style="display:inline"> | ||||
|       <rect | ||||
|          transform="scale(-1,1)" | ||||
|          ry="4" | ||||
|          rx="4" | ||||
|          y="74.5" | ||||
|          x="-859.5" | ||||
|          height="19" | ||||
|          width="63.000004" | ||||
|          id="rect17456" | ||||
|          style="color:#000000;fill:none;stroke:#2e3436;stroke-width:0.99999994;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;enable-background:accumulate" /> | ||||
|       <rect | ||||
|          transform="scale(-1,1)" | ||||
|          ry="4" | ||||
|          rx="4" | ||||
|          y="74" | ||||
|          x="-828" | ||||
|          height="20" | ||||
|          width="31" | ||||
|          id="rect17458" | ||||
|          style="fill:#000000;fill-opacity:1;stroke:#5f5f5f;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" /> | ||||
|       <g | ||||
|          transform="matrix(-1,0,0,1,1619.1239,-33.986291)" | ||||
|          id="g17460" | ||||
|          style="display:inline"> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1" | ||||
|            d="m 803.6322,115.48629 0,4.29495" | ||||
|            id="path17462" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" | ||||
|            d="m 806.62805,115.48629 0,4.29495" | ||||
|            id="path17464" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" | ||||
|            d="m 809.6239,115.48629 0,4.29495" | ||||
|            id="path17466" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|       </g> | ||||
|       <path | ||||
|          sodipodi:type="arc" | ||||
|          style="color:#000000;fill:none;stroke:#ffffff;stroke-width:1.96875012;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;enable-background:accumulate" | ||||
|          id="path18722" | ||||
|          sodipodi:cx="47.6875" | ||||
|          sodipodi:cy="11.5625" | ||||
|          sodipodi:rx="3.9375" | ||||
|          sodipodi:ry="3.9375" | ||||
|          d="m 51.625,11.5625 c 0,2.174621 -1.762879,3.9375 -3.9375,3.9375 -2.174621,0 -3.9375,-1.762879 -3.9375,-3.9375 0,-2.1746212 1.762879,-3.9375 3.9375,-3.9375 2.174621,0 3.9375,1.7628788 3.9375,3.9375 z" | ||||
|          transform="matrix(1.0158729,0,0,1.0158729,795.55556,72.25399)" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 4.7 KiB | 
| @@ -1,138 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="64" | ||||
|    height="22" | ||||
|    id="svg3012" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 6"> | ||||
|   <defs | ||||
|      id="defs3014"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3020" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective2997" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="32.000004" | ||||
|      inkscape:cy="10.999997" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata3017"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-343,-521.36218)"> | ||||
|     <g | ||||
|        id="g17454" | ||||
|        transform="translate(-453,448.36218)" | ||||
|        style="display:inline"> | ||||
|       <rect | ||||
|          transform="scale(-1,1)" | ||||
|          ry="4" | ||||
|          rx="4" | ||||
|          y="74.5" | ||||
|          x="-859.5" | ||||
|          height="19" | ||||
|          width="63.000004" | ||||
|          id="rect17456" | ||||
|          style="color:#000000;fill:none;stroke:#2e3436;stroke-width:0.99999994;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;enable-background:accumulate" /> | ||||
|       <rect | ||||
|          transform="scale(-1,1)" | ||||
|          ry="4" | ||||
|          rx="4" | ||||
|          y="74" | ||||
|          x="-828" | ||||
|          height="20" | ||||
|          width="31" | ||||
|          id="rect17458" | ||||
|          style="fill:#000000;fill-opacity:1;stroke:#5f5f5f;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" /> | ||||
|       <g | ||||
|          transform="matrix(-1,0,0,1,1619.1239,-33.986291)" | ||||
|          id="g17460" | ||||
|          style="display:inline"> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1" | ||||
|            d="m 803.6322,115.48629 0,4.29495" | ||||
|            id="path17462" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" | ||||
|            d="m 806.62805,115.48629 0,4.29495" | ||||
|            id="path17464" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" | ||||
|            d="m 809.6239,115.48629 0,4.29495" | ||||
|            id="path17466" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|       </g> | ||||
|       <g | ||||
|          style="font-size:8.95877075px;font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;text-indent:0;text-align:start;text-decoration:none;line-height:125%;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;block-progression:tb;writing-mode:lr-tb;text-anchor:start;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1px;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|          id="text17468" | ||||
|          transform="translate(0.34375,0)"> | ||||
|         <path | ||||
|            d="m 837.28518,80.750726 c 0.63282,6e-6 1.19566,0.123947 1.68852,0.371824 0.49284,0.247888 0.8807,0.609505 1.16359,1.084851 0.28287,0.472439 0.42431,1.022155 0.42431,1.649149 0,0.635748 -0.13853,1.200045 -0.41556,1.692892 -0.27706,0.489934 -0.66638,0.870507 -1.16797,1.141719 -0.5016,0.271213 -1.07756,0.406819 -1.72789,0.406819 -0.42869,0 -0.83551,-0.06562 -1.22045,-0.196848 -0.38495,-0.134148 -0.73053,-0.32808 -1.03674,-0.581795 -0.30329,-0.256631 -0.54534,-0.589085 -0.72615,-0.997363 -0.17789,-0.408276 -0.26684,-0.869045 -0.26683,-1.382311 -10e-6,-0.638658 0.13997,-1.200039 0.41994,-1.684144 0.27996,-0.487011 0.66782,-0.858835 1.16359,-1.115472 0.49576,-0.259541 1.06297,-0.389315 1.70164,-0.389321 m 0.57305,1.089225 c -0.20123,-0.05249 -0.40683,-0.07873 -0.61679,-0.07874 -0.20998,5e-6 -0.41412,0.02625 -0.61242,0.07874 -0.19831,0.04958 -0.38933,0.129779 -0.57305,0.240592 -0.18081,0.107907 -0.33974,0.242055 -0.47681,0.402445 -0.13706,0.160399 -0.24642,0.358705 -0.32808,0.594918 -0.0816,0.233306 -0.12248,0.491395 -0.12248,0.774269 0,0.67366 0.20851,1.214627 0.62554,1.622903 0.41702,0.408278 0.93758,0.612416 1.56166,0.612416 0.25954,0 0.51034,-0.04229 0.7524,-0.126858 0.24496,-0.08457 0.47097,-0.20997 0.67803,-0.376198 0.20705,-0.166226 0.37328,-0.392236 0.49868,-0.678032 0.12539,-0.285792 0.18809,-0.610956 0.1881,-0.975492 -10e-6,-0.297455 -0.0437,-0.568668 -0.13123,-0.813638 -0.0875,-0.247878 -0.20415,-0.453475 -0.34995,-0.61679 -0.14291,-0.163307 -0.31059,-0.301829 -0.50306,-0.415568 -0.18956,-0.11373 -0.38641,-0.195385 -0.59054,-0.244967" | ||||
|            style="line-height:125%;fill:#ffffff;fill-opacity:1;marker:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|            id="path18599" | ||||
|            inkscape:connector-curvature="0" /> | ||||
|         <path | ||||
|            d="m 843.5362,81.831203 0,1.17917 2.94834,0 0,1.014861 -2.94834,0 0,3.00713 -1.10673,0 0,-6.216022 4.31754,0 0,1.014861 -3.21081,0" | ||||
|            style="line-height:125%;fill:#ffffff;fill-opacity:1;marker:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|            id="path18601" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="ccccccccccc" /> | ||||
|         <path | ||||
|            d="m 849.71285,81.831203 0,1.17917 2.94834,0 0,1.014861 -2.94834,0 0,3.00713 -1.10672,0 0,-6.216022 4.31753,0 0,1.014861 -3.21081,0" | ||||
|            style="line-height:125%;fill:#ffffff;fill-opacity:1;marker:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|            id="path18603" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="ccccccccccc" /> | ||||
|       </g> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 7.2 KiB | 
| @@ -1,122 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="65" | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 11"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective3207" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3187" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="32.500004" | ||||
|      inkscape:cy="10.999997" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata3204"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        transform="translate(-453.5,448.36218)" | ||||
|        id="g16453"> | ||||
|       <rect | ||||
|          style="color:#000000;fill:#204a87;fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:0.99999994;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;enable-background:accumulate" | ||||
|          id="rect16256-9-4" | ||||
|          width="63.000004" | ||||
|          height="19" | ||||
|          x="-859.5" | ||||
|          y="74.5" | ||||
|          rx="4" | ||||
|          ry="4" | ||||
|          transform="scale(-1,1)" /> | ||||
|       <rect | ||||
|          style="fill:#000000;fill-opacity:1;stroke:#5f5f5f;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" | ||||
|          id="rect16258-5-4" | ||||
|          width="31" | ||||
|          height="20" | ||||
|          x="-860" | ||||
|          y="74" | ||||
|          rx="4" | ||||
|          ry="4" | ||||
|          transform="scale(-1,1)" /> | ||||
|       <g | ||||
|          style="display:inline" | ||||
|          id="g16298-3-6" | ||||
|          transform="matrix(-1,0,0,1,1651.1322,-33.986291)"> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-3-5" | ||||
|            d="m 803.6322,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-0-2-0" | ||||
|            d="m 806.62805,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-8-7-1" | ||||
|            d="m 809.6239,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" /> | ||||
|       </g> | ||||
|       <path | ||||
|          style="color:#000000;fill:#eeeeec;fill-opacity:1;fill-rule:nonzero;stroke:#ffffff;stroke-width:2;stroke-linecap:square;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" | ||||
|          d="m 16,27.9375 0,10.125" | ||||
|          id="path19232" | ||||
|          inkscape:connector-curvature="0" | ||||
|          transform="translate(796,51.00002)" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 4.5 KiB | 
| @@ -1,128 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="65" | ||||
|    height="22" | ||||
|    id="svg2857" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 2"> | ||||
|   <defs | ||||
|      id="defs2859"> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_z="744.09448 : 526.18109 : 1" | ||||
|        inkscape:persp3d-origin="372.04724 : 350.78739 : 1" | ||||
|        id="perspective2865" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective2843" | ||||
|        inkscape:persp3d-origin="0.5 : 0.33333333 : 1" | ||||
|        inkscape:vp_z="1 : 0.5 : 1" | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="-69.642856" | ||||
|      inkscape:cy="42.428569" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|   <metadata | ||||
|      id="metadata2862"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-444.64286,-781.36218)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        transform="translate(-351.35714,708.36218)" | ||||
|        id="g16453"> | ||||
|       <rect | ||||
|          style="color:#000000;fill:#204a87;fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:0.99999994;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;enable-background:accumulate" | ||||
|          id="rect16256-9-4" | ||||
|          width="63.000004" | ||||
|          height="19" | ||||
|          x="-859.5" | ||||
|          y="74.5" | ||||
|          rx="4" | ||||
|          ry="4" | ||||
|          transform="scale(-1,1)" /> | ||||
|       <rect | ||||
|          style="fill:#000000;fill-opacity:1;stroke:#5f5f5f;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" | ||||
|          id="rect16258-5-4" | ||||
|          width="31" | ||||
|          height="20" | ||||
|          x="-860" | ||||
|          y="74" | ||||
|          rx="4" | ||||
|          ry="4" | ||||
|          transform="scale(-1,1)" /> | ||||
|       <g | ||||
|          style="display:inline" | ||||
|          id="g16298-3-6" | ||||
|          transform="matrix(-1,0,0,1,1651.1322,-33.986291)"> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-3-5" | ||||
|            d="m 803.6322,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-0-2-0" | ||||
|            d="m 806.62805,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            id="path16265-8-7-1" | ||||
|            d="m 809.6239,115.48629 0,4.29495" | ||||
|            style="fill:none;stroke:#5f5f5f;stroke-width:1px;stroke-linecap:round;stroke-linejoin:miter;stroke-opacity:1;display:inline" /> | ||||
|       </g> | ||||
|       <g | ||||
|          style="font-size:8.95877075px;font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;text-indent:0;text-align:start;text-decoration:none;line-height:125%;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;block-progression:tb;writing-mode:lr-tb;text-anchor:start;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1px;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|          id="text42229-3-0"> | ||||
|         <path | ||||
|            d="m 808.01473,80.573953 c 0.63283,6e-6 1.19567,0.123947 1.68852,0.371824 0.49284,0.247888 0.88071,0.609505 1.16359,1.084851 0.28287,0.472439 0.42431,1.022155 0.42432,1.649149 -10e-6,0.635748 -0.13853,1.200045 -0.41557,1.692892 -0.27705,0.489934 -0.66637,0.870506 -1.16796,1.141719 -0.50161,0.271212 -1.07757,0.406819 -1.72789,0.406819 -0.4287,0 -0.83552,-0.06562 -1.22046,-0.196848 -0.38495,-0.134148 -0.73053,-0.32808 -1.03673,-0.581795 -0.3033,-0.256631 -0.54535,-0.589085 -0.72615,-0.997363 -0.1779,-0.408276 -0.26684,-0.869045 -0.26684,-1.382311 0,-0.638658 0.13998,-1.200039 0.41994,-1.684144 0.27996,-0.487011 0.66782,-0.858835 1.16359,-1.115472 0.49576,-0.259541 1.06298,-0.389315 1.70164,-0.389321 m 0.57305,1.089225 c -0.20123,-0.05249 -0.40682,-0.07873 -0.61679,-0.07874 -0.20998,5e-6 -0.41411,0.02625 -0.61242,0.07874 -0.19831,0.04958 -0.38932,0.129779 -0.57304,0.240592 -0.18081,0.107907 -0.33975,0.242055 -0.47681,0.402445 -0.13707,0.160399 -0.24643,0.358705 -0.32808,0.594918 -0.0817,0.233305 -0.12249,0.491395 -0.12249,0.774269 0,0.67366 0.20851,1.214627 0.62554,1.622902 0.41703,0.408279 0.93758,0.612417 1.56166,0.612416 0.25955,10e-7 0.51035,-0.04228 0.7524,-0.126857 0.24496,-0.08457 0.47097,-0.20997 0.67803,-0.376199 0.20705,-0.166225 0.37328,-0.392236 0.49868,-0.678031 0.1254,-0.285792 0.1881,-0.610956 0.1881,-0.975492 0,-0.297455 -0.0437,-0.568668 -0.13123,-0.813638 -0.0875,-0.247878 -0.20414,-0.453475 -0.34995,-0.61679 -0.1429,-0.163307 -0.31059,-0.301829 -0.50306,-0.415568 -0.18956,-0.11373 -0.38641,-0.195385 -0.59054,-0.244967" | ||||
|            style="line-height:125%;fill:#ffffff;fill-opacity:1;marker:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|            id="path18606" /> | ||||
|         <path | ||||
|            d="m 813.15903,80.639569 1.21608,0 3.4689,4.776844 0,-4.776844 1.10235,0 0,6.216022 -1.22921,0 -3.45577,-4.785594 0,4.785594 -1.10235,0 0,-6.216022" | ||||
|            style="line-height:125%;fill:#ffffff;fill-opacity:1;marker:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|            id="path18608" /> | ||||
|       </g> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 6.7 KiB | 
| @@ -1,376 +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" | ||||
|    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> | ||||
| Before Width: | Height: | Size: 13 KiB | 
| @@ -1,447 +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="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> | ||||
| Before Width: | Height: | Size: 16 KiB | 
| @@ -6,22 +6,6 @@ | ||||
|  | ||||
|   <name xml:lang="en">GNOME Shell</name> | ||||
|   <shortdesc xml:lang="en">Next generation GNOME desktop shell</shortdesc> | ||||
|   <description>GNOME Shell provides core user interface functions for the GNOME 3 | ||||
| desktop, like switching to windows and launching applications. | ||||
| GNOME Shell takes advantage of the capabilities of modern graphics | ||||
| hardware and introduces innovative user interface concepts to | ||||
| provide a visually attractive and easy to use experience. | ||||
|  | ||||
| Tarball releases are provided largely for distributions to build | ||||
| packages. If you are interested in building GNOME Shell from source, | ||||
| we would recommend building from version control using the build | ||||
| script described at: | ||||
|  | ||||
|  http://live.gnome.org/GnomeShell | ||||
|  | ||||
| Not only will that give you the very latest version of this rapidly | ||||
| changing project, it will be much easier than get GNOME Shell and | ||||
| its dependencies to build from tarballs.</description> | ||||
|   <!-- | ||||
|   <homepage rdf:resource="http://live.gnome.org/GnomeShell" /> | ||||
|   --> | ||||
| @@ -49,7 +33,7 @@ its dependencies to build from tarballs.</description> | ||||
|     <foaf:Person> | ||||
|       <foaf:name>Colin Walters</foaf:name> | ||||
|       <foaf:mbox rdf:resource="mailto:walters@verbum.org" /> | ||||
|       <gnome:userid>walters</gnome:userid> | ||||
|       <gnome:userid>cwalters</gnome:userid> | ||||
|     </foaf:Person> | ||||
|   </maintainer> | ||||
|   <maintainer> | ||||
|   | ||||
| @@ -1,67 +1 @@ | ||||
|  | ||||
| jsdir = $(pkgdatadir)/js | ||||
|  | ||||
| nobase_dist_js_DATA = 	\ | ||||
| 	misc/config.js		\ | ||||
| 	misc/docInfo.js		\ | ||||
| 	misc/fileUtils.js	\ | ||||
| 	misc/format.js		\ | ||||
| 	misc/gnomeSession.js	\ | ||||
| 	misc/history.js		\ | ||||
| 	misc/modemManager.js	\ | ||||
| 	misc/params.js		\ | ||||
| 	misc/util.js		\ | ||||
| 	perf/core.js		\ | ||||
| 	ui/altTab.js		\ | ||||
| 	ui/appDisplay.js	\ | ||||
| 	ui/appFavorites.js	\ | ||||
| 	ui/boxpointer.js	\ | ||||
| 	ui/calendar.js		\ | ||||
| 	ui/chrome.js		\ | ||||
| 	ui/ctrlAltTab.js	\ | ||||
| 	ui/dash.js		\ | ||||
| 	ui/dateMenu.js		\ | ||||
| 	ui/dnd.js		\ | ||||
| 	ui/docDisplay.js	\ | ||||
| 	ui/endSessionDialog.js	\ | ||||
| 	ui/environment.js	\ | ||||
| 	ui/extensionSystem.js	\ | ||||
| 	ui/iconGrid.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/notificationDaemon.js \ | ||||
| 	ui/overview.js		\ | ||||
| 	ui/panel.js		\ | ||||
| 	ui/panelMenu.js		\ | ||||
| 	ui/placeDisplay.js	\ | ||||
| 	ui/polkitAuthenticationAgent.js \ | ||||
| 	ui/popupMenu.js		\ | ||||
| 	ui/runDialog.js		\ | ||||
| 	ui/scripting.js		\ | ||||
| 	ui/search.js		\ | ||||
| 	ui/searchDisplay.js	\ | ||||
| 	ui/shellDBus.js		\ | ||||
| 	ui/statusIconDispatcher.js	\ | ||||
| 	ui/statusMenu.js	\ | ||||
| 	ui/status/accessibility.js	\ | ||||
| 	ui/status/keyboard.js	\ | ||||
| 	ui/status/network.js	\ | ||||
| 	ui/status/power.js	\ | ||||
| 	ui/status/volume.js	\ | ||||
| 	ui/status/bluetooth.js	\ | ||||
| 	ui/telepathyClient.js	\ | ||||
| 	ui/tweener.js		\ | ||||
| 	ui/viewSelector.js	\ | ||||
| 	ui/windowAttentionHandler.js	\ | ||||
| 	ui/windowManager.js	\ | ||||
| 	ui/workspace.js		\ | ||||
| 	ui/workspaceThumbnail.js	\ | ||||
| 	ui/workspacesView.js	\ | ||||
| 	ui/workspaceSwitcherPopup.js    \ | ||||
| 	ui/xdndHandler.js | ||||
| SUBDIRS = misc ui | ||||
|   | ||||
							
								
								
									
										4
									
								
								js/misc/Makefile.am
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| jsmiscdir = $(pkgdatadir)/js/misc | ||||
|  | ||||
| dist_jsmisc_DATA =		\ | ||||
| 	docInfo.js | ||||
| @@ -1,10 +0,0 @@ | ||||
| /* mode: js2; indent-tabs-mode: nil; tab-size: 4 */ | ||||
| /* The name of this package (not localized) */ | ||||
| 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@; | ||||
|  | ||||
| @@ -1,10 +1,13 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const St = imports.gi.St; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const Search = imports.ui.search; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const THUMBNAIL_ICON_MARGIN = 2; | ||||
|  | ||||
| @@ -14,40 +17,69 @@ function DocInfo(recentInfo) { | ||||
|  | ||||
| DocInfo.prototype = { | ||||
|     _init : function(recentInfo) { | ||||
|         this.recentInfo = recentInfo; | ||||
|         this._recentInfo = recentInfo; | ||||
|         // We actually used get_modified() instead of get_visited() | ||||
|         // here, as GtkRecentInfo doesn't updated get_visited() | ||||
|         // correctly. See http://bugzilla.gnome.org/show_bug.cgi?id=567094 | ||||
|         this.timestamp = recentInfo.get_modified(); | ||||
|         this.timestamp = recentInfo.get_modified().getTime() / 1000; | ||||
|         this.name = recentInfo.get_display_name(); | ||||
|         this._lowerName = this.name.toLowerCase(); | ||||
|         this.uri = recentInfo.get_uri(); | ||||
|         this.mimeType = recentInfo.get_mime_type(); | ||||
|     }, | ||||
|  | ||||
|     createIcon : function(size) { | ||||
|         return St.TextureCache.get_default().load_recent_thumbnail(size, this.recentInfo); | ||||
|         return Shell.TextureCache.get_default().load_recent_thumbnail(size, this._recentInfo); | ||||
|     }, | ||||
|  | ||||
|     launch : function(workspaceIndex) { | ||||
|         Shell.DocSystem.get_default().open(this.recentInfo, workspaceIndex); | ||||
|     }, | ||||
|     launch : function() { | ||||
|         // While using Gio.app_info_launch_default_for_uri() would be | ||||
|         // shorter in terms of lines of code, we are not doing so | ||||
|         // because that would duplicate the work of retrieving the | ||||
|         // mime type. | ||||
|  | ||||
|     matchTerms: function(terms) { | ||||
|         let mtype = Search.MatchType.NONE; | ||||
|         for (let i = 0; i < terms.length; i++) { | ||||
|             let term = terms[i]; | ||||
|             let idx = this._lowerName.indexOf(term); | ||||
|             if (idx == 0) { | ||||
|                 mtype = Search.MatchType.PREFIX; | ||||
|             } else if (idx > 0) { | ||||
|                 if (mtype == Search.MatchType.NONE) | ||||
|                     mtype = Search.MatchType.SUBSTRING; | ||||
|         let appInfo = Gio.app_info_get_default_for_type(this.mimeType, true); | ||||
|  | ||||
|         if (appInfo != null) { | ||||
|             appInfo.launch_uris([this.uri], Main.createAppLaunchContext()); | ||||
|         } else { | ||||
|             log("Failed to get default application info for mime type " + this.mimeType + | ||||
|                 ". Will try to use the last application that registered the document."); | ||||
|             let appName = this._recentInfo.last_application(); | ||||
|             let [success, appExec, count, time] = this._recentInfo.get_application_info(appName); | ||||
|             if (success) { | ||||
|                 log("Will open a document with the following command: " + appExec); | ||||
|                 // TODO: Change this once better support for creating | ||||
|                 // GAppInfo is added to GtkRecentInfo, as right now | ||||
|                 // this relies on the fact that the file uri is | ||||
|                 // already a part of appExec, so we don't supply any | ||||
|                 // files to appInfo.launch(). | ||||
|  | ||||
|                 // The 'command line' passed to | ||||
|                 // create_from_command_line is allowed to contain | ||||
|                 // '%<something>' macros that are expanded to file | ||||
|                 // name / icon name, etc, so we need to escape % as %% | ||||
|                 appExec = appExec.replace(/%/g, "%%"); | ||||
|  | ||||
|                 let appInfo = Gio.app_info_create_from_commandline(appExec, null, 0, null); | ||||
|  | ||||
|                 // The point of passing an app launch context to | ||||
|                 // launch() is mostly to get startup notification and | ||||
|                 // associated benefits like the app appearing on the | ||||
|                 // right desktop; but it doesn't really work for now | ||||
|                 // because with the way we create the appInfo we | ||||
|                 // aren't reading the application's desktop file, and | ||||
|                 // thus don't find the StartupNotify=true in it. So, | ||||
|                 // despite passing the app launch context, no startup | ||||
|                 // notification occurs. | ||||
|                 appInfo.launch([], Main.createAppLaunchContext()); | ||||
|             } else { | ||||
|                 return Search.MatchType.NONE; | ||||
|                 log("Failed to get application info for " + this.uri); | ||||
|             } | ||||
|         } | ||||
|         return mtype; | ||||
|     }, | ||||
|  | ||||
|     exists : function() { | ||||
|         return this._recentInfo.exists(); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| @@ -59,82 +91,51 @@ function getDocManager() { | ||||
|     return docManagerInstance; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * DocManager wraps the DocSystem, primarily to expose DocInfo objects. | ||||
|  */ | ||||
| function DocManager() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| DocManager.prototype = { | ||||
|     _init: function() { | ||||
|         this._docSystem = Shell.DocSystem.get_default(); | ||||
|         this._infosByTimestamp = []; | ||||
|         this._infosByUri = {}; | ||||
|         this._docSystem.connect('changed', Lang.bind(this, this._reload)); | ||||
|         this._recentManager = Gtk.RecentManager.get_default(); | ||||
|         this._items = {}; | ||||
|         this._recentManager.connect('changed', Lang.bind(this, function(recentManager) { | ||||
|             this._reload(); | ||||
|             this.emit('changed'); | ||||
|         })); | ||||
|         this._reload(); | ||||
|     }, | ||||
|  | ||||
|     _reload: function() { | ||||
|         let docs = this._docSystem.get_all(); | ||||
|         this._infosByTimestamp = []; | ||||
|         this._infosByUri = {}; | ||||
|         let docs = this._recentManager.get_items(); | ||||
|         let newItems = {}; | ||||
|         for (let i = 0; i < docs.length; i++) { | ||||
|             let recentInfo = docs[i]; | ||||
|             if (!recentInfo.exists()) | ||||
|                 continue; | ||||
|  | ||||
|             let docInfo = new DocInfo(recentInfo); | ||||
|             this._infosByTimestamp.push(docInfo); | ||||
|             this._infosByUri[docInfo.uri] = docInfo; | ||||
|  | ||||
|             // we use GtkRecentInfo URI as an item Id | ||||
|             newItems[docInfo.uri] = docInfo; | ||||
|         } | ||||
|         this.emit('changed'); | ||||
|         let deleted = {}; | ||||
|         for (var uri in this._items) { | ||||
|             if (!(uri in newItems)) | ||||
|                 deleted[uri] = this._items[uri]; | ||||
|         } | ||||
|         /* If we'd cached any thumbnail references that no longer exist, | ||||
|            dump them here */ | ||||
|         let texCache = Shell.TextureCache.get_default(); | ||||
|         for (var uri in deleted) { | ||||
|             texCache.evict_recent_thumbnail(this._items[uri]); | ||||
|         } | ||||
|         this._items = newItems; | ||||
|     }, | ||||
|  | ||||
|     getTimestampOrderedInfos: function() { | ||||
|         return this._infosByTimestamp; | ||||
|     }, | ||||
|  | ||||
|     getInfosByUri: function() { | ||||
|         return this._infosByUri; | ||||
|     }, | ||||
|  | ||||
|     lookupByUri: function(uri) { | ||||
|         return this._infosByUri[uri]; | ||||
|     }, | ||||
|  | ||||
|     queueExistenceCheck: function(count) { | ||||
|         return this._docSystem.queue_existence_check(count); | ||||
|     }, | ||||
|  | ||||
|     _searchDocs: function(items, terms) { | ||||
|         let multiplePrefixMatches = []; | ||||
|         let prefixMatches = []; | ||||
|         let multipleSubtringMatches = []; | ||||
|         let substringMatches = []; | ||||
|         for (let i = 0; i < items.length; i++) { | ||||
|             let item = items[i]; | ||||
|             let mtype = item.matchTerms(terms); | ||||
|             if (mtype == Search.MatchType.MULTIPLE_PREFIX) | ||||
|                 multiplePrefixMatches.push(item.uri); | ||||
|             else if (mtype == Search.MatchType.PREFIX) | ||||
|                 prefixMatches.push(item.uri); | ||||
|             else if (mtype == Search.MatchType.MULTIPLE_SUBSTRING) | ||||
|                 multipleSubtringMatches.push(item.uri); | ||||
|             else if (mtype == Search.MatchType.SUBSTRING) | ||||
|                 substringMatches.push(item.uri); | ||||
|          } | ||||
|         return multiplePrefixMatches.concat(prefixMatches.concat(multipleSubtringMatches.concat(substringMatches))); | ||||
|     }, | ||||
|  | ||||
|     initialSearch: function(terms) { | ||||
|         return this._searchDocs(this._infosByTimestamp, terms); | ||||
|     }, | ||||
|  | ||||
|     subsearch: function(previousResults, terms) { | ||||
|         return this._searchDocs(previousResults.map(Lang.bind(this, | ||||
|             function(url) { | ||||
|                 return this._infosByUri[url]; | ||||
|             })), terms); | ||||
|     getItems: function() { | ||||
|         return this._items; | ||||
|     } | ||||
| }; | ||||
| } | ||||
|  | ||||
| Signals.addSignalMethods(DocManager.prototype); | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
|  | ||||
| function listDirAsync(file, callback) { | ||||
|     let allFiles = []; | ||||
|     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); | ||||
|         function onNextFileComplete(obj, res) { | ||||
|             let files = obj.next_files_finish(res); | ||||
|             if (files.length) { | ||||
|                 allFiles = allFiles.concat(files); | ||||
|                 enumerator.next_files_async(100, GLib.PRIORITY_LOW, null, onNextFileComplete); | ||||
|             } else { | ||||
|                 enumerator.close(null); | ||||
|                 callback(allFiles); | ||||
|             } | ||||
|         } | ||||
|         enumerator.next_files_async(100, GLib.PRIORITY_LOW, null, onNextFileComplete); | ||||
|     }); | ||||
| } | ||||
| @@ -1,60 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 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); | ||||
|                 }); | ||||
| } | ||||
| @@ -1,125 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const DBus = imports.dbus; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const PresenceIface = { | ||||
|     name: 'org.gnome.SessionManager.Presence', | ||||
|     methods: [{ name: 'SetStatus', | ||||
|                 inSignature: 'u', | ||||
|                 outSignature: '' }], | ||||
|     properties: [{ name: 'status', | ||||
|                    signature: 'u', | ||||
|                    access: 'readwrite' }], | ||||
|     signals: [{ name: 'StatusChanged', | ||||
|                 inSignature: 'u' }] | ||||
| }; | ||||
|  | ||||
| const PresenceStatus = { | ||||
|     AVAILABLE: 0, | ||||
|     INVISIBLE: 1, | ||||
|     BUSY: 2, | ||||
|     IDLE: 3 | ||||
| }; | ||||
|  | ||||
| function Presence() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| Presence.prototype = { | ||||
|     _init: function() { | ||||
|         DBus.session.proxifyObject(this, 'org.gnome.SessionManager', '/org/gnome/SessionManager/Presence', this); | ||||
|     }, | ||||
|  | ||||
|     getStatus: function(callback) { | ||||
|         this.GetRemote('status', Lang.bind(this, | ||||
|             function(status, ex) { | ||||
|                 if (!ex) | ||||
|                     callback(this, status); | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     setStatus: function(status) { | ||||
|         this.SetStatusRemote(status); | ||||
|     } | ||||
| }; | ||||
| DBus.proxifyPrototype(Presence.prototype, PresenceIface); | ||||
|  | ||||
| // Note inhibitors are immutable objects, so they don't | ||||
| // change at runtime (changes always come in the form | ||||
| // of new inhibitors) | ||||
| const InhibitorIface = { | ||||
|     name: 'org.gnome.SessionManager.Inhibitor', | ||||
|     properties: [{ name: 'app_id', | ||||
|                    signature: 's', | ||||
|                    access: 'readonly' }, | ||||
|                  { name: 'client_id', | ||||
|                    signature: 's', | ||||
|                    access: 'readonly' }, | ||||
|                  { name: 'reason', | ||||
|                    signature: 's', | ||||
|                    access: 'readonly' }, | ||||
|                  { name: 'flags', | ||||
|                    signature: 'u', | ||||
|                    access: 'readonly' }, | ||||
|                  { name: 'toplevel_xid', | ||||
|                    signature: 'u', | ||||
|                    access: 'readonly' }, | ||||
|                  { name: 'cookie', | ||||
|                    signature: 'u', | ||||
|                    access: 'readonly' }], | ||||
| }; | ||||
|  | ||||
| function Inhibitor(objectPath) { | ||||
|     this._init(objectPath); | ||||
| } | ||||
|  | ||||
| Inhibitor.prototype = { | ||||
|     _init: function(objectPath) { | ||||
|         DBus.session.proxifyObject(this, | ||||
|                                    "org.gnome.SessionManager", | ||||
|                                    objectPath); | ||||
|         this.isLoaded = false; | ||||
|         this._loadingPropertiesCount = InhibitorIface.properties.length; | ||||
|         for (let i = 0; i < InhibitorIface.properties.length; i++) { | ||||
|             let propertyName = InhibitorIface.properties[i].name; | ||||
|             this.GetRemote(propertyName, Lang.bind(this, | ||||
|                 function(value, exception) { | ||||
|                     if (exception) | ||||
|                         return; | ||||
|  | ||||
|                     this[propertyName] = value; | ||||
|                     this._loadingPropertiesCount--; | ||||
|  | ||||
|                     if (this._loadingPropertiesCount == 0) { | ||||
|                         this.isLoaded = true; | ||||
|                         this.emit("is-loaded"); | ||||
|                     } | ||||
|                 })); | ||||
|         } | ||||
|     }, | ||||
| }; | ||||
| DBus.proxifyPrototype(Inhibitor.prototype, InhibitorIface); | ||||
| Signals.addSignalMethods(Inhibitor.prototype); | ||||
|  | ||||
|  | ||||
| // Not the full interface, only the methods we use | ||||
| const SessionManagerIface = { | ||||
|     name: 'org.gnome.SessionManager', | ||||
|     methods: [ | ||||
|         { name: 'Logout', inSignature: 'u', outSignature: '' }, | ||||
|         { name: 'Shutdown', inSignature: '', outSignature: '' } | ||||
|     ] | ||||
| }; | ||||
|  | ||||
| function SessionManager() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| SessionManager.prototype = { | ||||
|     _init: function() { | ||||
|         DBus.session.proxifyObject(this, 'org.gnome.SessionManager', '/org/gnome/SessionManager'); | ||||
|     } | ||||
| }; | ||||
| DBus.proxifyPrototype(SessionManager.prototype, SessionManagerIface); | ||||
| @@ -1,115 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| const DEFAULT_LIMIT = 512; | ||||
|  | ||||
| function HistoryManager(params) { | ||||
|     this._init(params); | ||||
| } | ||||
|  | ||||
| HistoryManager.prototype = { | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { gsettingsKey: null, | ||||
|                                         limit: DEFAULT_LIMIT, | ||||
|                                         entry: null }); | ||||
|  | ||||
|         this._key = params.gsettingsKey; | ||||
|         this._limit = params.limit; | ||||
|  | ||||
|         this._historyIndex = 0; | ||||
|         if (this._key) { | ||||
|             this._history = global.settings.get_strv(this._key); | ||||
|             global.settings.connect('changed::' + this._key, | ||||
|                                     Lang.bind(this, this._historyChanged)); | ||||
|  | ||||
|         } else { | ||||
|             this._history = []; | ||||
|         } | ||||
|  | ||||
|         this._entry = params.entry; | ||||
|  | ||||
|         if (this._entry) { | ||||
|             this._entry.connect('key-press-event',  | ||||
|                                 Lang.bind(this, this._onEntryKeyPress)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _historyChanged: function() { | ||||
|         this._history = global.settings.get_strv(this._key); | ||||
|         this._historyIndex = this._history.length; | ||||
|     }, | ||||
|  | ||||
|     prevItem: function(text) { | ||||
|         if (this._historyIndex <= 0) | ||||
|             return text; | ||||
|  | ||||
|         if (text) | ||||
|             this._history[this._historyIndex] = text; | ||||
|         this._historyIndex--; | ||||
|         return this._indexChanged(); | ||||
|     }, | ||||
|  | ||||
|     nextItem: function(text) { | ||||
|         if (this._historyIndex >= this._history.length) | ||||
|             return text; | ||||
|  | ||||
|         if (text) | ||||
|             this._history[this._historyIndex] = text; | ||||
|         this._historyIndex++; | ||||
|         return this._indexChanged(); | ||||
|     }, | ||||
|  | ||||
|     lastItem: function() { | ||||
|         if (this._historyIndex != this._history.length) { | ||||
|             this._historyIndex = this._history.length; | ||||
|             this._indexChanged(); | ||||
|         } | ||||
|  | ||||
|         return this._historyIndex[this._history.length]; | ||||
|     }, | ||||
|  | ||||
|     addItem: function(input) { | ||||
|         if (this._history.length == 0 || | ||||
|             this._history[this._history.length - 1] != input) { | ||||
|  | ||||
|             this._history.push(input); | ||||
|             this._historyIndex = this._history.length; | ||||
|             this._save(); | ||||
|         }    | ||||
|     }, | ||||
|  | ||||
|     _onEntryKeyPress: function(entry, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_Up) { | ||||
|             this.prevItem(entry.get_text()); | ||||
|             return true; | ||||
|         } else if (symbol == Clutter.KEY_Down) { | ||||
|             this.nextItem(entry.get_text()); | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _indexChanged: function() { | ||||
|         let current = this._history[this._historyIndex] || ''; | ||||
|         this.emit('changed', current); | ||||
|  | ||||
|         if (this._entry) | ||||
|             this._entry.set_text(current); | ||||
|  | ||||
|         return current; | ||||
|     }, | ||||
|  | ||||
|     _save: function() { | ||||
|         if (this._history.length > this._limit) | ||||
|             this._history.splice(0, this._history.length - this._limit); | ||||
|  | ||||
|         if (this._key) | ||||
|             global.settings.set_strv(this._key, this._history); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(HistoryManager.prototype); | ||||
| @@ -1,225 +0,0 @@ | ||||
| // -*- mode: js2; indent-tabs-mode: nil; js2-basic-offset: 4 -*- | ||||
|  | ||||
| const DBus = imports.dbus; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| // The following are not the complete interfaces, just the methods we need | ||||
| // (or may need in the future) | ||||
|  | ||||
| const ModemGsmNetworkInterface = { | ||||
|     name: 'org.freedesktop.ModemManager.Modem.Gsm.Network', | ||||
|     methods: [ | ||||
|         { name: 'GetRegistrationInfo', inSignature: '', outSignature: 'uss' }, | ||||
|         { name: 'GetSignalQuality', inSignature: '', outSignature: 'u' } | ||||
|     ], | ||||
|     properties: [ | ||||
|         { name: 'AccessTechnology', signature: 'u', access: 'read' } | ||||
|     ], | ||||
|     signals: [ | ||||
|         { name: 'SignalQuality', inSignature: 'u' }, | ||||
|         { name: 'RegistrationInfo', inSignature: 'uss' } | ||||
|     ] | ||||
| }; | ||||
| const ModemGsmNetworkProxy = DBus.makeProxyClass(ModemGsmNetworkInterface); | ||||
|  | ||||
| const ModemCdmaInterface = { | ||||
|     name: 'org.freedesktop.ModemManager.Modem.Cdma', | ||||
|     methods: [ | ||||
|         { name: 'GetSignalQuality', inSignature: '', outSignature: 'u' }, | ||||
|         { name: 'GetServingSystem', inSignature: '', outSignature: 'usu' } | ||||
|     ], | ||||
|     signals: [ | ||||
|         { name: 'SignalQuality', inSignature: 'u' } | ||||
|     ] | ||||
| }; | ||||
| const ModemCdmaProxy = DBus.makeProxyClass(ModemCdmaInterface); | ||||
|  | ||||
| let _providersTable; | ||||
| function _getProvidersTable() { | ||||
|     if (_providersTable) | ||||
|         return _providersTable; | ||||
|     let [providers, countryCodes] = Shell.mobile_providers_parse(); | ||||
|     return _providersTable = providers; | ||||
| } | ||||
|  | ||||
| function ModemGsm() { | ||||
|     this._init.apply(this, arguments); | ||||
| } | ||||
|  | ||||
| ModemGsm.prototype = { | ||||
|     _init: function(path) { | ||||
|         this._proxy = new ModemGsmNetworkProxy(DBus.system, 'org.freedesktop.ModemManager', path); | ||||
|  | ||||
|         this.signal_quality = 0; | ||||
|         this.operator_name = null; | ||||
|  | ||||
|         // Code is duplicated because the function have different signatures | ||||
|         this._proxy.connect('SignalQuality', Lang.bind(this, function(proxy, quality) { | ||||
|             this.signal_quality = quality; | ||||
|             this.emit('notify::signal-quality'); | ||||
|         })); | ||||
|         this._proxy.connect('RegistrationInfo', Lang.bind(this, function(proxy, status, code, name) { | ||||
|             this.operator_name = this._findOperatorName(name, code); | ||||
|             this.emit('notify::operator-name'); | ||||
|         })); | ||||
|         this._proxy.GetRegistrationInfoRemote(Lang.bind(this, function(result, err) { | ||||
|             if (err) { | ||||
|                 log(err); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             let [status, code, name] = result; | ||||
|             this.operator_name = this._findOperatorName(name, code); | ||||
|             this.emit('notify::operator-name'); | ||||
|         })); | ||||
|         this._proxy.GetSignalQualityRemote(Lang.bind(this, function(result, err) { | ||||
|             if (err) { | ||||
|                 // it will return an error if the device is not connected | ||||
|                 this.signal_quality = 0; | ||||
|             } else { | ||||
|                 let [quality] = result; | ||||
|                 this.signal_quality = quality; | ||||
|             } | ||||
|             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); | ||||
|  | ||||
| function ModemCdma() { | ||||
|     this._init.apply(this, arguments); | ||||
| } | ||||
|  | ||||
| ModemCdma.prototype = { | ||||
|     _init: function(path) { | ||||
|         this._proxy = new ModemCdmaProxy(DBus.system, 'org.freedesktop.ModemManager', path); | ||||
|  | ||||
|         this.signal_quality = 0; | ||||
|         this.operator_name = null; | ||||
|         this._proxy.connect('SignalQuality', Lang.bind(this, function(proxy, quality) { | ||||
|             this.signal_quality = quality; | ||||
|             this.emit('notify::signal-quality'); | ||||
|  | ||||
|             // receiving this signal means the device got activated | ||||
|             // and we can finally call GetServingSystem | ||||
|             if (this.operator_name == null) | ||||
|                 this._refreshServingSystem(); | ||||
|         })); | ||||
|         this._proxy.GetSignalQualityRemote(Lang.bind(this, function(result, err) { | ||||
|             if (err) { | ||||
|                 // it will return an error if the device is not connected | ||||
|                 this.signal_quality = 0; | ||||
|             } else { | ||||
|                 let [quality] = result; | ||||
|                 this.signal_quality = quality; | ||||
|             } | ||||
|             this.emit('notify::signal-quality'); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _refreshServingSystem: function() { | ||||
|         this._proxy.GetServingSystemRemote(Lang.bind(this, function(result, err) { | ||||
|             if (err) { | ||||
|                 // it will return an error if the device is not connected | ||||
|                 this.operator_name = null; | ||||
|             } else { | ||||
|                 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); | ||||
| @@ -1,35 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| // parse: | ||||
| // @params: caller-provided parameter object, or %null | ||||
| // @defaults: function-provided defaults object | ||||
| // @allowExtras: whether or not to allow properties not in @default | ||||
| // | ||||
| // Examines @params and fills in default values from @defaults for | ||||
| // any properties in @defaults that don't appear in @params. If | ||||
| // @allowExtras is not %true, it will throw an error if @params | ||||
| // contains any properties that aren't in @defaults. | ||||
| // | ||||
| // If @params is %null, this returns the values from @defaults. | ||||
| // | ||||
| // Return value: a new object, containing the merged parameters from | ||||
| // @params and @defaults | ||||
| function parse(params, defaults, allowExtras) { | ||||
|     let ret = {}, prop; | ||||
|  | ||||
|     if (!params) | ||||
|         params = {}; | ||||
|  | ||||
|     for (prop in params) { | ||||
|         if (!(prop in defaults) && !allowExtras) | ||||
|             throw new Error('Unrecognized parameter "' + prop + '"'); | ||||
|         ret[prop] = params[prop]; | ||||
|     } | ||||
|  | ||||
|     for (prop in defaults) { | ||||
|         if (!(prop in params)) | ||||
|             ret[prop] = defaults[prop]; | ||||
|     } | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
							
								
								
									
										213
									
								
								js/misc/util.js
									
									
									
									
									
								
							
							
						
						| @@ -1,213 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| /* http://daringfireball.net/2010/07/improved_regex_for_matching_urls */ | ||||
| const _urlRegexp = new RegExp('\\b(([a-z][\\w-]+:(/{1,3}|[a-z0-9%])|www\\d{0,3}[.]|[a-z0-9.\\-]+[.][a-z]{2,4}/)([^\\s()<>]+|\\(([^\\s()<>]+|(\\([^\\s()<>]+\\)))*\\))+(\\(([^\\s()<>]+|(\\([^\\s()<>]+\\)))*\\)|[^\\s`!()\\[\\]{};:\'\\".,<>?«»“”‘’]))', 'gi'); | ||||
|  | ||||
| // findUrls: | ||||
| // @str: string to find URLs in | ||||
| // | ||||
| // Searches @str for URLs and returns an array of objects with %url | ||||
| // properties showing the matched URL string, and %pos properties indicating | ||||
| // the position within @str where the URL was found. | ||||
| // | ||||
| // Return value: the list of match objects, as described above | ||||
| function findUrls(str) { | ||||
|     let res = [], match; | ||||
|     while ((match = _urlRegexp.exec(str))) | ||||
|         res.push({ url: match[0], pos: match.index }); | ||||
|     return res; | ||||
| } | ||||
|  | ||||
| // spawn: | ||||
| // @argv: an argv array | ||||
| // | ||||
| // Runs @argv in the background, handling any errors that occur | ||||
| // when trying to start the program. | ||||
| function spawn(argv) { | ||||
|     try { | ||||
|         trySpawn(argv); | ||||
|     } catch (err) { | ||||
|         _handleSpawnError(argv[0], err); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // spawnCommandLine: | ||||
| // @command_line: a command line | ||||
| // | ||||
| // Runs @command_line in the background, handling any errors that | ||||
| // occur when trying to parse or start the program. | ||||
| function spawnCommandLine(command_line) { | ||||
|     try { | ||||
|         let [success, argc, argv] = GLib.shell_parse_argv(command_line); | ||||
|         trySpawn(argv); | ||||
|     } catch (err) { | ||||
|         _handleSpawnError(command_line, err); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // trySpawn: | ||||
| // @argv: an argv array | ||||
| // | ||||
| // Runs @argv in the background. If launching @argv fails, | ||||
| // this will throw an error. | ||||
| function trySpawn(argv) | ||||
| { | ||||
|     try { | ||||
|         GLib.spawn_async(null, argv, null, | ||||
|                          GLib.SpawnFlags.SEARCH_PATH, | ||||
|                          null, null); | ||||
|     } catch (err) { | ||||
|         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.) | ||||
|             err.message = err.message.replace(/.*\((.+)\)/, '$1'); | ||||
|         } | ||||
|  | ||||
|         throw err; | ||||
|     } | ||||
| } | ||||
|  | ||||
| // trySpawnCommandLine: | ||||
| // @command_line: a command line | ||||
| // | ||||
| // Runs @command_line in the background. If launching @command_line | ||||
| // fails, this will throw an error. | ||||
| function trySpawnCommandLine(command_line) { | ||||
|     let success, argc, argv; | ||||
|  | ||||
|     try { | ||||
|         [success, argc, argv] = GLib.shell_parse_argv(command_line); | ||||
|     } catch (err) { | ||||
|         // Replace "Error invoking GLib.shell_parse_argv: " with | ||||
|         // something nicer | ||||
|         err.message = err.message.replace(/[^:]*: /, _("Could not parse command:") + "\n"); | ||||
|         throw err; | ||||
|     } | ||||
|  | ||||
|     trySpawn(argv); | ||||
| } | ||||
|  | ||||
| function _handleSpawnError(command, err) { | ||||
|     let title = _("Execution of '%s' failed:").format(command); | ||||
|     Main.notifyError(title, err.message); | ||||
| } | ||||
|  | ||||
| // killall: | ||||
| // @processName: a process name | ||||
| // | ||||
| // Kills @processName. If no process with the given name is found, | ||||
| // this will fail silently. | ||||
| function killall(processName) { | ||||
|     try { | ||||
|         // pkill is more portable than killall, but on Linux at least | ||||
|         // it won't match if you pass more than 15 characters of the | ||||
|         // process name... However, if you use the '-f' flag to match | ||||
|         // the entire command line, it will work, but we have to be | ||||
|         // careful in that case that we can match | ||||
|         // '/usr/bin/processName' but not 'gedit processName.c' or | ||||
|         // whatever... | ||||
|  | ||||
|         let argv = ['pkill', '-f', '^([^ ]*/)?' + processName + '($| )']; | ||||
|         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. | ||||
|     } catch (e) { | ||||
|         logError(e, 'Failed to kill ' + 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.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; 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(' '); | ||||
| } | ||||
							
								
								
									
										226
									
								
								js/perf/core.js
									
									
									
									
									
								
							
							
						
						| @@ -1,226 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Scripting = imports.ui.scripting; | ||||
|  | ||||
| // This performance script measure the most important (core) performance | ||||
| // metrics for the shell. By looking at the output metrics of this script | ||||
| // someone should be able to get an idea of how well the shell is performing | ||||
| // on a particular system. | ||||
|  | ||||
| let METRICS = { | ||||
|     overviewLatencyFirst: | ||||
|     { description: "Time to first frame after triggering overview, first time", | ||||
|       units: "us" }, | ||||
|     overviewFpsFirst: | ||||
|     { description: "Frame rate when going to the overview, first time", | ||||
|       units: "frames / s" }, | ||||
|     overviewLatencySubsequent: | ||||
|     { description: "Time to first frame after triggering overview, second time", | ||||
|       units: "us"}, | ||||
|     overviewFpsSubsequent: | ||||
|     { description: "Frames rate when going to the overview, second time", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps5Windows: | ||||
|     { description: "Frames rate when going to the overview, 5 windows open", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps10Windows: | ||||
|     { description: "Frames rate when going to the overview, 10 windows open", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps5Maximized: | ||||
|     { description: "Frames rate when going to the overview, 5 maximized windows open", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps10Maximized: | ||||
|     { description: "Frames rate when going to the overview, 10 maximized windows open", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps5Alpha: | ||||
|     { description: "Frames rate when going to the overview, 5 alpha-transparent windows open", | ||||
|       units: "frames / s" }, | ||||
|     overviewFps10Alpha: | ||||
|     { description: "Frames rate when going to the overview, 10 alpha-transparent windows open", | ||||
|       units: "frames / s" }, | ||||
|     usedAfterOverview: | ||||
|     { description: "Malloc'ed bytes after the overview is shown once", | ||||
|       units: "B" }, | ||||
|     leakedAfterOverview: | ||||
|     { description: "Additional malloc'ed bytes the second time the overview is shown", | ||||
|       units: "B" }, | ||||
|     applicationsShowTimeFirst: | ||||
|     { description: "Time to switch to applications view, first time", | ||||
|       units: "us" }, | ||||
|     applicationsShowTimeSubsequent: | ||||
|     { description: "Time to switch to applications view, second time", | ||||
|       units: "us"} | ||||
| }; | ||||
|  | ||||
| let WINDOW_CONFIGS = [ | ||||
|     { width: 640, height: 480, alpha: false, maximized: false, count: 1,  metric: 'overviewFpsSubsequent' }, | ||||
|     { width: 640, height: 480, alpha: false, maximized: false, count: 5,  metric: 'overviewFps5Windows'  }, | ||||
|     { width: 640, height: 480, alpha: false, maximized: false, count: 10, metric: 'overviewFps10Windows'  }, | ||||
|     { width: 640, height: 480, alpha: false, maximized: true,  count: 5,  metric: 'overviewFps5Maximized' }, | ||||
|     { width: 640, height: 480, alpha: false, maximized: true,  count: 10, metric: 'overviewFps10Maximized' }, | ||||
|     { width: 640, height: 480, alpha: true,  maximized: false, count: 5,  metric: 'overviewFps5Alpha' }, | ||||
|     { width: 640, height: 480, alpha: true,  maximized: false, count: 10, metric: 'overviewFps10Alpha' } | ||||
| ]; | ||||
|  | ||||
| function run() { | ||||
|     Scripting.defineScriptEvent("overviewShowStart", "Starting to show the overview"); | ||||
|     Scripting.defineScriptEvent("overviewShowDone", "Overview finished showing"); | ||||
|     Scripting.defineScriptEvent("afterShowHide", "After a show/hide cycle for the overview"); | ||||
|     Scripting.defineScriptEvent("applicationsShowStart", "Starting to switch to applications view"); | ||||
|     Scripting.defineScriptEvent("applicationsShowDone", "Done switching to applications view"); | ||||
|  | ||||
|     Main.overview.connect('shown', function() { | ||||
|                               Scripting.scriptEvent('overviewShowDone'); | ||||
|                           }); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     for (let i = 0; i < 2 * WINDOW_CONFIGS.length; i++) { | ||||
|         // We go to the overview twice for each configuration; the first time | ||||
|         // to calculate the mipmaps for the windows, the second time to get | ||||
|         // a clean set of numbers. | ||||
|         if ((i % 2) == 0) { | ||||
|             let config = WINDOW_CONFIGS[i / 2]; | ||||
|             yield Scripting.destroyTestWindows(); | ||||
|  | ||||
|             for (let k = 0; k < config.count; k++) | ||||
|                 yield Scripting.createTestWindow(config.width, config.height, config.alpha, config.maximized); | ||||
|  | ||||
|             yield Scripting.waitTestWindows(); | ||||
|             yield Scripting.sleep(1000); | ||||
|             yield Scripting.waitLeisure(); | ||||
|         } | ||||
|  | ||||
|         Scripting.scriptEvent('overviewShowStart'); | ||||
|         Main.overview.show(); | ||||
|  | ||||
|         yield Scripting.waitLeisure(); | ||||
|         Main.overview.hide(); | ||||
|         yield Scripting.waitLeisure(); | ||||
|  | ||||
|         global.gc(); | ||||
|         yield Scripting.sleep(1000); | ||||
|         Scripting.collectStatistics(); | ||||
|         Scripting.scriptEvent('afterShowHide'); | ||||
|     } | ||||
|  | ||||
|     yield Scripting.destroyTestWindows(); | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Main.overview.show(); | ||||
|     yield Scripting.waitLeisure(); | ||||
|  | ||||
|     for (let i = 0; i < 2; i++) { | ||||
|         Scripting.scriptEvent('applicationsShowStart'); | ||||
|         Main.overview.viewSelector.switchTab('applications'); | ||||
|         yield Scripting.waitLeisure(); | ||||
|         Scripting.scriptEvent('applicationsShowDone'); | ||||
|         Main.overview.viewSelector.switchTab('windows'); | ||||
|         yield Scripting.waitLeisure(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| let showingOverview = false; | ||||
| let finishedShowingOverview = false; | ||||
| let overviewShowStart; | ||||
| let overviewFrames; | ||||
| let overviewLatency; | ||||
| let mallocUsedSize = 0; | ||||
| let overviewShowCount = 0; | ||||
| let firstOverviewUsedSize; | ||||
| let haveSwapComplete = false; | ||||
| let applicationsShowStart; | ||||
| let applicationsShowCount = 0; | ||||
|  | ||||
| function script_overviewShowStart(time) { | ||||
|     showingOverview = true; | ||||
|     finishedShowingOverview = false; | ||||
|     overviewShowStart = time; | ||||
|     overviewFrames = 0; | ||||
| } | ||||
|  | ||||
| function script_overviewShowDone(time) { | ||||
|     // We've set up the state at the end of the zoom out, but we | ||||
|     // need to wait for one more frame to paint before we count | ||||
|     // ourselves as done. | ||||
|     finishedShowingOverview = true; | ||||
| } | ||||
|  | ||||
| function script_applicationsShowStart(time) { | ||||
|     applicationsShowStart = time; | ||||
| } | ||||
|  | ||||
| function script_applicationsShowDone(time) { | ||||
|     applicationsShowCount++; | ||||
|     if (applicationsShowCount == 1) | ||||
|         METRICS.applicationsShowTimeFirst.value = time - applicationsShowStart; | ||||
|     else | ||||
|         METRICS.applicationsShowTimeSubsequent.value = time - applicationsShowStart; | ||||
| } | ||||
|  | ||||
| function script_afterShowHide(time) { | ||||
|     if (overviewShowCount == 1) { | ||||
|         METRICS.usedAfterOverview.value = mallocUsedSize; | ||||
|     } else { | ||||
|         METRICS.leakedAfterOverview.value = mallocUsedSize - METRICS.usedAfterOverview.value; | ||||
|     } | ||||
| } | ||||
|  | ||||
| function malloc_usedSize(time, bytes) { | ||||
|     mallocUsedSize = bytes; | ||||
| } | ||||
|  | ||||
| function _frameDone(time) { | ||||
|     if (showingOverview) { | ||||
|         if (overviewFrames == 0) | ||||
|             overviewLatency = time - overviewShowStart; | ||||
|  | ||||
|         overviewFrames++; | ||||
|     } | ||||
|  | ||||
|     if (finishedShowingOverview) { | ||||
|         showingOverview = false; | ||||
|         finishedShowingOverview = false; | ||||
|         overviewShowCount++; | ||||
|  | ||||
|         let dt = (time - (overviewShowStart + overviewLatency)) / 1000000; | ||||
|  | ||||
|         // If we see a start frame and an end frame, that would | ||||
|         // be 1 frame for a FPS computation, hence the '- 1' | ||||
|         let fps = (overviewFrames - 1) / dt; | ||||
|  | ||||
|         if (overviewShowCount == 1) { | ||||
|             METRICS.overviewLatencyFirst.value = overviewLatency; | ||||
|             METRICS.overviewFpsFirst.value = fps; | ||||
|         } else if (overviewShowCount == 2) { | ||||
|             METRICS.overviewLatencySubsequent.value = overviewLatency; | ||||
|         } | ||||
|  | ||||
|         // Other than overviewFpsFirst, we collect FPS metrics the second | ||||
|         // we show each window configuration. overviewShowCount is 1,2,3... | ||||
|         if (overviewShowCount % 2 == 0) { | ||||
|             let config = WINDOW_CONFIGS[(overviewShowCount / 2) - 1]; | ||||
|             METRICS[config.metric].value = fps; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| function glx_swapComplete(time, swapTime) { | ||||
|     haveSwapComplete = true; | ||||
|  | ||||
|     _frameDone(swapTime); | ||||
| } | ||||
|  | ||||
| function clutter_stagePaintDone(time) { | ||||
|     // If we aren't receiving GLXBufferSwapComplete events, then we approximate | ||||
|     // the time the user sees a frame with the time we finished doing drawing | ||||
|     // commands for the frame. This doesn't take into account the time for | ||||
|     // the GPU to finish painting, and the time for waiting for the buffer | ||||
|     // swap, but if this are uniform - every frame takes the same time to draw - | ||||
|     // then it won't upset our FPS calculation, though the latency value | ||||
|     // will be slightly too low. | ||||
|  | ||||
|     if (!haveSwapComplete) | ||||
|         _frameDone(time); | ||||
| } | ||||
							
								
								
									
										26
									
								
								js/ui/Makefile.am
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| jsuidir = $(pkgdatadir)/js/ui | ||||
|  | ||||
| dist_jsui_DATA =		\ | ||||
| 	altTab.js		\ | ||||
| 	appDisplay.js		\ | ||||
| 	appIcon.js		\ | ||||
| 	button.js		\ | ||||
| 	chrome.js		\ | ||||
| 	dash.js			\ | ||||
| 	dnd.js			\ | ||||
| 	docDisplay.js		\ | ||||
| 	environment.js		\ | ||||
| 	genericDisplay.js	\ | ||||
| 	link.js			\ | ||||
| 	lookingGlass.js		\ | ||||
| 	main.js			\ | ||||
| 	overview.js		\ | ||||
| 	panel.js		\ | ||||
| 	places.js		\ | ||||
| 	runDialog.js		\ | ||||
| 	sidebar.js		\ | ||||
| 	tweener.js		\ | ||||
| 	widget.js		\ | ||||
| 	widgetBox.js		\ | ||||
| 	windowManager.js	\ | ||||
| 	workspaces.js | ||||
							
								
								
									
										1291
									
								
								js/ui/altTab.js
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										1661
									
								
								js/ui/appDisplay.js
									
									
									
									
									
								
							
							
						
						| @@ -1,135 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Shell = imports.gi.Shell; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| function AppFavorites() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| AppFavorites.prototype = { | ||||
|     FAVORITE_APPS_KEY: 'favorite-apps', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._favorites = {}; | ||||
|         global.settings.connect('changed::' + this.FAVORITE_APPS_KEY, Lang.bind(this, this._onFavsChanged)); | ||||
|         this._reload(); | ||||
|     }, | ||||
|  | ||||
|     _onFavsChanged: function() { | ||||
|         this._reload(); | ||||
|         this.emit('changed'); | ||||
|     }, | ||||
|  | ||||
|     _reload: function() { | ||||
|         let ids = global.settings.get_strv(this.FAVORITE_APPS_KEY); | ||||
|         let appSys = Shell.AppSystem.get_default(); | ||||
|         let apps = ids.map(function (id) { | ||||
|                 return appSys.get_app(id); | ||||
|             }).filter(function (app) { | ||||
|                 return app != null; | ||||
|             }); | ||||
|         this._favorites = {}; | ||||
|         for (let i = 0; i < apps.length; i++) { | ||||
|             let app = apps[i]; | ||||
|             this._favorites[app.get_id()] = app; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getIds: function() { | ||||
|         let ret = []; | ||||
|         for (let id in this._favorites) | ||||
|             ret.push(id); | ||||
|         return ret; | ||||
|     }, | ||||
|  | ||||
|     getFavoriteMap: function() { | ||||
|         return this._favorites; | ||||
|     }, | ||||
|  | ||||
|     getFavorites: function() { | ||||
|         let ret = []; | ||||
|         for (let id in this._favorites) | ||||
|             ret.push(this._favorites[id]); | ||||
|         return ret; | ||||
|     }, | ||||
|  | ||||
|     isFavorite: function(appId) { | ||||
|         return appId in this._favorites; | ||||
|     }, | ||||
|  | ||||
|     _addFavorite: function(appId, pos) { | ||||
|         if (appId in this._favorites) | ||||
|             return false; | ||||
|  | ||||
|         let app = Shell.AppSystem.get_default().get_app(appId); | ||||
|  | ||||
|         if (!app) | ||||
|             return false; | ||||
|  | ||||
|         let ids = this._getIds(); | ||||
|         if (pos == -1) | ||||
|             ids.push(appId); | ||||
|         else | ||||
|             ids.splice(pos, 0, appId); | ||||
|         global.settings.set_strv(this.FAVORITE_APPS_KEY, ids); | ||||
|         this._favorites[appId] = app; | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     addFavoriteAtPos: function(appId, pos) { | ||||
|         if (!this._addFavorite(appId, pos)) | ||||
|             return; | ||||
|  | ||||
|         let app = Shell.AppSystem.get_default().get_app(appId); | ||||
|  | ||||
|         Main.overview.shellInfo.setMessage(_("%s has been added to your favorites.").format(app.get_name()), Lang.bind(this, function () { | ||||
|             this._removeFavorite(appId); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     addFavorite: function(appId) { | ||||
|         this.addFavoriteAtPos(appId, -1); | ||||
|     }, | ||||
|  | ||||
|     moveFavoriteToPos: function(appId, pos) { | ||||
|         this._removeFavorite(appId); | ||||
|         this._addFavorite(appId, pos); | ||||
|     }, | ||||
|  | ||||
|     _removeFavorite: function(appId) { | ||||
|         if (!appId in this._favorites) | ||||
|             return false; | ||||
|  | ||||
|         let ids = this._getIds().filter(function (id) { return id != appId; }); | ||||
|         global.settings.set_strv(this.FAVORITE_APPS_KEY, ids); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     removeFavorite: function(appId) { | ||||
|         let ids = this._getIds(); | ||||
|         let pos = ids.indexOf(appId); | ||||
|  | ||||
|         let app = this._favorites[appId]; | ||||
|         if (!this._removeFavorite(appId)) | ||||
|             return; | ||||
|  | ||||
|         Main.overview.shellInfo.setMessage(_("%s has been removed from your favorites.").format(app.get_name()), | ||||
|                                          Lang.bind(this, function () { | ||||
|             this._addFavorite(appId, pos); | ||||
|         })); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(AppFavorites.prototype); | ||||
|  | ||||
| var appFavoritesInstance = null; | ||||
| function getAppFavorites() { | ||||
|     if (appFavoritesInstance == null) | ||||
|         appFavoritesInstance = new AppFavorites(); | ||||
|     return appFavoritesInstance; | ||||
| } | ||||
							
								
								
									
										133
									
								
								js/ui/appIcon.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,133 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Big = imports.gi.Big; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Lang = imports.lang; | ||||
|  | ||||
| const GenericDisplay = imports.ui.genericDisplay; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const GLOW_COLOR = new Clutter.Color(); | ||||
| GLOW_COLOR.from_pixel(0x4f6ba4ff); | ||||
| const GLOW_PADDING_HORIZONTAL = 3; | ||||
| const GLOW_PADDING_VERTICAL = 3; | ||||
|  | ||||
| const APP_ICON_SIZE = 48; | ||||
|  | ||||
| function AppIcon(appInfo) { | ||||
|     this._init(appInfo); | ||||
| } | ||||
|  | ||||
| AppIcon.prototype = { | ||||
|     _init : function(appInfo) { | ||||
|         this.appInfo = appInfo; | ||||
|         this.windows = Shell.AppMonitor.get_default().get_windows_for_app(appInfo.get_id()); | ||||
|         for (let i = 0; i < this.windows.length; i++) { | ||||
|             this.windows[i].connect('notify::user-time', Lang.bind(this, this._resortWindows)); | ||||
|         } | ||||
|         this._resortWindows(); | ||||
|  | ||||
|         this.actor = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                    corner_radius: 2, | ||||
|                                    padding: 1, | ||||
|                                    reactive: true }); | ||||
|  | ||||
|         let iconBox = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                     x_align: Big.BoxAlignment.CENTER, | ||||
|                                     y_align: Big.BoxAlignment.CENTER }); | ||||
|         this.icon = appInfo.create_icon_texture(APP_ICON_SIZE); | ||||
|         iconBox.append(this.icon, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         this.actor.append(iconBox, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         this._windows = Shell.AppMonitor.get_default().get_windows_for_app(appInfo.get_id()); | ||||
|  | ||||
|         let nameBox = new Shell.GenericContainer(); | ||||
|         nameBox.connect('get-preferred-width', Lang.bind(this, this._nameBoxGetPreferredWidth)); | ||||
|         nameBox.connect('get-preferred-height', Lang.bind(this, this._nameBoxGetPreferredHeight)); | ||||
|         nameBox.connect('allocate', Lang.bind(this, this._nameBoxAllocate)); | ||||
|         this._nameBox = nameBox; | ||||
|  | ||||
|         this._name = new Clutter.Text({ color: GenericDisplay.ITEM_DISPLAY_NAME_COLOR, | ||||
|                                         font_name: "Sans 12px", | ||||
|                                         line_alignment: Pango.Alignment.CENTER, | ||||
|                                         ellipsize: Pango.EllipsizeMode.END, | ||||
|                                         text: appInfo.get_name() }); | ||||
|         nameBox.add_actor(this._name); | ||||
|         this._glowBox = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL }); | ||||
|         let glowPath = GLib.filename_to_uri(global.imagedir + 'app-well-glow.png', ''); | ||||
|         for (let i = 0; i < this._windows.length && i < 3; i++) { | ||||
|             let glow = Shell.TextureCache.get_default().load_uri_sync(Shell.TextureCachePolicy.FOREVER, | ||||
|                                                                           glowPath, -1, -1); | ||||
|             glow.keep_aspect_ratio = false; | ||||
|             this._glowBox.append(glow, Big.BoxPackFlags.EXPAND); | ||||
|         } | ||||
|         this._nameBox.add_actor(this._glowBox); | ||||
|         this._glowBox.lower(this._name); | ||||
|         this.actor.append(nameBox, Big.BoxPackFlags.NONE); | ||||
|     }, | ||||
|  | ||||
|     _nameBoxGetPreferredWidth: function (nameBox, forHeight, alloc) { | ||||
|         let [min, natural] = this._name.get_preferred_width(forHeight); | ||||
|         alloc.min_size = min + GLOW_PADDING_HORIZONTAL * 2; | ||||
|         alloc.natural_size = natural + GLOW_PADDING_HORIZONTAL * 2; | ||||
|     }, | ||||
|  | ||||
|     _nameBoxGetPreferredHeight: function (nameBox, forWidth, alloc) { | ||||
|         let [min, natural] = this._name.get_preferred_height(forWidth); | ||||
|         alloc.min_size = min + GLOW_PADDING_VERTICAL * 2; | ||||
|         alloc.natural_size = natural + GLOW_PADDING_VERTICAL * 2; | ||||
|     }, | ||||
|  | ||||
|     _nameBoxAllocate: function (nameBox, box, flags) { | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         let [minWidth, naturalWidth] = this._name.get_preferred_width(-1); | ||||
|         let [minHeight, naturalHeight] = this._name.get_preferred_height(-1); | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|         let targetWidth = availWidth; | ||||
|         let xPadding = 0; | ||||
|         if (naturalWidth < availWidth) { | ||||
|             xPadding = Math.floor((availWidth - naturalWidth) / 2); | ||||
|         } | ||||
|         childBox.x1 = xPadding; | ||||
|         childBox.x2 = availWidth - xPadding; | ||||
|         childBox.y1 = GLOW_PADDING_VERTICAL; | ||||
|         childBox.y2 = availHeight - GLOW_PADDING_VERTICAL; | ||||
|         this._name.allocate(childBox, flags); | ||||
|  | ||||
|         // Now the glow | ||||
|         if (this._glowBox != null) { | ||||
|             let glowPaddingHoriz = Math.max(0, xPadding - GLOW_PADDING_HORIZONTAL); | ||||
|             glowPaddingHoriz = Math.max(GLOW_PADDING_HORIZONTAL, glowPaddingHoriz); | ||||
|             childBox.x1 = glowPaddingHoriz; | ||||
|             childBox.x2 = availWidth - glowPaddingHoriz; | ||||
|             childBox.y1 = 0; | ||||
|             childBox.y2 = availHeight; | ||||
|             this._glowBox.allocate(childBox, flags); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _resortWindows: function() { | ||||
|         this.windows.sort(function (a, b) { | ||||
|             let timeA = a.get_user_time(); | ||||
|             let timeB = b.get_user_time(); | ||||
|             if (timeA == timeB) | ||||
|                 return 0; | ||||
|             else if (timeA > timeB) | ||||
|                 return -1; | ||||
|             return 1; | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     getDragActor: function() { | ||||
|         return this.appInfo.create_icon_texture(APP_ICON_SIZE); | ||||
|     }, | ||||
|  | ||||
|     getDragActorSource: function() { | ||||
|         return this.icon; | ||||
|     } | ||||
| }; | ||||
| @@ -1,438 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const POPUP_ANIMATION_TIME = 0.15; | ||||
|  | ||||
| /** | ||||
|  * BoxPointer: | ||||
|  * @side: side to draw the arrow on | ||||
|  * @binProperties: Properties to set on contained bin | ||||
|  * | ||||
|  * 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(). | ||||
|  * | ||||
|  */ | ||||
| function BoxPointer(side, binProperties) { | ||||
|     this._init(side, binProperties); | ||||
| } | ||||
|  | ||||
| BoxPointer.prototype = { | ||||
|     _init: function(arrowSide, binProperties) { | ||||
|         this._arrowSide = arrowSide; | ||||
|         this._arrowOrigin = 0; | ||||
|         this.actor = new St.Bin({ x_fill: true, | ||||
|                                   y_fill: true }); | ||||
|         this._container = new Shell.GenericContainer(); | ||||
|         this.actor.set_child(this._container); | ||||
|         this._container.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth)); | ||||
|         this._container.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight)); | ||||
|         this._container.connect('allocate', Lang.bind(this, this._allocate)); | ||||
|         this.bin = new St.Bin(binProperties); | ||||
|         this._container.add_actor(this.bin); | ||||
|         this._border = new St.DrawingArea(); | ||||
|         this._border.connect('repaint', Lang.bind(this, this._drawBorder)); | ||||
|         this._container.add_actor(this._border); | ||||
|         this.bin.raise(this._border); | ||||
|         this._xOffset = 0; | ||||
|         this._yOffset = 0; | ||||
|         this._xPosition = 0; | ||||
|         this._yPosition = 0; | ||||
|     }, | ||||
|  | ||||
|     show: function(animate, onComplete) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|  | ||||
|         this.opacity = 0; | ||||
|         this.actor.show(); | ||||
|  | ||||
|         if (animate) { | ||||
|             switch (this._arrowSide) { | ||||
|                 case St.Side.TOP: | ||||
|                     this.yOffset = -rise; | ||||
|                     break; | ||||
|                 case St.Side.BOTTOM: | ||||
|                     this.yOffset = rise; | ||||
|                     break; | ||||
|                 case St.Side.LEFT: | ||||
|                     this.xOffset = -rise; | ||||
|                     break; | ||||
|                 case St.Side.RIGHT: | ||||
|                     this.xOffset = rise; | ||||
|                     break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Tweener.addTween(this, { opacity: 255, | ||||
|                                  xOffset: 0, | ||||
|                                  yOffset: 0, | ||||
|                                  transition: "linear", | ||||
|                                  onComplete: onComplete, | ||||
|                                  time: POPUP_ANIMATION_TIME }); | ||||
|     }, | ||||
|  | ||||
|     hide: function(animate, onComplete) { | ||||
|         let xOffset = 0; | ||||
|         let yOffset = 0; | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|  | ||||
|         if (animate) { | ||||
|             switch (this._arrowSide) { | ||||
|                 case St.Side.TOP: | ||||
|                     yOffset = rise; | ||||
|                     break; | ||||
|                 case St.Side.BOTTOM: | ||||
|                     yOffset = -rise; | ||||
|                     break; | ||||
|                 case St.Side.LEFT: | ||||
|                     xOffset = rise; | ||||
|                     break; | ||||
|                 case St.Side.RIGHT: | ||||
|                     xOffset = -rise; | ||||
|                     break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Tweener.addTween(this, { opacity: 0, | ||||
|                                  xOffset: xOffset, | ||||
|                                  yOffset: yOffset, | ||||
|                                  transition: "linear", | ||||
|                                  time: POPUP_ANIMATION_TIME, | ||||
|                                  onComplete: Lang.bind(this, function () { | ||||
|                                      this.actor.hide(); | ||||
|                                      this.xOffset = 0; | ||||
|                                      this.yOffset = 0; | ||||
|                                      if (onComplete) | ||||
|                                          onComplete(); | ||||
|                                  }) | ||||
|                                }); | ||||
|     }, | ||||
|  | ||||
|     _adjustAllocationForArrow: function(isWidth, alloc) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let borderWidth = themeNode.get_length('-arrow-border-width'); | ||||
|         alloc.min_size += borderWidth * 2; | ||||
|         alloc.natural_size += borderWidth * 2; | ||||
|         if ((!isWidth && (this._arrowSide == St.Side.TOP || this._arrowSide == St.Side.BOTTOM)) | ||||
|             || (isWidth && (this._arrowSide == St.Side.LEFT || this._arrowSide == St.Side.RIGHT))) { | ||||
|             let rise = themeNode.get_length('-arrow-rise'); | ||||
|             alloc.min_size += rise; | ||||
|             alloc.natural_size += rise; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         let [minInternalSize, natInternalSize] = this.bin.get_preferred_width(forHeight); | ||||
|         alloc.min_size = minInternalSize; | ||||
|         alloc.natural_size = natInternalSize; | ||||
|         this._adjustAllocationForArrow(true, alloc); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let [minSize, naturalSize] = this.bin.get_preferred_height(forWidth); | ||||
|         alloc.min_size = minSize; | ||||
|         alloc.natural_size = naturalSize; | ||||
|         this._adjustAllocationForArrow(false, alloc); | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let borderWidth = themeNode.get_length('-arrow-border-width'); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|  | ||||
|         childBox.x1 = 0; | ||||
|         childBox.y1 = 0; | ||||
|         childBox.x2 = availWidth; | ||||
|         childBox.y2 = availHeight; | ||||
|         this._border.allocate(childBox, flags); | ||||
|  | ||||
|         childBox.x1 = borderWidth; | ||||
|         childBox.y1 = borderWidth; | ||||
|         childBox.x2 = availWidth - borderWidth; | ||||
|         childBox.y2 = availHeight - borderWidth; | ||||
|         switch (this._arrowSide) { | ||||
|             case St.Side.TOP: | ||||
|                 childBox.y1 += rise; | ||||
|                 break; | ||||
|             case St.Side.BOTTOM: | ||||
|                 childBox.y2 -= rise; | ||||
|                 break; | ||||
|             case St.Side.LEFT: | ||||
|                 childBox.x1 += rise; | ||||
|                 break; | ||||
|             case St.Side.RIGHT: | ||||
|                 childBox.x2 -= rise; | ||||
|                 break; | ||||
|         } | ||||
|         this.bin.allocate(childBox, flags); | ||||
|  | ||||
|         if (this._sourceActor && this._sourceActor.mapped) | ||||
|             this._reposition(this._sourceActor, this._gap, this._alignment); | ||||
|     }, | ||||
|  | ||||
|     _drawBorder: function(area) { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|  | ||||
|         let borderWidth = themeNode.get_length('-arrow-border-width'); | ||||
|         let base = themeNode.get_length('-arrow-base'); | ||||
|         let rise = themeNode.get_length('-arrow-rise'); | ||||
|         let borderRadius = themeNode.get_length('-arrow-border-radius'); | ||||
|  | ||||
|         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(); | ||||
|         let [boxWidth, boxHeight] = [width, height]; | ||||
|         if (this._arrowSide == St.Side.TOP || this._arrowSide == St.Side.BOTTOM) { | ||||
|             boxHeight -= rise; | ||||
|         } else { | ||||
|             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 | ||||
|         if (this._arrowSide == St.Side.TOP) { | ||||
|             cr.translate(0, rise); | ||||
|         } else if (this._arrowSide == St.Side.LEFT) { | ||||
|             cr.translate(rise, 0); | ||||
|         } | ||||
|  | ||||
|         let [x1, y1] = [halfBorder, halfBorder]; | ||||
|         let [x2, y2] = [boxWidth - halfBorder, boxHeight - halfBorder]; | ||||
|  | ||||
|         cr.moveTo(x1 + borderRadius, y1); | ||||
|         if (this._arrowSide == St.Side.TOP) { | ||||
|             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); | ||||
|                 cr.lineTo(this._arrowOrigin + halfBase, y1); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         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 (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); | ||||
|                 cr.lineTo(x2, this._arrowOrigin + halfBase); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         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 (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); | ||||
|                 cr.lineTo(this._arrowOrigin - halfBase, y2); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         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 (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); | ||||
|                 cr.lineTo(x1, this._arrowOrigin - halfBase); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         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(); | ||||
|         Clutter.cairo_set_source_color(cr, borderColor); | ||||
|         cr.setLineWidth(borderWidth); | ||||
|         cr.stroke(); | ||||
|     }, | ||||
|  | ||||
|     setPosition: function(sourceActor, gap, alignment) { | ||||
|         // We need to show it now to force an allocation, | ||||
|         // so that we can query the correct size. | ||||
|         this.actor.show(); | ||||
|  | ||||
|         this._sourceActor = sourceActor; | ||||
|         this._gap = gap; | ||||
|         this._alignment = alignment; | ||||
|  | ||||
|         this._reposition(sourceActor, gap, alignment); | ||||
|     }, | ||||
|  | ||||
|     _reposition: function(sourceActor, gap, alignment) { | ||||
|         // Position correctly relative to the sourceActor | ||||
|         let sourceNode = sourceActor.get_theme_node(); | ||||
|         let sourceContentBox = sourceNode.get_content_box(sourceActor.get_allocation_box()); | ||||
|         let sourceAllocation = Shell.util_get_transformed_allocation(sourceActor); | ||||
|         let sourceCenterX = sourceAllocation.x1 + sourceContentBox.x1 + (sourceContentBox.x2 - sourceContentBox.x1) / 2; | ||||
|         let sourceCenterY = sourceAllocation.y1 + sourceContentBox.y1 + (sourceContentBox.y2 - sourceContentBox.y1) / 2; | ||||
|         let [minWidth, minHeight, natWidth, natHeight] = this.actor.get_preferred_size(); | ||||
|  | ||||
|         // We also want to keep it onscreen, and separated from the | ||||
|         // edge by the same distance as the main part of the box is | ||||
|         // separated from its sourceActor | ||||
|         let primary = global.get_primary_monitor(); | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         let borderWidth = themeNode.get_length('-arrow-border-width'); | ||||
|         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 resX, resY; | ||||
|  | ||||
|         switch (this._arrowSide) { | ||||
|         case St.Side.TOP: | ||||
|             resY = sourceAllocation.y2 + gap; | ||||
|             break; | ||||
|         case St.Side.BOTTOM: | ||||
|             resY = sourceAllocation.y1 - natHeight - gap; | ||||
|             break; | ||||
|         case St.Side.LEFT: | ||||
|             resX = sourceAllocation.x2 + gap; | ||||
|             break; | ||||
|         case St.Side.RIGHT: | ||||
|             resX = sourceAllocation.x1 - natWidth - gap; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         // 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, primary.x + 10); | ||||
|             resX = Math.min(resX, primary.x + primary.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, primary.y + 10); | ||||
|             resY = Math.min(resY, primary.y + primary.height - (10 + natHeight)); | ||||
|  | ||||
|             this.setArrowOrigin(sourceCenterY - resY); | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         let parent = this.actor.get_parent(); | ||||
|         let success, x, y; | ||||
|         while (!success) { | ||||
|             [success, x, y] = parent.transform_stage_point(resX, resY); | ||||
|             parent = parent.get_parent(); | ||||
|         } | ||||
|  | ||||
|         this._xPosition = Math.floor(x); | ||||
|         this._yPosition = Math.floor(y); | ||||
|         this._shiftActor(); | ||||
|     }, | ||||
|  | ||||
|     // @origin: Coordinate specifying middle of the arrow, along | ||||
|     // the Y axis for St.Side.LEFT, St.Side.RIGHT from the top and X axis from | ||||
|     // the left for St.Side.TOP and St.Side.BOTTOM. | ||||
|     setArrowOrigin: function(origin) { | ||||
|         if (this._arrowOrigin != origin) { | ||||
|             this._arrowOrigin = origin; | ||||
|             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 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. | ||||
|         this.actor.set_anchor_point(-(this._xPosition + this._xOffset), | ||||
|                                     -(this._yPosition + this._yOffset)); | ||||
|     }, | ||||
|  | ||||
|     set xOffset(offset) { | ||||
|         this._xOffset = offset; | ||||
|         this._shiftActor(); | ||||
|     }, | ||||
|  | ||||
|     get xOffset() { | ||||
|         return this._xOffset; | ||||
|     }, | ||||
|  | ||||
|     set yOffset(offset) { | ||||
|         this._yOffset = offset; | ||||
|         this._shiftActor(); | ||||
|     }, | ||||
|  | ||||
|     get yOffset() { | ||||
|         return this._yOffset; | ||||
|     }, | ||||
|  | ||||
|     set opacity(opacity) { | ||||
|         this.actor.opacity = opacity; | ||||
|     }, | ||||
|  | ||||
|     get opacity() { | ||||
|         return this.actor.opacity; | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										173
									
								
								js/ui/button.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,173 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Big = imports.gi.Big; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const DEFAULT_BUTTON_COLOR = new Clutter.Color(); | ||||
| DEFAULT_BUTTON_COLOR.from_pixel(0xeeddcc66); | ||||
|  | ||||
| const DEFAULT_PRESSED_BUTTON_COLOR = new Clutter.Color(); | ||||
| DEFAULT_PRESSED_BUTTON_COLOR.from_pixel(0xccbbaa66); | ||||
|  | ||||
| const DEFAULT_TEXT_COLOR = new Clutter.Color(); | ||||
| DEFAULT_TEXT_COLOR.from_pixel(0x000000ff); | ||||
|  | ||||
| const DEFAULT_FONT = 'Sans Bold 16px'; | ||||
|  | ||||
| // Padding on the left and right side of the button. | ||||
| const SIDE_PADDING = 14; | ||||
|  | ||||
| function Button(widget, buttonColor, pressedButtonColor, textColor, font) { | ||||
|     this._init(widget, buttonColor, pressedButtonColor, textColor, font); | ||||
| } | ||||
|  | ||||
| Button.prototype = { | ||||
|     _init : function(widgetOrText, buttonColor, pressedButtonColor, textColor, font) { | ||||
|         this._buttonColor = buttonColor | ||||
|         if (buttonColor == null) | ||||
|             this._buttonColor = DEFAULT_BUTTON_COLOR; | ||||
|  | ||||
|         this._pressedButtonColor = pressedButtonColor | ||||
|         if (pressedButtonColor == null) | ||||
|             this._pressedButtonColor = DEFAULT_PRESSED_BUTTON_COLOR; | ||||
|  | ||||
|         this._textColor = textColor; | ||||
|         if (textColor == null) | ||||
|             this._textColor = DEFAULT_TEXT_COLOR; | ||||
|  | ||||
|         this._font = font; | ||||
|         if (font == null) | ||||
|             this._font = DEFAULT_FONT; | ||||
|  | ||||
|         this._isBetweenPressAndRelease = false; | ||||
|         this._mouseIsOverButton = false; | ||||
|  | ||||
|         this.actor = new Shell.ButtonBox({ reactive: true, | ||||
|                                            corner_radius: 5, | ||||
|                                            padding_left: SIDE_PADDING, | ||||
|                                            padding_right: SIDE_PADDING, | ||||
|                                            orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                            y_align: Big.BoxAlignment.CENTER | ||||
|                                          }); | ||||
|         if (typeof widgetOrText == 'string') { | ||||
|             this._widget = new Clutter.Text({ font_name: this._font, | ||||
|                                               color: this._textColor, | ||||
|                                               text: widgetOrText }); | ||||
|         } else { | ||||
|             this._widget = widgetOrText; | ||||
|         } | ||||
|  | ||||
|         this.actor.append(this._widget, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         this.actor.connect('notify::hover', Lang.bind(this, this._updateColors)); | ||||
|         this.actor.connect('notify::pressed', Lang.bind(this, this._updateColors)); | ||||
|         this.actor.connect('notify::active', Lang.bind(this, this._updateColors)); | ||||
|     }, | ||||
|  | ||||
|     _updateColors : function() { | ||||
|         if (this.actor.active || this.actor.pressed) | ||||
|             this.actor.backgroundColor = this._pressedButtonColor; | ||||
|         else if (this.actor.hover) | ||||
|             this.actor.backgroundColor = this._buttonColor; | ||||
|         else | ||||
|             this.actor.backgroundColor = null; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| Signals.addSignalMethods(Button.prototype); | ||||
|  | ||||
| /* Delay before the icon should appear, in seconds after the pointer has entered the parent */ | ||||
| const ANIMATION_TIME = 0.25; | ||||
|  | ||||
| /* This is an icon button that fades in/out when mouse enters/leaves the parent. | ||||
|  * A delay is used before the fading starts. You can force it to be shown if needed. | ||||
|  * | ||||
|  * parent -- used to show/hide the button depending on mouse entering/leaving it | ||||
|  * size -- size in pixels of  both the button and the icon it contains | ||||
|  * texture -- optional, must be used if the texture for the icon is already created (else, use setIconFromName) | ||||
|  */ | ||||
| function iconButton(parent, size, texture) { | ||||
|     this._init(parent, size, texture); | ||||
| } | ||||
|  | ||||
| iconButton.prototype = { | ||||
|     _init : function(parent, size, texture) { | ||||
|         this._size = size; | ||||
|         if (texture) | ||||
|             this.actor = texture; | ||||
|         else | ||||
|             this.actor = new Clutter.Texture({ width: this._size, height: this._size }); | ||||
|         this.actor.set_reactive(true); | ||||
|         this.actor.set_opacity(0); | ||||
|         parent.connect("enter-event", Lang.bind(this, function(actor, event) { | ||||
|             this._shouldHide = false; | ||||
|             // Nothing to do if the cursor has come back from a child of the parent actor | ||||
|             if (actor.get_children().indexOf(event.get_related()) != -1) | ||||
|                 return; | ||||
|  | ||||
|             this._fadeIn(); | ||||
|         })); | ||||
|         parent.connect("leave-event", Lang.bind(this, function(actor, event) { | ||||
|             // Nothing to do if the cursor has merely entered a child of the parent actor | ||||
|             if (actor.get_children().indexOf(event.get_related()) != -1) | ||||
|                 return; | ||||
|  | ||||
|             // Remember that we should not be visible to hide the button if forceShow is unset | ||||
|             if (this._forceShow) { | ||||
|                 this._shouldHide = true; | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             this._fadeOut(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     /// Private methods /// | ||||
|  | ||||
|     setIconFromName : function(iconName) { | ||||
|         let iconTheme = Gtk.IconTheme.get_default(); | ||||
|         let iconInfo = iconTheme.lookup_icon(iconName, this._size, 0); | ||||
|         if (!iconInfo) | ||||
|             return; | ||||
|  | ||||
|         let iconPath = iconInfo.get_filename(); | ||||
|         this.actor.set_from_file(iconPath); | ||||
|     }, | ||||
|  | ||||
|     // Useful if we want to show the button immediately, | ||||
|     // e.g. in case the mouse is already in the parent when the button is created | ||||
|     show : function() { | ||||
|         this.actor.set_opacity(255); | ||||
|     }, | ||||
|  | ||||
|     // If show is true, prevents the button from fading out | ||||
|     forceShow : function(show) { | ||||
|         this._forceShow = show; | ||||
|         // Hide the button if it should have been hidden under normal conditions | ||||
|         if (!this._forceShow && this._shouldHide) { | ||||
|            this._fadeOut(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     /// Private methods /// | ||||
|  | ||||
|     _fadeIn : function() { | ||||
|         Tweener.removeTweens(this.actor); | ||||
|         Tweener.addTween(this.actor, { opacity: 255, | ||||
|                                        time: ANIMATION_TIME, | ||||
|                                        transition :"easeInQuad" }); | ||||
|     }, | ||||
|  | ||||
|     _fadeOut : function() { | ||||
|         Tweener.removeTweens(this.actor); | ||||
|         Tweener.addTween(this.actor, { opacity: 0, | ||||
|                                        time: ANIMATION_TIME, | ||||
|                                        transition :"easeOutQuad" }); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| @@ -1,774 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const DBus = imports.dbus; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const St = imports.gi.St; | ||||
| const Signals = imports.signals; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Gettext_gtk30 = imports.gettext.domain('gtk30'); | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
| const C_ = Gettext.pgettext; | ||||
| 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 | ||||
| const CLOCK_FORMAT_KEY        = 'clock-format'; | ||||
|  | ||||
| function _sameDay(dateA, dateB) { | ||||
|     return (dateA.getDate() == dateB.getDate() && | ||||
|             dateA.getMonth() == dateB.getMonth() && | ||||
|             dateA.getYear() == dateB.getYear()); | ||||
| } | ||||
|  | ||||
| function _sameYear(dateA, dateB) { | ||||
|     return (dateA.getYear() == dateB.getYear()); | ||||
| } | ||||
|  | ||||
| /* TODO: maybe needs config - right now we assume that Saturday and | ||||
|  * Sunday are non-work days (not true in e.g. Israel, it's Sunday and | ||||
|  * Monday there) | ||||
|  */ | ||||
| function _isWorkDay(date) { | ||||
|     return date.getDay() != 0 && date.getDay() != 6; | ||||
| } | ||||
|  | ||||
| function _getBeginningOfDay(date) { | ||||
|     let ret = new Date(date.getTime()); | ||||
|     ret.setHours(0); | ||||
|     ret.setMinutes(0); | ||||
|     ret.setSeconds(0); | ||||
|     ret.setMilliseconds(0); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| function _getEndOfDay(date) { | ||||
|     let ret = new Date(date.getTime()); | ||||
|     ret.setHours(23); | ||||
|     ret.setMinutes(59); | ||||
|     ret.setSeconds(59); | ||||
|     ret.setMilliseconds(999); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| function _formatEventTime(event, clockFormat) { | ||||
|     let ret; | ||||
|     if (event.allDay) { | ||||
|         /* Translators: Shown in calendar event list for all day events | ||||
|          * Keep it short, best if you can use less then 10 characters | ||||
|          */ | ||||
|         ret = C_("event list time", "All Day"); | ||||
|     } else { | ||||
|         switch (clockFormat) { | ||||
|         case '24h': | ||||
|             /* 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 */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%l:%M %p")); | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| function _getCalendarWeekForDate(date) { | ||||
|     // Based on the algorithms found here: | ||||
|     // http://en.wikipedia.org/wiki/Talk:ISO_week_date | ||||
|     let midnightDate = new Date(date.getFullYear(), date.getMonth(), date.getDate()); | ||||
|     // Need to get Monday to be 1 ... Sunday to be 7 | ||||
|     let dayOfWeek = 1 + ((midnightDate.getDay() + 6) % 7); | ||||
|     let nearestThursday = new Date(midnightDate.getFullYear(), midnightDate.getMonth(), | ||||
|                                    midnightDate.getDate() + (4 - dayOfWeek)); | ||||
|  | ||||
|     let jan1st = new Date(nearestThursday.getFullYear(), 0, 1); | ||||
|     let diffDate = nearestThursday - jan1st; | ||||
|     let dayNumber = Math.floor(Math.abs(diffDate) / MSECS_IN_DAY); | ||||
|     let weekNumber = Math.floor(dayNumber / 7) + 1; | ||||
|  | ||||
|     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. | ||||
|          * | ||||
|          * NOTE: These grid abbreviations are always shown together | ||||
|          * and in order, e.g. "S M T W T F S". | ||||
|          */ | ||||
|         C_("grid sunday", "S"), | ||||
|         /* Translators: Calendar grid abbreviation for Monday */ | ||||
|         C_("grid monday", "M"), | ||||
|         /* Translators: Calendar grid abbreviation for Tuesday */ | ||||
|         C_("grid tuesday", "T"), | ||||
|         /* Translators: Calendar grid abbreviation for Wednesday */ | ||||
|         C_("grid wednesday", "W"), | ||||
|         /* Translators: Calendar grid abbreviation for Thursday */ | ||||
|         C_("grid thursday", "T"), | ||||
|         /* Translators: Calendar grid abbreviation for Friday */ | ||||
|         C_("grid friday", "F"), | ||||
|         /* Translators: Calendar grid abbreviation for Saturday */ | ||||
|         C_("grid saturday", "S") | ||||
|     ]; | ||||
|     return abbreviations[dayNumber]; | ||||
| } | ||||
|  | ||||
| function _getEventDayAbbreviation(dayNumber) { | ||||
|     let abbreviations = [ | ||||
|         /* Translators: Event list abbreviation for Sunday. | ||||
|          * | ||||
|          * NOTE: These list abbreviations are normally not shown together | ||||
|          * so they need to be unique (e.g. Tuesday and Thursday cannot | ||||
|          * both be 'T'). | ||||
|          */ | ||||
|         C_("list sunday", "Su"), | ||||
|         /* Translators: Event list abbreviation for Monday */ | ||||
|         C_("list monday", "M"), | ||||
|         /* Translators: Event list abbreviation for Tuesday */ | ||||
|         C_("list tuesday", "T"), | ||||
|         /* Translators: Event list abbreviation for Wednesday */ | ||||
|         C_("list wednesday", "W"), | ||||
|         /* Translators: Event list abbreviation for Thursday */ | ||||
|         C_("list thursday", "Th"), | ||||
|         /* Translators: Event list abbreviation for Friday */ | ||||
|         C_("list friday", "F"), | ||||
|         /* Translators: Event list abbreviation for Saturday */ | ||||
|         C_("list saturday", "S") | ||||
|     ]; | ||||
|     return abbreviations[dayNumber]; | ||||
| } | ||||
|  | ||||
| // Abstraction for an appointment/event in a calendar | ||||
|  | ||||
| function CalendarEvent(date, end, summary, allDay) { | ||||
|     this._init(date, end, summary, allDay); | ||||
| } | ||||
|  | ||||
| CalendarEvent.prototype = { | ||||
|     _init: function(date, end, summary, allDay) { | ||||
|         this.date = date; | ||||
|         this.end = end; | ||||
|         this.summary = summary; | ||||
|         this.allDay = allDay; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| // Interface for appointments/events - e.g. the contents of a calendar | ||||
| // | ||||
|  | ||||
| // First, an implementation with no events | ||||
| function EmptyEventSource() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| EmptyEventSource.prototype = { | ||||
|     _init: function() { | ||||
|     }, | ||||
|  | ||||
|     requestRange: function(begin, end) { | ||||
|     }, | ||||
|  | ||||
|     getEvents: function(begin, end) { | ||||
|         let result = []; | ||||
|         return result; | ||||
|     }, | ||||
|  | ||||
|     hasEvents: function(day) { | ||||
|         return false; | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(EmptyEventSource.prototype); | ||||
|  | ||||
| const CalendarServerIface = { | ||||
|     name: 'org.gnome.Shell.CalendarServer', | ||||
|     methods: [{ name: 'GetEvents', | ||||
|                 inSignature: 'xxb', | ||||
|                 outSignature: 'a(sssbxxa{sv})' }], | ||||
|     signals: [{ name: 'Changed', | ||||
|                 inSignature: '' }] | ||||
| }; | ||||
|  | ||||
| const CalendarServer = function () { | ||||
|     this._init(); | ||||
| }; | ||||
|  | ||||
| CalendarServer.prototype = { | ||||
|      _init: function() { | ||||
|          DBus.session.proxifyObject(this, 'org.gnome.Shell.CalendarServer', '/org/gnome/Shell/CalendarServer'); | ||||
|      } | ||||
| }; | ||||
|  | ||||
| DBus.proxifyPrototype(CalendarServer.prototype, CalendarServerIface); | ||||
|  | ||||
| // an implementation that reads data from a session bus service | ||||
| function DBusEventSource(owner) { | ||||
|     this._init(owner); | ||||
| } | ||||
|  | ||||
| function _datesEqual(a, b) { | ||||
|     if (a < b) | ||||
|         return false; | ||||
|     else if (a > b) | ||||
|         return false; | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| function _dateIntervalsOverlap(a0, a1, b0, b1) | ||||
| { | ||||
|     if (a1 <= b0) | ||||
|         return false; | ||||
|     else if (b1 <= a0) | ||||
|         return false; | ||||
|     else | ||||
|         return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| DBusEventSource.prototype = { | ||||
|     _init: function(owner) { | ||||
|         this._resetCache(); | ||||
|  | ||||
|         this._dbusProxy = new CalendarServer(owner); | ||||
|         this._dbusProxy.connect('Changed', Lang.bind(this, this._onChanged)); | ||||
|  | ||||
|         DBus.session.watch_name('org.gnome.Shell.CalendarServer', | ||||
|                                 false, // do not launch a name-owner if none exists | ||||
|                                 Lang.bind(this, this._onNameAppeared), | ||||
|                                 Lang.bind(this, this._onNameVanished)); | ||||
|     }, | ||||
|  | ||||
|     _resetCache: function() { | ||||
|         this._events = []; | ||||
|         this._lastRequestBegin = null; | ||||
|         this._lastRequestEnd = null; | ||||
|     }, | ||||
|  | ||||
|     _onNameAppeared: function(owner) { | ||||
|         this._resetCache(); | ||||
|         this._loadEvents(true); | ||||
|     }, | ||||
|  | ||||
|     _onNameVanished: function(oldOwner) { | ||||
|         this._resetCache(); | ||||
|         this.emit('changed'); | ||||
|     }, | ||||
|  | ||||
|     _onChanged: function() { | ||||
|         this._loadEvents(false); | ||||
|     }, | ||||
|  | ||||
|     _onEventsReceived: function(appointments) { | ||||
|         let newEvents = []; | ||||
|         if (appointments != null) { | ||||
|             for (let n = 0; n < appointments.length; n++) { | ||||
|                 let a = appointments[n]; | ||||
|                 let date = new Date(a[4] * 1000); | ||||
|                 let end = new Date(a[5] * 1000); | ||||
|                 let summary = a[1]; | ||||
|                 let allDay = a[3]; | ||||
|                 let event = new CalendarEvent(date, end, summary, allDay); | ||||
|                 newEvents.push(event); | ||||
|             } | ||||
|             newEvents.sort(function(event1, event2) { | ||||
|                 return event1.date.getTime() - event2.date.getTime(); | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         this._events = newEvents; | ||||
|         this.emit('changed'); | ||||
|     }, | ||||
|  | ||||
|     _loadEvents: function(forceReload) { | ||||
|         if (this._curRequestBegin && this._curRequestEnd){ | ||||
|             let callFlags = 0; | ||||
|             if (forceReload) | ||||
|                 callFlags |= DBus.CALL_FLAG_START; | ||||
|             this._dbusProxy.GetEventsRemote(this._curRequestBegin.getTime() / 1000, | ||||
|                                             this._curRequestEnd.getTime() / 1000, | ||||
|                                             forceReload, | ||||
|                                             Lang.bind(this, this._onEventsReceived), | ||||
|                                             callFlags); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     requestRange: function(begin, end, forceReload) { | ||||
|         if (forceReload || !(_datesEqual(begin, this._lastRequestBegin) && _datesEqual(end, this._lastRequestEnd))) { | ||||
|             this._lastRequestBegin = begin; | ||||
|             this._lastRequestEnd = end; | ||||
|             this._curRequestBegin = begin; | ||||
|             this._curRequestEnd = end; | ||||
|             this._loadEvents(forceReload); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     getEvents: function(begin, end) { | ||||
|         let result = []; | ||||
|         for(let n = 0; n < this._events.length; n++) { | ||||
|             let event = this._events[n]; | ||||
|             if (_dateIntervalsOverlap (event.date, event.end, begin, end)) { | ||||
|                 result.push(event); | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     }, | ||||
|  | ||||
|     hasEvents: function(day) { | ||||
|         let dayBegin = _getBeginningOfDay(day); | ||||
|         let dayEnd = _getEndOfDay(day); | ||||
|  | ||||
|         let events = this.getEvents(dayBegin, dayEnd); | ||||
|  | ||||
|         if (events.length == 0) | ||||
|             return false; | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(DBusEventSource.prototype); | ||||
|  | ||||
| // Calendar: | ||||
| // @eventSource: is an object implementing the EventSource API, e.g. the | ||||
| // requestRange(), getEvents(), hasEvents() methods and the ::changed signal. | ||||
| function Calendar(eventSource) { | ||||
|     this._init(eventSource); | ||||
| } | ||||
|  | ||||
| Calendar.prototype = { | ||||
|     _init: function(eventSource) { | ||||
|         this._eventSource = eventSource; | ||||
|  | ||||
|         this._eventSource.connect('changed', Lang.bind(this, | ||||
|                                                        function() { | ||||
|                                                            this._update(false); | ||||
|                                                        })); | ||||
|  | ||||
|         // FIXME: This is actually the fallback method for GTK+ for the week start; | ||||
|         // GTK+ by preference uses nl_langinfo (NL_TIME_FIRST_WEEKDAY). We probably | ||||
|         // should add a C function so we can do the full handling. | ||||
|         this._weekStart = NaN; | ||||
|         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)); | ||||
|         this._useWeekdate = this._settings.get_boolean(SHOW_WEEKDATE_KEY); | ||||
|  | ||||
|         let weekStartString = Gettext_gtk30.gettext('calendar:week_start:0'); | ||||
|         if (weekStartString.indexOf('calendar:week_start:') == 0) { | ||||
|             this._weekStart = parseInt(weekStartString.substring(20)); | ||||
|         } | ||||
|  | ||||
|         if (isNaN(this._weekStart) || this._weekStart < 0 || this._weekStart > 6) { | ||||
|             log('Translation of "calendar:week_start:0" in GTK+ is not correct'); | ||||
|             this._weekStart = 0; | ||||
|         } | ||||
|  | ||||
|         // Find the ordering for month/year in the calendar heading | ||||
|         this._headerFormatWithoutYear = '%B'; | ||||
|         switch (Gettext_gtk30.gettext('calendar:MY')) { | ||||
|         case 'calendar:MY': | ||||
|             this._headerFormat = '%B %Y'; | ||||
|             break; | ||||
|         case 'calendar:YM': | ||||
|             this._headerFormat = '%Y %B'; | ||||
|             break; | ||||
|         default: | ||||
|             log('Translation of "calendar:MY" in GTK+ is not correct'); | ||||
|             this._headerFormat = '%B %Y'; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         // Start off with the current date | ||||
|         this._selectedDate = new Date(); | ||||
|  | ||||
|         this.actor = new St.Table({ homogeneous: false, | ||||
|                                     style_class: 'calendar', | ||||
|                                     reactive: true }); | ||||
|  | ||||
|         this.actor.connect('scroll-event', | ||||
|                            Lang.bind(this, this._onScroll)); | ||||
|  | ||||
|         this._buildHeader (); | ||||
|     }, | ||||
|  | ||||
|     // Sets the calendar to show a specific date | ||||
|     setDate: function(date, forceReload) { | ||||
|         if (!_sameDay(date, this._selectedDate)) { | ||||
|             this._selectedDate = date; | ||||
|             this._update(forceReload); | ||||
|             this.emit('selected-date-changed', new Date(this._selectedDate)); | ||||
|         } else { | ||||
|             if (forceReload) | ||||
|                 this._update(forceReload); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _buildHeader: function() { | ||||
|         let offsetCols = this._useWeekdate ? 1 : 0; | ||||
|         this.actor.destroy_children(); | ||||
|  | ||||
|         // Top line of the calendar '<| September 2009 |>' | ||||
|         this._topBox = new St.BoxLayout(); | ||||
|         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)); | ||||
|  | ||||
|         this._monthLabel = new St.Label({style_class: 'calendar-month-label'}); | ||||
|         this._topBox.add(this._monthLabel, { expand: true, x_fill: false, x_align: St.Align.MIDDLE }); | ||||
|  | ||||
|         let forward = new St.Button({ style_class: 'calendar-change-month-forward' }); | ||||
|         this._topBox.add(forward); | ||||
|         forward.connect('clicked', Lang.bind(this, this._onNextMonthButtonClicked)); | ||||
|  | ||||
|         // Add weekday labels... | ||||
|         // | ||||
|         // We need to figure out the abbreviated localized names for the days of the week; | ||||
|         // we do this by just getting the next 7 days starting from right now and then putting | ||||
|         // them in the right cell in the table. It doesn't matter if we add them in order | ||||
|         let iter = new Date(this._selectedDate); | ||||
|         iter.setSeconds(0); // Leap second protection. Hah! | ||||
|         iter.setHours(12); | ||||
|         for (let i = 0; i < 7; i++) { | ||||
|             // Could use iter.toLocaleFormat('%a') but that normally gives three characters | ||||
|             // and we want, ideally, a single character for e.g. S M T W T F S | ||||
|             let customDayAbbrev = _getCalendarDayAbbreviation(iter.getDay()); | ||||
|             let label = new St.Label({ style_class: 'calendar-day-base calendar-day-heading', | ||||
|                                        text: customDayAbbrev }); | ||||
|             this.actor.add(label, | ||||
|                            { row: 1, | ||||
|                              col: offsetCols + (7 + iter.getDay() - this._weekStart) % 7, | ||||
|                              x_fill: false, x_align: St.Align.MIDDLE }); | ||||
|             iter.setTime(iter.getTime() + MSECS_IN_DAY); | ||||
|         } | ||||
|  | ||||
|         // All the children after this are days, and get removed when we update the calendar | ||||
|         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) { | ||||
|         switch (event.get_scroll_direction()) { | ||||
|         case Clutter.ScrollDirection.UP: | ||||
|         case Clutter.ScrollDirection.LEFT: | ||||
|             this._onPrevMonthButtonClicked(); | ||||
|             break; | ||||
|         case Clutter.ScrollDirection.DOWN: | ||||
|         case Clutter.ScrollDirection.RIGHT: | ||||
|             this._onNextMonthButtonClicked(); | ||||
|             break; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onPrevMonthButtonClicked: function() { | ||||
|         let newDate = new Date(this._selectedDate); | ||||
|         let oldMonth = newDate.getMonth(); | ||||
|         if (oldMonth == 0) { | ||||
|             newDate.setMonth(11); | ||||
|             newDate.setFullYear(newDate.getFullYear() - 1); | ||||
|             if (newDate.getMonth() != 11) { | ||||
|                 let day = 32 - new Date(newDate.getFullYear() - 1, 11, 32).getDate(); | ||||
|                 newDate = new Date(newDate.getFullYear() - 1, 11, day); | ||||
|             } | ||||
|         } | ||||
|         else { | ||||
|             newDate.setMonth(oldMonth - 1); | ||||
|             if (newDate.getMonth() != oldMonth - 1) { | ||||
|                 let day = 32 - new Date(newDate.getFullYear(), oldMonth - 1, 32).getDate(); | ||||
|                 newDate = new Date(newDate.getFullYear(), oldMonth - 1, day); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this.setDate(newDate, false); | ||||
|    }, | ||||
|  | ||||
|    _onNextMonthButtonClicked: function() { | ||||
|         let newDate = new Date(this._selectedDate); | ||||
|         let oldMonth = newDate.getMonth(); | ||||
|         if (oldMonth == 11) { | ||||
|             newDate.setMonth(0); | ||||
|             newDate.setFullYear(newDate.getFullYear() + 1); | ||||
|             if (newDate.getMonth() != 0) { | ||||
|                 let day = 32 - new Date(newDate.getFullYear() + 1, 0, 32).getDate(); | ||||
|                 newDate = new Date(newDate.getFullYear() + 1, 0, day); | ||||
|             } | ||||
|         } | ||||
|         else { | ||||
|             newDate.setMonth(oldMonth + 1); | ||||
|             if (newDate.getMonth() != oldMonth + 1) { | ||||
|                 let day = 32 - new Date(newDate.getFullYear(), oldMonth + 1, 32).getDate(); | ||||
|                 newDate = new Date(newDate.getFullYear(), oldMonth + 1, day); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|        this.setDate(newDate, false); | ||||
|     }, | ||||
|  | ||||
|     _onSettingsChange: function() { | ||||
|         this._useWeekdate = this._settings.get_boolean(SHOW_WEEKDATE_KEY); | ||||
|         this._buildHeader(); | ||||
|         this._update(false); | ||||
|     }, | ||||
|  | ||||
|     _update: function(forceReload) { | ||||
|         let now = new Date(); | ||||
|  | ||||
|         if (_sameYear(this._selectedDate, now)) | ||||
|             this._monthLabel.text = this._selectedDate.toLocaleFormat(this._headerFormatWithoutYear); | ||||
|         else | ||||
|             this._monthLabel.text = this._selectedDate.toLocaleFormat(this._headerFormat); | ||||
|  | ||||
|         // Remove everything but the topBox and the weekday labels | ||||
|         let children = this.actor.get_children(); | ||||
|         for (let i = this._firstDayIndex; i < children.length; i++) | ||||
|             children[i].destroy(); | ||||
|  | ||||
|         // Start at the beginning of the week before the start of the month | ||||
|         let beginDate = new Date(this._selectedDate); | ||||
|         beginDate.setDate(1); | ||||
|         beginDate.setSeconds(0); | ||||
|         beginDate.setHours(12); | ||||
|         let daysToWeekStart = (7 + beginDate.getDay() - this._weekStart) % 7; | ||||
|         beginDate.setTime(beginDate.getTime() - daysToWeekStart * MSECS_IN_DAY); | ||||
|  | ||||
|         let iter = new Date(beginDate); | ||||
|         let row = 2; | ||||
|         while (true) { | ||||
|             let button = new St.Button({ label: iter.getDate().toString() }); | ||||
|  | ||||
|             let iterStr = iter.toUTCString(); | ||||
|             button.connect('clicked', Lang.bind(this, function() { | ||||
|                 let newlySelectedDate = new Date(iterStr); | ||||
|                 this.setDate(newlySelectedDate, false); | ||||
|             })); | ||||
|  | ||||
|             let hasEvents = this._eventSource.hasEvents(iter); | ||||
|             let styleClass = 'calendar-day-base calendar-day'; | ||||
|             if (_isWorkDay(iter)) | ||||
|                 styleClass += ' calendar-work-day' | ||||
|             else | ||||
|                 styleClass += ' calendar-nonwork-day' | ||||
|  | ||||
|             // Hack used in lieu of border-collapse - see gnome-shell.css | ||||
|             if (row == 2) | ||||
|                 styleClass = 'calendar-day-top ' + styleClass; | ||||
|             if (iter.getDay() == this._weekStart) | ||||
|                 styleClass = 'calendar-day-left ' + styleClass; | ||||
|  | ||||
|             if (_sameDay(now, iter)) | ||||
|                 styleClass += ' calendar-today'; | ||||
|             else if (iter.getMonth() != this._selectedDate.getMonth()) | ||||
|                 styleClass += ' calendar-other-month-day'; | ||||
|  | ||||
|             if (_sameDay(this._selectedDate, iter)) | ||||
|                 button.add_style_pseudo_class('active'); | ||||
|  | ||||
|             if (hasEvents) | ||||
|                 styleClass += ' calendar-day-with-events' | ||||
|  | ||||
|             button.style_class = styleClass; | ||||
|  | ||||
|             let offsetCols = this._useWeekdate ? 1 : 0; | ||||
|             this.actor.add(button, | ||||
|                            { row: row, col: offsetCols + (7 + iter.getDay() - this._weekStart) % 7 }); | ||||
|  | ||||
|             if (this._useWeekdate && iter.getDay() == 4) { | ||||
|                 let label = new St.Label({ text: _getCalendarWeekForDate(iter).toString(), | ||||
|                                            style_class: 'calendar-day-base calendar-week-number'}); | ||||
|                 this.actor.add(label, | ||||
|                                { row: row, col: 0, y_align: St.Align.MIDDLE }); | ||||
|             } | ||||
|  | ||||
|             iter.setTime(iter.getTime() + MSECS_IN_DAY); | ||||
|             if (iter.getDay() == this._weekStart) { | ||||
|                 // We stop on the first "first day of the week" after the month we are displaying | ||||
|                 if (iter.getMonth() > this._selectedDate.getMonth() || iter.getYear() > this._selectedDate.getYear()) | ||||
|                     break; | ||||
|                 row++; | ||||
|             } | ||||
|         } | ||||
|         // Signal to the event source that we are interested in events | ||||
|         // only from this date range | ||||
|         this._eventSource.requestRange(beginDate, iter, forceReload); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| Signals.addSignalMethods(Calendar.prototype); | ||||
|  | ||||
| function EventsList(eventSource) { | ||||
|     this._init(eventSource); | ||||
| } | ||||
|  | ||||
| EventsList.prototype = { | ||||
|     _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)); | ||||
|         let weekStartString = Gettext_gtk30.gettext('calendar:week_start:0'); | ||||
|         if (weekStartString.indexOf('calendar:week_start:') == 0) { | ||||
|             this._weekStart = parseInt(weekStartString.substring(20)); | ||||
|         } | ||||
|  | ||||
|         if (isNaN(this._weekStart) || | ||||
|                   this._weekStart < 0 || | ||||
|                   this._weekStart > 6) { | ||||
|             log('Translation of "calendar:week_start:0" in GTK+ is not correct'); | ||||
|             this._weekStart = 0; | ||||
|         } | ||||
|  | ||||
|         this._update(); | ||||
|     }, | ||||
|  | ||||
|     _addEvent: function(dayNameBox, timeBox, eventTitleBox, includeDayName, day, time, desc) { | ||||
|         if (includeDayName) { | ||||
|             dayNameBox.add(new St.Label( { style_class: 'events-day-dayname', | ||||
|                                            text: day } ), | ||||
|                            { x_fill: true } ); | ||||
|         } | ||||
|         timeBox.add(new St.Label( { style_class: 'events-day-time', | ||||
|                                     text: time} ), | ||||
|                     { x_fill: true } ); | ||||
|         eventTitleBox.add(new St.Label( { style_class: 'events-day-task', | ||||
|                                           text: desc} )); | ||||
|     }, | ||||
|  | ||||
|     _addPeriod: function(header, begin, end, includeDayName, showNothingScheduled) { | ||||
|         let events = this._eventSource.getEvents(begin, end); | ||||
|  | ||||
|         let clockFormat = this._desktopSettings.get_string(CLOCK_FORMAT_KEY);; | ||||
|  | ||||
|         if (events.length == 0 && !showNothingScheduled) | ||||
|             return; | ||||
|  | ||||
|         let vbox = new St.BoxLayout( {vertical: true} ); | ||||
|         this.actor.add(vbox); | ||||
|  | ||||
|         vbox.add(new St.Label({ style_class: 'events-day-header', text: header })); | ||||
|         let box = new St.BoxLayout({style_class: 'events-header-hbox'}); | ||||
|         let dayNameBox = new St.BoxLayout({ vertical: true, style_class: 'events-day-name-box' }); | ||||
|         let timeBox = new St.BoxLayout({ vertical: true, style_class: 'events-time-box' }); | ||||
|         let eventTitleBox = new St.BoxLayout({ vertical: true, style_class: 'events-event-box' }); | ||||
|         box.add(dayNameBox, {x_fill: false}); | ||||
|         box.add(timeBox, {x_fill: false}); | ||||
|         box.add(eventTitleBox, {expand: true}); | ||||
|         vbox.add(box); | ||||
|  | ||||
|         for (let n = 0; n < events.length; n++) { | ||||
|             let event = events[n]; | ||||
|             let dayString = _getEventDayAbbreviation(event.date.getDay()); | ||||
|             let timeString = _formatEventTime(event, clockFormat); | ||||
|             let summaryString = event.summary; | ||||
|             this._addEvent(dayNameBox, timeBox, eventTitleBox, includeDayName, dayString, timeString, summaryString); | ||||
|         } | ||||
|  | ||||
|         if (events.length == 0 && showNothingScheduled) { | ||||
|             let now = new Date(); | ||||
|             /* Translators: Text to show if there are no events */ | ||||
|             let nothingEvent = new CalendarEvent(now, now, _("Nothing Scheduled"), true); | ||||
|             let timeString = _formatEventTime(nothingEvent, clockFormat); | ||||
|             this._addEvent(dayNameBox, timeBox, eventTitleBox, false, "", timeString, nothingEvent.summary); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _showOtherDay: function(day) { | ||||
|         this.actor.destroy_children(); | ||||
|  | ||||
|         let dayBegin = _getBeginningOfDay(day); | ||||
|         let dayEnd = _getEndOfDay(day); | ||||
|  | ||||
|         let dayString; | ||||
|         let now = new Date(); | ||||
|         if (_sameYear(day, now)) | ||||
|             /* Translators: Shown on calendar heading when selected day occurs on current year */ | ||||
|             dayString = day.toLocaleFormat(C_("calendar heading", "%A, %B %d")); | ||||
|         else | ||||
|             /* Translators: Shown on calendar heading when selected day occurs on different year */ | ||||
|             dayString = day.toLocaleFormat(C_("calendar heading", "%A, %B %d, %Y")); | ||||
|         this._addPeriod(dayString, dayBegin, dayEnd, false, true); | ||||
|     }, | ||||
|  | ||||
|     _showToday: function() { | ||||
|         this.actor.destroy_children(); | ||||
|  | ||||
|         let now = new Date(); | ||||
|         let dayBegin = _getBeginningOfDay(now); | ||||
|         let dayEnd = _getEndOfDay(now); | ||||
|         this._addPeriod(_("Today"), dayBegin, dayEnd, false, true); | ||||
|  | ||||
|         let tomorrowBegin = new Date(dayBegin.getTime() + 86400 * 1000); | ||||
|         let tomorrowEnd = new Date(dayEnd.getTime() + 86400 * 1000); | ||||
|         this._addPeriod(_("Tomorrow"), tomorrowBegin, tomorrowEnd, false, true); | ||||
|  | ||||
|         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 + 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 | ||||
|              * "Next week" and include events up until and including *next* | ||||
|              * Saturday/Sunday | ||||
|              */ | ||||
|             let nextWeekBegin = new Date(dayBegin.getTime() + 2 * 86400 * 1000); | ||||
|             let nextWeekEnd = new Date(dayEnd.getTime() + (13 + this._weekStart - dayEnd.getDay()) * 86400 * 1000); | ||||
|             this._addPeriod(_("Next week"), nextWeekBegin, nextWeekEnd, true, false); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Sets the event list to show events from a specific date | ||||
|     setDate: function(date) { | ||||
|         if (!_sameDay(date, this._date)) { | ||||
|             this._date = date; | ||||
|             this._update(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _update: function() { | ||||
|         let today = new Date(); | ||||
|         if (_sameDay (this._date, today)) { | ||||
|             this._showToday(); | ||||
|         } else { | ||||
|             this._showOtherDay(this._date); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										413
									
								
								js/ui/chrome.js
									
									
									
									
									
								
							
							
						
						| @@ -1,25 +1,17 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| // This manages the shell "chrome"; the UI that's visible in the | ||||
| // normal mode (ie, outside the Overview), that surrounds the main | ||||
| // workspace content. | ||||
|  | ||||
| const defaultParams = { | ||||
|     visibleInOverview: false, | ||||
|     visibleInFullscreen: false, | ||||
|     affectsStruts: true, | ||||
|     affectsInputRegion: true | ||||
| }; | ||||
|  | ||||
| function Chrome() { | ||||
|     this._init(); | ||||
| } | ||||
| @@ -27,17 +19,15 @@ function Chrome() { | ||||
| Chrome.prototype = { | ||||
|     _init: function() { | ||||
|         // The group itself has zero size so it doesn't interfere with DND | ||||
|         this.actor = new Shell.GenericContainer({ width: 0, height: 0 }); | ||||
|         Main.uiGroup.add_actor(this.actor); | ||||
|         this.actor.connect('allocate', Lang.bind(this, this._allocated)); | ||||
|         this.actor = new Clutter.Group({ width: 0, height: 0 }); | ||||
|         global.stage.add_actor(this.actor); | ||||
|         this.nonOverviewActor = new Clutter.Group(); | ||||
|         this.actor.add_actor(this.nonOverviewActor); | ||||
|  | ||||
|         this._monitors = []; | ||||
|         this._inOverview = false; | ||||
|         this._obscuredByFullscreen = false; | ||||
|  | ||||
|         this._trackedActors = []; | ||||
|  | ||||
|         global.screen.connect('monitors-changed', | ||||
|                               Lang.bind(this, this._monitorsChanged)); | ||||
|         global.screen.connect('restacked', | ||||
|                               Lang.bind(this, this._windowsRestacked)); | ||||
|  | ||||
| @@ -50,20 +40,21 @@ Chrome.prototype = { | ||||
|         Main.overview.connect('hidden', | ||||
|                              Lang.bind(this, this._overviewHidden)); | ||||
|  | ||||
|         this._updateMonitors(); | ||||
|         this._updateFullscreen(); | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _allocated: function(actor, box, flags) { | ||||
|         let children = this.actor.get_children(); | ||||
|         for (let i = 0; i < children.length; i++) | ||||
|             children[i].allocate_preferred_size(flags); | ||||
|     _verifyAncestry: function(actor, ancestor) { | ||||
|         while (actor) { | ||||
|             if (actor == ancestor) | ||||
|                 return true; | ||||
|             actor = actor.get_parent(); | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     // addActor: | ||||
|     // @actor: an actor to add to the chrome layer | ||||
|     // @params: (optional) additional params | ||||
|     // @shapeActor: optional "shape actor". | ||||
|     // | ||||
|     // Adds @actor to the chrome layer and extends the input region | ||||
|     // and window manager struts to include it. (Window manager struts | ||||
| @@ -73,60 +64,59 @@ Chrome.prototype = { | ||||
|     // in its visibility will affect the input region, but NOT the | ||||
|     // struts. | ||||
|     // | ||||
|     // If %visibleInOverview is %true in @params, @actor will remain | ||||
|     // visible when the overview is brought up. Otherwise it will | ||||
|     // automatically be hidden. Likewise, if %visibleInFullscreen is | ||||
|     // %true, the actor will be visible even when a fullscreen window | ||||
|     // should be covering it. | ||||
|     // | ||||
|     // If %affectsStruts or %affectsInputRegion is %false, the actor | ||||
|     // will not have the indicated effect. | ||||
|     addActor: function(actor, params) { | ||||
|         this.actor.add_actor(actor); | ||||
|         this._trackActor(actor, params); | ||||
|     // If @shapeActor is provided, it will be used instead of @actor | ||||
|     // for the input region/strut shape. (This lets you have things like | ||||
|     // drop shadows in @actor that don't affect the struts.) It must | ||||
|     // be a child of @actor. Alternatively, you can pass %null for | ||||
|     // @shapeActor to indicate that @actor should not affect the input | ||||
|     // region or struts at all. | ||||
|     addActor: function(actor, shapeActor) { | ||||
|         if (shapeActor === undefined) | ||||
|             shapeActor = actor; | ||||
|         else if (shapeActor && !this._verifyAncestry(shapeActor, actor)) | ||||
|             throw new Error('shapeActor is not a descendent of actor'); | ||||
|  | ||||
|         this.nonOverviewActor.add_actor(actor); | ||||
|  | ||||
|         if (shapeActor) | ||||
|             this._trackActor(shapeActor, true, true); | ||||
|     }, | ||||
|  | ||||
|     // trackActor: | ||||
|     // @actor: a descendant of the chrome to begin tracking | ||||
|     // @params: parameters describing how to track @actor | ||||
|     // setVisibleInOverview: | ||||
|     // @actor: an actor in the chrome layer | ||||
|     // @visible: Overview visibility | ||||
|     // | ||||
|     // Tells the chrome to track @actor, which must be a descendant | ||||
|     // of an actor added via addActor(). This can be used to extend the | ||||
|     // struts or input region to cover specific children. | ||||
|     // | ||||
|     // @params can have any of the same values as in addActor(), though | ||||
|     // some possibilities don't make sense (eg, trying to have a | ||||
|     // %visibleInOverview child of a non-%visibleInOverview parent). | ||||
|     // By default, @actor has the same params as its chrome ancestor. | ||||
|     trackActor: function(actor, params) { | ||||
|         let ancestor = actor.get_parent(); | ||||
|         let index = this._findActor(ancestor); | ||||
|         while (ancestor && index == -1) { | ||||
|             ancestor = ancestor.get_parent(); | ||||
|             index = this._findActor(ancestor); | ||||
|         } | ||||
|         if (!ancestor) | ||||
|     // By default, actors in the chrome layer are automatically hidden | ||||
|     // when the Overview is shown. This can be used to override that | ||||
|     // behavior | ||||
|     setVisibleInOverview: function(actor, visible) { | ||||
|         if (!this._verifyAncestry(actor, this.actor)) | ||||
|             throw new Error('actor is not a descendent of the chrome layer'); | ||||
|  | ||||
|         let ancestorData = this._trackedActors[index]; | ||||
|         if (!params) | ||||
|             params = {}; | ||||
|         // We can't use Params.parse here because we want to drop | ||||
|         // the extra values like ancestorData.actor | ||||
|         for (let prop in defaultParams) { | ||||
|             if (!params[prop]) | ||||
|                 params[prop] = ancestorData[prop]; | ||||
|         } | ||||
|  | ||||
|         this._trackActor(actor, params); | ||||
|         if (visible) | ||||
|             actor.reparent(this.actor); | ||||
|         else | ||||
|             actor.reparent(this.nonOverviewActor); | ||||
|     }, | ||||
|  | ||||
|     // untrackActor: | ||||
|     // @actor: an actor previously tracked via trackActor() | ||||
|     // addInputRegionActor: | ||||
|     // @actor: an actor to add to the stage input region | ||||
|     // | ||||
|     // Undoes the effect of trackActor() | ||||
|     untrackActor: function(actor) { | ||||
|         this._untrackActor(actor); | ||||
|     // Adds @actor to the stage input region, as with addActor(), but | ||||
|     // for actors that are already descendants of the chrome layer. | ||||
|     addInputRegionActor: function(actor) { | ||||
|         if (!this._verifyAncestry(actor, this.actor)) | ||||
|             throw new Error('actor is not a descendent of the chrome layer'); | ||||
|  | ||||
|         this._trackActor(actor, true, false); | ||||
|     }, | ||||
|  | ||||
|     // removeInputRegionActor: | ||||
|     // @actor: an actor previously added to the stage input region | ||||
|     // | ||||
|     // Undoes the effect of addInputRegionActor() | ||||
|     removeInputRegionActor: function(actor) { | ||||
|         this._untrackActor(actor, true, false); | ||||
|     }, | ||||
|  | ||||
|     // removeActor: | ||||
| @@ -134,8 +124,11 @@ Chrome.prototype = { | ||||
|     // | ||||
|     // Removes @actor from the chrome layer | ||||
|     removeActor: function(actor) { | ||||
|         this.actor.remove_actor(actor); | ||||
|         this._untrackActor(actor); | ||||
|         if (actor.get_parent() == this.nonOverviewActor) | ||||
|             this.nonOverviewActor.remove_actor(actor); | ||||
|         else | ||||
|             this.actor.remove_actor(actor); | ||||
|         this._untrackActor(actor, true, true); | ||||
|     }, | ||||
|  | ||||
|     _findActor: function(actor) { | ||||
| @@ -147,126 +140,92 @@ Chrome.prototype = { | ||||
|         return -1; | ||||
|     }, | ||||
|  | ||||
|     _trackActor: function(actor, params) { | ||||
|         if (this._findActor(actor) != -1) | ||||
|             throw new Error('trying to re-track existing chrome actor'); | ||||
|     _trackActor: function(actor, inputRegion, strut) { | ||||
|         let actorData; | ||||
|         let i = this._findActor(actor); | ||||
|  | ||||
|         if (i != -1) { | ||||
|             actorData = this._trackedActors[i]; | ||||
|             if (inputRegion) | ||||
|                 actorData.inputRegion++; | ||||
|             if (strut) | ||||
|                 actorData.strut++; | ||||
|             if (!inputRegion && !strut) | ||||
|                 actorData.children++; | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         actorData = { actor: actor, | ||||
|                       inputRegion: inputRegion ? 1 : 0, | ||||
|                       strut: strut ? 1 : 0, | ||||
|                       children: 0 }; | ||||
|  | ||||
|         let actorData = Params.parse(params, defaultParams); | ||||
|         actorData.actor = actor; | ||||
|         actorData.visibleId = actor.connect('notify::visible', | ||||
|                                             Lang.bind(this, this._queueUpdateRegions)); | ||||
|         actorData.allocationId = actor.connect('notify::allocation', | ||||
|                                                Lang.bind(this, this._queueUpdateRegions)); | ||||
|         actorData.parentSetId = actor.connect('parent-set', | ||||
|                                               Lang.bind(this, this._actorReparented)); | ||||
|         // Note that destroying actor will unset its parent, so we don't | ||||
|         // need to connect to 'destroy' too. | ||||
|  | ||||
|         this._trackedActors.push(actorData); | ||||
|         this._queueUpdateRegions(); | ||||
|  | ||||
|         actor = actor.get_parent(); | ||||
|         if (actor != this.actor && actor != this.nonOverviewActor) | ||||
|             this._trackActor(actor, false, false); | ||||
|  | ||||
|         if (inputRegion || strut) | ||||
|             this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _untrackActor: function(actor) { | ||||
|     _untrackActor: function(actor, inputRegion, strut) { | ||||
|         let i = this._findActor(actor); | ||||
|  | ||||
|         if (i == -1) | ||||
|             return; | ||||
|         let actorData = this._trackedActors[i]; | ||||
|  | ||||
|         this._trackedActors.splice(i, 1); | ||||
|         actor.disconnect(actorData.visibleId); | ||||
|         actor.disconnect(actorData.allocationId); | ||||
|         actor.disconnect(actorData.parentSetId); | ||||
|         if (inputRegion) | ||||
|             actorData.inputRegion--; | ||||
|         if (strut) | ||||
|             actorData.strut--; | ||||
|         if (!inputRegion && !strut) | ||||
|             actorData.children--; | ||||
|  | ||||
|         this._queueUpdateRegions(); | ||||
|         if (actorData.inputRegion <= 0 && actorData.strut <= 0 && actorData.children <= 0) { | ||||
|             this._trackedActors.splice(i, 1); | ||||
|             actor.disconnect(actorData.visibleId); | ||||
|             actor.disconnect(actorData.allocationId); | ||||
|             actor.disconnect(actorData.parentSetId); | ||||
|  | ||||
|             actor = actor.get_parent(); | ||||
|             if (actor && actor != this.actor && actor != this.nonOverviewActor) | ||||
|                 this._untrackActor(actor, false, false); | ||||
|         } | ||||
|  | ||||
|         if (inputRegion || strut) | ||||
|             this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _actorReparented: function(actor, oldParent) { | ||||
|         if (!this.actor.contains(actor)) | ||||
|             this._untrackActor(actor); | ||||
|     }, | ||||
|  | ||||
|     _updateVisibility: function() { | ||||
|         for (let i = 0; i < this._trackedActors.length; i++) { | ||||
|             let actorData = this._trackedActors[i]; | ||||
|             if (this._inOverview && !actorData.visibleInOverview) | ||||
|                 this.actor.set_skip_paint(actorData.actor, true); | ||||
|             else if (!this._inOverview && !actorData.visibleInFullscreen && | ||||
|                      this._findMonitorForActor(actorData.actor).inFullscreen) | ||||
|                 this.actor.set_skip_paint(actorData.actor, true); | ||||
|             else | ||||
|                 this.actor.set_skip_paint(actorData.actor, false); | ||||
|         if (this._verifyAncestry(actor, this.actor)) { | ||||
|             let newParent = actor.get_parent(); | ||||
|             if (newParent != this.actor && newParent != this.nonOverviewActor) | ||||
|                 this._trackActor(newParent, false, false); | ||||
|         } | ||||
|         if (oldParent != this.actor && oldParent != this.nonOverviewActor) | ||||
|             this._untrackActor(oldParent, false, false); | ||||
|     }, | ||||
|  | ||||
|     _overviewShowing: function() { | ||||
|         this._inOverview = true; | ||||
|         this._updateVisibility(); | ||||
|         this.actor.show(); | ||||
|         this.nonOverviewActor.hide(); | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _overviewHidden: function() { | ||||
|         this._inOverview = false; | ||||
|         this._updateVisibility(); | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _updateMonitors: function() { | ||||
|         let monitors = global.get_monitors(); | ||||
|         let primary = global.get_primary_monitor(); | ||||
|         this._monitors = monitors; | ||||
|         for (let i = 0; i < monitors.length; i++) { | ||||
|             let monitor = monitors[i]; | ||||
|             if (monitor.x == primary.x && | ||||
|                 monitor.y == primary.y && | ||||
|                 monitor.width == primary.width && | ||||
|                 monitor.height == primary.height) | ||||
|                 this._primaryMonitor = monitor; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _findMonitorForRect: function(x, y, w, h) { | ||||
|         // First look at what monitor the center of the rectangle is at | ||||
|         let cx = x + w/2; | ||||
|         let cy = y + h/2; | ||||
|         for (let i = 0; i < this._monitors.length; i++) { | ||||
|             let monitor = this._monitors[i]; | ||||
|             if (cx >= monitor.x && cx < monitor.x + monitor.width && | ||||
|                 cy >= monitor.y && cy < monitor.y + monitor.height) | ||||
|                 return monitor; | ||||
|         } | ||||
|         // If the center is not on a monitor, return the first overlapping monitor | ||||
|         for (let i = 0; i < this._monitors.length; i++) { | ||||
|             let monitor = this._monitors[i]; | ||||
|             if (x + w > monitor.x && x < monitor.x + monitor.width && | ||||
|                 y + h > monitor.y && y < monitor.y + monitor.height) | ||||
|                 return monitor; | ||||
|         } | ||||
|         // otherwise on no monitor | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     _findMonitorForWindow: function(window) { | ||||
|         return this._findMonitorForRect(window.x, window.y, window.width, window.height); | ||||
|     }, | ||||
|  | ||||
|     // This call guarantees that we return some monitor to simplify usage of it | ||||
|     // In practice all tracked actors should be visible on some monitor anyway | ||||
|     _findMonitorForActor: function(actor) { | ||||
|         let [x, y] = actor.get_transformed_position(); | ||||
|         let [w, h] = actor.get_transformed_size(); | ||||
|         let monitor = this._findMonitorForRect(x, y, w, h); | ||||
|         if (monitor) | ||||
|             return monitor; | ||||
|         return this._primaryMonitor; // Not on any monitor, pretend its on the primary | ||||
|     }, | ||||
|  | ||||
|     _monitorsChanged: function() { | ||||
|         this._updateMonitors(); | ||||
|  | ||||
|         // Update everything that depends on monitor positions | ||||
|         this._updateFullscreen(); | ||||
|         this._updateVisibility(); | ||||
|         if (this._obscuredByFullscreen) | ||||
|             this.actor.hide(); | ||||
|         this.nonOverviewActor.show(); | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
| @@ -276,17 +235,13 @@ Chrome.prototype = { | ||||
|                                                        Meta.PRIORITY_BEFORE_REDRAW); | ||||
|     }, | ||||
|  | ||||
|     _updateFullscreen: function() { | ||||
|         let windows = Main.getWindowActorsForWorkspace(global.screen.get_active_workspace_index()); | ||||
|  | ||||
|         // Reset all monitors to not fullscreen | ||||
|         for (let i = 0; i < this._monitors.length; i++) | ||||
|             this._monitors[i].inFullscreen = false; | ||||
|     _windowsRestacked: function() { | ||||
|         let windows = global.get_windows(); | ||||
|  | ||||
|         // The chrome layer should be visible unless there is a window | ||||
|         // with layer FULLSCREEN, or a window with layer | ||||
|         // OVERRIDE_REDIRECT that covers the whole screen. | ||||
|         // ('override_redirect' is not actually a layer above all | ||||
|         // ("override_redirect" is not actually a layer above all | ||||
|         // other windows, but this seems to be how mutter treats it | ||||
|         // currently...) If we wanted to be extra clever, we could | ||||
|         // figure out when an OVERRIDE_REDIRECT window was trying to | ||||
| @@ -295,51 +250,30 @@ Chrome.prototype = { | ||||
|  | ||||
|         // @windows is sorted bottom to top. | ||||
|  | ||||
|         this._obscuredByFullscreen = false; | ||||
|         for (let i = windows.length - 1; i > -1; i--) { | ||||
|             let window = windows[i]; | ||||
|             let layer = window.get_meta_window().get_layer(); | ||||
|             let layer = windows[i].get_meta_window().get_layer(); | ||||
|  | ||||
|             if (layer == Meta.StackLayer.FULLSCREEN) { | ||||
|                 let monitor = this._findMonitorForWindow(window); | ||||
|                 if (monitor) | ||||
|                     monitor.inFullscreen = true; | ||||
|             } | ||||
|             if (layer == Meta.StackLayer.OVERRIDE_REDIRECT) { | ||||
|                 let monitor = this._findMonitorForWindow(window); | ||||
|                 if (monitor && | ||||
|                     window.x <= monitor.x && | ||||
|                     window.x + window.width >= monitor.x + monitor.width && | ||||
|                     window.y <= monitor.y && | ||||
|                     window.y + window.height >= monitor.y + monitor.height) | ||||
|                     monitor.inFullscreen = true; | ||||
|                 if (windows[i].x <= 0 && | ||||
|                     windows[i].x + windows[i].width >= global.screen_width && | ||||
|                     windows[i].y <= 0 && | ||||
|                     windows[i].y + windows[i].height >= global.screen_height) { | ||||
|                     this._obscuredByFullscreen = true; | ||||
|                     break; | ||||
|                 } | ||||
|             } else if (layer == Meta.StackLayer.FULLSCREEN) { | ||||
|                 this._obscuredByFullscreen = true; | ||||
|                 break; | ||||
|             } else | ||||
|                 break; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _windowsRestacked: function() { | ||||
|         let wasInFullscreen = []; | ||||
|         for (let i = 0; i < this._monitors.length; i++) | ||||
|             wasInFullscreen[i] = this._monitors[i].inFullscreen; | ||||
|  | ||||
|         this._updateFullscreen(); | ||||
|  | ||||
|         let changed = false; | ||||
|         for (let i = 0; i < wasInFullscreen.length; i++) { | ||||
|             if (wasInFullscreen[i] != this._monitors[i].inFullscreen) { | ||||
|                 changed = true; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         if (changed) { | ||||
|             this._updateVisibility(); | ||||
|         let shouldBeVisible = !this._obscuredByFullscreen || Main.overview.visible; | ||||
|         if (this.actor.visible != shouldBeVisible) { | ||||
|             this.actor.visible = shouldBeVisible; | ||||
|             this._queueUpdateRegions(); | ||||
|         } | ||||
|  | ||||
|         // Figure out where the pointer is in case we lost track of | ||||
|         // it during a grab. (In particular, if a trayicon popup menu | ||||
|         // is dismissed, see if we need to close the message tray.) | ||||
|         global.sync_pointer(); | ||||
|     }, | ||||
|  | ||||
|     _updateRegions: function() { | ||||
| @@ -349,7 +283,7 @@ Chrome.prototype = { | ||||
|  | ||||
|         for (i = 0; i < this._trackedActors.length; i++) { | ||||
|             let actorData = this._trackedActors[i]; | ||||
|             if (!actorData.affectsInputRegion && !actorData.affectsStruts) | ||||
|             if (!actorData.inputRegion && !actorData.strut) | ||||
|                 continue; | ||||
|  | ||||
|             let [x, y] = actorData.actor.get_transformed_position(); | ||||
| @@ -360,83 +294,47 @@ Chrome.prototype = { | ||||
|             h = Math.round(h); | ||||
|             let rect = new Meta.Rectangle({ x: x, y: y, width: w, height: h}); | ||||
|  | ||||
|             if (actorData.affectsInputRegion && | ||||
|                 actorData.actor.get_paint_visibility() && | ||||
|                 !this.actor.get_skip_paint(actorData.actor)) | ||||
|             if (actorData.inputRegion && actorData.actor.get_paint_visibility()) | ||||
|                 rects.push(rect); | ||||
|  | ||||
|             if (!actorData.affectsStruts) | ||||
|             if (!actorData.strut) | ||||
|                 continue; | ||||
|  | ||||
|             // Limit struts to the size of the screen | ||||
|             let x1 = Math.max(x, 0); | ||||
|             let x2 = Math.min(x + w, global.screen_width); | ||||
|             let y1 = Math.max(y, 0); | ||||
|             let y2 = Math.min(y + h, global.screen_height); | ||||
|  | ||||
|             // NetWM struts are not really powerful enought to handle | ||||
|             // a multi-monitor scenario, they only describe what happens | ||||
|             // around the outer sides of the full display region. However | ||||
|             // it can describe a partial region along each side, so | ||||
|             // we can support having the struts only affect the | ||||
|             // primary monitor. This should be enough as we only have | ||||
|             // chrome affecting the struts on the primary monitor so | ||||
|             // far. | ||||
|             // | ||||
|             // Metacity wants to know what side of the screen the | ||||
|             // strut is considered to be attached to. If the actor is | ||||
|             // only touching one edge, or is touching the entire | ||||
|             // border of the primary monitor, then it's obvious which | ||||
|             // side to call it. If it's in a corner, we pick a side | ||||
|             // width/height of one edge, then it's obvious which side | ||||
|             // to call it. If it's in a corner, we pick a side | ||||
|             // arbitrarily. If it doesn't touch any edges, or it spans | ||||
|             // the width/height across the middle of the screen, then | ||||
|             // we don't create a strut for it at all. | ||||
|             let side; | ||||
|             let primary = this._primaryMonitor; | ||||
|             if (x1 <= primary.x && x2 >= primary.x + primary.width) { | ||||
|                 if (y1 <= primary.y) | ||||
|             if (w >= global.screen_width) { | ||||
|                 if (y <= 0) | ||||
|                     side = Meta.Side.TOP; | ||||
|                 else if (y2 >= primary.y + primary.height) | ||||
|                 else if (y + h >= global.screen_height) | ||||
|                     side = Meta.Side.BOTTOM; | ||||
|                 else | ||||
|                     continue; | ||||
|             } else if (y1 <= primary.y && y2 >= primary.y + primary.height) { | ||||
|                 if (x1 <= 0) | ||||
|             } else if (h >= global.screen_height) { | ||||
|                 if (x <= 0) | ||||
|                     side = Meta.Side.LEFT; | ||||
|                 else if (x2 >= global.screen_width) | ||||
|                 else if (x + w >= global.screen_width) | ||||
|                     side = Meta.Side.RIGHT; | ||||
|                 else | ||||
|                     continue; | ||||
|             } else if (x1 <= 0) | ||||
|             } else if (x <= 0) | ||||
|                 side = Meta.Side.LEFT; | ||||
|             else if (y1 <= 0) | ||||
|             else if (y <= 0) | ||||
|                 side = Meta.Side.TOP; | ||||
|             else if (x2 >= global.screen_width) | ||||
|             else if (x + w >= global.screen_width) | ||||
|                 side = Meta.Side.RIGHT; | ||||
|             else if (y2 >= global.screen_height) | ||||
|             else if (y + h >= global.screen_height) | ||||
|                 side = Meta.Side.BOTTOM; | ||||
|             else | ||||
|                 continue; | ||||
|  | ||||
|             // Ensure that the strut rects goes all the way to the screen edge, | ||||
|             // as this really what mutter expects. | ||||
|             switch (side) { | ||||
|             case Meta.Side.TOP: | ||||
|                 y1 = 0; | ||||
|                 break; | ||||
|             case Meta.Side.BOTTOM: | ||||
|                 y2 = global.screen_height; | ||||
|                 break; | ||||
|             case Meta.Side.LEFT: | ||||
|                 x1 = 0; | ||||
|                 break; | ||||
|             case Meta.Side.RIGHT: | ||||
|                 x2 = global.screen_width; | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|             let strutRect = new Meta.Rectangle({ x: x1, y: y1, width: x2 - x1, height: y2 - y1}); | ||||
|             let strut = new Meta.Strut({ rect: strutRect, side: side }); | ||||
|             let strut = new Meta.Strut({ rect: rect, side: side }); | ||||
|             struts.push(strut); | ||||
|         } | ||||
|  | ||||
| @@ -451,4 +349,3 @@ Chrome.prototype = { | ||||
|         return false; | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(Chrome.prototype); | ||||
|   | ||||
| @@ -1,328 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 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 Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const POPUP_APPICON_SIZE = 96; | ||||
| const POPUP_FADE_TIME = 0.1; // seconds | ||||
|  | ||||
| const SortGroup = { | ||||
|     TOP:    0, | ||||
|     MIDDLE: 1, | ||||
|     BOTTOM: 2 | ||||
| }; | ||||
|  | ||||
| function CtrlAltTabManager() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| CtrlAltTabManager.prototype = { | ||||
|     _init: function() { | ||||
|         this._items = []; | ||||
|         this._focusManager = St.FocusManager.get_for_stage(global.stage); | ||||
|     }, | ||||
|  | ||||
|     addGroup: function(root, name, icon, params) { | ||||
|         let item = Params.parse(params, { sortGroup: SortGroup.MIDDLE, | ||||
|                                           proxy: root, | ||||
|                                           focusCallback: null }); | ||||
|  | ||||
|         item.root = root; | ||||
|         item.name = name; | ||||
|         item.iconName = icon; | ||||
|  | ||||
|         this._items.push(item); | ||||
|         root.connect('destroy', Lang.bind(this, function() { this.removeGroup(root); })); | ||||
|         this._focusManager.add_group(root); | ||||
|     }, | ||||
|  | ||||
|     removeGroup: function(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); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     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, | ||||
|     // and everything else in between, sorted by X coordinate, so that | ||||
|     // they will have the same left-to-right ordering in the | ||||
|     // Ctrl-Alt-Tab dialog as they do onscreen. | ||||
|     _sortItems: function(a, b) { | ||||
|         if (a.sortGroup != b.sortGroup) | ||||
|             return a.sortGroup - b.sortGroup; | ||||
|  | ||||
|         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 a.x - b.x; | ||||
|     }, | ||||
|  | ||||
|     popup: function(backwards) { | ||||
|         // Start with the set of focus groups that are currently mapped | ||||
|         let items = this._items.filter(function (item) { return item.proxy.mapped; }); | ||||
|  | ||||
|         // And add the windows metacity would show in its Ctrl-Alt-Tab list | ||||
|         if (!Main.overview.visible) { | ||||
|             let screen = global.screen; | ||||
|             let display = screen.get_display(); | ||||
|             let windows = display.get_tab_list(Meta.TabList.DOCKS, screen, screen.get_active_workspace ()); | ||||
|             let windowTracker = Shell.WindowTracker.get_default(); | ||||
|             let textureCache = St.TextureCache.get_default(); | ||||
|             for (let i = 0; i < windows.length; i++) { | ||||
|                 let icon; | ||||
|                 let app = windowTracker.get_window_app(windows[i]); | ||||
|                 if (app) | ||||
|                     icon = app.create_icon_texture(POPUP_APPICON_SIZE); | ||||
|                 else | ||||
|                     icon = textureCache.bind_pixbuf_property(windows[i], 'icon'); | ||||
|                 items.push({ window: windows[i], | ||||
|                              name: windows[i].title, | ||||
|                              iconActor: icon, | ||||
|                              sortGroup: SortGroup.MIDDLE }); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!items.length) | ||||
|             return; | ||||
|  | ||||
|         items.sort(Lang.bind(this, this._sortItems)); | ||||
|         new CtrlAltTabPopup().show(items, backwards); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| function mod(a, b) { | ||||
|     return (a + b) % b; | ||||
| } | ||||
|  | ||||
| function CtrlAltTabPopup() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| CtrlAltTabPopup.prototype = { | ||||
|     _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._selection = 0; | ||||
|  | ||||
|         Main.uiGroup.add_actor(this.actor); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function (actor, forHeight, alloc) { | ||||
|         let primary = global.get_primary_monitor(); | ||||
|  | ||||
|         alloc.min_size = primary.width; | ||||
|         alloc.natural_size = primary.width; | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function (actor, forWidth, alloc) { | ||||
|         let primary = global.get_primary_monitor(); | ||||
|  | ||||
|         alloc.min_size = primary.height; | ||||
|         alloc.natural_size = primary.height; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function (actor, box, flags) { | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         let primary = global.get_primary_monitor(); | ||||
|  | ||||
|         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.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) { | ||||
|         if (!Main.pushModal(this.actor)) | ||||
|             return false; | ||||
|         this._haveModal = true; | ||||
|  | ||||
|         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 & Gdk.ModifierType.MOD1_MASK)) { | ||||
|             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; | ||||
|     }, | ||||
|  | ||||
|     _next : function() { | ||||
|         return mod(this._selection + 1, this._items.length); | ||||
|     }, | ||||
|  | ||||
|     _previous : function() { | ||||
|         return mod(this._selection - 1, this._items.length); | ||||
|     }, | ||||
|  | ||||
|     _keyPressEvent : function(actor, event) { | ||||
|         let keysym = event.get_key_symbol(); | ||||
|         let shift = (Shell.get_event_state(event) & 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; | ||||
|     }, | ||||
|  | ||||
|     _keyReleaseEvent : function(actor, event) { | ||||
|         let [x, y, mods] = global.get_pointer(); | ||||
|         let state = mods & Clutter.ModifierType.MOD1_MASK; | ||||
|  | ||||
|         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); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| function CtrlAltTabSwitcher(items) { | ||||
|     this._init(items); | ||||
| } | ||||
|  | ||||
| CtrlAltTabSwitcher.prototype = { | ||||
|     __proto__ : AltTab.SwitcherList.prototype, | ||||
|  | ||||
|     _init : function(items) { | ||||
|         AltTab.SwitcherList.prototype._init.call(this, true); | ||||
|  | ||||
|         for (let i = 0; i < items.length; i++) | ||||
|             this._addIcon(items[i]); | ||||
|     }, | ||||
|  | ||||
|     _addIcon : function(item) { | ||||
|         let box = new St.BoxLayout({ style_class: 'alt-tab-app', | ||||
|                                      vertical: true }); | ||||
|  | ||||
|         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 } ); | ||||
|  | ||||
|         let text = new St.Label({ text: item.name }); | ||||
|         box.add(text, { x_fill: false }); | ||||
|  | ||||
|         this.addItem(box); | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										1577
									
								
								js/ui/dash.js
									
									
									
									
									
								
							
							
						
						| @@ -1,212 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Cairo = imports.cairo; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Util = imports.misc.util; | ||||
| const Main = imports.ui.main; | ||||
| const PanelMenu = imports.ui.panelMenu; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
| const Calendar = imports.ui.calendar; | ||||
|  | ||||
| // 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) | ||||
| { | ||||
|     let cr = area.get_context(); | ||||
|     let themeNode = area.get_theme_node(); | ||||
|     let [width, height] = area.get_surface_size(); | ||||
|     let stippleColor = themeNode.get_color('-stipple-color'); | ||||
|     let stippleWidth = themeNode.get_length('-stipple-width'); | ||||
|     let x = Math.floor(width/2) + 0.5; | ||||
|     cr.moveTo(x, 0); | ||||
|     cr.lineTo(x, height); | ||||
|     Clutter.cairo_set_source_color(cr, stippleColor); | ||||
|     cr.setDash([1, 3], 1); // Hard-code for now | ||||
|     cr.setLineWidth(stippleWidth); | ||||
|     cr.stroke(); | ||||
| }; | ||||
|  | ||||
| function DateMenuButton() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| DateMenuButton.prototype = { | ||||
|     __proto__: PanelMenu.Button.prototype, | ||||
|  | ||||
|     _init: function() { | ||||
|         let item; | ||||
|         let hbox; | ||||
|         let vbox; | ||||
|  | ||||
|         this._eventSource = new Calendar.DBusEventSource(); | ||||
|  | ||||
|         let menuAlignment = 0.25; | ||||
|         if (St.Widget.get_default_direction() == St.TextDirection.RTL) | ||||
|             menuAlignment = 1.0 - menuAlignment; | ||||
|         PanelMenu.Button.prototype._init.call(this, menuAlignment); | ||||
|  | ||||
|         this._clock = new St.Label(); | ||||
|         this.actor.set_child(this._clock); | ||||
|  | ||||
|         hbox = new St.BoxLayout({name: 'calendarArea'}); | ||||
|         this.menu.addActor(hbox); | ||||
|  | ||||
|         // Fill up the first column | ||||
|  | ||||
|         vbox = new St.BoxLayout({vertical: true}); | ||||
|         hbox.add(vbox); | ||||
|  | ||||
|         // Date | ||||
|         this._date = new St.Label(); | ||||
|         this._date.style_class = 'datemenu-date-label'; | ||||
|         vbox.add(this._date); | ||||
|  | ||||
|         this._eventList = new Calendar.EventsList(this._eventSource); | ||||
|  | ||||
|         // Calendar | ||||
|         this._calendar = new Calendar.Calendar(this._eventSource); | ||||
|         this._calendar.connect('selected-date-changed', | ||||
|                                Lang.bind(this, function(calendar, date) { | ||||
|                                    this._eventList.setDate(date); | ||||
|                                })); | ||||
|         vbox.add(this._calendar.actor); | ||||
|  | ||||
|         item = new PopupMenu.PopupSeparatorMenuItem(); | ||||
|         item.setColumnWidths(1); | ||||
|         vbox.add(item.actor, {y_align: St.Align.END, expand: true, y_fill: false}); | ||||
|         item = new PopupMenu.PopupMenuItem(_("Date and Time Settings")); | ||||
|         item.connect('activate', Lang.bind(this, this._onPreferencesActivate)); | ||||
|         item.actor.can_focus = false; | ||||
|         vbox.add(item.actor); | ||||
|  | ||||
|         // Add vertical separator | ||||
|  | ||||
|         item = new St.DrawingArea({ style_class: 'calendar-vertical-separator', | ||||
|                                     pseudo_class: 'highlighted' }); | ||||
|         item.connect('repaint', Lang.bind(this, _onVertSepRepaint)); | ||||
|         hbox.add(item); | ||||
|  | ||||
|         // Fill up the second column | ||||
|  | ||||
|         vbox = new St.BoxLayout({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) { | ||||
|             if (isOpen) { | ||||
|                 let now = new Date(); | ||||
|                 /* Passing true to setDate() forces events to be reloaded. We | ||||
|                  * want this behavior, because | ||||
|                  * | ||||
|                  *   o It will cause activation of the calendar server which is | ||||
|                  *     useful if it has crashed | ||||
|                  * | ||||
|                  *   o It will cause the calendar server to reload events which | ||||
|                  *     is useful if dynamic updates are not supported or not | ||||
|                  *     properly working | ||||
|                  * | ||||
|                  * Since this only happens when the menu is opened, the cost | ||||
|                  * isn't very big. | ||||
|                  */ | ||||
|                 this._calendar.setDate(now, true); | ||||
|                 // No need to update this._eventList as ::selected-date-changed | ||||
|                 // signal will fire | ||||
|             } | ||||
|         })); | ||||
|  | ||||
|         // Done with hbox for calendar and event list | ||||
|  | ||||
|         // 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)); | ||||
|  | ||||
|         // Start the clock | ||||
|         this._updateClockAndDate(); | ||||
|     }, | ||||
|  | ||||
|     _updateClockAndDate: function() { | ||||
|         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"). | ||||
|          */ | ||||
|         dateFormat = _("%A %B %e, %Y"); | ||||
|         this._date.set_text(displayDate.toLocaleFormat(dateFormat)); | ||||
|  | ||||
|         Mainloop.timeout_add_seconds(1, Lang.bind(this, this._updateClockAndDate)); | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onPreferencesActivate: function() { | ||||
|         this.menu.close(); | ||||
|         let app = Shell.AppSystem.get_default().get_app('gnome-datetime-panel.desktop'); | ||||
|         app.activate(-1); | ||||
|     }, | ||||
|  | ||||
|     _onOpenCalendarActivate: function() { | ||||
|         this.menu.close(); | ||||
|         // TODO: pass the selected day | ||||
|         Util.spawn(['evolution', '-c', 'calendar']); | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										586
									
								
								js/ui/dnd.js
									
									
									
									
									
								
							
							
						
						| @@ -1,134 +1,34 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; tab-width: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const St = imports.gi.St; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| // Time to scale down to maxDragActorSize | ||||
| const SCALE_ANIMATION_TIME = 0.25; | ||||
| // Time to animate to original position on cancel | ||||
| const SNAP_BACK_ANIMATION_TIME = 0.25; | ||||
| // Time to animate to original position on success | ||||
| const REVERT_ANIMATION_TIME = 0.75; | ||||
|  | ||||
| const DragMotionResult = { | ||||
|     NO_DROP:   0, | ||||
|     COPY_DROP: 1, | ||||
|     MOVE_DROP: 2, | ||||
|     CONTINUE:  3 | ||||
| }; | ||||
|  | ||||
| const DRAG_CURSOR_MAP = { | ||||
|     0: Shell.Cursor.DND_UNSUPPORTED_TARGET, | ||||
|     1: Shell.Cursor.DND_COPY, | ||||
|     2: Shell.Cursor.DND_MOVE | ||||
| }; | ||||
|  | ||||
| const DragDropResult = { | ||||
|     FAILURE:  0, | ||||
|     SUCCESS:  1, | ||||
|     CONTINUE: 2 | ||||
| }; | ||||
|  | ||||
| let eventHandlerActor = null; | ||||
| let currentDraggable = null; | ||||
| let dragMonitors = []; | ||||
|  | ||||
| function _getEventHandlerActor() { | ||||
|     if (!eventHandlerActor) { | ||||
|         eventHandlerActor = new Clutter.Rectangle(); | ||||
|         eventHandlerActor.width = 0; | ||||
|         eventHandlerActor.height = 0; | ||||
|         Main.uiGroup.add_actor(eventHandlerActor); | ||||
|         // We connect to 'event' rather than 'captured-event' because the capturing phase doesn't happen | ||||
|         // when you've grabbed the pointer. | ||||
|         eventHandlerActor.connect('event', | ||||
|                                   function(actor, event) { | ||||
|                                       return currentDraggable._onEvent(actor, event); | ||||
|                                   }); | ||||
|     } | ||||
|     return eventHandlerActor; | ||||
| } | ||||
|  | ||||
| function addDragMonitor(monitor) { | ||||
|     dragMonitors.push(monitor); | ||||
| } | ||||
|  | ||||
| function removeMonitor(monitor) { | ||||
|     for (let i = 0; i < dragMonitors.length; i++) | ||||
|         if (dragMonitors[i] == monitor) { | ||||
|             dragMonitors.splice(i, 1); | ||||
|             return; | ||||
|         } | ||||
| } | ||||
|  | ||||
| function _Draggable(actor, params) { | ||||
|     this._init(actor, params); | ||||
| function _Draggable(actor, manualMode) { | ||||
|     this._init(actor, manualMode); | ||||
| } | ||||
|  | ||||
| _Draggable.prototype = { | ||||
|     _init : function(actor, params) { | ||||
|         params = Params.parse(params, { manualMode: false, | ||||
|                                         restoreOnSuccess: false, | ||||
|                                         dragActorMaxSize: undefined, | ||||
|                                         dragActorOpacity: undefined }); | ||||
|  | ||||
|     _init : function(actor, manualMode) { | ||||
|         this.actor = actor; | ||||
|         if (!params.manualMode) | ||||
|         if (!manualMode) | ||||
|             this.actor.connect('button-press-event', | ||||
|                                Lang.bind(this, this._onButtonPress)); | ||||
|  | ||||
|         this.actor.connect('destroy', Lang.bind(this, function() { | ||||
|             this._actorDestroyed = true; | ||||
|             // If the drag actor is destroyed and we were going to fix | ||||
|             // up its hover state, fix up the parent hover state instead | ||||
|             if (this.actor == this._firstLeaveActor) | ||||
|                 this._firstLeaveActor = this._dragOrigParent; | ||||
|             if (this._dragInProgress) | ||||
|                 this._cancelDrag(global.get_current_time()); | ||||
|             this.disconnectAll(); | ||||
|         })); | ||||
|         this._onEventId = null; | ||||
|  | ||||
|         this._restoreOnSuccess = params.restoreOnSuccess; | ||||
|         this._dragActorMaxSize = params.dragActorMaxSize; | ||||
|         this._dragActorOpacity = params.dragActorOpacity; | ||||
|  | ||||
|         this._buttonDown = false; // The mouse button has been pressed and has not yet been released. | ||||
|         this._dragInProgress = false; // The drag has been started, and has not been dropped or cancelled yet. | ||||
|         this._animationInProgress = false; // The drag is over and the item is in the process of animating to its original position (snapping back or reverting). | ||||
|  | ||||
|         // During the drag, we eat enter/leave events so that actors don't prelight or show | ||||
|         // tooltips. But we remember the actors that we first left/last entered so we can | ||||
|         // fix up the hover state after the drag ends. | ||||
|         this._firstLeaveActor = null; | ||||
|         this._lastEnterActor = null; | ||||
|  | ||||
|         this._eventsGrabbed = false; | ||||
|         this._haveSourceGrab = false; | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress : function (actor, event) { | ||||
|         if (event.get_button() != 1) | ||||
|             return false; | ||||
|  | ||||
|         // FIXME: we should make sure it's button 1, but we can't currently | ||||
|         // check that from JavaScript | ||||
|         if (Tweener.getTweenCount(actor)) | ||||
|             return false; | ||||
|  | ||||
|         this._buttonDown = true; | ||||
|         // special case St.Button: grabbing the pointer would mess up the | ||||
|         // internal state, so we start the drag manually on hover change | ||||
|         if (this.actor instanceof St.Button) | ||||
|             this.actor.connect('notify::hover', | ||||
|                                Lang.bind(this, this._onButtonHoverChanged)); | ||||
|         else | ||||
|             this._grabActor(); | ||||
|         this._haveSourceGrab = true; | ||||
|         this._grabActor(actor); | ||||
|  | ||||
|         let [stageX, stageY] = event.get_coords(); | ||||
|         this._dragStartX = stageX; | ||||
| @@ -136,92 +36,42 @@ _Draggable.prototype = { | ||||
|  | ||||
|         return false; | ||||
|     }, | ||||
|      | ||||
|     _grabActor : function (actor) { | ||||
|         Clutter.grab_pointer(actor); | ||||
|  | ||||
|     _onButtonHoverChanged: function(button) { | ||||
|         if (button.hover || !button.pressed) | ||||
|             return; | ||||
|  | ||||
|         button.fake_release(); | ||||
|         this.startDrag(this._dragStartX, this._dragStartY, | ||||
|                        global.get_current_time()); | ||||
|         // We intercept motion and button-release events so that when | ||||
|         // you release after dragging, the app doesn't see that and | ||||
|         // think you just clicked. We connect to 'event' rather than | ||||
|         // 'captured-event' because the capturing phase doesn't happen | ||||
|         // when you've grabbed the pointer. | ||||
|         this._onEventId = actor.connect('event', | ||||
|                                         Lang.bind(this, this._onEvent)); | ||||
|     }, | ||||
|  | ||||
|     _grabActor: function() { | ||||
|         Clutter.grab_pointer(this.actor); | ||||
|         this._onEventId = this.actor.connect('event', | ||||
|                                              Lang.bind(this, this._onEvent)); | ||||
|     }, | ||||
|  | ||||
|     _ungrabActor: function() { | ||||
|     _ungrabActor : function (actor) { | ||||
|         Clutter.ungrab_pointer(); | ||||
|         if (!this._onEventId) | ||||
|             return; | ||||
|         this.actor.disconnect(this._onEventId); | ||||
|         this._onEventId = null; | ||||
|         actor.disconnect(this._onEventId); | ||||
|     }, | ||||
|  | ||||
|     _grabEvents: function() { | ||||
|         if (!this._eventsGrabbed) { | ||||
|             Clutter.grab_pointer(_getEventHandlerActor()); | ||||
|             Clutter.grab_keyboard(_getEventHandlerActor()); | ||||
|             this._eventsGrabbed = true; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _ungrabEvents: function() { | ||||
|         if (this._eventsGrabbed) { | ||||
|             Clutter.ungrab_pointer(); | ||||
|             Clutter.ungrab_keyboard(); | ||||
|             this._eventsGrabbed = false; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onEvent: function(actor, event) { | ||||
|         // We intercept BUTTON_RELEASE event to know that the button was released in case we | ||||
|         // didn't start the drag, to drop the draggable in case the drag was in progress, and | ||||
|         // to complete the drag and ensure that whatever happens to be under the pointer does | ||||
|         // not get triggered if the drag was cancelled with Esc. | ||||
|         if (event.type() == Clutter.EventType.BUTTON_RELEASE) { | ||||
|             this._buttonDown = false; | ||||
|             if (this._dragInProgress) { | ||||
|                 return this._dragActorDropped(event); | ||||
|             } else if (this._dragActor != null && !this._animationInProgress) { | ||||
|                 // Drag must have been cancelled with Esc. | ||||
|                 this._dragComplete(); | ||||
|                 return true; | ||||
|             } else { | ||||
|                 // Drag has never started. | ||||
|                 this._ungrabActor(); | ||||
|     _onEvent : function (actor, event) { | ||||
|         if (this._dragActor) { | ||||
|             if (actor != this._dragActor ) | ||||
|                 return false; | ||||
|             } | ||||
|         // We intercept MOTION event to figure out if the drag has started and to draw | ||||
|         // this._dragActor under the pointer when dragging is in progress | ||||
|         } else if (event.type() == Clutter.EventType.MOTION) { | ||||
|             if (this._dragInProgress) { | ||||
|                 return this._updateDragPosition(event); | ||||
|             } else if (this._dragActor == null) { | ||||
|                 return this._maybeStartDrag(event); | ||||
|             } | ||||
|         // We intercept KEY_PRESS event so that we can process Esc key press to cancel | ||||
|         // dragging and ignore all other key presses. | ||||
|         } else if (event.type() == Clutter.EventType.KEY_PRESS && this._dragInProgress) { | ||||
|             let symbol = event.get_key_symbol(); | ||||
|             if (symbol == Clutter.Escape) { | ||||
|                 this._cancelDrag(event.get_time()); | ||||
|                 return true; | ||||
|             } | ||||
|         } else if (event.type() == Clutter.EventType.LEAVE) { | ||||
|             if (this._firstLeaveActor == null) | ||||
|                 this._firstLeaveActor = event.get_source(); | ||||
|         } else if (event.type() == Clutter.EventType.ENTER) { | ||||
|             this._lastEnterActor = event.get_source(); | ||||
|         } | ||||
|         } else if (actor != this.actor) | ||||
|             return false; | ||||
|  | ||||
|         return false; | ||||
|         if (event.type() == Clutter.EventType.BUTTON_RELEASE) | ||||
|             return this._onButtonRelease(actor, event); | ||||
|         else if (event.type() == Clutter.EventType.MOTION) | ||||
|             return this._onMotion(actor, event); | ||||
|         else | ||||
|             return false; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * startDrag: | ||||
|      * @actor: Origin actor for drag and drop | ||||
|      * @stageX: X coordinate of event | ||||
|      * @stageY: Y coordinate of event | ||||
|      * @time: Event timestamp | ||||
| @@ -230,18 +80,11 @@ _Draggable.prototype = { | ||||
|      * This function is useful to call if you've specified manualMode | ||||
|      * for the draggable. | ||||
|      */ | ||||
|     startDrag: function (stageX, stageY, time) { | ||||
|         currentDraggable = this; | ||||
|         this._dragInProgress = true; | ||||
|  | ||||
|     startDrag: function (actor, stageX, stageY, time) { | ||||
|         this.emit('drag-begin', time); | ||||
|         if (this._onEventId) | ||||
|             this._ungrabActor(); | ||||
|         this._grabEvents(); | ||||
|         global.set_cursor(Shell.Cursor.DND_IN_DRAG); | ||||
|  | ||||
|         this._dragX = this._dragStartX = stageX; | ||||
|         this._dragY = this._dragStartY = stageY; | ||||
|         this._dragStartX = stageX; | ||||
|         this._dragStartY = stageY; | ||||
|  | ||||
|         if (this.actor._delegate && this.actor._delegate.getDragActor) { | ||||
|             this._dragActor = this.actor._delegate.getDragActor(this._dragStartX, this._dragStartY); | ||||
| @@ -254,9 +97,10 @@ _Draggable.prototype = { | ||||
|                 // the dragActor over it. Otherwise, center it | ||||
|                 // around the pointer | ||||
|                 let [sourceX, sourceY] = this._dragActorSource.get_transformed_position(); | ||||
|                 let [sourceWidth, sourceHeight] = this._dragActorSource.get_transformed_size(); | ||||
|                 let x, y; | ||||
|                 if (stageX > sourceX && stageX <= sourceX + this._dragActor.width && | ||||
|                     stageY > sourceY && stageY <= sourceY + this._dragActor.height) { | ||||
|                 if (stageX > sourceX && stageX <= sourceX + sourceWidth && | ||||
|                     stageY > sourceY && stageY <= sourceY + sourceHeight) { | ||||
|                     x = sourceX; | ||||
|                     y = sourceY; | ||||
|                 } else { | ||||
| @@ -268,288 +112,135 @@ _Draggable.prototype = { | ||||
|                 this._dragActorSource = this.actor; | ||||
|             } | ||||
|             this._dragOrigParent = undefined; | ||||
|             if (this._haveSourceGrab) { | ||||
|                 this._haveSourceGrab = false; | ||||
|                 this._ungrabActor(actor); | ||||
|             } | ||||
|             this._grabActor(this._dragActor); | ||||
|  | ||||
|             this._dragOffsetX = this._dragActor.x - this._dragStartX; | ||||
|             this._dragOffsetY = this._dragActor.y - this._dragStartY; | ||||
|         } else { | ||||
|             this._dragActor = this.actor; | ||||
|             this._dragActor = actor; | ||||
|             this._dragActorSource = undefined; | ||||
|             this._dragOrigParent = this.actor.get_parent(); | ||||
|             this._dragOrigParent = actor.get_parent(); | ||||
|             this._dragOrigX = this._dragActor.x; | ||||
|             this._dragOrigY = this._dragActor.y; | ||||
|             this._dragOrigScale = this._dragActor.scale_x; | ||||
|  | ||||
|             let [actorStageX, actorStageY] = this.actor.get_transformed_position(); | ||||
|             let [actorStageX, actorStageY] = actor.get_transformed_position(); | ||||
|             this._dragOffsetX = actorStageX - this._dragStartX; | ||||
|             this._dragOffsetY = actorStageY - this._dragStartY; | ||||
|  | ||||
|             // Set the actor's scale such that it will keep the same | ||||
|             // transformed size when it's reparented to the stage | ||||
|             let [scaledWidth, scaledHeight] = this.actor.get_transformed_size(); | ||||
|             this.actor.set_scale(scaledWidth / this.actor.width, | ||||
|                                  scaledHeight / this.actor.height); | ||||
|             let [scaledWidth, scaledHeight] = actor.get_transformed_size(); | ||||
|             actor.set_scale(scaledWidth / actor.width, | ||||
|                            scaledHeight / actor.height); | ||||
|         } | ||||
|  | ||||
|         this._dragActor.reparent(this.actor.get_stage()); | ||||
|         this._dragActor.reparent(actor.get_stage()); | ||||
|         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; | ||||
|  | ||||
|         this._snapBackX = this._dragStartX + this._dragOffsetX; | ||||
|         this._snapBackY = this._dragStartY + this._dragOffsetY; | ||||
|         this._snapBackScale = this._dragActor.scale_x; | ||||
|  | ||||
|         if (this._dragActorMaxSize != undefined) { | ||||
|             let [scaledWidth, scaledHeight] = this._dragActor.get_transformed_size(); | ||||
|             let currentSize = Math.max(scaledWidth, scaledHeight); | ||||
|             if (currentSize > this._dragActorMaxSize) { | ||||
|                 let scale = this._dragActorMaxSize / currentSize; | ||||
|                 let origScale =  this._dragActor.scale_x; | ||||
|                 let origDragOffsetX = this._dragOffsetX; | ||||
|                 let origDragOffsetY = this._dragOffsetY; | ||||
|  | ||||
|                 // The position of the actor changes as we scale | ||||
|                 // around the drag position, but we can't just tween | ||||
|                 // to the final position because that tween would | ||||
|                 // fight with updates as the user continues dragging | ||||
|                 // the mouse; instead we do the position computations in | ||||
|                 // an onUpdate() function. | ||||
|                 Tweener.addTween(this._dragActor, | ||||
|                                  { scale_x: scale * origScale, | ||||
|                                    scale_y: scale * origScale, | ||||
|                                    time: SCALE_ANIMATION_TIME, | ||||
|                                    transition: 'easeOutQuad', | ||||
|                                    onUpdate: function() { | ||||
|                                        let currentScale = this._dragActor.scale_x / origScale; | ||||
|                                        this._dragOffsetX = currentScale * origDragOffsetX; | ||||
|                                        this._dragOffsetY = currentScale * origDragOffsetY; | ||||
|                                        this._dragActor.set_position(this._dragX + this._dragOffsetX, | ||||
|                                                                     this._dragY + this._dragOffsetY); | ||||
|                                    }, | ||||
|                                    onUpdateScope: this }); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _maybeStartDrag:  function(event) { | ||||
|     _onMotion : function (actor, event) { | ||||
|         let [stageX, stageY] = event.get_coords(); | ||||
|  | ||||
|         // See if the user has moved the mouse enough to trigger a drag | ||||
|         // If we haven't begun a drag, see if the user has moved the | ||||
|         // mouse enough to trigger a drag | ||||
|         let threshold = Gtk.Settings.get_default().gtk_dnd_drag_threshold; | ||||
|         if ((Math.abs(stageX - this._dragStartX) > threshold || | ||||
|         if (!this._dragActor && | ||||
|             (Math.abs(stageX - this._dragStartX) > threshold || | ||||
|              Math.abs(stageY - this._dragStartY) > threshold)) { | ||||
|                 this.startDrag(stageX, stageY, event.get_time()); | ||||
|                 this._updateDragPosition(event); | ||||
|                 this.startDrag(actor, stageX, stageY, event.get_time()); | ||||
|         } | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _updateDragPosition : function (event) { | ||||
|         let [stageX, stageY] = event.get_coords(); | ||||
|         this._dragX = stageX; | ||||
|         this._dragY = stageY; | ||||
|  | ||||
|         // If we are dragging, update the position | ||||
|         if (this._dragActor) { | ||||
|             this._dragActor.set_position(stageX + this._dragOffsetX, | ||||
|                                          stageY + this._dragOffsetY); | ||||
|  | ||||
|             let target = this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL, | ||||
|                                                                       stageX, stageY); | ||||
|  | ||||
|             // We call observers only once per motion with the innermost | ||||
|             // target actor. If necessary, the observer can walk the | ||||
|             // parent itself. | ||||
|             let dragEvent = { | ||||
|                 x: stageX, | ||||
|                 y: stageY, | ||||
|                 dragActor: this._dragActor, | ||||
|                 source: this.actor._delegate, | ||||
|                 targetActor: target | ||||
|             }; | ||||
|             for (let i = 0; i < dragMonitors.length; i++) { | ||||
|                 let motionFunc = dragMonitors[i].dragMotion; | ||||
|                 if (motionFunc) { | ||||
|                     let result = motionFunc(dragEvent); | ||||
|                     if (result != DragMotionResult.CONTINUE) { | ||||
|                         global.set_cursor(DRAG_CURSOR_MAP[result]); | ||||
|                         return true; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             // Because we want to find out what other actor is located at the current position of this._dragActor, | ||||
|             // we have to temporarily hide this._dragActor. | ||||
|             this._dragActor.hide();  | ||||
|             let target = actor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL, | ||||
|                                                             stageX + this._dragOffsetX, | ||||
|                                                             stageY + this._dragOffsetY); | ||||
|             this._dragActor.show(); | ||||
|             while (target) { | ||||
|                 if (target._delegate && target._delegate.handleDragOver) { | ||||
|                     let [r, targX, targY] = target.transform_stage_point(stageX, stageY); | ||||
|                     let [targX, targY] = target.get_transformed_position(); | ||||
|                     // We currently loop through all parents on drag-over even if one of the children has handled it. | ||||
|                     // We can check the return value of the function and break the loop if it's true if we don't want | ||||
|                     // to continue checking the parents. | ||||
|                     let result = target._delegate.handleDragOver(this.actor._delegate, | ||||
|                                                                  this._dragActor, | ||||
|                                                                  targX, | ||||
|                                                                  targY, | ||||
|                                                                  event.get_time()); | ||||
|                     if (result != DragMotionResult.CONTINUE) { | ||||
|                         global.set_cursor(DRAG_CURSOR_MAP[result]); | ||||
|                         return true; | ||||
|                     } | ||||
|                     target._delegate.handleDragOver(this.actor._delegate, actor, | ||||
|                                                     (stageX + this._dragOffsetX - targX) / target.scale_x, | ||||
|                                                     (stageY + this._dragOffsetY - targY) / target.scale_y, | ||||
|                                                     event.get_time()); | ||||
|                 } | ||||
|                 target = target.get_parent(); | ||||
|             } | ||||
|             global.set_cursor(Shell.Cursor.DND_IN_DRAG); | ||||
|         } | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _dragActorDropped: function(event) { | ||||
|     _onButtonRelease : function (actor, event) { | ||||
|         this._ungrabActor(actor); | ||||
|  | ||||
|         let dragging = (actor == this._dragActor); | ||||
|         this._dragActor = undefined; | ||||
|  | ||||
|         if (!dragging) | ||||
|             return false; | ||||
|  | ||||
|         // Find a drop target | ||||
|         actor.hide(); | ||||
|         let [dropX, dropY] = event.get_coords(); | ||||
|         let target = this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL, | ||||
|                                                                   dropX, dropY); | ||||
|  | ||||
|         // We call observers only once per motion with the innermost | ||||
|         // target actor. If necessary, the observer can walk the | ||||
|         // parent itself. | ||||
|         let dropEvent = { | ||||
|             dropActor: this._dragActor, | ||||
|             targetActor: target, | ||||
|             clutterEvent: event | ||||
|         }; | ||||
|         for (let i = 0; i < dragMonitors.length; i++) { | ||||
|             let dropFunc = dragMonitors[i].dragDrop; | ||||
|             if (dropFunc) | ||||
|                 switch (dropFunc(dropEvent)) { | ||||
|                     case DragDropResult.FAILURE: | ||||
|                     case DragDropResult.SUCCESS: | ||||
|                         return true; | ||||
|                     case DragDropResult.CONTINUE: | ||||
|                         continue; | ||||
|                 } | ||||
|         } | ||||
|  | ||||
|         let target = actor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL, | ||||
|                                                         dropX, dropY); | ||||
|         actor.show(); | ||||
|         while (target) { | ||||
|             if (target._delegate && target._delegate.acceptDrop) { | ||||
|                 let [r, targX, targY] = target.transform_stage_point(dropX, dropY); | ||||
|                 if (target._delegate.acceptDrop(this.actor._delegate, | ||||
|                                                 this._dragActor, | ||||
|                                                 targX, | ||||
|                                                 targY, | ||||
|                 let [targX, targY] = target.get_transformed_position(); | ||||
|                 if (target._delegate.acceptDrop(this.actor._delegate, actor, | ||||
|                                                 (dropX - targX) / target.scale_x, | ||||
|                                                 (dropY - targY) / target.scale_y, | ||||
|                                                 event.get_time())) { | ||||
|                     if (this._actorDestroyed) | ||||
|                         return true; | ||||
|                     // If it accepted the drop without taking the actor, | ||||
|                     // handle it ourselves. | ||||
|                     if (this._dragActor.get_parent() == this._dragActor.get_stage()) { | ||||
|                         if (this._restoreOnSuccess) { | ||||
|                             this._restoreDragActor(event.get_time()); | ||||
|                             return true; | ||||
|                         } else | ||||
|                             this._dragActor.destroy(); | ||||
|                     } | ||||
|                     // destroy it. | ||||
|                     if (actor.get_parent() == actor.get_stage()) | ||||
|                         actor.destroy(); | ||||
|  | ||||
|                     this._dragInProgress = false; | ||||
|                     global.unset_cursor(); | ||||
|                     this.emit('drag-end', event.get_time(), true); | ||||
|                     this._dragComplete(); | ||||
|                     return true; | ||||
|                 } | ||||
|             } | ||||
|             target = target.get_parent(); | ||||
|         } | ||||
|  | ||||
|         this._cancelDrag(event.get_time()); | ||||
|         // Snap back to the actor source if the source is still around, snap back  | ||||
|         // to the original location if the actor itself was being dragged or the | ||||
|         // source is no longer around. | ||||
|         let snapBackX = this._dragStartX + this._dragOffsetX; | ||||
|         let snapBackY = this._dragStartY + this._dragOffsetY; | ||||
|         if (this._dragActorSource && this._dragActorSource.visible) { | ||||
|             [snapBackX, snapBackY] = this._dragActorSource.get_transformed_position(); | ||||
|         } | ||||
|  | ||||
|         // No target, so snap back | ||||
|         Tweener.addTween(actor, | ||||
|                          { x: snapBackX, | ||||
|                            y: snapBackY, | ||||
|                            time: SNAP_BACK_ANIMATION_TIME, | ||||
|                            transition: "easeOutQuad", | ||||
|                            onComplete: this._onSnapBackComplete, | ||||
|                            onCompleteScope: this, | ||||
|                            onCompleteParams: [actor, event.get_time()] | ||||
|                          }); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _getRestoreLocation: function() { | ||||
|         let x, y, scale; | ||||
|  | ||||
|         if (this._dragActorSource && this._dragActorSource.visible) { | ||||
|             // Snap the clone back to its source | ||||
|             [x, y] = this._dragActorSource.get_transformed_position(); | ||||
|             let [sourceScaledWidth, sourceScaledHeight] = this._dragActorSource.get_transformed_size(); | ||||
|             scale = this._dragActor.width / sourceScaledWidth; | ||||
|         } else if (this._dragOrigParent) { | ||||
|             // Snap the actor back to its original position within | ||||
|             // its parent, adjusting for the fact that the parent | ||||
|             // may have been moved or scaled | ||||
|             let [parentX, parentY] = this._dragOrigParent.get_transformed_position(); | ||||
|             let [parentWidth, parentHeight] = this._dragOrigParent.get_size(); | ||||
|             let [parentScaledWidth, parentScaledHeight] = this._dragOrigParent.get_transformed_size(); | ||||
|             let parentScale = 1.0; | ||||
|             if (parentWidth != 0) | ||||
|                 parentScale = parentScaledWidth / parentWidth; | ||||
|  | ||||
|             x = parentX + parentScale * this._dragOrigX; | ||||
|             y = parentY + parentScale * this._dragOrigY; | ||||
|             scale = this._dragOrigScale * parentScale; | ||||
|         } else { | ||||
|             // Snap back actor to its original stage position | ||||
|             x = this._snapBackX; | ||||
|             y = this._snapBackY; | ||||
|             scale = this._snapBackScale; | ||||
|         } | ||||
|  | ||||
|         return [x, y, scale]; | ||||
|     }, | ||||
|  | ||||
|     _cancelDrag: function(eventTime) { | ||||
|         this.emit('drag-cancelled', eventTime); | ||||
|         this._dragInProgress = false; | ||||
|         let [snapBackX, snapBackY, snapBackScale] = this._getRestoreLocation(); | ||||
|  | ||||
|         if (this._actorDestroyed) { | ||||
|             global.unset_cursor(); | ||||
|             if (!this._buttonDown) | ||||
|                 this._dragComplete(); | ||||
|             this.emit('drag-end', eventTime, false); | ||||
|             if (!this._dragOrigParent) | ||||
|                 this._dragActor.destroy(); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._animationInProgress = true; | ||||
|         // No target, so snap back | ||||
|         Tweener.addTween(this._dragActor, | ||||
|                          { x: snapBackX, | ||||
|                            y: snapBackY, | ||||
|                            scale_x: snapBackScale, | ||||
|                            scale_y: snapBackScale, | ||||
|                            opacity: this._dragOrigOpacity, | ||||
|                            time: SNAP_BACK_ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: this._onAnimationComplete, | ||||
|                            onCompleteScope: this, | ||||
|                            onCompleteParams: [this._dragActor, eventTime] | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     _restoreDragActor: function(eventTime) { | ||||
|         this._dragInProgress = false; | ||||
|         [restoreX, restoreY, restoreScale] = this._getRestoreLocation(); | ||||
|  | ||||
|         // fade the actor back in at its original location | ||||
|         this._dragActor.set_position(restoreX, restoreY); | ||||
|         this._dragActor.set_scale(restoreScale, restoreScale); | ||||
|         this._dragActor.opacity = 0; | ||||
|  | ||||
|         this._animationInProgress = true; | ||||
|         Tweener.addTween(this._dragActor, | ||||
|                          { opacity: this._dragOrigOpacity, | ||||
|                            time: REVERT_ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: this._onAnimationComplete, | ||||
|                            onCompleteScope: this, | ||||
|                            onCompleteParams: [this._dragActor, eventTime] | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     _onAnimationComplete : function (dragActor, eventTime) { | ||||
|     _onSnapBackComplete : function (dragActor, eventTime) { | ||||
|         if (this._dragOrigParent) { | ||||
|             dragActor.reparent(this._dragOrigParent); | ||||
|             dragActor.set_scale(this._dragOrigScale, this._dragOrigScale); | ||||
| @@ -557,44 +248,7 @@ _Draggable.prototype = { | ||||
|         } else { | ||||
|             dragActor.destroy(); | ||||
|         } | ||||
|         global.unset_cursor(); | ||||
|         this.emit('drag-end', eventTime, false); | ||||
|  | ||||
|         this._animationInProgress = false; | ||||
|         if (!this._buttonDown) | ||||
|             this._dragComplete(); | ||||
|     }, | ||||
|  | ||||
|     // Actor is an actor we have entered or left during the drag; call | ||||
|     // st_widget_sync_hover on all StWidget ancestors | ||||
|     _syncHover: function(actor) { | ||||
|         while (actor) { | ||||
|             let parent = actor.get_parent(); | ||||
|             if (actor instanceof St.Widget) | ||||
|                 actor.sync_hover(); | ||||
|  | ||||
|             actor = parent; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _dragComplete: function() { | ||||
|         if (!this._actorDestroyed) | ||||
|             Shell.util_set_hidden_from_pick(this._dragActor, false); | ||||
|  | ||||
|         this._ungrabEvents(); | ||||
|  | ||||
|         if (this._firstLeaveActor) { | ||||
|             this._syncHover(this._firstLeaveActor); | ||||
|             this._firstLeaveActor = null; | ||||
|         } | ||||
|  | ||||
|         if (this._lastEnterActor) { | ||||
|             this._syncHover(this._lastEnterActor); | ||||
|             this._lastEnterActor = null; | ||||
|         } | ||||
|  | ||||
|         this._dragActor = undefined; | ||||
|         currentDraggable = null; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| @@ -603,24 +257,10 @@ Signals.addSignalMethods(_Draggable.prototype); | ||||
| /** | ||||
|  * makeDraggable: | ||||
|  * @actor: Source actor | ||||
|  * @params: (optional) Additional parameters | ||||
|  * @manualMode: If given, do not automatically start drag and drop on click | ||||
|  * | ||||
|  * Create an object which controls drag and drop for the given actor. | ||||
|  * | ||||
|  * If %manualMode is %true in @params, do not automatically start | ||||
|  * drag and drop on click | ||||
|  * | ||||
|  * If %dragActorMaxSize is present in @params, the drag actor will | ||||
|  * be scaled down to be no larger than that size in pixels. | ||||
|  * | ||||
|  * If %dragActorOpacity is present in @params, the drag actor will | ||||
|  * will be set to have that opacity during the drag. | ||||
|  * | ||||
|  * Note that when the drag actor is the source actor and the drop | ||||
|  * succeeds, the actor scale and opacity aren't reset; if the drop | ||||
|  * target wants to reuse the actor, it's up to the drop target to | ||||
|  * reset these values. | ||||
|  */ | ||||
| function makeDraggable(actor, params) { | ||||
|     return new _Draggable(actor, params); | ||||
| function makeDraggable(actor, manualMode) { | ||||
|     return new _Draggable(actor, manualMode); | ||||
| } | ||||
|   | ||||
| @@ -1,50 +1,436 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
| const Big = imports.gi.Big; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const Mainloop = imports.mainloop; | ||||
|  | ||||
| const DocInfo = imports.misc.docInfo; | ||||
| const Params = imports.misc.params; | ||||
| const Search = imports.ui.search; | ||||
| const DND = imports.ui.dnd; | ||||
| const GenericDisplay = imports.ui.genericDisplay; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const DASH_DOCS_ICON_SIZE = 16; | ||||
|  | ||||
| function DocSearchProvider() { | ||||
| const DEFAULT_SPACING = 4; | ||||
|  | ||||
| /* This class represents a single display item containing information about a document. | ||||
|  * We take the current number of seconds in the constructor to avoid looking up the current | ||||
|  * time for every item when they are created in a batch. | ||||
|  * | ||||
|  * docInfo - DocInfo object containing information about the document | ||||
|  * currentSeconds - current number of seconds since the epoch | ||||
|  */ | ||||
| function DocDisplayItem(docInfo, currentSecs) { | ||||
|     this._init(docInfo, currentSecs); | ||||
| } | ||||
|  | ||||
| DocDisplayItem.prototype = { | ||||
|     __proto__:  GenericDisplay.GenericDisplayItem.prototype, | ||||
|  | ||||
|     _init : function(docInfo, currentSecs) { | ||||
|         GenericDisplay.GenericDisplayItem.prototype._init.call(this); | ||||
|         this._docInfo = docInfo; | ||||
|  | ||||
|         this._setItemInfo(docInfo.name, ""); | ||||
|  | ||||
|         this._timeoutTime = -1; | ||||
|         this._resetTimeDisplay(currentSecs); | ||||
|     }, | ||||
|  | ||||
|     //// Public methods //// | ||||
|  | ||||
|     getUpdateTimeoutTime: function() { | ||||
|         return this._timeoutTime; | ||||
|     }, | ||||
|  | ||||
|     // Update any relative-time based displays for this item. | ||||
|     redisplay: function(currentSecs) { | ||||
|         this._resetTimeDisplay(currentSecs); | ||||
|     }, | ||||
|  | ||||
|     //// Public method overrides //// | ||||
|  | ||||
|     // Opens a document represented by this display item. | ||||
|     launch : function() { | ||||
|         this._docInfo.launch(); | ||||
|     }, | ||||
|  | ||||
|     //// Protected method overrides //// | ||||
|  | ||||
|     // Returns an icon for the item. | ||||
|     _createIcon : function() { | ||||
|         return this._docInfo.createIcon(GenericDisplay.ITEM_DISPLAY_ICON_SIZE); | ||||
|     }, | ||||
|  | ||||
|     // Returns a preview icon for the item. | ||||
|     _createPreviewIcon : function() { | ||||
|         return this._docInfo.createIcon(GenericDisplay.PREVIEW_ICON_SIZE); | ||||
|     }, | ||||
|  | ||||
|     // Creates and returns a large preview icon, but only if this._docInfo is an image file | ||||
|     // and we were able to generate a pixbuf from it successfully. | ||||
|     _createLargePreviewIcon : function() { | ||||
|         if (this._docInfo.mimeType == null || this._docInfo.mimeType.indexOf("image/") != 0) | ||||
|             return null; | ||||
|  | ||||
|         try { | ||||
|             return Shell.TextureCache.get_default().load_uri_sync(Shell.TextureCachePolicy.NONE, | ||||
|                                                                   this._docInfo.uri, -1, -1); | ||||
|         } catch (e) { | ||||
|             // An exception will be raised when the image format isn't know | ||||
|             /* FIXME: http://bugzilla.gnome.org/show_bug.cgi?id=591480: should | ||||
|              *        only ignore GDK_PIXBUF_ERROR_UNKNOWN_TYPE. */ | ||||
|             return null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     //// Drag and Drop //// | ||||
|  | ||||
|     shellWorkspaceLaunch: function() { | ||||
|         this.launch(); | ||||
|     }, | ||||
|  | ||||
|     //// Private Methods //// | ||||
|  | ||||
|     // Updates the last visited time displayed in the description text for the item. | ||||
|     _resetTimeDisplay: function(currentSecs) { | ||||
|         let lastSecs = this._docInfo.timestamp; | ||||
|         let timeDelta = currentSecs - lastSecs; | ||||
|         let [text, nextUpdate] = global.format_time_relative_pretty(timeDelta); | ||||
|         this._timeoutTime = currentSecs + nextUpdate; | ||||
|         this._setDescriptionText(text); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /* This class represents a display containing a collection of document items. | ||||
|  * The documents are sorted by how recently they were last visited. | ||||
|  */ | ||||
| function DocDisplay() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| DocSearchProvider.prototype = { | ||||
|     __proto__: Search.SearchProvider.prototype, | ||||
| DocDisplay.prototype = { | ||||
|     __proto__:  GenericDisplay.GenericDisplay.prototype, | ||||
|  | ||||
|     _init : function() { | ||||
|         GenericDisplay.GenericDisplay.prototype._init.call(this); | ||||
|         // We keep a single timeout callback for updating last visited times | ||||
|         // for all the items in the display. This avoids creating individual | ||||
|         // callbacks for each item in the display. So proper time updates | ||||
|         // for individual items and item details depend on the item being  | ||||
|         // associated with one of the displays. | ||||
|         this._updateTimeoutTargetTime = -1; | ||||
|         this._updateTimeoutId = 0; | ||||
|  | ||||
|     _init: function(name) { | ||||
|         Search.SearchProvider.prototype._init.call(this, _("RECENT ITEMS")); | ||||
|         this._docManager = DocInfo.getDocManager(); | ||||
|         this._docsStale = true; | ||||
|         this._docManager.connect('changed', Lang.bind(this, function(mgr, userData) { | ||||
|             this._docsStale = true; | ||||
|             // Changes in local recent files should not happen when we are in the Overview mode, | ||||
|             // but redisplaying right away is cool when we use Zephyr. | ||||
|             // Also, we might be displaying remote documents, like Google Docs, in the future | ||||
|             // which might be edited by someone else. | ||||
|             this._redisplay(false); | ||||
|         })); | ||||
|  | ||||
|         this.connect('destroy', Lang.bind(this, function (o) { | ||||
|             if (this._updateTimeoutId > 0) | ||||
|                 Mainloop.source_remove(this._updateTimeoutId); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     getResultMeta: function(resultId) { | ||||
|         let docInfo = this._docManager.lookupByUri(resultId); | ||||
|         if (!docInfo) | ||||
|             return null; | ||||
|         return { 'id': resultId, | ||||
|                  'name': docInfo.name, | ||||
|                  'createIcon': function(size) { | ||||
|                                    return docInfo.createIcon(size); | ||||
|                                } | ||||
|                }; | ||||
|     //// Protected method overrides //// | ||||
|  | ||||
|     // Gets the list of recent items from the recent items manager. | ||||
|     _refreshCache : function() { | ||||
|         if (!this._docsStale) | ||||
|             return; | ||||
|         this._allItems = this._docManager.getItems(); | ||||
|         this._docsStale = false; | ||||
|     }, | ||||
|  | ||||
|     activateResult: function(id, params) { | ||||
|         params = Params.parse(params, { workspace: null, | ||||
|                                         timestamp: null }); | ||||
|     // Sets the list of the displayed items based on how recently they were last visited. | ||||
|     _setDefaultList : function() { | ||||
|         // It seems to be an implementation detail of the Mozilla JavaScript that object | ||||
|         // properties are returned during the iteration in the same order in which they were | ||||
|         // defined, but it is not a guarantee according to this  | ||||
|         // https://developer.mozilla.org/en/Core_JavaScript_1.5_Reference/Statements/for...in | ||||
|         // While this._allItems associative array seems to always be ordered by last added, | ||||
|         // as the results of this._recentManager.get_items() based on which it is constructed are, | ||||
|         // we should do the sorting manually because we want the order to be based on last visited. | ||||
|         // | ||||
|         // This function is called each time the search string is set back to '' or we display | ||||
|         // the Overview, so we are doing the sorting over the same items multiple times if the list | ||||
|         // of recent items didn't change. We could store an additional array of doc ids and sort | ||||
|         // them once when they are returned by this._recentManager.get_items() to avoid having to do  | ||||
|         // this sorting each time, but the sorting seems to be very fast anyway, so there is no need | ||||
|         // to introduce an additional class variable. | ||||
|         this._matchedItems = []; | ||||
|         let docIdsToRemove = []; | ||||
|         for (docId in this._allItems) { | ||||
|             // this._allItems[docId].exists() checks if the resource still exists | ||||
|             if (this._allItems[docId].exists())  | ||||
|                 this._matchedItems.push(docId); | ||||
|             else  | ||||
|                 docIdsToRemove.push(docId); | ||||
|         } | ||||
|  | ||||
|         let docInfo = this._docManager.lookupByUri(id); | ||||
|         docInfo.launch(params.workspace ? params.workspace.index() : -1); | ||||
|         for (docId in docIdsToRemove) { | ||||
|             delete this._allItems[docId]; | ||||
|         } | ||||
|  | ||||
|         this._matchedItems.sort(Lang.bind(this, function (a,b) { return this._compareItems(a,b); })); | ||||
|     }, | ||||
|  | ||||
|     getInitialResultSet: function(terms) { | ||||
|         return this._docManager.initialSearch(terms); | ||||
|     // Compares items associated with the item ids based on how recently the items | ||||
|     // were last visited. | ||||
|     // Returns an integer value indicating the result of the comparison. | ||||
|    _compareItems : function(itemIdA, itemIdB) { | ||||
|         let docA = this._allItems[itemIdA]; | ||||
|         let docB = this._allItems[itemIdB]; | ||||
|  | ||||
|         return docB.timestamp - docA.timestamp; | ||||
|     }, | ||||
|  | ||||
|     getSubsearchResultSet: function(previousResults, terms) { | ||||
|         return this._docManager.subsearch(previousResults, terms); | ||||
|     // Checks if the item info can be a match for the search string by checking | ||||
|     // the name of the document. Item info is expected to be GtkRecentInfo. | ||||
|     // Returns a boolean flag indicating if itemInfo is a match. | ||||
|     _isInfoMatching : function(itemInfo, search) { | ||||
|         if (!itemInfo.exists()) | ||||
|             return false; | ||||
|   | ||||
|         if (search == null || search == '') | ||||
|             return true; | ||||
|  | ||||
|         let name = itemInfo.name.toLowerCase(); | ||||
|         if (name.indexOf(search) >= 0) | ||||
|             return true; | ||||
|         // TODO: we can also check doc URIs, so that | ||||
|         // if you search for a directory name, we display recent files from it | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     // Creates a DocDisplayItem based on itemInfo, which is expected to be a DocInfo object. | ||||
|     _createDisplayItem: function(itemInfo) { | ||||
|         let currentSecs = new Date().getTime() / 1000; | ||||
|         let docDisplayItem = new DocDisplayItem(itemInfo, currentSecs); | ||||
|         this._updateTimeoutCallback(docDisplayItem, currentSecs); | ||||
|         return docDisplayItem; | ||||
|     }, | ||||
|  | ||||
|     //// Private Methods //// | ||||
|  | ||||
|     // A callback function that redisplays the items, updating their descriptions, | ||||
|     // and sets up a new timeout callback. | ||||
|     _docTimeout: function () { | ||||
|         let currentSecs = new Date().getTime() / 1000; | ||||
|         this._updateTimeoutId = 0; | ||||
|         this._updateTimeoutTargetTime = -1; | ||||
|         for (let docId in this._displayedItems) { | ||||
|             let docDisplayItem = this._displayedItems[docId]; | ||||
|             docDisplayItem.redisplay(currentSecs);           | ||||
|             this._updateTimeoutCallback(docDisplayItem, currentSecs); | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     // Updates the timeout callback if the timeout time for the docDisplayItem  | ||||
|     // is earlier than the target time for the current timeout callback. | ||||
|     _updateTimeoutCallback: function (docDisplayItem, currentSecs) { | ||||
|         let timeoutTime = docDisplayItem.getUpdateTimeoutTime(); | ||||
|         if (this._updateTimeoutTargetTime < 0 || timeoutTime < this._updateTimeoutTargetTime) { | ||||
|             if (this._updateTimeoutId > 0) | ||||
|                 Mainloop.source_remove(this._updateTimeoutId); | ||||
|             this._updateTimeoutId = Mainloop.timeout_add_seconds(timeoutTime - currentSecs, Lang.bind(this, this._docTimeout)); | ||||
|             this._updateTimeoutTargetTime = timeoutTime; | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| Signals.addSignalMethods(DocDisplay.prototype); | ||||
|  | ||||
| function DashDocDisplayItem(docInfo) { | ||||
|     this._init(docInfo); | ||||
| } | ||||
|  | ||||
| DashDocDisplayItem.prototype = { | ||||
|     _init: function(docInfo) { | ||||
|         this._info = docInfo; | ||||
|         this.actor = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                    spacing: DEFAULT_SPACING, | ||||
|                                    reactive: true }); | ||||
|         this.actor.connect('button-release-event', Lang.bind(this, function () { | ||||
|             docInfo.launch(); | ||||
|             Main.overview.hide(); | ||||
|         })); | ||||
|  | ||||
|         this._icon = docInfo.createIcon(DASH_DOCS_ICON_SIZE); | ||||
|         let iconBox = new Big.Box({ y_align: Big.BoxAlignment.CENTER }); | ||||
|         iconBox.append(this._icon, Big.BoxPackFlags.NONE); | ||||
|         this.actor.append(iconBox, Big.BoxPackFlags.NONE); | ||||
|         let name = new Clutter.Text({ font_name: "Sans 14px", | ||||
|                                       color: GenericDisplay.ITEM_DISPLAY_NAME_COLOR, | ||||
|                                       ellipsize: Pango.EllipsizeMode.END, | ||||
|                                       text: docInfo.name }); | ||||
|         this.actor.append(name, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         let draggable = DND.makeDraggable(this.actor); | ||||
|         this.actor._delegate = this; | ||||
|     }, | ||||
|  | ||||
|     getDragActorSource: function() { | ||||
|         return this._icon; | ||||
|     }, | ||||
|  | ||||
|     getDragActor: function(stageX, stageY) { | ||||
|         this.dragActor = this._info.createIcon(DASH_DOCS_ICON_SIZE); | ||||
|         return this.dragActor; | ||||
|     }, | ||||
|  | ||||
|     //// Drag and drop functions //// | ||||
|  | ||||
|     shellWorkspaceLaunch: function () { | ||||
|         this._info.launch(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Class used to display two column recent documents in the dash | ||||
|  */ | ||||
| function DashDocDisplay() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| DashDocDisplay.prototype = { | ||||
|     _init: function() { | ||||
|         this.actor = new Shell.GenericContainer(); | ||||
|         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._docManager = DocInfo.getDocManager(); | ||||
|         this._docManager.connect('changed', Lang.bind(this, function(mgr) { | ||||
|             this._redisplay(); | ||||
|         })); | ||||
|         this._redisplay(); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         let children = actor.get_children(); | ||||
|  | ||||
|         // We use two columns maximum.  Just take the min and natural size of the | ||||
|         // first two items, even though strictly speaking it's not correct; we'd | ||||
|         // need to calculate how many items we could fit for the height, then | ||||
|         // take the biggest preferred width for each column. | ||||
|         // In practice the dash gets a fixed width anyways. | ||||
|  | ||||
|         // If we have one child, add its minimum and natural size | ||||
|         if (children.length > 0) { | ||||
|             let [minSize, naturalSize] = children[0].get_preferred_width(forHeight); | ||||
|             alloc.min_size += minSize; | ||||
|             alloc.natural_size += naturalSize; | ||||
|         } | ||||
|         // If we have two, add its size, plus DEFAULT_SPACING | ||||
|         if (children.length > 1) { | ||||
|             let [minSize, naturalSize] = children[1].get_preferred_width(forHeight); | ||||
|             alloc.min_size += DEFAULT_SPACING + minSize; | ||||
|             alloc.natural_size += DEFAULT_SPACING + naturalSize; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let children = actor.get_children(); | ||||
|  | ||||
|         // Two columns, where we go vertically down first.  So just take | ||||
|         // the height of half of the children as our preferred height. | ||||
|  | ||||
|         let firstColumnChildren = children.length / 2; | ||||
|  | ||||
|         alloc.min_size = 0; | ||||
|         for (let i = 0; i < firstColumnChildren; i++) { | ||||
|             let child = children[i]; | ||||
|             let [minSize, naturalSize] = child.get_preferred_height(forWidth); | ||||
|             alloc.natural_size += naturalSize; | ||||
|  | ||||
|             if (i > 0 && i < children.length - 1) { | ||||
|                 alloc.min_size += DEFAULT_SPACING; | ||||
|                 alloc.natural_size += DEFAULT_SPACING; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
|         let width = box.x2 - box.x1; | ||||
|         let height = box.y2 - box.y1; | ||||
|  | ||||
|         let children = actor.get_children(); | ||||
|  | ||||
|         // The width of an item is our allocated width, minus spacing, divided in half. | ||||
|         let itemWidth = Math.floor((width - DEFAULT_SPACING) / 2); | ||||
|         let x = box.x1; | ||||
|         let y = box.y1; | ||||
|         let columnIndex = 0; | ||||
|         let i = 0; | ||||
|         // Loop over the children, going vertically down first.  When we run | ||||
|         // out of vertical space (our y variable is bigger than box.y2), switch | ||||
|         // to the second column. | ||||
|         for (; i < children.length; i++) { | ||||
|             let child = children[i]; | ||||
|  | ||||
|             let [minSize, naturalSize] = child.get_preferred_height(-1); | ||||
|  | ||||
|             if (y + naturalSize > box.y2) { | ||||
|                 // Is this the second column?  Ok, break. | ||||
|                 if (columnIndex == 1) { | ||||
|                     break; | ||||
|                 } | ||||
|                 // Set x to the halfway point. | ||||
|                 columnIndex += 1; | ||||
|                 x = x + itemWidth + DEFAULT_SPACING; | ||||
|                 // And y is back to the top. | ||||
|                 y = box.y1; | ||||
|             } | ||||
|  | ||||
|             let childBox = new Clutter.ActorBox(); | ||||
|             childBox.x1 = x; | ||||
|             childBox.y1 = y; | ||||
|             childBox.x2 = childBox.x1 + itemWidth; | ||||
|             childBox.y2 = y + naturalSize; | ||||
|  | ||||
|             y = childBox.y2 + DEFAULT_SPACING; | ||||
|  | ||||
|             child.show(); | ||||
|             child.allocate(childBox, flags); | ||||
|         } | ||||
|  | ||||
|         // Everything else didn't fit, just hide it. | ||||
|         for (; i < children.length; i++) { | ||||
|             children[i].hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _redisplay: function() { | ||||
|         this.actor.remove_all(); | ||||
|  | ||||
|         let docs = this._docManager.getItems(); | ||||
|         let docUrls = []; | ||||
|         for (let url in docs) { | ||||
|             docUrls.push(url); | ||||
|         } | ||||
|         docUrls.sort(function (urlA, urlB) { return docs[urlB].timestamp - docs[urlA].timestamp; }); | ||||
|         let textureCache = Shell.TextureCache.get_default(); | ||||
|  | ||||
|         for (let i = 0; i < docUrls.length; i++) { | ||||
|             let url = docUrls[i]; | ||||
|             let docInfo = docs[url]; | ||||
|             let display = new DashDocDisplayItem(docInfo); | ||||
|             this.actor.add_actor(display.actor); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,536 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- | ||||
|  * | ||||
|  * Copyright 2010 Red Hat, Inc | ||||
|  * | ||||
|  * 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 DBus = imports.dbus; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gdm = imports.gi.Gdm; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Pango = imports.gi.Pango; | ||||
| 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; | ||||
|  | ||||
| let _endSessionDialog = null; | ||||
|  | ||||
| const _ITEM_ICON_SIZE = 48; | ||||
| const _DIALOG_ICON_SIZE = 32; | ||||
|  | ||||
| const GSM_SESSION_MANAGER_LOGOUT_FORCE = 2; | ||||
|  | ||||
| const EndSessionDialogIface = { | ||||
|     name: 'org.gnome.SessionManager.EndSessionDialog', | ||||
|     methods: [{ name: 'Open', | ||||
|                 inSignature: 'uuuao', | ||||
|                 outSignature: '' | ||||
|               } | ||||
|              ], | ||||
|     signals: [{ name: 'Canceled', | ||||
|                 inSignature: '', | ||||
|               }], | ||||
|     properties: [] | ||||
| }; | ||||
|  | ||||
| const logoutDialogContent = { | ||||
|     subjectWithUser: _("Log Out %s"), | ||||
|     subject: _("Log Out"), | ||||
|     inhibitedDescription: _("Click Log Out to quit these applications and log out of the system."), | ||||
|     uninhibitedDescriptionWithUser: _("%s will be logged out automatically in %d seconds."), | ||||
|     uninhibitedDescription: _("You will be logged out automatically in %d seconds."), | ||||
|     endDescription: _("Logging out of the system."), | ||||
|     confirmButtons: [{ signal: 'ConfirmedLogout', | ||||
|                        label:  _("Log Out") }], | ||||
|     iconStyleClass: 'end-session-dialog-logout-icon' | ||||
| }; | ||||
|  | ||||
| const shutdownDialogContent = { | ||||
|     subject: _("Power Off"), | ||||
|     inhibitedDescription: _("Click Power Off to quit these applications and power off the system."), | ||||
|     uninhibitedDescription: _("The system will power off automatically in %d seconds."), | ||||
|     endDescription: _("Powering off the system."), | ||||
|     confirmButtons: [{ signal: 'ConfirmedReboot', | ||||
|                        label:  _("Restart") }, | ||||
|                      { signal: 'ConfirmedShutdown', | ||||
|                        label:  _("Power Off") }], | ||||
|     iconName: 'system-shutdown', | ||||
|     iconStyleClass: 'end-session-dialog-shutdown-icon' | ||||
| }; | ||||
|  | ||||
| const restartDialogContent = { | ||||
|     subject: _("Restart"), | ||||
|     inhibitedDescription: _("Click Restart to quit these applications and restart the system."), | ||||
|     uninhibitedDescription: _("The system will restart automatically in %d seconds."), | ||||
|     endDescription: _("Restarting the system."), | ||||
|     confirmButtons: [{ signal: 'ConfirmedReboot', | ||||
|                        label:  _("Restart") }], | ||||
|     iconName: 'system-shutdown', | ||||
|     iconStyleClass: 'end-session-dialog-shutdown-icon' | ||||
| }; | ||||
|  | ||||
| const DialogContent = { | ||||
|     0 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_LOGOUT */: logoutDialogContent, | ||||
|     1 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_SHUTDOWN */: shutdownDialogContent, | ||||
|     2 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_RESTART */: restartDialogContent | ||||
| }; | ||||
|  | ||||
| function findAppFromInhibitor(inhibitor) { | ||||
|     let desktopFile = inhibitor.app_id; | ||||
|  | ||||
|     if (!GLib.str_has_suffix(desktopFile, '.desktop')) | ||||
|       desktopFile += '.desktop'; | ||||
|  | ||||
|     let candidateDesktopFiles = []; | ||||
|  | ||||
|     candidateDesktopFiles.push(desktopFile); | ||||
|     candidateDesktopFiles.push('gnome-' + desktopFile); | ||||
|  | ||||
|     let appSystem = Shell.AppSystem.get_default(); | ||||
|     let app = null; | ||||
|     for (let i = 0; i < candidateDesktopFiles.length; i++) { | ||||
|         try { | ||||
|             app = appSystem.get_app(candidateDesktopFiles[i]); | ||||
|  | ||||
|             if (app) | ||||
|                 break; | ||||
|         } catch(e) { | ||||
|             // ignore errors | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return app; | ||||
| } | ||||
|  | ||||
| function ListItem(app, reason) { | ||||
|     this._init(app, reason); | ||||
| } | ||||
|  | ||||
| ListItem.prototype = { | ||||
|     _init: function(app, reason) { | ||||
|         this._app = app; | ||||
|         this._reason = reason; | ||||
|  | ||||
|         if (this._reason == null) | ||||
|           this._reason = ''; | ||||
|  | ||||
|         let layout = new St.BoxLayout({ vertical: false}); | ||||
|  | ||||
|         this.actor = new St.Button({ style_class: 'end-session-dialog-app-list-item', | ||||
|                                      can_focus:   true, | ||||
|                                      child:       layout, | ||||
|                                      reactive:    true, | ||||
|                                      x_align:     St.Align.START, | ||||
|                                      x_fill:      true }); | ||||
|  | ||||
|         this._icon = this._app.create_icon_texture(_ITEM_ICON_SIZE); | ||||
|  | ||||
|         let iconBin = new St.Bin({ style_class: 'end-session-dialog-app-list-item-icon', | ||||
|                                    child:       this._icon }); | ||||
|         layout.add(iconBin); | ||||
|  | ||||
|         let textLayout = new St.BoxLayout({ style_class: 'end-session-dialog-app-list-item-text-box', | ||||
|                                             vertical:    true }); | ||||
|         layout.add(textLayout); | ||||
|  | ||||
|         this._nameLabel = new St.Label({ text:        this._app.get_name(), | ||||
|                                          style_class: 'end-session-dialog-app-list-item-name' }); | ||||
|         textLayout.add(this._nameLabel, | ||||
|                        { expand: false, | ||||
|                          x_fill: true }); | ||||
|  | ||||
|         this._descriptionLabel = new St.Label({ text:        this._reason, | ||||
|                                                 style_class: 'end-session-dialog-app-list-item-description' }); | ||||
|         textLayout.add(this._descriptionLabel, | ||||
|                        { expand: true, | ||||
|                          x_fill: true }); | ||||
|  | ||||
|         this.actor.connect('clicked', Lang.bind(this, this._onClicked)); | ||||
|     }, | ||||
|  | ||||
|     _onClicked: function() { | ||||
|         this.emit('activate'); | ||||
|         this._app.activate(-1); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(ListItem.prototype); | ||||
|  | ||||
| // The logout timer only shows updates every 10 seconds | ||||
| // until the last 10 seconds, then it shows updates every | ||||
| // second.  This function takes a given time and returns | ||||
| // what we should show to the user for that time. | ||||
| function _roundSecondsToInterval(totalSeconds, secondsLeft, interval) { | ||||
|     let time; | ||||
|  | ||||
|     time = Math.ceil(secondsLeft); | ||||
|  | ||||
|     // Final count down is in decrements of 1 | ||||
|     if (time <= interval) | ||||
|         return time; | ||||
|  | ||||
|     // Round up higher than last displayable time interval | ||||
|     time += interval - 1; | ||||
|  | ||||
|     // Then round down to that time interval | ||||
|     if (time > totalSeconds) | ||||
|         time = Math.ceil(totalSeconds); | ||||
|     else | ||||
|         time -= time % interval; | ||||
|  | ||||
|     return time; | ||||
| } | ||||
|  | ||||
| function _setLabelText(label, text) { | ||||
|     if (text) { | ||||
|         label.set_text(text); | ||||
|         label.show(); | ||||
|     } else { | ||||
|         label.set_text(''); | ||||
|         label.hide(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function EndSessionDialog() { | ||||
|     if (_endSessionDialog == null) { | ||||
|         this._init(); | ||||
|         DBus.session.exportObject('/org/gnome/SessionManager/EndSessionDialog', | ||||
|                                   this); | ||||
|         _endSessionDialog = this; | ||||
|     } | ||||
|  | ||||
|     return _endSessionDialog; | ||||
| } | ||||
|  | ||||
| function init() { | ||||
|     // This always returns the same singleton object | ||||
|     // By instantiating it initially, we register the | ||||
|     // bus object, etc. | ||||
|     let dialog = new EndSessionDialog(); | ||||
| } | ||||
|  | ||||
| EndSessionDialog.prototype = { | ||||
|     __proto__: ModalDialog.ModalDialog.prototype, | ||||
|  | ||||
|     _init: function() { | ||||
|         ModalDialog.ModalDialog.prototype._init.call(this, { styleClass: 'end-session-dialog' }); | ||||
|  | ||||
|         this._user = Gdm.UserManager.ref_default().get_user(GLib.get_user_name()); | ||||
|  | ||||
|         this._secondsLeft = 0; | ||||
|         this._totalSecondsToStayOpen = 0; | ||||
|         this._inhibitors = []; | ||||
|  | ||||
|         this.connect('destroy', | ||||
|                      Lang.bind(this, this._onDestroy)); | ||||
|         this.connect('opened', | ||||
|                      Lang.bind(this, this._onOpened)); | ||||
|  | ||||
|         this._userLoadedId = this._user.connect('notify::is_loaded', | ||||
|                                                 Lang.bind(this, this._updateContent)); | ||||
|  | ||||
|         this._userChangedId = this._user.connect('changed', | ||||
|                                                  Lang.bind(this, this._updateContent)); | ||||
|  | ||||
|         let mainContentLayout = new St.BoxLayout({ vertical: false }); | ||||
|         this.contentLayout.add(mainContentLayout, | ||||
|                                { x_fill: true, | ||||
|                                  y_fill: false }); | ||||
|  | ||||
|         this._iconBin = new St.Bin(); | ||||
|         mainContentLayout.add(this._iconBin, | ||||
|                               { x_fill:  true, | ||||
|                                 y_fill:  false, | ||||
|                                 x_align: St.Align.END, | ||||
|                                 y_align: St.Align.START }); | ||||
|  | ||||
|         let messageLayout = new St.BoxLayout({ vertical: true }); | ||||
|         mainContentLayout.add(messageLayout, | ||||
|                               { y_align: St.Align.START }); | ||||
|  | ||||
|         this._subjectLabel = new St.Label({ style_class: 'end-session-dialog-subject' }); | ||||
|  | ||||
|         messageLayout.add(this._subjectLabel, | ||||
|                           { y_fill:  false, | ||||
|                             y_align: St.Align.START }); | ||||
|  | ||||
|         this._descriptionLabel = new St.Label({ style_class: 'end-session-dialog-description' }); | ||||
|         this._descriptionLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this._descriptionLabel.clutter_text.line_wrap = true; | ||||
|  | ||||
|         messageLayout.add(this._descriptionLabel, | ||||
|                           { y_fill:  true, | ||||
|                             y_align: St.Align.START }); | ||||
|  | ||||
|         let scrollView = new St.ScrollView({ style_class: 'end-session-dialog-app-list'}); | ||||
|         scrollView.set_policy(Gtk.PolicyType.NEVER, | ||||
|                               Gtk.PolicyType.AUTOMATIC); | ||||
|         this.contentLayout.add(scrollView, | ||||
|                                { x_fill: true, | ||||
|                                  y_fill: true }); | ||||
|         scrollView.hide(); | ||||
|  | ||||
|         this._applicationList = new St.BoxLayout({ vertical: true }); | ||||
|         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_children().length == 1) | ||||
|                                               scrollView.show(); | ||||
|                                       })); | ||||
|  | ||||
|         this._applicationList.connect('actor-removed', | ||||
|                                       Lang.bind(this, function() { | ||||
|                                           if (this._applicationList.get_children().length == 0) | ||||
|                                               scrollView.hide(); | ||||
|                                       })); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         this._user.disconnect(this._userLoadedId); | ||||
|         this._user.disconnect(this._userChangedId); | ||||
|     }, | ||||
|  | ||||
|     _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 + '");'); | ||||
|         } 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; | ||||
|  | ||||
|         let dialogContent = DialogContent[this._type]; | ||||
|  | ||||
|         let subject = dialogContent.subject; | ||||
|         let description; | ||||
|  | ||||
|         if (this._user.is_loaded && !dialogContent.iconName) { | ||||
|             let iconFile = this._user.get_icon_file(); | ||||
|  | ||||
|             this._setIconFromFile(iconFile, dialogContent.iconStyleClass); | ||||
|         } else if (dialogContent.iconName) { | ||||
|             this._setIconFromName(dialogContent.iconName, | ||||
|                                   dialogContent.iconStyleClass); | ||||
|         } | ||||
|  | ||||
|         if (this._inhibitors.length > 0) { | ||||
|             this._stopTimer(); | ||||
|             description = dialogContent.inhibitedDescription; | ||||
|         } else if (this._secondsLeft > 0 && this._inhibitors.length == 0) { | ||||
|             let displayTime = _roundSecondsToInterval(this._totalSecondsToStayOpen, | ||||
|                                                       this._secondsLeft, | ||||
|                                                       10); | ||||
|  | ||||
|             if (this._user.is_loaded) { | ||||
|                 let realName = this._user.get_real_name(); | ||||
|  | ||||
|                 if (realName != null) { | ||||
|                     if (dialogContent.subjectWithUser) | ||||
|                         subject = dialogContent.subjectWithUser.format(realName); | ||||
|  | ||||
|                     if (dialogContent.uninhibitedDescriptionWithUser) | ||||
|                         description = dialogContent.uninhibitedDescriptionWithUser.format(realName, displayTime); | ||||
|                     else | ||||
|                         description = dialogContent.uninhibitedDescription.format(displayTime); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (!description) | ||||
|                 description = dialogContent.uninhibitedDescription.format(displayTime); | ||||
|         } else { | ||||
|             description = dialogContent.endDescription; | ||||
|         } | ||||
|  | ||||
|         _setLabelText(this._subjectLabel, subject); | ||||
|         _setLabelText(this._descriptionLabel, description); | ||||
|     }, | ||||
|  | ||||
|     _updateButtons: function() { | ||||
|         if (this.state != ModalDialog.State.OPENING && | ||||
|             this.state != ModalDialog.State.OPENED) | ||||
|             return; | ||||
|  | ||||
|         let dialogContent = DialogContent[this._type]; | ||||
|         let buttons = [{ action: Lang.bind(this, this.cancel), | ||||
|                          label:  _("Cancel"), | ||||
|                          key:    Clutter.Escape }]; | ||||
|  | ||||
|         for (let i = 0; i < dialogContent.confirmButtons.length; i++) { | ||||
|             let signal = dialogContent.confirmButtons[i].signal; | ||||
|             let label = dialogContent.confirmButtons[i].label; | ||||
|             buttons.push({ action: Lang.bind(this, function() { | ||||
|                                        this._confirm(signal); | ||||
|                                    }), | ||||
|                            label: label }); | ||||
|         } | ||||
|  | ||||
|         this.setButtons(buttons); | ||||
|     }, | ||||
|  | ||||
|     close: function() { | ||||
|         ModalDialog.ModalDialog.prototype.close.call(this); | ||||
|         DBus.session.emit_signal('/org/gnome/SessionManager/EndSessionDialog', | ||||
|                                  'org.gnome.SessionManager.EndSessionDialog', | ||||
|                                  'Closed', '', []); | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|         this._stopTimer(); | ||||
|         DBus.session.emit_signal('/org/gnome/SessionManager/EndSessionDialog', | ||||
|                                  'org.gnome.SessionManager.EndSessionDialog', | ||||
|                                  'Canceled', '', []); | ||||
|         this.close(global.get_current_time()); | ||||
|     }, | ||||
|  | ||||
|     _confirm: function(signal) { | ||||
|         this._fadeOutDialog(); | ||||
|         this._stopTimer(); | ||||
|         DBus.session.emit_signal('/org/gnome/SessionManager/EndSessionDialog', | ||||
|                                  'org.gnome.SessionManager.EndSessionDialog', | ||||
|                                  signal, '', []); | ||||
|     }, | ||||
|  | ||||
|     _onOpened: function() { | ||||
|         if (this._inhibitors.length == 0) | ||||
|             this._startTimer(); | ||||
|     }, | ||||
|  | ||||
|     _startTimer: function() { | ||||
|         this._secondsLeft = this._totalSecondsToStayOpen; | ||||
|         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() { | ||||
|         Tweener.removeTweens(this); | ||||
|         this._secondsLeft = 0; | ||||
|     }, | ||||
|  | ||||
|     _onInhibitorLoaded: function(inhibitor) { | ||||
|         if (this._inhibitors.indexOf(inhibitor) < 0) { | ||||
|             // Stale inhibitor | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let app = findAppFromInhibitor(inhibitor); | ||||
|  | ||||
|         if (app) { | ||||
|             let item = new ListItem(app, inhibitor.reason); | ||||
|             item.connect('activate', | ||||
|                          Lang.bind(this, function() { | ||||
|                              this.close(global.get_current_time()); | ||||
|                          })); | ||||
|             this._applicationList.add(item.actor, { x_fill: true }); | ||||
|             this._stopTimer(); | ||||
|         } else { | ||||
|             // inhibiting app is a service, not an application | ||||
|             this._inhibitors.splice(this._inhibitors.indexOf(inhibitor), 1); | ||||
|         } | ||||
|  | ||||
|         this._updateContent(); | ||||
|     }, | ||||
|  | ||||
|     OpenAsync: function(type, timestamp, totalSecondsToStayOpen, inhibitorObjectPaths, callback) { | ||||
|         this._totalSecondsToStayOpen = totalSecondsToStayOpen; | ||||
|         this._inhibitors = []; | ||||
|         this._applicationList.destroy_children(); | ||||
|         this._type = type; | ||||
|  | ||||
|         if (!(this._type in DialogContent)) | ||||
|             throw new DBus.DBusError('org.gnome.Shell.ModalDialog.TypeError', | ||||
|                                      "Unknown dialog type requested"); | ||||
|  | ||||
|         for (let i = 0; i < inhibitorObjectPaths.length; i++) { | ||||
|             let inhibitor = new GnomeSession.Inhibitor(inhibitorObjectPaths[i]); | ||||
|  | ||||
|             inhibitor.connect('is-loaded', | ||||
|                               Lang.bind(this, function() { | ||||
|                                   this._onInhibitorLoaded(inhibitor); | ||||
|                               })); | ||||
|             this._inhibitors.push(inhibitor); | ||||
|         } | ||||
|  | ||||
|         if (!this.open(timestamp)) | ||||
|             throw new DBus.DBusError('org.gnome.Shell.ModalDialog.GrabError', | ||||
|                                      "Cannot grab pointer and keyboard"); | ||||
|  | ||||
|         this._updateButtons(); | ||||
|         this._updateContent(); | ||||
|  | ||||
|         let signalId = this.connect('opened', | ||||
|                                     Lang.bind(this, function() { | ||||
|                                         callback(); | ||||
|                                         this.disconnect(signalId); | ||||
|                                     })); | ||||
|     } | ||||
| }; | ||||
| DBus.conformExport(EndSessionDialog.prototype, EndSessionDialogIface); | ||||
| @@ -1,15 +1,9 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter;; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
| const Gettext_gtk30 = imports.gettext.domain('gtk30'); | ||||
| const Nbtk = imports.gi.Nbtk; | ||||
|  | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const Format = imports.misc.format; | ||||
|  | ||||
| // "monkey patch" in some varargs ClutterContainer methods; we need | ||||
| // to do this per-container class since there is no representation | ||||
| // of interfaces in Javascript | ||||
| @@ -17,7 +11,7 @@ function _patchContainerClass(containerClass) { | ||||
|     // This one is a straightforward mapping of the C method | ||||
|     containerClass.prototype.child_set = function(actor, props) { | ||||
|         let meta = this.get_child_meta(actor); | ||||
|         for (let prop in props) | ||||
|         for (prop in props) | ||||
|             meta[prop] = props[prop]; | ||||
|     }; | ||||
|  | ||||
| @@ -31,74 +25,8 @@ function _patchContainerClass(containerClass) { | ||||
|     }; | ||||
| } | ||||
|  | ||||
| // Replace @method with something that throws an error instead | ||||
| function _blockMethod(method, replacement, reason) { | ||||
|     let match = method.match(/^(.+)\.([^.]+)$/); | ||||
|     if (!match) | ||||
|         throw new Error('Bad method name "' + method + '"'); | ||||
|     let proto = 'imports.gi.' + match[1] + '.prototype'; | ||||
|     let property = match[2]; | ||||
|  | ||||
|     if (!global.set_property_mutable(proto, property, true)) | ||||
|         throw new Error('Bad method name "' + method + '"'); | ||||
|  | ||||
|     // eval() is evil in general, but we know it's safe here since | ||||
|     // set_property_mutable() would have failed if proto was | ||||
|     // malformed. | ||||
|     let node = eval(proto); | ||||
|  | ||||
|     let msg = 'Do not use "' + method + '".'; | ||||
|     if (replacement) | ||||
|         msg += ' Use "' + replacement + '" instead.'; | ||||
|     if (reason) | ||||
|         msg += ' (' + reason + ')'; | ||||
|  | ||||
|     node[property] = function() { | ||||
|         throw new Error(msg); | ||||
|     }; | ||||
|  | ||||
|     global.set_property_mutable(proto, property, false); | ||||
| } | ||||
| _patchContainerClass(Nbtk.BoxLayout); | ||||
|  | ||||
| function init() { | ||||
|     Tweener.init(); | ||||
|     String.prototype.format = Format.format; | ||||
|  | ||||
|     // Work around https://bugzilla.mozilla.org/show_bug.cgi?id=508783 | ||||
|     Date.prototype.toLocaleFormat = function(format) { | ||||
|         return Shell.util_format_date(format, this.getTime()); | ||||
|     }; | ||||
|  | ||||
|     // Set the default direction for St widgets (this needs to be done before any use of St) | ||||
|     if (Gettext_gtk30.gettext('default:LTR') == 'default:RTL') { | ||||
|         St.Widget.set_default_direction(St.TextDirection.RTL); | ||||
|     } | ||||
|  | ||||
|     let slowdownEnv = GLib.getenv('GNOME_SHELL_SLOWDOWN_FACTOR'); | ||||
|     if (slowdownEnv) { | ||||
|         let factor = parseFloat(slowdownEnv); | ||||
|         if (!isNaN(factor) && factor > 0.0) | ||||
|             St.set_slow_down_factor(factor); | ||||
|     } | ||||
|  | ||||
|     _patchContainerClass(St.BoxLayout); | ||||
|     _patchContainerClass(St.Table); | ||||
|  | ||||
|     Clutter.Actor.prototype.toString = function() { | ||||
|         return St.describe_actor(this); | ||||
|     }; | ||||
|  | ||||
|     if (window.global === undefined) // test environment | ||||
|         return; | ||||
|  | ||||
|     _blockMethod('Clutter.Event.get_state', 'Shell.get_event_state', | ||||
|                  'gjs\'s handling of Clutter.ModifierType is broken. See bug 597292.'); | ||||
|     _blockMethod('Gdk.Window.get_device_position', 'global.get_pointer', | ||||
|                  'gjs\'s handling of Gdk.ModifierType is broken. See bug 597292.'); | ||||
|  | ||||
|     // Now close the back door to prevent extensions from trying to | ||||
|     // abuse it. We can't actually delete it since | ||||
|     // Shell.Global.prototype itself is read-only. | ||||
|     global.set_property_mutable('imports.gi.Shell.Global.prototype', 'set_property_mutable', true); | ||||
|     Shell.Global.prototype.set_property_mutable = undefined; | ||||
| } | ||||
|   | ||||
| @@ -1,207 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
|  | ||||
| const ExtensionState = { | ||||
|     ENABLED: 1, | ||||
|     DISABLED: 2, | ||||
|     ERROR: 3, | ||||
|     OUT_OF_DATE: 4 | ||||
| }; | ||||
|  | ||||
| const ExtensionType = { | ||||
|     SYSTEM: 1, | ||||
|     PER_USER: 2 | ||||
| }; | ||||
|  | ||||
| // Maps uuid -> metadata object | ||||
| const extensionMeta = {}; | ||||
| // Maps uuid -> importer object (extension directory tree) | ||||
| const extensions = {}; | ||||
| // Array of uuids | ||||
| var disabledExtensions; | ||||
| // GFile for user extensions | ||||
| var userExtensionsDir = null; | ||||
|  | ||||
| /** | ||||
|  * versionCheck: | ||||
|  * @required: an array of versions we're compatible with | ||||
|  * @current: the version we have | ||||
|  * | ||||
|  * Check if a component is compatible for an extension. | ||||
|  * @required is an array, and at least one version must match. | ||||
|  * @current must be in the format <major>.<minor>.<point>.<micro> | ||||
|  * <micro> is always ignored | ||||
|  * <point> is ignored if <minor> is even (so you can target the | ||||
|  * whole stable release) | ||||
|  * <minor> and <major> must match | ||||
|  * Each target version must be at least <major> and <minor> | ||||
|  */ | ||||
| function versionCheck(required, current) { | ||||
|     let currentArray = current.split('.'); | ||||
|     let major = currentArray[0]; | ||||
|     let minor = currentArray[1]; | ||||
|     let point = currentArray[2]; | ||||
|     for (let i = 0; i < required.length; i++) { | ||||
|         let requiredArray = required[i].split('.'); | ||||
|         if (requiredArray[0] == major && | ||||
|             requiredArray[1] == minor && | ||||
|             (requiredArray[2] == point || | ||||
|              (requiredArray[2] == undefined && parseInt(minor) % 2 == 0))) | ||||
|             return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| function loadExtension(dir, enabled, type) { | ||||
|     let info; | ||||
|     let baseErrorString = 'While loading extension from "' + dir.get_parse_name() + '": '; | ||||
|  | ||||
|     let metadataFile = dir.get_child('metadata.json'); | ||||
|     if (!metadataFile.query_exists(null)) { | ||||
|         global.logError(baseErrorString + 'Missing metadata.json'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     let [success, metadataContents, len, etag] = metadataFile.load_contents(null); | ||||
|     let meta; | ||||
|     try { | ||||
|         meta = JSON.parse(metadataContents); | ||||
|     } catch (e) { | ||||
|         global.logError(baseErrorString + 'Failed to parse metadata.json: ' + e); | ||||
|         return; | ||||
|     } | ||||
|     let requiredProperties = ['uuid', 'name', 'description', 'shell-version']; | ||||
|     for (let i = 0; i < requiredProperties.length; i++) { | ||||
|         let prop = requiredProperties[i]; | ||||
|         if (!meta[prop]) { | ||||
|             global.logError(baseErrorString + 'missing "' + prop + '" property in metadata.json'); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (extensions[meta.uuid] != undefined) { | ||||
|         global.logError(baseErrorString + "extension already loaded"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Encourage people to add this | ||||
|     if (!meta['url']) { | ||||
|         global.log(baseErrorString + 'Warning: Missing "url" property in metadata.json'); | ||||
|     } | ||||
|  | ||||
|     let base = dir.get_basename(); | ||||
|     if (base != meta.uuid) { | ||||
|         global.logError(baseErrorString + 'uuid "' + meta.uuid + '" from metadata.json does not match directory name "' + base + '"'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!versionCheck(meta['shell-version'], Config.PACKAGE_VERSION) || | ||||
|         (meta['js-version'] && !versionCheck(meta['js-version'], Config.GJS_VERSION))) { | ||||
|         global.logError(baseErrorString + 'extension is not compatible with current GNOME Shell and/or GJS version'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     extensionMeta[meta.uuid] = meta; | ||||
|     extensionMeta[meta.uuid].type = type; | ||||
|     extensionMeta[meta.uuid].path = dir.get_path(); | ||||
|     if (!enabled) { | ||||
|         extensionMeta[meta.uuid].state = ExtensionState.DISABLED; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Default to error, we set success as the last step | ||||
|     extensionMeta[meta.uuid].state = ExtensionState.ERROR; | ||||
|  | ||||
|     let extensionJs = dir.get_child('extension.js'); | ||||
|     if (!extensionJs.query_exists(null)) { | ||||
|         global.logError(baseErrorString + '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) { | ||||
|             global.logError(baseErrorString + 'Stylesheet parse error: ' + e); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let extensionModule; | ||||
|     try { | ||||
|         global.add_extension_importer('imports.ui.extensionSystem.extensions', meta.uuid, dir.get_path()); | ||||
|         extensionModule = extensions[meta.uuid].extension; | ||||
|     } catch (e) { | ||||
|         if (stylesheetPath != null) | ||||
|             theme.unload_stylesheet(stylesheetPath); | ||||
|         global.logError(baseErrorString + e); | ||||
|         return; | ||||
|     } | ||||
|     if (!extensionModule.main) { | ||||
|         global.logError(baseErrorString + 'missing \'main\' function'); | ||||
|         return; | ||||
|     } | ||||
|     try { | ||||
|         extensionModule.main(meta); | ||||
|     } catch (e) { | ||||
|         if (stylesheetPath != null) | ||||
|             theme.unload_stylesheet(stylesheetPath); | ||||
|         global.logError(baseErrorString + 'Failed to evaluate main function:' + e); | ||||
|         return; | ||||
|     } | ||||
|     extensionMeta[meta.uuid].state = ExtensionState.ENABLED; | ||||
|     global.log('Loaded extension ' + meta.uuid); | ||||
| } | ||||
|  | ||||
| function init() { | ||||
|     let userExtensionsPath = GLib.build_filenamev([global.userdatadir, 'extensions']); | ||||
|     userExtensionsDir = Gio.file_new_for_path(userExtensionsPath); | ||||
|     try { | ||||
|         userExtensionsDir.make_directory_with_parents(null); | ||||
|     } catch (e) { | ||||
|         global.logError('' + e); | ||||
|     } | ||||
|  | ||||
|     disabledExtensions = global.settings.get_strv('disabled-extensions', -1); | ||||
| } | ||||
|  | ||||
| function _loadExtensionsIn(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) | ||||
|             continue; | ||||
|         let name = info.get_name(); | ||||
|         let enabled = disabledExtensions.indexOf(name) < 0; | ||||
|         let child = dir.get_child(name); | ||||
|         loadExtension(child, enabled, type); | ||||
|     } | ||||
|     fileEnum.close(null); | ||||
| } | ||||
|  | ||||
| function loadExtensions() { | ||||
|     _loadExtensionsIn(userExtensionsDir, ExtensionType.PER_USER); | ||||
|     let systemDataDirs = GLib.get_system_data_dirs(); | ||||
|     for (let i = 0; i < systemDataDirs.length; i++) { | ||||
|         let dirPath = systemDataDirs[i] + '/gnome-shell/extensions'; | ||||
|         let dir = Gio.file_new_for_path(dirPath); | ||||
|         if (dir.query_exists(null)) | ||||
|             _loadExtensionsIn(dir, ExtensionType.SYSTEM); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										785
									
								
								js/ui/genericDisplay.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,785 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Big = imports.gi.Big; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Signals = imports.signals; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Button = imports.ui.button; | ||||
| const DND = imports.ui.dnd; | ||||
| const Link = imports.ui.link; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const ITEM_DISPLAY_NAME_COLOR = new Clutter.Color(); | ||||
| ITEM_DISPLAY_NAME_COLOR.from_pixel(0xffffffff); | ||||
| const ITEM_DISPLAY_DESCRIPTION_COLOR = new Clutter.Color(); | ||||
| ITEM_DISPLAY_DESCRIPTION_COLOR.from_pixel(0xffffffbb); | ||||
| const ITEM_DISPLAY_BACKGROUND_COLOR = new Clutter.Color(); | ||||
| ITEM_DISPLAY_BACKGROUND_COLOR.from_pixel(0x00000000); | ||||
| const ITEM_DISPLAY_SELECTED_BACKGROUND_COLOR = new Clutter.Color(); | ||||
| ITEM_DISPLAY_SELECTED_BACKGROUND_COLOR.from_pixel(0x4f6fadaa); | ||||
| const DISPLAY_CONTROL_SELECTED_COLOR = new Clutter.Color(); | ||||
| DISPLAY_CONTROL_SELECTED_COLOR.from_pixel(0x112288ff); | ||||
| const PREVIEW_BOX_BACKGROUND_COLOR = new Clutter.Color(); | ||||
| PREVIEW_BOX_BACKGROUND_COLOR.from_pixel(0xADADADf0); | ||||
|  | ||||
| const DEFAULT_PADDING = 4; | ||||
|  | ||||
| const ITEM_DISPLAY_HEIGHT = 50; | ||||
| const ITEM_DISPLAY_ICON_SIZE = 48; | ||||
| const ITEM_DISPLAY_PADDING = 1; | ||||
| const ITEM_DISPLAY_PADDING_RIGHT = 2; | ||||
| const DEFAULT_COLUMN_GAP = 6; | ||||
|  | ||||
| const PREVIEW_ICON_SIZE = 96; | ||||
| const PREVIEW_BOX_PADDING = 6; | ||||
| const PREVIEW_BOX_SPACING = DEFAULT_PADDING; | ||||
| const PREVIEW_BOX_CORNER_RADIUS = 10; | ||||
| // how far relative to the full item width the preview box should be placed | ||||
| const PREVIEW_PLACING = 3/4; | ||||
| const PREVIEW_DETAILS_MIN_WIDTH = PREVIEW_ICON_SIZE * 2; | ||||
|  | ||||
| const INFORMATION_BUTTON_SIZE = 16; | ||||
|  | ||||
| /* This is a virtual class that represents a single display item containing | ||||
|  * a name, a description, and an icon. It allows selecting an item and represents | ||||
|  * it by highlighting it with a different background color than the default. | ||||
|  */ | ||||
| function GenericDisplayItem() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| GenericDisplayItem.prototype = { | ||||
|     _init: function() { | ||||
|         this.actor = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                    spacing: ITEM_DISPLAY_PADDING, | ||||
|                                    reactive: true, | ||||
|                                    background_color: ITEM_DISPLAY_BACKGROUND_COLOR, | ||||
|                                    corner_radius: 4, | ||||
|                                    height: ITEM_DISPLAY_HEIGHT }); | ||||
|  | ||||
|         this.actor._delegate = this; | ||||
|         this.actor.connect('button-release-event', | ||||
|                            Lang.bind(this, | ||||
|                                      function() { | ||||
|                                          // Activates the item by launching it | ||||
|                                          this.emit('activate'); | ||||
|                                          return true;   | ||||
|                                      })); | ||||
|  | ||||
|         let draggable = DND.makeDraggable(this.actor); | ||||
|         draggable.connect('drag-begin', Lang.bind(this, this._onDragBegin)); | ||||
|  | ||||
|         this._infoContent = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                           spacing: DEFAULT_PADDING }); | ||||
|         this.actor.append(this._infoContent, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         this._iconBox = new Big.Box(); | ||||
|         this._infoContent.append(this._iconBox, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         this._infoText = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                        spacing: DEFAULT_PADDING }); | ||||
|         this._infoContent.append(this._infoText, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         let infoIconUri = "file://" + global.imagedir + "info.svg"; | ||||
|         let infoIcon = Shell.TextureCache.get_default().load_uri_sync(Shell.TextureCachePolicy.FOREVER, | ||||
|                                                                       infoIconUri, | ||||
|                                                                       INFORMATION_BUTTON_SIZE, | ||||
|                                                                       INFORMATION_BUTTON_SIZE); | ||||
|         this._informationButton = new Button.iconButton(this.actor, INFORMATION_BUTTON_SIZE, infoIcon); | ||||
|         let buttonBox = new Big.Box({ width: INFORMATION_BUTTON_SIZE + 2 * DEFAULT_PADDING, | ||||
|                                       height: INFORMATION_BUTTON_SIZE, | ||||
|                                       padding_left: DEFAULT_PADDING, padding_right: DEFAULT_PADDING, | ||||
|                                       y_align: Big.BoxAlignment.CENTER }); | ||||
|         buttonBox.append(this._informationButton.actor, Big.BoxPackFlags.NONE); | ||||
|         this.actor.append(buttonBox, Big.BoxPackFlags.END); | ||||
|  | ||||
|         // Connecting to the button-press-event for the information button ensures that the actor, | ||||
|         // which is a draggable actor, does not get the button-press-event and doesn't initiate | ||||
|         // the dragging, which then prevents us from getting the button-release-event for the button. | ||||
|         this._informationButton.actor.connect('button-press-event', | ||||
|                                               Lang.bind(this, | ||||
|                                                         function() { | ||||
|                                                             return true; | ||||
|                                                         })); | ||||
|         this._informationButton.actor.connect('button-release-event', | ||||
|                                               Lang.bind(this, | ||||
|                                                         function() { | ||||
|                                                             // Selects the item by highlighting it and displaying its details | ||||
|                                                             this.emit('show-details'); | ||||
|                                                             return true; | ||||
|                                                         })); | ||||
|  | ||||
|         this._name = null; | ||||
|         this._description = null; | ||||
|         this._icon = null; | ||||
|  | ||||
|         // An array of details description actors that we create over time for the item. | ||||
|         // It is used for updating the description text inside the details actor when | ||||
|         // the description text for the item is updated. | ||||
|         this._detailsDescriptions = []; | ||||
|     }, | ||||
|  | ||||
|     //// Draggable object interface //// | ||||
|  | ||||
|     // Returns a cloned texture of the item's icon to represent the item as it  | ||||
|     // is being dragged.  | ||||
|     getDragActor: function(stageX, stageY) { | ||||
|         return this._createIcon(); | ||||
|     }, | ||||
|  | ||||
|     // Returns the item icon, a separate copy of which is used to | ||||
|     // represent the item as it is being dragged. This is used to | ||||
|     // determine a snap-back location for the drag icon if it does | ||||
|     // not get accepted by any drop target. | ||||
|     getDragActorSource: function() { | ||||
|         return this._icon; | ||||
|     }, | ||||
|  | ||||
|     //// Public methods //// | ||||
|  | ||||
|     // Shows the information button when the item was drawn under the mouse pointer. | ||||
|     onDrawnUnderPointer: function() { | ||||
|         this._informationButton.show(); | ||||
|     }, | ||||
|  | ||||
|     // Highlights the item by setting a different background color than the default  | ||||
|     // if isSelected is true, removes the highlighting otherwise. | ||||
|     markSelected: function(isSelected) { | ||||
|        let color; | ||||
|        if (isSelected) { | ||||
|            color = ITEM_DISPLAY_SELECTED_BACKGROUND_COLOR; | ||||
|            this._informationButton.forceShow(true); | ||||
|        } | ||||
|        else { | ||||
|            color = ITEM_DISPLAY_BACKGROUND_COLOR; | ||||
|            this._informationButton.forceShow(false); | ||||
|        } | ||||
|        this.actor.background_color = color; | ||||
|     }, | ||||
|  | ||||
|     /* | ||||
|      * Returns an actor containing item details. In the future details can have more information than what | ||||
|      * the preview pop-up has and be item-type specific. | ||||
|      */ | ||||
|     createDetailsActor: function() { | ||||
|  | ||||
|         let details = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                     spacing: PREVIEW_BOX_SPACING }); | ||||
|  | ||||
|         let mainDetails = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                         spacing: PREVIEW_BOX_SPACING }); | ||||
|  | ||||
|         // Inner box with name and description | ||||
|         let textDetails = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                         spacing: PREVIEW_BOX_SPACING }); | ||||
|         let detailsName = new Clutter.Text({ color: ITEM_DISPLAY_NAME_COLOR, | ||||
|                                              font_name: "Sans bold 14px", | ||||
|                                              line_wrap: true, | ||||
|                                              text: this._name.text }); | ||||
|         textDetails.append(detailsName, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         let detailsDescription = new Clutter.Text({ color: ITEM_DISPLAY_NAME_COLOR, | ||||
|                                                     font_name: "Sans 14px", | ||||
|                                                     line_wrap: true, | ||||
|                                                     text: this._description.text }); | ||||
|         textDetails.append(detailsDescription, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         this._detailsDescriptions.push(detailsDescription); | ||||
|  | ||||
|         mainDetails.append(textDetails, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         let previewIcon = this._createPreviewIcon(); | ||||
|         let largePreviewIcon = this._createLargePreviewIcon(); | ||||
|  | ||||
|         if (previewIcon != null && largePreviewIcon == null) { | ||||
|             mainDetails.prepend(previewIcon, Big.BoxPackFlags.NONE); | ||||
|         } | ||||
|  | ||||
|         details.append(mainDetails, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         if (largePreviewIcon != null) { | ||||
|             let largePreview = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL }); | ||||
|             largePreview.append(largePreviewIcon, Big.BoxPackFlags.NONE); | ||||
|             details.append(largePreview, Big.BoxPackFlags.NONE); | ||||
|         } | ||||
|     | ||||
|         return details; | ||||
|     }, | ||||
|  | ||||
|     // Destroys the item. | ||||
|     destroy: function() { | ||||
|       this.actor.destroy(); | ||||
|     }, | ||||
|  | ||||
|     //// Pure virtual public methods //// | ||||
|  | ||||
|     // Performes an action associated with launching this item, such as opening a file or an application. | ||||
|     launch: function() { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     //// Protected methods //// | ||||
|  | ||||
|     /* | ||||
|      * Creates the graphical elements for the item based on the item information. | ||||
|      * | ||||
|      * nameText - name of the item | ||||
|      * descriptionText - short description of the item | ||||
|      */ | ||||
|     _setItemInfo: function(nameText, descriptionText) { | ||||
|         if (this._name != null) { | ||||
|             // this also removes this._name from the parent container, | ||||
|             // so we don't need to call this.actor.remove_actor(this._name) directly | ||||
|             this._name.destroy(); | ||||
|             this._name = null; | ||||
|         }  | ||||
|         if (this._description != null) { | ||||
|             this._description.destroy(); | ||||
|             this._description = null; | ||||
|         }  | ||||
|         if (this._icon != null) { | ||||
|             // though we get the icon from elsewhere, we assume its ownership here, | ||||
|             // and therefore should be responsible for distroying it | ||||
|             this._icon.destroy(); | ||||
|             this._icon = null; | ||||
|         } | ||||
|  | ||||
|         this._icon = this._createIcon(); | ||||
|         this._iconBox.append(this._icon, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         this._name = new Clutter.Text({ color: ITEM_DISPLAY_NAME_COLOR, | ||||
|                                         font_name: "Sans 14px", | ||||
|                                         ellipsize: Pango.EllipsizeMode.END, | ||||
|                                         text: nameText }); | ||||
|         this._infoText.append(this._name, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         this._description = new Clutter.Text({ color: ITEM_DISPLAY_DESCRIPTION_COLOR, | ||||
|                                                font_name: "Sans 12px", | ||||
|                                                ellipsize: Pango.EllipsizeMode.END, | ||||
|                                                text: descriptionText ? descriptionText : "" | ||||
|                                             }); | ||||
|         this._infoText.append(this._description, Big.BoxPackFlags.EXPAND); | ||||
|     }, | ||||
|  | ||||
|     // Sets the description text for the item, including the description text | ||||
|     // in the details actors that have been created for the item. | ||||
|     _setDescriptionText: function(text) { | ||||
|         this._description.text = text; | ||||
|         for (let i = 0; i < this._detailsDescriptions.length; i++) { | ||||
|             let detailsDescription = this._detailsDescriptions[i]; | ||||
|             if (detailsDescription != null) { | ||||
|                 detailsDescription.text = text; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     //// Virtual protected methods //// | ||||
|  | ||||
|     // Creates and returns a large preview icon, but only if we have a detailed image. | ||||
|     _createLargePreviewIcon : function() { | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     //// Pure virtual protected methods //// | ||||
|  | ||||
|     // Returns an icon for the item. | ||||
|     _createIcon: function() { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     // Returns a preview icon for the item. | ||||
|     _createPreviewIcon: function() { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     //// Private methods //// | ||||
|  | ||||
|     // Hides the information button once the item starts being dragged. | ||||
|     _onDragBegin : function (draggable, time) { | ||||
|         // For some reason, we are not getting leave-event signal when we are dragging an item, | ||||
|         // so we should remove the link manually. | ||||
|         this._informationButton.actor.hide(); | ||||
|     }  | ||||
| }; | ||||
|  | ||||
| Signals.addSignalMethods(GenericDisplayItem.prototype); | ||||
|  | ||||
| /* This is a virtual class that represents a display containing a collection of items | ||||
|  * that can be filtered with a search string. | ||||
|  */ | ||||
| function GenericDisplay() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| GenericDisplay.prototype = { | ||||
|     _init : function() { | ||||
|         this._search = ''; | ||||
|         this._expanded = false; | ||||
|  | ||||
|         this._maxItemsPerPage = null; | ||||
|         this._list = new Shell.OverflowList({ spacing: 6.0, | ||||
|                                               item_height: ITEM_DISPLAY_HEIGHT }); | ||||
|  | ||||
|         this._list.connect('notify::n-pages', Lang.bind(this, function () { | ||||
|             this._updateDisplayControl(true); | ||||
|         })); | ||||
|         this._list.connect('notify::page', Lang.bind(this, function () { | ||||
|             this._updateDisplayControl(false); | ||||
|         })); | ||||
|  | ||||
|         // map<itemId, Object> where Object represents the item info | ||||
|         this._allItems = {}; | ||||
|         // an array of itemIds of items that match the current request | ||||
|         // in the order in which the items should be displayed | ||||
|         this._matchedItems = []; | ||||
|         // map<itemId, GenericDisplayItem> | ||||
|         this._displayedItems = {}; | ||||
|         this._openDetailIndex = -1; | ||||
|         this._selectedIndex = -1; | ||||
|         // These two are public - .actor is the normal "actor subclass" property, | ||||
|         // but we also expose a .displayControl actor which is separate. | ||||
|         // See also getNavigationArea. | ||||
|         this.actor = this._list; | ||||
|         this.displayControl = new Big.Box({ background_color: ITEM_DISPLAY_BACKGROUND_COLOR, | ||||
|                                             spacing: 12, | ||||
|                                             orientation: Big.BoxOrientation.HORIZONTAL}); | ||||
|     }, | ||||
|  | ||||
|     //// Public methods //// | ||||
|  | ||||
|     // Sets the search string and displays the matching items. | ||||
|     setSearch: function(text) { | ||||
|         this._search = text.toLowerCase(); | ||||
|         this._redisplay(true); | ||||
|     }, | ||||
|  | ||||
|     // Launches the item that is currently selected, closing the Overview | ||||
|     activateSelected: function() { | ||||
|         if (this._selectedIndex != -1) { | ||||
|             let selected = this._findDisplayedByIndex(this._selectedIndex); | ||||
|             selected.launch(); | ||||
|             this.unsetSelected(); | ||||
|             Main.overview.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Moves the selection one up. If the selection was already on the top item, it's moved | ||||
|     // to the bottom one. Returns true if the selection actually moved up, false if it wrapped  | ||||
|     // around to the bottom. | ||||
|     selectUp: function() { | ||||
|         let count = this._list.displayedCount; | ||||
|         let selectedUp = true; | ||||
|         let prev = this._selectedIndex - 1; | ||||
|         if (this._selectedIndex <= 0) { | ||||
|             prev = count - 1; | ||||
|             selectedUp = false;  | ||||
|         } | ||||
|         this._selectIndex(prev); | ||||
|         return selectedUp; | ||||
|     }, | ||||
|  | ||||
|     // Moves the selection one down. If the selection was already on the bottom item, it's moved | ||||
|     // to the top one. Returns true if the selection actually moved down, false if it wrapped  | ||||
|     // around to the top. | ||||
|     selectDown: function() { | ||||
|         let count = this._list.displayedCount; | ||||
|         let selectedDown = true; | ||||
|         let next = this._selectedIndex + 1; | ||||
|         if (this._selectedIndex == count - 1) { | ||||
|             next = 0; | ||||
|             selectedDown = false; | ||||
|         } | ||||
|         this._selectIndex(next); | ||||
|         return selectedDown; | ||||
|     }, | ||||
|  | ||||
|     // Selects the first item among the displayed items. | ||||
|     selectFirstItem: function() { | ||||
|         if (this.hasItems()) | ||||
|             this._selectIndex(0); | ||||
|     }, | ||||
|  | ||||
|     // Selects the last item among the displayed items. | ||||
|     selectLastItem: function() { | ||||
|         let count = this._list.displayedCount; | ||||
|         if (this.hasItems()) | ||||
|             this._selectIndex(count - 1); | ||||
|     }, | ||||
|  | ||||
|     // Returns true if the display has some item selected. | ||||
|     hasSelected: function() { | ||||
|         return this._selectedIndex != -1; | ||||
|     }, | ||||
|  | ||||
|     // Removes selection if some display item is selected. | ||||
|     unsetSelected: function() { | ||||
|         this._selectIndex(-1); | ||||
|     }, | ||||
|  | ||||
|     // Returns true if the display has any displayed items. | ||||
|     hasItems: function() { | ||||
|         // TODO: figure out why this._list.displayedCount is returning a | ||||
|         // positive number when this._mathedItems.length is 0 | ||||
|         // This can be triggered if a search string is entered for which there are no matches. | ||||
|         // log("this._mathedItems.length: " + this._matchedItems.length + " this._list.displayedCount " + this._list.displayedCount); | ||||
|         return this._matchedItems.length > 0; | ||||
|     }, | ||||
|  | ||||
|     getMatchedItemsCount: function() { | ||||
|         return this._matchedItems.length; | ||||
|     }, | ||||
|  | ||||
|     // Load the initial state | ||||
|     load: function() { | ||||
|         this._redisplay(true); | ||||
|     }, | ||||
|  | ||||
|     // Should be called when the display is closed | ||||
|     resetState: function() { | ||||
|         this._filterReset(); | ||||
|         this._openDetailIndex = -1; | ||||
|     }, | ||||
|  | ||||
|     // Returns an actor which acts as a sidebar; this is used for | ||||
|     // the applications category view | ||||
|     getNavigationArea: function () { | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     createDetailsForIndex: function(index) { | ||||
|         let item = this._findDisplayedByIndex(index); | ||||
|         return item.createDetailsActor(); | ||||
|     }, | ||||
|  | ||||
|     // Displays the page specified by the pageNumber argument. | ||||
|     displayPage: function(pageNumber) { | ||||
|         // Cleanup from the previous selection, but don't unset this._selectedIndex | ||||
|         if (this.hasSelected()) { | ||||
|             this._findDisplayedByIndex(this._selectedIndex).markSelected(false); | ||||
|         } | ||||
|         this._list.page = pageNumber; | ||||
|     }, | ||||
|  | ||||
|     //// Protected methods //// | ||||
|  | ||||
|     /* | ||||
|      * Displays items that match the current request and should show up on the current page. | ||||
|      * Updates the display control to reflect the matched items set and the page selected. | ||||
|      * | ||||
|      * resetDisplayControl - indicates if the display control should be re-created because  | ||||
|      *                       the results or the space allocated for them changed. If it's false, | ||||
|      *                       the existing display control is used and only the page links are | ||||
|      *                       updated to reflect the current page selection. | ||||
|      */ | ||||
|     _displayMatchedItems: function(resetDisplayControl) { | ||||
|         // When generating a new list to display, we first remove all the old | ||||
|         // displayed items which will unset the selection. So we need  | ||||
|         // to keep a flag which indicates if this display had the selection. | ||||
|         let hadSelected = this.hasSelected(); | ||||
|  | ||||
|         this._removeAllDisplayItems(); | ||||
|         for (let i = 0; i < this._matchedItems.length; i++) { | ||||
|             this._addDisplayItem(this._matchedItems[i]); | ||||
|         } | ||||
|  | ||||
|         if (hadSelected) { | ||||
|             this._selectedIndex = -1; | ||||
|             this.selectFirstItem(); | ||||
|         } | ||||
|  | ||||
|         // Check if the pointer is over one of the items and display the information button if it is. | ||||
|         // We want to do this between finishing our changes to the display and the point where | ||||
|         // the display is redrawn. | ||||
|         Mainloop.idle_add(Lang.bind(this, | ||||
|                                     function() { | ||||
|                                         let [child, x, y, mask] = Gdk.Screen.get_default().get_root_window().get_pointer(); | ||||
|                                         let actor = global.stage.get_actor_at_pos(Clutter.PickMode.REACTIVE, | ||||
|                                                                                   x, y); | ||||
|                                         if (actor != null) { | ||||
|                                             let item = this._findDisplayedByActor(actor); | ||||
|                                             if (item != null) { | ||||
|                                                 item.onDrawnUnderPointer(); | ||||
|                                             } | ||||
|                                         } | ||||
|                                         return false; | ||||
|                                     }), | ||||
|                           Meta.PRIORITY_BEFORE_REDRAW); | ||||
|     }, | ||||
|  | ||||
|     // Creates a display item based on the information associated with itemId  | ||||
|     // and adds it to the displayed items. | ||||
|     _addDisplayItem : function(itemId) { | ||||
|         if (this._displayedItems.hasOwnProperty(itemId)) { | ||||
|             log("Tried adding a display item for " + itemId + ", but an item with this item id is already among displayed items."); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let itemInfo = this._allItems[itemId]; | ||||
|         let displayItem = this._createDisplayItem(itemInfo); | ||||
|  | ||||
|         displayItem.connect('activate', | ||||
|                             Lang.bind(this, | ||||
|                                       function() { | ||||
|                                           // update the selection | ||||
|                                           this._selectIndex(this._list.get_actor_index(displayItem.actor)); | ||||
|                                           this.activateSelected(); | ||||
|                                       })); | ||||
|  | ||||
|         displayItem.connect('show-details', | ||||
|                             Lang.bind(this, | ||||
|                                       function() { | ||||
|                                           let index = this._list.get_actor_index(displayItem.actor); | ||||
|                                           /* Close the details pane if already open */ | ||||
|                                           if (index == this._openDetailIndex) { | ||||
|                                               this._openDetailIndex = -1; | ||||
|                                               this.emit('show-details', -1); | ||||
|                                           } else { | ||||
|                                               this._openDetailIndex = index; | ||||
|                                               this.emit('show-details', index); | ||||
|                                           } | ||||
|                                       })); | ||||
|         this._list.add_actor(displayItem.actor); | ||||
|         this._displayedItems[itemId] = displayItem; | ||||
|     }, | ||||
|  | ||||
|     // Removes an item identifed by the itemId from the displayed items. | ||||
|     _removeDisplayItem: function(itemId) { | ||||
|         let count = this._list.displayedCount; | ||||
|         let displayItem = this._displayedItems[itemId]; | ||||
|         let displayItemIndex = this._list.get_actor_index(displayItem.actor); | ||||
|  | ||||
|         if (this.hasSelected() && count == 1) { | ||||
|             this.unsetSelected(); | ||||
|         } else if (this.hasSelected() && displayItemIndex < this._selectedIndex) { | ||||
|             this.selectUp(); | ||||
|         } | ||||
|  | ||||
|         displayItem.destroy(); | ||||
|  | ||||
|         delete this._displayedItems[itemId]; | ||||
|     }, | ||||
|  | ||||
|     // Removes all displayed items. | ||||
|     _removeAllDisplayItems: function() { | ||||
|         this.unsetSelected(); | ||||
|         for (itemId in this._displayedItems) | ||||
|             this._removeDisplayItem(itemId); | ||||
|      }, | ||||
|  | ||||
|     // Return true if there's an active search or other constraint | ||||
|     // on the list | ||||
|     _filterActive: function() { | ||||
|         return this._search != ""; | ||||
|     }, | ||||
|  | ||||
|     // Called when we are resetting state | ||||
|     _filterReset: function() { | ||||
|         this.unsetSelected(); | ||||
|     }, | ||||
|  | ||||
|     /* | ||||
|      * Updates the displayed items, applying the search string if one exists. | ||||
|      * | ||||
|      * resetPage - indicates if the page selection should be reset when displaying the matching results. | ||||
|      *             We reset the page selection when the change in results was initiated by the user by   | ||||
|      *             entering a different search criteria or by viewing the results list in a different | ||||
|      *             size mode, but we keep the page selection the same if the results got updated on | ||||
|      *             their own while the user was browsing through the result pages. | ||||
|      */ | ||||
|     _redisplay: function(resetPage) { | ||||
|         this._refreshCache(); | ||||
|         if (!this._filterActive()) | ||||
|             this._setDefaultList(); | ||||
|         else | ||||
|             this._doSearchFilter(); | ||||
|  | ||||
|         if (resetPage) | ||||
|             this._list.page = 0; | ||||
|  | ||||
|         this._displayMatchedItems(true); | ||||
|  | ||||
|         this.emit('redisplayed'); | ||||
|     }, | ||||
|  | ||||
|     //// Pure virtual protected methods //// | ||||
|   | ||||
|     // Performs the steps needed to have the latest information about the items. | ||||
|     _refreshCache: function() { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     // Sets the list of the displayed items based on the default sorting order. | ||||
|     // The default sorting order is specific to each implementing class. | ||||
|     _setDefaultList: function() { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
| 	// Compares items associated with the item ids based on the order in which the | ||||
| 	// items should be displayed. | ||||
| 	// Intended to be used as a compareFunction for array.sort(). | ||||
| 	// Returns an integer value indicating the result of the comparison. | ||||
|     _compareItems: function(itemIdA, itemIdB) { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     // Checks if the item info can be a match for the search string.  | ||||
|     // Returns a boolean flag indicating if that's the case. | ||||
|     _isInfoMatching: function(itemInfo, search) { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     // Creates a display item based on itemInfo. | ||||
|     _createDisplayItem: function(itemInfo) { | ||||
|         throw new Error("Not implemented"); | ||||
|     }, | ||||
|  | ||||
|     //// Private methods //// | ||||
|  | ||||
|     _getSearchMatchedItems: function() { | ||||
|         let matchedItemsForSearch = {}; | ||||
|         // Break the search up into terms, and search for each | ||||
|         // individual term.  Keep track of the number of terms | ||||
|         // each item matched. | ||||
|         let terms = this._search.split(/\s+/); | ||||
|         for (let i = 0; i < terms.length; i++) { | ||||
|             let term = terms[i]; | ||||
|             for (itemId in this._allItems) { | ||||
|                 let item = this._allItems[itemId]; | ||||
|                 if (this._isInfoMatching(item, term)) { | ||||
|                     let count = matchedItemsForSearch[itemId]; | ||||
|                     if (!count) | ||||
|                         count = 0; | ||||
|                     count += 1; | ||||
|                     matchedItemsForSearch[itemId] = count; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return matchedItemsForSearch; | ||||
|     }, | ||||
|  | ||||
|     // Applies the search string to the list of items to find matches, | ||||
|     // and displays the matching items. | ||||
|     _doSearchFilter: function() { | ||||
|         let matchedItemsForSearch; | ||||
|  | ||||
|         if (this._filterActive()) { | ||||
|             matchedItemsForSearch = this._getSearchMatchedItems(); | ||||
|         } else { | ||||
|             matchedItemsForSearch = {}; | ||||
|             for (let itemId in this._allItems) { | ||||
|                 matchedItemsForSearch[itemId] = 1; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this._matchedItems = []; | ||||
|         for (itemId in matchedItemsForSearch) { | ||||
|             this._matchedItems.push(itemId); | ||||
|         } | ||||
|         this._matchedItems.sort(Lang.bind(this, function (a, b) { | ||||
|             let countA = matchedItemsForSearch[a]; | ||||
|             let countB = matchedItemsForSearch[b]; | ||||
|             if (countA > countB) | ||||
|                 return -1; | ||||
|             else if (countA < countB) | ||||
|                 return 1; | ||||
|             else | ||||
|                 return this._compareItems(a, b); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     /* | ||||
|      * Updates the display control to reflect the matched items set and the page selected. | ||||
|      * | ||||
|      * resetDisplayControl - indicates if the display control should be re-created because  | ||||
|      *                       the results or the space allocated for them changed. If it's false, | ||||
|      *                       the existing display control is used and only the page links are | ||||
|      *                       updated to reflect the current page selection. | ||||
|      */ | ||||
|     _updateDisplayControl: function(resetDisplayControl) { | ||||
|         if (resetDisplayControl) { | ||||
|             this.displayControl.remove_all(); | ||||
|             let nPages = this._list.n_pages; | ||||
|             // Don't show the page indicator if there is only one page. | ||||
|             if (nPages == 1) | ||||
|                 return; | ||||
|             let pageNumber = this._list.page; | ||||
|             for (let i = 0; i < nPages; i++) { | ||||
|                 let pageControl = new Link.Link({ color: (i == pageNumber) ? DISPLAY_CONTROL_SELECTED_COLOR : ITEM_DISPLAY_DESCRIPTION_COLOR, | ||||
|                                                   font_name: "Sans Bold 16px", | ||||
|                                                   text: (i+1) + "", | ||||
|                                                   reactive: (i == pageNumber) ? false : true}); | ||||
|                 this.displayControl.append(pageControl.actor, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|                 // we use pageNumberLocalScope to get the page number right in the callback function | ||||
|                 let pageNumberLocalScope = i; | ||||
|                 pageControl.connect('clicked', | ||||
|                                     Lang.bind(this, | ||||
|                                               function(o, event) { | ||||
|                                                   this.displayPage(pageNumberLocalScope); | ||||
|                                               })); | ||||
|             } | ||||
|         } else { | ||||
|             let pageControlActors = this.displayControl.get_children(); | ||||
|             for (let i = 0; i < pageControlActors.length; i++) { | ||||
|                 let pageControlActor = pageControlActors[i]; | ||||
|                 if (i == this._list.page) { | ||||
|                     pageControlActor.color =  DISPLAY_CONTROL_SELECTED_COLOR; | ||||
|                     pageControlActor.reactive = false; | ||||
|                 } else { | ||||
|                     pageControlActor.color =  ITEM_DISPLAY_DESCRIPTION_COLOR; | ||||
|                     pageControlActor.reactive = true; | ||||
|                 } | ||||
|             }  | ||||
|         } | ||||
|         if (this.hasSelected()) { | ||||
|             this.selectFirstItem(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Returns a display item based on its index in the ordering of the | ||||
|     // display children. | ||||
|     _findDisplayedByIndex: function(index) { | ||||
|         let actor = this._list.get_displayed_actor(index); | ||||
|         return this._findDisplayedByActor(actor); | ||||
|     }, | ||||
|  | ||||
|     // Returns a display item based on the actor that represents it in  | ||||
|     // the display. | ||||
|     _findDisplayedByActor: function(actor) { | ||||
|         for (itemId in this._displayedItems) { | ||||
|             let item = this._displayedItems[itemId]; | ||||
|             if (item.actor == actor) { | ||||
|                 return item; | ||||
|             } | ||||
|         } | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     // Selects (e.g. highlights) a display item at the provided index, | ||||
|     // updates this.selectedItemDetails actor, and emits 'selected' signal. | ||||
|     _selectIndex: function(index) { | ||||
|         // Cleanup from the previous item | ||||
|         if (this.hasSelected()) { | ||||
|             this._findDisplayedByIndex(this._selectedIndex).markSelected(false); | ||||
|         } | ||||
|  | ||||
|         this._selectedIndex = index; | ||||
|         if (index < 0) | ||||
|             return | ||||
|  | ||||
|         // Mark the new item as selected and create its details pane | ||||
|         let item = this._findDisplayedByIndex(index); | ||||
|         item.markSelected(true); | ||||
|         this.emit('selected'); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| Signals.addSignalMethods(GenericDisplay.prototype); | ||||
| @@ -1,327 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| const ICON_SIZE = 48; | ||||
|  | ||||
|  | ||||
| function BaseIcon(label, createIcon) { | ||||
|     this._init(label, createIcon); | ||||
| } | ||||
|  | ||||
| BaseIcon.prototype = { | ||||
|     _init : function(label, params) { | ||||
|         params = Params.parse(params, { createIcon: null, | ||||
|                                         setSizeManually: false, | ||||
|                                         showLabel: true }); | ||||
|         this.actor = new St.Bin({ style_class: 'overview-icon', | ||||
|                                   x_fill: true, | ||||
|                                   y_fill: true }); | ||||
|         this.actor._delegate = this; | ||||
|         this.actor.connect('style-changed', | ||||
|                            Lang.bind(this, this._onStyleChanged)); | ||||
|  | ||||
|         this._spacing = 0; | ||||
|  | ||||
|         let box = new Shell.GenericContainer(); | ||||
|         box.connect('allocate', Lang.bind(this, this._allocate)); | ||||
|         box.connect('get-preferred-width', | ||||
|                     Lang.bind(this, this._getPreferredWidth)); | ||||
|         box.connect('get-preferred-height', | ||||
|                     Lang.bind(this, this._getPreferredHeight)); | ||||
|         this.actor.set_child(box); | ||||
|  | ||||
|         this.iconSize = ICON_SIZE; | ||||
|         this._iconBin = new St.Bin(); | ||||
|  | ||||
|         box.add_actor(this._iconBin); | ||||
|  | ||||
|         if (params.showLabel) { | ||||
|             this._name = new St.Label({ text: label }); | ||||
|             box.add_actor(this._name); | ||||
|         } else { | ||||
|             this._name = null; | ||||
|         } | ||||
|  | ||||
|         if (params.createIcon) | ||||
|             this.createIcon = params.createIcon; | ||||
|         this._setSizeManually = params.setSizeManually; | ||||
|  | ||||
|         this.icon = null; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|  | ||||
|         let iconSize = availHeight; | ||||
|  | ||||
|         let [iconMinHeight, iconNatHeight] = this._iconBin.get_preferred_height(-1); | ||||
|         let [iconMinWidth, iconNatWidth] = this._iconBin.get_preferred_width(-1); | ||||
|         let preferredHeight = iconNatHeight; | ||||
|  | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         if (this._name) { | ||||
|             let [labelMinHeight, labelNatHeight] = this._name.get_preferred_height(-1); | ||||
|             preferredHeight += this._spacing + labelNatHeight; | ||||
|  | ||||
|             let labelHeight = availHeight >= preferredHeight ? labelNatHeight | ||||
|                                                              : labelMinHeight; | ||||
|             iconSize -= this._spacing + labelHeight; | ||||
|  | ||||
|             childBox.x1 = 0; | ||||
|             childBox.x2 = availWidth; | ||||
|             childBox.y1 = iconSize + this._spacing; | ||||
|             childBox.y2 = childBox.y1 + labelHeight; | ||||
|             this._name.allocate(childBox, flags); | ||||
|         } | ||||
|  | ||||
|         childBox.x1 = Math.floor((availWidth - iconNatWidth) / 2); | ||||
|         childBox.y1 = Math.floor((iconSize - iconNatHeight) / 2); | ||||
|         childBox.x2 = childBox.x1 + iconNatWidth; | ||||
|         childBox.y2 = childBox.y1 + iconNatHeight; | ||||
|         this._iconBin.allocate(childBox, flags); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         this._getPreferredHeight(actor, -1, alloc); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let [iconMinHeight, iconNatHeight] = this._iconBin.get_preferred_height(forWidth); | ||||
|         alloc.min_size = iconMinHeight; | ||||
|         alloc.natural_size = iconNatHeight; | ||||
|  | ||||
|         if (this._name) { | ||||
|             let [labelMinHeight, labelNatHeight] = this._name.get_preferred_height(forWidth); | ||||
|             alloc.min_size += this._spacing + labelMinHeight; | ||||
|             alloc.natural_size += this._spacing + labelNatHeight; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // This can be overridden by a subclass, or by the createIcon | ||||
|     // parameter to _init() | ||||
|     createIcon: function(size) { | ||||
|         throw new Error('no implementation of createIcon in ' + this); | ||||
|     }, | ||||
|  | ||||
|     setIconSize: function(size) { | ||||
|         if (!this._setSizeManually) | ||||
|             throw new Error('setSizeManually has to be set to use setIconsize'); | ||||
|  | ||||
|         if (size == this.iconSize) | ||||
|             return; | ||||
|  | ||||
|         this._createIconTexture(size); | ||||
|     }, | ||||
|  | ||||
|     _createIconTexture: function(size) { | ||||
|         if (this.icon) | ||||
|             this.icon.destroy(); | ||||
|         this.iconSize = size; | ||||
|         this.icon = this.createIcon(this.iconSize); | ||||
|  | ||||
|         // The icon returned by createIcon() might actually be smaller than | ||||
|         // the requested icon size (for instance StTextureCache does this | ||||
|         // for fallback icons), so set the size explicitly. | ||||
|         this.icon.set_size(this.iconSize, this.iconSize); | ||||
|  | ||||
|         this._iconBin.child = this.icon; | ||||
|     }, | ||||
|  | ||||
|     _onStyleChanged: function() { | ||||
|         let node = this.actor.get_theme_node(); | ||||
|         this._spacing = node.get_length('spacing'); | ||||
|  | ||||
|         let size; | ||||
|         if (this._setSizeManually) { | ||||
|             size = this.iconSize; | ||||
|         } else { | ||||
|             let [found, len] = node.lookup_length('icon-size', false); | ||||
|             size = found ? len : ICON_SIZE; | ||||
|         } | ||||
|  | ||||
|         this._createIconTexture(size); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| function IconGrid(params) { | ||||
|     this._init(params); | ||||
| } | ||||
|  | ||||
| IconGrid.prototype = { | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { rowLimit: null, | ||||
|                                         columnLimit: null, | ||||
|                                         xAlign: St.Align.MIDDLE }); | ||||
|         this._rowLimit = params.rowLimit; | ||||
|         this._colLimit = params.columnLimit; | ||||
|         this._xAlign = params.xAlign; | ||||
|  | ||||
|         this.actor = new St.BoxLayout({ style_class: 'icon-grid', | ||||
|                                         vertical: true }); | ||||
|         // Pulled from CSS, but hardcode some defaults here | ||||
|         this._spacing = 0; | ||||
|         this._item_size = ICON_SIZE; | ||||
|         this._grid = new Shell.GenericContainer(); | ||||
|         this.actor.add(this._grid, { expand: true, y_align: St.Align.START }); | ||||
|         this.actor.connect('style-changed', Lang.bind(this, this._onStyleChanged)); | ||||
|  | ||||
|         this._grid.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth)); | ||||
|         this._grid.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight)); | ||||
|         this._grid.connect('allocate', Lang.bind(this, this._allocate)); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function (grid, forHeight, alloc) { | ||||
|         let children = this._grid.get_children(); | ||||
|         let nColumns = this._colLimit ? Math.min(this._colLimit, | ||||
|                                                  children.length) | ||||
|                                       : children.length; | ||||
|         let totalSpacing = Math.max(0, nColumns - 1) * this._spacing; | ||||
|         // Kind of a lie, but not really an issue right now.  If | ||||
|         // we wanted to support some sort of hidden/overflow that would | ||||
|         // need higher level design | ||||
|         alloc.min_size = this._item_size; | ||||
|         alloc.natural_size = nColumns * this._item_size + totalSpacing; | ||||
|     }, | ||||
|  | ||||
|     _getVisibleChildren: function() { | ||||
|         let children = this._grid.get_children(); | ||||
|         children = children.filter(function(actor) { | ||||
|             return actor.visible; | ||||
|         }); | ||||
|         return children; | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function (grid, forWidth, alloc) { | ||||
|         let children = this._getVisibleChildren(); | ||||
|         let [nColumns, usedWidth] = this._computeLayout(forWidth); | ||||
|         let nRows; | ||||
|         if (nColumns > 0) | ||||
|             nRows = Math.ceil(children.length / nColumns); | ||||
|         else | ||||
|             nRows = 0; | ||||
|         if (this._rowLimit) | ||||
|             nRows = Math.min(nRows, this._rowLimit); | ||||
|         let totalSpacing = Math.max(0, nRows - 1) * this._spacing; | ||||
|         let height = nRows * this._item_size + totalSpacing; | ||||
|         alloc.min_size = height; | ||||
|         alloc.natural_size = height; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function (grid, box, flags) { | ||||
|         let children = this._getVisibleChildren(); | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|  | ||||
|         let [nColumns, usedWidth] = this._computeLayout(availWidth); | ||||
|  | ||||
|         let leftPadding; | ||||
|         switch(this._xAlign) { | ||||
|             case St.Align.START: | ||||
|                 leftPadding = 0; | ||||
|                 break; | ||||
|             case St.Align.MIDDLE: | ||||
|                 leftPadding = Math.floor((availWidth - usedWidth) / 2); | ||||
|                 break; | ||||
|             case St.Align.END: | ||||
|                 leftPadding = availWidth - usedWidth; | ||||
|         } | ||||
|  | ||||
|         let x = box.x1 + leftPadding; | ||||
|         let y = box.y1; | ||||
|         let columnIndex = 0; | ||||
|         let rowIndex = 0; | ||||
|         for (let i = 0; i < children.length; i++) { | ||||
|             let [childMinWidth, childMinHeight, childNaturalWidth, childNaturalHeight] | ||||
|                 = children[i].get_preferred_size(); | ||||
|  | ||||
|             /* Center the item in its allocation horizontally */ | ||||
|             let width = Math.min(this._item_size, childNaturalWidth); | ||||
|             let childXSpacing = Math.max(0, width - childNaturalWidth) / 2; | ||||
|             let height = Math.min(this._item_size, childNaturalHeight); | ||||
|             let childYSpacing = Math.max(0, height - childNaturalHeight) / 2; | ||||
|  | ||||
|             let childBox = new Clutter.ActorBox(); | ||||
|             if (St.Widget.get_default_direction() == St.TextDirection.RTL) { | ||||
|                 let _x = box.x2 - (x + width); | ||||
|                 childBox.x1 = Math.floor(_x - childXSpacing); | ||||
|             } else { | ||||
|                 childBox.x1 = Math.floor(x + childXSpacing); | ||||
|             } | ||||
|             childBox.y1 = Math.floor(y + childYSpacing); | ||||
|             childBox.x2 = childBox.x1 + width; | ||||
|             childBox.y2 = childBox.y1 + height; | ||||
|  | ||||
|             if (this._rowLimit && rowIndex >= this._rowLimit) { | ||||
|                 this._grid.set_skip_paint(children[i], true); | ||||
|             } else { | ||||
|                 children[i].allocate(childBox, flags); | ||||
|                 this._grid.set_skip_paint(children[i], false); | ||||
|             } | ||||
|  | ||||
|             columnIndex++; | ||||
|             if (columnIndex == nColumns) { | ||||
|                 columnIndex = 0; | ||||
|                 rowIndex++; | ||||
|             } | ||||
|  | ||||
|             if (columnIndex == 0) { | ||||
|                 y += this._item_size + this._spacing; | ||||
|                 x = box.x1 + leftPadding; | ||||
|             } else { | ||||
|                 x += this._item_size + this._spacing; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     childrenInRow: function(rowWidth) { | ||||
|         return this._computeLayout(rowWidth)[0]; | ||||
|     }, | ||||
|  | ||||
|     _computeLayout: function (forWidth) { | ||||
|         let nColumns = 0; | ||||
|         let usedWidth = 0; | ||||
|         while ((this._colLimit == null || nColumns < this._colLimit) && | ||||
|                (usedWidth + this._item_size <= forWidth)) { | ||||
|             usedWidth += this._item_size + this._spacing; | ||||
|             nColumns += 1; | ||||
|         } | ||||
|  | ||||
|         if (nColumns > 0) | ||||
|             usedWidth -= this._spacing; | ||||
|  | ||||
|         return [nColumns, usedWidth]; | ||||
|     }, | ||||
|  | ||||
|     _onStyleChanged: function() { | ||||
|         let themeNode = this.actor.get_theme_node(); | ||||
|         this._spacing = themeNode.get_length('spacing'); | ||||
|         this._item_size = themeNode.get_length('-shell-grid-item-size'); | ||||
|         this._grid.queue_relayout(); | ||||
|     }, | ||||
|  | ||||
|     removeAll: function () { | ||||
|         this._grid.get_children().forEach(Lang.bind(this, function (child) { | ||||
|             child.destroy(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     addItem: function(actor) { | ||||
|         this._grid.add_actor(actor); | ||||
|     }, | ||||
|  | ||||
|     getItemAtIndex: function(index) { | ||||
|         return this._grid.get_children()[index]; | ||||
|     }, | ||||
|  | ||||
|     visibleItemsCount: function() { | ||||
|         return this._grid.get_children().length - this._grid.get_n_skip_paint(); | ||||
|     } | ||||
| }; | ||||
| @@ -1,199 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| /** | ||||
|  * Lightbox: | ||||
|  * @container: parent Clutter.Container | ||||
|  * @params: (optional) additional parameters: | ||||
|  *           - inhibitEvents: whether to inhibit events for @container | ||||
|  *           - width: shade actor width | ||||
|  *           - height: shade actor height | ||||
|  *           - fadeTime: seconds used to fade in/out | ||||
|  * | ||||
|  * Lightbox creates a dark translucent "shade" actor to hide the | ||||
|  * contents of @container, and allows you to specify particular actors | ||||
|  * in @container to highlight by bringing them above the shade. It | ||||
|  * tracks added and removed actors in @container while the lightboxing | ||||
|  * is active, and ensures that all actors are returned to their | ||||
|  * original stacking order when the lightboxing is removed. (However, | ||||
|  * if actors are restacked by outside code while the lightboxing is | ||||
|  * active, the lightbox may later revert them back to their original | ||||
|  * order.) | ||||
|  * | ||||
|  * By default, the shade window will have the height and width of | ||||
|  * @container and will track any changes in its size. You can override | ||||
|  * this by passing an explicit width and height in @params. | ||||
|  */ | ||||
| function Lightbox(container, params) { | ||||
|     this._init(container, params); | ||||
| } | ||||
|  | ||||
| Lightbox.prototype = { | ||||
|     _init : function(container, params) { | ||||
|         params = Params.parse(params, { inhibitEvents: false, | ||||
|                                         width: null, | ||||
|                                         height: null, | ||||
|                                         fadeTime: null | ||||
|                                       }); | ||||
|  | ||||
|         this._container = container; | ||||
|         this._children = container.get_children(); | ||||
|         this._fadeTime = params.fadeTime; | ||||
|         this.actor = new St.Bin({ x: 0, | ||||
|                                   y: 0, | ||||
|                                   style_class: 'lightbox', | ||||
|                                   reactive: params.inhibitEvents }); | ||||
|  | ||||
|         container.add_actor(this.actor); | ||||
|         this.actor.raise_top(); | ||||
|         this.actor.hide(); | ||||
|  | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         if (params.width && params.height) { | ||||
|             this.actor.width = params.width; | ||||
|             this.actor.height = params.height; | ||||
|             this._allocationChangedSignalId = 0; | ||||
|         } else { | ||||
|             this.actor.width = container.width; | ||||
|             this.actor.height = container.height; | ||||
|             this._allocationChangedSignalId = container.connect('allocation-changed', Lang.bind(this, this._allocationChanged)); | ||||
|         } | ||||
|  | ||||
|         this._actorAddedSignalId = container.connect('actor-added', Lang.bind(this, this._actorAdded)); | ||||
|         this._actorRemovedSignalId = container.connect('actor-removed', Lang.bind(this, this._actorRemoved)); | ||||
|  | ||||
|         this._highlighted = null; | ||||
|     }, | ||||
|  | ||||
|     _allocationChanged : function(container, box, flags) { | ||||
|         Meta.later_add(Meta.LaterType.BEFORE_REDRAW, Lang.bind(this, function() { | ||||
|             this.actor.width = this.width; | ||||
|             this.actor.height = this.height; | ||||
|             return false; | ||||
|         })); | ||||
|         this.width = this._container.width; | ||||
|         this.height = this._container.height; | ||||
|     }, | ||||
|  | ||||
|     _actorAdded : function(container, newChild) { | ||||
|         let children = this._container.get_children(); | ||||
|         let myIndex = children.indexOf(this.actor); | ||||
|         let newChildIndex = children.indexOf(newChild); | ||||
|  | ||||
|         if (newChildIndex > myIndex) { | ||||
|             // The child was added above the shade (presumably it was | ||||
|             // made the new top-most child). Move it below the shade, | ||||
|             // and add it to this._children as the new topmost actor. | ||||
|             newChild.lower(this.actor); | ||||
|             this._children.push(newChild); | ||||
|         } else if (newChildIndex == 0) { | ||||
|             // Bottom of stack | ||||
|             this._children.unshift(newChild); | ||||
|         } else { | ||||
|             // Somewhere else; insert it into the correct spot | ||||
|             let prevChild = this._children.indexOf(children[newChildIndex - 1]); | ||||
|             if (prevChild != -1) // paranoia | ||||
|                 this._children.splice(prevChild + 1, 0, newChild); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     show: function() { | ||||
|         if (this._fadeTime) { | ||||
|             this.actor.opacity = 0; | ||||
|             Tweener.addTween(this.actor, | ||||
|                              { opacity: 255, | ||||
|                                time: this._fadeTime, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
|         } else { | ||||
|             this.actor.opacity = 255; | ||||
|         } | ||||
|         this.actor.show(); | ||||
|     }, | ||||
|  | ||||
|     hide: function() { | ||||
|         if (this._fadeTime) { | ||||
|             Tweener.addTween(this.actor, | ||||
|                              { opacity: 0, | ||||
|                                time: this._fadeTime, | ||||
|                                transition: 'easeOutQuad', | ||||
|                                onComplete: Lang.bind(this, function() { | ||||
|                                    this.actor.hide(); | ||||
|                                }) | ||||
|                              }); | ||||
|         } else { | ||||
|             this.actor.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _actorRemoved : function(container, child) { | ||||
|         let index = this._children.indexOf(child); | ||||
|         if (index != -1) // paranoia | ||||
|             this._children.splice(index, 1); | ||||
|  | ||||
|         if (child == this._highlighted) | ||||
|             this._highlighted = null; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * highlight: | ||||
|      * @window: actor to highlight | ||||
|      * | ||||
|      * Highlights the indicated actor and unhighlights any other | ||||
|      * currently-highlighted actor. With no arguments or a false/null | ||||
|      * argument, all actors will be unhighlighted. | ||||
|      */ | ||||
|     highlight : function(window) { | ||||
|         if (this._highlighted == window) | ||||
|             return; | ||||
|  | ||||
|         // Walk this._children raising and lowering actors as needed. | ||||
|         // Things get a little tricky if the to-be-raised and | ||||
|         // to-be-lowered actors were originally adjacent, in which | ||||
|         // case we may need to indicate some *other* actor as the new | ||||
|         // sibling of the to-be-lowered one. | ||||
|  | ||||
|         let below = this.actor; | ||||
|         for (let i = this._children.length - 1; i >= 0; i--) { | ||||
|             if (this._children[i] == window) | ||||
|                 this._children[i].raise_top(); | ||||
|             else if (this._children[i] == this._highlighted) | ||||
|                 this._children[i].lower(below); | ||||
|             else | ||||
|                 below = this._children[i]; | ||||
|         } | ||||
|  | ||||
|         this._highlighted = window; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * destroy: | ||||
|      * | ||||
|      * Destroys the lightbox. | ||||
|      */ | ||||
|     destroy : function() { | ||||
|         this.actor.destroy(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * _onDestroy: | ||||
|      * | ||||
|      * This is called when the lightbox' actor is destroyed, either | ||||
|      * by destroying its container or by explicitly calling this.destroy(). | ||||
|      */ | ||||
|     _onDestroy: function() { | ||||
|         if (this._allocationChangedSignalId != 0) | ||||
|             this._container.disconnect(this._allocationChangedSignalId); | ||||
|         this._container.disconnect(this._actorAddedSignalId); | ||||
|         this._container.disconnect(this._actorRemovedSignalId); | ||||
|  | ||||
|         this.highlight(null); | ||||
|     } | ||||
| }; | ||||
| @@ -1,23 +1,80 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| // Link is a clickable link. Right now it just handles properly capturing | ||||
| // press and release events and short-circuiting the button handling in | ||||
| // ClutterText, but more features like different colors for hover/pressed states | ||||
| // or a different mouse cursor could be implemented. | ||||
| // | ||||
| // The properties passed in are forwarded to the Clutter.Text() constructor, | ||||
| // so can include, 'text', 'font_name', etc. | ||||
| function Link(props) { | ||||
|     this._init(props); | ||||
| } | ||||
|  | ||||
| Link.prototype = { | ||||
|     _init : function(props) { | ||||
|         let realProps = { reactive: true, | ||||
|                           track_hover: true, | ||||
|                           style_class: 'shell-link' }; | ||||
| 	let realProps = { reactive: true }; | ||||
|         // 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); | ||||
| 	Lang.copyProperties(props, realProps); | ||||
|  | ||||
|         this.actor = new St.Button(realProps); | ||||
| 	this.actor = new Clutter.Text(realProps); | ||||
| 	this.actor._delegate = this; | ||||
| 	this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress)); | ||||
| 	this.actor.connect('button-release-event', Lang.bind(this, this._onButtonRelease)); | ||||
| 	this.actor.connect('enter-event', Lang.bind(this, this._onEnter)); | ||||
| 	this.actor.connect('leave-event', Lang.bind(this, this._onLeave)); | ||||
|  | ||||
| 	this._buttonDown = false; | ||||
| 	this._havePointer = false; | ||||
|     }, | ||||
|  | ||||
|     // Update the text of the link | ||||
|     setText : function(text) { | ||||
| 	this.actor.text = text; | ||||
|     }, | ||||
|  | ||||
|     // We want to react on buttonDown, but if we override button-release-event for | ||||
|     // ClutterText, but not button-press-event, we get a stuck grab. Tracking | ||||
|     // buttonDown and doing the grab isn't really necessary, but doing it makes | ||||
|     // the behavior perfectly correct if the user clicks on one actor, drags | ||||
|     // to another and releases - that should not trigger either actor. | ||||
|     _onButtonPress : function(actor, event) { | ||||
| 	this._buttonDown = true; | ||||
| 	this._havePointer = true; // Hack to work around poor enter/leave tracking in Clutter | ||||
| 	Clutter.grab_pointer(actor); | ||||
|  | ||||
| 	return true; | ||||
|     }, | ||||
|  | ||||
|     _onButtonRelease : function(actor, event) { | ||||
| 	if (this._buttonDown) { | ||||
| 	    this._buttonDown = false; | ||||
| 	    Clutter.ungrab_pointer(actor); | ||||
|  | ||||
| 	    if (this._havePointer) | ||||
| 		this.emit('clicked'); | ||||
| 	} | ||||
|  | ||||
| 	return true; | ||||
|     }, | ||||
|  | ||||
|     _onEnter : function(actor, event) { | ||||
| 	if (event.get_source() == actor) | ||||
| 	    this._havePointer = true; | ||||
|  | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onLeave : function(actor, event) { | ||||
| 	if (event.get_source() == actor) | ||||
| 	    this._havePointer = false; | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										1429
									
								
								js/ui/magnifier.js
									
									
									
									
									
								
							
							
						
						| @@ -1,383 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const DBus = imports.dbus; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const MAG_SERVICE_NAME = 'org.gnome.Magnifier'; | ||||
| const MAG_SERVICE_PATH = '/org/gnome/Magnifier'; | ||||
| const ZOOM_SERVICE_NAME = 'org.gnome.Magnifier.ZoomRegion'; | ||||
| const ZOOM_SERVICE_PATH = '/org/gnome/Magnifier/ZoomRegion'; | ||||
|  | ||||
| // Subset of gnome-mag's Magnifier dbus interface -- to be expanded.  See: | ||||
| // http://git.gnome.org/browse/gnome-mag/tree/xml/...Magnifier.xml | ||||
| const MagnifierIface = { | ||||
|     name: MAG_SERVICE_NAME, | ||||
|     methods: [ | ||||
|                 { name: 'setActive', inSignature: 'b', outSignature: '' }, | ||||
|                 { name: 'isActive', inSignature: '', outSignature: 'b' }, | ||||
|                 { name: 'showCursor', inSignature: '', outSignature: '' }, | ||||
|                 { name: 'hideCursor', inSignature: '', outSignature: ''  }, | ||||
|                 { name: 'createZoomRegion', inSignature: 'ddaiai', outSignature: 'o' }, | ||||
|                 { name: 'addZoomRegion', inSignature: 'o', outSignature: 'b' }, | ||||
|                 { name: 'getZoomRegions', inSignature: '', outSignature: 'ao' }, | ||||
|                 { name: 'clearAllZoomRegions', inSignature: '', outSignature: '' }, | ||||
|                 { name: 'fullScreenCapable', inSignature: '', outSignature: 'b' }, | ||||
|  | ||||
|                 { name: 'setCrosswireSize', inSignature: 'i', outSignature: '' }, | ||||
|                 { name: 'getCrosswireSize', inSignature: '', outSignature: 'i' }, | ||||
|                 { name: 'setCrosswireLength', inSignature: 'i', outSignature: '' }, | ||||
|                 { name: 'getCrosswireLength', inSignature: '', outSignature: 'i' }, | ||||
|                 { name: 'setCrosswireClip', inSignature: 'b', outSignature: '' }, | ||||
|                 { name: 'getCrosswireClip', inSignature: '', outSignature: 'b' }, | ||||
|                 { name: 'setCrosswireColor', inSignature: 'u', outSignature: '' }, | ||||
|                 { name: 'getCrosswireColor', inSignature: '', outSignature: 'u' } | ||||
|              ], | ||||
|     signals: [], | ||||
|     properties: [] | ||||
| }; | ||||
|  | ||||
| // Subset of gnome-mag's ZoomRegion dbus interface -- to be expanded.  See: | ||||
| // http://git.gnome.org/browse/gnome-mag/tree/xml/...ZoomRegion.xml | ||||
| const ZoomRegionIface = { | ||||
|     name: ZOOM_SERVICE_NAME, | ||||
|     methods: [ | ||||
|                 { name: 'setMagFactor', inSignature: 'dd', outSignature: ''}, | ||||
|                 { name: 'getMagFactor', inSignature: '', outSignature: 'dd' }, | ||||
|                 { name: 'setRoi', inSignature: 'ai', outSignature: '' }, | ||||
|                 { name: 'getRoi', inSignature: '', outSignature: 'ai' }, | ||||
|                 { name: 'shiftContentsTo', inSignature: 'ii', outSignature: 'b' }, | ||||
|                 { name: 'moveResize', inSignature: 'ai', outSignature: '' } | ||||
|              ], | ||||
|     signals: [], | ||||
|     properties: [] | ||||
| }; | ||||
|  | ||||
| // For making unique ZoomRegion DBus proxy object paths of the form: | ||||
| // '/org/gnome/Magnifier/ZoomRegion/zoomer0', | ||||
| // '/org/gnome/Magnifier/ZoomRegion/zoomer1', etc. | ||||
| let _zoomRegionInstanceCount = 0; | ||||
|  | ||||
| function ShellMagnifier() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| ShellMagnifier.prototype = { | ||||
|     _init: function() { | ||||
|         this._zoomers = {}; | ||||
|         DBus.session.exportObject(MAG_SERVICE_PATH, this); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setActive: | ||||
|      * @activate:   Boolean to activate or de-activate the magnifier. | ||||
|      */ | ||||
|     setActive: function(activate) { | ||||
|         Main.magnifier.setActive(activate); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * isActive: | ||||
|      * @return  Whether the magnifier is active (boolean). | ||||
|      */ | ||||
|     isActive: function() { | ||||
|         return Main.magnifier.isActive(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * showCursor: | ||||
|      * Show the system mouse pointer. | ||||
|      */ | ||||
|     showCursor: function() { | ||||
|         Main.magnifier.showSystemCursor(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * hideCursor: | ||||
|      * Hide the system mouse pointer. | ||||
|      */ | ||||
|     hideCursor: function() { | ||||
|         Main.magnifier.hideSystemCursor(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * createZoomRegion: | ||||
|      * Create a new ZoomRegion and return its object path. | ||||
|      * @xMagFactor:     The power to set horizontal magnification of the | ||||
|      *                  ZoomRegion.  A value of 1.0 means no magnification.  A | ||||
|      *                  value of 2.0 doubles the size. | ||||
|      * @yMagFactor:     The power to set the vertical magnification of the | ||||
|      *                  ZoomRegion. | ||||
|      * @roi             Array of integers defining the region of the | ||||
|      *                  screen/desktop to magnify.  The array has the form | ||||
|      *                  [left, top, right, bottom]. | ||||
|      * @viewPort        Array of integers, [left, top, right, bottom] that defines | ||||
|      *                  the position of the ZoomRegion on screen. | ||||
|      * | ||||
|      * FIXME: The arguments here are redundant, since the width and height of | ||||
|      *   the ROI are determined by the viewport and magnification factors. | ||||
|      *   We ignore the passed in width and height. | ||||
|      * | ||||
|      * @return          The newly created ZoomRegion. | ||||
|      */ | ||||
|     createZoomRegion: function(xMagFactor, yMagFactor, roi, viewPort) { | ||||
|         let ROI = { x: roi[0], y: roi[1], width: roi[2] - roi[0], height: roi[3] - roi[1] }; | ||||
|         let viewBox = { x: viewPort[0], y: viewPort[1], width: viewPort[2] - viewPort[0], height: viewPort[3] - viewPort[1] }; | ||||
|         let realZoomRegion = Main.magnifier.createZoomRegion(xMagFactor, yMagFactor, ROI, viewBox); | ||||
|         let objectPath = ZOOM_SERVICE_PATH + '/zoomer' + _zoomRegionInstanceCount; | ||||
|         _zoomRegionInstanceCount++; | ||||
|  | ||||
|         let zoomRegionProxy = new ShellMagnifierZoomRegion(objectPath, realZoomRegion); | ||||
|         let proxyAndZoomRegion = {}; | ||||
|         proxyAndZoomRegion.proxy = zoomRegionProxy; | ||||
|         proxyAndZoomRegion.zoomRegion = realZoomRegion; | ||||
|         this._zoomers[objectPath] = proxyAndZoomRegion; | ||||
|         return objectPath; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * addZoomRegion: | ||||
|      * Append the given ZoomRegion to the magnifier's list of ZoomRegions. | ||||
|      * @zoomerObjectPath:   The object path for the zoom region proxy. | ||||
|      */ | ||||
|     addZoomRegion: function(zoomerObjectPath) { | ||||
|         let proxyAndZoomRegion = this._zoomers[zoomerObjectPath]; | ||||
|         if (proxyAndZoomRegion && proxyAndZoomRegion.zoomRegion) { | ||||
|             Main.magnifier.addZoomRegion(proxyAndZoomRegion.zoomRegion); | ||||
|             return true; | ||||
|         } | ||||
|         else | ||||
|             return false; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getZoomRegions: | ||||
|      * Return a list of ZoomRegion object paths for this Magnifier. | ||||
|      * @return:     The Magnifier's zoom region list as an array of DBus object | ||||
|      *              paths. | ||||
|      */ | ||||
|     getZoomRegions: function() { | ||||
|         // There may be more ZoomRegions in the magnifier itself than have | ||||
|         // been added through dbus.  Make sure all of them are associated with | ||||
|         // an object path and proxy. | ||||
|         let zoomRegions = Main.magnifier.getZoomRegions(); | ||||
|         let objectPaths = []; | ||||
|         let thoseZoomers = this._zoomers; | ||||
|         zoomRegions.forEach (function(aZoomRegion, index, array) { | ||||
|             let found = false; | ||||
|             for (let objectPath in thoseZoomers) { | ||||
|                 let proxyAndZoomRegion = thoseZoomers[objectPath]; | ||||
|                 if (proxyAndZoomRegion.zoomRegion === aZoomRegion) { | ||||
|                     objectPaths.push(objectPath); | ||||
|                     found = true; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|             if (!found) { | ||||
|                 // Got a ZoomRegion with no DBus proxy, make one. | ||||
|                 let newPath =  ZOOM_SERVICE_PATH + '/zoomer' + _zoomRegionInstanceCount; | ||||
|                 _zoomRegionInstanceCount++; | ||||
|                 let zoomRegionProxy = new ShellMagnifierZoomRegion(newPath, aZoomRegion); | ||||
|                 let proxyAndZoomer = {}; | ||||
|                 proxyAndZoomer.proxy = zoomRegionProxy; | ||||
|                 proxyAndZoomer.zoomRegion = aZoomRegion; | ||||
|                 thoseZoomers[newPath] = proxyAndZoomer; | ||||
|                 objectPaths.push(newPath); | ||||
|             } | ||||
|         }); | ||||
|         return objectPaths; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * clearAllZoomRegions: | ||||
|      * Remove all the zoom regions from this Magnfier's ZoomRegion list. | ||||
|      */ | ||||
|     clearAllZoomRegions: function() { | ||||
|         Main.magnifier.clearAllZoomRegions(); | ||||
|         for (let objectPath in this._zoomers) { | ||||
|             let proxyAndZoomer = this._zoomers[objectPath]; | ||||
|             proxyAndZoomer.proxy = null; | ||||
|             proxyAndZoomer.zoomRegion = null; | ||||
|             delete this._zoomers[objectPath]; | ||||
|             DBus.session.unexportObject(proxyAndZoomer); | ||||
|         } | ||||
|         this._zoomers = {}; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * fullScreenCapable: | ||||
|      * Consult if the Magnifier can magnify in full-screen mode. | ||||
|      * @return  Always return true. | ||||
|      */ | ||||
|     fullScreenCapable: function() { | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setCrosswireSize: | ||||
|      * Set the crosswire size of all ZoomRegions. | ||||
|      * @size:   The thickness of each line in the cross wire. | ||||
|      */ | ||||
|      setCrosswireSize: function(size) { | ||||
|         Main.magnifier.setCrosshairsThickness(size); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * getCrosswireSize: | ||||
|      * Get the crosswire size of all ZoomRegions. | ||||
|      * @return:   The thickness of each line in the cross wire. | ||||
|      */ | ||||
|      getCrosswireSize: function() { | ||||
|         return Main.magnifier.getCrosshairsThickness(); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * setCrosswireLength: | ||||
|      * Set the crosswire length of all zoom-regions.. | ||||
|      * @size:   The length of each line in the cross wire. | ||||
|      */ | ||||
|      setCrosswireLength: function(length) { | ||||
|         Main.magnifier.setCrosshairsLength(length); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * setCrosswireSize: | ||||
|      * Set the crosswire size of all zoom-regions. | ||||
|      * @size:   The thickness of each line in the cross wire. | ||||
|      */ | ||||
|      getCrosswireLength: function() { | ||||
|         return Main.magnifier.getCrosshairsLength(); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * setCrosswireClip: | ||||
|      * Set if the crosswire will be clipped by the cursor image.. | ||||
|      * @clip:   Flag to indicate whether to clip the crosswire. | ||||
|      */ | ||||
|      setCrosswireClip: function(clip) { | ||||
|         Main.magnifier.setCrosshairsClip(clip); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * getCrosswireClip: | ||||
|      * Get the crosswire clip value. | ||||
|      * @return:   Whether the crosswire is clipped by the cursor image. | ||||
|      */ | ||||
|      getCrosswireClip: function() { | ||||
|         return Main.magnifier.getCrosshairsClip(); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * setCrosswireColor: | ||||
|      * Set the crosswire color of all ZoomRegions. | ||||
|      * @color:   Unsigned int of the form rrggbbaa. | ||||
|      */ | ||||
|      setCrosswireColor: function(color) { | ||||
|         Main.magnifier.setCrosshairsColor('#%08x'.format(color)); | ||||
|      }, | ||||
|  | ||||
|     /** | ||||
|      * getCrosswireClip: | ||||
|      * Get the crosswire color of all ZoomRegions. | ||||
|      * @return:   The crosswire color as an unsigned int in the form rrggbbaa. | ||||
|      */ | ||||
|      getCrosswireColor: function() { | ||||
|         let colorString = Main.magnifier.getCrosshairsColor(); | ||||
|         // Drop the leading '#'. | ||||
|         return parseInt(colorString.slice(1), 16); | ||||
|      } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * ShellMagnifierZoomRegion: | ||||
|  * Object that implements the DBus ZoomRegion interface. | ||||
|  * @zoomerObjectPath:   String that is the path to a DBus ZoomRegion. | ||||
|  * @zoomRegion:         The actual zoom region associated with the object path. | ||||
|  */ | ||||
| function ShellMagnifierZoomRegion(zoomerObjectPath, zoomRegion) { | ||||
|     this._init(zoomerObjectPath, zoomRegion); | ||||
| } | ||||
|  | ||||
| ShellMagnifierZoomRegion.prototype = { | ||||
|     _init: function(zoomerObjectPath, zoomRegion) { | ||||
|         this._zoomRegion = zoomRegion; | ||||
|         DBus.session.proxifyObject(this, ZOOM_SERVICE_NAME, zoomerObjectPath); | ||||
|         DBus.session.exportObject(zoomerObjectPath, this); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setMagFactor: | ||||
|      * @xMagFactor:     The power to set the horizontal magnification factor to | ||||
|      *                  of the magnified view.  A value of 1.0 means no | ||||
|      *                  magnification.  A value of 2.0 doubles the size. | ||||
|      * @yMagFactor:     The power to set the vertical magnification factor to | ||||
|      *                  of the magnified view. | ||||
|      */ | ||||
|     setMagFactor: function(xMagFactor, yMagFactor) { | ||||
|         this._zoomRegion.setMagFactor(xMagFactor, yMagFactor); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getMagFactor: | ||||
|      * @return  an array, [xMagFactor, yMagFactor], containing the horizontal | ||||
|      *          and vertical magnification powers.  A value of 1.0 means no | ||||
|      *          magnification.  A value of 2.0 means the contents are doubled | ||||
|      *          in size, and so on. | ||||
|      */ | ||||
|     getMagFactor: function() { | ||||
|         return this._zoomRegion.getMagFactor(); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * setRoi: | ||||
|      * Sets the "region of interest" that the ZoomRegion is magnifying. | ||||
|      * @roi     Array, [left, top, right, bottom], defining the region of the | ||||
|      *          screen to magnify. The values are in screen (unmagnified) | ||||
|      *          coordinate space. | ||||
|      */ | ||||
|     setRoi: function(roi) { | ||||
|         let roiObject = { x: roi[0], y: roi[1], width: roi[2] - roi[0], height: roi[3] - roi[1] }; | ||||
|         this._zoomRegion.setROI(roiObject); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * getRoi: | ||||
|      * Retrieves the "region of interest" -- the rectangular bounds of that part | ||||
|      * of the desktop that the magnified view is showing (x, y, width, height). | ||||
|      * The bounds are given in non-magnified coordinates. | ||||
|      * @return  an array, [left, top, right, bottom], representing the bounding | ||||
|      *          rectangle of what is shown in the magnified view. | ||||
|      */ | ||||
|     getRoi: function() { | ||||
|         let roi = this._zoomRegion.getROI(); | ||||
|         roi[2] += roi[0]; | ||||
|         roi[3] += roi[1]; | ||||
|         return roi; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * Set the "region of interest" by centering the given screen coordinate | ||||
|      * within the zoom region. | ||||
|      * @x       The x-coord of the point to place at the center of the zoom region. | ||||
|      * @y       The y-coord. | ||||
|      * @return  Whether the shift was successful (for GS-mag, this is always | ||||
|      *          true). | ||||
|      */ | ||||
|     shiftContentsTo: function(x, y) { | ||||
|         this._zoomRegion.scrollContentsTo(x, y); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
|      * moveResize | ||||
|      * Sets the position and size of the ZoomRegion on screen. | ||||
|      * @viewPort    Array, [left, top, right, bottom], defining the position and | ||||
|      *              size on screen to place the zoom region. | ||||
|      */ | ||||
|     moveResize: function(viewPort) { | ||||
|         let viewRect = { x: viewPort[0], y: viewPort[1], width: viewPort[2] - viewPort[0], height: viewPort[3] - viewPort[1] }; | ||||
|         this._zoomRegion.setViewPort(viewRect); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| DBus.conformExport(ShellMagnifier.prototype, MagnifierIface); | ||||
| DBus.conformExport(ShellMagnifierZoomRegion.prototype, ZoomRegionIface); | ||||
							
								
								
									
										868
									
								
								js/ui/main.js
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										2178
									
								
								js/ui/messageTray.js
									
									
									
									
									
								
							
							
						
						| @@ -1,277 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 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 Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Pango = imports.gi.Pango; | ||||
| const St = imports.gi.St; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| const Lightbox = imports.ui.lightbox; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const OPEN_AND_CLOSE_TIME = 0.1; | ||||
| const FADE_OUT_DIALOG_TIME = 1.0; | ||||
|  | ||||
| const State = { | ||||
|     OPENED: 0, | ||||
|     CLOSED: 1, | ||||
|     OPENING: 2, | ||||
|     CLOSING: 3, | ||||
|     FADED_OUT: 4 | ||||
| }; | ||||
|  | ||||
| function ModalDialog() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| ModalDialog.prototype = { | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { styleClass: null }); | ||||
|  | ||||
|         this.state = State.CLOSED; | ||||
|         this._hasModal = false; | ||||
|  | ||||
|         this._group = new St.Group({ visible: false, | ||||
|                                      x: 0, | ||||
|                                      y: 0 }); | ||||
|         Main.uiGroup.add_actor(this._group); | ||||
|  | ||||
|         let constraint = new Clutter.BindConstraint({ source: global.stage, | ||||
|                                                       coordinate: Clutter.BindCoordinate.POSITION | Clutter.BindCoordinate.SIZE }); | ||||
|         this._group.add_constraint(constraint); | ||||
|  | ||||
|         global.focus_manager.add_group(this._group); | ||||
|         this._initialKeyFocus = this._group; | ||||
|         this._savedKeyFocus = null; | ||||
|  | ||||
|         this._group.connect('destroy', Lang.bind(this, this._onGroupDestroy)); | ||||
|  | ||||
|         this._actionKeys = {}; | ||||
|         this._group.connect('key-press-event', Lang.bind(this, this._onKeyPressEvent)); | ||||
|  | ||||
|         this._lightbox = new Lightbox.Lightbox(this._group, | ||||
|                                                { inhibitEvents: true }); | ||||
|  | ||||
|         this._backgroundBin = new St.Bin(); | ||||
|  | ||||
|         this._group.add_actor(this._backgroundBin); | ||||
|         this._lightbox.highlight(this._backgroundBin); | ||||
|  | ||||
|         this._backgroundStack = new Shell.Stack(); | ||||
|         this._backgroundBin.child = this._backgroundStack; | ||||
|  | ||||
|         this._eventBlocker = new Clutter.Group({ reactive: true }); | ||||
|         this._backgroundStack.add_actor(this._eventBlocker); | ||||
|  | ||||
|         this._dialogLayout = new St.BoxLayout({ style_class: 'modal-dialog', | ||||
|                                                 vertical:    true }); | ||||
|         if (params.styleClass != null) { | ||||
|             this._dialogLayout.add_style_class_name(params.styleClass); | ||||
|         } | ||||
|         this._backgroundStack.add_actor(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._buttonLayout = new St.BoxLayout({ style_class: 'modal-dialog-button-box', | ||||
|                                                 opacity:     220, | ||||
|                                                 vertical:    false }); | ||||
|         this._dialogLayout.add(this._buttonLayout, | ||||
|                                { expand:  true, | ||||
|                                  x_align: St.Align.MIDDLE, | ||||
|                                  y_align: St.Align.END }); | ||||
|     }, | ||||
|  | ||||
|     setButtons: function(buttons) { | ||||
|         this._buttonLayout.destroy_children(); | ||||
|         this._actionKeys = {}; | ||||
|  | ||||
|         let i = 0; | ||||
|         for (let index in buttons) { | ||||
|             let buttonInfo = buttons[index]; | ||||
|             let label = buttonInfo['label']; | ||||
|             let action = buttonInfo['action']; | ||||
|             let key = buttonInfo['key']; | ||||
|  | ||||
|             let button = new St.Button({ style_class: 'modal-dialog-button', | ||||
|                                          reactive:    true, | ||||
|                                          can_focus:   true, | ||||
|                                          label:       label }); | ||||
|  | ||||
|             let x_alignment; | ||||
|             if (buttons.length == 1) | ||||
|                 x_alignment = St.Align.END; | ||||
|             else if (i == 0) | ||||
|                 x_alignment = St.Align.START; | ||||
|             else if (i == buttons.length - 1) | ||||
|                 x_alignment = St.Align.END; | ||||
|             else | ||||
|                 x_alignment = St.Align.MIDDLE; | ||||
|  | ||||
|             this._initialKeyFocus = button; | ||||
|             this._buttonLayout.add(button, | ||||
|                                    { expand: true, | ||||
|                                      x_fill: false, | ||||
|                                      y_fill: false, | ||||
|                                      x_align: x_alignment, | ||||
|                                      y_align: St.Align.MIDDLE }); | ||||
|  | ||||
|             button.connect('clicked', action); | ||||
|  | ||||
|             if (key) | ||||
|                 this._actionKeys[key] = action; | ||||
|             i++; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onKeyPressEvent: function(object, keyPressEvent) { | ||||
|         let symbol = keyPressEvent.get_key_symbol(); | ||||
|         let action = this._actionKeys[symbol]; | ||||
|  | ||||
|         if (action) | ||||
|             action(); | ||||
|     }, | ||||
|  | ||||
|     _onGroupDestroy: function() { | ||||
|         this.emit('destroy'); | ||||
|     }, | ||||
|  | ||||
|     _fadeOpen: function() { | ||||
|         let monitor = global.get_focus_monitor(); | ||||
|  | ||||
|         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._lightbox.show(); | ||||
|         this._group.opacity = 0; | ||||
|         this._group.show(); | ||||
|         Tweener.addTween(this._group, | ||||
|                          { opacity: 255, | ||||
|                            time: OPEN_AND_CLOSE_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, | ||||
|                                function() { | ||||
|                                    this.state = State.OPENED; | ||||
|                                    this.emit('opened'); | ||||
|                                }) | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     setInitialKeyFocus: function(actor) { | ||||
|         this._initialKeyFocus = actor; | ||||
|     }, | ||||
|  | ||||
|     open: function(timestamp) { | ||||
|         if (this.state == State.OPENED || this.state == State.OPENING) | ||||
|             return true; | ||||
|  | ||||
|         if (!this.pushModal(timestamp)) | ||||
|             return false; | ||||
|  | ||||
|         this._fadeOpen(); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     close: function(timestamp) { | ||||
|         if (this.state == State.CLOSED || this.state == State.CLOSING) | ||||
|             return; | ||||
|  | ||||
|         this.state = State.CLOSING; | ||||
|         this.popModal(timestamp); | ||||
|  | ||||
|         Tweener.addTween(this._group, | ||||
|                          { opacity: 0, | ||||
|                            time: OPEN_AND_CLOSE_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, | ||||
|                                function() { | ||||
|                                    this.state = State.CLOSED; | ||||
|                                    this._group.hide(); | ||||
|                                }) | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     // Drop modal status without closing the dialog; this makes the | ||||
|     // dialog insensitive as well, so it needs to be followed shortly | ||||
|     // by either a close() or a pushModal() | ||||
|     popModal: function(timestamp) { | ||||
|         if (!this._hasModal) | ||||
|             return; | ||||
|  | ||||
|         let focus = global.stage.key_focus; | ||||
|         if (focus && this._group.contains(focus)) | ||||
|             this._savedKeyFocus = focus; | ||||
|         else | ||||
|             this._savedKeyFocus = null; | ||||
|         Main.popModal(this._group, timestamp); | ||||
|         global.gdk_screen.get_display().sync(); | ||||
|         this._hasModal = false; | ||||
|  | ||||
|         this._eventBlocker.raise_top(); | ||||
|     }, | ||||
|  | ||||
|     pushModal: function (timestamp) { | ||||
|         if (this._hasModal) | ||||
|             return true; | ||||
|         if (!Main.pushModal(this._group, timestamp)) | ||||
|             return false; | ||||
|  | ||||
|         this._hasModal = true; | ||||
|         if (this._savedKeyFocus) { | ||||
|             this._savedKeyFocus.grab_key_focus(); | ||||
|             this._savedKeyFocus = null; | ||||
|         } else | ||||
|             this._initialKeyFocus.grab_key_focus(); | ||||
|  | ||||
|         this._eventBlocker.lower_bottom(); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     // This method is like close, but fades the dialog out much slower, | ||||
|     // and leaves the lightbox in place. Once in the faded out state, | ||||
|     // the dialog can be brought back by an open call, or the lightbox | ||||
|     // can be dismissed by a close call. | ||||
|     // | ||||
|     // The main point of this method is to give some indication to the user | ||||
|     // that the dialog reponse has been acknowledged but will take a few | ||||
|     // moments before being processed. | ||||
|     // e.g., if a user clicked "Log Out" then the dialog should go away | ||||
|     // imediately, but the lightbox should remain until the logout is | ||||
|     // complete. | ||||
|     _fadeOutDialog: function(timestamp) { | ||||
|         if (this.state == State.CLOSED || this.state == State.CLOSING) | ||||
|             return; | ||||
|  | ||||
|         if (this.state == State.FADED_OUT) | ||||
|             return; | ||||
|  | ||||
|         this.popModal(timestamp); | ||||
|         Tweener.addTween(this._dialogLayout, | ||||
|                          { opacity: 0, | ||||
|                            time:    FADE_OUT_DIALOG_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, | ||||
|                                function() { | ||||
|                                    this.state = State.FADED_OUT; | ||||
|                                }) | ||||
|                          }); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(ModalDialog.prototype); | ||||
| @@ -1,558 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const DBus = imports.dbus; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Mainloop = imports.mainloop; | ||||
| const St = imports.gi.St; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
| const Main = imports.ui.main; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
| const Params = imports.misc.params; | ||||
| const Util = imports.misc.util; | ||||
|  | ||||
| let nextNotificationId = 1; | ||||
|  | ||||
| // Should really be defined in dbus.js | ||||
| const BusIface = { | ||||
|     name: 'org.freedesktop.DBus', | ||||
|     methods: [{ name: 'GetConnectionUnixProcessID', | ||||
|                 inSignature: 's', | ||||
|                 outSignature: 'i' }] | ||||
| }; | ||||
|  | ||||
| const Bus = function () { | ||||
|     this._init(); | ||||
| }; | ||||
|  | ||||
| Bus.prototype = { | ||||
|      _init: function() { | ||||
|          DBus.session.proxifyObject(this, 'org.freedesktop.DBus', '/org/freedesktop/DBus'); | ||||
|      } | ||||
| }; | ||||
|  | ||||
| DBus.proxifyPrototype(Bus.prototype, BusIface); | ||||
|  | ||||
| const NotificationDaemonIface = { | ||||
|     name: 'org.freedesktop.Notifications', | ||||
|     methods: [{ name: 'Notify', | ||||
|                 inSignature: 'susssasa{sv}i', | ||||
|                 outSignature: 'u' | ||||
|               }, | ||||
|               { name: 'CloseNotification', | ||||
|                 inSignature: 'u', | ||||
|                 outSignature: '' | ||||
|               }, | ||||
|               { name: 'GetCapabilities', | ||||
|                 inSignature: '', | ||||
|                 outSignature: 'as' | ||||
|               }, | ||||
|               { name: 'GetServerInformation', | ||||
|                 inSignature: '', | ||||
|                 outSignature: 'ssss' | ||||
|               }], | ||||
|     signals: [{ name: 'NotificationClosed', | ||||
|                 inSignature: 'uu' }, | ||||
|               { name: 'ActionInvoked', | ||||
|                 inSignature: 'us' }] | ||||
| }; | ||||
|  | ||||
| const NotificationClosedReason = { | ||||
|     EXPIRED: 1, | ||||
|     DISMISSED: 2, | ||||
|     APP_CLOSED: 3, | ||||
|     UNDEFINED: 4 | ||||
| }; | ||||
|  | ||||
| const Urgency = { | ||||
|     LOW: 0, | ||||
|     NORMAL: 1, | ||||
|     CRITICAL: 2 | ||||
| }; | ||||
|  | ||||
| const rewriteRules = { | ||||
|     'XChat': [ | ||||
|         { pattern:     /^XChat: Private message from: (\S*) \(.*\)$/, | ||||
|           replacement: '<$1>' }, | ||||
|         { pattern:     /^XChat: New public message from: (\S*) \((.*)\)$/, | ||||
|           replacement: '$2 <$1>' }, | ||||
|         { pattern:     /^XChat: Highlighted message from: (\S*) \((.*)\)$/, | ||||
|           replacement: '$2 <$1>' } | ||||
|     ] | ||||
| }; | ||||
|  | ||||
| function NotificationDaemon() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| NotificationDaemon.prototype = { | ||||
|     _init: function() { | ||||
|         DBus.session.exportObject('/org/freedesktop/Notifications', this); | ||||
|  | ||||
|         this._sources = {}; | ||||
|         this._senderToPid = {}; | ||||
|         this._notifications = {}; | ||||
|         this._busProxy = new Bus(); | ||||
|  | ||||
|         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)); | ||||
|     }, | ||||
|  | ||||
|     _iconForNotificationData: function(icon, hints, size) { | ||||
|         let textureCache = St.TextureCache.get_default(); | ||||
|  | ||||
|         if (icon) { | ||||
|             if (icon.substr(0, 7) == 'file://') | ||||
|                 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, data.length, hasAlpha, | ||||
|                                               width, height, rowStride, 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 }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Returns the source associated with ndata.notification if it is set. | ||||
|     // Otherwise, returns the source associated with the pid if one is | ||||
|     // stored in this._sources and the notification is not transient. | ||||
|     // Otherwise, creates a new source as long as pid is provided. | ||||
|     // | ||||
|     // Either a pid or ndata.notification is needed to retrieve or | ||||
|     // create a source. | ||||
|     _getSource: function(title, pid, ndata) { | ||||
|         if (!pid && !(ndata && ndata.notification)) | ||||
|             return null; | ||||
|  | ||||
|         // We use notification's source for the notifications we still have | ||||
|         // around that are getting replaced because we don't keep sources | ||||
|         // for transient notifications in this._sources, but we still want | ||||
|         // the notification associated with them to get replaced correctly. | ||||
|         if (ndata && ndata.notification) | ||||
|             return ndata.notification.source; | ||||
|  | ||||
|         let isForTransientNotification = (ndata && ndata.hints['transient'] == true); | ||||
|  | ||||
|         // We don't want to override a persistent notification | ||||
|         // with a transient one from the same sender, so we | ||||
|         // always create a new source object for new transient notifications | ||||
|         // and never add it to this._sources . | ||||
|         if (!isForTransientNotification && this._sources[pid]) | ||||
|             return this._sources[pid]; | ||||
|  | ||||
|         let source = new Source(title, pid); | ||||
|         source.setTransient(isForTransientNotification); | ||||
|  | ||||
|         if (!isForTransientNotification) { | ||||
|             this._sources[pid] = source; | ||||
|             source.connect('destroy', Lang.bind(this, | ||||
|                 function() { | ||||
|                     delete this._sources[pid]; | ||||
|                 })); | ||||
|         } | ||||
|  | ||||
|         Main.messageTray.add(source); | ||||
|         return source; | ||||
|     }, | ||||
|  | ||||
|     Notify: function(appName, replacesId, icon, summary, body, | ||||
|                      actions, hints, timeout) { | ||||
|         let id; | ||||
|  | ||||
|         // Filter out notifications from Empathy, since we | ||||
|         // handle that information from telepathyClient.js | ||||
|         if (appName == 'Empathy') { | ||||
|             // Ignore replacesId since we already sent back a | ||||
|             // NotificationClosed for that id. | ||||
|             id = nextNotificationId++; | ||||
|             Mainloop.idle_add(Lang.bind(this, | ||||
|                                         function () { | ||||
|                                             this._emitNotificationClosed(id, NotificationClosedReason.DISMISSED); | ||||
|                                         })); | ||||
|             return id; | ||||
|         } | ||||
|  | ||||
|         let rewrites = rewriteRules[appName]; | ||||
|         if (rewrites) { | ||||
|             for (let i = 0; i < rewrites.length; i++) { | ||||
|                 let rule = rewrites[i]; | ||||
|                 if (summary.search(rule.pattern) != -1) | ||||
|                     summary = summary.replace(rule.pattern, rule.replacement); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         hints = Params.parse(hints, { urgency: Urgency.NORMAL }, true); | ||||
|  | ||||
|         // Be compatible with the various hints for image data | ||||
|         // 'image-data' is the latest name of this hint, introduced in 1.2 | ||||
|         if (!hints['image-data']) { | ||||
|             if (hints['image_data']) | ||||
|                 hints['image-data'] = hints['image_data']; // version 1.1 of the spec | ||||
|             else if (hints['icon_data']) | ||||
|                 hints['image-data'] = hints['icon_data']; // previous versions of the spec | ||||
|         } | ||||
|  | ||||
|         let ndata = { appName: appName, | ||||
|                       icon: icon, | ||||
|                       summary: summary, | ||||
|                       body: body, | ||||
|                       actions: actions, | ||||
|                       hints: hints, | ||||
|                       timeout: timeout }; | ||||
|         if (replacesId != 0 && this._notifications[replacesId]) { | ||||
|             ndata.id = id = replacesId; | ||||
|             ndata.notification = this._notifications[replacesId].notification; | ||||
|         } else { | ||||
|             replacesId = 0; | ||||
|             ndata.id = id = nextNotificationId++; | ||||
|         } | ||||
|         this._notifications[id] = ndata; | ||||
|  | ||||
|         let sender = DBus.getCurrentMessageContext().sender; | ||||
|         let pid = this._senderToPid[sender]; | ||||
|  | ||||
|         let source = this._getSource(appName, pid, ndata); | ||||
|  | ||||
|         if (source) { | ||||
|             this._notifyForSource(source, ndata); | ||||
|             return id; | ||||
|         } | ||||
|  | ||||
|         if (replacesId) { | ||||
|             // There's already a pending call to GetConnectionUnixProcessID, | ||||
|             // which will see the new notification data when it finishes, | ||||
|             // so we don't have to do anything. | ||||
|             return id; | ||||
|         } | ||||
|  | ||||
|         this._busProxy.GetConnectionUnixProcessIDRemote(sender, Lang.bind(this, | ||||
|             function (pid, ex) { | ||||
|                 // The app may have updated or removed the notification | ||||
|                 ndata = this._notifications[id]; | ||||
|                 if (!ndata) | ||||
|                     return; | ||||
|  | ||||
|                 source = this._getSource(appName, pid, ndata); | ||||
|  | ||||
|                 // We only store sender-pid entries for persistent sources. | ||||
|                 // Removing the entries once the source is destroyed | ||||
|                 // would result in the entries associated with transient | ||||
|                 // sources removed once the notification is shown anyway. | ||||
|                 // However, keeping these pairs would mean that we would | ||||
|                 // possibly remove an entry associated with a persistent | ||||
|                 // source when a transient source for the same sender is | ||||
|                 // distroyed. | ||||
|                 if (!source.isTransient) { | ||||
|                     this._senderToPid[sender] = pid; | ||||
|                     source.connect('destroy', Lang.bind(this, | ||||
|                         function() { | ||||
|                             delete this._senderToPid[sender]; | ||||
|                         })); | ||||
|                 } | ||||
|                 this._notifyForSource(source, ndata); | ||||
|             })); | ||||
|  | ||||
|         return id; | ||||
|     }, | ||||
|  | ||||
|     _notifyForSource: function(source, ndata) { | ||||
|         let [id, icon, summary, body, actions, hints, notification] = | ||||
|             [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, summary, body, | ||||
|                                                         { icon: iconActor, | ||||
|                                                           bannerMarkup: true }); | ||||
|             ndata.notification = notification; | ||||
|             notification.connect('destroy', Lang.bind(this, | ||||
|                 function(n, reason) { | ||||
|                     delete this._notifications[id]; | ||||
|                     let notificationClosedReason; | ||||
|                     switch (reason) { | ||||
|                         case MessageTray.NotificationDestroyedReason.EXPIRED: | ||||
|                             notificationClosedReason = NotificationClosedReason.EXPIRED; | ||||
|                             break; | ||||
|                         case MessageTray.NotificationDestroyedReason.DISMISSED: | ||||
|                             notificationClosedReason = NotificationClosedReason.DISMISSED; | ||||
|                             break; | ||||
|                         case MessageTray.NotificationDestroyedReason.SOURCE_CLOSED: | ||||
|                             notificationClosedReason = NotificationClosedReason.APP_CLOSED; | ||||
|                             break; | ||||
|                     } | ||||
|                     this._emitNotificationClosed(id, notificationClosedReason); | ||||
|                 })); | ||||
|             notification.connect('action-invoked', Lang.bind(this, | ||||
|                 function(n, actionId) { | ||||
|                     this._emitActionInvoked(id, actionId); | ||||
|                 })); | ||||
|         } else { | ||||
|             notification.update(summary, body, { icon: iconActor, | ||||
|                                                  bannerMarkup: true, | ||||
|                                                  clear: true }); | ||||
|         } | ||||
|  | ||||
|         if (actions.length) { | ||||
|             notification.setUseActionIcons(hints['action-icons'] == true); | ||||
|             for (let i = 0; i < actions.length - 1; i += 2) | ||||
|                 notification.addButton(actions[i], actions[i + 1]); | ||||
|         } | ||||
|         switch (hints.urgency) { | ||||
|             case Urgency.LOW: | ||||
|                 notification.setUrgency(MessageTray.Urgency.LOW); | ||||
|                 break; | ||||
|             case Urgency.NORMAL: | ||||
|                 notification.setUrgency(MessageTray.Urgency.NORMAL); | ||||
|                 break; | ||||
|             case Urgency.CRITICAL: | ||||
|                 notification.setUrgency(MessageTray.Urgency.CRITICAL); | ||||
|                 break; | ||||
|         } | ||||
|         notification.setResident(hints.resident == true); | ||||
|         // 'transient' is a reserved keyword in JS, so we have to retrieve the value | ||||
|         // of the 'transient' hint with hints['transient'] rather than hints.transient | ||||
|         notification.setTransient(hints['transient'] == true); | ||||
|  | ||||
|         let sourceIconActor = source.useNotificationIcon ? this._iconForNotificationData(icon, hints, source.ICON_SIZE) : null; | ||||
|         source.processNotification(notification, sourceIconActor); | ||||
|     }, | ||||
|  | ||||
|     CloseNotification: function(id) { | ||||
|         let ndata = this._notifications[id]; | ||||
|         if (ndata) { | ||||
|             if (ndata.notification) | ||||
|                 ndata.notification.destroy(MessageTray.NotificationDestroyedReason.SOURCE_CLOSED); | ||||
|             delete this._notifications[id]; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     GetCapabilities: function() { | ||||
|         return [ | ||||
|             'actions', | ||||
|             'action-icons', | ||||
|             'body', | ||||
|             // 'body-hyperlinks', | ||||
|             // 'body-images', | ||||
|             'body-markup', | ||||
|             // 'icon-multi', | ||||
|             'icon-static', | ||||
|             'persistence', | ||||
|             // 'sound', | ||||
|         ]; | ||||
|     }, | ||||
|  | ||||
|     GetServerInformation: function() { | ||||
|         return [ | ||||
|             Config.PACKAGE_NAME, | ||||
|             'GNOME', | ||||
|             Config.PACKAGE_VERSION, | ||||
|             '1.2' | ||||
|         ]; | ||||
|     }, | ||||
|  | ||||
|     _onFocusAppChanged: function() { | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         if (!tracker.focus_app) | ||||
|             return; | ||||
|  | ||||
|         for (let id in this._sources) { | ||||
|             let source = this._sources[id]; | ||||
|             if (source.app == tracker.focus_app) { | ||||
|                 source.destroyNonResidentNotifications(); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _emitNotificationClosed: function(id, reason) { | ||||
|         DBus.session.emit_signal('/org/freedesktop/Notifications', | ||||
|                                  'org.freedesktop.Notifications', | ||||
|                                  'NotificationClosed', 'uu', | ||||
|                                  [id, reason]); | ||||
|     }, | ||||
|  | ||||
|     _emitActionInvoked: function(id, action) { | ||||
|         DBus.session.emit_signal('/org/freedesktop/Notifications', | ||||
|                                  'org.freedesktop.Notifications', | ||||
|                                  'ActionInvoked', 'us', | ||||
|                                  [id, action]); | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconAdded: function(o, icon) { | ||||
|         let source = this._getSource(icon.title || icon.wm_class || _("Unknown"), icon.pid, null); | ||||
|         source.setTrayIcon(icon); | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconRemoved: function(o, icon) { | ||||
|         let source = this._sources[icon.pid]; | ||||
|         if (source) | ||||
|             source.destroy(); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| DBus.conformExport(NotificationDaemon.prototype, NotificationDaemonIface); | ||||
|  | ||||
| function Source(title, pid) { | ||||
|     this._init(title, pid); | ||||
| } | ||||
|  | ||||
| Source.prototype = { | ||||
|     __proto__:  MessageTray.Source.prototype, | ||||
|  | ||||
|     _init: function(title, pid) { | ||||
|         MessageTray.Source.prototype._init.call(this, title); | ||||
|  | ||||
|         this._pid = pid; | ||||
|         this._appStateChangedId = 0; | ||||
|         this._setApp(); | ||||
|         if (this.app) | ||||
|             this.title = this.app.get_name(); | ||||
|         else | ||||
|             this.useNotificationIcon = true; | ||||
|         this._trayIcon = null; | ||||
|     }, | ||||
|  | ||||
|     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) | ||||
|             this.pushNotification(notification); | ||||
|         else | ||||
|             this.notify(notification); | ||||
|     }, | ||||
|  | ||||
|     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 (event.get_button() == 1 && | ||||
|             this.notifications.length > 0) | ||||
|             return false; | ||||
|  | ||||
|         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); | ||||
|         } | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _setApp: function() { | ||||
|         if (this.app) | ||||
|             return; | ||||
|  | ||||
|         this.app = Shell.WindowTracker.get_default().get_app_from_pid(this._pid); | ||||
|         if (!this.app) | ||||
|             return; | ||||
|  | ||||
|         // We only update the app if this.app is null, so we can't disconnect the old this._appStateChangedId | ||||
|         // even if it were non-zero for some reason. | ||||
|         this._appStateChangedId = this.app.connect('notify::state', Lang.bind(this,  this._appStateChanged)); | ||||
|  | ||||
|         // Only override the icon if we were previously using | ||||
|         // notification-based icons (ie, not a trayicon) or if it was unset before | ||||
|         if (!this._trayIcon) { | ||||
|             this.useNotificationIcon = false; | ||||
|             this._setSummaryIcon(this.app.create_icon_texture (this.ICON_SIZE)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     setTrayIcon: function(icon) { | ||||
|         this._setSummaryIcon(icon); | ||||
|         this.useNotificationIcon = false; | ||||
|         this._trayIcon = icon; | ||||
|     }, | ||||
|  | ||||
|     open: function(notification) { | ||||
|         this.destroyNonResidentNotifications(); | ||||
|         this.openApp(); | ||||
|     }, | ||||
|  | ||||
|     _lastNotificationRemoved: function() { | ||||
|         if (!this._trayIcon) | ||||
|             this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     _appStateChanged: function() { | ||||
|         // Destroy notification sources when their apps exit. | ||||
|         // The app exiting would normally result in a tray icon being removed, | ||||
|         // so the associated source would be destroyed through the code path | ||||
|         // that handles the tray icon being removed. We should not destroy | ||||
|         // the source associated with a tray icon when the application state | ||||
|         // is Shell.AppState.STOPPED because running applications that have | ||||
|         // no open windows would also have that state. This is often the case | ||||
|         // for applications that use tray icons. | ||||
|         if (!this._trayIcon && this.app.get_state() == Shell.AppState.STOPPED) | ||||
|             this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     openApp: function() { | ||||
|         if (this.app == null) | ||||
|             return; | ||||
|  | ||||
|         let windows = this.app.get_windows(); | ||||
|         if (windows.length > 0) { | ||||
|             let mostRecentWindow = windows[0]; | ||||
|             Main.activateWindow(mostRecentWindow); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         if (this.app && this._appStateChangedId) { | ||||
|             this.app.disconnect(this._appStateChangedId); | ||||
|             this._appStateChangedId = 0; | ||||
|         } | ||||
|         MessageTray.Source.prototype.destroy.call(this); | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										1311
									
								
								js/ui/panel.js
									
									
									
									
									
								
							
							
						
						| @@ -1,117 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| function Button(menuAlignment) { | ||||
|     this._init(menuAlignment); | ||||
| } | ||||
|  | ||||
| Button.prototype = { | ||||
|     _init: function(menuAlignment) { | ||||
|         this.actor = new St.Bin({ style_class: 'panel-button', | ||||
|                                   reactive: true, | ||||
|                                   can_focus: true, | ||||
|                                   x_fill: true, | ||||
|                                   y_fill: false, | ||||
|                                   track_hover: true }); | ||||
|         this.actor._delegate = this; | ||||
|         this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress)); | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this._onSourceKeyPress)); | ||||
|         this.menu = new PopupMenu.PopupMenu(this.actor, menuAlignment, St.Side.TOP, 0); | ||||
|         this.menu.connect('open-state-changed', Lang.bind(this, this._onOpenStateChanged)); | ||||
|         this.menu.actor.connect('key-press-event', Lang.bind(this, this._onMenuKeyPress)); | ||||
|         Main.chrome.addActor(this.menu.actor, { visibleInOverview: true, | ||||
|                                                 affectsStruts: false }); | ||||
|         this.menu.actor.hide(); | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress: function(actor, event) { | ||||
|         this.menu.toggle(); | ||||
|     }, | ||||
|  | ||||
|     _onSourceKeyPress: function(actor, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_space || symbol == Clutter.KEY_Return) { | ||||
|             this.menu.toggle(); | ||||
|             return true; | ||||
|         } else if (symbol == Clutter.KEY_Escape && this.menu.isOpen) { | ||||
|             this.menu.close(); | ||||
|             return true; | ||||
|         } else if (symbol == Clutter.KEY_Down) { | ||||
|             if (!this.menu.isOpen) | ||||
|                 this.menu.toggle(); | ||||
|             this.menu.actor.navigate_focus(this.actor, Gtk.DirectionType.DOWN, false); | ||||
|             return true; | ||||
|         } else | ||||
|             return false; | ||||
|     }, | ||||
|  | ||||
|     _onMenuKeyPress: function(actor, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_Left || symbol == Clutter.KEY_Right) { | ||||
|             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); | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onOpenStateChanged: function(menu, open) { | ||||
|         if (open) | ||||
|             this.actor.add_style_pseudo_class('active'); | ||||
|         else | ||||
|             this.actor.remove_style_pseudo_class('active'); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /* SystemStatusButton: | ||||
|  * | ||||
|  * This class manages one System Status indicator (network, keyboard, | ||||
|  * volume, bluetooth...), which is just a PanelMenuButton with an | ||||
|  * icon and a tooltip | ||||
|  */ | ||||
| function SystemStatusButton() { | ||||
|     this._init.apply(this, arguments); | ||||
| } | ||||
|  | ||||
| SystemStatusButton.prototype = { | ||||
|     __proto__: Button.prototype, | ||||
|  | ||||
|     _init: function(iconName,tooltipText) { | ||||
|         Button.prototype._init.call(this, 0.0); | ||||
|         this._iconActor = new St.Icon({ icon_name: iconName, | ||||
|                                         icon_type: St.IconType.SYMBOLIC, | ||||
|                                         style_class: 'system-status-icon' }); | ||||
|         this.actor.set_child(this._iconActor); | ||||
|         this.setTooltip(tooltipText); | ||||
|     }, | ||||
|  | ||||
|     setIcon: function(iconName) { | ||||
|         this._iconActor.icon_name = iconName; | ||||
|     }, | ||||
|  | ||||
|     setGIcon: function(gicon) { | ||||
|         this._iconActor.gicon = gicon; | ||||
|     }, | ||||
|  | ||||
|     setTooltip: function(text) { | ||||
|         if (text != null) { | ||||
|             this.tooltip = text; | ||||
|             this.actor.has_tooltip = true; | ||||
|             this.actor.tooltip_text = text; | ||||
|         } else { | ||||
|             this.actor.has_tooltip = false; | ||||
|             this.tooltip = null; | ||||
|         } | ||||
|     } | ||||
| }; | ||||
| @@ -1,442 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const DND = imports.ui.dnd; | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const Search = imports.ui.search; | ||||
| const Util = imports.misc.util; | ||||
|  | ||||
| /** | ||||
|  * Represents a place object, which is most normally a bookmark entry, | ||||
|  * a mount/volume, or a special place like the Home Folder, Computer, and Network. | ||||
|  * | ||||
|  * @name: String title | ||||
|  * @iconFactory: A JavaScript callback which will create an icon texture given a size parameter | ||||
|  * @launch: A JavaScript callback to launch the entry | ||||
|  */ | ||||
| function PlaceInfo(id, name, iconFactory, launch) { | ||||
|     this._init(id, name, iconFactory, launch); | ||||
| } | ||||
|  | ||||
| PlaceInfo.prototype = { | ||||
|     _init: function(id, name, iconFactory, launch) { | ||||
|         this.id = id; | ||||
|         this.name = name; | ||||
|         this._lowerName = name.toLowerCase(); | ||||
|         this.iconFactory = iconFactory; | ||||
|         this.launch = launch; | ||||
|     }, | ||||
|  | ||||
|     matchTerms: function(terms) { | ||||
|         let mtype = Search.MatchType.NONE; | ||||
|         for (let i = 0; i < terms.length; i++) { | ||||
|             let term = terms[i]; | ||||
|             let idx = this._lowerName.indexOf(term); | ||||
|             if (idx == 0) { | ||||
|                 mtype = Search.MatchType.PREFIX; | ||||
|             } else if (idx > 0) { | ||||
|                 if (mtype == Search.MatchType.NONE) | ||||
|                     mtype = Search.MatchType.SUBSTRING; | ||||
|             } else { | ||||
|                 return Search.MatchType.NONE; | ||||
|             } | ||||
|         } | ||||
|         return mtype; | ||||
|     }, | ||||
|  | ||||
|     isRemovable: function() { | ||||
|         return false; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| // Helper function to translate launch parameters into a GAppLaunchContext | ||||
| function _makeLaunchContext(params) | ||||
| { | ||||
|     params = Params.parse(params, { workspace: null, | ||||
|                                     timestamp: null }); | ||||
|  | ||||
|     let launchContext = global.create_app_launch_context(); | ||||
|     if (params.workspace != null) | ||||
|         launchContext.set_desktop(params.workspace.index()); | ||||
|     if (params.timestamp != null) | ||||
|         launchContext.set_timestamp(params.timestamp); | ||||
|  | ||||
|     return launchContext; | ||||
| } | ||||
|  | ||||
| function PlaceDeviceInfo(mount) { | ||||
|     this._init(mount); | ||||
| } | ||||
|  | ||||
| PlaceDeviceInfo.prototype = { | ||||
|     __proto__: PlaceInfo.prototype, | ||||
|  | ||||
|     _init: function(mount) { | ||||
|         this._mount = mount; | ||||
|         this.name = mount.get_name(); | ||||
|         this._lowerName = this.name.toLowerCase(); | ||||
|         this.id = 'mount:' + mount.get_root().get_uri(); | ||||
|     }, | ||||
|  | ||||
|     iconFactory: function(size) { | ||||
|         let icon = this._mount.get_icon(); | ||||
|         return St.TextureCache.get_default().load_gicon(null, icon, size); | ||||
|     }, | ||||
|  | ||||
|     launch: function(params) { | ||||
|         Gio.app_info_launch_default_for_uri(this._mount.get_root().get_uri(), | ||||
|                                             _makeLaunchContext(params)); | ||||
|     }, | ||||
|  | ||||
|     isRemovable: function() { | ||||
|         return this._mount.can_unmount(); | ||||
|     }, | ||||
|  | ||||
|     remove: function() { | ||||
|         if (!this.isRemovable()) | ||||
|             return; | ||||
|  | ||||
|         if (this._mount.can_eject()) | ||||
|             this._mount.eject(0, null, Lang.bind(this, this._removeFinish)); | ||||
|         else | ||||
|             this._mount.unmount(0, null, Lang.bind(this, this._removeFinish)); | ||||
|     }, | ||||
|  | ||||
|     _removeFinish: function(o, res, data) { | ||||
|         try { | ||||
|             if (this._mount.can_eject()) | ||||
|                 this._mount.eject_finish(res); | ||||
|             else | ||||
|                 this._mount.unmount_finish(res); | ||||
|         } catch (e) { | ||||
|             let message = _("Failed to unmount '%s'").format(o.get_name()); | ||||
|             Main.overview.shellInfo.setMessage(message, | ||||
|                                              Lang.bind(this, this.remove), | ||||
|                                              _("Retry")); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| function PlacesManager() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| PlacesManager.prototype = { | ||||
|     _init: function() { | ||||
|         this._defaultPlaces = []; | ||||
|         this._mounts = []; | ||||
|         this._bookmarks = []; | ||||
|  | ||||
|         let homeFile = Gio.file_new_for_path (GLib.get_home_dir()); | ||||
|         let homeUri = homeFile.get_uri(); | ||||
|         let homeLabel = Shell.util_get_label_for_uri (homeUri); | ||||
|         let homeIcon = Shell.util_get_icon_for_uri (homeUri); | ||||
|         this._home = new PlaceInfo('special:home', homeLabel, | ||||
|             function(size) { | ||||
|                 return St.TextureCache.get_default().load_gicon(null, homeIcon, size); | ||||
|             }, | ||||
|             function(params) { | ||||
|                 Gio.app_info_launch_default_for_uri(homeUri, _makeLaunchContext(params)); | ||||
|             }); | ||||
|  | ||||
|         let desktopPath = GLib.get_user_special_dir(GLib.UserDirectory.DIRECTORY_DESKTOP); | ||||
|         let desktopFile = Gio.file_new_for_path (desktopPath); | ||||
|         let desktopUri = desktopFile.get_uri(); | ||||
|         let desktopLabel = Shell.util_get_label_for_uri (desktopUri); | ||||
|         let desktopIcon = Shell.util_get_icon_for_uri (desktopUri); | ||||
|         this._desktopMenu = new PlaceInfo('special:desktop', desktopLabel, | ||||
|             function(size) { | ||||
|                 return St.TextureCache.get_default().load_gicon(null, desktopIcon, size); | ||||
|             }, | ||||
|             function(params) { | ||||
|                 Gio.app_info_launch_default_for_uri(desktopUri, _makeLaunchContext(params)); | ||||
|             }); | ||||
|  | ||||
|         this._connect = new PlaceInfo('special:connect', _("Connect to..."), | ||||
|             function (size) { | ||||
|                 return new St.Icon({ icon_name: 'applications-internet', | ||||
|                                      icon_type: St.IconType.FULLCOLOR, | ||||
|                                      icon_size: size }); | ||||
|             }, | ||||
|             function (params) { | ||||
|                 // BUG: nautilus-connect-server doesn't have a desktop file, so we can't | ||||
|                 // launch it with the workspace from params. It's probably pretty rare | ||||
|                 // and odd to drag this place onto a workspace in any case | ||||
|  | ||||
|                 Util.spawn(['nautilus-connect-server']); | ||||
|             }); | ||||
|  | ||||
|         this._defaultPlaces.push(this._home); | ||||
|         this._defaultPlaces.push(this._desktopMenu); | ||||
|         this._defaultPlaces.push(this._connect); | ||||
|  | ||||
|         /* | ||||
|         * Show devices, code more or less ported from nautilus-places-sidebar.c | ||||
|         */ | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
|         this._volumeMonitor.connect('volume-added', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('volume-removed',Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('volume-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-added', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-removed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-connected', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-disconnected', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._updateDevices(); | ||||
|  | ||||
|         this._bookmarksPath = GLib.build_filenamev([GLib.get_home_dir(), '.gtk-bookmarks']); | ||||
|         this._bookmarksFile = Gio.file_new_for_path(this._bookmarksPath); | ||||
|         let monitor = this._bookmarksFile.monitor_file(Gio.FileMonitorFlags.NONE, null); | ||||
|         this._bookmarkTimeoutId = 0; | ||||
|         monitor.connect('changed', Lang.bind(this, function () { | ||||
|             if (this._bookmarkTimeoutId > 0) | ||||
|                 return; | ||||
|             /* Defensive event compression */ | ||||
|             this._bookmarkTimeoutId = Mainloop.timeout_add(100, Lang.bind(this, function () { | ||||
|                 this._bookmarkTimeoutId = 0; | ||||
|                 this._reloadBookmarks(); | ||||
|                 return false; | ||||
|             })); | ||||
|         })); | ||||
|  | ||||
|         this._reloadBookmarks(); | ||||
|     }, | ||||
|  | ||||
|     _updateDevices: function() { | ||||
|         this._mounts = []; | ||||
|  | ||||
|         /* first go through all connected drives */ | ||||
|         let drives = this._volumeMonitor.get_connected_drives(); | ||||
|         for (let i = 0; i < drives.length; i++) { | ||||
|             let volumes = drives[i].get_volumes(); | ||||
|             for(let j = 0; j < volumes.length; j++) { | ||||
|                 let mount = volumes[j].get_mount(); | ||||
|                 if(mount != null) { | ||||
|                     this._addMount(mount); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* add all volumes that is not associated with a drive */ | ||||
|         let volumes = this._volumeMonitor.get_volumes(); | ||||
|         for(let i = 0; i < volumes.length; i++) { | ||||
|             if(volumes[i].get_drive() != null) | ||||
|                 continue; | ||||
|  | ||||
|             let mount = volumes[i].get_mount(); | ||||
|             if(mount != null) { | ||||
|                 this._addMount(mount); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* add mounts that have no volume (/etc/mtab mounts, ftp, sftp,...) */ | ||||
|         let mounts = this._volumeMonitor.get_mounts(); | ||||
|         for(let i = 0; i < mounts.length; i++) { | ||||
|             if(mounts[i].is_shadowed()) | ||||
|                 continue; | ||||
|  | ||||
|             if(mounts[i].get_volume()) | ||||
|                 continue; | ||||
|  | ||||
|             this._addMount(mounts[i]); | ||||
|         } | ||||
|  | ||||
|         /* We emit two signals, one for a generic 'all places' update | ||||
|          * and the other for one specific to mounts. We do this because | ||||
|          * clients like PlaceDisplay may only care about places in general | ||||
|          * being updated while clients like DashPlaceDisplay care which | ||||
|          * specific type of place got updated. | ||||
|          */ | ||||
|         this.emit('mounts-updated'); | ||||
|         this.emit('places-updated'); | ||||
|  | ||||
|     }, | ||||
|  | ||||
|     _reloadBookmarks: function() { | ||||
|  | ||||
|         this._bookmarks = []; | ||||
|  | ||||
|         if (!GLib.file_test(this._bookmarksPath, GLib.FileTest.EXISTS)) | ||||
|             return; | ||||
|  | ||||
|         let [success, bookmarksContent, len] = GLib.file_get_contents(this._bookmarksPath); | ||||
|  | ||||
|         if (!success) | ||||
|             return; | ||||
|  | ||||
|         let bookmarks = bookmarksContent.split('\n'); | ||||
|  | ||||
|         let bookmarksToLabel = {}; | ||||
|         let bookmarksOrder = []; | ||||
|         for (let i = 0; i < bookmarks.length; i++) { | ||||
|             let bookmarkLine = bookmarks[i]; | ||||
|             let components = bookmarkLine.split(' '); | ||||
|             let bookmark = components[0]; | ||||
|             if (bookmark in bookmarksToLabel) | ||||
|                 continue; | ||||
|             let label = null; | ||||
|             if (components.length > 1) | ||||
|                 label = components.slice(1).join(' '); | ||||
|             bookmarksToLabel[bookmark] = label; | ||||
|             bookmarksOrder.push(bookmark); | ||||
|         } | ||||
|  | ||||
|         for (let i = 0; i < bookmarksOrder.length; i++) { | ||||
|             let bookmark = bookmarksOrder[i]; | ||||
|             let label = bookmarksToLabel[bookmark]; | ||||
|             let file = Gio.file_new_for_uri(bookmark); | ||||
|             if (!file.query_exists(null)) | ||||
|                 continue; | ||||
|             if (label == null) | ||||
|                 label = Shell.util_get_label_for_uri(bookmark); | ||||
|             if (label == null) | ||||
|                 continue; | ||||
|             let icon = Shell.util_get_icon_for_uri(bookmark); | ||||
|  | ||||
|             let item = new PlaceInfo('bookmark:' + bookmark, label, | ||||
|                 function(size) { | ||||
|                     return St.TextureCache.get_default().load_gicon(null, icon, size); | ||||
|                 }, | ||||
|                 function(params) { | ||||
|                     Gio.app_info_launch_default_for_uri(bookmark, _makeLaunchContext(params)); | ||||
|                 }); | ||||
|             this._bookmarks.push(item); | ||||
|         } | ||||
|  | ||||
|         /* See comment in _updateDevices for explanation why there are two signals. */ | ||||
|         this.emit('bookmarks-updated'); | ||||
|         this.emit('places-updated'); | ||||
|     }, | ||||
|  | ||||
|     _addMount: function(mount) { | ||||
|         let devItem = new PlaceDeviceInfo(mount); | ||||
|         this._mounts.push(devItem); | ||||
|     }, | ||||
|  | ||||
|     getAllPlaces: function () { | ||||
|         return this.getDefaultPlaces().concat(this.getBookmarks(), this.getMounts()); | ||||
|     }, | ||||
|  | ||||
|     getDefaultPlaces: function () { | ||||
|         return this._defaultPlaces; | ||||
|     }, | ||||
|  | ||||
|     getBookmarks: function () { | ||||
|         return this._bookmarks; | ||||
|     }, | ||||
|  | ||||
|     getMounts: function () { | ||||
|         return this._mounts; | ||||
|     }, | ||||
|  | ||||
|     _lookupIndexById: function(sourceArray, id) { | ||||
|         for (let i = 0; i < sourceArray.length; i++) { | ||||
|             let place = sourceArray[i]; | ||||
|             if (place.id == id) | ||||
|                 return i; | ||||
|         } | ||||
|         return -1; | ||||
|     }, | ||||
|  | ||||
|     lookupPlaceById: function(id) { | ||||
|         let colonIdx = id.indexOf(':'); | ||||
|         let type = id.substring(0, colonIdx); | ||||
|         let sourceArray = null; | ||||
|         if (type == 'special') | ||||
|             sourceArray = this._defaultPlaces; | ||||
|         else if (type == 'mount') | ||||
|             sourceArray = this._mounts; | ||||
|         else if (type == 'bookmark') | ||||
|             sourceArray = this._bookmarks; | ||||
|         return sourceArray[this._lookupIndexById(sourceArray, id)]; | ||||
|     }, | ||||
|  | ||||
|     _removeById: function(sourceArray, id) { | ||||
|         sourceArray.splice(this._lookupIndexById(sourceArray, id), 1); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(PlacesManager.prototype); | ||||
|  | ||||
|  | ||||
| function PlaceSearchProvider() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| PlaceSearchProvider.prototype = { | ||||
|     __proto__: Search.SearchProvider.prototype, | ||||
|  | ||||
|     _init: function() { | ||||
|         Search.SearchProvider.prototype._init.call(this, _("PLACES & DEVICES")); | ||||
|     }, | ||||
|  | ||||
|     getResultMeta: function(resultId) { | ||||
|         let placeInfo = Main.placesManager.lookupPlaceById(resultId); | ||||
|         if (!placeInfo) | ||||
|             return null; | ||||
|         return { 'id': resultId, | ||||
|                  'name': placeInfo.name, | ||||
|                  'createIcon': function(size) { | ||||
|                                    return placeInfo.iconFactory(size); | ||||
|                                } | ||||
|                }; | ||||
|     }, | ||||
|  | ||||
|     activateResult: function(id, params) { | ||||
|         let placeInfo = Main.placesManager.lookupPlaceById(id); | ||||
|         placeInfo.launch(params); | ||||
|     }, | ||||
|  | ||||
|     _compareResultMeta: function (idA, idB) { | ||||
|         let infoA = Main.placesManager.lookupPlaceById(idA); | ||||
|         let infoB = Main.placesManager.lookupPlaceById(idB); | ||||
|         return infoA.name.localeCompare(infoB.name); | ||||
|     }, | ||||
|  | ||||
|     _searchPlaces: function(places, terms) { | ||||
|         let multiplePrefixResults = []; | ||||
|         let prefixResults = []; | ||||
|         let multipleSubstringResults = []; | ||||
|         let substringResults = []; | ||||
|  | ||||
|         terms = terms.map(String.toLowerCase); | ||||
|  | ||||
|         for (let i = 0; i < places.length; i++) { | ||||
|             let place = places[i]; | ||||
|             let mtype = place.matchTerms(terms); | ||||
|             if (mtype == Search.MatchType.MULTIPLE_PREFIX) | ||||
|                 multiplePrefixResults.push(place.id); | ||||
|             else if (mtype == Search.MatchType.PREFIX) | ||||
|                 prefixResults.push(place.id); | ||||
|             else if (mtype == Search.MatchType.MULTIPLE_SUBSTRING) | ||||
|                 multipleSubstringResults.push(place.id); | ||||
|             else if (mtype == Search.MatchType.SUBSTRING) | ||||
|                 substringResults.push(place.id); | ||||
|         } | ||||
|         multiplePrefixResults.sort(this._compareResultMeta); | ||||
|         prefixResults.sort(this._compareResultMeta); | ||||
|         multipleSubstringResults.sort(this._compareResultMeta); | ||||
|         substringResults.sort(this._compareResultMeta); | ||||
|         return multiplePrefixResults.concat(prefixResults.concat(multipleSubstringResults.concat(substringResults))); | ||||
|     }, | ||||
|  | ||||
|     getInitialResultSet: function(terms) { | ||||
|         let places = Main.placesManager.getAllPlaces(); | ||||
|         return this._searchPlaces(places, terms); | ||||
|     }, | ||||
|  | ||||
|     getSubsearchResultSet: function(previousResults, terms) { | ||||
|         let places = previousResults.map(function (id) { return Main.placesManager.lookupPlaceById(id); }); | ||||
|         return this._searchPlaces(places, terms); | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										278
									
								
								js/ui/places.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,278 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ | ||||
|  | ||||
| const Big = imports.gi.Big; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Pango = imports.gi.Pango; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const DND = imports.ui.dnd; | ||||
| const Main = imports.ui.main; | ||||
| const GenericDisplay = imports.ui.genericDisplay; | ||||
|  | ||||
| const PLACES_VSPACING = 8; | ||||
| const PLACES_ICON_SIZE = 16; | ||||
|  | ||||
| /** | ||||
|  * An entry in the places menu. | ||||
|  * @name: String title | ||||
|  * @iconFactory: A JavaScript callback which will create an icon texture | ||||
|  * @onActivate: A JavaScript callback to launch the entry | ||||
|  */ | ||||
| function PlaceDisplay(name, iconFactory, onActivate) { | ||||
|     this._init(name, iconFactory, onActivate); | ||||
| } | ||||
|  | ||||
| PlaceDisplay.prototype = { | ||||
|     _init : function(name, iconFactory, onActivate) { | ||||
|         this.actor = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                    reactive: true, | ||||
|                                    spacing: 4 }); | ||||
|         this.actor.connect('button-release-event', Lang.bind(this, function (b, e) { | ||||
|             onActivate(this); | ||||
|             Main.overview.hide(); | ||||
|         })); | ||||
|         let text = new Clutter.Text({ font_name: "Sans 14px", | ||||
|                                       ellipsize: Pango.EllipsizeMode.END, | ||||
|                                       color: GenericDisplay.ITEM_DISPLAY_NAME_COLOR, | ||||
|                                       text: name }); | ||||
|         let iconBox = new Big.Box({ y_align: Big.BoxAlignment.CENTER }); | ||||
|         this._icon = iconFactory(); | ||||
|         iconBox.append(this._icon, Big.BoxPackFlags.NONE); | ||||
|         this.actor.append(iconBox, Big.BoxPackFlags.NONE); | ||||
|         this.actor.append(text, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         this._iconFactory = iconFactory; | ||||
|         this._onActivate = onActivate; | ||||
|  | ||||
|         this.actor._delegate = this; | ||||
|         let draggable = DND.makeDraggable(this.actor); | ||||
|     }, | ||||
|  | ||||
|     getDragActorSource: function() { | ||||
|         return this._icon; | ||||
|     }, | ||||
|  | ||||
|     getDragActor: function(stageX, stageY) { | ||||
|         return this._iconFactory(); | ||||
|     }, | ||||
|  | ||||
|     //// Drag and drop methods //// | ||||
|  | ||||
|     shellWorkspaceLaunch : function() { | ||||
|         this._onActivate(); | ||||
|     } | ||||
| }; | ||||
| Signals.addSignalMethods(PlaceDisplay.prototype); | ||||
|  | ||||
| function Places() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| Places.prototype = { | ||||
|     _init : function() { | ||||
|         this.actor = new Big.Box({ orientation: Big.BoxOrientation.HORIZONTAL, | ||||
|                                    spacing: 4 }); | ||||
|         this._menuBox = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                       spacing: PLACES_VSPACING }); | ||||
|         this.actor.append(this._menuBox, Big.BoxPackFlags.EXPAND); | ||||
|         this._devBox = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                       spacing: PLACES_VSPACING }); | ||||
|  | ||||
|         this._dirsBox = new Big.Box({ orientation: Big.BoxOrientation.VERTICAL, | ||||
|                                       spacing: PLACES_VSPACING }); | ||||
|         this.actor.append(this._dirsBox, Big.BoxPackFlags.EXPAND); | ||||
|  | ||||
|         let homeFile = Gio.file_new_for_path (GLib.get_home_dir()); | ||||
|         let homeUri = homeFile.get_uri(); | ||||
|         let homeLabel = Shell.util_get_label_for_uri (homeUri); | ||||
|         let homeIcon = Shell.util_get_icon_for_uri (homeUri); | ||||
|         let home = new PlaceDisplay(homeLabel, | ||||
|             function() { | ||||
|                 return Shell.TextureCache.get_default().load_gicon(homeIcon, PLACES_ICON_SIZE); | ||||
|             }, | ||||
|             function() { | ||||
|                 Gio.app_info_launch_default_for_uri(homeUri, Main.createAppLaunchContext()); | ||||
|             }); | ||||
|  | ||||
|         this._menuBox.append(home.actor, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         /* | ||||
|         * Show devices, code more or less ported from nautilus-places-sidebar.c | ||||
|         */ | ||||
|  | ||||
|         this._menuBox.append(this._devBox, Big.BoxPackFlags.NONE); | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
|         this._volumeMonitor.connect('volume-added', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('volume-removed',Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('volume-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-added', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-removed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('mount-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-connected', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-disconnected', Lang.bind(this, this._updateDevices)); | ||||
|         this._volumeMonitor.connect('drive-changed', Lang.bind(this, this._updateDevices)); | ||||
|         this._updateDevices(); | ||||
|  | ||||
|         let networkApp = null; | ||||
|         try { | ||||
|             networkApp = Shell.AppSystem.get_default().load_from_desktop_file('gnome-network-scheme.desktop'); | ||||
|         } catch(e) { | ||||
|             try { | ||||
|                 networkApp = Shell.AppSystem.get_default().load_from_desktop_file('network-scheme.desktop'); | ||||
|             } catch(e) { | ||||
|                 log("Cannot create \"Network\" item, .desktop file not found or corrupt."); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (networkApp != null) { | ||||
|             let network = new PlaceDisplay(networkApp.get_name(), | ||||
|                 function() { | ||||
|                     return networkApp.create_icon_texture(PLACES_ICON_SIZE); | ||||
|                 }, | ||||
|                 function () { | ||||
|                     networkApp.launch(); | ||||
|                 }); | ||||
|             this._menuBox.append(network.actor, Big.BoxPackFlags.NONE); | ||||
|         } | ||||
|  | ||||
|         let connect = new PlaceDisplay('Connect to...', | ||||
|             function () { | ||||
|                 return Shell.TextureCache.get_default().load_icon_name("applications-internet", PLACES_ICON_SIZE); | ||||
|             }, | ||||
|             function () { | ||||
|                 new Shell.Process({ args: ['nautilus-connect-server'] }).run(); | ||||
|             }); | ||||
|         this._menuBox.append(connect.actor, Big.BoxPackFlags.NONE); | ||||
|  | ||||
|         this._bookmarksPath = GLib.build_filenamev([GLib.get_home_dir(), ".gtk-bookmarks"]); | ||||
|         this._bookmarksFile = Gio.file_new_for_path(this._bookmarksPath); | ||||
|         let monitor = this._bookmarksFile.monitor_file(Gio.FileMonitorFlags.NONE, null); | ||||
|         this._bookmarkTimeoutId = 0; | ||||
|         monitor.connect('changed', Lang.bind(this, function () { | ||||
|             if (this._bookmarkTimeoutId > 0) | ||||
|                 return; | ||||
|             /* Defensive event compression */ | ||||
|             this._bookmarkTimeoutId = Mainloop.timeout_add(100, Lang.bind(this, function () { | ||||
|                 this._bookmarkTimeoutId = 0; | ||||
|                 this._reloadBookmarks(); | ||||
|                 return false; | ||||
|             })); | ||||
|         })); | ||||
|  | ||||
|         this._reloadBookmarks(); | ||||
|     }, | ||||
|  | ||||
|     _reloadBookmarks: function() { | ||||
|  | ||||
|         this._dirsBox.remove_all(); | ||||
|  | ||||
|         if (!GLib.file_test(this._bookmarksPath, GLib.FileTest.EXISTS)) | ||||
|           return; | ||||
|  | ||||
|         let [success, bookmarksContent, len] = GLib.file_get_contents(this._bookmarksPath); | ||||
|  | ||||
|         if (!success) | ||||
|             return; | ||||
|  | ||||
|         let bookmarks = bookmarksContent.split('\n'); | ||||
|  | ||||
|         let bookmarksToLabel = {}; | ||||
|         let bookmarksOrder = []; | ||||
|         for (let i = 0; i < bookmarks.length; i++) { | ||||
|             let bookmarkLine = bookmarks[i]; | ||||
|             let components = bookmarkLine.split(' '); | ||||
|             let bookmark = components[0]; | ||||
|             if (bookmark in bookmarksToLabel) | ||||
|                 continue; | ||||
|             let label = null; | ||||
|             if (components.length > 1) | ||||
|                 label = components.slice(1).join(' '); | ||||
|             bookmarksToLabel[bookmark] = label; | ||||
|             bookmarksOrder.push(bookmark); | ||||
|         } | ||||
|  | ||||
|         for (let i = 0; i < bookmarksOrder.length; i++) { | ||||
|             let bookmark = bookmarksOrder[i]; | ||||
|             let label = bookmarksToLabel[bookmark]; | ||||
|             let file = Gio.file_new_for_uri(bookmark); | ||||
|             if (!file.query_exists(null)) | ||||
|                 continue; | ||||
|             if (label == null) | ||||
|                 label = Shell.util_get_label_for_uri(bookmark); | ||||
|             if (label == null) | ||||
|                 continue; | ||||
|             let icon = Shell.util_get_icon_for_uri(bookmark); | ||||
|  | ||||
|             let item = new PlaceDisplay(label, | ||||
|                 function() { | ||||
|                     return Shell.TextureCache.get_default().load_gicon(icon, PLACES_ICON_SIZE); | ||||
|                 }, | ||||
|                 function() { | ||||
|                     Gio.app_info_launch_default_for_uri(bookmark, Main.createAppLaunchContext()); | ||||
|                 }); | ||||
|             this._dirsBox.append(item.actor, Big.BoxPackFlags.NONE); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateDevices: function() { | ||||
|         this._devBox.remove_all(); | ||||
|  | ||||
|         /* first go through all connected drives */ | ||||
|         let drives = this._volumeMonitor.get_connected_drives(); | ||||
|         for (let i = 0; i < drives.length; i++) { | ||||
|                 let volumes = drives[i].get_volumes(); | ||||
|                 for(let j = 0; j < volumes.length; j++) { | ||||
|                         let mount = volumes[j].get_mount(); | ||||
|                         if(mount != null) { | ||||
|                                 this._addMount(mount); | ||||
|                         } | ||||
|                 } | ||||
|         } | ||||
|  | ||||
|         /* add all volumes that is not associated with a drive */ | ||||
|         let volumes = this._volumeMonitor.get_volumes(); | ||||
|         for(let i = 0; i < volumes.length; i++) { | ||||
|                 if(volumes[i].get_drive() != null) | ||||
|                         continue; | ||||
|  | ||||
|                 let mount = volumes[i].get_mount(); | ||||
|                 if(mount != null) { | ||||
|                         this._addMount(mount); | ||||
|                 } | ||||
|         } | ||||
|  | ||||
|         /* add mounts that have no volume (/etc/mtab mounts, ftp, sftp,...) */ | ||||
|         let mounts = this._volumeMonitor.get_mounts(); | ||||
|         for(let i = 0; i < mounts.length; i++) { | ||||
|                 if(mounts[i].is_shadowed()) | ||||
|                         continue; | ||||
|  | ||||
|                 if(mounts[i].get_volume()) | ||||
|                         continue; | ||||
|  | ||||
|                 this._addMount(mounts[i]); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _addMount: function(mount) { | ||||
|         let mountLabel = mount.get_name(); | ||||
|         let mountIcon = mount.get_icon(); | ||||
|         let root = mount.get_root(); | ||||
|         let mountUri = root.get_uri(); | ||||
|         let devItem = new PlaceDisplay(mountLabel, | ||||
|                function() { | ||||
|                         return Shell.TextureCache.get_default().load_gicon(mountIcon, PLACES_ICON_SIZE); | ||||
|                }, | ||||
|                function() { | ||||
|                         Gio.app_info_launch_default_for_uri(mountUri, Main.createAppLaunchContext()); | ||||
|                }); | ||||
|         this._devBox.append(devItem.actor, Big.BoxPackFlags.NONE); | ||||
|     } | ||||
|  | ||||
| }; | ||||
| Signals.addSignalMethods(Places.prototype); | ||||
| @@ -1,412 +0,0 @@ | ||||
| /* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- | ||||
|  * | ||||
|  * Copyright 2010 Red Hat, Inc | ||||
|  * | ||||
|  * 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. | ||||
| * | ||||
|  * Author: David Zeuthen <davidz@redhat.com> | ||||
|  */ | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const Gettext = imports.gettext.domain('gnome-shell'); | ||||
| const _ = Gettext.gettext; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const St = imports.gi.St; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Gdm = imports.gi.Gdm; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Polkit = imports.gi.Polkit; | ||||
| const PolkitAgent = imports.gi.PolkitAgent; | ||||
|  | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
|  | ||||
| function AuthenticationDialog(actionId, message, cookie, userNames) { | ||||
|     this._init(actionId, message, cookie, userNames); | ||||
| } | ||||
|  | ||||
| AuthenticationDialog.prototype = { | ||||
|     __proto__: ModalDialog.ModalDialog.prototype, | ||||
|  | ||||
|     _init: function(actionId, message, cookie, userNames) { | ||||
|         ModalDialog.ModalDialog.prototype._init.call(this, { styleClass: 'polkit-dialog' }); | ||||
|  | ||||
|         this.actionId = actionId; | ||||
|         this.message = message; | ||||
|         this.userNames = userNames; | ||||
|         this._wasDismissed = false; | ||||
|         this._completed = false; | ||||
|  | ||||
|         let mainContentBox = new St.BoxLayout({ style_class: 'polkit-dialog-main-layout', | ||||
|                                                 vertical: false }); | ||||
|         this.contentLayout.add(mainContentBox, | ||||
|                                { x_fill: true, | ||||
|                                  y_fill: true }); | ||||
|  | ||||
|         let icon = new St.Icon({ icon_name: 'dialog-password-symbolic' }); | ||||
|         mainContentBox.add(icon, | ||||
|                            { x_fill:  true, | ||||
|                              y_fill:  false, | ||||
|                              x_align: St.Align.END, | ||||
|                              y_align: St.Align.START }); | ||||
|  | ||||
|         let messageBox = new St.BoxLayout({ style_class: 'polkit-dialog-message-layout', | ||||
|                                             vertical: true }); | ||||
|         mainContentBox.add(messageBox, | ||||
|                            { y_align: St.Align.START }); | ||||
|  | ||||
|         this._subjectLabel = new St.Label({ style_class: 'polkit-dialog-headline', | ||||
|                                             text: _("Authentication Required") }); | ||||
|  | ||||
|         messageBox.add(this._subjectLabel, | ||||
|                        { y_fill:  false, | ||||
|                          y_align: St.Align.START }); | ||||
|  | ||||
|         this._descriptionLabel = new St.Label({ style_class: 'polkit-dialog-description', | ||||
|                                                 text: message }); | ||||
|         this._descriptionLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this._descriptionLabel.clutter_text.line_wrap = true; | ||||
|  | ||||
|         messageBox.add(this._descriptionLabel, | ||||
|                        { y_fill:  true, | ||||
|                          y_align: St.Align.START }); | ||||
|  | ||||
|         if (userNames.length > 1) { | ||||
|             log('polkitAuthenticationAgent: Received ' + userNames.length + | ||||
|                 ' identities that can be used for authentication. Only ' + | ||||
|                 'considering the first one.'); | ||||
|         } | ||||
|  | ||||
|         let userName = userNames[0]; | ||||
|  | ||||
|         this._user = Gdm.UserManager.ref_default().get_user(userName); | ||||
|         let userRealName = this._user.get_real_name() | ||||
|         this._userLoadedId = this._user.connect('notify::is_loaded', | ||||
|                                                 Lang.bind(this, this._onUserChanged)); | ||||
|         this._userChangedId = this._user.connect('changed', | ||||
|                                                  Lang.bind(this, this._onUserChanged)); | ||||
|  | ||||
|         // Special case 'root' | ||||
|         let userIsRoot = false; | ||||
|         if (userName == 'root') { | ||||
|             userIsRoot = true; | ||||
|             userRealName = _("Administrator"); | ||||
|         } | ||||
|  | ||||
|         if (userIsRoot) { | ||||
|             let userLabel = new St.Label(({ style_class: 'polkit-dialog-user-root-label', | ||||
|                                             text: userRealName })); | ||||
|             messageBox.add(userLabel); | ||||
|         } else { | ||||
|             let userBox = new St.BoxLayout({ style_class: 'polkit-dialog-user-layout', | ||||
|                                              vertical: false }); | ||||
|             messageBox.add(userBox); | ||||
|             this._userIcon = new St.Icon(); | ||||
|             this._userIcon.hide(); | ||||
|             userBox.add(this._userIcon, | ||||
|                         { x_fill:  true, | ||||
|                           y_fill:  false, | ||||
|                           x_align: St.Align.END, | ||||
|                           y_align: St.Align.START }); | ||||
|             let userLabel = new St.Label(({ style_class: 'polkit-dialog-user-label', | ||||
|                                             text: userRealName })); | ||||
|             userBox.add(userLabel, | ||||
|                         { x_fill:  true, | ||||
|                           y_fill:  false, | ||||
|                           x_align: St.Align.END, | ||||
|                           y_align: St.Align.MIDDLE }); | ||||
|         } | ||||
|  | ||||
|         this._onUserChanged(); | ||||
|  | ||||
|         this._passwordBox = new St.BoxLayout({ vertical: false }); | ||||
|         messageBox.add(this._passwordBox); | ||||
|         this._passwordLabel = new St.Label(({ style_class: 'polkit-dialog-password-label' })); | ||||
|         this._passwordBox.add(this._passwordLabel); | ||||
|         this._passwordEntry = new St.Entry({ style_class: 'polkit-dialog-password-entry', | ||||
|                                              text: "", | ||||
|                                              can_focus: true}); | ||||
|         this._passwordEntry.clutter_text.connect('activate', Lang.bind(this, this._onEntryActivate)); | ||||
|         this._passwordBox.add(this._passwordEntry, | ||||
|                               {expand: true }); | ||||
|         this._passwordBox.hide(); | ||||
|  | ||||
|         this._errorMessageLabel = new St.Label({ style_class: 'polkit-dialog-error-label' }); | ||||
|         this._errorMessageLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this._errorMessageLabel.clutter_text.line_wrap = true; | ||||
|         messageBox.add(this._errorMessageLabel); | ||||
|         this._errorMessageLabel.hide(); | ||||
|  | ||||
|         this._infoMessageLabel = new St.Label({ style_class: 'polkit-dialog-info-label' }); | ||||
|         this._infoMessageLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this._infoMessageLabel.clutter_text.line_wrap = true; | ||||
|         messageBox.add(this._infoMessageLabel); | ||||
|         this._infoMessageLabel.hide(); | ||||
|  | ||||
|         /* text is intentionally non-blank otherwise the height is not the same as for | ||||
|          * infoMessage and errorMessageLabel - but it is still invisible because | ||||
|          * gnome-shell.css sets the color to be transparent | ||||
|          */ | ||||
|         this._nullMessageLabel = new St.Label({ style_class: 'polkit-dialog-null-label', | ||||
|                                                 text: 'abc'}); | ||||
|         this._nullMessageLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         this._nullMessageLabel.clutter_text.line_wrap = true; | ||||
|         messageBox.add(this._nullMessageLabel); | ||||
|         this._nullMessageLabel.show(); | ||||
|  | ||||
|         this.setButtons([{ label: _("Cancel"), | ||||
|                            action: Lang.bind(this, this.cancel), | ||||
|                            key:    Clutter.Escape | ||||
|                          }, | ||||
|                          { label:  _("Authenticate"), | ||||
|                            action: Lang.bind(this, this._onAuthenticateButtonPressed) | ||||
|                          }]); | ||||
|  | ||||
|         this._doneEmitted = false; | ||||
|  | ||||
|         this._identityToAuth = Polkit.UnixUser.new_for_name(userName); | ||||
|         this._cookie = cookie; | ||||
|  | ||||
|         this._session = new PolkitAgent.Session({ identity: this._identityToAuth, | ||||
|                                                   cookie: this._cookie }); | ||||
|         this._session.connect('completed', Lang.bind(this, this._onSessionCompleted)); | ||||
|         this._session.connect('request', Lang.bind(this, this._onSessionRequest)); | ||||
|         this._session.connect('show-error', Lang.bind(this, this._onSessionShowError)); | ||||
|         this._session.connect('show-info', Lang.bind(this, this._onSessionShowInfo)); | ||||
|  | ||||
|         // Delay focus grab to avoid ModalDialog stealing focus with | ||||
|         // its buttons | ||||
|         this.connect('opened', | ||||
|                      Lang.bind(this, function() { | ||||
|                          this._passwordEntry.grab_key_focus(); | ||||
|                      })); | ||||
|     }, | ||||
|  | ||||
|     startAuthentication: function() { | ||||
|         this._session.initiate(); | ||||
|     }, | ||||
|  | ||||
|     _ensureOpen: function() { | ||||
|         // NOTE: ModalDialog.open() is safe to call if the dialog is | ||||
|         // already open - it just returns true without side-effects | ||||
|         if (!this.open(global.get_current_time())) { | ||||
|             // This can fail if e.g. unable to get input grab | ||||
|             // | ||||
|             // In an ideal world this wouldn't happen (because the | ||||
|             // Shell is in complete control of the session) but that's | ||||
|             // just not how things work right now. | ||||
|             // | ||||
|             // One way to make this happen is by running 'sleep 3; | ||||
|             // pkexec bash' and then opening a popup menu. | ||||
|             // | ||||
|             // We could add retrying if this turns out to be a problem | ||||
|  | ||||
|             log('polkitAuthenticationAgent: Failed to show modal dialog.' + | ||||
|                 ' Dismissing authentication request for action-id ' + this.actionId + | ||||
|                 ' cookie ' + this._cookie); | ||||
|             this._emitDone(false, true); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _emitDone: function(keepVisible, dismissed) { | ||||
|         if (!this._doneEmitted) { | ||||
|             this._doneEmitted = true; | ||||
|             this.emit('done', keepVisible, dismissed); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onEntryActivate: function() { | ||||
|         let response = this._passwordEntry.get_text(); | ||||
|         this._session.response(response); | ||||
|         // When the user responds, dismiss already shown info and | ||||
|         // error texts (if any) | ||||
|         this._errorMessageLabel.hide(); | ||||
|         this._infoMessageLabel.hide(); | ||||
|         this._nullMessageLabel.show(); | ||||
|     }, | ||||
|  | ||||
|     _onAuthenticateButtonPressed: function() { | ||||
|         this._onEntryActivate(); | ||||
|     }, | ||||
|  | ||||
|     _onSessionCompleted: function(session, gainedAuthorization) { | ||||
|         if (this._completed) | ||||
|             return; | ||||
|  | ||||
|         this._completed = true; | ||||
|  | ||||
|         if (!gainedAuthorization) { | ||||
|             /* Unless we are showing an existing error message from the PAM | ||||
|              * module (the PAM module could be reporting the authentication | ||||
|              * error providing authentication-method specific information), | ||||
|              * show "Sorry, that didn't work. Please try again." | ||||
|              */ | ||||
|             if (!this._errorMessageLabel.visible && !this._wasDismissed) { | ||||
|                 /* Translators: "that didn't work" refers to the fact that the | ||||
|                  * requested authentication was not gained; this can happen | ||||
|                  * because of an authentication error (like invalid password), | ||||
|                  * for instance. */ | ||||
|                 this._errorMessageLabel.set_text(_("Sorry, that didn\'t work. Please try again.")); | ||||
|                 this._errorMessageLabel.show(); | ||||
|                 this._infoMessageLabel.hide(); | ||||
|                 this._nullMessageLabel.hide(); | ||||
|             } | ||||
|         } | ||||
|         this._emitDone(!gainedAuthorization, false); | ||||
|     }, | ||||
|  | ||||
|     _onSessionRequest: function(session, request, echo_on) { | ||||
|         // Cheap localization trick | ||||
|         if (request == 'Password:') | ||||
|             this._passwordLabel.set_text(_("Password:")); | ||||
|         else | ||||
|             this._passwordLabel.set_text(request); | ||||
|  | ||||
|         if (echo_on) | ||||
|             this._passwordEntry.clutter_text.set_password_char(''); | ||||
|         else | ||||
|             this._passwordEntry.clutter_text.set_password_char('\u25cf'); // ● U+25CF BLACK CIRCLE | ||||
|  | ||||
|         this._passwordBox.show(); | ||||
|         this._passwordEntry.set_text(''); | ||||
|         this._passwordEntry.grab_key_focus(); | ||||
|         this._ensureOpen(); | ||||
|     }, | ||||
|  | ||||
|     _onSessionShowError: function(session, text) { | ||||
|         this._passwordEntry.set_text(''); | ||||
|         this._errorMessageLabel.set_text(text); | ||||
|         this._errorMessageLabel.show(); | ||||
|         this._infoMessageLabel.hide(); | ||||
|         this._nullMessageLabel.hide(); | ||||
|         this._ensureOpen(); | ||||
|     }, | ||||
|  | ||||
|     _onSessionShowInfo: function(session, text) { | ||||
|         this._passwordEntry.set_text(''); | ||||
|         this._infoMessageLabel.set_text(text); | ||||
|         this._infoMessageLabel.show(); | ||||
|         this._errorMessageLabel.hide(); | ||||
|         this._nullMessageLabel.hide(); | ||||
|         this._ensureOpen(); | ||||
|     }, | ||||
|  | ||||
|     destroySession: function() { | ||||
|         if (this._session) { | ||||
|             if (!this._completed) | ||||
|                 this._session.cancel(); | ||||
|             this._session = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onUserChanged: function() { | ||||
|         if (this._user.is_loaded) { | ||||
|             if (this._userIcon) { | ||||
|                 let iconFileName = this._user.get_icon_file(); | ||||
|                 let iconFile = Gio.file_new_for_path(iconFileName); | ||||
|                 let icon; | ||||
|                 if (iconFile.query_exists(null)) { | ||||
|                     icon = new Gio.FileIcon({file: iconFile}); | ||||
|                 } else { | ||||
|                     icon = new Gio.ThemedIcon({name: 'avatar-default'}); | ||||
|                 } | ||||
|                 this._userIcon.set_gicon (icon); | ||||
|                 this._userIcon.show(); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|         this._wasDismissed = true; | ||||
|         this.close(global.get_current_time()); | ||||
|         this._emitDone(false, true); | ||||
|     }, | ||||
|  | ||||
| }; | ||||
| Signals.addSignalMethods(AuthenticationDialog.prototype); | ||||
|  | ||||
| function AuthenticationAgent() { | ||||
|     this._init(); | ||||
| } | ||||
|  | ||||
| AuthenticationAgent.prototype = { | ||||
|     _init: function() { | ||||
|         this._native = new Shell.PolkitAuthenticationAgent(); | ||||
|         this._native.connect('initiate', Lang.bind(this, this._onInitiate)); | ||||
|         this._native.connect('cancel', Lang.bind(this, this._onCancel)); | ||||
|         this._currentDialog = null; | ||||
|         this._isCompleting = false; | ||||
|     }, | ||||
|  | ||||
|     _onInitiate: function(nativeAgent, actionId, message, iconName, cookie, userNames) { | ||||
|         this._currentDialog = new AuthenticationDialog(actionId, message, cookie, userNames); | ||||
|  | ||||
|         // We actually don't want to open the dialog until we know for | ||||
|         // sure that we're going to interact with the user. For | ||||
|         // example, if the password for the identity to auth is blank | ||||
|         // (which it will be on a live CD) then there will be no | ||||
|         // conversation at all... of course, we don't *know* that | ||||
|         // until we actually try it. | ||||
|         // | ||||
|         // See https://bugzilla.gnome.org/show_bug.cgi?id=643062 for more | ||||
|         // discussion. | ||||
|  | ||||
|         this._currentDialog.connect('done', Lang.bind(this, this._onDialogDone)); | ||||
|         this._currentDialog.startAuthentication(); | ||||
|     }, | ||||
|  | ||||
|     _onCancel: function(nativeAgent) { | ||||
|         this._completeRequest(false, false); | ||||
|     }, | ||||
|  | ||||
|     _onDialogDone: function(dialog, keepVisible, dismissed) { | ||||
|         this._completeRequest(keepVisible, dismissed); | ||||
|     }, | ||||
|  | ||||
|     _reallyCompleteRequest: function(dismissed) { | ||||
|         this._currentDialog.close(); | ||||
|         this._currentDialog.destroySession(); | ||||
|         this._currentDialog = null; | ||||
|         this._isCompleting = false; | ||||
|  | ||||
|         this._native.complete(dismissed) | ||||
|     }, | ||||
|  | ||||
|     _completeRequest: function(keepVisible, wasDismissed) { | ||||
|         if (this._isCompleting) | ||||
|             return; | ||||
|  | ||||
|         this._isCompleting = true; | ||||
|  | ||||
|         if (keepVisible) { | ||||
|             // Give the user 2 seconds to read 'Authentication Failure' before | ||||
|             // dismissing the dialog | ||||
|             Mainloop.timeout_add(2000, | ||||
|                                  Lang.bind(this, | ||||
|                                            function() { | ||||
|                                                this._reallyCompleteRequest(wasDismissed); | ||||
|                                            })); | ||||
|         } else { | ||||
|             this._reallyCompleteRequest(wasDismissed); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| function init() { | ||||
|     let agent = new AuthenticationAgent(); | ||||
| } | ||||