Compare commits
	
		
			708 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | df81c98b65 | ||
|   | 43ada376f8 | ||
|   | e317bfdab7 | ||
|   | f38263a67f | ||
|   | 3a03a36533 | ||
|   | 9f3a7ab432 | ||
|   | f3f887da73 | ||
|   | 443e0e6b1f | ||
|   | 8b65871c90 | ||
|   | 59b9ffa864 | ||
|   | c9965f43c1 | ||
|   | fada6645a2 | ||
|   | e516b40c8c | ||
|   | 2c795f7d05 | ||
|   | 94438e9d69 | ||
|   | 4d12baea02 | ||
|   | f34b5a04a1 | ||
|   | 5c5ed37b8b | ||
|   | 3e40168e54 | ||
|   | a2a7ca6882 | ||
|   | ce5a82a294 | ||
|   | 462bfccc7e | ||
|   | 61be034c6d | ||
|   | da49b8110e | ||
|   | 7b3eeefc85 | ||
|   | fc1849b475 | ||
|   | ef999930f6 | ||
|   | 402676be84 | ||
|   | c68abfbacf | ||
|   | 2758c71cf7 | ||
|   | b772a14cbd | ||
|   | 8d66fff2aa | ||
|   | 75e0894de0 | ||
|   | fbef56b602 | ||
|   | c398e02700 | ||
|   | 48cfd35b65 | ||
|   | ce38ae64cc | ||
|   | 98e6fd9cfb | ||
|   | b30ecd29d8 | ||
|   | a0df7aa2b8 | ||
|   | e2500092fd | ||
|   | cbeebbf264 | ||
|   | aebf641d2d | ||
|   | 1c8af221af | ||
|   | f0ef47a70a | ||
|   | 5f0030edf7 | ||
|   | dde91ab9a6 | ||
|   | 3680f7c814 | ||
|   | a27ca0765a | ||
|   | fa76e39717 | ||
|   | aa5c722eca | ||
|   | 1fd968d520 | ||
|   | ef3285d5e7 | ||
|   | e71b0a57fb | ||
|   | 333becef45 | ||
|   | 3732a6891d | ||
|   | 6a5e3ea12d | ||
|   | 4cf6293066 | ||
|   | ee673cc5c4 | ||
|   | 604bb1fe82 | ||
|   | 1f83911bb6 | ||
|   | 95a6e52f1a | ||
|   | 5e9d491480 | ||
|   | 068ebbf2a6 | ||
|   | 6bc222f37b | ||
|   | 6cf53a8d1c | ||
|   | 95e141e47e | ||
|   | b053f4c74b | ||
|   | d0d60fbd93 | ||
|   | a6f572e49a | ||
|   | bd49559357 | ||
|   | ec6841429f | ||
|   | 98ccc45f80 | ||
|   | 868a199144 | ||
|   | 7be4238aa2 | ||
|   | cb698f9a09 | ||
|   | d2dad4f0fa | ||
|   | cdbb39b9e4 | ||
|   | bfbb442b9d | ||
|   | 3531052e69 | ||
|   | c9f9ac0f35 | ||
|   | 3ccdc5f8ba | ||
|   | ac3e315a47 | ||
|   | 6972bc90f2 | ||
|   | a2a5a3c908 | ||
|   | d8b2cd5f8e | ||
|   | 3b759f7966 | ||
|   | 0ded0dbfd5 | ||
|   | bca8e371b0 | ||
|   | c2f5813463 | ||
|   | f4301147bb | ||
|   | ea9e5bc52c | ||
|   | f08e2b72d2 | ||
|   | 240cdc6565 | ||
|   | aa2ae29ad8 | ||
|   | a4fb55b4ca | ||
|   | 5a8923ef95 | ||
|   | a71ab9abb2 | ||
|   | 5cddab3305 | ||
|   | c4a775e94a | ||
|   | c7031d9e56 | ||
|   | d113c92b5f | ||
|   | 3ba8a578e8 | ||
|   | 7aa75f8eb0 | ||
|   | 98d3a6772f | ||
|   | 625a38f605 | ||
|   | 8d617166fe | ||
|   | ae4e82c82b | ||
|   | 7bb7734869 | ||
|   | 2f554e27f2 | ||
|   | b4c8a00e38 | ||
|   | f755e73a48 | ||
|   | 12cb023050 | ||
|   | 299ec4f368 | ||
|   | 4da7f20759 | ||
|   | ea66329423 | ||
|   | 39d3ff6caf | ||
|   | 513d1a195d | ||
|   | d54b87c455 | ||
|   | 397460d64c | ||
|   | 1900468846 | ||
|   | fb6598ac10 | ||
|   | 12b00501cc | ||
|   | c647be375a | ||
|   | 7b31d2146a | ||
|   | 3b846f0e9c | ||
|   | 21056a967f | ||
|   | ead4c2b863 | ||
|   | 086fcd4a32 | ||
|   | b82f11acf3 | ||
|   | 1da9546c40 | ||
|   | e69cc20fc7 | ||
|   | c363e2a322 | ||
|   | dd0c3296bb | ||
|   | 52048c03da | ||
|   | c8765eeebd | ||
|   | b9b9faacbd | ||
|   | d04340c675 | ||
|   | c556d28b98 | ||
|   | a333bfec19 | ||
|   | 19fbbfd74d | ||
|   | f393bb41aa | ||
|   | 4bf53cd507 | ||
|   | fd45d3589b | ||
|   | 6efa9e46ca | ||
|   | 9f5a2d8aca | ||
|   | ce35d523a2 | ||
|   | 0bfaa5c6a3 | ||
|   | 67ec1e5519 | ||
|   | cc05d303d8 | ||
|   | 554de3fb24 | ||
|   | 76315fca9f | ||
|   | af889168f0 | ||
|   | d8ca374a06 | ||
|   | 19c7da7b08 | ||
|   | 42ccfe0ccd | ||
|   | 538272bb23 | ||
|   | 985e909876 | ||
|   | d183f13456 | ||
|   | e150a9cff2 | ||
|   | 60b3d43ea7 | ||
|   | e3a8a284a4 | ||
|   | 3136ca086b | ||
|   | 3f913b2895 | ||
|   | 8345fe006e | ||
|   | 67a807e849 | ||
|   | cea946e745 | ||
|   | 966ae2c370 | ||
|   | 897144baba | ||
|   | f585fee4cf | ||
|   | 85ec77eb26 | ||
|   | 2d372e6201 | ||
|   | cb84c3a921 | ||
|   | fce3cd534a | ||
|   | 3291281957 | ||
|   | 01c6392c13 | ||
|   | 186f9b2e2c | ||
|   | 2c8e405e27 | ||
|   | 94446eb8c4 | ||
|   | 6803528810 | ||
|   | 2aa4fb02dd | ||
|   | 906106c700 | ||
|   | 0257a23c31 | ||
|   | e0eebc90e0 | ||
|   | ddeac2386f | ||
|   | 7b9be2188a | ||
|   | 117a81e06f | ||
|   | 522fe8e8bf | ||
|   | 73480255b1 | ||
|   | 09d3f8eb71 | ||
|   | 0592adead5 | ||
|   | b21f5c5566 | ||
|   | 7602260462 | ||
|   | 0448effd82 | ||
|   | 90dedccc87 | ||
|   | b29e8a1a3d | ||
|   | 9689a9ba8b | ||
|   | 4eb0a672db | ||
|   | cad56c871a | ||
|   | 0acc7ed4c2 | ||
|   | 2940ef07e9 | ||
|   | deddac8748 | ||
|   | f4cc3327e8 | ||
|   | 977448b6bc | ||
|   | d69eceab3b | ||
|   | ed7649c93a | ||
|   | af6ed9e02a | ||
|   | 983314333f | ||
|   | 26389f92f5 | ||
|   | 1e8e5d4671 | ||
|   | 31d375093e | ||
|   | 546ae00854 | ||
|   | e00bfcc2cf | ||
|   | 943f6c9d28 | ||
|   | dbf16d6ea3 | ||
|   | 77b0c07410 | ||
|   | e36eb02ab0 | ||
|   | 2935fd4ffe | ||
|   | 5961f162fa | ||
|   | 5d440fa8d9 | ||
|   | 3d4408dce8 | ||
|   | ea707dcd75 | ||
|   | 196429a18f | ||
|   | 487b5cd0a9 | ||
|   | 891f30ff63 | ||
|   | 45e42d7b9e | ||
|   | facaea6850 | ||
|   | e467a734a1 | ||
|   | 050378743e | ||
|   | 25bf2ff87a | ||
|   | 551b96d79f | ||
|   | 1291bcd0c8 | ||
|   | 73362beb0b | ||
|   | 52503c4c38 | ||
|   | 219b0bf987 | ||
|   | cbfd6a8128 | ||
|   | 21d11b3638 | ||
|   | 455595f2b2 | ||
|   | 316f825b2a | ||
|   | 35fcd16992 | ||
|   | 084f7a36b1 | ||
|   | 1fe3ab8f68 | ||
|   | be3c3c64c1 | ||
|   | 568454abb8 | ||
|   | 5014434b9a | ||
|   | b05ab0eaeb | ||
|   | e402ed1fe4 | ||
|   | 4801d1dd7f | ||
|   | 24c0a1a1d4 | ||
|   | 4a1f939e2b | ||
|   | 111a9e13f2 | ||
|   | 926f76635e | ||
|   | 72a43a6001 | ||
|   | 5180ab262c | ||
|   | 5fca85cd28 | ||
|   | fb5b368ca7 | ||
|   | 62b6419332 | ||
|   | 65c136f4ed | ||
|   | 34606c0a8c | ||
|   | 260657c0b8 | ||
|   | f5cc579272 | ||
|   | b4ce0e7208 | ||
|   | eb3fc7815e | ||
|   | 2f5a226bc2 | ||
|   | 01eb79a3cc | ||
|   | 7ea364ae8b | ||
|   | 48a6e6f309 | ||
|   | 0f63ad0fc1 | ||
|   | 49c4ba5656 | ||
|   | 642bf2b778 | ||
|   | 328bb1c21b | ||
|   | 38add2e78b | ||
|   | 2dc41c944e | ||
|   | ea552ea157 | ||
|   | a72dca3610 | ||
|   | 5e5035a0f7 | ||
|   | ffdb85e003 | ||
|   | 6a969b934f | ||
|   | 8d3ff56846 | ||
|   | d8d046f2b3 | ||
|   | 0bec76b6ee | ||
|   | aa3aea7520 | ||
|   | 49fcc93a4b | ||
|   | eaec459ee9 | ||
|   | 72c6f0025d | ||
|   | aa3caff714 | ||
|   | 522ccaba3f | ||
|   | 3d00864599 | ||
|   | bd3dfb8f6a | ||
|   | 1b3c89f32b | ||
|   | a4475465f1 | ||
|   | 84f14cb61c | ||
|   | d0de411a59 | ||
|   | 2b9645b268 | ||
|   | 4f08bebbf0 | ||
|   | 4e8c476153 | ||
|   | 1d374ac8bd | ||
|   | 83f0f4ea36 | ||
|   | 18f569280c | ||
|   | fc0c98805f | ||
|   | 6722b69b8a | ||
|   | f9d909f985 | ||
|   | dd42cfa853 | ||
|   | 4ed4bb330e | ||
|   | f02b007337 | ||
|   | 0035de4ab7 | ||
|   | 28552d4b38 | ||
|   | 42066a7c46 | ||
|   | 45847470d1 | ||
|   | 72b0a3d78f | ||
|   | fc3ad390b7 | ||
|   | 926de53c0c | ||
|   | e6cd112379 | ||
|   | 2cefc8be27 | ||
|   | 991179835e | ||
|   | 0ca2fee54f | ||
|   | 778ef365d9 | ||
|   | 45937ed85d | ||
|   | 65a9c09c9c | ||
|   | 67b14d5fe5 | ||
|   | f976e46c46 | ||
|   | 7026209eae | ||
|   | 1e82d6edc2 | ||
|   | 50ac1dd197 | ||
|   | 9d933356e1 | ||
|   | c2a5c00111 | ||
|   | edb561a6ff | ||
|   | 14c2460fe4 | ||
|   | 598362448a | ||
|   | 845273d4c0 | ||
|   | 3aebfdc319 | ||
|   | f423c371c8 | ||
|   | 6fe1ad62ba | ||
|   | 63efe1db11 | ||
|   | 002ae0e7f1 | ||
|   | 4f5e5e1999 | ||
|   | f4feb409ff | ||
|   | 05a035e6fd | ||
|   | 7c2a3cc233 | ||
|   | 7f660fd4d7 | ||
|   | c0730b610e | ||
|   | b0cfcc4842 | ||
|   | 1afc9fefcc | ||
|   | 90bea5785e | ||
|   | 61dacf3d91 | ||
|   | 3ae45bddad | ||
|   | 9ad4070d58 | ||
|   | 1981b21ea8 | ||
|   | 9896135c97 | ||
|   | 0d340099da | ||
|   | cacc780111 | ||
|   | a9a6da08a1 | ||
|   | ccb6b93b65 | ||
|   | 5623f3b4b5 | ||
|   | a227d595f5 | ||
|   | 012955e341 | ||
|   | a41cd7d3cc | ||
|   | 11d947f1e5 | ||
|   | 2bb717b0b7 | ||
|   | 770f8c6538 | ||
|   | e76e0042a8 | ||
|   | 91e266113c | ||
|   | ed2956a2ea | ||
|   | aa6f352cb0 | ||
|   | 5a42a91ee0 | ||
|   | 4d682c7861 | ||
|   | 20fec42496 | ||
|   | 437df2f0bd | ||
|   | 445aa54622 | ||
|   | 7fa9ca0a9c | ||
|   | 0fe7ae1810 | ||
|   | 9c251d85cf | ||
|   | 959b5fe5cf | ||
|   | 75a8bf626a | ||
|   | eb514f335c | ||
|   | ad77b4ddfc | ||
|   | 534bf2b000 | ||
|   | 3821fd04a4 | ||
|   | b55b9cc7a9 | ||
|   | ac6462c7c6 | ||
|   | aa7b0a285b | ||
|   | 0a63de8c8d | ||
|   | 0992bd41ed | ||
|   | ed52a5a58b | ||
|   | 79bf770783 | ||
|   | c7e3b68dcd | ||
|   | 1b9318c82f | ||
|   | 6a142c4260 | ||
|   | c49f2e1384 | ||
|   | 9208473e59 | ||
|   | 08ece4c186 | ||
|   | 99c29366ee | ||
|   | 7a11964dfb | ||
|   | c4922f6624 | ||
|   | 4f2070a7c6 | ||
|   | 03177dc474 | ||
|   | b9b4886a6f | ||
|   | 33e35f269f | ||
|   | b886656f61 | ||
|   | 8589bfb62e | ||
|   | 6a36a68f32 | ||
|   | a0a701757e | ||
|   | effe6fab3a | ||
|   | 8e560f98d1 | ||
|   | 58aabfcf5b | ||
|   | 19e3f794f8 | ||
|   | 2b1077aaa1 | ||
|   | 547cdf86cc | ||
|   | 7653175c6f | ||
|   | a4c1b55111 | ||
|   | 93205eefb9 | ||
|   | ec3f8d4b85 | ||
|   | 0258c7a518 | ||
|   | ebfdb8ec3c | ||
|   | 6823bad2d8 | ||
|   | 491100c7ee | ||
|   | 762e770c84 | ||
|   | c214e3b07a | ||
|   | b656b1c22f | ||
|   | d211680534 | ||
|   | d44c3d0cd3 | ||
|   | 6cb1692841 | ||
|   | 593acb954d | ||
|   | d209bc69b6 | ||
|   | 4d64bbcf7d | ||
|   | 5bd4329b11 | ||
|   | 02870e5363 | ||
|   | 2a59478b37 | ||
|   | 04030f22a6 | ||
|   | 2cc4558018 | ||
|   | 215eb5c65f | ||
|   | 5faef316b8 | ||
|   | 6751ca4c18 | ||
|   | 84e0a20701 | ||
|   | e04e507659 | ||
|   | 6a48dee037 | ||
|   | ea3fd0cf65 | ||
|   | 9c614057be | ||
|   | 8096e71c53 | ||
|   | 73977795a6 | ||
|   | 3b3445146d | ||
|   | 650dea017b | ||
|   | 70099872ab | ||
|   | dc5618558f | ||
|   | 62481f4b7c | ||
|   | d3418f6381 | ||
|   | 57dd862e35 | ||
|   | db76fb8ff9 | ||
|   | 9e5704b498 | ||
|   | 3deaeb4a90 | ||
|   | afb7f08e6b | ||
|   | 38c70e73b9 | ||
|   | e60bf44c2e | ||
|   | 746a8692ac | ||
|   | a00762ddd9 | ||
|   | 3981b27d8f | ||
|   | 62786c09a8 | ||
|   | 67c216a6fe | ||
|   | e933302ae4 | ||
|   | 75d5e84a4b | ||
|   | f160dda187 | ||
|   | 9c474a635a | ||
|   | d445bd17eb | ||
|   | 584f8c8381 | ||
|   | 032a688a72 | ||
|   | 285a7467d0 | ||
|   | 3289105ac4 | ||
|   | 51861b1e6b | ||
|   | d85f97c744 | ||
|   | 98847f2279 | ||
|   | 1d58ea25ab | ||
|   | 9de05994db | ||
|   | c9f3afc38f | ||
|   | 3526bc0f0a | ||
|   | 079809af1d | ||
|   | 26719b02e4 | ||
|   | 3a45ddcaad | ||
|   | f839100bc2 | ||
|   | c93767768c | ||
|   | 0011755b41 | ||
|   | ec932b2306 | ||
|   | 1c0c38574d | ||
|   | 78b81650f5 | ||
|   | 62bfde45aa | ||
|   | 5b624a34b8 | ||
|   | 9c008ab998 | ||
|   | 6edcd82103 | ||
|   | d5aa276c41 | ||
|   | d54efe0838 | ||
|   | d0fe1211f2 | ||
|   | 3e20843d9c | ||
|   | 7c9d90b0aa | ||
|   | b6e6e097b7 | ||
|   | 3842981c35 | ||
|   | 926177785d | ||
|   | 7d80647170 | ||
|   | c2a21bb885 | ||
|   | eaff1e9290 | ||
|   | 93c5e6d97e | ||
|   | 20fc9735fa | ||
|   | d450b74e10 | ||
|   | e8fa2b6417 | ||
|   | c459ef6888 | ||
|   | d1a3a000af | ||
|   | 5ce8980db3 | ||
|   | 8c32cff6fc | ||
|   | bc99db9fd3 | ||
|   | a5eae8e3d8 | ||
|   | 82a764ee93 | ||
|   | ce2c90a534 | ||
|   | 28cc0da151 | ||
|   | f1957dccb7 | ||
|   | 0af4dc0b4c | ||
|   | 687e1ebf69 | ||
|   | 805b686576 | ||
|   | 101daf6791 | ||
|   | 5d12ab415c | ||
|   | 0810ab62db | ||
|   | 6f00d81abf | ||
|   | 4184edc7f8 | ||
|   | 05f9f991d8 | ||
|   | 3df7ef6ce6 | ||
|   | d836194e31 | ||
|   | 8ed3e2117f | ||
|   | 62fcda5d91 | ||
|   | 0a780376f3 | ||
|   | efb9f167bd | ||
|   | eb69f3aa76 | ||
|   | c15a885418 | ||
|   | fe60db64e0 | ||
|   | b90cc5ff26 | ||
|   | e9f95ca605 | ||
|   | 017c2468ee | ||
|   | 8c6a2874ff | ||
|   | 557130d2f2 | ||
|   | 84cbbafaae | ||
|   | 72a663f554 | ||
|   | 7c762ef9df | ||
|   | 3182aba744 | ||
|   | e6339fbb45 | ||
|   | a6d8c25494 | ||
|   | cd4eda8bef | ||
|   | a0bd4a02e4 | ||
|   | 42b54aaa21 | ||
|   | bbfa616f27 | ||
|   | 7e31015ba2 | ||
|   | aa2fc3c858 | ||
|   | d4f0b5bdf3 | ||
|   | eda27d5194 | ||
|   | 589e6c29f3 | ||
|   | 4b46533ce8 | ||
|   | 6687b9b739 | ||
|   | 91c4408d23 | ||
|   | 83adb2a864 | ||
|   | 5918faddf7 | ||
|   | 916c02a2f5 | ||
|   | a84fb99c0a | ||
|   | 69d5cef3b2 | ||
|   | 38d05a8285 | ||
|   | f8899cf274 | ||
|   | dbbf4097a5 | ||
|   | acb1497f4f | ||
|   | e545ec59b9 | ||
|   | da26a9daf8 | ||
|   | a2f263dcbb | ||
|   | ce5cd3bf30 | ||
|   | 9a05aea76f | ||
|   | c9f6d5e2a1 | ||
|   | eba2b999ed | ||
|   | cceac0d8fb | ||
|   | 14eedf8651 | ||
|   | 9c6180afa2 | ||
|   | f1b1dbcb00 | ||
|   | ee23b8dbe0 | ||
|   | c9e00bee08 | ||
|   | 9970671bb1 | ||
|   | bb4502dca8 | ||
|   | d77c7a407c | ||
|   | 554001c0ed | ||
|   | 4d153bc96f | ||
|   | 476394809a | ||
|   | b6f3e15037 | ||
|   | 46c86e093c | ||
|   | c6350aa557 | ||
|   | 4e56af39da | ||
|   | 9fff972946 | ||
|   | e30925995f | ||
|   | f0d4260c81 | ||
|   | a7f82745c6 | ||
|   | a6fa6519d5 | ||
|   | a944dca60e | ||
|   | fdc443aebe | ||
|   | 7e08e1e0e7 | ||
|   | bd7938e02f | ||
|   | a583f45cc6 | ||
|   | 1071ac5d25 | ||
|   | 45793d0e47 | ||
|   | ea3866a07a | ||
|   | e41879a5c4 | ||
|   | 24dc926660 | ||
|   | 932b895127 | ||
|   | 9c4ffc4bf3 | ||
|   | 1ea2e2bcab | ||
|   | a8b15dd2cf | ||
|   | b1b8147ab8 | ||
|   | 39c210abed | ||
|   | 4bb2a364d3 | ||
|   | 6d3ebdcb5e | ||
|   | e865db57e0 | ||
|   | 19ba9a98b8 | ||
|   | 4a39af7f98 | ||
|   | c326aad9d7 | ||
|   | dc94f7b9f5 | ||
|   | e5be41b667 | ||
|   | 775bd961b6 | ||
|   | 3c67d012e7 | ||
|   | 2fbd8f063e | ||
|   | f285f2c69f | ||
|   | e375e1a857 | ||
|   | 910c95fa9b | ||
|   | b95c0682b0 | ||
|   | 9f460a36f6 | ||
|   | 41a3f10938 | ||
|   | d850c8599e | ||
|   | ec288d0e68 | ||
|   | 0b92cd0772 | ||
|   | c7f5f172dd | ||
|   | 84bc445593 | ||
|   | 365bfcae12 | ||
|   | 47c9243271 | ||
|   | 8c67a70db0 | ||
|   | 5f4591e24c | ||
|   | 37ef0e4bed | ||
|   | 7d7b92419f | ||
|   | 309d40a92b | ||
|   | 02718357da | ||
|   | cfef107114 | ||
|   | b742b1eed2 | ||
|   | d58be565a1 | ||
|   | 522ed3c21d | ||
|   | 2fb8781f30 | ||
|   | 2bda6db30f | ||
|   | 8abd18363c | ||
|   | 256bb532a2 | ||
|   | 6077e28f95 | ||
|   | 83cb26d70e | ||
|   | 6d66afc14e | ||
|   | 88faee4c79 | ||
|   | 66f5e4b44d | ||
|   | 772d8692e7 | ||
|   | 17f481f6fe | ||
|   | b057e786a4 | ||
|   | 8b9904b6d0 | ||
|   | 43ae3b8140 | ||
|   | f76dd4d6b2 | ||
|   | 079cc39166 | ||
|   | bf0c7f731d | ||
|   | 5a8a293614 | ||
|   | c768ee6175 | ||
|   | 75c2a723d9 | ||
|   | 32240df141 | ||
|   | c532e3f1a5 | ||
|   | b04c79643d | ||
|   | 4eca992db8 | ||
|   | c36ca625e6 | ||
|   | 238466b3d6 | ||
|   | 4f28840a59 | ||
|   | d98c1ba522 | ||
|   | 9bbb3e9c85 | ||
|   | 2710c56827 | ||
|   | e22ff0e42d | ||
|   | 8c74a4fee0 | ||
|   | 234b90ac86 | ||
|   | ce46b06f36 | ||
|   | 585930123d | ||
|   | 5d00c1a5ee | ||
|   | f288c43e6e | ||
|   | b981a591c7 | ||
|   | 3b7756b610 | ||
|   | 292f87caf7 | ||
|   | ae2751a68b | ||
|   | 589becbc79 | ||
|   | 4c7fcf272c | ||
|   | e51aecee03 | ||
|   | d98e7dbd4a | ||
|   | 1b97778925 | ||
|   | 829e7623df | ||
|   | 89675c9061 | ||
|   | f9df83802d | ||
|   | e51eb723fc | ||
|   | 5b61f2d642 | ||
|   | 094669baee | ||
|   | f6b5385495 | ||
|   | 524e2df708 | ||
|   | 9f887d9a28 | ||
|   | 214a41793f | ||
|   | 2e40ffc558 | ||
|   | 3584887938 | ||
|   | 0d6c002b8e | ||
|   | ec714864f2 | ||
|   | 1b77149ec9 | ||
|   | 93c9e031e3 | ||
|   | 3ff4277f86 | ||
|   | 2a63267be0 | ||
|   | 2b365627ed | ||
|   | b8f0d0f0dc | ||
|   | df1bed941d | ||
|   | 604085fdb9 | 
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -23,11 +23,15 @@ data/gnome-shell-wayland.desktop | ||||
| data/gnome-shell-wayland.desktop.in | ||||
| data/gnome-shell-extension-prefs.desktop | ||||
| data/gnome-shell-extension-prefs.desktop.in | ||||
| data/gnome-shell-theme.gresource | ||||
| data/gschemas.compiled | ||||
| data/perf-background.xml | ||||
| data/org.gnome.shell.gschema.xml | ||||
| data/org.gnome.shell.gschema.valid | ||||
| data/org.gnome.shell.evolution.calendar.gschema.xml | ||||
| data/org.gnome.shell.evolution.calendar.gschema.valid | ||||
| data/org.gnome.Shell.PortalHelper.desktop | ||||
| data/org.gnome.Shell.PortalHelper.service | ||||
| docs/reference/*/*.args | ||||
| docs/reference/*/*.bak | ||||
| docs/reference/*/*.hierarchy | ||||
| @@ -80,7 +84,9 @@ src/gnome-shell-extension-tool | ||||
| src/gnome-shell-hotplug-sniffer | ||||
| src/gnome-shell-perf-helper | ||||
| src/gnome-shell-perf-tool | ||||
| src/gnome-shell-portal-helper | ||||
| src/hotplug-sniffer/org.gnome.Shell.HotplugSniffer.service | ||||
| src/org-gtk-application.[ch] | ||||
| src/run-js-test | ||||
| src/test-recorder | ||||
| src/test-recorder.ogg | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +1,6 @@ | ||||
| [submodule "src/gvc"] | ||||
| 	path = src/gvc | ||||
| 	url = git://git.gnome.org/libgnome-volume-control | ||||
| [submodule "data/theme/gnome-shell-sass"] | ||||
| 	path = data/theme/gnome-shell-sass | ||||
| 	url = https://git.gnome.org/browse/gnome-shell-sass | ||||
|   | ||||
							
								
								
									
										264
									
								
								NEWS
									
									
									
									
									
								
							
							
						
						| @@ -1,3 +1,267 @@ | ||||
| 3.15.4 | ||||
| ====== | ||||
| * Add mode parameter to AcceleratorActivated signal [Florian; #711682] | ||||
| * Fix PID based window/app association [Sebastian; #736527] | ||||
| * Fix current day highlight on day change [Sebastian; #742492] | ||||
| * Switch to vp9 for screencast recordings [Adel; #742744] | ||||
| * Disable IBus input sources on password entries [Takao; #730628] | ||||
| * Make slider scrolling smoother [Adel; #742648] | ||||
| * Allow move-up shortcut to move window above top workspace [Florian; #665764] | ||||
| * Misc. bug fixes [Adel, Rui; #742748, #742824, #741114] | ||||
|  | ||||
| Contributors: | ||||
|   Cosimo Cecchi, Takao Fujiwara, Adel Gadllah, Sebastian Keller, Rui Matos, | ||||
|   Florian Müllner, Jasper St. Pierre | ||||
|  | ||||
| Translations: | ||||
|   Andika Triwidada [id], Matej Urbančič [sl], Saibal Ray [bn_IN], | ||||
|   Inaki Larranaga Murgoitio [eu], Stas Solovey [ru], Kjartan Maraas [nb], | ||||
|   Balázs Úr [hu], Marek Černocký [cs], Rafael Ferreira [pt_BR], | ||||
|   Bernd Homuth [de], Daniel Mustieles [es], Fabio Tomat [fur] | ||||
|  | ||||
| 3.15.3 | ||||
| ====== | ||||
| * Add support for high-contrast themes [Florian; #740447] | ||||
| * Fix banner message on login screen without user list [Ray; #703972] | ||||
| * Fix flicker when activating windows on another workspace [Florian; #741680] | ||||
| * Misc. bug fixes [Giovanni, Florian; #735308, #740237] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Florian Müllner, Jasper St. Pierre, Ray Strode | ||||
|  | ||||
| Translations: | ||||
|   Balázs Úr [hu], Josef Andersson [sv], Muhammet Kara [tr], | ||||
|   Baurzhan Muftakhidinov [kk], Inaki Larranaga Murgoitio [eu] | ||||
|  | ||||
| 3.15.2 | ||||
| ====== | ||||
| * Fix visual glitch of window preview outline in overview [Chris; #699044] | ||||
| * Change user facing name of "Captive Portal" to "Network Login" [Elad; #737198] | ||||
| * Port to Python 3 [Slavek; #732478] | ||||
| * Hide Airplane mode indicator when g-s-d says so [Cosimo; #736292] | ||||
| * Allow translators to change non-work days [Lavi; #664645] | ||||
| * Delay invocation of caribou daemon until really needed [Daiki; #739712] | ||||
| * Don't lock screen after crash if locking is disabled [Adel; #704884] | ||||
| * Improve layout of extension installation dialog [William; #739888] | ||||
| * Fix workspace changes from app picker [Yuki; #737534] | ||||
| * Preload all ibus input sources in user configuration [Takao; #695428] | ||||
| * Properly remove network connections from list [Ryan; #740227] | ||||
| * Support CSS margin property [Carlos; #728437] | ||||
| * Improve handling of BUSY application state [Phillip; #736492] | ||||
| * Fix erroneous week numbers in calendar [Florian; #736722] | ||||
| * Misc. bugfixes and cleanups [Darcy, Yuki, Alexander, Eskild, Bastien, Cosimo, Colin, | ||||
|   Ray; #738725, #739497, #739241, #672500, #739822, #740074, #704163, #740141] | ||||
|  | ||||
| Contributors: | ||||
|   Yuki, Lavi .A, Elad Alfassa, Cosimo Cecchi, Takao Fujiwara, Adel Gadllah, | ||||
|   Eskild Hustvedt, Chris Johns, William Jon McCann, Slavek Kabrda, Ryan Lortie, | ||||
|   Florian Müllner, Bastien Nocera, Christian Persch, Carlos Soriano, | ||||
|   Jasper St. Pierre, Ray Strode, Alexander Tsoy, Daiki Ueno, Colin Walters, | ||||
|   Phillip Wood | ||||
|  | ||||
| Translations: | ||||
|   Jorge Perez Perez [an], Daniel Martinez [an], Daniel Mustieles [es], | ||||
|   Trần Ngọc Quân [vi], Changwoo Ryu [ko], Kjartan Maraas [nb], | ||||
|   Yosef Or Boczko [he], Marek Černocký [cs] | ||||
|  | ||||
| 3.15.1 | ||||
| ====== | ||||
| * Use GResources for theme loading [Cosimo; #736936] | ||||
| * Reset the OSK to primary on monitor changes [Rui; #738536] | ||||
| * Use LC_TIME locale for format string translations [Florian; #738640] | ||||
| * Summarize queued up notifications [Devyani; #702460] | ||||
| * Improve handling of multi-day events [Andreas; #727302] | ||||
| * Support EXTERNAL scroll policy type [Florian; #739379] | ||||
| * Misc. bugfixes [Owen, Rui; #738652, #739252] | ||||
|  | ||||
| Contributors: | ||||
|   Andreas Brauchli, Cosimo Cecchi, Devyani Kota, Rui Matos, Florian Müllner, | ||||
|   Jasper St. Pierre, Owen W. Taylor | ||||
|  | ||||
| Translations: | ||||
|   Bahodir Mansurov [uz@cyrillic] | ||||
|  | ||||
| 3.14.1 | ||||
| ====== | ||||
| * Fix pulse animation for scrolled app folders [Florian; #736885] | ||||
| * Fix updating background on file changes [Owen; #710756] | ||||
| * Obtain keyboard variant from IBus [Jinkyu; #735066] | ||||
| * Implement Ctrl-u / Ctrl-k keybindings in entries [Florian; #737346] | ||||
| * Pass VPN hints to auth dialog [Dan; #737592] | ||||
| * Only allow one screenshot request at a time [Adel; #737456] | ||||
| * Respect disable-save-to-disc lockdown setting [Florian; #737846] | ||||
| * Respect scaling-factor for profile pictures [Darcy; #735419] | ||||
| * Focus login screen after lifting the lock screen shield [Ray; #708105] | ||||
| * Speed up pulse animation for few items [Carlos S.; #737017] | ||||
| * Fix gap between workspace switcher and screen edge [Florian; #728899] | ||||
| * Disable unredirection during recordings [Adel; #738226] | ||||
| * Ensure there's always at least one input source [Rui; #738303] | ||||
| * Restrict width of dash icons' context menus [Adel; #738054] | ||||
| * Misc. bug fixes [Jasper, Florian, Carlos G., Owen; #736999, #737382, #737001, | ||||
|   #738314, #738256, #738147] | ||||
|  | ||||
| Contributors: | ||||
|   Darcy Beurle, Cosimo Cecchi, Adel Gadllah, Carlos Garnacho, Rui Matos, | ||||
|   Florian Müllner, Carlos Soriano, Jasper St. Pierre, Ray Strode, Patrick Ward, | ||||
|   Dan Williams, Owen W. Taylor, Jinkyu Yi | ||||
|  | ||||
| Translations: | ||||
|   Мирослав Николић po/sr, sr@latin.po, Fran Diéguez [gl], Marek Černocký [cs], | ||||
|   Saibal Ray [bn_IN], Rajesh Ranjan [hi], Friedel Wolff [af], | ||||
|   Zhou Fang [zh_CN], Krishnababu Krothapalli [te], Kjartan Maraas [nb], | ||||
|   Rūdolfs Mazurs [lv], Sweta Kothari [gu], Christian Kirbach [de], | ||||
|   Cheng-Chia Tseng [zh_TW], Pedro Albuquerque [pt], Daniel Mustieles [es], | ||||
|   Luca Ferretti [it], Baurzhan Muftakhidinov [kk], Arash Mousavi [fa], | ||||
|   Milo Casagrande [it] | ||||
|  | ||||
| 3.14.0 | ||||
| ====== | ||||
| * Fix exposure of the accessible tree [Alejandro; #736821] | ||||
| * Hide empty app folders in app picker [Florian; #736910] | ||||
|  | ||||
| Contributors: | ||||
|   Florian Müllner, Alejandro Piñeiro | ||||
|  | ||||
| Translations: | ||||
|   Yuri Myasoedov [ru], Pawan Chitrakar [ne], Manoj Kumar Giri [or], | ||||
|   Daniel Mustieles [es], GNOME Translation Robot [de], Rajesh Ranjan [hi], | ||||
|   Shankar Prasad [kn], Kenneth Nielsen [da], Daniel Korostil [uk], | ||||
|   Changwoo Ryu [ko], A S Alam [pa], Tom Tryfonidis [el], Petr Kovar [cs] | ||||
|  | ||||
| 3.13.92 | ||||
| ======= | ||||
| * Fix submenu arrow animations [Hashem; #728927] | ||||
| * Always initialize clutter accessibility [Alejandro; #735908] | ||||
| * Adapt to mutter background changes [Owen; #735638] | ||||
| * Improve handling of outOfDate extensions in prefs [Florian; #736185] | ||||
| * Port offline updates to PackageKit's DBus interface [Kalev; #736337] | ||||
| * location: Translate accuracy levels for geoclue [Zeeshan; #736479] | ||||
| * Implement input source switching [Rui; #736435] | ||||
| * Fix crash when dragging window from workspace switcher [Carlos G.; #735972] | ||||
| * Clean out list of default favorites [Elad; #735682] | ||||
| * Add settings link to location submenu [Florian; #736542] | ||||
| * Fix keynav of message tray menu button [Florian; #707799] | ||||
| * Misc. bug fixes [Carlos G., Florian, Rui; #736110, #736329, #736343, | ||||
|   #735927, #735976] | ||||
|  | ||||
| Contributors: | ||||
|   Elad Alfassa, Zeeshan Ali (Khattak), Michael Catanzaro, Adel Gadllah, | ||||
|   Carlos Garnacho, Kalev Lember, Rui Matos, Florian Müllner, Hashem Nasarat, | ||||
|   Alejandro Piñeiro, Carlos Soriano, Jasper St. Pierre, Owen W. Taylor | ||||
|  | ||||
| Translations: | ||||
|   Piotr Drąg [pl], Changwoo Ryu [ko], Yuri Myasoedov [ru], Zhou Fang [ja], | ||||
|   Peter Mráz [sk], Ville-Pekka Vainio [fi], Sweta Kothari [gu], | ||||
|   Marek Černocký [cs], A S Alam [pa], Christian Kirbach [de], | ||||
|   Alexandre Franke [fr], Aurimas Černius [lt], Khaled Hosny [ar], | ||||
|   Enrico Nicoletto [pt_BR], Andika Triwidada [id], Shantha kumar [ta], | ||||
|   Matej Urbančič [sl], Pawan Chitrakar [ne], Yosef Or Boczko [he], | ||||
|   Balázs Úr [hu], Dušan Kazik [sk], Gil Forcada [ca], | ||||
|   Carles Ferrando [ca@valencia], Nilamdyuti Goswami [as], Ivaylo Valkov [bg], | ||||
|   Sandeep Sheshrao Shedmake [mr], Umarzuki Bin Mochlis Moktar [ms], | ||||
|   Muhammet Kara [tr], Jiro Matsuzawa [ja], Kris Thomsen [da], | ||||
|   Mattias Eriksson [sv] | ||||
|  | ||||
| 3.13.91 | ||||
| ======= | ||||
| * Fix keynav into session menu on login screen [Florian; #735614] | ||||
| * Add gesture to summon message tray [Carlos G.; #735625] | ||||
| * Accept scrolls/swipes either way on the screen shield [Jasper; #734874] | ||||
| * Animate opening the app picker and improve app folder animations | ||||
|   [Carlos S.; #734726] | ||||
| * Animate app icons on launching a new window [Carlos S., Florian; #734726] | ||||
| * Show the on-screen keyboard when touch input is being used [David; #702015] | ||||
| * Misc. bug fixes [Bastien, Owen, Florian, Carlos G.; #735190, #735385, | ||||
|   #735608, #735681] | ||||
|  | ||||
| Contributors: | ||||
|   Carlos Garnacho, David King, Kalev Lember, Florian Müllner, Bastien Nocera, | ||||
|   Carlos Soriano, Jasper St. Pierre, Owen W. Taylor | ||||
|  | ||||
| Translations: | ||||
|   Fran Diéguez [gl], Sweta Kothari [gu], Manoj Kumar Giri [or], | ||||
|   Alain Lojewski [fr], Praveen Illa [te], Arash Mousavi [fa], | ||||
|   Andika Triwidada [id] | ||||
|  | ||||
| 3.13.90 | ||||
| ======= | ||||
| * Make use of GLSL optional [Adel; #733623] | ||||
| * Update on-screen-keyboard position on monitor changes [Cosimo; #733790] | ||||
| * Improve window manager animations [Giovanni; #732857] | ||||
| * Handle touch events [Carlos G.; #733633] | ||||
| * Try to not show "New Window" action for single-window apps | ||||
|   [Giovanni; #722554] | ||||
| * Fix overview exceeding screen size with many apps installed | ||||
|   [Carlos S.; #723496] | ||||
| * Add Software to default favorites [Mathieu; #734406] | ||||
| * Improve app picker <-> desktop transition [Carlos S.; #732901] | ||||
| * Remove <shift>-magic for switcher popups [Christophe; #732296] | ||||
| * Add a special background to use for performance testing [Owen; #734610] | ||||
| * Add support for default disabled search providers [Giovanni; #734110] | ||||
| * Fix portals that don't redirect properly [Giovanni; #733848] | ||||
| * Fix history trimming in chat notifications [Giovanni; #733899] | ||||
| * Try to use default calendar application [Florian; #722333] | ||||
| * Only show location menu when geolocation is in use [Zeeshan; #731122] | ||||
| * Misc. bug fixes and cleanups [Giovanni, Carlos G., Zeeshan, Carlos S., | ||||
|   Cosimo; #711682, #733840, #734483, #734680, #733813, #735062] | ||||
|  | ||||
| Contributors: | ||||
|   Zeeshan Ali (Khattak), Mathieu Bridon, Giovanni Campagna, Cosimo Cecchi, | ||||
|   Piotr Drąg, Christophe Fergeau, Adel Gadllah, Carlos Garnacho, | ||||
|   Florian Müllner, Carlos Soriano, Jasper St. Pierre, Olav Vitters, | ||||
|   Owen W. Taylor | ||||
|  | ||||
| Translations: | ||||
|   Aurimas Černius [lt], MarMav [el], Inaki Larranaga Murgoitio [eu], | ||||
|   Reinout van Schouwen [nl], ngoswami [as], Fabio Tomat [fur], | ||||
|   Chao-Hsiung Liao [zh_HK, zh_TW] | ||||
|  | ||||
| 3.13.4 | ||||
| ====== | ||||
| * Handle portal login requests [Giovanni; #704416] | ||||
| * Scale fonts on wayland on hiDPI devices [Adel; #732537] | ||||
| * Fix default ibus candidate index labels [Rui; #702944] | ||||
| * Add gestures for various system actions [Carlos G.] | ||||
| * Add performance test script for the perf.gnome.org [Owen; #732350] | ||||
| * Use new restart framework to improve restart visuals [Owen; #733026] | ||||
| * Improve keynav in app folder popups [Carlos S.; #731477] | ||||
| * Fix truncation of app search results [Carlos S.; #732416] | ||||
| * Automatically update renamed desktop files in favorites [Kalev; #729429] | ||||
| * Misc. bug fixes and cleanups [Giovanni, Yosef, Owen, Bastien, Javier; | ||||
|   #729823, #726401, #732301, #732348, #732349, #733498, #733540] | ||||
|  | ||||
| Contributors: | ||||
|   Giovanni Campagna, Adel Gadllah, Carlos Garnacho, Javier Hernández, | ||||
|   Kalev Lember, Rui Matos, Florian Müllner, Bastien Nocera, Yosef Or Boczko, | ||||
|   Carlos Soriano, Jasper St. Pierre, Owen W. Taylor | ||||
|  | ||||
| Translations: | ||||
|   Yuri Myasoedov [ru], Daniel Mustieles [es], Fran Diéguez [gl], | ||||
|   Cheng-Chia Tseng [zh_TW], A S Alam [pa], Benjamin Steinwender [de], | ||||
|   Enrico Nicoletto [pt_BR], MarMav [el], Yosef Or Boczko [he], | ||||
|   Kjartan Maraas [nb] | ||||
|  | ||||
| 3.13.3 | ||||
| ====== | ||||
| * Don't allow closing windows with attached modals [Florian; #729886] | ||||
| * Fix self-restarting on OpenBSD [Antoine; #727763] | ||||
| * Improve behavior of window buttons with compositor menus [Florian; #731058] | ||||
| * Work around atspi-related performance regression [Alejandro; #730118] | ||||
| * Misc bug fixes and cleanups [Florian, Lan, Jasper, Christophe, Debarshi, | ||||
|   Zeeshan; #728271, #726460, #703833, #731118, #731220, #695487, #730527, | ||||
|   #728170, #731619, #731738, #731882, #731923] | ||||
|  | ||||
| Contributors: | ||||
|   Zeeshan Ali (Khattak), Christophe Fergeau, Adel Gadllah, Antoine Jacoutot, | ||||
|   Ting-Wei Lan, Florian Müllner, Alejandro Piñeiro, Debarshi Ray, | ||||
|   Carlos Soriano, Jasper St. Pierre, Wim Taymans, Rico Tzschichholz | ||||
|  | ||||
| Translations: | ||||
|   Philip Withnall [en_GB], Milo Casagrande [it], Aurimas Černius [lt], | ||||
|   Enrico Nicoletto [pt_BR], Kjartan Maraas [nb], Balázs Meskó [hu], | ||||
|   Muhammet Kara [tr], Daniel Mustieles [es], Yosef Or Boczko [he], | ||||
|   Matej Urbančič [sl], Mattias Eriksson [sv] | ||||
|  | ||||
| 3.13.2 | ||||
| ====== | ||||
| * Make airplane mode menu insensitive in lock screen [Giovanni; #729224] | ||||
|   | ||||
| @@ -14,7 +14,7 @@ PKG_NAME="gnome-shell" | ||||
| } | ||||
|  | ||||
| # Fetch submodules if needed | ||||
| if test ! -f src/gvc/Makefile.am; | ||||
| if test ! -f src/gvc/Makefile.am || test ! -f data/theme/gnome-shell-sass/COPYING; | ||||
| then | ||||
|   echo "+ Setting up submodules" | ||||
|   git submodule init | ||||
|   | ||||
							
								
								
									
										23
									
								
								configure.ac
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,5 @@ | ||||
| AC_PREREQ(2.63) | ||||
| AC_INIT([gnome-shell],[3.13.2],[https://bugzilla.gnome.org/enter_bug.cgi?product=gnome-shell],[gnome-shell]) | ||||
| AC_INIT([gnome-shell],[3.15.4],[https://bugzilla.gnome.org/enter_bug.cgi?product=gnome-shell],[gnome-shell]) | ||||
|  | ||||
| AC_CONFIG_HEADERS([config.h]) | ||||
| AC_CONFIG_SRCDIR([src/shell-global.c]) | ||||
| @@ -37,7 +37,7 @@ AC_PATH_PROG([XSLTPROC], [xsltproc]) | ||||
| GLIB_GSETTINGS | ||||
|  | ||||
| # Get a value to substitute into gnome-shell.in | ||||
| AM_PATH_PYTHON([2.5]) | ||||
| AM_PATH_PYTHON([3]) | ||||
| AC_SUBST(PYTHON) | ||||
|  | ||||
| # We need at least this, since gst_plugin_register_static() was added | ||||
| @@ -76,11 +76,11 @@ AC_MSG_RESULT($enable_systemd) | ||||
| CLUTTER_MIN_VERSION=1.15.90 | ||||
| GOBJECT_INTROSPECTION_MIN_VERSION=0.10.1 | ||||
| GJS_MIN_VERSION=1.39.0 | ||||
| MUTTER_MIN_VERSION=3.13.2 | ||||
| GTK_MIN_VERSION=3.7.9 | ||||
| MUTTER_MIN_VERSION=3.15.4 | ||||
| GTK_MIN_VERSION=3.15.0 | ||||
| GIO_MIN_VERSION=2.37.0 | ||||
| LIBECAL_MIN_VERSION=3.5.3 | ||||
| LIBEDATASERVER_MIN_VERSION=3.5.3 | ||||
| LIBEDATASERVER_MIN_VERSION=3.13.90 | ||||
| TELEPATHY_GLIB_MIN_VERSION=0.17.5 | ||||
| POLKIT_MIN_VERSION=0.100 | ||||
| STARTUP_NOTIFICATION_MIN_VERSION=0.11 | ||||
| @@ -97,7 +97,6 @@ SHARED_PCS="gio-unix-2.0 >= $GIO_MIN_VERSION | ||||
|             gjs-internals-1.0 >= $GJS_MIN_VERSION | ||||
|             $recorder_modules | ||||
|             gdk-x11-3.0 libsoup-2.4 | ||||
|             xtst | ||||
|             clutter-x11-1.0 >= $CLUTTER_MIN_VERSION | ||||
|             clutter-glx-1.0 >= $CLUTTER_MIN_VERSION | ||||
|             libstartup-notification-1.0 >= $STARTUP_NOTIFICATION_MIN_VERSION | ||||
| @@ -119,7 +118,7 @@ PKG_CHECK_MODULES(SHELL_PERF_HELPER, gtk+-3.0 gio-2.0) | ||||
| PKG_CHECK_MODULES(SHELL_HOTPLUG_SNIFFER, gio-2.0 gdk-pixbuf-2.0) | ||||
| PKG_CHECK_MODULES(TRAY, gtk+-3.0) | ||||
| PKG_CHECK_MODULES(GVC, libpulse >= $PULSE_MIN_VERS libpulse-mainloop-glib gobject-2.0) | ||||
| PKG_CHECK_MODULES(DESKTOP_SCHEMAS, gsettings-desktop-schemas >= 3.7.4) | ||||
| PKG_CHECK_MODULES(DESKTOP_SCHEMAS, gsettings-desktop-schemas >= 3.13.1) | ||||
| PKG_CHECK_MODULES(CARIBOU, caribou-1.0 >= 0.4.8) | ||||
|  | ||||
| AC_ARG_ENABLE(browser-plugin, | ||||
| @@ -232,9 +231,17 @@ esac | ||||
| AM_CFLAGS="$AM_CFLAGS $WARN_CFLAGS" | ||||
| AC_SUBST(AM_CFLAGS) | ||||
|  | ||||
| BROWSER_PLUGIN_DIR="${BROWSER_PLUGIN_DIR:-"\${libdir}/mozilla/plugins"}" | ||||
| if test -z "${BROWSER_PLUGIN_DIR}"; then | ||||
|   BROWSER_PLUGIN_DIR="\${libdir}/mozilla/plugins" | ||||
| fi | ||||
| AC_ARG_VAR([BROWSER_PLUGIN_DIR],[Where to install the plugin to]) | ||||
|  | ||||
| AC_ARG_VAR([GDBUS_CODEGEN],[the gdbus-codegen programme]) | ||||
| AC_PATH_PROG([GDBUS_CODEGEN],[gdbus-codegen],[]) | ||||
| if test -z "$GDBUS_CODEGEN"; then | ||||
|   AC_MSG_ERROR([gdbus-codegen not found]) | ||||
| fi | ||||
|  | ||||
| AC_CONFIG_FILES([ | ||||
|   Makefile | ||||
|   data/Makefile | ||||
|   | ||||
| @@ -1,6 +1,24 @@ | ||||
| CLEANFILES = | ||||
|  | ||||
| desktopdir=$(datadir)/applications | ||||
| desktop_DATA = gnome-shell.desktop gnome-shell-wayland.desktop  gnome-shell-extension-prefs.desktop | ||||
|  | ||||
| if HAVE_NETWORKMANAGER | ||||
| desktop_DATA += org.gnome.Shell.PortalHelper.desktop | ||||
|  | ||||
| servicedir = $(datadir)/dbus-1/services | ||||
| service_DATA = org.gnome.Shell.PortalHelper.service | ||||
|  | ||||
| CLEANFILES += \ | ||||
| 	org.gnome.Shell.PortalHelper.service \ | ||||
| 	org.gnome.Shell.PortalHelper.desktop | ||||
|  | ||||
| endif | ||||
|  | ||||
| %.service: %.service.in | ||||
| 	$(AM_V_GEN) sed -e "s|@libexecdir[@]|$(libexecdir)|" \ | ||||
| 	    $< > $@ || rm $@ | ||||
|  | ||||
| # We substitute in bindir so it works as an autostart | ||||
| # file when built in a non-system prefix | ||||
| %.desktop.in:%.desktop.in.in | ||||
| @@ -17,45 +35,18 @@ introspection_DATA =				\ | ||||
| 	org.gnome.ShellSearchProvider.xml	\ | ||||
| 	org.gnome.ShellSearchProvider2.xml | ||||
|  | ||||
| themedir = $(pkgdatadir)/theme | ||||
| dist_theme_DATA =				\ | ||||
| 	theme/calendar-arrow-left.svg		\ | ||||
| 	theme/calendar-arrow-right.svg		\ | ||||
| 	theme/calendar-today.svg		\ | ||||
| 	theme/checkbox-focused.svg		\ | ||||
| 	theme/checkbox-off-focused.svg		\ | ||||
| 	theme/checkbox-off.svg			\ | ||||
| 	theme/checkbox.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/logged-in-indicator.svg		\ | ||||
| 	theme/menu-arrow-symbolic.svg		    \ | ||||
| 	theme/message-tray-background.png	\ | ||||
| 	theme/more-results.svg			\ | ||||
| 	theme/noise-texture.png			\ | ||||
| 	theme/page-indicator-active.svg		\ | ||||
| 	theme/page-indicator-inactive.svg	\ | ||||
| 	theme/page-indicator-checked.svg	\ | ||||
| 	theme/page-indicator-hover.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/source-button-border.svg		\ | ||||
| 	theme/summary-counter.svg		\ | ||||
| 	theme/toggle-off-us.svg			\ | ||||
| 	theme/toggle-off-intl.svg		\ | ||||
| 	theme/toggle-on-us.svg			\ | ||||
| 	theme/toggle-on-intl.svg		\ | ||||
| 	theme/ws-switch-arrow-up.png		\ | ||||
| 	theme/ws-switch-arrow-down.png | ||||
| resource_files = $(shell $(GLIB_COMPILE_RESOURCES) --sourcedir=$(srcdir)/theme --generate-dependencies $(srcdir)/gnome-shell-theme.gresource.xml) | ||||
| gnome-shell-theme.gresource: gnome-shell-theme.gresource.xml $(resource_files) | ||||
| 	$(AM_V_GEN) $(GLIB_COMPILE_RESOURCES) --target=$@ --sourcedir=$(srcdir)/theme $< | ||||
| resourcedir = $(pkgdatadir) | ||||
| resource_DATA = gnome-shell-theme.gresource | ||||
|  | ||||
| backgrounddir = $(pkgdatadir) | ||||
| background_DATA = perf-background.xml | ||||
|  | ||||
| perf-background.xml: perf-background.xml.in | ||||
| 	$(AM_V_GEN) sed -e "s|@datadir[@]|$(datadir)|" \ | ||||
| 	    $< > $@ || rm $@ | ||||
|  | ||||
| keysdir = @GNOME_KEYBINDINGS_KEYSDIR@ | ||||
| keys_in_files = 50-gnome-shell-system.xml.in | ||||
| @@ -89,15 +80,22 @@ EXTRA_DIST =						\ | ||||
| 	$(menu_DATA)					\ | ||||
| 	$(convert_DATA)					\ | ||||
| 	$(keys_in_files)				\ | ||||
| 	org.gnome.shell.gschema.xml.in.in | ||||
| 	perf-background.xml.in				\ | ||||
| 	org.gnome.Shell.PortalHelper.desktop.in		\ | ||||
| 	org.gnome.Shell.PortalHelper.service.in		\ | ||||
| 	org.gnome.shell.gschema.xml.in.in		\ | ||||
| 	gnome-shell-theme.gresource.xml 		\ | ||||
| 	$(resource_files) | ||||
|  | ||||
| CLEANFILES =						\ | ||||
| CLEANFILES +=						\ | ||||
| 	gnome-shell.desktop.in				\ | ||||
| 	gnome-shell-wayland.desktop.in			\ | ||||
| 	gnome-shell-extension-prefs.in			\ | ||||
| 	$(desktop_DATA)					\ | ||||
| 	$(keys_DATA)					\ | ||||
| 	$(gsettings_SCHEMAS)				\ | ||||
| 	perf-background.xml				\ | ||||
| 	gschemas.compiled				\ | ||||
| 	org.gnome.shell.gschema.valid			\ | ||||
| 	org.gnome.shell.gschema.xml.in | ||||
| 	org.gnome.shell.gschema.xml.in			\ | ||||
| 	gnome-shell-theme.gresource | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| [org.gnome.shell.overrides] | ||||
| attach-modal-dialogs = /desktop/gnome/shell/windows/attach_modal_dialogs | ||||
| button-layout = /desktop/gnome/shell/windows/button_layout | ||||
| edge-tiling = /desktop/gnome/shell/windows/edge_tiling | ||||
| workspaces-only-on-primary = /desktop/gnome/shell/windows/workspaces_only_on_primary | ||||
|   | ||||
							
								
								
									
										38
									
								
								data/gnome-shell-theme.gresource.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <gresources> | ||||
|   <gresource prefix="/org/gnome/shell/theme"> | ||||
|     <file>calendar-arrow-left.svg</file> | ||||
|     <file>calendar-arrow-right.svg</file> | ||||
|     <file>calendar-today.svg</file> | ||||
|     <file>checkbox-focused.svg</file> | ||||
|     <file>checkbox-off-focused.svg</file> | ||||
|     <file>checkbox-off.svg</file> | ||||
|     <file>checkbox.svg</file> | ||||
|     <file>close-window.svg</file> | ||||
|     <file>close.svg</file> | ||||
|     <file>corner-ripple-ltr.png</file> | ||||
|     <file>corner-ripple-rtl.png</file> | ||||
|     <file>dash-placeholder.svg</file> | ||||
|     <file>filter-selected-ltr.svg</file> | ||||
|     <file>filter-selected-rtl.svg</file> | ||||
|     <file>gnome-shell.css</file> | ||||
|     <file>gnome-shell-high-contrast.css</file> | ||||
|     <file>logged-in-indicator.svg</file> | ||||
|     <file>more-results.svg</file> | ||||
|     <file>noise-texture.png</file> | ||||
|     <file>page-indicator-active.svg</file> | ||||
|     <file>page-indicator-inactive.svg</file> | ||||
|     <file>page-indicator-checked.svg</file> | ||||
|     <file>page-indicator-hover.svg</file> | ||||
|     <file>process-working.svg</file> | ||||
|     <file>running-indicator.svg</file> | ||||
|     <file>source-button-border.svg</file> | ||||
|     <file>summary-counter.svg</file> | ||||
|     <file>toggle-off-us.svg</file> | ||||
|     <file>toggle-off-intl.svg</file> | ||||
|     <file>toggle-on-us.svg</file> | ||||
|     <file>toggle-on-intl.svg</file> | ||||
|     <file>ws-switch-arrow-up.png</file> | ||||
|     <file>ws-switch-arrow-down.png</file> | ||||
|   </gresource> | ||||
| </gresources> | ||||
							
								
								
									
										9
									
								
								data/org.gnome.Shell.PortalHelper.desktop.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| [Desktop Entry] | ||||
| _Name=Network Login | ||||
| Type=Application | ||||
| Exec=gapplication launch org.gnome.Shell.PortalHelper | ||||
| DBusActivatable=true | ||||
| NoDisplay=true | ||||
| Icon=network-workgroup | ||||
| StartupNotify=true | ||||
| OnlyShowIn=GNOME; | ||||
							
								
								
									
										3
									
								
								data/org.gnome.Shell.PortalHelper.service.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| [D-BUS Service] | ||||
| Name=org.gnome.Shell.PortalHelper | ||||
| Exec=@libexecdir@/gnome-shell-portal-helper | ||||
| @@ -38,7 +38,6 @@ | ||||
|     <method name="Screencast"> | ||||
|       <arg type="s" direction="in" name="file_template"/> | ||||
|       <arg type="a{sv}" direction="in" name="options"/> | ||||
|       <arg type="b" direction="in" name="flash"/> | ||||
|       <arg type="b" direction="out" name="success"/> | ||||
|       <arg type="s" direction="out" name="filename_used"/> | ||||
|     </method> | ||||
|   | ||||
| @@ -31,7 +31,7 @@ | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="favorite-apps" type="as"> | ||||
|       <default>[ 'epiphany.desktop', 'evolution.desktop', 'empathy.desktop', 'rhythmbox.desktop', 'shotwell.desktop', 'libreoffice-writer.desktop', 'org.gnome.Nautilus.desktop', 'org.gnome.Documents.desktop' ]</default> | ||||
|       <default>[ 'epiphany.desktop', 'evolution.desktop', 'rhythmbox.desktop', 'shotwell.desktop', 'org.gnome.Nautilus.desktop', 'org.gnome.Software.desktop' ]</default> | ||||
|       <_summary>List of desktop file IDs for favorite applications</_summary> | ||||
|       <_description> | ||||
|         The applications corresponding to these identifiers | ||||
| @@ -40,10 +40,10 @@ | ||||
|     </key> | ||||
|     <key name="app-picker-view" type="u"> | ||||
|       <default>0</default> | ||||
|       <summary>App Picker View</summary> | ||||
|       <description> | ||||
|       <_summary>App Picker View</_summary> | ||||
|       <_description> | ||||
|         Index of the currently selected view in the application picker. | ||||
|       </description> | ||||
|       </_description> | ||||
|     </key> | ||||
|     <key name="command-history" type="as"> | ||||
|       <default>[]</default> | ||||
| @@ -74,7 +74,6 @@ | ||||
|     <child name="calendar" schema="org.gnome.shell.calendar"/> | ||||
|     <child name="keybindings" schema="org.gnome.shell.keybindings"/> | ||||
|     <child name="keyboard" schema="org.gnome.shell.keyboard"/> | ||||
|     <child name="location" schema="org.gnome.shell.location"/> | ||||
|   </schema> | ||||
|  | ||||
|   <schema id="org.gnome.shell.calendar" path="/org/gnome/shell/calendar/" | ||||
| @@ -128,8 +127,8 @@ | ||||
|     </key> | ||||
|     <key name="pause-resume-tweens" type="as"> | ||||
|       <default>[]</default> | ||||
|       <summary>Keybinding that pauses and resumes all running tweens, for debugging purposes</summary> | ||||
|       <description></description> | ||||
|       <_summary>Keybinding that pauses and resumes all running tweens, for debugging purposes</_summary> | ||||
|       <_description></_description> | ||||
|     </key> | ||||
|   </schema> | ||||
|  | ||||
| @@ -144,42 +143,16 @@ | ||||
|     </key> | ||||
|   </schema> | ||||
|  | ||||
|   <enum id="org.gnome.shell.geoclue.AccuracyLevel"> | ||||
|     <value value="0" nick="off"/> | ||||
|     <value value="1" nick="country"/> | ||||
|     <value value="4" nick="city"/> | ||||
|     <value value="5" nick="neighborhood"/> | ||||
|     <value value="6" nick="street"/> | ||||
|     <value value="8" nick="exact"/> | ||||
|   </enum> | ||||
|   <schema id="org.gnome.shell.location" | ||||
|           path="/org/gnome/shell/location/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key name="max-accuracy-level" enum="org.gnome.shell.geoclue.AccuracyLevel"> | ||||
|       <default>'exact'</default> | ||||
|       <_summary>The maximum accuracy level of location.</_summary> | ||||
|       <_description> | ||||
|         Configures the maximum level of location accuracy applications are | ||||
|         allowed to see. Valid options are 'off' (disable location tracking), | ||||
|         'country', 'city', 'neighborhood', 'street', and 'exact' (typically | ||||
|         requires GPS receiver). Please keep in mind that this only controls | ||||
|         what GeoClue will allow applications to see and they can find user's | ||||
|         location on their own using network resources (albeit with street-level | ||||
|         accuracy at best). | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
|  | ||||
|   <schema id="org.gnome.shell.app-switcher" | ||||
|           path="/org/gnome/shell/app-switcher/" | ||||
|           gettext-domain="@GETTEXT_PACKAGE@"> | ||||
|     <key type="b" name="current-workspace-only"> | ||||
|       <default>false</default> | ||||
|       <summary>Limit switcher to current workspace.</summary> | ||||
|       <description> | ||||
|       <_summary>Limit switcher to current workspace.</_summary> | ||||
|       <_description> | ||||
| 	If true, only applications that have windows on the current workspace are shown in the switcher. | ||||
| 	Otherwise, all applications are included. | ||||
|       </description> | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
|  | ||||
| @@ -202,11 +175,11 @@ | ||||
|     </key> | ||||
|     <key type="b" name="current-workspace-only"> | ||||
|       <default>true</default> | ||||
|       <summary>Limit switcher to current workspace.</summary> | ||||
|       <description> | ||||
|       <_summary>Limit switcher to current workspace.</_summary> | ||||
|       <_description> | ||||
| 	If true, only windows from the current workspace are shown in the switcher. | ||||
| 	Otherwise, all windows are included. | ||||
|       </description> | ||||
|       </_description> | ||||
|     </key> | ||||
|   </schema> | ||||
|  | ||||
|   | ||||
							
								
								
									
										31
									
								
								data/perf-background.xml.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| <!-- With an animated background, performance will differ depending on whether | ||||
|      one layer or two layers are being blended together. This messes up our | ||||
|      benchmarks. We could just benchmark a single image, but since blended | ||||
|      images are present for much of the day with the GNOME default background, | ||||
|      we want to make sure that also performs well; for that reason we ship | ||||
|      an "animated" background that animates super-slowly to use during | ||||
|      performance tests; it will be in the blended state until 2030. --> | ||||
| <background> | ||||
|   <starttime> | ||||
|     <year>1990</year> | ||||
|     <month>1</month> | ||||
|     <day>1</day> | ||||
|     <hour>0</hour> | ||||
|     <minute>00</minute> | ||||
|     <second>00</second> | ||||
|   </starttime> | ||||
|  | ||||
| <!-- One transition that takes 40 years --> | ||||
| <transition type="overlay"> | ||||
| <duration>1261440000.0</duration> | ||||
| <from>@datadir@/backgrounds/gnome/adwaita-morning.jpg</from> | ||||
| <to>@datadir@/backgrounds/gnome/adwaita-day.jpg</to> | ||||
| </transition> | ||||
|  | ||||
| <!-- A single slide doesn't work, so another slide for 1 minute after 40 years --> | ||||
| <static> | ||||
| <duration>60</duration> | ||||
| <file>/usr/share/backgrounds/gnome/Sandstone.jpg</file> | ||||
| </static> | ||||
|  | ||||
| </background> | ||||
							
								
								
									
										3
									
								
								data/theme/HACKING
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| To generate the css files, from the project directory: | ||||
|  | ||||
| sass --sourcemap=none --update . | ||||
| @@ -14,7 +14,7 @@ | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="checkbox.svg"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
| @@ -132,51 +132,54 @@ | ||||
|        xlink:href="#linearGradient10597-5" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        y2="-388.72955" | ||||
|        x2="-93.031357" | ||||
|        y1="-396.34738" | ||||
|        x1="-93.031357" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.685418)" | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5581-5-2-4-6-8-7-35-8" | ||||
|        id="linearGradient11811" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="linearGradient14219-6" | ||||
|        xlink:href="#linearGradient15404-9" | ||||
|        inkscape:collect="always" /> | ||||
|        gradientTransform="matrix(3.0317573,0,0,1.0053174,-102.66338,-0.82153381)" | ||||
|        x1="63.568954" | ||||
|        y1="127.16142" | ||||
|        x2="63.568954" | ||||
|        y2="152.6618" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient15404-9" | ||||
|        inkscape:collect="always"> | ||||
|        id="linearGradient5581-5-2-4-6-8-7-35-8"> | ||||
|       <stop | ||||
|          id="stop15406-6" | ||||
|          id="stop5583-0-92-8-0-7-6-5-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#515151;stop-opacity:1" /> | ||||
|          style="stop-color:#454c4c;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          id="stop15408-7" | ||||
|          style="stop-color:#393f3f;stop-opacity:1;" | ||||
|          offset="0.40000001" | ||||
|          id="stop5585-4-7-2-7-9-9-92-0" /> | ||||
|       <stop | ||||
|          id="stop5587-6-7-2-0-3-1-21-5" | ||||
|          offset="1" | ||||
|          style="stop-color:#292929;stop-opacity:1" /> | ||||
|          style="stop-color:#2d3232;stop-opacity:1;" /> | ||||
|     </linearGradient> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      pagecolor="#a2a2a2" | ||||
|      bordercolor="#2d2d2d" | ||||
|      borderopacity="1" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="71.516955" | ||||
|      inkscape:cy="5.8710559" | ||||
|      inkscape:zoom="22.627417" | ||||
|      inkscape:cx="9.6447897" | ||||
|      inkscape:cy="12.591409" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1412" | ||||
|      inkscape:window-height="1067" | ||||
|      inkscape:window-x="2635" | ||||
|      inkscape:window-y="226" | ||||
|      inkscape:window-x="184" | ||||
|      inkscape:window-y="233" | ||||
|      inkscape:window-maximized="0" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|      showborder="true"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid14843" | ||||
| @@ -203,87 +206,56 @@ | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|     <g | ||||
|        transform="matrix(0.80230061,0,0,0.80230061,-87.624044,-453.10297)" | ||||
|        id="g14586-0" | ||||
|        style="stroke-width:2.3714385;stroke-miterlimit:4;stroke-dasharray:none"> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="matrix(1.9969286,0,0,1.9969286,-397.05491,877.00482)" | ||||
|          id="g15291-9-6" | ||||
|          style="stroke-width:1.18754292;stroke-miterlimit:4;stroke-dasharray:none;display:inline;enable-background:new"> | ||||
|         <g | ||||
|            transform="translate(877.50354,-102.83507)" | ||||
|            id="g16853-4-9" | ||||
|            style="stroke-width:1.18754292;stroke-miterlimit:4;stroke-dasharray:none;enable-background:new"> | ||||
|           <rect | ||||
|              transform="scale(1,-1)" | ||||
|              style="color:#000000;fill:url(#linearGradient14219-6);fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:1.24833274;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:new" | ||||
|              id="rect6506-6" | ||||
|              width="11.281681" | ||||
|              height="11.26221" | ||||
|              x="-409.59354" | ||||
|              y="-284.40115" | ||||
|              rx="1.0052766" | ||||
|              ry="1.0052764" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="translate(343.99999,987.99997)" | ||||
|          id="g5886-5" | ||||
|          style="stroke-width:2.3714385;stroke-miterlimit:4;stroke-dasharray:none;display:inline;enable-background:new" /> | ||||
|     </g> | ||||
|     <g | ||||
|        transform="matrix(0.84337,0,0,0.84337,-110.16632,-503.56182)" | ||||
|        id="g14586"> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="matrix(1.9969286,0,0,1.9969286,-397.05491,877.00482)" | ||||
|          id="g15291-9" | ||||
|          style="display:inline;enable-background:new"> | ||||
|         <g | ||||
|            transform="translate(877.50354,-102.83507)" | ||||
|            id="g16853-4" | ||||
|            style="enable-background:new" /> | ||||
|       </g> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="translate(343.99999,987.99997)" | ||||
|          id="g5886" | ||||
|          style="display:inline;enable-background:new"> | ||||
|         <path | ||||
|            style="fill:none;stroke:url(#linearGradient5891-0-4);stroke-width:7.11431503;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|            d="m 198.5,240 5.25,5.25 13.98616,-14.43081" | ||||
|            id="path5835" | ||||
|            inkscape:path-effect="#path-effect5837-4-6" | ||||
|            inkscape:original-d="m 198.5,240 5.25,5.25 13.98616,-14.43081" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="ccc" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            inkscape:original-d="m 198.5,240 5.25,5.25 13.91205,-14.31964" | ||||
|            inkscape:path-effect="#path-effect5837-4-6" | ||||
|            id="path5880" | ||||
|            d="m 198.5,240 5.25,5.25 13.91205,-14.31964" | ||||
|            style="fill:none;stroke:#4787c8;stroke-width:3.55715752;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|            sodipodi:nodetypes="ccc" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#7ea7d3;stroke-width:1.18571913px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" | ||||
|            d="m 197.45937,240.47455 c -0.17828,-0.29362 -0.20087,-0.67548 -0.0603,-0.98892 0.14055,-0.31344 0.43739,-0.54812 0.77144,-0.62817 0.33405,-0.08 0.69314,-0.01 0.99635,0.15175 0.30321,0.16144 0.55146,0.40727 0.79165,0.65284 l 3.66429,3.74643 12.87946,-12.98973 c 0.20796,-0.20974 0.42306,-0.41969 0.68548,-0.55522 0.26242,-0.13553 0.57293,-0.19052 0.85827,-0.11426 0.14267,0.0381 0.27708,0.10787 0.38874,0.20452 0.11167,0.0966 0.20021,0.22004 0.25479,0.35726 0.0546,0.13722 0.075,0.28793 0.0585,0.43468 -0.0165,0.14674 -0.07,0.28919 -0.15422,0.41052" | ||||
|            id="path5882" | ||||
|            inkscape:path-effect="#path-effect5884-4-7" | ||||
|            inkscape:original-d="m 197.45937,240.47455 c 0.65604,-0.56057 2.02485,-1.34847 2.49911,-0.8125 l 3.66429,3.74643 12.87946,-12.98973 c 0.6875,-0.6875 2.09152,0.7375 2.09152,0.7375" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="csccc" /> | ||||
|       </g> | ||||
|        style="display:inline" | ||||
|        id="use5671" | ||||
|        transform="matrix(1.3594109,0,0,1.3564242,319.2059,481.99447)"> | ||||
|       <rect | ||||
|          transform="matrix(0.47304779,0,0,0.4807373,-6.3607039,-29.396216)" | ||||
|          rx="4.4136767" | ||||
|          y="125.3458" | ||||
|          x="50.440369" | ||||
|          height="29.154205" | ||||
|          width="29.559635" | ||||
|          id="rect11803" | ||||
|          style="color:#000000;fill:url(#linearGradient11811);fill-opacity:1;stroke:#3465a4;stroke-width:1.54426003000000001;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          ry="4.4233952" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11809" | ||||
|          d="m 17.87105,33.844107 0,-0.773112 c 0,-1.031264 0.807171,-1.836142 1.811982,-1.836142 l 9.612456,0 c 1.004811,0 1.787822,0.804878 1.787822,1.836142 l 0,0.773112 c 0,-1.031264 -0.783011,-1.836142 -1.787822,-1.836142 l -9.612456,0 c -1.004811,0 -1.811982,0.804878 -1.811982,1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|       <path | ||||
|          sodipodi:nodetypes="csssscssc" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          d="m 17.87105,41.158551 0,0.773112 c 0,1.031264 0.807171,1.836142 1.811982,1.836142 l 9.612456,0 c 1.004811,0 1.787822,-0.804878 1.787822,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          id="path11867" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11869" | ||||
|          d="m 17.87105,41.895784 0,0.773112 c 0,1.031264 0.644622,1.836142 1.649433,1.836142 l 10.067593,0 c 1.004811,0 1.495234,-0.804878 1.495234,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#000000;fill-opacity:0.85253451;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|     </g> | ||||
|     <rect | ||||
|        style="color:#000000;fill:none;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect17347" | ||||
|        width="21.943846" | ||||
|        height="21.943846" | ||||
|        x="342.29913" | ||||
|        y="521.58435" /> | ||||
|     <path | ||||
|        inkscape:connector-curvature="0" | ||||
|        style="opacity:0.8;color:#000000;fill:#000000;fill-opacity:1;stroke:none;stroke-width:0.99999994;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        d="m 359.97505,524.8252 -7.88606,7.71465 -2.57155,-2.57155 -2.91442,-0.0427 0,2.35727 4.02875,3.98587 c 0.80342,0.80309 2.111,0.80309 2.91442,0 l 8.18609,-8.22894 0,-0.38573 c 0,-1.24128 0.19944,-1.76801 -0.82915,-2.29836 z" | ||||
|        id="rect5147-9-1-5-7-6-5-8-7" | ||||
|        sodipodi:nodetypes="ccccccccscc" /> | ||||
|     <path | ||||
|        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;baseline-shift:baseline;color:#000000;fill:#eeeeec;fill-opacity:1;stroke:none;stroke-width:3;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:sans-serif;-inkscape-font-specification:sans-serif" | ||||
|        d="m 361.65223,524.52745 -9.5602,9.36735 -2.56345,-2.56344 -2.92846,-0.0214 0.0153,2.32639 4.02203,4.02206 c 0.80341,0.80309 2.10565,0.80309 2.90906,0 l 10.95049,-11.05765 0.003,-2.1502 z" | ||||
|        id="path12830-4-17-0" | ||||
|        inkscape:connector-curvature="0" | ||||
|        sodipodi:nodetypes="cccccccccc" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 12 KiB | 
| @@ -14,22 +14,10 @@ | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="checkbox-off.svg"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
|     <linearGradient | ||||
|        id="linearGradient15404" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop15406" | ||||
|          offset="0" | ||||
|          style="stop-color:#515151;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop15408" | ||||
|          offset="1" | ||||
|          style="stop-color:#292929;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
| @@ -56,16 +44,6 @@ | ||||
|        effect="spiro" | ||||
|        id="path-effect5884-4-7" | ||||
|        is_visible="true" /> | ||||
|     <linearGradient | ||||
|        y2="-388.72955" | ||||
|        x2="-93.031357" | ||||
|        y1="-396.34738" | ||||
|        x1="-93.031357" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.685418)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="linearGradient14219" | ||||
|        xlink:href="#linearGradient15404" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient10013-4-63-6"> | ||||
| @@ -110,30 +88,55 @@ | ||||
|        id="linearGradient15376" | ||||
|        xlink:href="#linearGradient10597-5" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5581-5-2-4-6-8-7-35-8" | ||||
|        id="linearGradient11811" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(3.0317573,0,0,1.0053174,-102.66338,-0.82153381)" | ||||
|        x1="63.568954" | ||||
|        y1="127.16142" | ||||
|        x2="63.568954" | ||||
|        y2="152.6618" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient5581-5-2-4-6-8-7-35-8"> | ||||
|       <stop | ||||
|          id="stop5583-0-92-8-0-7-6-5-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#454c4c;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          style="stop-color:#393f3f;stop-opacity:1;" | ||||
|          offset="0.40000001" | ||||
|          id="stop5585-4-7-2-7-9-9-92-0" /> | ||||
|       <stop | ||||
|          id="stop5587-6-7-2-0-3-1-21-5" | ||||
|          offset="1" | ||||
|          style="stop-color:#2d3232;stop-opacity:1;" /> | ||||
|     </linearGradient> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      pagecolor="#a2a2a2" | ||||
|      bordercolor="#2d2d2d" | ||||
|      borderopacity="1" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="6.1225392" | ||||
|      inkscape:cy="3.6003241" | ||||
|      inkscape:zoom="5.6568542" | ||||
|      inkscape:cx="19.79113" | ||||
|      inkscape:cy="11.232334" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1412" | ||||
|      inkscape:window-height="1067" | ||||
|      inkscape:window-x="2116" | ||||
|      inkscape:window-y="261" | ||||
|      inkscape:window-x="184" | ||||
|      inkscape:window-y="233" | ||||
|      inkscape:window-maximized="0" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|      showborder="true"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid14843" | ||||
| @@ -160,39 +163,44 @@ | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|     <g | ||||
|        transform="matrix(0.80230061,0,0,0.80230061,-87.624044,-453.10297)" | ||||
|        id="g14586" | ||||
|        style="stroke-width:2.3714385;stroke-miterlimit:4;stroke-dasharray:none"> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="matrix(1.9969286,0,0,1.9969286,-397.05491,877.00482)" | ||||
|          id="g15291-9" | ||||
|          style="stroke-width:1.18754292;stroke-miterlimit:4;stroke-dasharray:none;display:inline;enable-background:new"> | ||||
|         <g | ||||
|            transform="translate(877.50354,-102.83507)" | ||||
|            id="g16853-4" | ||||
|            style="stroke-width:1.18754292;stroke-miterlimit:4;stroke-dasharray:none;enable-background:new"> | ||||
|           <rect | ||||
|              transform="scale(1,-1)" | ||||
|              style="color:#000000;fill:url(#linearGradient14219);fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:1.24833274;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:new" | ||||
|              id="rect6506-6" | ||||
|              width="11.281681" | ||||
|              height="11.26221" | ||||
|              x="-409.59354" | ||||
|              y="-284.40115" | ||||
|              rx="1.0052766" | ||||
|              ry="1.0052764" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="translate(343.99999,987.99997)" | ||||
|          id="g5886" | ||||
|          style="stroke-width:2.3714385;stroke-miterlimit:4;stroke-dasharray:none;display:inline;enable-background:new" /> | ||||
|        style="display:inline" | ||||
|        id="use5671" | ||||
|        transform="matrix(1.3594109,0,0,1.3564242,319.2059,481.99447)"> | ||||
|       <rect | ||||
|          transform="matrix(0.47304779,0,0,0.4807373,-6.3607039,-29.396216)" | ||||
|          rx="4.4136767" | ||||
|          y="125.3458" | ||||
|          x="50.440369" | ||||
|          height="29.154205" | ||||
|          width="29.559635" | ||||
|          id="rect11803" | ||||
|          style="color:#000000;fill:url(#linearGradient11811);fill-opacity:1;stroke:#3465a4;stroke-width:1.54426003000000001;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          ry="4.4233952" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11809" | ||||
|          d="m 17.87105,33.844107 0,-0.773112 c 0,-1.031264 0.807171,-1.836142 1.811982,-1.836142 l 9.612456,0 c 1.004811,0 1.787822,0.804878 1.787822,1.836142 l 0,0.773112 c 0,-1.031264 -0.783011,-1.836142 -1.787822,-1.836142 l -9.612456,0 c -1.004811,0 -1.811982,0.804878 -1.811982,1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|       <path | ||||
|          sodipodi:nodetypes="csssscssc" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          d="m 17.87105,41.158551 0,0.773112 c 0,1.031264 0.807171,1.836142 1.811982,1.836142 l 9.612456,0 c 1.004811,0 1.787822,-0.804878 1.787822,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          id="path11867" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11869" | ||||
|          d="m 17.87105,41.895784 0,0.773112 c 0,1.031264 0.644622,1.836142 1.649433,1.836142 l 10.067593,0 c 1.004811,0 1.495234,-0.804878 1.495234,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#000000;fill-opacity:0.85253451;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|     </g> | ||||
|     <rect | ||||
|        style="color:#000000;fill:none;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect17347" | ||||
|        width="21.943846" | ||||
|        height="21.943846" | ||||
|        x="342.29913" | ||||
|        y="521.58435" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 6.5 KiB After Width: | Height: | Size: 8.9 KiB | 
| @@ -14,22 +14,10 @@ | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="checkbox.svg"> | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="checkbox-focused.svg"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
|     <linearGradient | ||||
|        id="linearGradient15404" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop15406" | ||||
|          offset="0" | ||||
|          style="stop-color:#515151;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop15408" | ||||
|          offset="1" | ||||
|          style="stop-color:#292929;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:perspective | ||||
|        sodipodi:type="inkscape:persp3d" | ||||
|        inkscape:vp_x="0 : 526.18109 : 1" | ||||
| @@ -44,27 +32,6 @@ | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5872-5-1" | ||||
|        id="linearGradient5891-0-4" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        x1="205.84143" | ||||
|        y1="246.7094" | ||||
|        x2="206.74803" | ||||
|        y2="231.24142" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient5872-5-1"> | ||||
|       <stop | ||||
|          style="stop-color:#0b2e52;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop5874-4-4" /> | ||||
|       <stop | ||||
|          style="stop-color:#1862af;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop5876-0-5" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect5837-4-6" | ||||
| @@ -77,16 +44,6 @@ | ||||
|        effect="spiro" | ||||
|        id="path-effect5884-4-7" | ||||
|        is_visible="true" /> | ||||
|     <linearGradient | ||||
|        y2="-388.72955" | ||||
|        x2="-93.031357" | ||||
|        y1="-396.34738" | ||||
|        x1="-93.031357" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.685418)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="linearGradient14219" | ||||
|        xlink:href="#linearGradient15404" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient10013-4-63-6"> | ||||
| @@ -131,30 +88,55 @@ | ||||
|        id="linearGradient15376" | ||||
|        xlink:href="#linearGradient10597-5" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5581-5-2-4-6-8-7-35-8" | ||||
|        id="linearGradient11811" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(3.0317573,0,0,1.0053174,-102.66338,-0.82153381)" | ||||
|        x1="63.568954" | ||||
|        y1="127.16142" | ||||
|        x2="63.568954" | ||||
|        y2="152.6618" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient5581-5-2-4-6-8-7-35-8"> | ||||
|       <stop | ||||
|          id="stop5583-0-92-8-0-7-6-5-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#454c4c;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          style="stop-color:#393f3f;stop-opacity:1;" | ||||
|          offset="0.40000001" | ||||
|          id="stop5585-4-7-2-7-9-9-92-0" /> | ||||
|       <stop | ||||
|          id="stop5587-6-7-2-0-3-1-21-5" | ||||
|          offset="1" | ||||
|          style="stop-color:#2d3232;stop-opacity:1;" /> | ||||
|     </linearGradient> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      pagecolor="#a2a2a2" | ||||
|      bordercolor="#2d2d2d" | ||||
|      borderopacity="1" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="4" | ||||
|      inkscape:cx="71.247925" | ||||
|      inkscape:cy="33.339093" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="9.6447897" | ||||
|      inkscape:cy="12.591409" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1412" | ||||
|      inkscape:window-height="1067" | ||||
|      inkscape:window-x="2116" | ||||
|      inkscape:window-y="261" | ||||
|      inkscape:window-x="184" | ||||
|      inkscape:window-y="233" | ||||
|      inkscape:window-maximized="0" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|      showborder="true"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid14843" | ||||
| @@ -181,38 +163,44 @@ | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|     <g | ||||
|        transform="matrix(0.84337,0,0,0.84337,-110.16632,-503.56182)" | ||||
|        id="g14586"> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="matrix(1.9969286,0,0,1.9969286,-397.05491,877.00482)" | ||||
|          id="g15291-9" | ||||
|          style="display:inline;enable-background:new"> | ||||
|         <g | ||||
|            transform="translate(877.50354,-102.83507)" | ||||
|            id="g16853-4" | ||||
|            style="enable-background:new"> | ||||
|           <rect | ||||
|              transform="scale(1,-1)" | ||||
|              style="color:#000000;fill:url(#linearGradient14219);fill-opacity:1;fill-rule:nonzero;stroke:#868686;stroke-width:0.59377144999999998;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:new" | ||||
|              id="rect6506-6" | ||||
|              width="11.281681" | ||||
|              height="11.26221" | ||||
|              x="-409.59354" | ||||
|              y="-284.40115" | ||||
|              rx="0.95632279" | ||||
|              ry="0.95632273" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="translate(343.99999,987.99997)" | ||||
|          id="g5886" | ||||
|          style="display:inline;enable-background:new" /> | ||||
|        style="display:inline" | ||||
|        id="use5671" | ||||
|        transform="matrix(1.3594109,0,0,1.3564242,319.2059,481.99447)"> | ||||
|       <rect | ||||
|          transform="matrix(0.47304779,0,0,0.4807373,-6.3607039,-29.396216)" | ||||
|          rx="4.4136767" | ||||
|          y="125.3458" | ||||
|          x="50.440369" | ||||
|          height="29.154205" | ||||
|          width="29.559635" | ||||
|          id="rect11803" | ||||
|          style="color:#000000;fill:url(#linearGradient11811);fill-opacity:1;stroke:#1c1f1f;stroke-width:1.54426003;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          ry="4.4233952" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11809" | ||||
|          d="m 17.87105,33.844107 0,-0.773112 c 0,-1.031264 0.807171,-1.836142 1.811982,-1.836142 l 9.612456,0 c 1.004811,0 1.787822,0.804878 1.787822,1.836142 l 0,0.773112 c 0,-1.031264 -0.783011,-1.836142 -1.787822,-1.836142 l -9.612456,0 c -1.004811,0 -1.811982,0.804878 -1.811982,1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|       <path | ||||
|          sodipodi:nodetypes="csssscssc" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          d="m 17.87105,41.158551 0,0.773112 c 0,1.031264 0.807171,1.836142 1.811982,1.836142 l 9.612456,0 c 1.004811,0 1.787822,-0.804878 1.787822,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          id="path11867" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11869" | ||||
|          d="m 17.87105,41.895784 0,0.773112 c 0,1.031264 0.644622,1.836142 1.649433,1.836142 l 10.067593,0 c 1.004811,0 1.495234,-0.804878 1.495234,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#000000;fill-opacity:0.85253451;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|     </g> | ||||
|     <rect | ||||
|        style="color:#000000;fill:none;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect17347" | ||||
|        width="21.943846" | ||||
|        height="21.943846" | ||||
|        x="342.29913" | ||||
|        y="521.58435" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 6.8 KiB After Width: | Height: | Size: 8.9 KiB | 
| @@ -14,8 +14,8 @@ | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    sodipodi:docname="checkbox-focused.svg"> | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="checkbox.svg"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
|     <linearGradient | ||||
| @@ -131,30 +131,55 @@ | ||||
|        id="linearGradient15376" | ||||
|        xlink:href="#linearGradient10597-5" | ||||
|        inkscape:collect="always" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5581-5-2-4-6-8-7-35-8" | ||||
|        id="linearGradient11811" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(3.0317573,0,0,1.0053174,-102.66338,-0.82153381)" | ||||
|        x1="63.568954" | ||||
|        y1="127.16142" | ||||
|        x2="63.568954" | ||||
|        y2="152.6618" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient5581-5-2-4-6-8-7-35-8"> | ||||
|       <stop | ||||
|          id="stop5583-0-92-8-0-7-6-5-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#454c4c;stop-opacity:1;" /> | ||||
|       <stop | ||||
|          style="stop-color:#393f3f;stop-opacity:1;" | ||||
|          offset="0.40000001" | ||||
|          id="stop5585-4-7-2-7-9-9-92-0" /> | ||||
|       <stop | ||||
|          id="stop5587-6-7-2-0-3-1-21-5" | ||||
|          offset="1" | ||||
|          style="stop-color:#2d3232;stop-opacity:1;" /> | ||||
|     </linearGradient> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#000000" | ||||
|      pagecolor="#a2a2a2" | ||||
|      bordercolor="#2d2d2d" | ||||
|      borderopacity="1" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="64.516955" | ||||
|      inkscape:cy="13.871056" | ||||
|      inkscape:cx="-0.17876005" | ||||
|      inkscape:cy="11.944326" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1412" | ||||
|      inkscape:window-height="1067" | ||||
|      inkscape:window-x="2635" | ||||
|      inkscape:window-y="226" | ||||
|      inkscape:window-maximized="0" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1375" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|      showborder="true"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid14843" | ||||
| @@ -171,7 +196,7 @@ | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
| @@ -181,63 +206,56 @@ | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|     <g | ||||
|        transform="matrix(0.84337,0,0,0.84337,-110.16632,-503.56182)" | ||||
|        id="g14586"> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="matrix(1.9969286,0,0,1.9969286,-397.05491,877.00482)" | ||||
|          id="g15291-9" | ||||
|          style="display:inline;enable-background:new"> | ||||
|         <g | ||||
|            transform="translate(877.50354,-102.83507)" | ||||
|            id="g16853-4" | ||||
|            style="enable-background:new"> | ||||
|           <rect | ||||
|              transform="scale(1,-1)" | ||||
|              style="color:#000000;fill:url(#linearGradient14219);fill-opacity:1;fill-rule:nonzero;stroke:#868686;stroke-width:0.59377144999999998;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:new" | ||||
|              id="rect6506-6" | ||||
|              width="11.281681" | ||||
|              height="11.26221" | ||||
|              x="-409.59354" | ||||
|              y="-284.40115" | ||||
|              rx="0.95632279" | ||||
|              ry="0.95632273" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <g | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/jimmac/SparkleShare/gnome-mockups/boxes/interactive/img/checkbox-on.png" | ||||
|          transform="translate(343.99999,987.99997)" | ||||
|          id="g5886" | ||||
|          style="display:inline;enable-background:new"> | ||||
|         <path | ||||
|            style="fill:none;stroke:url(#linearGradient5891-0-4);stroke-width:7.11431503;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|            d="m 198.5,240 5.25,5.25 13.98616,-14.43081" | ||||
|            id="path5835" | ||||
|            inkscape:path-effect="#path-effect5837-4-6" | ||||
|            inkscape:original-d="m 198.5,240 5.25,5.25 13.98616,-14.43081" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="ccc" /> | ||||
|         <path | ||||
|            inkscape:connector-curvature="0" | ||||
|            inkscape:original-d="m 198.5,240 5.25,5.25 13.91205,-14.31964" | ||||
|            inkscape:path-effect="#path-effect5837-4-6" | ||||
|            id="path5880" | ||||
|            d="m 198.5,240 5.25,5.25 13.91205,-14.31964" | ||||
|            style="fill:none;stroke:#4787c8;stroke-width:3.55715752;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" | ||||
|            sodipodi:nodetypes="ccc" /> | ||||
|         <path | ||||
|            style="fill:none;stroke:#7ea7d3;stroke-width:1.18571913px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" | ||||
|            d="m 197.45937,240.47455 c -0.17828,-0.29362 -0.20087,-0.67548 -0.0603,-0.98892 0.14055,-0.31344 0.43739,-0.54812 0.77144,-0.62817 0.33405,-0.08 0.69314,-0.01 0.99635,0.15175 0.30321,0.16144 0.55146,0.40727 0.79165,0.65284 l 3.66429,3.74643 12.87946,-12.98973 c 0.20796,-0.20974 0.42306,-0.41969 0.68548,-0.55522 0.26242,-0.13553 0.57293,-0.19052 0.85827,-0.11426 0.14267,0.0381 0.27708,0.10787 0.38874,0.20452 0.11167,0.0966 0.20021,0.22004 0.25479,0.35726 0.0546,0.13722 0.075,0.28793 0.0585,0.43468 -0.0165,0.14674 -0.07,0.28919 -0.15422,0.41052" | ||||
|            id="path5882" | ||||
|            inkscape:path-effect="#path-effect5884-4-7" | ||||
|            inkscape:original-d="m 197.45937,240.47455 c 0.65604,-0.56057 2.02485,-1.34847 2.49911,-0.8125 l 3.66429,3.74643 12.87946,-12.98973 c 0.6875,-0.6875 2.09152,0.7375 2.09152,0.7375" | ||||
|            inkscape:connector-curvature="0" | ||||
|            sodipodi:nodetypes="csccc" /> | ||||
|       </g> | ||||
|        style="display:inline" | ||||
|        id="use5671" | ||||
|        transform="matrix(1.3594109,0,0,1.3564242,319.2059,481.99447)"> | ||||
|       <rect | ||||
|          transform="matrix(0.47304779,0,0,0.4807373,-6.3607039,-29.396216)" | ||||
|          rx="4.4136767" | ||||
|          y="125.3458" | ||||
|          x="50.440369" | ||||
|          height="29.154205" | ||||
|          width="29.559635" | ||||
|          id="rect11803" | ||||
|          style="color:#000000;fill:url(#linearGradient11811);fill-opacity:1;stroke:#1c1f1f;stroke-width:1.54426003;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|          ry="4.4233952" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11809" | ||||
|          d="m 17.87105,33.844107 0,-0.773112 c 0,-1.031264 0.807171,-1.836142 1.811982,-1.836142 l 9.612456,0 c 1.004811,0 1.787822,0.804878 1.787822,1.836142 l 0,0.773112 c 0,-1.031264 -0.783011,-1.836142 -1.787822,-1.836142 l -9.612456,0 c -1.004811,0 -1.811982,0.804878 -1.811982,1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|       <path | ||||
|          sodipodi:nodetypes="csssscssc" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          d="m 17.87105,41.158551 0,0.773112 c 0,1.031264 0.807171,1.836142 1.811982,1.836142 l 9.612456,0 c 1.004811,0 1.787822,-0.804878 1.787822,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          id="path11867" | ||||
|          inkscape:connector-curvature="0" /> | ||||
|       <path | ||||
|          inkscape:connector-curvature="0" | ||||
|          id="path11869" | ||||
|          d="m 17.87105,41.895784 0,0.773112 c 0,1.031264 0.644622,1.836142 1.649433,1.836142 l 10.067593,0 c 1.004811,0 1.495234,-0.804878 1.495234,-1.836142 l 0,-0.773112 c 0,1.031264 -0.783011,1.836142 -1.787822,1.836142 l -9.612456,0 c -1.004811,0 -1.811982,-0.804878 -1.811982,-1.836142 z" | ||||
|          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;baseline-shift:baseline;opacity:0.15;color:#000000;fill:#000000;fill-opacity:0.85253451;fill-rule:nonzero;stroke:none;stroke-width:1.00000012;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:Sans;-inkscape-font-specification:Sans" | ||||
|          sodipodi:nodetypes="csssscssc" /> | ||||
|     </g> | ||||
|     <rect | ||||
|        style="color:#000000;fill:none;stroke:none;stroke-width:2;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        id="rect17347" | ||||
|        width="21.943846" | ||||
|        height="21.943846" | ||||
|        x="342.29913" | ||||
|        y="521.58435" /> | ||||
|     <path | ||||
|        inkscape:connector-curvature="0" | ||||
|        style="opacity:0.8;color:#000000;fill:#000000;fill-opacity:1;stroke:none;stroke-width:0.99999994;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate" | ||||
|        d="m 359.97505,524.8252 -7.88606,7.71465 -2.57155,-2.57155 -2.91442,-0.0427 0,2.35727 4.02875,3.98587 c 0.80342,0.80309 2.111,0.80309 2.91442,0 l 8.18609,-8.22894 0,-0.38573 c 0,-1.24128 0.19944,-1.76801 -0.82915,-2.29836 z" | ||||
|        id="rect5147-9-1-5-7-6-5-8-7" | ||||
|        sodipodi:nodetypes="ccccccccscc" /> | ||||
|     <path | ||||
|        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;baseline-shift:baseline;color:#000000;fill:#eeeeec;fill-opacity:1;stroke:none;stroke-width:3;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:accumulate;font-family:sans-serif;-inkscape-font-specification:sans-serif" | ||||
|        d="m 361.65223,524.52745 -9.5602,9.36735 -2.56345,-2.56344 -2.92846,-0.0214 0.0153,2.32639 4.02203,4.02206 c 0.80341,0.80309 2.10565,0.80309 2.90906,0 l 10.95049,-11.05765 0.003,-2.1502 z" | ||||
|        id="path12830-4-17-0" | ||||
|        inkscape:connector-curvature="0" | ||||
|        sodipodi:nodetypes="cccccccccc" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 8.8 KiB After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										1678
									
								
								data/theme/gnome-shell-high-contrast.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										8
									
								
								data/theme/gnome-shell-high-contrast.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| @import "gnome-shell-sass/_high-contrast-colors"; //use gtk colors | ||||
| @import "gnome-shell-sass/_drawing"; | ||||
| @import "gnome-shell-sass/_common"; | ||||
|  | ||||
| //force symbolic icons | ||||
| stage { | ||||
|   -st-icon-style: symbolic; | ||||
| } | ||||
							
								
								
									
										1
									
								
								data/theme/gnome-shell-sass
									
									
									
									
									
										Submodule
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										5
									
								
								data/theme/gnome-shell.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| $variant: 'dark'; | ||||
|  | ||||
| @import "gnome-shell-sass/_colors"; //use gtk colors | ||||
| @import "gnome-shell-sass/_drawing"; | ||||
| @import "gnome-shell-sass/_common"; | ||||
| @@ -1,90 +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="svg3863" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.4 r9939" | ||||
|    sodipodi:docname="menu-arrow.svg"> | ||||
|   <defs | ||||
|      id="defs3865" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="15.836083" | ||||
|      inkscape:cx="-3.1641676" | ||||
|      inkscape:cy="11.823817" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      inkscape:grid-bbox="true" | ||||
|      inkscape:document-units="px" | ||||
|      showguides="true" | ||||
|      inkscape:guide-bbox="true" | ||||
|      inkscape:window-width="1366" | ||||
|      inkscape:window-height="702" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:snap-bbox="true"> | ||||
|     <sodipodi:guide | ||||
|        orientation="1,0" | ||||
|        position="15.996443,16.922964" | ||||
|        id="guide3873" /> | ||||
|     <sodipodi:guide | ||||
|        orientation="0,1" | ||||
|        position="28.041217,3.1256134" | ||||
|        id="guide3875" /> | ||||
|     <sodipodi:guide | ||||
|        orientation="0,1" | ||||
|        position="-0.80372916,24.469088" | ||||
|        id="guide3877" /> | ||||
|     <sodipodi:guide | ||||
|        orientation="1,0" | ||||
|        position="3.0363102,34.649657" | ||||
|        id="guide3879" /> | ||||
|     <sodipodi:guide | ||||
|        orientation="1,0" | ||||
|        position="29.023553,28.577037" | ||||
|        id="guide3881" /> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid2988" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata3868"> | ||||
|     <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)"> | ||||
|     <path | ||||
|        style="fill:#ffffff;fill-opacity:1;stroke:none" | ||||
|        d="m 4,23 8,0 -4,5 z" | ||||
|        id="path3883" | ||||
|        inkscape:connector-curvature="0" | ||||
|        sodipodi:nodetypes="cccc" /> | ||||
|   </g> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.5 KiB | 
| Before Width: | Height: | Size: 25 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="17" | ||||
|    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="7" | ||||
|        height="2" | ||||
|        x="5" | ||||
|        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 | 
| @@ -12,7 +12,7 @@ | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    id="svg5369" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48+devel r10053 custom" | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    width="96" | ||||
|    height="48" | ||||
|    sodipodi:docname="process-working.svg" | ||||
| @@ -25,12 +25,597 @@ | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title></dc:title> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <defs | ||||
|      id="defs5373" /> | ||||
|      id="defs5373"> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient35326" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient8231-1-4-4-1"> | ||||
|       <stop | ||||
|          id="stop8233-28-5-27-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop8235-7-3-94-3" /> | ||||
|       <stop | ||||
|          id="stop8237-7-8-20-2" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop8239-2-9-1-9" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient35230" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient5767-6"> | ||||
|       <stop | ||||
|          id="stop5769-0" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop5771-1" /> | ||||
|       <stop | ||||
|          id="stop5773-7" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop5775-8" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10255" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10257"> | ||||
|       <stop | ||||
|          id="stop10259" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10261" /> | ||||
|       <stop | ||||
|          id="stop10263" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10265" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10267" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10269"> | ||||
|       <stop | ||||
|          id="stop10271" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10273" /> | ||||
|       <stop | ||||
|          id="stop10275" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10277" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10279" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10281"> | ||||
|       <stop | ||||
|          id="stop10283" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10285" /> | ||||
|       <stop | ||||
|          id="stop10287" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10289" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10291" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10293"> | ||||
|       <stop | ||||
|          id="stop10295" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10297" /> | ||||
|       <stop | ||||
|          id="stop10299" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10301" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10303" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10305"> | ||||
|       <stop | ||||
|          id="stop10307" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10309" /> | ||||
|       <stop | ||||
|          id="stop10311" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10313" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10315" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10317"> | ||||
|       <stop | ||||
|          id="stop10319" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10321" /> | ||||
|       <stop | ||||
|          id="stop10323" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10325" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10327" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10329"> | ||||
|       <stop | ||||
|          id="stop10331" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10333" /> | ||||
|       <stop | ||||
|          id="stop10335" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10337" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10339" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10341"> | ||||
|       <stop | ||||
|          id="stop10343" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10345" /> | ||||
|       <stop | ||||
|          id="stop10347" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10349" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10351" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10353"> | ||||
|       <stop | ||||
|          id="stop10355" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10357" /> | ||||
|       <stop | ||||
|          id="stop10359" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10361" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10363" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10365"> | ||||
|       <stop | ||||
|          id="stop10367" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10369" /> | ||||
|       <stop | ||||
|          id="stop10371" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10373" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10375" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10377"> | ||||
|       <stop | ||||
|          id="stop10379" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10381" /> | ||||
|       <stop | ||||
|          id="stop10383" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10385" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10387" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10389"> | ||||
|       <stop | ||||
|          id="stop10391" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10393" /> | ||||
|       <stop | ||||
|          id="stop10395" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10397" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10399" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10401"> | ||||
|       <stop | ||||
|          id="stop10403" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10405" /> | ||||
|       <stop | ||||
|          id="stop10407" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10409" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10411" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10413"> | ||||
|       <stop | ||||
|          id="stop10415" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10417" /> | ||||
|       <stop | ||||
|          id="stop10419" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10421" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10423" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10425"> | ||||
|       <stop | ||||
|          id="stop10427" | ||||
|          offset="0" | ||||
|          style="stop-color:#ffffff;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0.15428571" | ||||
|          offset="0.31861392" | ||||
|          id="stop10429" /> | ||||
|       <stop | ||||
|          id="stop10431" | ||||
|          offset="0.54270232" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.33714285" /> | ||||
|       <stop | ||||
|          id="stop10433" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10435" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient10437"> | ||||
|       <stop | ||||
|          id="stop10439" | ||||
|          offset="0" | ||||
|          style="stop-color:#bebebe;stop-opacity:0" /> | ||||
|       <stop | ||||
|          style="stop-color:#ffffff;stop-opacity:0" | ||||
|          offset="0.31861392" | ||||
|          id="stop10441" /> | ||||
|       <stop | ||||
|          id="stop10443" | ||||
|          offset="0.75051737" | ||||
|          style="stop-color:#ffffff;stop-opacity:0.42857143" /> | ||||
|       <stop | ||||
|          id="stop10445" | ||||
|          offset="1" | ||||
|          style="stop-color:#ffffff;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10709" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10711" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient8231-1-4-4-1" | ||||
|        id="radialGradient10713" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(2.229454,-1.2865389,1.2087442,2.0939897,-228.90301,-208.08725)" | ||||
|        cx="-0.067823187" | ||||
|        cy="188.51917" | ||||
|        fx="-0.067823187" | ||||
|        fy="188.51917" | ||||
|        r="27.330345" /> | ||||
|     <radialGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient5767-6" | ||||
|        id="radialGradient10715" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.1252013,-0.60470548,0.56813832,1.0568583,-107.67128,-11.948108)" | ||||
|        cx="0.053942412" | ||||
|        cy="189.15244" | ||||
|        fx="0.053942412" | ||||
|        fy="189.15244" | ||||
|        r="27.330345" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      pagecolor="#808080" | ||||
|      bordercolor="#666666" | ||||
| @@ -46,13 +631,15 @@ | ||||
|      showgrid="true" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:zoom="16" | ||||
|      inkscape:cx="53.997662" | ||||
|      inkscape:cy="22.367695" | ||||
|      inkscape:window-x="1600" | ||||
|      inkscape:zoom="5.6568542" | ||||
|      inkscape:cx="40.82607" | ||||
|      inkscape:cy="30.594699" | ||||
|      inkscape:window-x="2560" | ||||
|      inkscape:window-y="33" | ||||
|      inkscape:window-maximized="0" | ||||
|      inkscape:current-layer="layer2"> | ||||
|      inkscape:current-layer="layer2" | ||||
|      inkscape:snap-bbox="true" | ||||
|      inkscape:snap-nodes="false"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid11933" | ||||
| @@ -128,134 +715,97 @@ | ||||
|      id="layer2" | ||||
|      inkscape:label="spinner"> | ||||
|     <g | ||||
|        transform="matrix(0.28240106,0,0,0.28240106,146.92015,-382.52444)" | ||||
|        id="g10450-5" | ||||
|        transform="matrix(0.43142675,0,0,0.43298814,218.13188,-592.92581)" | ||||
|        id="g10450-5-3" | ||||
|        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" /> | ||||
|          inkscape:export-ydpi="90" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-filename="/home/hbons/Moblin/git/carrick-ng/data/icons/network-connecting.png" | ||||
|          sodipodi:open="true" | ||||
|          sodipodi:end="4.712389" | ||||
|          sodipodi:start="0.23191105" | ||||
|          sodipodi:type="arc" | ||||
|          style="fill:none;stroke:url(#radialGradient10713);stroke-width:12.18051815;stroke-linecap:round;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" | ||||
|          id="path10452-5" | ||||
|          sodipodi:cx="-25.809397" | ||||
|          sodipodi:cy="179.43886" | ||||
|          sodipodi:rx="22.98097" | ||||
|          sodipodi:ry="22.98097" | ||||
|          d="m -3.4436513,184.72075 c -2.9171108,12.35226 -15.2953817,20.00096 -27.6476417,17.08385 -12.35226,-2.91711 -20.00096,-15.29538 -17.083849,-27.64764 2.449452,-10.372 11.708437,-17.69907 22.365746,-17.69907" | ||||
|          transform="matrix(-0.16397381,0.61157081,-0.61162275,-0.16377992,-372.32298,1442.5061)" /> | ||||
|       <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" /> | ||||
|          transform="matrix(-0.63300818,0.01438356,-0.01458424,-0.63300359,-491.4014,1510.996)" | ||||
|          d="m -3.4436513,184.72075 c -2.9171108,12.35226 -15.2953817,20.00096 -27.6476417,17.08385 -12.35226,-2.91711 -20.00096,-15.29538 -17.083849,-27.64764 2.449452,-10.372 11.708437,-17.69907 22.365746,-17.69907" | ||||
|          sodipodi:ry="22.98097" | ||||
|          sodipodi:rx="22.98097" | ||||
|          sodipodi:cy="179.43886" | ||||
|          sodipodi:cx="-25.809397" | ||||
|          id="path10454-7" | ||||
|          style="fill:none;stroke:url(#radialGradient10715);stroke-width:12.18051815;stroke-linecap:butt;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0;display:inline" | ||||
|          sodipodi:type="arc" | ||||
|          sodipodi:start="0.23191105" | ||||
|          sodipodi:end="4.712389" | ||||
|          sodipodi:open="true" | ||||
|          inkscape:export-filename="/home/hbons/Moblin/git/carrick-ng/data/icons/network-connecting.png" | ||||
|          inkscape:export-xdpi="90" | ||||
|          inkscape:export-ydpi="90" /> | ||||
|     </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" /> | ||||
|        xlink:href="#g10450-5-3" | ||||
|        id="use13294" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,35.986458,-4.9737924)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13294" | ||||
|        id="use13314" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,43.036943,-21.933639)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13314" | ||||
|        id="use13334" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,50.085328,-38.904987)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13334" | ||||
|        id="use13354" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,-38.894841,-31.888724)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13354" | ||||
|        id="use13374" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,52.971072,2.0670843)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13374" | ||||
|        id="use13394" | ||||
|        transform="matrix(0.70710678,0.70710679,-0.70710679,0.70710678,60.017834,-14.929741)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|     <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" /> | ||||
|        xlink:href="#use13394" | ||||
|        id="use13414" | ||||
|        transform="matrix(0.86602541,0.50000001,-0.50000001,0.86602541,50.044124,-25.16226)" | ||||
|        width="96" | ||||
|        height="48" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 9.8 KiB After Width: | Height: | Size: 26 KiB | 
| @@ -7,51 +7,127 @@ | ||||
|    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="65" | ||||
|    height="22" | ||||
|    id="svg3273" | ||||
|    id="svg2857" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 14"> | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="toggle-off-us.svg"> | ||||
|   <defs | ||||
|      id="defs3275"> | ||||
|      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="perspective3281" /> | ||||
|        id="perspective2865" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3261" | ||||
|        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" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient76469-7-7-4" | ||||
|        id="linearGradient38024" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0215462,0,0,1.0322581,717.22867,428.68472)" | ||||
|        x1="6" | ||||
|        y1="102.95528" | ||||
|        x2="6" | ||||
|        y2="84.505203" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient76469-7-7-4"> | ||||
|       <stop | ||||
|          style="stop-color:#2e3232;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop76471-7-1-5" /> | ||||
|       <stop | ||||
|          style="stop-color:#3e4545;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop76473-9-0-0" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4" | ||||
|        is_visible="true" /> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4-0" | ||||
|        is_visible="true" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient37802-8" | ||||
|        id="linearGradient12311-3-1-0-5-4" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.071426)" | ||||
|        x1="610.13782" | ||||
|        y1="501.43866" | ||||
|        x2="610.13782" | ||||
|        y2="492.52756" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient37802-8" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop37804-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#2c2c2c;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop37806-8" | ||||
|          offset="1" | ||||
|          style="stop-color:#16191a;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <linearGradient | ||||
|        y2="492.52756" | ||||
|        x2="610.13782" | ||||
|        y1="501.43866" | ||||
|        x1="610.13782" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-900.56122,-423.92857)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="linearGradient13602" | ||||
|        xlink:href="#linearGradient37802-8" | ||||
|        inkscape:collect="always" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="32.000004" | ||||
|      inkscape:cy="10.999997" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="-5.0602834" | ||||
|      inkscape:cy="16.473273" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      inkscape:current-layer="g37994" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1375" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid12954" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata3278"> | ||||
|      id="metadata2862"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
| @@ -66,61 +142,68 @@ | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-343,-521.36218)"> | ||||
|      transform="translate(-444.64286,-781.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" /> | ||||
|        transform="matrix(0.6526046,0,0,0.80554422,99.592644,-636.32172)" | ||||
|        id="g37994"> | ||||
|       <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" /> | ||||
|          id="g37996" | ||||
|          transform="translate(-115,1277)"> | ||||
|         <rect | ||||
|            style="color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964000000001;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;opacity:0.19591837" | ||||
|            id="rect13475" | ||||
|            width="98" | ||||
|            height="25" | ||||
|            x="644.5" | ||||
|            y="484.61118" | ||||
|            rx="4.7429576" | ||||
|            ry="3.8424656" /> | ||||
|         <rect | ||||
|            ry="3.8424656" | ||||
|            rx="4.7429576" | ||||
|            y="483.5" | ||||
|            x="644.5" | ||||
|            height="25" | ||||
|            width="98" | ||||
|            id="rect38000" | ||||
|            style="color:#000000;fill:url(#linearGradient12311-3-1-0-5-4);fill-opacity:1;fill-rule:nonzero;stroke:#16191a;stroke-width:1.37920942;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:new" /> | ||||
|       </g> | ||||
|       <g | ||||
|          transform="translate(-49.946213,-1.890275)" | ||||
|          id="g38002"> | ||||
|         <g | ||||
|            transform="translate(-115,1247)" | ||||
|            style="display:inline" | ||||
|            id="g38004"> | ||||
|           <rect | ||||
|              ry="3.7972314" | ||||
|              rx="4.6871223" | ||||
|              y="515.5" | ||||
|              x="694.53046" | ||||
|              height="25" | ||||
|              width="45.969578" | ||||
|              id="rect38006" | ||||
|              style="color:#000000;fill:url(#linearGradient38024);fill-opacity:1;fill-rule:nonzero;stroke:#1f2020;stroke-width:1.37920964;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" /> | ||||
|           <path | ||||
|              sodipodi:nodetypes="cc" | ||||
|              style="opacity:0.1;color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964px;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" | ||||
|              d="m 699.09675,516.7365 36.86904,0" | ||||
|              id="path38016" | ||||
|              inkscape:path-effect="#path-effect77541-4" | ||||
|              inkscape:original-d="m 699.09675,516.7365 36.86904,0" | ||||
|              inkscape:connector-curvature="0" /> | ||||
|         </g> | ||||
|       </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)" /> | ||||
|          style="color:#000000;fill:none;stroke:#ffffff;stroke-width:2.15627193;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:new" | ||||
|          id="path13479" | ||||
|          sodipodi:cx="16.4375" | ||||
|          sodipodi:cy="10.8125" | ||||
|          sodipodi:rx="4.3125" | ||||
|          sodipodi:ry="4.3125" | ||||
|          d="m 20.75,10.8125 a 4.3125,4.3125 0 1 1 -8.625,0 4.3125,4.3125 0 1 1 8.625,0 z" | ||||
|          transform="matrix(1.4212691,0,0,1.1514287,577.38488,1761.1138)" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 4.7 KiB After Width: | Height: | Size: 7.5 KiB | 
| @@ -7,51 +7,171 @@ | ||||
|    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="65" | ||||
|    height="22" | ||||
|    id="svg3012" | ||||
|    id="svg2857" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.47 r22583" | ||||
|    sodipodi:docname="New document 6"> | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="toggle-on-intl.svg"> | ||||
|   <defs | ||||
|      id="defs3014"> | ||||
|      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="perspective3020" /> | ||||
|        id="perspective2865" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective2997" | ||||
|        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" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient76469-7-7-4" | ||||
|        id="linearGradient38024" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0215462,0,0,1.0322581,717.22867,428.68472)" | ||||
|        x1="6" | ||||
|        y1="102.95528" | ||||
|        x2="6" | ||||
|        y2="84.505203" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient76469-7-7-4"> | ||||
|       <stop | ||||
|          style="stop-color:#2e3232;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop76471-7-1-5" /> | ||||
|       <stop | ||||
|          style="stop-color:#3e4545;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop76473-9-0-0" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4" | ||||
|        is_visible="true" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient37802" | ||||
|        id="linearGradient12311-3-1-0-5" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.071426)" | ||||
|        x1="610.13782" | ||||
|        y1="501.43866" | ||||
|        x2="610.13782" | ||||
|        y2="492.52756" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient37802" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop37804" | ||||
|          offset="0" | ||||
|          style="stop-color:#2c2c2c;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop37806" | ||||
|          offset="1" | ||||
|          style="stop-color:#16191a;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient76469-7-7-4-3" | ||||
|        id="linearGradient77680" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1,0,0,1.0322581,717.71949,428.68472)" | ||||
|        x1="6" | ||||
|        y1="102.95528" | ||||
|        x2="6" | ||||
|        y2="84.505203" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient76469-7-7-4-3"> | ||||
|       <stop | ||||
|          style="stop-color:#2e3232;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop76471-7-1-5-7" /> | ||||
|       <stop | ||||
|          style="stop-color:#3e4545;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop76473-9-0-0-9" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4-0" | ||||
|        is_visible="true" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient37802-8" | ||||
|        id="linearGradient12311-3-1-0-5-4" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-256.56122,59.071426)" | ||||
|        x1="610.13782" | ||||
|        y1="501.43866" | ||||
|        x2="610.13782" | ||||
|        y2="492.52756" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient37802-8" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop37804-1" | ||||
|          offset="0" | ||||
|          style="stop-color:#2c2c2c;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop37806-8" | ||||
|          offset="1" | ||||
|          style="stop-color:#16191a;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <linearGradient | ||||
|        y2="492.52756" | ||||
|        x2="610.13782" | ||||
|        y1="501.43866" | ||||
|        x1="610.13782" | ||||
|        gradientTransform="matrix(1.5918367,0,0,0.85714285,-900.56122,-423.92857)" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        id="linearGradient13602" | ||||
|        xlink:href="#linearGradient37802-8" | ||||
|        inkscape:collect="always" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="0.35" | ||||
|      inkscape:cx="32.000004" | ||||
|      inkscape:cy="10.999997" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="16.760995" | ||||
|      inkscape:cy="21.955673" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      inkscape:current-layer="g37994" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="609" | ||||
|      inkscape:window-height="501" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1375" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid12954" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata3017"> | ||||
|      id="metadata2862"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
| @@ -66,73 +186,70 @@ | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-343,-521.36218)"> | ||||
|      transform="translate(-444.64286,-781.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" /> | ||||
|        transform="matrix(0.6526046,0,0,0.80554422,99.592644,-636.32172)" | ||||
|        id="g37994"> | ||||
|       <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" /> | ||||
|          id="g37996" | ||||
|          transform="translate(-115,1277)"> | ||||
|         <rect | ||||
|            style="color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964000000001;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;opacity:0.19591837" | ||||
|            id="rect13475" | ||||
|            width="98" | ||||
|            height="25" | ||||
|            x="644.5" | ||||
|            y="484.61118" | ||||
|            rx="4.7429576" | ||||
|            ry="3.8424656" /> | ||||
|         <rect | ||||
|            ry="3.8424656" | ||||
|            rx="4.7429576" | ||||
|            y="483.5" | ||||
|            x="644.5" | ||||
|            height="25" | ||||
|            width="98" | ||||
|            id="rect38000" | ||||
|            style="color:#000000;fill:url(#linearGradient12311-3-1-0-5-4);fill-opacity:1;fill-rule:nonzero;stroke:#16191a;stroke-width:1.37920942;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:new" /> | ||||
|       </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" /> | ||||
|          transform="translate(-49.946213,-1.890275)" | ||||
|          id="g38002"> | ||||
|         <g | ||||
|            transform="translate(-115,1247)" | ||||
|            style="display:inline" | ||||
|            id="g38004"> | ||||
|           <rect | ||||
|              ry="3.7972314" | ||||
|              rx="4.6871223" | ||||
|              y="515.5" | ||||
|              x="694.53046" | ||||
|              height="25" | ||||
|              width="45.969578" | ||||
|              id="rect38006" | ||||
|              style="color:#000000;fill:url(#linearGradient38024);fill-opacity:1;fill-rule:nonzero;stroke:#1f2020;stroke-width:1.37920964;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" /> | ||||
|           <path | ||||
|              sodipodi:nodetypes="cc" | ||||
|              style="opacity:0.1;color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964px;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" | ||||
|              d="m 699.09675,516.7365 36.86904,0" | ||||
|              id="path38016" | ||||
|              inkscape:path-effect="#path-effect77541-4" | ||||
|              inkscape:original-d="m 699.09675,516.7365 36.86904,0" | ||||
|              inkscape:connector-curvature="0" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <text | ||||
|          xml:space="preserve" | ||||
|          style="font-size:13.79166794px;font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;line-height:125%;letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|          x="520.29974" | ||||
|          y="1997.0011" | ||||
|          id="text75614" | ||||
|          sodipodi:linespacing="125%" | ||||
|          transform="scale(1.1236771,0.88993537)"><tspan | ||||
|            sodipodi:role="line" | ||||
|            id="tspan75616" | ||||
|            x="520.29974" | ||||
|            y="1997.0011">OFF</tspan></text> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 7.2 KiB After Width: | Height: | Size: 8.8 KiB | 
| @@ -7,51 +7,113 @@ | ||||
|    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="65" | ||||
|    height="22" | ||||
|    id="svg3199" | ||||
|    id="svg2857" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="toggle-on-intl.svg"> | ||||
|   <defs | ||||
|      id="defs3201"> | ||||
|      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="perspective3207" /> | ||||
|        id="perspective2865" /> | ||||
|     <inkscape:perspective | ||||
|        id="perspective3187" | ||||
|        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" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient77461" | ||||
|        id="linearGradient77551" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.3066667,0,0,1,-841.64667,-483)" | ||||
|        x1="1164.7644" | ||||
|        y1="962.93695" | ||||
|        x2="1164.7644" | ||||
|        y2="970.51404" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient77461" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop77463" | ||||
|          offset="0" | ||||
|          style="stop-color:#182f4c;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop77465" | ||||
|          offset="1" | ||||
|          style="stop-color:#204a87;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient76469-7-7-4" | ||||
|        id="linearGradient38024" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0215462,0,0,1.0322581,717.22867,428.68472)" | ||||
|        x1="6" | ||||
|        y1="102.95528" | ||||
|        x2="6" | ||||
|        y2="84.505203" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient76469-7-7-4"> | ||||
|       <stop | ||||
|          style="stop-color:#2e3232;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop76471-7-1-5" /> | ||||
|       <stop | ||||
|          style="stop-color:#3e4545;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop76473-9-0-0" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4" | ||||
|        is_visible="true" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="49.147112" | ||||
|      inkscape:cy="17.532036" | ||||
|      inkscape:zoom="32" | ||||
|      inkscape:cx="17.255148" | ||||
|      inkscape:cy="8.9252639" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="1412" | ||||
|      inkscape:window-height="1067" | ||||
|      inkscape:current-layer="g37994" | ||||
|      showgrid="true" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1375" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="26" | ||||
|      inkscape:window-maximized="0" /> | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid12954" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata3204"> | ||||
|      id="metadata2862"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
| @@ -66,57 +128,65 @@ | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(-342.5,-521.36218)"> | ||||
|      transform="translate(-444.64286,-781.36218)"> | ||||
|     <g | ||||
|        style="display:inline" | ||||
|        transform="translate(-453.5,448.36218)" | ||||
|        id="g16453"> | ||||
|       <rect | ||||
|          style="color:#000000;fill:#4a90d9;fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:0.99999994000000003;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)" /> | ||||
|        transform="matrix(0.6526046,0,0,0.80554422,99.592644,-636.32172)" | ||||
|        id="g37994"> | ||||
|       <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" /> | ||||
|          id="g37996" | ||||
|          transform="translate(-115,1277)"> | ||||
|         <rect | ||||
|            style="color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964000000001;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;opacity:0.19591837" | ||||
|            id="rect13475" | ||||
|            width="98" | ||||
|            height="25" | ||||
|            x="644.5" | ||||
|            y="484.61118" | ||||
|            rx="4.7429576" | ||||
|            ry="3.8424656" /> | ||||
|         <rect | ||||
|            ry="3.8424656" | ||||
|            rx="4.7429576" | ||||
|            y="483.5" | ||||
|            x="644.5" | ||||
|            height="25" | ||||
|            width="98" | ||||
|            id="rect38000" | ||||
|            style="color:#000000;fill:url(#linearGradient77551);fill-opacity:1;fill-rule:nonzero;stroke:#182f4c;stroke-width:1.37920964;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" /> | ||||
|       </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 | ||||
|          transform="translate(2.0625,-2)" | ||||
|          id="g38002"> | ||||
|         <g | ||||
|            transform="translate(-115,1247)" | ||||
|            style="display:inline" | ||||
|            id="g38004"> | ||||
|           <rect | ||||
|              ry="3.7972314" | ||||
|              rx="4.6871223" | ||||
|              y="515.5" | ||||
|              x="694.53046" | ||||
|              height="25" | ||||
|              width="45.969578" | ||||
|              id="rect38006" | ||||
|              style="color:#000000;fill:url(#linearGradient38024);fill-opacity:1;fill-rule:nonzero;stroke:#1f2020;stroke-width:1.37920964;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" /> | ||||
|           <path | ||||
|              sodipodi:nodetypes="cc" | ||||
|              style="opacity:0.1;color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964px;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" | ||||
|              d="m 699.09675,516.7365 36.86904,0" | ||||
|              id="path38016" | ||||
|              inkscape:path-effect="#path-effect77541-4" | ||||
|              inkscape:original-d="m 699.09675,516.7365 36.86904,0" | ||||
|              inkscape:connector-curvature="0" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <rect | ||||
|          style="color:#000000;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.99999994;marker:none;visibility:visible;display:inline;overflow:visible;enable-background:new" | ||||
|          id="rect13678" | ||||
|          width="3.0646207" | ||||
|          height="12.414008" | ||||
|          x="554.77728" | ||||
|          y="1767.3566" /> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 4.5 KiB After Width: | Height: | Size: 6.8 KiB | 
| @@ -7,13 +7,14 @@ | ||||
|    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="65" | ||||
|    height="22" | ||||
|    id="svg2857" | ||||
|    version="1.1" | ||||
|    inkscape:version="0.48.1 r9760" | ||||
|    inkscape:version="0.48.5 r10040" | ||||
|    sodipodi:docname="toggle-on-us.svg"> | ||||
|   <defs | ||||
|      id="defs2859"> | ||||
| @@ -31,27 +32,86 @@ | ||||
|        inkscape:vp_y="0 : 1000 : 0" | ||||
|        inkscape:vp_x="0 : 0.5 : 1" | ||||
|        sodipodi:type="inkscape:persp3d" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient77461" | ||||
|        id="linearGradient77551" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.3066667,0,0,1,-841.64667,-483)" | ||||
|        x1="1164.7644" | ||||
|        y1="962.93695" | ||||
|        x2="1164.7644" | ||||
|        y2="970.51404" /> | ||||
|     <linearGradient | ||||
|        id="linearGradient77461" | ||||
|        inkscape:collect="always"> | ||||
|       <stop | ||||
|          id="stop77463" | ||||
|          offset="0" | ||||
|          style="stop-color:#182f4c;stop-opacity:1" /> | ||||
|       <stop | ||||
|          id="stop77465" | ||||
|          offset="1" | ||||
|          style="stop-color:#204a87;stop-opacity:1" /> | ||||
|     </linearGradient> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        xlink:href="#linearGradient76469-7-7-4" | ||||
|        id="linearGradient38024" | ||||
|        gradientUnits="userSpaceOnUse" | ||||
|        gradientTransform="matrix(1.0215462,0,0,1.0322581,717.22867,428.68472)" | ||||
|        x1="6" | ||||
|        y1="102.95528" | ||||
|        x2="6" | ||||
|        y2="84.505203" /> | ||||
|     <linearGradient | ||||
|        inkscape:collect="always" | ||||
|        id="linearGradient76469-7-7-4"> | ||||
|       <stop | ||||
|          style="stop-color:#2e3232;stop-opacity:1" | ||||
|          offset="0" | ||||
|          id="stop76471-7-1-5" /> | ||||
|       <stop | ||||
|          style="stop-color:#3e4545;stop-opacity:1" | ||||
|          offset="1" | ||||
|          id="stop76473-9-0-0" /> | ||||
|     </linearGradient> | ||||
|     <inkscape:path-effect | ||||
|        effect="spiro" | ||||
|        id="path-effect77541-4" | ||||
|        is_visible="true" /> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      pagecolor="#000000" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageopacity="1" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1" | ||||
|      inkscape:cx="19.689855" | ||||
|      inkscape:cy="2.0517979" | ||||
|      inkscape:cx="33.380898" | ||||
|      inkscape:cy="6.9658271" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      inkscape:current-layer="g37994" | ||||
|      showgrid="false" | ||||
|      inkscape:window-width="941" | ||||
|      inkscape:window-height="751" | ||||
|      inkscape:window-x="2577" | ||||
|      inkscape:window-y="206" | ||||
|      inkscape:window-maximized="0" | ||||
|      inkscape:window-width="2560" | ||||
|      inkscape:window-height="1375" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="27" | ||||
|      inkscape:window-maximized="1" | ||||
|      borderlayer="true" | ||||
|      inkscape:showpageshadow="false" /> | ||||
|      inkscape:showpageshadow="false" | ||||
|      inkscape:snap-nodes="false" | ||||
|      inkscape:snap-bbox="true" | ||||
|      showborder="false"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid12954" | ||||
|        empspacing="5" | ||||
|        visible="true" | ||||
|        enabled="true" | ||||
|        snapvisiblegridlinesonly="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata2862"> | ||||
|     <rdf:RDF> | ||||
| @@ -70,61 +130,68 @@ | ||||
|      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:#4a90d9;fill-opacity:1;fill-rule:nonzero;stroke:#3465a4;stroke-width:0.99999994000000003;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)" /> | ||||
|        transform="matrix(0.6526046,0,0,0.80554422,99.592644,-636.32172)" | ||||
|        id="g37994"> | ||||
|       <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" /> | ||||
|          id="g37996" | ||||
|          transform="translate(-115,1277)"> | ||||
|         <rect | ||||
|            style="color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964000000001;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;opacity:0.19591837" | ||||
|            id="rect13475" | ||||
|            width="98" | ||||
|            height="25" | ||||
|            x="644.5" | ||||
|            y="484.61118" | ||||
|            rx="4.7429576" | ||||
|            ry="3.8424656" /> | ||||
|         <rect | ||||
|            ry="3.8424656" | ||||
|            rx="4.7429576" | ||||
|            y="483.5" | ||||
|            x="644.5" | ||||
|            height="25" | ||||
|            width="98" | ||||
|            id="rect38000" | ||||
|            style="color:#000000;fill:url(#linearGradient77551);fill-opacity:1;fill-rule:nonzero;stroke:#182f4c;stroke-width:1.37920964;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" /> | ||||
|       </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" /> | ||||
|          transform="translate(2.0625,-2)" | ||||
|          id="g38002"> | ||||
|         <g | ||||
|            transform="translate(-115,1247)" | ||||
|            style="display:inline" | ||||
|            id="g38004"> | ||||
|           <rect | ||||
|              ry="3.7972314" | ||||
|              rx="4.6871223" | ||||
|              y="515.5" | ||||
|              x="694.53046" | ||||
|              height="25" | ||||
|              width="45.969578" | ||||
|              id="rect38006" | ||||
|              style="color:#000000;fill:url(#linearGradient38024);fill-opacity:1;fill-rule:nonzero;stroke:#1f2020;stroke-width:1.37920964;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" /> | ||||
|           <path | ||||
|              sodipodi:nodetypes="cc" | ||||
|              style="opacity:0.1;color:#000000;fill:none;stroke:#ffffff;stroke-width:1.37920964px;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" | ||||
|              d="m 699.09675,516.7365 36.86904,0" | ||||
|              id="path38016" | ||||
|              inkscape:path-effect="#path-effect77541-4" | ||||
|              inkscape:original-d="m 699.09675,516.7365 36.86904,0" | ||||
|              inkscape:connector-curvature="0" /> | ||||
|         </g> | ||||
|       </g> | ||||
|       <text | ||||
|          transform="scale(1.1000946,0.90901274)" | ||||
|          sodipodi:linespacing="125%" | ||||
|          id="text38018" | ||||
|          y="1955.5205" | ||||
|          x="495.94223" | ||||
|          style="font-size:13.29953671px;font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;line-height:125%;letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:none;font-family:Cantarell;-inkscape-font-specification:Cantarell Bold" | ||||
|          xml:space="preserve"><tspan | ||||
|            y="1955.5205" | ||||
|            x="495.94223" | ||||
|            id="tspan38020" | ||||
|            sodipodi:role="line">ON</tspan></text> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 6.8 KiB After Width: | Height: | Size: 7.0 KiB | 
| @@ -66,6 +66,7 @@ IGNORE_HFILES=					\ | ||||
| 	gactionmuxer.h				\ | ||||
| 	gactionobservable.h			\ | ||||
| 	gactionobserver.h			\ | ||||
| 	shell-network-agent.h			\ | ||||
| 	shell-recorder-src.h | ||||
|  | ||||
| if !BUILD_RECORDER | ||||
|   | ||||
| @@ -17,7 +17,6 @@ | ||||
|   <chapter> | ||||
|     <title>Actors</title> | ||||
|     <xi:include href="xml/shell-generic-container.xml"/> | ||||
|     <xi:include href="xml/shell-slicer.xml"/> | ||||
|     <xi:include href="xml/shell-stack.xml"/> | ||||
|   </chapter> | ||||
|   <chapter> | ||||
| @@ -46,11 +45,10 @@ | ||||
|     <xi:include href="doc-gen-org.gnome.Shell.SearchProvider.xml"/> | ||||
|     <xi:include href="doc-gen-org.gnome.Shell.SearchProvider2.xml"/> | ||||
|     <xi:include href="xml/shell-global.xml"/> | ||||
|     <xi:include href="xml/shell-keybinding-modes.xml"/> | ||||
|     <xi:include href="xml/shell-action-modes.xml"/> | ||||
|     <xi:include href="xml/shell-wm.xml"/> | ||||
|     <xi:include href="xml/shell-util.xml"/> | ||||
|     <xi:include href="xml/shell-mount-operation.xml"/> | ||||
|     <xi:include href="xml/shell-network-agent.xml"/> | ||||
|     <xi:include href="xml/shell-polkit-authentication-agent.xml"/> | ||||
|     <xi:include href="xml/shell-tp-client.xml"/> | ||||
|   </chapter> | ||||
|   | ||||
| @@ -27,7 +27,9 @@ its dependencies to build from tarballs.</description> | ||||
|   <download-page rdf:resource="http://download.gnome.org/sources/gnome-shell/" /> | ||||
|   <bug-database rdf:resource="https://bugzilla.gnome.org/browse.cgi?product=gnome-shell" /> | ||||
|  | ||||
|   <category rdf:resource="http://api.gnome.org/doap-extensions#desktop" /> | ||||
|   <category rdf:resource="http://api.gnome.org/doap-extensions#core" /> | ||||
|   <programming-language>JavaScript</programming-language> | ||||
|   <programming-language>C</programming-language> | ||||
|  | ||||
|   <maintainer> | ||||
|     <foaf:Person> | ||||
|   | ||||
| @@ -13,7 +13,7 @@ misc/config.js: misc/config.js.in Makefile | ||||
| 	    -e "s|[@]sysconfdir@|$(sysconfdir)|g" \ | ||||
|                $< > $@ | ||||
|  | ||||
| js_resource_files = $(shell $(GLIB_COMPILE_RESOURCES) --sourcedir=$(srcdir) --generate-dependencies $(srcdir)/js-resources.gresource.xml) | ||||
| js_resource_files = $(shell $(GLIB_COMPILE_RESOURCES) --sourcedir=$(srcdir) --sourcedir=$(builddir) --generate-dependencies $(srcdir)/js-resources.gresource.xml) | ||||
| js-resources.h: js-resources.gresource.xml $(js_resource_files) misc/config.js | ||||
| 	$(AM_V_GEN) $(GLIB_COMPILE_RESOURCES) --target=$@ --sourcedir=$(srcdir) --sourcedir=$(builddir) --generate --c-name shell_js_resources $< | ||||
| js-resources.c: js-resources.gresource.xml $(js_resource_files) misc/config.js | ||||
|   | ||||
| @@ -54,6 +54,7 @@ const Application = new Lang.Class({ | ||||
|         this._startupUuid = null; | ||||
|         this._loaded = false; | ||||
|         this._skipMainWindow = false; | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|     }, | ||||
|  | ||||
|     _extensionAvailable: function(uuid) { | ||||
| @@ -62,9 +63,6 @@ const Application = new Lang.Class({ | ||||
|         if (!extension) | ||||
|             return false; | ||||
|  | ||||
|         if (ExtensionUtils.isOutOfDate(extension)) | ||||
|             return false; | ||||
|  | ||||
|         if (!extension.dir.get_child('prefs.js').query_exists(null)) | ||||
|             return false; | ||||
|  | ||||
| @@ -278,11 +276,15 @@ const ExtensionRow = new Lang.Class({ | ||||
|  | ||||
|         this.uuid = uuid; | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.shell' }); | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|         this._settings.connect('changed::enabled-extensions', Lang.bind(this, | ||||
|             function() { | ||||
|                 this._switch.state = this._isEnabled(); | ||||
|             })); | ||||
|         this._settings.connect('changed::disable-extension-version-validation', | ||||
|             Lang.bind(this, function() { | ||||
|                 this._switch.sensitive = this._canEnable(); | ||||
|             })); | ||||
|  | ||||
|         this._buildUI(); | ||||
|     }, | ||||
| @@ -321,6 +323,7 @@ const ExtensionRow = new Lang.Class({ | ||||
|         this.prefsButton = button; | ||||
|  | ||||
|         this._switch = new Gtk.Switch({ valign: Gtk.Align.CENTER, | ||||
|                                         sensitive: this._canEnable(), | ||||
|                                         state: this._isEnabled() }); | ||||
|         this._switch.connect('notify::active', Lang.bind(this, | ||||
|             function() { | ||||
| @@ -333,6 +336,13 @@ const ExtensionRow = new Lang.Class({ | ||||
|         hbox.add(this._switch); | ||||
|     }, | ||||
|  | ||||
|     _canEnable: function() { | ||||
|         let extension = ExtensionUtils.extensions[this.uuid]; | ||||
|         let checkVersion = !this._settings.get_boolean('disable-extension-version-validation'); | ||||
|  | ||||
|         return !(checkVersion && ExtensionUtils.isOutOfDate(extension)); | ||||
|     }, | ||||
|  | ||||
|     _isEnabled: function() { | ||||
|         let extensions = this._settings.get_strv('enabled-extensions'); | ||||
|         return extensions.indexOf(this.uuid) != -1; | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
| @@ -126,7 +127,7 @@ const AuthPrompt = new Lang.Class({ | ||||
|  | ||||
|         this._initButtons(); | ||||
|  | ||||
|         let spinnerIcon = global.datadir + '/theme/process-working.svg'; | ||||
|         let spinnerIcon = Gio.File.new_for_uri('resource:///org/gnome/shell/theme/process-working.svg'); | ||||
|         this._spinner = new Animation.AnimatedIcon(spinnerIcon, DEFAULT_BUTTON_WELL_ICON_SIZE); | ||||
|         this._spinner.actor.opacity = 0; | ||||
|         this._spinner.actor.show(); | ||||
| @@ -134,13 +135,12 @@ const AuthPrompt = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         this._userVerifier.clear(); | ||||
|         this._userVerifier.disconnectAll(); | ||||
|         this._userVerifier.destroy(); | ||||
|         this._userVerifier = null; | ||||
|     }, | ||||
|  | ||||
|     _initButtons: function() { | ||||
|         this.cancelButton = new St.Button({ style_class: 'modal-dialog-button', | ||||
|         this.cancelButton = new St.Button({ style_class: 'modal-dialog-button button', | ||||
|                                             button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|                                             reactive: true, | ||||
|                                             can_focus: true, | ||||
| @@ -162,7 +162,7 @@ const AuthPrompt = new Lang.Class({ | ||||
|                               y_fill: false, | ||||
|                               x_align: St.Align.END, | ||||
|                               y_align: St.Align.MIDDLE }); | ||||
|         this.nextButton = new St.Button({ style_class: 'modal-dialog-button', | ||||
|         this.nextButton = new St.Button({ style_class: 'modal-dialog-button button', | ||||
|                                           button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|                                           reactive: true, | ||||
|                                           can_focus: true, | ||||
| @@ -261,6 +261,7 @@ const AuthPrompt = new Lang.Class({ | ||||
|  | ||||
|     _onVerificationComplete: function() { | ||||
|         this.verificationStatus = AuthPromptStatus.VERIFICATION_SUCCEEDED; | ||||
| 	this.cancelButton.reactive = false; | ||||
|     }, | ||||
|  | ||||
|     _onReset: function() { | ||||
| @@ -419,17 +420,20 @@ const AuthPrompt = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     setUser: function(user) { | ||||
|         let oldChild = this._userWell.get_child(); | ||||
|         if (oldChild) | ||||
|             oldChild.destroy(); | ||||
|  | ||||
|         if (user) { | ||||
|             let userWidget = new UserWidget.UserWidget(user); | ||||
|             this._userWell.set_child(userWidget.actor); | ||||
|         } else { | ||||
|             this._userWell.set_child(null); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     reset: function() { | ||||
|         let oldStatus = this.verificationStatus; | ||||
|         this.verificationStatus = AuthPromptStatus.NOT_VERIFYING; | ||||
|         this.cancelButton.reactive = true; | ||||
|  | ||||
|         if (oldStatus == AuthPromptStatus.VERIFYING) | ||||
|             this._userVerifier.cancel(); | ||||
| @@ -498,6 +502,9 @@ const AuthPrompt = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|         if (this.verificationStatus == AuthPromptStatus.NOT_VERIFYING || this.verificationStatus == AuthPromptStatus.VERIFICATION_SUCCEEDED) { | ||||
|             return; | ||||
|         } | ||||
|         this.reset(); | ||||
|         this.emit('cancelled'); | ||||
|     } | ||||
|   | ||||
| @@ -26,6 +26,7 @@ const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
| @@ -48,8 +49,6 @@ const _SCROLL_ANIMATION_TIME = 0.5; | ||||
| const _TIMED_LOGIN_IDLE_THRESHOLD = 5.0; | ||||
| const _LOGO_ICON_HEIGHT = 48; | ||||
|  | ||||
| let _loginDialog = null; | ||||
|  | ||||
| const UserListItem = new Lang.Class({ | ||||
|     Name: 'UserListItem', | ||||
|  | ||||
| @@ -66,6 +65,8 @@ const UserListItem = new Lang.Class({ | ||||
|                                      reactive: true, | ||||
|                                      x_align: St.Align.START, | ||||
|                                      x_fill: true }); | ||||
|         this.actor.connect('destroy', | ||||
|                            Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         this._userWidget = new UserWidget.UserWidget(this.user); | ||||
|         layout.add(this._userWidget.actor); | ||||
| @@ -89,6 +90,10 @@ const UserListItem = new Lang.Class({ | ||||
|             this.actor.remove_style_pseudo_class('logged-in'); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
|         this._user.disconnect(this._userChangedId); | ||||
|     }, | ||||
|  | ||||
|     _onClicked: function() { | ||||
|         this.emit('activate'); | ||||
|     }, | ||||
| @@ -360,31 +365,19 @@ const LoginDialog = new Lang.Class({ | ||||
|     Name: 'LoginDialog', | ||||
|  | ||||
|     _init: function(parentActor) { | ||||
|         this.actor = new St.Widget({ accessible_role: Atk.Role.WINDOW, | ||||
|                                      layout_manager: new Clutter.BinLayout(), | ||||
|                                      style_class: 'login-dialog', | ||||
|                                      visible: false }); | ||||
|         this.actor = new Shell.GenericContainer({ style_class: 'login-dialog', | ||||
|                                                   visible: false }); | ||||
|         this.actor.get_accessible().set_role(Atk.Role.WINDOW); | ||||
|  | ||||
|         this.actor.add_constraint(new Layout.MonitorConstraint({ primary: true })); | ||||
|         this.actor.connect('allocate', Lang.bind(this, this._onAllocate)); | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|         parentActor.add_child(this.actor); | ||||
|  | ||||
|         this._userManager = AccountsService.UserManager.get_default() | ||||
|         let gdmClient = new Gdm.Client(); | ||||
|         this._gdmClient = new Gdm.Client(); | ||||
|  | ||||
|         if (GLib.getenv('GDM_GREETER_TEST') != '1') { | ||||
|             this._greeter = gdmClient.get_greeter_sync(null); | ||||
|  | ||||
|             this._greeter.connect('default-session-name-changed', | ||||
|                                   Lang.bind(this, this._onDefaultSessionChanged)); | ||||
|  | ||||
|             this._greeter.connect('session-opened', | ||||
|                                   Lang.bind(this, this._onSessionOpened)); | ||||
|             this._greeter.connect('timed-login-requested', | ||||
|                                   Lang.bind(this, this._onTimedLoginRequested)); | ||||
|         } | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: GdmUtil.LOGIN_SCREEN_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: GdmUtil.LOGIN_SCREEN_SCHEMA }); | ||||
|  | ||||
|         this._settings.connect('changed::' + GdmUtil.BANNER_MESSAGE_KEY, | ||||
|                                Lang.bind(this, this._updateBanner)); | ||||
| @@ -396,30 +389,23 @@ const LoginDialog = new Lang.Class({ | ||||
|                                Lang.bind(this, this._updateLogo)); | ||||
|  | ||||
|         this._textureCache = St.TextureCache.get_default(); | ||||
|         this._textureCache.connect('texture-file-changed', | ||||
|                                    Lang.bind(this, this._updateLogoTexture)); | ||||
|         this._updateLogoTextureId = this._textureCache.connect('texture-file-changed', | ||||
|                                                                Lang.bind(this, this._updateLogoTexture)); | ||||
|  | ||||
|         this._userSelectionBox = new St.BoxLayout({ style_class: 'login-dialog-user-selection-box', | ||||
|                                                     x_align: Clutter.ActorAlign.CENTER, | ||||
|                                                     y_align: Clutter.ActorAlign.CENTER, | ||||
|                                                     x_expand: true, | ||||
|                                                     y_expand: true, | ||||
|                                                     vertical: true, | ||||
|                                                     visible: false }); | ||||
|         this.actor.add_child(this._userSelectionBox); | ||||
|  | ||||
|         this._bannerLabel = new St.Label({ style_class: 'login-dialog-banner', | ||||
|                                            text: '' }); | ||||
|         this._userSelectionBox.add(this._bannerLabel); | ||||
|         this._updateBanner(); | ||||
|  | ||||
|         this._userList = new UserList(); | ||||
|         this._userSelectionBox.add(this._userList.actor, | ||||
|                                    { expand: true, | ||||
|                                      x_fill: true, | ||||
|                                      y_fill: true }); | ||||
|  | ||||
|         this._authPrompt = new AuthPrompt.AuthPrompt(gdmClient, AuthPrompt.AuthPromptMode.UNLOCK_OR_LOG_IN); | ||||
|         this._authPrompt = new AuthPrompt.AuthPrompt(this._gdmClient, AuthPrompt.AuthPromptMode.UNLOCK_OR_LOG_IN); | ||||
|         this._authPrompt.connect('prompted', Lang.bind(this, this._onPrompted)); | ||||
|         this._authPrompt.connect('reset', Lang.bind(this, this._onReset)); | ||||
|         this._authPrompt.hide(); | ||||
| @@ -447,11 +433,25 @@ const LoginDialog = new Lang.Class({ | ||||
|                                      x_align: St.Align.START, | ||||
|                                      x_fill: true }); | ||||
|  | ||||
|         this._bannerView = new St.ScrollView({ style_class: 'login-dialog-banner-view', | ||||
|                                                opacity: 0, | ||||
|                                                vscrollbar_policy: Gtk.PolicyType.AUTOMATIC, | ||||
|                                                hscrollbar_policy: Gtk.PolicyType.NEVER }); | ||||
|         this.actor.add_child(this._bannerView); | ||||
|  | ||||
|         let bannerBox = new St.BoxLayout({ vertical: true }); | ||||
|  | ||||
|         this._bannerView.add_actor(bannerBox); | ||||
|         this._bannerLabel = new St.Label({ style_class: 'login-dialog-banner', | ||||
|                                            text: '' }); | ||||
|         this._bannerLabel.clutter_text.line_wrap = true; | ||||
|         this._bannerLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         bannerBox.add_child(this._bannerLabel); | ||||
|         this._updateBanner(); | ||||
|  | ||||
|         this._logoBin = new St.Widget({ style_class: 'login-dialog-logo-bin', | ||||
|                                         x_align: Clutter.ActorAlign.CENTER, | ||||
|                                         y_align: Clutter.ActorAlign.END, | ||||
|                                         x_expand: true, | ||||
|                                         y_expand: true }); | ||||
|                                         y_align: Clutter.ActorAlign.END }); | ||||
|         this.actor.add_child(this._logoBin); | ||||
|         this._updateLogo(); | ||||
|  | ||||
| @@ -478,8 +478,182 @@ const LoginDialog = new Lang.Class({ | ||||
|         // If the user list is enabled, it should take key focus; make sure the | ||||
|         // screen shield is initialized first to prevent it from stealing the | ||||
|         // focus later | ||||
|         Main.layoutManager.connect('startup-complete', | ||||
|                                    Lang.bind(this, this._updateDisableUserList)); | ||||
|         this._startupCompleteId = Main.layoutManager.connect('startup-complete', | ||||
|                                                              Lang.bind(this, this._updateDisableUserList)); | ||||
|     }, | ||||
|  | ||||
|     _getBannerAllocation: function (dialogBox) { | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [minWidth, minHeight, natWidth, natHeight] = this._bannerView.get_preferred_size(); | ||||
|         let centerX = dialogBox.x1 + (dialogBox.x2 - dialogBox.x1) / 2; | ||||
|  | ||||
|         actorBox.x1 = centerX - natWidth / 2; | ||||
|         actorBox.y1 = dialogBox.y1 + Main.layoutManager.panelBox.height; | ||||
|         actorBox.x2 = actorBox.x1 + natWidth; | ||||
|         actorBox.y2 = actorBox.y1 + natHeight; | ||||
|  | ||||
|         return actorBox; | ||||
|     }, | ||||
|  | ||||
|     _getLogoBinAllocation: function (dialogBox) { | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [minWidth, minHeight, natWidth, natHeight] = this._logoBin.get_preferred_size(); | ||||
|         let centerX = dialogBox.x1 + (dialogBox.x2 - dialogBox.x1) / 2; | ||||
|  | ||||
|         actorBox.x1 = centerX - natWidth / 2; | ||||
|         actorBox.y1 = dialogBox.y2 - natHeight; | ||||
|         actorBox.x2 = actorBox.x1 + natWidth; | ||||
|         actorBox.y2 = actorBox.y1 + natHeight; | ||||
|  | ||||
|         return actorBox; | ||||
|     }, | ||||
|  | ||||
|     _getCenterActorAllocation: function (dialogBox, actor) { | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [minWidth, minHeight, natWidth, natHeight] = actor.get_preferred_size(); | ||||
|         let centerX = dialogBox.x1 + (dialogBox.x2 - dialogBox.x1) / 2; | ||||
|         let centerY = dialogBox.y1 + (dialogBox.y2 - dialogBox.y1) / 2; | ||||
|  | ||||
|         actorBox.x1 = centerX - natWidth / 2; | ||||
|         actorBox.y1 = centerY - natHeight / 2; | ||||
|         actorBox.x2 = actorBox.x1 + natWidth; | ||||
|         actorBox.y2 = actorBox.y1 + natHeight; | ||||
|  | ||||
|         return actorBox; | ||||
|     }, | ||||
|  | ||||
|     _onAllocate: function (actor, dialogBox, flags) { | ||||
|         let dialogWidth = dialogBox.x2 - dialogBox.x1; | ||||
|         let dialogHeight = dialogBox.y2 - dialogBox.y1; | ||||
|  | ||||
|         // First find out what space the children require | ||||
|         let bannerAllocation = null; | ||||
|         let bannerHeight = 0; | ||||
|         let bannerWidth = 0; | ||||
|         if (this._bannerView.visible) { | ||||
|             bannerAllocation = this._getBannerAllocation(dialogBox, this._bannerView); | ||||
|             bannerHeight = bannerAllocation.y2 - bannerAllocation.y1; | ||||
|             bannerWidth = bannerAllocation.x2 - bannerAllocation.x1; | ||||
|         } | ||||
|  | ||||
|         let authPromptAllocation = null; | ||||
|         let authPromptHeight = 0; | ||||
|         let authPromptWidth = 0; | ||||
|         if (this._authPrompt.actor.visible) { | ||||
|             authPromptAllocation = this._getCenterActorAllocation(dialogBox, this._authPrompt.actor); | ||||
|             authPromptHeight = authPromptAllocation.y2 - authPromptAllocation.y1; | ||||
|             authPromptWidth = authPromptAllocation.x2 - authPromptAllocation.x1; | ||||
|         } | ||||
|  | ||||
|         let userSelectionAllocation = null; | ||||
|         let userSelectionHeight = 0; | ||||
|         if (this._userSelectionBox.visible) { | ||||
|             userSelectionAllocation = this._getCenterActorAllocation(dialogBox, this._userSelectionBox); | ||||
|             userSelectionHeight = userSelectionAllocation.y2 - userSelectionAllocation.y1; | ||||
|         } | ||||
|  | ||||
|         let logoAllocation = null; | ||||
|         let logoHeight = 0; | ||||
|         if (this._logoBin.visible) { | ||||
|             logoAllocation = this._getLogoBinAllocation(dialogBox); | ||||
|             logoHeight = logoAllocation.y2 - logoAllocation.y1; | ||||
|         } | ||||
|  | ||||
|         // Then figure out if we're overly constrained and need to | ||||
|         // try a different layout, or if we have what extra space we | ||||
|         // can hand out | ||||
|         if (bannerAllocation) { | ||||
|             let leftOverYSpace = dialogHeight - bannerHeight - authPromptHeight - logoHeight; | ||||
|  | ||||
|             if (leftOverYSpace > 0) { | ||||
|                  // First figure out how much left over space is up top | ||||
|                  let leftOverTopSpace = leftOverYSpace / 2; | ||||
|  | ||||
|                  // Then, shift the banner into the middle of that extra space | ||||
|                  let yShift = leftOverTopSpace / 2; | ||||
|  | ||||
|                  bannerAllocation.y1 += yShift; | ||||
|                  bannerAllocation.y2 += yShift; | ||||
|             } else { | ||||
|                  // Then figure out how much space there would be if we switched to a | ||||
|                  // wide layout with banner on one side and authprompt on the other. | ||||
|                  let leftOverXSpace = dialogWidth - authPromptWidth; | ||||
|  | ||||
|                  // In a wide view, half of the available space goes to the banner, | ||||
|                  // and the other half goes to the margins. | ||||
|                  let wideBannerWidth = leftOverXSpace / 2; | ||||
|                  let wideSpacing  = leftOverXSpace - wideBannerWidth; | ||||
|  | ||||
|                  // If we do go with a wide layout, we need there to be at least enough | ||||
|                  // space for the banner and the auth prompt to be the same width, | ||||
|                  // so it doesn't look unbalanced. | ||||
|                  if (authPromptWidth > 0 && wideBannerWidth > authPromptWidth) { | ||||
|                      let centerX = dialogBox.x1 + dialogWidth / 2; | ||||
|                      let centerY = dialogBox.y1 + dialogHeight / 2; | ||||
|  | ||||
|                      // A small portion of the spacing goes down the center of the | ||||
|                      // screen to help delimit the two columns of the wide view | ||||
|                      let centerGap = wideSpacing / 8; | ||||
|  | ||||
|                      // place the banner along the left edge of the center margin | ||||
|                      bannerAllocation.x2 = centerX - centerGap / 2; | ||||
|                      bannerAllocation.x1 = bannerAllocation.x2 - wideBannerWidth; | ||||
|  | ||||
|                      // figure out how tall it would like to be and try to accomodate | ||||
|                      // but don't let it get too close to the logo | ||||
|                      let [wideMinHeight, wideBannerHeight] = this._bannerView.get_preferred_height(wideBannerWidth); | ||||
|  | ||||
|                      let maxWideHeight = dialogHeight - 3 * logoHeight; | ||||
|                      wideBannerHeight = Math.min(maxWideHeight, wideBannerHeight); | ||||
|                      bannerAllocation.y1 = centerY - wideBannerHeight / 2; | ||||
|                      bannerAllocation.y2 = bannerAllocation.y1 + wideBannerHeight; | ||||
|  | ||||
|                      // place the auth prompt along the right edge of the center margin | ||||
|                      authPromptAllocation.x1 = centerX + centerGap / 2; | ||||
|                      authPromptAllocation.x2 = authPromptAllocation.x1 + authPromptWidth; | ||||
|                  } else { | ||||
|                      // If we aren't going to do a wide view, then we need to limit | ||||
|                      // the height of the banner so it will present scrollbars | ||||
|  | ||||
|                      // First figure out how much space there is without the banner | ||||
|                      leftOverYSpace += bannerHeight; | ||||
|  | ||||
|                      // Then figure out how much of that space is up top | ||||
|                      let availableTopSpace = leftOverYSpace / 2; | ||||
|  | ||||
|                      // Then give all of that space to the banner | ||||
|                      bannerAllocation.y2 = bannerAllocation.y1 + availableTopSpace; | ||||
|                  } | ||||
|             } | ||||
|         } else if (userSelectionAllocation) { | ||||
|             // Grow the user list to fill the space | ||||
|             let leftOverYSpace = dialogHeight - userSelectionHeight - logoHeight; | ||||
|  | ||||
|             if (leftOverYSpace > 0) { | ||||
|                 let topExpansion = leftOverYSpace / 2; | ||||
|                 let bottomExpansion = topExpansion; | ||||
|  | ||||
|                 userSelectionAllocation.y1 -= topExpansion; | ||||
|                 userSelectionAllocation.y2 += bottomExpansion; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Finally hand out the allocations | ||||
|         if (bannerAllocation) { | ||||
|             this._bannerView.allocate(bannerAllocation, flags); | ||||
|         } | ||||
|  | ||||
|         if (authPromptAllocation) | ||||
|             this._authPrompt.actor.allocate(authPromptAllocation, flags); | ||||
|  | ||||
|         if (userSelectionAllocation) | ||||
|             this._userSelectionBox.allocate(userSelectionAllocation, flags); | ||||
|  | ||||
|         if (logoAllocation) | ||||
|             this._logoBin.allocate(logoAllocation, flags); | ||||
|     }, | ||||
|  | ||||
|     _ensureUserListLoaded: function() { | ||||
| @@ -534,24 +708,38 @@ const LoginDialog = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateLogoTexture: function(cache, uri) { | ||||
|         if (this._logoFileUri != uri) | ||||
|     _fadeInBannerView: function() { | ||||
|         this._bannerView.show(); | ||||
|         Tweener.addTween(this._bannerView, | ||||
|                          { opacity: 255, | ||||
|                            time: _FADE_ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad' }); | ||||
|     }, | ||||
|  | ||||
|     _hideBannerView: function() { | ||||
|         Tweener.removeTweens(this._bannerView); | ||||
|         this._bannerView.opacity = 0; | ||||
|         this._bannerView.hide(); | ||||
|     }, | ||||
|  | ||||
|     _updateLogoTexture: function(cache, file) { | ||||
|         if (this._logoFile && !this._logoFile.equal(file)) | ||||
|             return; | ||||
|  | ||||
|         this._logoBin.destroy_all_children(); | ||||
|         if (this._logoFileUri) { | ||||
|         if (this._logoFile) { | ||||
|             let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|             this._logoBin.add_child(this._textureCache.load_uri_async(this._logoFileUri, | ||||
|                                                                       -1, _LOGO_ICON_HEIGHT, | ||||
|                                                                       scaleFactor)); | ||||
|             this._logoBin.add_child(this._textureCache.load_file_async(this._logoFile, | ||||
|                                                                        -1, _LOGO_ICON_HEIGHT, | ||||
|                                                                        scaleFactor)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateLogo: function() { | ||||
|         let path = this._settings.get_string(GdmUtil.LOGO_KEY); | ||||
|  | ||||
|         this._logoFileUri = path ? Gio.file_new_for_path(path).get_uri() : null; | ||||
|         this._updateLogoTexture(this._textureCache, this._logoFileUri); | ||||
|         this._logoFile = path ? Gio.file_new_for_path(path) : null; | ||||
|         this._updateLogoTexture(this._textureCache, this._logoFile); | ||||
|     }, | ||||
|  | ||||
|     _onPrompted: function() { | ||||
| @@ -562,7 +750,24 @@ const LoginDialog = new Lang.Class({ | ||||
|         this._showPrompt(); | ||||
|     }, | ||||
|  | ||||
|     _resetGreeterProxy: function() { | ||||
|         if (GLib.getenv('GDM_GREETER_TEST') != '1') { | ||||
|             if (this._greeter) { | ||||
|                 this._greeter.run_dispose(); | ||||
|             } | ||||
|             this._greeter = this._gdmClient.get_greeter_sync(null); | ||||
|  | ||||
|             this._defaultSessionChangedId = this._greeter.connect('default-session-name-changed', | ||||
|                                                                   Lang.bind(this, this._onDefaultSessionChanged)); | ||||
|             this._sessionOpenedId = this._greeter.connect('session-opened', | ||||
|                                                           Lang.bind(this, this._onSessionOpened)); | ||||
|             this._timedLoginRequestedId = this._greeter.connect('timed-login-requested', | ||||
|                                                                 Lang.bind(this, this._onTimedLoginRequested)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onReset: function(authPrompt, beginRequest) { | ||||
|         this._resetGreeterProxy(); | ||||
|         this._sessionMenuButton.updateSensitivity(true); | ||||
|  | ||||
|         this._user = null; | ||||
| @@ -601,6 +806,7 @@ const LoginDialog = new Lang.Class({ | ||||
|                          { opacity: 255, | ||||
|                            time: _FADE_ANIMATION_TIME, | ||||
|                            transition: 'easeOutQuad' }); | ||||
|         this._fadeInBannerView(); | ||||
|     }, | ||||
|  | ||||
|     _showRealmLoginHint: function(realmManager, hint) { | ||||
| @@ -640,6 +846,8 @@ const LoginDialog = new Lang.Class({ | ||||
|                                                         realmManager.release(); | ||||
|                                                     })); | ||||
|         this._updateCancelButton(); | ||||
|  | ||||
|         this._authPrompt.updateSensitivity(true); | ||||
|         this._showPrompt(); | ||||
|     }, | ||||
|  | ||||
| @@ -667,10 +875,12 @@ const LoginDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _gotGreeterSessionProxy: function(proxy) { | ||||
|         proxy.connect('g-properties-changed', Lang.bind(this, function() { | ||||
|             if (proxy.Active) | ||||
|                 this._loginScreenSessionActivated(); | ||||
|         })); | ||||
|         this._greeterSessionProxy = proxy; | ||||
|         this._greeterSessionProxyChangedId = | ||||
|             proxy.connect('g-properties-changed', Lang.bind(this, function() { | ||||
|                 if (proxy.Active) | ||||
|                     this._loginScreenSessionActivated(); | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     _startSession: function(serviceName) { | ||||
| @@ -853,6 +1063,7 @@ const LoginDialog = new Lang.Class({ | ||||
|     _showUserList: function() { | ||||
|         this._ensureUserListLoaded(); | ||||
|         this._authPrompt.hide(); | ||||
|         this._hideBannerView(); | ||||
|         this._sessionMenuButton.close(); | ||||
|         this._setUserListExpanded(true); | ||||
|         this._notListedButton.show(); | ||||
| @@ -892,6 +1103,30 @@ const LoginDialog = new Lang.Class({ | ||||
|             this._userManager.disconnect(this._userManagerLoadedId); | ||||
|             this._userManagerLoadedId = 0; | ||||
|         } | ||||
|         if (this._userAddedId) { | ||||
|             this._userManager.disconnect(this._userAddedId); | ||||
|             this._userAddedId = 0; | ||||
|         } | ||||
|         if (this._userRemovedId) { | ||||
|             this._userManager.disconnect(this._userRemovedId); | ||||
|             this._userRemovedId = 0; | ||||
|         } | ||||
|         this._textureCache.disconnect(this._updateLogoTextureId); | ||||
|         Main.layoutManager.disconnect(this._startupCompleteId); | ||||
|         if (this._settings) { | ||||
|             this._settings.run_dispose(); | ||||
|             this._settings = null; | ||||
|         } | ||||
|         if (this._greeter) { | ||||
|             this._greeter.disconnect(this._defaultSessionChangedId); | ||||
|             this._greeter.disconnect(this._sessionOpenedId); | ||||
|             this._greeter.disconnect(this._timedLoginRequestedId); | ||||
|             this._greeter = null; | ||||
|         } | ||||
|         if (this._greeterSessionProxy) { | ||||
|             this._greeterSessionProxy.disconnect(this._greeterSessionProxyChangedId); | ||||
|             this._greeterSessionProxy = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _loadUserList: function() { | ||||
| @@ -906,15 +1141,15 @@ const LoginDialog = new Lang.Class({ | ||||
|             this._userList.addUser(users[i]); | ||||
|         } | ||||
|  | ||||
|         this._userManager.connect('user-added', | ||||
|                                   Lang.bind(this, function(userManager, user) { | ||||
|                                       this._userList.addUser(user); | ||||
|                                   })); | ||||
|         this._userAddedId = this._userManager.connect('user-added', | ||||
|                                                       Lang.bind(this, function(userManager, user) { | ||||
|                                                           this._userList.addUser(user); | ||||
|                                                       })); | ||||
|  | ||||
|         this._userManager.connect('user-removed', | ||||
|                                   Lang.bind(this, function(userManager, user) { | ||||
|                                       this._userList.removeUser(user); | ||||
|                                   })); | ||||
|         this._userRemovedId = this._userManager.connect('user-removed', | ||||
|                                                         Lang.bind(this, function(userManager, user) { | ||||
|                                                             this._userList.removeUser(user); | ||||
|                                                         })); | ||||
|  | ||||
|         return GLib.SOURCE_REMOVE; | ||||
|     }, | ||||
| @@ -928,6 +1163,8 @@ const LoginDialog = new Lang.Class({ | ||||
|         this.actor.show(); | ||||
|         this.actor.opacity = 0; | ||||
|  | ||||
|         Main.pushModal(this.actor, { actionMode: Shell.ActionMode.LOGIN_SCREEN }); | ||||
|  | ||||
|         Tweener.addTween(this.actor, | ||||
|                          { opacity: 255, | ||||
|                            time: 1, | ||||
| @@ -937,7 +1174,8 @@ const LoginDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     close: function() { | ||||
|         Main.ctrlAltTabManager.removeGroup(this.dialogLayout); | ||||
|         Main.popModal(this.actor); | ||||
|         Main.ctrlAltTabManager.removeGroup(this.actor); | ||||
|     }, | ||||
|  | ||||
|     cancel: function() { | ||||
|   | ||||
| @@ -128,7 +128,7 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|  | ||||
|         this._client = client; | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: LOGIN_SCREEN_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: LOGIN_SCREEN_SCHEMA }); | ||||
|         this._settings.connect('changed', | ||||
|                                Lang.bind(this, this._updateDefaultService)); | ||||
|         this._updateDefaultService(); | ||||
| @@ -142,10 +142,10 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|         // after a user has been picked. | ||||
|         this._checkForSmartcard(); | ||||
|  | ||||
|         this._smartcardManager.connect('smartcard-inserted', | ||||
|                                        Lang.bind(this, this._checkForSmartcard)); | ||||
|         this._smartcardManager.connect('smartcard-removed', | ||||
|                                        Lang.bind(this, this._checkForSmartcard)); | ||||
|         this._smartcardInsertedId = this._smartcardManager.connect('smartcard-inserted', | ||||
|                                                                    Lang.bind(this, this._checkForSmartcard)); | ||||
|         this._smartcardRemovedId = this._smartcardManager.connect('smartcard-removed', | ||||
|                                                                   Lang.bind(this, this._checkForSmartcard)); | ||||
|  | ||||
|         this._messageQueue = []; | ||||
|         this._messageQueueTimeoutId = 0; | ||||
| @@ -159,8 +159,8 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|         if (this._oVirtCredentialsManager.hasToken()) | ||||
|             this._oVirtUserAuthenticated(this._oVirtCredentialsManager.getToken()); | ||||
|  | ||||
|         this._oVirtCredentialsManager.connect('user-authenticated', | ||||
|                                               Lang.bind(this, this._oVirtUserAuthenticated)); | ||||
|         this._oVirtUserAuthenticatedId = this._oVirtCredentialsManager.connect('user-authenticated', | ||||
|                                                                                Lang.bind(this, this._oVirtUserAuthenticated)); | ||||
|     }, | ||||
|  | ||||
|     begin: function(userName, hold) { | ||||
| @@ -191,20 +191,37 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _clearUserVerifier: function() { | ||||
|         if (this._userVerifier) { | ||||
|             this._userVerifier.run_dispose(); | ||||
|             this._userVerifier = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     clear: function() { | ||||
|         if (this._cancellable) { | ||||
|             this._cancellable.cancel(); | ||||
|             this._cancellable = null; | ||||
|         } | ||||
|  | ||||
|         if (this._userVerifier) { | ||||
|             this._userVerifier.run_dispose(); | ||||
|             this._userVerifier = null; | ||||
|         } | ||||
|  | ||||
|         this._clearUserVerifier(); | ||||
|         this._clearMessageQueue(); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         this.clear(); | ||||
|  | ||||
|         this._settings.run_dispose(); | ||||
|         this._settings = null; | ||||
|  | ||||
|         this._smartcardManager.disconnect(this._smartcardInsertedId); | ||||
|         this._smartcardManager.disconnect(this._smartcardRemovedId); | ||||
|         this._smartcardManager = null; | ||||
|  | ||||
|         this._oVirtCredentialsManager.disconnect(this._oVirtUserAuthenticatedId); | ||||
|         this._oVirtCredentialsManager = null; | ||||
|     }, | ||||
|  | ||||
|     answerQuery: function(serviceName, answer) { | ||||
|         if (!this.hasPendingMessages) { | ||||
|             this._userVerifier.call_answer_query(serviceName, answer, this._cancellable, null); | ||||
| @@ -283,9 +300,10 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|  | ||||
|         this._fprintManager.GetDefaultDeviceRemote(Gio.DBusCallFlags.NONE, this._cancellable, Lang.bind(this, | ||||
|             function(device, error) { | ||||
|                 if (!error && device) | ||||
|                 if (!error && device) { | ||||
|                     this._haveFingerprintReader = true; | ||||
|                     this._updateDefaultService(); | ||||
|                 } | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
| @@ -326,6 +344,7 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|  | ||||
|     _reauthenticationChannelOpened: function(client, result) { | ||||
|         try { | ||||
|             this._clearUserVerifier(); | ||||
|             this._userVerifier = client.open_reauthentication_channel_finish(result); | ||||
|         } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|             return; | ||||
| @@ -349,6 +368,7 @@ const ShellUserVerifier = new Lang.Class({ | ||||
|  | ||||
|     _userVerifierGot: function(client, result) { | ||||
|         try { | ||||
|             this._clearUserVerifier(); | ||||
|             this._userVerifier = client.get_user_verifier_finish(result); | ||||
|         } catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) { | ||||
|             return; | ||||
|   | ||||
| @@ -16,7 +16,9 @@ | ||||
|     <file>misc/fileUtils.js</file> | ||||
|     <file>misc/gnomeSession.js</file> | ||||
|     <file>misc/history.js</file> | ||||
|     <file>misc/ibusManager.js</file> | ||||
|     <file>misc/jsParse.js</file> | ||||
|     <file>misc/keyboardManager.js</file> | ||||
|     <file>misc/loginManager.js</file> | ||||
|     <file>misc/modemManager.js</file> | ||||
|     <file>misc/objectManager.js</file> | ||||
| @@ -25,6 +27,9 @@ | ||||
|     <file>misc/util.js</file> | ||||
|  | ||||
|     <file>perf/core.js</file> | ||||
|     <file>perf/hwtest.js</file> | ||||
|  | ||||
|     <file>portalHelper/main.js</file> | ||||
|  | ||||
|     <file>ui/altTab.js</file> | ||||
|     <file>ui/animation.js</file> | ||||
| @@ -39,6 +44,7 @@ | ||||
|     <file>ui/dash.js</file> | ||||
|     <file>ui/dateMenu.js</file> | ||||
|     <file>ui/dnd.js</file> | ||||
|     <file>ui/edgeDragAction.js</file> | ||||
|     <file>ui/endSessionDialog.js</file> | ||||
|     <file>ui/environment.js</file> | ||||
|     <file>ui/extensionDownloader.js</file> | ||||
| @@ -58,6 +64,7 @@ | ||||
|     <file>ui/modalDialog.js</file> | ||||
|     <file>ui/notificationDaemon.js</file> | ||||
|     <file>ui/osdWindow.js</file> | ||||
|     <file>ui/osdMonitorLabeler.js</file> | ||||
|     <file>ui/overview.js</file> | ||||
|     <file>ui/overviewControls.js</file> | ||||
|     <file>ui/panel.js</file> | ||||
|   | ||||
| @@ -43,7 +43,7 @@ function getCurrentExtension() { | ||||
|     let path = match[1]; | ||||
|     let file = Gio.File.new_for_path(path); | ||||
|  | ||||
|     // Walk up the directory tree, looking for an extesion with | ||||
|     // Walk up the directory tree, looking for an extension with | ||||
|     // the same UUID as a directory name. | ||||
|     while (file != null) { | ||||
|         let extension = extensions[file.get_basename()]; | ||||
|   | ||||
							
								
								
									
										234
									
								
								js/misc/ibusManager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,234 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| try { | ||||
|     var IBus = imports.gi.IBus; | ||||
|     _checkIBusVersion(1, 5, 2); | ||||
|     const IBusCandidatePopup = imports.ui.ibusCandidatePopup; | ||||
| } catch (e) { | ||||
|     var IBus = null; | ||||
|     log(e); | ||||
| } | ||||
|  | ||||
| let _ibusManager = null; | ||||
|  | ||||
| function _checkIBusVersion(requiredMajor, requiredMinor, requiredMicro) { | ||||
|     if ((IBus.MAJOR_VERSION > requiredMajor) || | ||||
|         (IBus.MAJOR_VERSION == requiredMajor && IBus.MINOR_VERSION > requiredMinor) || | ||||
|         (IBus.MAJOR_VERSION == requiredMajor && IBus.MINOR_VERSION == requiredMinor && | ||||
|          IBus.MICRO_VERSION >= requiredMicro)) | ||||
|         return; | ||||
|  | ||||
|     throw "Found IBus version %d.%d.%d but required is %d.%d.%d". | ||||
|         format(IBus.MAJOR_VERSION, IBus.MINOR_VERSION, IBus.MINOR_VERSION, | ||||
|                requiredMajor, requiredMinor, requiredMicro); | ||||
| } | ||||
|  | ||||
| function getIBusManager() { | ||||
|     if (_ibusManager == null) | ||||
|         _ibusManager = new IBusManager(); | ||||
|     return _ibusManager; | ||||
| } | ||||
|  | ||||
| const IBusManager = new Lang.Class({ | ||||
|     Name: 'IBusManager', | ||||
|  | ||||
|     // This is the longest we'll keep the keyboard frozen until an input | ||||
|     // source is active. | ||||
|     _MAX_INPUT_SOURCE_ACTIVATION_TIME: 4000, // ms | ||||
|     _PRELOAD_ENGINES_DELAY_TIME: 30, // sec | ||||
|  | ||||
|     _init: function() { | ||||
|         if (!IBus) | ||||
|             return; | ||||
|  | ||||
|         IBus.init(); | ||||
|  | ||||
|         this._candidatePopup = new IBusCandidatePopup.CandidatePopup(); | ||||
|  | ||||
|         this._panelService = null; | ||||
|         this._engines = {}; | ||||
|         this._ready = false; | ||||
|         this._registerPropertiesId = 0; | ||||
|         this._currentEngineName = null; | ||||
|         this._preloadEnginesId = 0; | ||||
|  | ||||
|         this._ibus = IBus.Bus.new_async(); | ||||
|         this._ibus.connect('connected', Lang.bind(this, this._onConnected)); | ||||
|         this._ibus.connect('disconnected', Lang.bind(this, this._clear)); | ||||
|         // Need to set this to get 'global-engine-changed' emitions | ||||
|         this._ibus.set_watch_ibus_signal(true); | ||||
|         this._ibus.connect('global-engine-changed', Lang.bind(this, this._engineChanged)); | ||||
|  | ||||
|         this._spawn(); | ||||
|     }, | ||||
|  | ||||
|     _spawn: function() { | ||||
|         try { | ||||
|             Gio.Subprocess.new(['ibus-daemon', '--xim', '--panel', 'disable'], | ||||
|                                Gio.SubprocessFlags.NONE); | ||||
|         } catch(e) { | ||||
|             log('Failed to launch ibus-daemon: ' + e.message); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _clear: function() { | ||||
|         if (this._panelService) | ||||
|             this._panelService.destroy(); | ||||
|  | ||||
|         this._panelService = null; | ||||
|         this._candidatePopup.setPanelService(null); | ||||
|         this._engines = {}; | ||||
|         this._ready = false; | ||||
|         this._registerPropertiesId = 0; | ||||
|         this._currentEngineName = null; | ||||
|  | ||||
|         this.emit('ready', false); | ||||
|  | ||||
|         this._spawn(); | ||||
|     }, | ||||
|  | ||||
|     _onConnected: function() { | ||||
|         this._ibus.list_engines_async(-1, null, Lang.bind(this, this._initEngines)); | ||||
|         this._ibus.request_name_async(IBus.SERVICE_PANEL, | ||||
|                                       IBus.BusNameFlag.REPLACE_EXISTING, | ||||
|                                       -1, null, | ||||
|                                       Lang.bind(this, this._initPanelService)); | ||||
|     }, | ||||
|  | ||||
|     _initEngines: function(ibus, result) { | ||||
|         let enginesList = this._ibus.list_engines_async_finish(result); | ||||
|         if (enginesList) { | ||||
|             for (let i = 0; i < enginesList.length; ++i) { | ||||
|                 let name = enginesList[i].get_name(); | ||||
|                 this._engines[name] = enginesList[i]; | ||||
|             } | ||||
|             this._updateReadiness(); | ||||
|         } else { | ||||
|             this._clear(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _initPanelService: function(ibus, result) { | ||||
|         let success = this._ibus.request_name_async_finish(result); | ||||
|         if (success) { | ||||
|             this._panelService = new IBus.PanelService({ connection: this._ibus.get_connection(), | ||||
|                                                          object_path: IBus.PATH_PANEL }); | ||||
|             this._candidatePopup.setPanelService(this._panelService); | ||||
|             this._panelService.connect('update-property', Lang.bind(this, this._updateProperty)); | ||||
|             try { | ||||
|                 // IBus versions older than 1.5.10 have a bug which | ||||
|                 // causes spurious set-content-type emissions when | ||||
|                 // switching input focus that temporarily lose purpose | ||||
|                 // and hints defeating its intended semantics and | ||||
|                 // confusing users. We thus don't use it in that case. | ||||
|                 _checkIBusVersion(1, 5, 10); | ||||
|                 this._panelService.connect('set-content-type', Lang.bind(this, this._setContentType)); | ||||
|             } catch (e) { | ||||
|             } | ||||
|             // If an engine is already active we need to get its properties | ||||
|             this._ibus.get_global_engine_async(-1, null, Lang.bind(this, function(i, result) { | ||||
|                 let engine; | ||||
|                 try { | ||||
|                     engine = this._ibus.get_global_engine_async_finish(result); | ||||
|                     if (!engine) | ||||
|                         return; | ||||
|                 } catch(e) { | ||||
|                     return; | ||||
|                 } | ||||
|                 this._engineChanged(this._ibus, engine.get_name()); | ||||
|             })); | ||||
|             this._updateReadiness(); | ||||
|         } else { | ||||
|             this._clear(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateReadiness: function() { | ||||
|         this._ready = (Object.keys(this._engines).length > 0 && | ||||
|                        this._panelService != null); | ||||
|         this.emit('ready', this._ready); | ||||
|     }, | ||||
|  | ||||
|     _engineChanged: function(bus, engineName) { | ||||
|         if (!this._ready) | ||||
|             return; | ||||
|  | ||||
|         this._currentEngineName = engineName; | ||||
|  | ||||
|         if (this._registerPropertiesId != 0) | ||||
|             return; | ||||
|  | ||||
|         this._registerPropertiesId = | ||||
|             this._panelService.connect('register-properties', Lang.bind(this, function(p, props) { | ||||
|                 if (!props.get(0)) | ||||
|                     return; | ||||
|  | ||||
|                 this._panelService.disconnect(this._registerPropertiesId); | ||||
|                 this._registerPropertiesId = 0; | ||||
|  | ||||
|                 this.emit('properties-registered', this._currentEngineName, props); | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     _updateProperty: function(panel, prop) { | ||||
|         this.emit('property-updated', this._currentEngineName, prop); | ||||
|     }, | ||||
|  | ||||
|     _setContentType: function(panel, purpose, hints) { | ||||
|         this.emit('set-content-type', purpose, hints); | ||||
|     }, | ||||
|  | ||||
|     activateProperty: function(key, state) { | ||||
|         this._panelService.property_activate(key, state); | ||||
|     }, | ||||
|  | ||||
|     getEngineDesc: function(id) { | ||||
|         if (!IBus || !this._ready) | ||||
|             return null; | ||||
|  | ||||
|         return this._engines[id]; | ||||
|     }, | ||||
|  | ||||
|     setEngine: function(id, callback) { | ||||
|         // Send id even if id == this._currentEngineName | ||||
|         // because 'properties-registered' signal can be emitted | ||||
|         // while this._ibusSources == null on a lock screen. | ||||
|         if (!IBus || !this._ready) { | ||||
|             if (callback) | ||||
|                 callback(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._ibus.set_global_engine_async(id, this._MAX_INPUT_SOURCE_ACTIVATION_TIME, | ||||
|                                            null, callback); | ||||
|     }, | ||||
|  | ||||
|     preloadEngines: function(ids) { | ||||
|         if (!IBus || !this._ibus || ids.length == 0) | ||||
|             return; | ||||
|  | ||||
|         if (this._preloadEnginesId != 0) { | ||||
|             Mainloop.source_remove(this._preloadEnginesId); | ||||
|             this._preloadEnginesId = 0; | ||||
|         } | ||||
|  | ||||
|         this._preloadEnginesId = | ||||
|             Mainloop.timeout_add_seconds(this._PRELOAD_ENGINES_DELAY_TIME, | ||||
|                                          Lang.bind(this, function() { | ||||
|                                              this._ibus.preload_engines_async( | ||||
|                                                  ids, | ||||
|                                                  -1, | ||||
|                                                  null, | ||||
|                                                  null); | ||||
|                                              this._preloadEnginesId = 0; | ||||
|                                              return GLib.SOURCE_REMOVE; | ||||
|                                          })); | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(IBusManager.prototype); | ||||
							
								
								
									
										153
									
								
								js/misc/keyboardManager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,153 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const GLib = imports.gi.GLib; | ||||
| const GnomeDesktop = imports.gi.GnomeDesktop; | ||||
| const Lang = imports.lang; | ||||
| const Meta = imports.gi.Meta; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const DEFAULT_LOCALE = 'en_US'; | ||||
| const DEFAULT_LAYOUT = 'us'; | ||||
| const DEFAULT_VARIANT = ''; | ||||
|  | ||||
| let _xkbInfo = null; | ||||
|  | ||||
| function getXkbInfo() { | ||||
|     if (_xkbInfo == null) | ||||
|         _xkbInfo = new GnomeDesktop.XkbInfo(); | ||||
|     return _xkbInfo; | ||||
| } | ||||
|  | ||||
| let _keyboardManager = null; | ||||
|  | ||||
| function getKeyboardManager() { | ||||
|     if (_keyboardManager == null) | ||||
|         _keyboardManager = new KeyboardManager(); | ||||
|     return _keyboardManager; | ||||
| } | ||||
|  | ||||
| function releaseKeyboard() { | ||||
|     if (Main.modalCount > 0) | ||||
|         global.display.unfreeze_keyboard(global.get_current_time()); | ||||
|     else | ||||
|         global.display.ungrab_keyboard(global.get_current_time()); | ||||
| } | ||||
|  | ||||
| function holdKeyboard() { | ||||
|     global.display.freeze_keyboard(global.get_current_time()); | ||||
| } | ||||
|  | ||||
| const KeyboardManager = new Lang.Class({ | ||||
|     Name: 'KeyboardManager', | ||||
|  | ||||
|     // The XKB protocol doesn't allow for more that 4 layouts in a | ||||
|     // keymap. Wayland doesn't impose this limit and libxkbcommon can | ||||
|     // handle up to 32 layouts but since we need to support X clients | ||||
|     // even as a Wayland compositor, we can't bump this. | ||||
|     MAX_LAYOUTS_PER_GROUP: 4, | ||||
|  | ||||
|     _init: function() { | ||||
|         this._xkbInfo = getXkbInfo(); | ||||
|         this._current = null; | ||||
|         this._localeLayoutInfo = this._getLocaleLayout(); | ||||
|         this._layoutInfos = {}; | ||||
|     }, | ||||
|  | ||||
|     _applyLayoutGroup: function(group) { | ||||
|         let options = this._buildOptionsString(); | ||||
|         let [layouts, variants] = this._buildGroupStrings(group); | ||||
|         Meta.get_backend().set_keymap(layouts, variants, options); | ||||
|     }, | ||||
|  | ||||
|     _applyLayoutGroupIndex: function(idx) { | ||||
|         Meta.get_backend().lock_layout_group(idx); | ||||
|     }, | ||||
|  | ||||
|     apply: function(id) { | ||||
|         let info = this._layoutInfos[id]; | ||||
|         if (!info) | ||||
|             return; | ||||
|  | ||||
|         if (this._current && this._current.group == info.group) { | ||||
|             if (this._current.groupIndex != info.groupIndex) | ||||
|                 this._applyLayoutGroupIndex(info.groupIndex); | ||||
|         } else { | ||||
|             this._applyLayoutGroup(info.group); | ||||
|             this._applyLayoutGroupIndex(info.groupIndex); | ||||
|         } | ||||
|  | ||||
|         this._current = info; | ||||
|     }, | ||||
|  | ||||
|     reapply: function() { | ||||
|         if (!this._current) | ||||
|             return; | ||||
|  | ||||
|         this._applyLayoutGroup(this._current.group); | ||||
|         this._applyLayoutGroupIndex(this._current.groupIndex); | ||||
|     }, | ||||
|  | ||||
|     setUserLayouts: function(ids) { | ||||
|         this._current = null; | ||||
|         this._layoutInfos = {}; | ||||
|  | ||||
|         for (let i = 0; i < ids.length; ++i) { | ||||
|             let [found, , , _layout, _variant] = this._xkbInfo.get_layout_info(ids[i]); | ||||
|             if (found) | ||||
|                 this._layoutInfos[ids[i]] = { id: ids[i], layout: _layout, variant: _variant }; | ||||
|         } | ||||
|  | ||||
|         let i = 0; | ||||
|         let group = []; | ||||
|         for (let id in this._layoutInfos) { | ||||
|             // We need to leave one slot on each group free so that we | ||||
|             // can add a layout containing the symbols for the | ||||
|             // language used in UI strings to ensure that toolkits can | ||||
|             // handle mnemonics like Alt+Ф even if the user is | ||||
|             // actually typing in a different layout. | ||||
|             let groupIndex = i % (this.MAX_LAYOUTS_PER_GROUP - 1); | ||||
|             if (groupIndex == 0) | ||||
|                 group = []; | ||||
|  | ||||
|             let info = this._layoutInfos[id]; | ||||
|             group[groupIndex] = info; | ||||
|             info.group = group; | ||||
|             info.groupIndex = groupIndex; | ||||
|  | ||||
|             i += 1; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getLocaleLayout: function() { | ||||
|         let locale = GLib.get_language_names()[0]; | ||||
|         if (locale.indexOf('_') == -1) | ||||
|             locale = DEFAULT_LOCALE; | ||||
|  | ||||
|         let [found, , id] = GnomeDesktop.get_input_source_from_locale(locale); | ||||
|         if (!found) | ||||
|             [, , id] = GnomeDesktop.get_input_source_from_locale(DEFAULT_LOCALE); | ||||
|  | ||||
|         let [found, , , _layout, _variant] = this._xkbInfo.get_layout_info(id); | ||||
|         if (found) | ||||
|             return { layout: _layout, variant: _variant }; | ||||
|         else | ||||
|             return { layout: DEFAULT_LAYOUT, variant: DEFAULT_VARIANT }; | ||||
|     }, | ||||
|  | ||||
|     _buildGroupStrings: function(_group) { | ||||
|         let group = _group.concat(this._localeLayoutInfo); | ||||
|         let layouts = group.map(function(g) { return g.layout; }).join(','); | ||||
|         let variants = group.map(function(g) { return g.variant; }).join(','); | ||||
|         return [layouts, variants]; | ||||
|     }, | ||||
|  | ||||
|     setKeyboardOptions: function(options) { | ||||
|         this._xkbOptions = options; | ||||
|     }, | ||||
|  | ||||
|     _buildOptionsString: function() { | ||||
|         let options = this._xkbOptions.join(','); | ||||
|         return options; | ||||
|     } | ||||
| }); | ||||
| @@ -72,6 +72,9 @@ function run() { | ||||
|     Scripting.defineScriptEvent("applicationsShowStart", "Starting to switch to applications view"); | ||||
|     Scripting.defineScriptEvent("applicationsShowDone", "Done switching to applications view"); | ||||
|  | ||||
|     // Enable recording of timestamps for different points in the frame cycle | ||||
|     global.frame_timestamps = true; | ||||
|  | ||||
|     Main.overview.connect('shown', function() { | ||||
|                               Scripting.scriptEvent('overviewShowDone'); | ||||
|                           }); | ||||
| @@ -87,7 +90,10 @@ function run() { | ||||
|             yield Scripting.destroyTestWindows(); | ||||
|  | ||||
|             for (let k = 0; k < config.count; k++) | ||||
|                 yield Scripting.createTestWindow(config.width, config.height, config.alpha, config.maximized); | ||||
|                 yield Scripting.createTestWindow({ width: config.width, | ||||
|                                                    height: config.height, | ||||
|                                                    alpha: config.alpha, | ||||
|                                                    maximized: config.maximized }); | ||||
|  | ||||
|             yield Scripting.waitTestWindows(); | ||||
|             yield Scripting.sleep(1000); | ||||
|   | ||||
							
								
								
									
										308
									
								
								js/perf/hwtest.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,308 @@ | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Main = imports.ui.main; | ||||
| const Scripting = imports.ui.scripting; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| let METRICS = { | ||||
|     timeToDesktop: | ||||
|     { description: "Time from starting graphical.target to desktop showing", | ||||
|       units: "us" }, | ||||
|  | ||||
|     overviewShowTime: | ||||
|     { description: "Time to switch to overview view, first time", | ||||
|       units: "us" }, | ||||
|  | ||||
|     applicationsShowTime: | ||||
|     { description: "Time to switch to applications view, first time", | ||||
|       units: "us" }, | ||||
|  | ||||
|     mainViewRedrawTime: | ||||
|     { description: "Time to redraw the main view, full screen", | ||||
|       units: "us" }, | ||||
|  | ||||
|     overviewRedrawTime: | ||||
|     { description: "Time to redraw the overview, full screen, 5 windows", | ||||
|       units: "us" }, | ||||
|  | ||||
|     applicationRedrawTime: | ||||
|     { description: "Time to redraw frame with a maximized application update", | ||||
|       units: "us" }, | ||||
|  | ||||
|     geditStartTime: | ||||
|     { description: "Time from gedit launch to window drawn", | ||||
|       units: "us" }, | ||||
| } | ||||
|  | ||||
| function waitAndDraw(milliseconds) { | ||||
|     let cb; | ||||
|  | ||||
|     let timeline = new Clutter.Timeline({ duration: milliseconds }); | ||||
|     timeline.start(); | ||||
|  | ||||
|     timeline.connect('new-frame', | ||||
|         function(timeline, frame) { | ||||
|             global.stage.queue_redraw(); | ||||
|         }); | ||||
|  | ||||
|     timeline.connect('completed', | ||||
|         function() { | ||||
|             timeline.stop(); | ||||
|             if (cb) | ||||
|                 cb(); | ||||
|         }); | ||||
|  | ||||
|     return function(callback) { | ||||
|         cb = callback; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| function waitSignal(object, signal) { | ||||
|     let cb; | ||||
|  | ||||
|     let id = object.connect(signal, function() { | ||||
|         object.disconnect(id); | ||||
|         if (cb) | ||||
|             cb(); | ||||
|     }); | ||||
|  | ||||
|     return function(callback) { | ||||
|         cb = callback; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| function extractBootTimestamp() { | ||||
|     let sp = Gio.Subprocess.new(['journalctl', '-b', | ||||
|                                  'MESSAGE_ID=7d4958e842da4a758f6c1cdc7b36dcc5', | ||||
|                                  'UNIT=graphical.target', | ||||
|                                  '-o', | ||||
|                                  'json'], | ||||
|                                 Gio.SubprocessFlags.STDOUT_PIPE); | ||||
|     let result = null; | ||||
|  | ||||
|     let datastream = Gio.DataInputStream.new(sp.get_stdout_pipe()); | ||||
|     while (true) { | ||||
|         let [line, length] = datastream.read_line_utf8(null); | ||||
|         if (line === null) | ||||
|             break; | ||||
|  | ||||
|         let fields = JSON.parse(line); | ||||
|         result = Number(fields['__MONOTONIC_TIMESTAMP']); | ||||
|     } | ||||
|     datastream.close(null); | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| function run() { | ||||
|     Scripting.defineScriptEvent("desktopShown", "Finished initial animation"); | ||||
|     Scripting.defineScriptEvent("overviewShowStart", "Starting to show the overview"); | ||||
|     Scripting.defineScriptEvent("overviewShowDone", "Overview finished showing"); | ||||
|     Scripting.defineScriptEvent("applicationsShowStart", "Starting to switch to applications view"); | ||||
|     Scripting.defineScriptEvent("applicationsShowDone", "Done switching to applications view"); | ||||
|     Scripting.defineScriptEvent("mainViewDrawStart", "Drawing main view"); | ||||
|     Scripting.defineScriptEvent("mainViewDrawDone", "Ending timing main view drawing"); | ||||
|     Scripting.defineScriptEvent("overviewDrawStart", "Drawing overview"); | ||||
|     Scripting.defineScriptEvent("overviewDrawDone", "Ending timing overview drawing"); | ||||
|     Scripting.defineScriptEvent("redrawTestStart", "Drawing application window"); | ||||
|     Scripting.defineScriptEvent("redrawTestDone", "Ending timing application window drawing"); | ||||
|     Scripting.defineScriptEvent("collectTimings", "Accumulate frame timings from redraw tests"); | ||||
|     Scripting.defineScriptEvent("geditLaunch", "gedit application launch"); | ||||
|     Scripting.defineScriptEvent("geditFirstFrame", "first frame of gedit window drawn"); | ||||
|  | ||||
|     yield Scripting.waitLeisure(); | ||||
|     Scripting.scriptEvent('desktopShown'); | ||||
|  | ||||
|     Gtk.Settings.get_default().gtk_enable_animations = false; | ||||
|  | ||||
|     Scripting.scriptEvent('overviewShowStart'); | ||||
|     Main.overview.show(); | ||||
|     yield Scripting.waitLeisure(); | ||||
|     Scripting.scriptEvent('overviewShowDone'); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Scripting.scriptEvent('applicationsShowStart'); | ||||
|     Main.overview._dash.showAppsButton.checked = true; | ||||
|  | ||||
|     yield Scripting.waitLeisure(); | ||||
|     Scripting.scriptEvent('applicationsShowDone'); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Main.overview.hide(); | ||||
|     yield Scripting.waitLeisure(); | ||||
|  | ||||
|     //////////////////////////////////////// | ||||
|     // Tests of redraw speed | ||||
|     //////////////////////////////////////// | ||||
|  | ||||
|     global.frame_timestamps = true; | ||||
|     global.frame_finish_timestamp = true; | ||||
|  | ||||
|     for (let k = 0; k < 5; k++) | ||||
|         yield Scripting.createTestWindow(640, 480, | ||||
|                                          { maximized: true }); | ||||
|     yield Scripting.waitTestWindows(); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Scripting.scriptEvent('mainViewDrawStart'); | ||||
|     yield waitAndDraw(1000); | ||||
|     Scripting.scriptEvent('mainViewDrawDone'); | ||||
|  | ||||
|     Main.overview.show(); | ||||
|     Scripting.waitLeisure(); | ||||
|  | ||||
|     yield Scripting.sleep(1500); | ||||
|  | ||||
|     Scripting.scriptEvent('overviewDrawStart'); | ||||
|     yield waitAndDraw(1000); | ||||
|     Scripting.scriptEvent('overviewDrawDone'); | ||||
|  | ||||
|     yield Scripting.destroyTestWindows(); | ||||
|     Main.overview.hide(); | ||||
|  | ||||
|     yield Scripting.createTestWindow(640, 480, | ||||
|                                      { maximized: true, | ||||
|                                        redraws: true}); | ||||
|     yield Scripting.waitTestWindows(); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Scripting.scriptEvent('redrawTestStart'); | ||||
|     yield Scripting.sleep(1000); | ||||
|     Scripting.scriptEvent('redrawTestDone'); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|     Scripting.scriptEvent('collectTimings'); | ||||
|  | ||||
|     yield Scripting.destroyTestWindows(); | ||||
|  | ||||
|     global.frame_timestamps = false; | ||||
|     global.frame_finish_timestamp = false; | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     //////////////////////////////////////// | ||||
|  | ||||
|     let appSys = Shell.AppSystem.get_default(); | ||||
|     let app = appSys.lookup_app('org.gnome.gedit.desktop'); | ||||
|  | ||||
|     Scripting.scriptEvent('geditLaunch'); | ||||
|     app.activate(); | ||||
|  | ||||
|     let windows = app.get_windows(); | ||||
|     if (windows.length > 0) | ||||
|         throw new Error('gedit was already running'); | ||||
|  | ||||
|     while (windows.length == 0) { | ||||
|         yield waitSignal(global.display, 'window-created'); | ||||
|         windows = app.get_windows(); | ||||
|     } | ||||
|  | ||||
|     let actor = windows[0].get_compositor_private(); | ||||
|     yield waitSignal(actor, 'first-frame'); | ||||
|     Scripting.scriptEvent('geditFirstFrame'); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     windows[0].delete(global.get_current_time()); | ||||
|  | ||||
|     yield Scripting.sleep(1000); | ||||
|  | ||||
|     Gtk.Settings.get_default().gtk_enable_animations = true; | ||||
| } | ||||
|  | ||||
| let overviewShowStart; | ||||
| let applicationsShowStart; | ||||
| let stagePaintStart; | ||||
| let redrawTiming; | ||||
| let redrawTimes = {}; | ||||
| let geditLaunchTime; | ||||
|  | ||||
| function script_desktopShown(time) { | ||||
|     let bootTimestamp = extractBootTimestamp(); | ||||
|     METRICS.timeToDesktop.value = time - bootTimestamp; | ||||
| } | ||||
|  | ||||
| function script_overviewShowStart(time) { | ||||
|     overviewShowStart = time; | ||||
| } | ||||
|  | ||||
| function script_overviewShowDone(time) { | ||||
|     METRICS.overviewShowTime.value = time - overviewShowStart; | ||||
| } | ||||
|  | ||||
| function script_applicationsShowStart(time) { | ||||
|     applicationsShowStart = time; | ||||
| } | ||||
|  | ||||
| function script_applicationsShowDone(time) { | ||||
|     METRICS.applicationsShowTime.value = time - applicationsShowStart; | ||||
| } | ||||
|  | ||||
| function script_mainViewDrawStart(time) { | ||||
|     redrawTiming = 'mainView'; | ||||
| } | ||||
|  | ||||
| function script_mainViewDrawDone(time) { | ||||
|     redrawTiming = null; | ||||
| } | ||||
|  | ||||
| function script_overviewDrawStart(time) { | ||||
|     redrawTiming = 'overview'; | ||||
| } | ||||
|  | ||||
| function script_overviewDrawDone(time) { | ||||
|     redrawTiming = null; | ||||
| } | ||||
|  | ||||
| function script_redrawTestStart(time) { | ||||
|     redrawTiming = 'application'; | ||||
| } | ||||
|  | ||||
| function script_redrawTestDone(time) { | ||||
|     redrawTiming = null; | ||||
| } | ||||
|  | ||||
| function script_collectTimings(time) { | ||||
|     for (let timing in redrawTimes) { | ||||
|         let times = redrawTimes[timing]; | ||||
|         times.sort(function(a, b) { return a - b }); | ||||
|  | ||||
|         let len = times.length; | ||||
|         let median; | ||||
|  | ||||
|         if (len == 0) | ||||
|             median = -1; | ||||
|         else if (len % 2 == 1) | ||||
|             median = times[(len - 1)/ 2]; | ||||
|         else | ||||
|             median = Math.round((times[len / 2 - 1] + times[len / 2]) / 2); | ||||
|  | ||||
|         METRICS[timing + 'RedrawTime'].value = median; | ||||
|     } | ||||
| } | ||||
|  | ||||
| function script_geditLaunch(time) { | ||||
|     geditLaunchTime = time; | ||||
| } | ||||
|  | ||||
| function script_geditFirstFrame(time) { | ||||
|     METRICS.geditStartTime.value = time - geditLaunchTime; | ||||
| } | ||||
|  | ||||
| function clutter_stagePaintStart(time) { | ||||
|     stagePaintStart = time; | ||||
| } | ||||
|  | ||||
| function clutter_paintCompletedTimestamp(time) { | ||||
|     if (redrawTiming != null && stagePaintStart != null) { | ||||
|         if (!(redrawTiming in redrawTimes)) | ||||
|             redrawTimes[redrawTiming] = []; | ||||
|         redrawTimes[redrawTiming].push(time - stagePaintStart); | ||||
|     } | ||||
|     stagePaintStart = null; | ||||
| } | ||||
							
								
								
									
										247
									
								
								js/portalHelper/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,247 @@ | ||||
| const Format = imports.format; | ||||
| const Gettext = imports.gettext; | ||||
| const GLib = imports.gi.GLib; | ||||
| const GObject = imports.gi.GObject; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Pango = imports.gi.Pango; | ||||
| const Soup = imports.gi.Soup; | ||||
| const WebKit = imports.gi.WebKit2; | ||||
|  | ||||
| const _ = Gettext.gettext; | ||||
|  | ||||
| const Config = imports.misc.config; | ||||
|  | ||||
| const PortalHelperResult = { | ||||
|     CANCELLED: 0, | ||||
|     COMPLETED: 1, | ||||
|     RECHECK: 2 | ||||
| }; | ||||
|  | ||||
| const INACTIVITY_TIMEOUT = 30000; //ms | ||||
| const CONNECTIVITY_RECHECK_RATELIMIT_TIMEOUT = 30 * GLib.USEC_PER_SEC; | ||||
|  | ||||
| const HelperDBusInterface = '<node> \ | ||||
| <interface name="org.gnome.Shell.PortalHelper"> \ | ||||
| <method name="Authenticate"> \ | ||||
|     <arg type="o" direction="in" name="connection" /> \ | ||||
|     <arg type="s" direction="in" name="url" /> \ | ||||
|     <arg type="u" direction="in" name="timestamp" /> \ | ||||
| </method> \ | ||||
| <method name="Close"> \ | ||||
|     <arg type="o" direction="in" name="connection" /> \ | ||||
| </method> \ | ||||
| <method name="Refresh"> \ | ||||
|     <arg type="o" direction="in" name="connection" /> \ | ||||
| </method> \ | ||||
| <signal name="Done"> \ | ||||
|     <arg type="o" name="connection" /> \ | ||||
|     <arg type="u" name="result" /> \ | ||||
| </signal> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
| const PortalWindow = new Lang.Class({ | ||||
|     Name: 'PortalWindow', | ||||
|     Extends: Gtk.ApplicationWindow, | ||||
|  | ||||
|     _init: function(application, url, timestamp, doneCallback) { | ||||
|         this.parent({ application: application }); | ||||
|  | ||||
|         if (!url) { | ||||
|             url = 'http://www.gnome.org'; | ||||
|             this._originalUrlWasGnome = true; | ||||
|         } else { | ||||
|             this._originalUrlWasGnome = false; | ||||
|         } | ||||
|         this._uri = new Soup.URI(url); | ||||
|         this._everSeenRedirect = false; | ||||
|         this._originalUrl = url; | ||||
|         this._doneCallback = doneCallback; | ||||
|         this._lastRecheck = 0; | ||||
|         this._recheckAtExit = false; | ||||
|  | ||||
|         this._webView = new WebKit.WebView(); | ||||
|         this._webView.connect('decide-policy', Lang.bind(this, this._onDecidePolicy)); | ||||
|         this._webView.load_uri(url); | ||||
|         this._webView.connect('notify::title', Lang.bind(this, this._syncTitle)); | ||||
|         this._syncTitle(); | ||||
|  | ||||
|         this.add(this._webView); | ||||
|         this._webView.show(); | ||||
|         this.maximize(); | ||||
|         this.present_with_time(timestamp); | ||||
|     }, | ||||
|  | ||||
|     _syncTitle: function() { | ||||
|         let title = this._webView.title; | ||||
|  | ||||
|         if (title) { | ||||
|             this.title = title; | ||||
|         } else { | ||||
|             /* TRANSLATORS: this is the title of the wifi captive portal login | ||||
|              * window, until we know the title of the actual login page */ | ||||
|             this.title = _("Web Authentication Redirect"); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     refresh: function() { | ||||
|         this._everSeenRedirect = false; | ||||
|         this._webView.load_uri(this._originalUrl); | ||||
|     }, | ||||
|  | ||||
|     vfunc_delete_event: function(event) { | ||||
|         if (this._recheckAtExit) | ||||
|             this._doneCallback(PortalHelperResult.RECHECK); | ||||
|         else | ||||
|             this._doneCallback(PortalHelperResult.CANCELLED); | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _onDecidePolicy: function(view, decision, type) { | ||||
|         if (type == WebKit.PolicyDecisionType.NEW_WINDOW_ACTION) { | ||||
|             decision.ignore(); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         if (type != WebKit.PolicyDecisionType.NAVIGATION_ACTION) | ||||
|             return false; | ||||
|  | ||||
|         let request = decision.get_request(); | ||||
|         let uri = new Soup.URI(request.get_uri()); | ||||
|  | ||||
|         if (!uri.host_equal(this._uri) && this._originalUrlWasGnome) { | ||||
|             if (uri.get_host() == 'www.gnome.org' && this._everSeenRedirect) { | ||||
|                 // Yay, we got to gnome! | ||||
|                 decision.ignore(); | ||||
|                 this._doneCallback(PortalHelperResult.COMPLETED); | ||||
|                 return true; | ||||
|             } else if (uri.get_host() != 'www.gnome.org') { | ||||
|                 this._everSeenRedirect = true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // We *may* have finished here, but we don't know for | ||||
|         // sure. Tell gnome-shell to run another connectivity check | ||||
|         // (but ratelimit the checks, we don't want to spam | ||||
|         // nmcheck.gnome.org for portals that have 10 or more internal | ||||
|         // redirects - and unfortunately they exist) | ||||
|         // If we hit the rate limit, we also queue a recheck | ||||
|         // when the window is closed, just in case we miss the | ||||
|         // final check and don't realize we're connected | ||||
|         // This should not be a problem in the cancelled logic, | ||||
|         // because if the user doesn't want to start the login, | ||||
|         // we should not see any redirect at all, outside this._uri | ||||
|  | ||||
|         let now = GLib.get_monotonic_time(); | ||||
|         let shouldRecheck = (now - this._lastRecheck) > | ||||
|             CONNECTIVITY_RECHECK_RATELIMIT_TIMEOUT; | ||||
|  | ||||
|         if (shouldRecheck) { | ||||
|             this._lastRecheck = now; | ||||
|             this._recheckAtExit = false; | ||||
|             this._doneCallback(PortalHelperResult.RECHECK); | ||||
|         } else { | ||||
|             this._recheckAtExit = true; | ||||
|         } | ||||
|  | ||||
|         // Update the URI, in case of chained redirects, so we still | ||||
|         // think we're doing the login until gnome-shell kills us | ||||
|         this._uri = uri; | ||||
|  | ||||
|         decision.use(); | ||||
|         return true; | ||||
|     }, | ||||
| }); | ||||
|  | ||||
| const WebPortalHelper = new Lang.Class({ | ||||
|     Name: 'WebPortalHelper', | ||||
|     Extends: Gtk.Application, | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent({ application_id: 'org.gnome.Shell.PortalHelper', | ||||
|                       flags: Gio.ApplicationFlags.IS_SERVICE, | ||||
|                       inactivity_timeout: 30000 }); | ||||
|  | ||||
|         this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(HelperDBusInterface, this); | ||||
|         this._queue = []; | ||||
|     }, | ||||
|  | ||||
|     vfunc_dbus_register: function(connection, path) { | ||||
|         this._dbusImpl.export(connection, path); | ||||
|         this.parent(connection, path); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     vfunc_dbus_unregister: function(connection, path) { | ||||
|         this._dbusImpl.unexport_from_connection(connection); | ||||
|         this.parent(connection, path); | ||||
|     }, | ||||
|  | ||||
|     vfunc_activate: function() { | ||||
|         // If launched manually (for example for testing), force a dummy authentication | ||||
|         // session with the default url | ||||
|         this.Authenticate('/org/gnome/dummy', '', 0); | ||||
|     }, | ||||
|  | ||||
|     Authenticate: function(connection, url, timestamp) { | ||||
|         this._queue.push({ connection: connection, url: url, timestamp: timestamp }); | ||||
|  | ||||
|         this._processQueue(); | ||||
|     }, | ||||
|  | ||||
|     Close: function(connection) { | ||||
|         for (let i = 0; i < this._queue.length; i++) { | ||||
|             let obj = this._queue[i]; | ||||
|  | ||||
|             if (obj.connection == connection) { | ||||
|                 if (obj.window) | ||||
|                     obj.window.destroy(); | ||||
|                 this._queue.splice(i, 1); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this._processQueue(); | ||||
|     }, | ||||
|  | ||||
|     Refresh: function(connection) { | ||||
|         for (let i = 0; i < this._queue.length; i++) { | ||||
|             let obj = this._queue[i]; | ||||
|  | ||||
|             if (obj.connection == connection) { | ||||
|                 if (obj.window) | ||||
|                     obj.window.refresh(); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _processQueue: function() { | ||||
|         if (this._queue.length == 0) | ||||
|             return; | ||||
|  | ||||
|         let top = this._queue[0]; | ||||
|         if (top.window != null) | ||||
|             return; | ||||
|  | ||||
|         top.window = new PortalWindow(this, top.uri, top.timestamp, Lang.bind(this, function(result) { | ||||
|             this._dbusImpl.emit_signal('Done', new GLib.Variant('(ou)', [top.connection, result])); | ||||
|         })); | ||||
|     }, | ||||
| }); | ||||
|  | ||||
| function initEnvironment() { | ||||
|     String.prototype.format = Format.format; | ||||
| } | ||||
|  | ||||
| function main(argv) { | ||||
|     initEnvironment(); | ||||
|  | ||||
|     Gettext.bindtextdomain(Config.GETTEXT_PACKAGE, Config.LOCALEDIR); | ||||
|     Gettext.textdomain(Config.GETTEXT_PACKAGE); | ||||
|  | ||||
|     let app = new WebPortalHelper(); | ||||
|     return app.run(argv); | ||||
| } | ||||
| @@ -58,6 +58,14 @@ const AppSwitcherPopup = new Lang.Class({ | ||||
|         this._currentWindow = -1; | ||||
|  | ||||
|         this.thumbnailsVisible = false; | ||||
|  | ||||
|         let apps = Shell.AppSystem.get_default().get_running (); | ||||
|  | ||||
|         if (apps.length == 0) | ||||
|             return; | ||||
|  | ||||
|         this._switcherList = new AppSwitcher(apps, this); | ||||
|         this._items = this._switcherList.icons; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function (actor, box, flags) { | ||||
| @@ -73,7 +81,6 @@ const AppSwitcherPopup = new Lang.Class({ | ||||
|             let leftPadding = this.actor.get_theme_node().get_padding(St.Side.LEFT); | ||||
|             let rightPadding = this.actor.get_theme_node().get_padding(St.Side.RIGHT); | ||||
|             let bottomPadding = this.actor.get_theme_node().get_padding(St.Side.BOTTOM); | ||||
|             let vPadding = this.actor.get_theme_node().get_vertical_padding(); | ||||
|             let hPadding = leftPadding + rightPadding; | ||||
|  | ||||
|             let icon = this._items[this._selectedIndex].actor; | ||||
| @@ -99,20 +106,6 @@ const AppSwitcherPopup = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _createSwitcher: function() { | ||||
|         let apps = Shell.AppSystem.get_default().get_running (); | ||||
|  | ||||
|         if (apps.length == 0) | ||||
|             return false; | ||||
|  | ||||
|         this._switcherList = new AppSwitcher(apps, this); | ||||
|         this._items = this._switcherList.icons; | ||||
|         if (this._items.length == 0) | ||||
|             return false; | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _initialSelection: function(backward, binding) { | ||||
|         if (binding == 'switch-group') { | ||||
|             if (backward) { | ||||
| @@ -151,13 +144,13 @@ const AppSwitcherPopup = new Lang.Class({ | ||||
|                                  this._items[this._selectedIndex].cachedWindows.length); | ||||
|     }, | ||||
|  | ||||
|     _keyPressHandler: function(keysym, backwards, action) { | ||||
|     _keyPressHandler: function(keysym, action) { | ||||
|         if (action == Meta.KeyBindingAction.SWITCH_GROUP) { | ||||
|             this._select(this._selectedIndex, backwards ? this._previousWindow() : this._nextWindow()); | ||||
|             this._select(this._selectedIndex, this._nextWindow()); | ||||
|         } else if (action == Meta.KeyBindingAction.SWITCH_GROUP_BACKWARD) { | ||||
|             this._select(this._selectedIndex, this._previousWindow()); | ||||
|         } else if (action == Meta.KeyBindingAction.SWITCH_APPLICATIONS) { | ||||
|             this._select(backwards ? this._previous() : this._next()); | ||||
|             this._select(this._next()); | ||||
|         } else if (action == Meta.KeyBindingAction.SWITCH_APPLICATIONS_BACKWARD) { | ||||
|             this._select(this._previous()); | ||||
|         } else if (this._thumbnailsFocused) { | ||||
| @@ -365,44 +358,28 @@ const WindowSwitcherPopup = new Lang.Class({ | ||||
|     Name: 'WindowSwitcherPopup', | ||||
|     Extends: SwitcherPopup.SwitcherPopup, | ||||
|  | ||||
|     _init: function(items) { | ||||
|         this.parent(items); | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.shell.window-switcher' }); | ||||
|     }, | ||||
|     _init: function() { | ||||
|         this.parent(); | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell.window-switcher' }); | ||||
|  | ||||
|     _getWindowList: function() { | ||||
|         let workspace = this._settings.get_boolean('current-workspace-only') ? global.screen.get_active_workspace() : null; | ||||
|         return global.display.get_tab_list(Meta.TabList.NORMAL, global.screen, workspace); | ||||
|     }, | ||||
|  | ||||
|     _createSwitcher: function() { | ||||
|         let windows = this._getWindowList(); | ||||
|  | ||||
|         if (windows.length == 0) | ||||
|             return false; | ||||
|             return; | ||||
|  | ||||
|         let mode = this._settings.get_enum('app-icon-mode'); | ||||
|         this._switcherList = new WindowList(windows, mode); | ||||
|         this._items = this._switcherList.icons; | ||||
|  | ||||
|         if (this._items.length == 0) | ||||
|             return false; | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _initialSelection: function(backward, binding) { | ||||
|         if (binding == 'switch-windows-backward' || backward) | ||||
|             this._select(this._items.length - 1); | ||||
|         else if (this._items.length == 1) | ||||
|             this._select(0); | ||||
|         else | ||||
|             this._select(1); | ||||
|     _getWindowList: function() { | ||||
|         let workspace = this._settings.get_boolean('current-workspace-only') ? global.screen.get_active_workspace() : null; | ||||
|         return global.display.get_tab_list(Meta.TabList.NORMAL, workspace); | ||||
|     }, | ||||
|  | ||||
|     _keyPressHandler: function(keysym, backwards, action) { | ||||
|     _keyPressHandler: function(keysym, action) { | ||||
|         if (action == Meta.KeyBindingAction.SWITCH_WINDOWS) { | ||||
|             this._select(backwards ? this._previous() : this._next()); | ||||
|             this._select(this._next()); | ||||
|         } else if (action == Meta.KeyBindingAction.SWITCH_WINDOWS_BACKWARD) { | ||||
|             this._select(this._previous()); | ||||
|         } else { | ||||
| @@ -456,11 +433,10 @@ const AppSwitcher = new Lang.Class({ | ||||
|         this._arrows = []; | ||||
|  | ||||
|         let windowTracker = Shell.WindowTracker.get_default(); | ||||
|         let settings = new Gio.Settings({ schema: 'org.gnome.shell.app-switcher' }); | ||||
|         let settings = new Gio.Settings({ schema_id: 'org.gnome.shell.app-switcher' }); | ||||
|         let workspace = settings.get_boolean('current-workspace-only') ? global.screen.get_active_workspace() | ||||
|                                                                        : null; | ||||
|         let allWindows = global.display.get_tab_list(Meta.TabList.NORMAL, | ||||
|                                                      global.screen, workspace); | ||||
|         let allWindows = global.display.get_tab_list(Meta.TabList.NORMAL, workspace); | ||||
|  | ||||
|         // Construct the AppIcons, add to the popup | ||||
|         for (let i = 0; i < apps.length; i++) { | ||||
|   | ||||
| @@ -12,7 +12,7 @@ const ANIMATED_ICON_UPDATE_TIMEOUT = 100; | ||||
| const Animation = new Lang.Class({ | ||||
|     Name: 'Animation', | ||||
|  | ||||
|     _init: function(filename, width, height, speed) { | ||||
|     _init: function(file, width, height, speed) { | ||||
|         this.actor = new St.Bin(); | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|         this._speed = speed; | ||||
| @@ -23,7 +23,7 @@ const Animation = new Lang.Class({ | ||||
|         this._frame = 0; | ||||
|  | ||||
|         let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|         this._animations = St.TextureCache.get_default().load_sliced_image (filename, width, height, scaleFactor, | ||||
|         this._animations = St.TextureCache.get_default().load_sliced_image (file, width, height, scaleFactor, | ||||
|                                                                             Lang.bind(this, this._animationsLoaded)); | ||||
|         this.actor.set_child(this._animations); | ||||
|     }, | ||||
| @@ -82,7 +82,7 @@ const AnimatedIcon = new Lang.Class({ | ||||
|     Name: 'AnimatedIcon', | ||||
|     Extends: Animation, | ||||
|  | ||||
|     _init: function(filename, size) { | ||||
|         this.parent(filename, size, size, ANIMATED_ICON_UPDATE_TIMEOUT); | ||||
|     _init: function(file, size) { | ||||
|         this.parent(file, size, size, ANIMATED_ICON_UPDATE_TIMEOUT); | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -16,6 +16,7 @@ const Atk = imports.gi.Atk; | ||||
| const AppFavorites = imports.ui.appFavorites; | ||||
| const BoxPointer = imports.ui.boxpointer; | ||||
| const DND = imports.ui.dnd; | ||||
| const GrabHelper = imports.ui.grabHelper; | ||||
| const IconGrid = imports.ui.iconGrid; | ||||
| const Main = imports.ui.main; | ||||
| const Overview = imports.ui.overview; | ||||
| @@ -33,7 +34,9 @@ const MIN_COLUMNS = 4; | ||||
| const MIN_ROWS = 4; | ||||
|  | ||||
| const INACTIVE_GRID_OPACITY = 77; | ||||
| const INACTIVE_GRID_OPACITY_ANIMATION_TIME = 0.40; | ||||
| // This time needs to be less than IconGrid.EXTRA_SPACE_ANIMATION_TIME | ||||
| // to not clash with other animations | ||||
| const INACTIVE_GRID_OPACITY_ANIMATION_TIME = 0.24; | ||||
| const FOLDER_SUBICON_FRACTION = .4; | ||||
|  | ||||
| const MIN_FREQUENT_APPS_COUNT = 3; | ||||
| @@ -42,6 +45,16 @@ const INDICATORS_BASE_TIME = 0.25; | ||||
| const INDICATORS_ANIMATION_DELAY = 0.125; | ||||
| const INDICATORS_ANIMATION_MAX_TIME = 0.75; | ||||
|  | ||||
| // Follow iconGrid animations approach and divide by 2 to animate out to | ||||
| // not annoy the user when the user wants to quit appDisplay. | ||||
| // Also, make sure we don't exceed iconGrid animation total time or | ||||
| // views switch time. | ||||
| const INDICATORS_BASE_TIME_OUT = 0.125; | ||||
| const INDICATORS_ANIMATION_DELAY_OUT = 0.0625; | ||||
| const INDICATORS_ANIMATION_MAX_TIME_OUT = | ||||
|     Math.min (VIEWS_SWITCH_TIME, | ||||
|               IconGrid.ANIMATION_TIME_OUT + IconGrid.ANIMATION_MAX_DELAY_OUT_FOR_ITEM); | ||||
|  | ||||
| const PAGE_SWITCH_TIME = 0.3; | ||||
|  | ||||
| const VIEWS_SWITCH_TIME = 0.4; | ||||
| @@ -176,26 +189,98 @@ const BaseAppView = new Lang.Class({ | ||||
|                 this.selectApp(id); | ||||
|             })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _doSpringAnimation: function(animationDirection) { | ||||
|         this._grid.actor.opacity = 255; | ||||
|         this._grid.animateSpring(animationDirection, | ||||
|                                  Main.overview.getShowAppsButton()); | ||||
|     }, | ||||
|  | ||||
|     animate: function(animationDirection, onComplete) { | ||||
|         if (animationDirection == IconGrid.AnimationDirection.IN) { | ||||
|             let toAnimate = this._grid.actor.connect('notify::allocation', Lang.bind(this, | ||||
|                 function() { | ||||
|                     this._grid.actor.disconnect(toAnimate); | ||||
|                     // We need to hide the grid temporary to not flash it | ||||
|                     // for a frame | ||||
|                     this._grid.actor.opacity = 0; | ||||
|                     Meta.later_add(Meta.LaterType.BEFORE_REDRAW, | ||||
|                                    Lang.bind(this, function() { | ||||
|                                        this._doSpringAnimation(animationDirection) | ||||
|                                   })); | ||||
|                 })); | ||||
|         } else { | ||||
|             this._doSpringAnimation(animationDirection); | ||||
|         } | ||||
|  | ||||
|         if (onComplete) { | ||||
|             let animationDoneId = this._grid.connect('animation-done', Lang.bind(this, | ||||
|                 function () { | ||||
|                     this._grid.disconnect(animationDoneId); | ||||
|                     onComplete(); | ||||
|             })); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     animateSwitch: function(animationDirection) { | ||||
|         Tweener.removeTweens(this.actor); | ||||
|         Tweener.removeTweens(this._grid.actor); | ||||
|  | ||||
|         let params = { time: VIEWS_SWITCH_TIME, | ||||
|                        transition: 'easeOutQuad' }; | ||||
|         if (animationDirection == IconGrid.AnimationDirection.IN) { | ||||
|             this.actor.show(); | ||||
|             params.opacity = 255; | ||||
|             params.delay = VIEWS_SWITCH_ANIMATION_DELAY; | ||||
|         } else { | ||||
|             params.opacity = 0; | ||||
|             params.delay = 0; | ||||
|             params.onComplete = Lang.bind(this, function() { this.actor.hide() }); | ||||
|         } | ||||
|  | ||||
|         Tweener.addTween(this._grid.actor, params); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(BaseAppView.prototype); | ||||
|  | ||||
| const PageIndicatorsActor = new Lang.Class({ | ||||
|     Name:'PageIndicatorsActor', | ||||
|     Extends: St.BoxLayout, | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent({ style_class: 'page-indicators', | ||||
|                       vertical: true, | ||||
|                       x_expand: true, y_expand: true, | ||||
|                       x_align: Clutter.ActorAlign.END, | ||||
|                       y_align: Clutter.ActorAlign.CENTER, | ||||
|                       reactive: true, | ||||
|                       clip_to_allocation: true }); | ||||
|     }, | ||||
|  | ||||
|     vfunc_get_preferred_height: function(forWidth) { | ||||
|         // We want to request the natural height of all our children as our | ||||
|         // natural height, so we chain up to St.BoxLayout, but we only request 0 | ||||
|         // as minimum height, since it's not that important if some indicators | ||||
|         // are not shown | ||||
|         let [, natHeight] = this.parent(forWidth); | ||||
|         return [0, natHeight]; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const PageIndicators = new Lang.Class({ | ||||
|     Name:'PageIndicators', | ||||
|  | ||||
|     _init: function() { | ||||
|         this.actor = new St.BoxLayout({ style_class: 'page-indicators', | ||||
|                                         vertical: true, | ||||
|                                         x_expand: true, y_expand: true, | ||||
|                                         x_align: Clutter.ActorAlign.END, | ||||
|                                         y_align: Clutter.ActorAlign.CENTER, | ||||
|                                         reactive: true }); | ||||
|         this.actor = new PageIndicatorsActor(); | ||||
|         this._nPages = 0; | ||||
|         this._currentPage = undefined; | ||||
|  | ||||
|         this.actor.connect('notify::mapped', | ||||
|                            Lang.bind(this, this._animateIndicators)); | ||||
|                            Lang.bind(this, function() { | ||||
|                                this.animateIndicators(IconGrid.AnimationDirection.IN); | ||||
|                            }) | ||||
|                           ); | ||||
|     }, | ||||
|  | ||||
|     setNPages: function(nPages) { | ||||
| @@ -236,7 +321,7 @@ const PageIndicators = new Lang.Class({ | ||||
|             children[i].set_checked(i == this._currentPage); | ||||
|     }, | ||||
|  | ||||
|     _animateIndicators: function() { | ||||
|     animateIndicators: function(animationDirection) { | ||||
|         if (!this.actor.mapped) | ||||
|             return; | ||||
|  | ||||
| @@ -244,24 +329,32 @@ const PageIndicators = new Lang.Class({ | ||||
|         if (children.length == 0) | ||||
|             return; | ||||
|  | ||||
|         for (let i = 0; i < this._nPages; i++) | ||||
|             Tweener.removeTweens(children[i]); | ||||
|  | ||||
|         let offset; | ||||
|         if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) | ||||
|             offset = -children[0].width; | ||||
|         else | ||||
|             offset = children[0].width; | ||||
|  | ||||
|         let delay = INDICATORS_ANIMATION_DELAY; | ||||
|         let totalAnimationTime = INDICATORS_BASE_TIME + INDICATORS_ANIMATION_DELAY * this._nPages; | ||||
|         if (totalAnimationTime > INDICATORS_ANIMATION_MAX_TIME) | ||||
|             delay -= (totalAnimationTime - INDICATORS_ANIMATION_MAX_TIME) / this._nPages; | ||||
|         let isAnimationIn = animationDirection == IconGrid.AnimationDirection.IN; | ||||
|         let delay = isAnimationIn ? INDICATORS_ANIMATION_DELAY : | ||||
|                                     INDICATORS_ANIMATION_DELAY_OUT; | ||||
|         let baseTime = isAnimationIn ? INDICATORS_BASE_TIME : INDICATORS_BASE_TIME_OUT; | ||||
|         let totalAnimationTime = baseTime + delay * this._nPages; | ||||
|         let maxTime = isAnimationIn ? INDICATORS_ANIMATION_MAX_TIME : | ||||
|                                       INDICATORS_ANIMATION_MAX_TIME_OUT; | ||||
|         if (totalAnimationTime > maxTime) | ||||
|             delay -= (totalAnimationTime - maxTime) / this._nPages; | ||||
|  | ||||
|         for (let i = 0; i < this._nPages; i++) { | ||||
|             children[i].translation_x = offset; | ||||
|             children[i].translation_x = isAnimationIn ? offset : 0; | ||||
|             Tweener.addTween(children[i], | ||||
|                              { translation_x: 0, | ||||
|                                time: INDICATORS_BASE_TIME + delay * i, | ||||
|                              { translation_x: isAnimationIn ? 0 : offset, | ||||
|                                time: baseTime + delay * i, | ||||
|                                transition: 'easeInOutQuad', | ||||
|                                delay: VIEWS_SWITCH_ANIMATION_DELAY | ||||
|                                delay: isAnimationIn ? VIEWS_SWITCH_ANIMATION_DELAY : 0 | ||||
|                              }); | ||||
|         } | ||||
|     } | ||||
| @@ -273,6 +366,8 @@ const AllView = new Lang.Class({ | ||||
|     Extends: BaseAppView, | ||||
|  | ||||
|     _init: function() { | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|  | ||||
|         this.parent({ usePagination: true }, null); | ||||
|         this._scrollView = new St.ScrollView({ style_class: 'all-apps', | ||||
|                                                x_expand: true, | ||||
| @@ -286,9 +381,7 @@ const AllView = new Lang.Class({ | ||||
|         this.actor.add_actor(this._scrollView); | ||||
|  | ||||
|         this._scrollView.set_policy(Gtk.PolicyType.NEVER, | ||||
|                                     Gtk.PolicyType.AUTOMATIC); | ||||
|         // we are only using ScrollView for the fade effect, hide scrollbars | ||||
|         this._scrollView.vscroll.hide(); | ||||
|                                     Gtk.PolicyType.EXTERNAL); | ||||
|         this._adjustment = this._scrollView.vscroll.adjustment; | ||||
|  | ||||
|         this._pageIndicators = new PageIndicators(); | ||||
| @@ -304,7 +397,7 @@ const AllView = new Lang.Class({ | ||||
|         this._stack = new St.Widget({ layout_manager: new Clutter.BinLayout() }); | ||||
|         let box = new St.BoxLayout({ vertical: true }); | ||||
|  | ||||
|         this._currentPage = 0; | ||||
|         this._grid.currentPage = 0; | ||||
|         this._stack.add_actor(this._grid.actor); | ||||
|         this._eventBlocker = new St.Widget({ x_expand: true, y_expand: true }); | ||||
|         this._stack.add_actor(this._eventBlocker); | ||||
| @@ -370,7 +463,7 @@ const AllView = new Lang.Class({ | ||||
|         Shell.AppSystem.get_default().connect('installed-changed', Lang.bind(this, function() { | ||||
|             Main.queueDeferredWork(this._redisplayWorkId); | ||||
|         })); | ||||
|         this._folderSettings = new Gio.Settings({ schema: 'org.gnome.desktop.app-folders' }); | ||||
|         this._folderSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.app-folders' }); | ||||
|         this._folderSettings.connect('changed::folder-children', Lang.bind(this, function() { | ||||
|             Main.queueDeferredWork(this._redisplayWorkId); | ||||
|         })); | ||||
| @@ -426,9 +519,19 @@ const AllView = new Lang.Class({ | ||||
|             this.folderIcons.push(icon); | ||||
|         })); | ||||
|  | ||||
|         // Allow dragging of the icon only if the Dash would accept a drop to | ||||
|         // change favorite-apps. There are no other possible drop targets from | ||||
|         // the app picker, so there's no other need for a drag to start, | ||||
|         // at least on single-monitor setups. | ||||
|         // This also disables drag-to-launch on multi-monitor setups, | ||||
|         // but we hope that is not used much. | ||||
|         let favoritesWritable = this._settings.is_writable('favorite-apps'); | ||||
|  | ||||
|         apps.forEach(Lang.bind(this, function(appId) { | ||||
|             let app = appSys.lookup_app(appId); | ||||
|             let icon = new AppIcon(app); | ||||
|  | ||||
|             let icon = new AppIcon(app, | ||||
|                                    { isDraggable: favoritesWritable }); | ||||
|             this.addItem(icon); | ||||
|         })); | ||||
|  | ||||
| @@ -436,14 +539,52 @@ const AllView = new Lang.Class({ | ||||
|         this._refilterApps(); | ||||
|     }, | ||||
|  | ||||
|     // Overriden from BaseAppView | ||||
|     animate: function (animationDirection, onComplete) { | ||||
|         if (animationDirection == IconGrid.AnimationDirection.OUT && | ||||
|             this._displayingPopup && this._currentPopup) { | ||||
|             this._currentPopup.popdown(); | ||||
|             let spaceClosedId = this._grid.connect('space-closed', Lang.bind(this, | ||||
|                 function() { | ||||
|                     this._grid.disconnect(spaceClosedId); | ||||
|                     // Given that we can't call this.parent() inside the | ||||
|                     // signal handler, call again animate which will | ||||
|                     // call the parent given that popup is already | ||||
|                     // closed. | ||||
|                     this.animate(animationDirection, onComplete); | ||||
|                 })); | ||||
|         } else { | ||||
|             this.parent(animationDirection, onComplete); | ||||
|             if (animationDirection == IconGrid.AnimationDirection.OUT) | ||||
|                 this._pageIndicators.animateIndicators(animationDirection); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     animateSwitch: function(animationDirection) { | ||||
|         this.parent(animationDirection); | ||||
|  | ||||
|         if (this._currentPopup && this._displayingPopup && | ||||
|             animationDirection == IconGrid.AnimationDirection.OUT) | ||||
|             Tweener.addTween(this._currentPopup.actor, | ||||
|                              { time: VIEWS_SWITCH_TIME, | ||||
|                                transition: 'easeOutQuad', | ||||
|                                opacity: 0, | ||||
|                                onComplete: function() { | ||||
|                                   this.opacity = 255; | ||||
|                                } }); | ||||
|  | ||||
|         if (animationDirection == IconGrid.AnimationDirection.OUT) | ||||
|             this._pageIndicators.animateIndicators(animationDirection); | ||||
|     }, | ||||
|  | ||||
|     getCurrentPageY: function() { | ||||
|         return this._grid.getPageY(this._currentPage); | ||||
|         return this._grid.getPageY(this._grid.currentPage); | ||||
|     }, | ||||
|  | ||||
|     goToPage: function(pageNumber) { | ||||
|         pageNumber = clamp(pageNumber, 0, this._grid.nPages() - 1); | ||||
|  | ||||
|         if (this._currentPage == pageNumber && this._displayingPopup && this._currentPopup) | ||||
|         if (this._grid.currentPage == pageNumber && this._displayingPopup && this._currentPopup) | ||||
|             return; | ||||
|         if (this._displayingPopup && this._currentPopup) | ||||
|             this._currentPopup.popdown(); | ||||
| @@ -463,7 +604,7 @@ const AllView = new Lang.Class({ | ||||
|         let time; | ||||
|         // Only take the velocity into account on page changes, otherwise | ||||
|         // return smoothly to the current page using the default velocity | ||||
|         if (this._currentPage != pageNumber) { | ||||
|         if (this._grid.currentPage != pageNumber) { | ||||
|             let minVelocity = totalHeight / (PAGE_SWITCH_TIME * 1000); | ||||
|             velocity = Math.max(minVelocity, velocity); | ||||
|             time = (diffToPage / velocity) / 1000; | ||||
| @@ -474,9 +615,9 @@ const AllView = new Lang.Class({ | ||||
|         // longer than PAGE_SWITCH_TIME | ||||
|         time = Math.min(time, PAGE_SWITCH_TIME); | ||||
|  | ||||
|         this._currentPage = pageNumber; | ||||
|         this._grid.currentPage = pageNumber; | ||||
|         Tweener.addTween(this._adjustment, | ||||
|                          { value: this._grid.getPageY(this._currentPage), | ||||
|                          { value: this._grid.getPageY(this._grid.currentPage), | ||||
|                            time: time, | ||||
|                            transition: 'easeOutQuad' }); | ||||
|         this._pageIndicators.setCurrentPage(pageNumber); | ||||
| @@ -505,9 +646,9 @@ const AllView = new Lang.Class({ | ||||
|  | ||||
|         let direction = event.get_scroll_direction(); | ||||
|         if (direction == Clutter.ScrollDirection.UP) | ||||
|             this.goToPage(this._currentPage - 1); | ||||
|             this.goToPage(this._grid.currentPage - 1); | ||||
|         else if (direction == Clutter.ScrollDirection.DOWN) | ||||
|             this.goToPage(this._currentPage + 1); | ||||
|             this.goToPage(this._grid.currentPage + 1); | ||||
|  | ||||
|         return Clutter.EVENT_STOP; | ||||
|     }, | ||||
| @@ -547,10 +688,10 @@ const AllView = new Lang.Class({ | ||||
|             return Clutter.EVENT_STOP; | ||||
|  | ||||
|         if (event.get_key_symbol() == Clutter.Page_Up) { | ||||
|             this.goToPage(this._currentPage - 1); | ||||
|             this.goToPage(this._grid.currentPage - 1); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (event.get_key_symbol() == Clutter.Page_Down) { | ||||
|             this.goToPage(this._currentPage + 1); | ||||
|             this.goToPage(this._grid.currentPage + 1); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } | ||||
|  | ||||
| @@ -611,6 +752,7 @@ const AllView = new Lang.Class({ | ||||
|  | ||||
|         if (this._availWidth != availWidth || this._availHeight != availHeight || oldNPages != this._grid.nPages()) { | ||||
|             this._adjustment.value = 0; | ||||
|             this._grid.currentPage = 0; | ||||
|             Meta.later_add(Meta.LaterType.BEFORE_REDRAW, Lang.bind(this, | ||||
|                 function() { | ||||
|                     this._pageIndicators.setNPages(this._grid.nPages()); | ||||
| @@ -633,6 +775,9 @@ const FrequentView = new Lang.Class({ | ||||
|  | ||||
|     _init: function() { | ||||
|         this.parent(null, { fillParent: true }); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|  | ||||
|         this.actor = new St.Widget({ style_class: 'frequent-apps', | ||||
|                                      layout_manager: new Clutter.BinLayout(), | ||||
|                                      x_expand: true, y_expand: true }); | ||||
| @@ -669,10 +814,19 @@ const FrequentView = new Lang.Class({ | ||||
|         if(!hasUsefulData) | ||||
|             return; | ||||
|  | ||||
|         // Allow dragging of the icon only if the Dash would accept a drop to | ||||
|         // change favorite-apps. There are no other possible drop targets from | ||||
|         // the app picker, so there's no other need for a drag to start, | ||||
|         // at least on single-monitor setups. | ||||
|         // This also disables drag-to-launch on multi-monitor setups, | ||||
|         // but we hope that is not used much. | ||||
|         let favoritesWritable = this._settings.is_writable('favorite-apps'); | ||||
|  | ||||
|         for (let i = 0; i < mostUsed.length; i++) { | ||||
|             if (!mostUsed[i].get_app_info().should_show()) | ||||
|                 continue; | ||||
|             let appIcon = new AppIcon(mostUsed[i]); | ||||
|             let appIcon = new AppIcon(mostUsed[i], | ||||
|                                       { isDraggable: favoritesWritable }); | ||||
|             this._grid.addItem(appIcon, -1); | ||||
|         } | ||||
|     }, | ||||
| @@ -740,7 +894,7 @@ const AppDisplay = new Lang.Class({ | ||||
|     Name: 'AppDisplay', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._privacySettings = new Gio.Settings({ schema: 'org.gnome.desktop.privacy' }); | ||||
|         this._privacySettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.privacy' }); | ||||
|         this._privacySettings.connect('changed::remember-app-usage', | ||||
|                                       Lang.bind(this, this._updateFrequentVisibility)); | ||||
|  | ||||
| @@ -749,14 +903,14 @@ const AppDisplay = new Lang.Class({ | ||||
|         let view, button; | ||||
|         view = new FrequentView(); | ||||
|         button = new St.Button({ label: _("Frequent"), | ||||
|                                  style_class: 'app-view-control', | ||||
|                                  style_class: 'app-view-control button', | ||||
|                                  can_focus: true, | ||||
|                                  x_expand: true }); | ||||
|         this._views[Views.FREQUENT] = { 'view': view, 'control': button }; | ||||
|  | ||||
|         view = new AllView(); | ||||
|         button = new St.Button({ label: _("All"), | ||||
|                                  style_class: 'app-view-control', | ||||
|                                  style_class: 'app-view-control button', | ||||
|                                  can_focus: true, | ||||
|                                  x_expand: true }); | ||||
|         this._views[Views.ALL] = { 'view': view, 'control': button }; | ||||
| @@ -772,6 +926,19 @@ const AppDisplay = new Lang.Class({ | ||||
|         let layout = new ControlsBoxLayout({ homogeneous: true }); | ||||
|         this._controls = new St.Widget({ style_class: 'app-view-controls', | ||||
|                                          layout_manager: layout }); | ||||
|         this._controls.connect('notify::mapped', Lang.bind(this, | ||||
|             function() { | ||||
|                 // controls are faded either with their parent or | ||||
|                 // explicitly in animate(); we can't know how they'll be | ||||
|                 // shown next, so make sure to restore their opacity | ||||
|                 // when they are hidden | ||||
|                 if (this._controls.mapped) | ||||
|                   return; | ||||
|  | ||||
|                 Tweener.removeTweens(this._controls); | ||||
|                 this._controls.opacity = 255; | ||||
|             })); | ||||
|  | ||||
|         layout.hookup_style(this._controls); | ||||
|         this.actor.add_actor(new St.Bin({ child: this._controls })); | ||||
|  | ||||
| @@ -795,23 +962,39 @@ const AppDisplay = new Lang.Class({ | ||||
|         this._updateFrequentVisibility(); | ||||
|     }, | ||||
|  | ||||
|     animate: function(animationDirection, onComplete) { | ||||
|         let currentView = this._views[global.settings.get_uint('app-picker-view')].view; | ||||
|  | ||||
|         // Animate controls opacity using iconGrid animation time, since | ||||
|         // it will be the time the AllView or FrequentView takes to show | ||||
|         // it entirely. | ||||
|         let finalOpacity; | ||||
|         if (animationDirection == IconGrid.AnimationDirection.IN) { | ||||
|             this._controls.opacity = 0; | ||||
|             finalOpacity = 255; | ||||
|         } else { | ||||
|             finalOpacity = 0 | ||||
|         } | ||||
|  | ||||
|         Tweener.addTween(this._controls, | ||||
|                          { time: IconGrid.ANIMATION_TIME_IN, | ||||
|                            transition: 'easeInOutQuad', | ||||
|                            opacity: finalOpacity, | ||||
|                           }); | ||||
|  | ||||
|         currentView.animate(animationDirection, onComplete); | ||||
|     }, | ||||
|  | ||||
|     _showView: function(activeIndex) { | ||||
|         for (let i = 0; i < this._views.length; i++) { | ||||
|             let actor = this._views[i].view.actor; | ||||
|  | ||||
|             let params = { time: VIEWS_SWITCH_TIME, | ||||
|                            opacity: (i == activeIndex) ? 255 : 0, | ||||
|                            delay: (i == activeIndex) ? VIEWS_SWITCH_ANIMATION_DELAY : 0 }; | ||||
|             if (i == activeIndex) | ||||
|                 actor.visible = true; | ||||
|             else | ||||
|                 params.onComplete = function() { actor.hide(); }; | ||||
|             Tweener.addTween(actor, params); | ||||
|  | ||||
|             if (i == activeIndex) | ||||
|                 this._views[i].control.add_style_pseudo_class('checked'); | ||||
|             else | ||||
|                 this._views[i].control.remove_style_pseudo_class('checked'); | ||||
|  | ||||
|             let animationDirection = i == activeIndex ? IconGrid.AnimationDirection.IN : | ||||
|                                                         IconGrid.AnimationDirection.OUT; | ||||
|             this._views[i].view.animateSwitch(animationDirection); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
| @@ -893,26 +1076,6 @@ const AppSearchProvider = new Lang.Class({ | ||||
|         this.getInitialResultSet(terms, callback, cancellable); | ||||
|     }, | ||||
|  | ||||
|     activateResult: function(result) { | ||||
|         let app = this._appSys.lookup_app(result); | ||||
|         let event = Clutter.get_current_event(); | ||||
|         let modifiers = event ? event.get_state() : 0; | ||||
|         let openNewWindow = modifiers & Clutter.ModifierType.CONTROL_MASK; | ||||
|  | ||||
|         if (openNewWindow) | ||||
|             app.open_new_window(-1); | ||||
|         else | ||||
|             app.activate(); | ||||
|     }, | ||||
|  | ||||
|     dragActivateResult: function(id, params) { | ||||
|         params = Params.parse(params, { workspace: -1, | ||||
|                                         timestamp: 0 }); | ||||
|  | ||||
|         let app = this._appSys.lookup_app(id); | ||||
|         app.open_new_window(workspace); | ||||
|     }, | ||||
|  | ||||
|     createResultObject: function (resultMeta) { | ||||
|         let app = this._appSys.lookup_app(resultMeta['id']); | ||||
|         return new AppIcon(app); | ||||
| @@ -944,14 +1107,20 @@ const FolderView = new Lang.Class({ | ||||
|         Util.ensureActorVisibleInScrollView(this.actor, actor); | ||||
|     }, | ||||
|  | ||||
|     // Overriden from BaseAppView | ||||
|     animate: function(animationDirection) { | ||||
|         this._grid.animatePulse(animationDirection); | ||||
|     }, | ||||
|  | ||||
|     createFolderIcon: function(size) { | ||||
|         let layout = new Clutter.TableLayout(); | ||||
|         let layout = new Clutter.GridLayout(); | ||||
|         let icon = new St.Widget({ layout_manager: layout, | ||||
|                                    style_class: 'app-folder-icon' }); | ||||
|         layout.hookup_style(icon); | ||||
|         let subSize = Math.floor(FOLDER_SUBICON_FRACTION * size); | ||||
|  | ||||
|         let numItems = this._allItems.length; | ||||
|         let rtl = icon.get_text_direction() == Clutter.TextDirection.RTL; | ||||
|         for (let i = 0; i < 4; i++) { | ||||
|             let bin; | ||||
|             if (i < numItems) { | ||||
| @@ -960,7 +1129,7 @@ const FolderView = new Lang.Class({ | ||||
|             } else { | ||||
|                 bin = new St.Bin({ width: subSize, height: subSize }); | ||||
|             } | ||||
|             layout.pack(bin, i % 2, Math.floor(i / 2)); | ||||
|             layout.attach(bin, rtl ? (i + 1) % 2 : i % 2, Math.floor(i / 2), 1, 1); | ||||
|         } | ||||
|  | ||||
|         return icon; | ||||
| @@ -1120,6 +1289,7 @@ const FolderIcon = new Lang.Class({ | ||||
|             addAppId(appInfo.get_id()); | ||||
|         }); | ||||
|  | ||||
|         this.actor.visible = this.view.getAllItems().length > 0; | ||||
|         this.view.loadGrid(); | ||||
|         this.emit('apps-changed'); | ||||
|     }, | ||||
| @@ -1249,18 +1419,53 @@ const AppFolderPopup = new Lang.Class({ | ||||
|             function() { | ||||
|                 this.actor.destroy(); | ||||
|             })); | ||||
|         this._grabHelper = new GrabHelper.GrabHelper(this.actor); | ||||
|         this._grabHelper.addActor(Main.layoutManager.overviewGroup); | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this._onKeyPress)); | ||||
|     }, | ||||
|  | ||||
|     _onKeyPress: function(actor, event) { | ||||
|         if (!this._isOpen) | ||||
|         if (global.stage.get_key_focus() != actor) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         if (event.get_key_symbol() != Clutter.KEY_Escape) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|         // Since we need to only grab focus on one item child when the user | ||||
|         // actually press a key we don't use navigate_focus when opening | ||||
|         // the popup. | ||||
|         // Instead of that, grab the focus on the AppFolderPopup actor | ||||
|         // and actually moves the focus to a child only when the user | ||||
|         // actually press a key. | ||||
|         // It should work with just grab_key_focus on the AppFolderPopup | ||||
|         // actor, but since the arrow keys are not wrapping_around the focus | ||||
|         // is not grabbed by a child when the widget that has the current focus | ||||
|         // is the same that is requesting focus, so to make it works with arrow | ||||
|         // keys we need to connect to the key-press-event and navigate_focus | ||||
|         // when that happens using TAB_FORWARD or TAB_BACKWARD instead of arrow | ||||
|         // keys | ||||
|  | ||||
|         this.popdown(); | ||||
|         return Clutter.EVENT_STOP; | ||||
|         // Use TAB_FORWARD for down key and right key | ||||
|         // and TAB_BACKWARD for up key and left key on ltr | ||||
|         // languages | ||||
|         let direction; | ||||
|         let isLtr = Clutter.get_default_text_direction() == Clutter.TextDirection.LTR; | ||||
|         switch (event.get_key_symbol()) { | ||||
|             case Clutter.Down: | ||||
|                 direction = Gtk.DirectionType.TAB_FORWARD; | ||||
|                 break; | ||||
|             case Clutter.Right: | ||||
|                 direction = isLtr ? Gtk.DirectionType.TAB_FORWARD : | ||||
|                                     Gtk.DirectionType.TAB_BACKWARD; | ||||
|                 break; | ||||
|             case Clutter.Up: | ||||
|                 direction = Gtk.DirectionType.TAB_BACKWARD; | ||||
|                 break; | ||||
|             case Clutter.Left: | ||||
|                 direction = isLtr ? Gtk.DirectionType.TAB_BACKWARD : | ||||
|                                     Gtk.DirectionType.TAB_FORWARD; | ||||
|                 break; | ||||
|             default: | ||||
|                 return Clutter.EVENT_PROPAGATE; | ||||
|         } | ||||
|         return actor.navigate_focus(null, direction, false); | ||||
|     }, | ||||
|  | ||||
|     toggle: function() { | ||||
| @@ -1274,15 +1479,27 @@ const AppFolderPopup = new Lang.Class({ | ||||
|         if (this._isOpen) | ||||
|             return; | ||||
|  | ||||
|         this._isOpen = this._grabHelper.grab({ actor: this.actor, | ||||
|                                                onUngrab: Lang.bind(this, this.popdown) }); | ||||
|  | ||||
|         if (!this._isOpen) | ||||
|             return; | ||||
|  | ||||
|         this.actor.show(); | ||||
|  | ||||
|         this._boxPointer.setArrowActor(this._source.actor); | ||||
|         // We need to hide the icons of the view until the boxpointer animation | ||||
|         // is completed so we can animate the icons after as we like without | ||||
|         // showing them while boxpointer is animating. | ||||
|         this._view.actor.opacity = 0; | ||||
|         this._boxPointer.show(BoxPointer.PopupAnimation.FADE | | ||||
|                               BoxPointer.PopupAnimation.SLIDE); | ||||
|                               BoxPointer.PopupAnimation.SLIDE, | ||||
|                               Lang.bind(this, | ||||
|             function() { | ||||
|                 this._view.actor.opacity = 255; | ||||
|                 this._view.animate(IconGrid.AnimationDirection.IN); | ||||
|             })); | ||||
|  | ||||
|         this.actor.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false); | ||||
|  | ||||
|         this._isOpen = true; | ||||
|         this.emit('open-state-changed', true); | ||||
|     }, | ||||
|  | ||||
| @@ -1290,6 +1507,8 @@ const AppFolderPopup = new Lang.Class({ | ||||
|         if (!this._isOpen) | ||||
|             return; | ||||
|  | ||||
|         this._grabHelper.ungrab({ actor: this.actor }); | ||||
|  | ||||
|         this._boxPointer.hide(BoxPointer.PopupAnimation.FADE | | ||||
|                               BoxPointer.PopupAnimation.SLIDE); | ||||
|         this._isOpen = false; | ||||
| @@ -1328,47 +1547,70 @@ const AppIcon = new Lang.Class({ | ||||
|                                      can_focus: true, | ||||
|                                      x_fill: true, | ||||
|                                      y_fill: true }); | ||||
|  | ||||
|         this._dot = new St.Widget({ style_class: 'app-well-app-running-dot', | ||||
|                                     layout_manager: new Clutter.BinLayout(), | ||||
|                                     x_expand: true, y_expand: true, | ||||
|                                     x_align: Clutter.ActorAlign.CENTER, | ||||
|                                     y_align: Clutter.ActorAlign.END }); | ||||
|  | ||||
|         this._iconContainer = new St.Widget({ layout_manager: new Clutter.BinLayout(), | ||||
|                                               x_expand: true, y_expand: true }); | ||||
|  | ||||
|         this.actor.set_child(this._iconContainer); | ||||
|         this._iconContainer.add_child(this._dot); | ||||
|  | ||||
|         this.actor._delegate = this; | ||||
|  | ||||
|         if (!iconParams) | ||||
|             iconParams = {}; | ||||
|  | ||||
|         // Get the isDraggable property without passing it on to the BaseIcon: | ||||
|         let appIconParams = Params.parse(iconParams, { isDraggable: true }, true); | ||||
|         let isDraggable = appIconParams['isDraggable']; | ||||
|         delete iconParams['isDraggable']; | ||||
|  | ||||
|         iconParams['createIcon'] = Lang.bind(this, this._createIcon); | ||||
|         iconParams['setSizeManually'] = true; | ||||
|         this.icon = new IconGrid.BaseIcon(app.get_name(), iconParams); | ||||
|         this.actor.set_child(this.icon.actor); | ||||
|         this._iconContainer.add_child(this.icon.actor); | ||||
|  | ||||
|         this.actor.label_actor = this.icon.label; | ||||
|  | ||||
|         this.actor.connect('leave-event', Lang.bind(this, this._onLeaveEvent)); | ||||
|         this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress)); | ||||
|         this.actor.connect('touch-event', Lang.bind(this, this._onTouchEvent)); | ||||
|         this.actor.connect('clicked', Lang.bind(this, this._onClicked)); | ||||
|         this.actor.connect('popup-menu', Lang.bind(this, this._onKeyboardPopupMenu)); | ||||
|  | ||||
|         this._menu = null; | ||||
|         this._menuManager = new PopupMenu.PopupMenuManager(this); | ||||
|  | ||||
|         this._draggable = DND.makeDraggable(this.actor); | ||||
|         this._draggable.connect('drag-begin', Lang.bind(this, | ||||
|             function () { | ||||
|                 this._removeMenuTimeout(); | ||||
|                 Main.overview.beginItemDrag(this); | ||||
|             })); | ||||
|         this._draggable.connect('drag-cancelled', Lang.bind(this, | ||||
|             function () { | ||||
|                 Main.overview.cancelledItemDrag(this); | ||||
|             })); | ||||
|         this._draggable.connect('drag-end', Lang.bind(this, | ||||
|             function () { | ||||
|                Main.overview.endItemDrag(this); | ||||
|             })); | ||||
|         if (isDraggable) { | ||||
|             this._draggable = DND.makeDraggable(this.actor); | ||||
|             this._draggable.connect('drag-begin', Lang.bind(this, | ||||
|                 function () { | ||||
|                     this._removeMenuTimeout(); | ||||
|                     Main.overview.beginItemDrag(this); | ||||
|                 })); | ||||
|             this._draggable.connect('drag-cancelled', Lang.bind(this, | ||||
|                 function () { | ||||
|                     Main.overview.cancelledItemDrag(this); | ||||
|                 })); | ||||
|             this._draggable.connect('drag-end', Lang.bind(this, | ||||
|                 function () { | ||||
|                    Main.overview.endItemDrag(this); | ||||
|                 })); | ||||
|         } | ||||
|  | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         this._menuTimeoutId = 0; | ||||
|         this._stateChangedId = this.app.connect('notify::state', | ||||
|                                                 Lang.bind(this, | ||||
|                                                           this._onStateChanged)); | ||||
|         this._onStateChanged(); | ||||
|         this._stateChangedId = this.app.connect('notify::state', Lang.bind(this, | ||||
|             function () { | ||||
|                 this._updateRunningStyle(); | ||||
|             })); | ||||
|         this._updateRunningStyle(); | ||||
|     }, | ||||
|  | ||||
|     _onDestroy: function() { | ||||
| @@ -1389,24 +1631,33 @@ const AppIcon = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onStateChanged: function() { | ||||
|     _updateRunningStyle: function() { | ||||
|         if (this.app.state != Shell.AppState.STOPPED) | ||||
|             this.actor.add_style_class_name('running'); | ||||
|             this._dot.show(); | ||||
|         else | ||||
|             this.actor.remove_style_class_name('running'); | ||||
|             this._dot.hide(); | ||||
|     }, | ||||
|  | ||||
|     _setPopupTimeout: function() { | ||||
|         this._removeMenuTimeout(); | ||||
|         this._menuTimeoutId = Mainloop.timeout_add(MENU_POPUP_TIMEOUT, | ||||
|             Lang.bind(this, function() { | ||||
|                 this._menuTimeoutId = 0; | ||||
|                 this.popupMenu(); | ||||
|                 return GLib.SOURCE_REMOVE; | ||||
|             })); | ||||
|         GLib.Source.set_name_by_id(this._menuTimeoutId, '[gnome-shell] this.popupMenu'); | ||||
|     }, | ||||
|  | ||||
|     _onLeaveEvent: function(actor, event) { | ||||
|         this.actor.fake_release(); | ||||
|         this._removeMenuTimeout(); | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress: function(actor, event) { | ||||
|         let button = event.get_button(); | ||||
|         if (button == 1) { | ||||
|             this._removeMenuTimeout(); | ||||
|             this._menuTimeoutId = Mainloop.timeout_add(MENU_POPUP_TIMEOUT, | ||||
|                 Lang.bind(this, function() { | ||||
|                     this._menuTimeoutId = 0; | ||||
|                     this.popupMenu(); | ||||
|                     return GLib.SOURCE_REMOVE; | ||||
|                 })); | ||||
|             GLib.Source.set_name_by_id(this._menuTimeoutId, '[gnome-shell] this.popupMenu'); | ||||
|             this._setPopupTimeout(); | ||||
|         } else if (button == 3) { | ||||
|             this.popupMenu(); | ||||
|             return Clutter.EVENT_STOP; | ||||
| @@ -1414,16 +1665,16 @@ const AppIcon = new Lang.Class({ | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onTouchEvent: function (actor, event) { | ||||
|         if (event.type() == Clutter.EventType.TOUCH_BEGIN) | ||||
|             this._setPopupTimeout(); | ||||
|  | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onClicked: function(actor, button) { | ||||
|         this._removeMenuTimeout(); | ||||
|  | ||||
|         if (button == 1) { | ||||
|             this._onActivate(Clutter.get_current_event()); | ||||
|         } else if (button == 2) { | ||||
|             this.app.open_new_window(-1); | ||||
|             Main.overview.hide(); | ||||
|         } | ||||
|         return false; | ||||
|         this.activate(button); | ||||
|     }, | ||||
|  | ||||
|     _onKeyboardPopupMenu: function() { | ||||
| @@ -1438,7 +1689,9 @@ const AppIcon = new Lang.Class({ | ||||
|     popupMenu: function() { | ||||
|         this._removeMenuTimeout(); | ||||
|         this.actor.fake_release(); | ||||
|         this._draggable.fakeRelease(); | ||||
|  | ||||
|         if (this._draggable) | ||||
|             this._draggable.fakeRelease(); | ||||
|  | ||||
|         if (!this._menu) { | ||||
|             this._menu = new AppIconMenu(this); | ||||
| @@ -1477,19 +1730,29 @@ const AppIcon = new Lang.Class({ | ||||
|         this.emit('menu-state-changed', false); | ||||
|     }, | ||||
|  | ||||
|     _onActivate: function (event) { | ||||
|         let modifiers = event.get_state(); | ||||
|     activate: function (button) { | ||||
|         let event = Clutter.get_current_event(); | ||||
|         let modifiers = event ? event.get_state() : 0; | ||||
|         let openNewWindow = this.app.can_open_new_window () && | ||||
|                             modifiers & Clutter.ModifierType.CONTROL_MASK && | ||||
|                             this.app.state == Shell.AppState.RUNNING || | ||||
|                             button && button == 2; | ||||
|  | ||||
|         if (modifiers & Clutter.ModifierType.CONTROL_MASK | ||||
|             && this.app.state == Shell.AppState.RUNNING) { | ||||
|         if (this.app.state == Shell.AppState.STOPPED || openNewWindow) | ||||
|             this.animateLaunch(); | ||||
|  | ||||
|         if (openNewWindow) | ||||
|             this.app.open_new_window(-1); | ||||
|         } else { | ||||
|         else | ||||
|             this.app.activate(); | ||||
|         } | ||||
|  | ||||
|         Main.overview.hide(); | ||||
|     }, | ||||
|  | ||||
|     animateLaunch: function() { | ||||
|         this.icon.animateZoomOut(); | ||||
|     }, | ||||
|  | ||||
|     shellWorkspaceLaunch : function(params) { | ||||
|         params = Params.parse(params, { workspace: -1, | ||||
|                                         timestamp: 0 }); | ||||
| @@ -1531,6 +1794,8 @@ const AppIconMenu = new Lang.Class({ | ||||
|  | ||||
|         this.actor.add_style_class_name('app-well-menu'); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|  | ||||
|         // Chain our visibility and lifecycle to that of the source | ||||
|         source.actor.connect('notify::mapped', Lang.bind(this, function () { | ||||
|             if (!source.actor.mapped) | ||||
| @@ -1568,15 +1833,21 @@ const AppIconMenu = new Lang.Class({ | ||||
|         if (!this._source.app.is_window_backed()) { | ||||
|             this._appendSeparator(); | ||||
|  | ||||
|             this._newWindowMenuItem = this._appendMenuItem(_("New Window")); | ||||
|             this._newWindowMenuItem.connect('activate', Lang.bind(this, function() { | ||||
|                 this._source.app.open_new_window(-1); | ||||
|                 this.emit('activate-window', null); | ||||
|             })); | ||||
|             this._appendSeparator(); | ||||
|  | ||||
|             let appInfo = this._source.app.get_app_info(); | ||||
|             let actions = appInfo.list_actions(); | ||||
|             if (this._source.app.can_open_new_window() && | ||||
|                 actions.indexOf('new-window') == -1) { | ||||
|                 this._newWindowMenuItem = this._appendMenuItem(_("New Window")); | ||||
|                 this._newWindowMenuItem.connect('activate', Lang.bind(this, function() { | ||||
|                     if (this._source.app.state == Shell.AppState.STOPPED) | ||||
|                         this._source.animateLaunch(); | ||||
|  | ||||
|                     this._source.app.open_new_window(-1); | ||||
|                     this.emit('activate-window', null); | ||||
|                 })); | ||||
|                 this._appendSeparator(); | ||||
|             } | ||||
|  | ||||
|             for (let i = 0; i < actions.length; i++) { | ||||
|                 let action = actions[i]; | ||||
|                 let item = this._appendMenuItem(appInfo.get_action_name(action)); | ||||
| @@ -1585,22 +1856,27 @@ const AppIconMenu = new Lang.Class({ | ||||
|                     this.emit('activate-window', null); | ||||
|                 })); | ||||
|             } | ||||
|             this._appendSeparator(); | ||||
|  | ||||
|             let isFavorite = AppFavorites.getAppFavorites().isFavorite(this._source.app.get_id()); | ||||
|             let canFavorite = this._settings.is_writable('favorite-apps'); | ||||
|  | ||||
|             if (isFavorite) { | ||||
|                 let item = this._appendMenuItem(_("Remove from Favorites")); | ||||
|                 item.connect('activate', Lang.bind(this, function() { | ||||
|                     let favs = AppFavorites.getAppFavorites(); | ||||
|                     favs.removeFavorite(this._source.app.get_id()); | ||||
|                 })); | ||||
|             } else { | ||||
|                 let item = this._appendMenuItem(_("Add to Favorites")); | ||||
|                 item.connect('activate', Lang.bind(this, function() { | ||||
|                     let favs = AppFavorites.getAppFavorites(); | ||||
|                     favs.addFavorite(this._source.app.get_id()); | ||||
|                 })); | ||||
|             if (canFavorite) { | ||||
|                 this._appendSeparator(); | ||||
|  | ||||
|                 let isFavorite = AppFavorites.getAppFavorites().isFavorite(this._source.app.get_id()); | ||||
|  | ||||
|                 if (isFavorite) { | ||||
|                     let item = this._appendMenuItem(_("Remove from Favorites")); | ||||
|                     item.connect('activate', Lang.bind(this, function() { | ||||
|                         let favs = AppFavorites.getAppFavorites(); | ||||
|                         favs.removeFavorite(this._source.app.get_id()); | ||||
|                     })); | ||||
|                 } else { | ||||
|                     let item = this._appendMenuItem(_("Add to Favorites")); | ||||
|                     item.connect('activate', Lang.bind(this, function() { | ||||
|                         let favs = AppFavorites.getAppFavorites(); | ||||
|                         favs.addFavorite(this._source.app.get_id()); | ||||
|                     })); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (Shell.AppSystem.get_default().lookup_app('org.gnome.Software.desktop')) { | ||||
|   | ||||
| @@ -6,6 +6,36 @@ const Signals = imports.signals; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const RENAMED_DESKTOP_IDS = { | ||||
|     'baobab.desktop': 'org.gnome.baobab.desktop', | ||||
|     'cheese.desktop': 'org.gnome.Cheese.desktop', | ||||
|     'dconf-editor.desktop': 'ca.desrt.dconf-editor.desktop', | ||||
|     'file-roller.desktop': 'org.gnome.FileRoller.desktop', | ||||
|     'gcalctool.desktop': 'gnome-calculator.desktop', | ||||
|     'gedit.desktop': 'org.gnome.gedit.desktop', | ||||
|     'glchess.desktop': 'gnome-chess.desktop', | ||||
|     'glines.desktop': 'five-or-more.desktop', | ||||
|     'gnect.desktop': 'four-in-a-row.desktop', | ||||
|     'gnibbles.desktop': 'gnome-nibbles.desktop', | ||||
|     'gnobots2.desktop': 'gnome-robots.desktop', | ||||
|     'gnome-boxes.desktop': 'org.gnome.Boxes.desktop', | ||||
|     'gnome-clocks.desktop': 'org.gnome.clocks.desktop', | ||||
|     'gnome-contacts.desktop': 'org.gnome.Contacts.desktop', | ||||
|     'gnome-documents.desktop': 'org.gnome.Documents.desktop', | ||||
|     'gnome-font-viewer.desktop': 'org.gnome.font-viewer.desktop', | ||||
|     'gnome-photos.desktop': 'org.gnome.Photos.desktop', | ||||
|     'gnome-screenshot.desktop': 'org.gnome.Screenshot.desktop', | ||||
|     'gnome-software.desktop': 'org.gnome.Software.desktop', | ||||
|     'gnome-weather.desktop': 'org.gnome.Weather.Application.desktop', | ||||
|     'gnomine.desktop': 'gnome-mines.desktop', | ||||
|     'gnotravex.desktop': 'gnome-tetravex.desktop', | ||||
|     'gnotski.desktop': 'gnome-klotski.desktop', | ||||
|     'gtali.desktop': 'tali.desktop', | ||||
|     'nautilus.desktop': 'org.gnome.Nautilus.desktop', | ||||
|     'polari.desktop': 'org.gnome.Polari.desktop', | ||||
|     'totem.desktop': 'org.gnome.Totem.desktop', | ||||
| }; | ||||
|  | ||||
| const AppFavorites = new Lang.Class({ | ||||
|     Name: 'AppFavorites', | ||||
|  | ||||
| @@ -24,6 +54,21 @@ const AppFavorites = new Lang.Class({ | ||||
|  | ||||
|     reload: function() { | ||||
|         let ids = global.settings.get_strv(this.FAVORITE_APPS_KEY); | ||||
|  | ||||
|         // Map old desktop file names to the current ones | ||||
|         let updated = false; | ||||
|         ids = ids.map(function (id) { | ||||
|             let newId = RENAMED_DESKTOP_IDS[id]; | ||||
|             if (newId !== undefined) { | ||||
|                 updated = true; | ||||
|                 return newId; | ||||
|             } | ||||
|             return id; | ||||
|         }); | ||||
|         // ... and write back the updated desktop file names | ||||
|         if (updated) | ||||
|             global.settings.set_strv(this.FAVORITE_APPS_KEY, ids); | ||||
|  | ||||
|         let appSys = Shell.AppSystem.get_default(); | ||||
|         let apps = ids.map(function (id) { | ||||
|                 return appSys.lookup_app(id); | ||||
|   | ||||
| @@ -33,8 +33,7 @@ function addBackgroundMenu(actor, layoutManager) { | ||||
|     actor._backgroundManager = new PopupMenu.PopupMenuManager({ actor: actor }); | ||||
|     actor._backgroundManager.addMenu(actor._backgroundMenu); | ||||
|  | ||||
|     function openMenu() { | ||||
|         let [x, y] = global.get_pointer(); | ||||
|     function openMenu(x, y) { | ||||
|         Main.layoutManager.setDummyCursorGeometry(x, y, 0, 0); | ||||
|         actor._backgroundMenu.open(BoxPointer.PopupAnimation.NONE); | ||||
|     } | ||||
| @@ -42,22 +41,32 @@ function addBackgroundMenu(actor, layoutManager) { | ||||
|     let clickAction = new Clutter.ClickAction(); | ||||
|     clickAction.connect('long-press', function(action, actor, state) { | ||||
|         if (state == Clutter.LongPressState.QUERY) | ||||
|             return action.get_button() == 1 && !actor._backgroundMenu.isOpen; | ||||
|             return ((action.get_button() == 0 || | ||||
|                      action.get_button() == 1) && | ||||
|                     !actor._backgroundMenu.isOpen); | ||||
|         if (state == Clutter.LongPressState.ACTIVATE) { | ||||
|             openMenu(); | ||||
|             let [x, y] = action.get_coords(); | ||||
|             openMenu(x, y); | ||||
|             actor._backgroundManager.ignoreRelease(); | ||||
|         } | ||||
|         return true; | ||||
|     }); | ||||
|     clickAction.connect('clicked', function(action) { | ||||
|         if (action.get_button() == 3) | ||||
|             openMenu(); | ||||
|         if (action.get_button() == 3) { | ||||
|             let [x, y] = action.get_coords(); | ||||
|             openMenu(x, y); | ||||
|         } | ||||
|     }); | ||||
|     actor.add_action(clickAction); | ||||
|  | ||||
|     let grabOpBeginId = global.display.connect('grab-op-begin', function () { | ||||
|         clickAction.release(); | ||||
|     }); | ||||
|  | ||||
|     actor.connect('destroy', function() { | ||||
|         actor._backgroundMenu.destroy(); | ||||
|         actor._backgroundMenu = null; | ||||
|         actor._backgroundManager = null; | ||||
|         global.display.disconnect(grabOpBeginId); | ||||
|     }); | ||||
| } | ||||
|   | ||||
| @@ -13,9 +13,11 @@ const Shell = imports.gi.Shell; | ||||
|  | ||||
| const MSECS_IN_DAY = 24 * 60 * 60 * 1000; | ||||
| const SHOW_WEEKDATE_KEY = 'show-weekdate'; | ||||
| const ELLIPSIS_CHAR = '\u2026'; | ||||
|  | ||||
| // alias to prevent xgettext from picking up strings translated in GTK+ | ||||
| const gtk30_ = Gettext_gtk30.gettext; | ||||
| const NC_ = function(context, str) { return str; }; | ||||
|  | ||||
| // in org.gnome.desktop.interface | ||||
| const CLOCK_FORMAT_KEY        = 'clock-format'; | ||||
| @@ -32,12 +34,10 @@ function _sameDay(dateA, dateB) { | ||||
|     return _sameMonth(dateA, dateB) && (dateA.getDate() == dateB.getDate()); | ||||
| } | ||||
|  | ||||
| /* 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; | ||||
|     /* Translators: Enter 0-6 (Sunday-Saturday) for non-work days. Examples: "0" (Sunday) "6" (Saturday) "06" (Sunday and Saturday). */ | ||||
|     let days = C_('calendar-no-work', "06"); | ||||
|     return days.indexOf(date.getDay().toString()) == -1; | ||||
| } | ||||
|  | ||||
| function _getBeginningOfDay(date) { | ||||
| @@ -58,19 +58,21 @@ function _getEndOfDay(date) { | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| function _formatEventTime(event, clockFormat) { | ||||
| function _formatEventTime(event, clockFormat, periodBegin, periodEnd) { | ||||
|     let ret; | ||||
|     if (event.allDay) { | ||||
|     let allDay = (event.allDay || (event.date <= periodBegin && event.end >= periodEnd)); | ||||
|     if (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 { | ||||
|         let date = event.date >= periodBegin ? event.date : event.end; | ||||
|         switch (clockFormat) { | ||||
|         case '24h': | ||||
|             /* Translators: Shown in calendar event list, if 24h format, | ||||
|                \u2236 is a ratio character, similar to : */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%H\u2236%M")); | ||||
|             ret = date.toLocaleFormat(C_("event list time", "%H\u2236%M")); | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
| @@ -79,30 +81,13 @@ function _formatEventTime(event, clockFormat) { | ||||
|             /* Translators: Shown in calendar event list, if 12h format, | ||||
|                \u2236 is a ratio character, similar to : and \u2009 is | ||||
|                a thin space */ | ||||
|             ret = event.date.toLocaleFormat(C_("event list time", "%l\u2236%M\u2009%p")); | ||||
|             ret = date.toLocaleFormat(C_("event list time", "%l\u2236%M\u2009%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 _getCalendarDayAbbreviation(dayNumber) { | ||||
|     let abbreviations = [ | ||||
|         /* Translators: Calendar grid abbreviation for Sunday. | ||||
| @@ -248,11 +233,24 @@ const DBusEventSource = new Lang.Class({ | ||||
|         this._initialized = false; | ||||
|         this._dbusProxy = new CalendarServer(); | ||||
|         this._dbusProxy.init_async(GLib.PRIORITY_DEFAULT, null, Lang.bind(this, function(object, result) { | ||||
|             let loaded = false; | ||||
|  | ||||
|             try { | ||||
|                 this._dbusProxy.init_finish(result); | ||||
|                 loaded = true; | ||||
|             } catch(e) { | ||||
|                 log('Error loading calendars: ' + e.message); | ||||
|                 return; | ||||
|                 if (e.matches(Gio.DBusError, Gio.DBusError.TIMED_OUT)) { | ||||
|                     // Ignore timeouts and install signals as normal, because with high | ||||
|                     // probability the service will appear later on, and we will get a | ||||
|                     // NameOwnerChanged which will finish loading | ||||
|                     // | ||||
|                     // (But still _initialized to false, because the proxy does not know | ||||
|                     // about the HasCalendars property and would cause an exception trying | ||||
|                     // to read it) | ||||
|                 } else { | ||||
|                     log('Error loading calendars: ' + e.message); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             this._dbusProxy.connectSignal('Changed', Lang.bind(this, this._onChanged)); | ||||
| @@ -268,9 +266,11 @@ const DBusEventSource = new Lang.Class({ | ||||
|                 this.emit('notify::has-calendars'); | ||||
|             })); | ||||
|  | ||||
|             this._initialized = true; | ||||
|             this.emit('notify::has-calendars'); | ||||
|             this._onNameAppeared(); | ||||
|             this._initialized = loaded; | ||||
|             if (loaded) { | ||||
|                 this.emit('notify::has-calendars'); | ||||
|                 this._onNameAppeared(); | ||||
|             } | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
| @@ -292,6 +292,7 @@ const DBusEventSource = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onNameAppeared: function(owner) { | ||||
|         this._initialized = true; | ||||
|         this._resetCache(); | ||||
|         this._loadEvents(true); | ||||
|     }, | ||||
| @@ -361,6 +362,12 @@ const DBusEventSource = new Lang.Class({ | ||||
|                 result.push(event); | ||||
|             } | ||||
|         } | ||||
|         result.sort(function(event1, event2) { | ||||
|             // sort events by end time on ending day | ||||
|             let d1 = event1.date < begin && event1.end <= end ? event1.end : event1.date; | ||||
|             let d2 = event2.date < begin && event2.end <= end ? event2.end : event2.date; | ||||
|             return d1.getTime() - d2.getTime(); | ||||
|         }); | ||||
|         return result; | ||||
|     }, | ||||
|  | ||||
| @@ -383,7 +390,7 @@ const Calendar = new Lang.Class({ | ||||
|  | ||||
|     _init: function() { | ||||
|         this._weekStart = Shell.util_get_week_start(); | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.shell.calendar' }); | ||||
|         this._settings = new Gio.Settings({ schema_id: '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); | ||||
| @@ -408,9 +415,9 @@ const Calendar = new Lang.Class({ | ||||
|  | ||||
|         this._shouldDateGrabFocus = false; | ||||
|  | ||||
|         this.actor = new St.Table({ homogeneous: false, | ||||
|                                     style_class: 'calendar', | ||||
|                                     reactive: true }); | ||||
|         this.actor = new St.Widget({ style_class: 'calendar', | ||||
|                                      layout_manager: new Clutter.GridLayout(), | ||||
|                                      reactive: true }); | ||||
|  | ||||
|         this.actor.connect('scroll-event', | ||||
|                            Lang.bind(this, this._onScroll)); | ||||
| @@ -441,15 +448,15 @@ const Calendar = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _buildHeader: function() { | ||||
|         let layout = this.actor.layout_manager; | ||||
|         let offsetCols = this._useWeekdate ? 1 : 0; | ||||
|         this.actor.destroy_all_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 }); | ||||
|         layout.attach(this._topBox, 0, 0, offsetCols + 7, 1); | ||||
|  | ||||
|         this._backButton = new St.Button({ style_class: 'calendar-change-month-back', | ||||
|         this._backButton = new St.Button({ style_class: 'calendar-change-month-back pager-button', | ||||
|                                            accessible_name: _("Previous month"), | ||||
|                                            can_focus: true }); | ||||
|         this._topBox.add(this._backButton); | ||||
| @@ -459,7 +466,7 @@ const Calendar = new Lang.Class({ | ||||
|                                          can_focus: true }); | ||||
|         this._topBox.add(this._monthLabel, { expand: true, x_fill: false, x_align: St.Align.MIDDLE }); | ||||
|  | ||||
|         this._forwardButton = new St.Button({ style_class: 'calendar-change-month-forward', | ||||
|         this._forwardButton = new St.Button({ style_class: 'calendar-change-month-forward pager-button', | ||||
|                                               accessible_name: _("Next month"), | ||||
|                                               can_focus: true }); | ||||
|         this._topBox.add(this._forwardButton); | ||||
| @@ -479,10 +486,12 @@ const Calendar = new Lang.Class({ | ||||
|             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 }); | ||||
|             let col; | ||||
|             if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) | ||||
|                 col = 6 - (7 + iter.getDay() - this._weekStart) % 7; | ||||
|             else | ||||
|                 col = offsetCols + (7 + iter.getDay() - this._weekStart) % 7; | ||||
|             layout.attach(label, col, 1, 1, 1); | ||||
|             iter.setTime(iter.getTime() + MSECS_IN_DAY); | ||||
|         } | ||||
|  | ||||
| @@ -592,6 +601,7 @@ const Calendar = new Lang.Class({ | ||||
|         beginDate.setHours(12); | ||||
|  | ||||
|         this._calendarBegin = new Date(beginDate); | ||||
|         this._markedAsToday = now; | ||||
|  | ||||
|         let year = beginDate.getYear(); | ||||
|  | ||||
| @@ -601,6 +611,7 @@ const Calendar = new Lang.Class({ | ||||
|  | ||||
|         beginDate.setTime(beginDate.getTime() - (weekPadding + daysToWeekStart) * MSECS_IN_DAY); | ||||
|  | ||||
|         let layout = this.actor.layout_manager; | ||||
|         let iter = new Date(beginDate); | ||||
|         let row = 2; | ||||
|         // nRows here means 6 weeks + one header + one navbar | ||||
| @@ -648,16 +659,19 @@ const Calendar = new Lang.Class({ | ||||
|             button.style_class = styleClass; | ||||
|  | ||||
|             let offsetCols = this._useWeekdate ? 1 : 0; | ||||
|             this.actor.add(button, | ||||
|                            { row: row, col: offsetCols + (7 + iter.getDay() - this._weekStart) % 7 }); | ||||
|             let col; | ||||
|             if (rtl) | ||||
|                 col = 6 - (7 + iter.getDay() - this._weekStart) % 7; | ||||
|             else | ||||
|                 col = offsetCols + (7 + iter.getDay() - this._weekStart) % 7; | ||||
|             layout.attach(button, col, row, 1, 1); | ||||
|  | ||||
|             this._buttons.push(button); | ||||
|  | ||||
|             if (this._useWeekdate && iter.getDay() == 4) { | ||||
|                 let label = new St.Label({ text: _getCalendarWeekForDate(iter).toString(), | ||||
|                 let label = new St.Label({ text: iter.toLocaleFormat('%V'), | ||||
|                                            style_class: 'calendar-day-base calendar-week-number'}); | ||||
|                 this.actor.add(label, | ||||
|                                { row: row, col: 0, y_align: St.Align.MIDDLE }); | ||||
|                 layout.attach(label, rtl ? 7 : 0, row, 1, 1); | ||||
|             } | ||||
|  | ||||
|             iter.setTime(iter.getTime() + MSECS_IN_DAY); | ||||
| @@ -679,7 +693,7 @@ const Calendar = new Lang.Class({ | ||||
|         else | ||||
|             this._monthLabel.text = this._selectedDate.toLocaleFormat(this._headerFormat); | ||||
|  | ||||
|         if (!this._calendarBegin || !_sameMonth(this._selectedDate, this._calendarBegin)) | ||||
|         if (!this._calendarBegin || !_sameMonth(this._selectedDate, this._calendarBegin) || !_sameDay(now, this._markedAsToday)) | ||||
|             this._rebuildCalendar(); | ||||
|  | ||||
|         this._buttons.forEach(Lang.bind(this, function(button) { | ||||
| @@ -700,9 +714,12 @@ const EventsList = new Lang.Class({ | ||||
|     Name: 'EventsList', | ||||
|  | ||||
|     _init: function() { | ||||
|         this.actor = new St.Table({ style_class: 'events-table' }); | ||||
|         let layout = new Clutter.GridLayout({ orientation: Clutter.Orientation.VERTICAL }); | ||||
|         this.actor = new St.Widget({ style_class: 'events-table', | ||||
|                                      layout_manager: layout }); | ||||
|         layout.hookup_style(this.actor); | ||||
|         this._date = new Date(); | ||||
|         this._desktopSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' }); | ||||
|         this._desktopSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.interface' }); | ||||
|         this._desktopSettings.connect('changed', Lang.bind(this, this._update)); | ||||
|         this._weekStart = Shell.util_get_week_start(); | ||||
|     }, | ||||
| @@ -712,68 +729,82 @@ const EventsList = new Lang.Class({ | ||||
|         this._eventSource.connect('changed', Lang.bind(this, this._update)); | ||||
|     }, | ||||
|  | ||||
|     _addEvent: function(event, index, includeDayName) { | ||||
|     _addEvent: function(event, index, includeDayName, periodBegin, periodEnd) { | ||||
|         let dayString; | ||||
|         if (includeDayName) | ||||
|             dayString = _getEventDayAbbreviation(event.date.getDay()); | ||||
|         else | ||||
|         if (includeDayName) { | ||||
|             if (event.date >= periodBegin) | ||||
|                 dayString = _getEventDayAbbreviation(event.date.getDay()); | ||||
|             else /* show event end day if it began earlier */ | ||||
|                 dayString = _getEventDayAbbreviation(event.end.getDay()); | ||||
|         } else { | ||||
|             dayString = ''; | ||||
|         } | ||||
|  | ||||
|         let dayLabel = new St.Label({ style_class: 'events-day-dayname', | ||||
|                                       text: dayString }); | ||||
|                                       text: dayString, | ||||
|                                       x_align: Clutter.ActorAlign.END, | ||||
|                                       y_align: Clutter.ActorAlign.START }); | ||||
|         dayLabel.clutter_text.line_wrap = false; | ||||
|         dayLabel.clutter_text.ellipsize = false; | ||||
|  | ||||
|         this.actor.add(dayLabel, { row: index, col: 0, | ||||
|                                    x_expand: false, x_align: St.Align.END, | ||||
|                                    y_fill: false, y_align: St.Align.START }); | ||||
|         let rtl = this.actor.get_text_direction() == Clutter.TextDirection.RTL; | ||||
|  | ||||
|         let layout = this.actor.layout_manager; | ||||
|         layout.attach(dayLabel, rtl ? 2 : 0, index, 1, 1); | ||||
|         let clockFormat = this._desktopSettings.get_string(CLOCK_FORMAT_KEY); | ||||
|         let timeString = _formatEventTime(event, clockFormat); | ||||
|         let timeString = _formatEventTime(event, clockFormat, periodBegin, periodEnd); | ||||
|         let timeLabel = new St.Label({ style_class: 'events-day-time', | ||||
|                                        text: timeString }); | ||||
|                                        text: timeString, | ||||
|                                        y_align: Clutter.ActorAlign.START }); | ||||
|         timeLabel.clutter_text.line_wrap = false; | ||||
|         timeLabel.clutter_text.ellipsize = false; | ||||
|  | ||||
|         this.actor.add(timeLabel, { row: index, col: 1, | ||||
|                                     x_expand: false, x_align: St.Align.MIDDLE, | ||||
|                                     y_fill: false, y_align: St.Align.START }); | ||||
|         let preEllipsisLabel = new St.Label({ style_class: 'events-day-time-ellipses', | ||||
|                                               text: ELLIPSIS_CHAR, | ||||
|                                               y_align: Clutter.ActorAlign.START }); | ||||
|         let postEllipsisLabel = new St.Label({ style_class: 'events-day-time-ellipses', | ||||
|                                                text: ELLIPSIS_CHAR, | ||||
|                                                y_align: Clutter.ActorAlign.START }); | ||||
|         if (event.allDay || event.date >= periodBegin) | ||||
|             preEllipsisLabel.opacity = 0; | ||||
|         if (event.allDay || event.end <= periodEnd) | ||||
|             postEllipsisLabel.opacity = 0; | ||||
|  | ||||
|         let timeLabelBoxLayout = new St.BoxLayout(); | ||||
|         timeLabelBoxLayout.add(preEllipsisLabel); | ||||
|         timeLabelBoxLayout.add(timeLabel); | ||||
|         timeLabelBoxLayout.add(postEllipsisLabel); | ||||
|         layout.attach(timeLabelBoxLayout, 1, index, 1, 1); | ||||
|  | ||||
|         let titleLabel = new St.Label({ style_class: 'events-day-task', | ||||
|                                         text: event.summary }); | ||||
|                                         text: event.summary, | ||||
|                                         x_expand: true }); | ||||
|         titleLabel.clutter_text.line_wrap = true; | ||||
|         titleLabel.clutter_text.ellipsize = false; | ||||
|  | ||||
|         this.actor.add(titleLabel, { row: index, col: 2, | ||||
|                                      x_expand: true, x_align: St.Align.START, | ||||
|                                      y_fill: false, y_align: St.Align.START }); | ||||
|         layout.attach(titleLabel, rtl ? 0 : 2, index, 1, 1); | ||||
|     }, | ||||
|  | ||||
|     _addPeriod: function(header, index, begin, end, includeDayName, showNothingScheduled) { | ||||
|         let events = this._eventSource.getEvents(begin, end); | ||||
|     _addPeriod: function(header, index, periodBegin, periodEnd, includeDayName, showNothingScheduled) { | ||||
|         let events = this._eventSource.getEvents(periodBegin, periodEnd); | ||||
|  | ||||
|         if (events.length == 0 && !showNothingScheduled) | ||||
|             return index; | ||||
|  | ||||
|         this.actor.add(new St.Label({ style_class: 'events-day-header', text: header }), | ||||
|                        { row: index, col: 0, col_span: 3, | ||||
|                          // In theory, x_expand should be true here, but x_expand | ||||
|                          // is a property of the column for StTable, ie all day cells | ||||
|                          // get it too | ||||
|                          x_expand: false, x_align: St.Align.START, | ||||
|                          y_fill: false, y_align: St.Align.START }); | ||||
|         let label = new St.Label({ style_class: 'events-day-header', text: header }); | ||||
|         let layout = this.actor.layout_manager; | ||||
|         layout.attach(label, 0, index, 3, 1); | ||||
|         index++; | ||||
|  | ||||
|         for (let n = 0; n < events.length; n++) { | ||||
|             this._addEvent(events[n], index, includeDayName); | ||||
|             this._addEvent(events[n], index, includeDayName, periodBegin, periodEnd); | ||||
|             index++; | ||||
|         } | ||||
|  | ||||
|         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); | ||||
|             this._addEvent(nothingEvent, index, false); | ||||
|             let nothingEvent = new CalendarEvent(periodBegin, periodBegin, _("Nothing Scheduled"), true); | ||||
|             this._addEvent(nothingEvent, index, false, periodBegin, periodEnd); | ||||
|             index++; | ||||
|         } | ||||
|  | ||||
| @@ -786,14 +817,17 @@ const EventsList = new Lang.Class({ | ||||
|         let dayBegin = _getBeginningOfDay(day); | ||||
|         let dayEnd = _getEndOfDay(day); | ||||
|  | ||||
|         let dayString; | ||||
|         let dayFormat; | ||||
|         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")); | ||||
|             dayFormat = Shell.util_translate_time_string(NC_("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")); | ||||
|             dayFormat = Shell.util_translate_time_string(NC_("calendar heading", | ||||
|                                                              "%A, %B %d, %Y")); | ||||
|         let dayString = day.toLocaleFormat(dayFormat); | ||||
|         this._addPeriod(dayString, 0, dayBegin, dayEnd, false, true); | ||||
|     }, | ||||
|  | ||||
|   | ||||
| @@ -23,7 +23,7 @@ const AutomountManager = new Lang.Class({ | ||||
|     Name: 'AutomountManager', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: SETTINGS_SCHEMA }); | ||||
|         this._volumeQueue = []; | ||||
|         this._session = new GnomeSession.SessionManager(); | ||||
|         this._session.connectSignal('InhibitorAdded', | ||||
|   | ||||
| @@ -24,11 +24,11 @@ const AutorunSetting = { | ||||
| }; | ||||
|  | ||||
| // misc utils | ||||
| function shouldAutorunMount(mount, forTransient) { | ||||
| function shouldAutorunMount(mount) { | ||||
|     let root = mount.get_root(); | ||||
|     let volume = mount.get_volume(); | ||||
|  | ||||
|     if (!volume || (!volume.allowAutorun && forTransient)) | ||||
|     if (!volume || !volume.allowAutorun) | ||||
|         return false; | ||||
|  | ||||
|     if (root.is_native() && isMountRootHidden(root)) | ||||
| @@ -96,7 +96,7 @@ const ContentTypeDiscoverer = new Lang.Class({ | ||||
|  | ||||
|     _init: function(callback) { | ||||
|         this._callback = callback; | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: SETTINGS_SCHEMA }); | ||||
|     }, | ||||
|  | ||||
|     guessContentTypes: function(mount) { | ||||
| @@ -167,281 +167,43 @@ const AutorunManager = new Lang.Class({ | ||||
|         this._session = new GnomeSession.SessionManager(); | ||||
|         this._volumeMonitor = Gio.VolumeMonitor.get(); | ||||
|  | ||||
|         this._transDispatcher = new AutorunTransientDispatcher(this); | ||||
|     }, | ||||
|  | ||||
|     _ensureResidentSource: function() { | ||||
|         if (this._residentSource) | ||||
|             return; | ||||
|  | ||||
|         this._residentSource = new AutorunResidentSource(this); | ||||
|         let destroyId = this._residentSource.connect('destroy', Lang.bind(this, function() { | ||||
|             this._residentSource.disconnect(destroyId); | ||||
|             this._residentSource = null; | ||||
|         })); | ||||
|         this._dispatcher = new AutorunDispatcher(this); | ||||
|     }, | ||||
|  | ||||
|     enable: function() { | ||||
|         this._scanMounts(); | ||||
|  | ||||
|         this._mountAddedId = this._volumeMonitor.connect('mount-added', Lang.bind(this, this._onMountAdded)); | ||||
|         this._mountRemovedId = this._volumeMonitor.connect('mount-removed', Lang.bind(this, this._onMountRemoved)); | ||||
|     }, | ||||
|  | ||||
|     disable: function() { | ||||
|         if (this._residentSource) | ||||
|             this._residentSource.destroy(); | ||||
|         this._volumeMonitor.disconnect(this._mountAddedId); | ||||
|         this._volumeMonitor.disconnect(this._mountRemovedId); | ||||
|     }, | ||||
|  | ||||
|     _processMount: function(mount, hotplug) { | ||||
|         let discoverer = new ContentTypeDiscoverer(Lang.bind(this, function(mount, apps, contentTypes) { | ||||
|             this._ensureResidentSource(); | ||||
|             this._residentSource.addMount(mount, apps); | ||||
|  | ||||
|             if (hotplug) | ||||
|                 this._transDispatcher.addMount(mount, apps, contentTypes); | ||||
|         })); | ||||
|         discoverer.guessContentTypes(mount); | ||||
|     }, | ||||
|  | ||||
|     _scanMounts: function() { | ||||
|         let mounts = this._volumeMonitor.get_mounts(); | ||||
|         mounts.forEach(Lang.bind(this, function(mount) { | ||||
|             this._processMount(mount, false); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _onMountAdded: function(monitor, mount) { | ||||
|         // don't do anything if our session is not the currently | ||||
|         // active one | ||||
|         if (!this._session.SessionIsActive) | ||||
|             return; | ||||
|  | ||||
|         this._processMount(mount, true); | ||||
|         let discoverer = new ContentTypeDiscoverer(Lang.bind(this, function(mount, apps, contentTypes) { | ||||
|             this._dispatcher.addMount(mount, apps, contentTypes); | ||||
|         })); | ||||
|         discoverer.guessContentTypes(mount); | ||||
|     }, | ||||
|  | ||||
|     _onMountRemoved: function(monitor, mount) { | ||||
|         this._transDispatcher.removeMount(mount); | ||||
|         if (this._residentSource) | ||||
|             this._residentSource.removeMount(mount); | ||||
|     }, | ||||
|  | ||||
|     ejectMount: function(mount) { | ||||
|         let mountOp = new ShellMountOperation.ShellMountOperation(mount); | ||||
|  | ||||
|         // first, see if we have a drive | ||||
|         let drive = mount.get_drive(); | ||||
|         let volume = mount.get_volume(); | ||||
|  | ||||
|         if (drive && | ||||
|             drive.get_start_stop_type() == Gio.DriveStartStopType.SHUTDOWN && | ||||
|             drive.can_stop()) { | ||||
|             drive.stop(0, mountOp.mountOp, null, | ||||
|                        Lang.bind(this, this._onStop)); | ||||
|         } else { | ||||
|             if (mount.can_eject()) { | ||||
|                 mount.eject_with_operation(0, mountOp.mountOp, null, | ||||
|                                            Lang.bind(this, this._onEject)); | ||||
|             } else if (volume && volume.can_eject()) { | ||||
|                 volume.eject_with_operation(0, mountOp.mountOp, null, | ||||
|                                             Lang.bind(this, this._onEject)); | ||||
|             } else if (drive && drive.can_eject()) { | ||||
|                 drive.eject_with_operation(0, mountOp.mountOp, null, | ||||
|                                            Lang.bind(this, this._onEject)); | ||||
|             } else if (mount.can_unmount()) { | ||||
|                 mount.unmount_with_operation(0, mountOp.mountOp, null, | ||||
|                                              Lang.bind(this, this._onUnmount)); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onUnmount: function(mount, res) { | ||||
|         try { | ||||
|             mount.unmount_with_operation_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to eject the mount ' + mount.get_name()  | ||||
|                     + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onEject: function(source, res) { | ||||
|         try { | ||||
|             source.eject_with_operation_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to eject the drive ' + source.get_name() | ||||
|                     + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onStop: function(drive, res) { | ||||
|         try { | ||||
|             drive.stop_finish(res); | ||||
|         } catch (e) { | ||||
|             if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.FAILED_HANDLED)) | ||||
|                 log('Unable to stop the drive ' + drive.get_name()  | ||||
|                     + ': ' + e.toString()); | ||||
|         } | ||||
|     }, | ||||
| }); | ||||
|  | ||||
| const AutorunResidentSource = new Lang.Class({ | ||||
|     Name: 'AutorunResidentSource', | ||||
|     Extends: MessageTray.Source, | ||||
|  | ||||
|     _init: function(manager) { | ||||
|         this.parent(_("Removable Devices"), 'media-removable'); | ||||
|         this.resident = true; | ||||
|  | ||||
|         this._mounts = []; | ||||
|  | ||||
|         this._manager = manager; | ||||
|         this._notification = new AutorunResidentNotification(this._manager, this); | ||||
|     }, | ||||
|  | ||||
|     _createPolicy: function() { | ||||
|         return new MessageTray.NotificationPolicy({ showInLockScreen: false }); | ||||
|     }, | ||||
|  | ||||
|     buildRightClickMenu: function() { | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     addMount: function(mount, apps) { | ||||
|         if (!shouldAutorunMount(mount, false)) | ||||
|             return; | ||||
|  | ||||
|         let filtered = this._mounts.filter(function (element) { | ||||
|             return (element.mount == mount); | ||||
|         }); | ||||
|  | ||||
|         if (filtered.length != 0) | ||||
|             return; | ||||
|  | ||||
|         let element = { mount: mount, apps: apps }; | ||||
|         this._mounts.push(element); | ||||
|         this._redisplay(); | ||||
|     }, | ||||
|  | ||||
|     removeMount: function(mount) { | ||||
|         this._mounts = | ||||
|             this._mounts.filter(function (element) { | ||||
|                 return (element.mount != mount); | ||||
|             }); | ||||
|  | ||||
|         this._redisplay(); | ||||
|     }, | ||||
|  | ||||
|     _redisplay: function() { | ||||
|         if (this._mounts.length == 0) { | ||||
|             this._notification.destroy(); | ||||
|             this.destroy(); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         this._notification.updateForMounts(this._mounts); | ||||
|  | ||||
|         // add ourselves as a source, and push the notification | ||||
|         if (!Main.messageTray.contains(this)) { | ||||
|             Main.messageTray.add(this); | ||||
|             this.pushNotification(this._notification); | ||||
|         } | ||||
|         this._dispatcher.removeMount(mount); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const AutorunResidentNotification = new Lang.Class({ | ||||
|     Name: 'AutorunResidentNotification', | ||||
|     Extends: MessageTray.Notification, | ||||
|  | ||||
|     _init: function(manager, source) { | ||||
|         this.parent(source, source.title, null, { customContent: true }); | ||||
|  | ||||
|         // set the notification as resident | ||||
|         this.setResident(true); | ||||
|  | ||||
|         this._layout = new St.BoxLayout ({ style_class: 'hotplug-resident-box', | ||||
|                                            vertical: true }); | ||||
|         this._manager = manager; | ||||
|  | ||||
|         this.addActor(this._layout, | ||||
|                       { x_expand: true, | ||||
|                         x_fill: true }); | ||||
|     }, | ||||
|  | ||||
|     updateForMounts: function(mounts) { | ||||
|         // remove all the layout content | ||||
|         this._layout.destroy_all_children(); | ||||
|  | ||||
|         for (let idx = 0; idx < mounts.length; idx++) { | ||||
|             let element = mounts[idx]; | ||||
|  | ||||
|             let actor = this._itemForMount(element.mount, element.apps); | ||||
|             this._layout.add(actor, { x_fill: true, | ||||
|                                       expand: true }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _itemForMount: function(mount, apps) { | ||||
|         let item = new St.BoxLayout(); | ||||
|  | ||||
|         // prepare the mount button content | ||||
|         let mountLayout = new St.BoxLayout(); | ||||
|  | ||||
|         let mountIcon = new St.Icon({ gicon: mount.get_icon(), | ||||
|                                       style_class: 'hotplug-resident-mount-icon' }); | ||||
|         mountLayout.add_actor(mountIcon); | ||||
|  | ||||
|         let labelBin = new St.Bin({ y_align: St.Align.MIDDLE }); | ||||
|         let mountLabel = | ||||
|             new St.Label({ text: mount.get_name(), | ||||
|                            style_class: 'hotplug-resident-mount-label', | ||||
|                            track_hover: true, | ||||
|                            reactive: true }); | ||||
|         labelBin.add_actor(mountLabel); | ||||
|         mountLayout.add_actor(labelBin); | ||||
|  | ||||
|         let mountButton = new St.Button({ child: mountLayout, | ||||
|                                           x_align: St.Align.START, | ||||
|                                           x_fill: true, | ||||
|                                           style_class: 'hotplug-resident-mount', | ||||
|                                           button_mask: St.ButtonMask.ONE }); | ||||
|         item.add(mountButton, { x_align: St.Align.START, | ||||
|                                 expand: true }); | ||||
|  | ||||
|         let ejectIcon =  | ||||
|             new St.Icon({ icon_name: 'media-eject-symbolic', | ||||
|                           style_class: 'hotplug-resident-eject-icon' }); | ||||
|  | ||||
|         let ejectButton = | ||||
|             new St.Button({ style_class: 'hotplug-resident-eject-button', | ||||
|                             button_mask: St.ButtonMask.ONE, | ||||
|                             child: ejectIcon }); | ||||
|         item.add(ejectButton, { x_align: St.Align.END }); | ||||
|  | ||||
|         // now connect signals | ||||
|         mountButton.connect('clicked', Lang.bind(this, function(actor, event) { | ||||
|             startAppForMount(apps[0], mount); | ||||
|         })); | ||||
|  | ||||
|         ejectButton.connect('clicked', Lang.bind(this, function() { | ||||
|             this._manager.ejectMount(mount); | ||||
|         })); | ||||
|  | ||||
|         return item; | ||||
|     }, | ||||
| }); | ||||
|  | ||||
| const AutorunTransientDispatcher = new Lang.Class({ | ||||
|     Name: 'AutorunTransientDispatcher', | ||||
| const AutorunDispatcher = new Lang.Class({ | ||||
|     Name: 'AutorunDispatcher', | ||||
|  | ||||
|     _init: function(manager) { | ||||
|         this._manager = manager; | ||||
|         this._sources = []; | ||||
|         this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: SETTINGS_SCHEMA }); | ||||
|     }, | ||||
|  | ||||
|     _getAutorunSettingForType: function(contentType) { | ||||
| @@ -482,7 +244,7 @@ const AutorunTransientDispatcher = new Lang.Class({ | ||||
|             return; | ||||
|       | ||||
|         // add a new source | ||||
|         this._sources.push(new AutorunTransientSource(this._manager, mount, apps)); | ||||
|         this._sources.push(new AutorunSource(this._manager, mount, apps)); | ||||
|     }, | ||||
|  | ||||
|     addMount: function(mount, apps, contentTypes) { | ||||
| @@ -491,7 +253,7 @@ const AutorunTransientDispatcher = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         // if the mount doesn't want to be autorun, return | ||||
|         if (!shouldAutorunMount(mount, true)) | ||||
|         if (!shouldAutorunMount(mount)) | ||||
|             return; | ||||
|  | ||||
|         let setting = this._getAutorunSettingForType(contentTypes[0]); | ||||
| @@ -531,8 +293,8 @@ const AutorunTransientDispatcher = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const AutorunTransientSource = new Lang.Class({ | ||||
|     Name: 'AutorunTransientSource', | ||||
| const AutorunSource = new Lang.Class({ | ||||
|     Name: 'AutorunSource', | ||||
|     Extends: MessageTray.Source, | ||||
|  | ||||
|     _init: function(manager, mount, apps) { | ||||
| @@ -542,7 +304,7 @@ const AutorunTransientSource = new Lang.Class({ | ||||
|  | ||||
|         this.parent(mount.get_name()); | ||||
|  | ||||
|         this._notification = new AutorunTransientNotification(this._manager, this); | ||||
|         this._notification = new AutorunNotification(this._manager, this); | ||||
|  | ||||
|         // add ourselves as a source, and popup the notification | ||||
|         Main.messageTray.add(this); | ||||
| @@ -554,34 +316,24 @@ const AutorunTransientSource = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const AutorunTransientNotification = new Lang.Class({ | ||||
|     Name: 'AutorunTransientNotification', | ||||
| const AutorunNotification = new Lang.Class({ | ||||
|     Name: 'AutorunNotification', | ||||
|     Extends: MessageTray.Notification, | ||||
|  | ||||
|     _init: function(manager, source) { | ||||
|         this.parent(source, source.title, null, { customContent: true }); | ||||
|         this.parent(source, source.title); | ||||
|  | ||||
|         this._manager = manager; | ||||
|         this._box = new St.BoxLayout({ style_class: 'hotplug-transient-box', | ||||
|                                        vertical: true }); | ||||
|         this.addActor(this._box); | ||||
|  | ||||
|         this._mount = source.mount; | ||||
|  | ||||
|         source.apps.forEach(Lang.bind(this, function (app) { | ||||
|             let actor = this._buttonForApp(app); | ||||
|  | ||||
|             if (actor) | ||||
|                 this._box.add(actor, { x_fill: true, | ||||
|                                        x_align: St.Align.START }); | ||||
|                 this.addButton(actor); | ||||
|         })); | ||||
|  | ||||
|         this._box.add(this._buttonForEject(), { x_fill: true, | ||||
|                                                 x_align: St.Align.START }); | ||||
|  | ||||
|         // set the notification to transient and urgent, so that it | ||||
|         // expands out | ||||
|         this.setTransient(true); | ||||
|         // set the notification to urgent, so that it expands out | ||||
|         this.setUrgency(MessageTray.Urgency.CRITICAL); | ||||
|     }, | ||||
|  | ||||
| @@ -601,7 +353,7 @@ const AutorunTransientNotification = new Lang.Class({ | ||||
|                                      x_fill: true, | ||||
|                                      x_align: St.Align.START, | ||||
|                                      button_mask: St.ButtonMask.ONE, | ||||
|                                      style_class: 'hotplug-notification-item' }); | ||||
|                                      style_class: 'hotplug-notification-item button' }); | ||||
|  | ||||
|         button.connect('clicked', Lang.bind(this, function() { | ||||
|             startAppForMount(app, this._mount); | ||||
| @@ -611,29 +363,11 @@ const AutorunTransientNotification = new Lang.Class({ | ||||
|         return button; | ||||
|     }, | ||||
|  | ||||
|     _buttonForEject: function() { | ||||
|         let box = new St.BoxLayout(); | ||||
|         let icon = new St.Icon({ icon_name: 'media-eject-symbolic', | ||||
|                                  style_class: 'hotplug-notification-item-icon' }); | ||||
|         box.add(icon); | ||||
|     _onClicked: function() { | ||||
|         this.parent(); | ||||
|  | ||||
|         let label = new St.Bin({ y_align: St.Align.MIDDLE, | ||||
|                                  child: new St.Label | ||||
|                                  ({ text: _("Eject") }) | ||||
|                                }); | ||||
|         box.add(label); | ||||
|  | ||||
|         let button = new St.Button({ child: box, | ||||
|                                      x_fill: true, | ||||
|                                      x_align: St.Align.START, | ||||
|                                      button_mask: St.ButtonMask.ONE, | ||||
|                                      style_class: 'hotplug-notification-item' }); | ||||
|  | ||||
|         button.connect('clicked', Lang.bind(this, function() { | ||||
|             this._manager.ejectMount(this._mount); | ||||
|         })); | ||||
|  | ||||
|         return button; | ||||
|         let app = Gio.app_info_get_default_for_type('inode/directory', false); | ||||
|         startAppForMount(app, this._mount); | ||||
|     } | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -41,7 +41,7 @@ const KeyringDialog = new Lang.Class({ | ||||
|         mainContentBox.add(this._messageBox, | ||||
|                            { y_align: St.Align.START, expand: true, x_fill: true, y_fill: true }); | ||||
|  | ||||
|         let subject = new St.Label({ style_class: 'prompt-dialog-headline' }); | ||||
|         let subject = new St.Label({ style_class: 'prompt-dialog-headline headline' }); | ||||
|         this.prompt.bind_property('message', subject, 'text', GObject.BindingFlags.SYNC_CREATE); | ||||
|  | ||||
|         this._messageBox.add(subject, | ||||
| @@ -80,44 +80,58 @@ const KeyringDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _buildControlTable: function() { | ||||
|         let layout = new Clutter.TableLayout(); | ||||
|         let layout = new Clutter.GridLayout({ orientation: Clutter.Orientation.VERTICAL }); | ||||
|         let table = new St.Widget({ style_class: 'keyring-dialog-control-table', | ||||
|                                     layout_manager: layout }); | ||||
|         layout.hookup_style(table); | ||||
|         let rtl = table.get_text_direction() == Clutter.TextDirection.RTL; | ||||
|         let row = 0; | ||||
|  | ||||
|         if (this.prompt.password_visible) { | ||||
|             let label = new St.Label({ style_class: 'prompt-dialog-password-label' }); | ||||
|             let label = new St.Label({ style_class: 'prompt-dialog-password-label', | ||||
|                                        x_align: Clutter.ActorAlign.START, | ||||
|                                        y_align: Clutter.ActorAlign.CENTER }); | ||||
|             label.set_text(_("Password:")); | ||||
|             label.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|             layout.pack(label, 0, row); | ||||
|             layout.child_set(label, { x_expand: false, y_fill: false, | ||||
|                                       x_align: Clutter.TableAlignment.START }); | ||||
|             this._passwordEntry = new St.Entry({ style_class: 'prompt-dialog-password-entry', | ||||
|                                                  text: '', | ||||
|                                                  can_focus: true }); | ||||
|                                                  can_focus: true, | ||||
|                                                  x_expand: true }); | ||||
|             this._passwordEntry.clutter_text.set_password_char('\u25cf'); // ● U+25CF BLACK CIRCLE | ||||
|             ShellEntry.addContextMenu(this._passwordEntry, { isPassword: true }); | ||||
|             this._passwordEntry.clutter_text.connect('activate', Lang.bind(this, this._onPasswordActivate)); | ||||
|             layout.pack(this._passwordEntry, 1, row); | ||||
|  | ||||
|             if (rtl) { | ||||
|                 layout.attach(this._passwordEntry, 0, row, 1, 1); | ||||
|                 layout.attach(label, 1, row, 1, 1); | ||||
|             } else { | ||||
|                 layout.attach(label, 0, row, 1, 1); | ||||
|                 layout.attach(this._passwordEntry, 1, row, 1, 1); | ||||
|             } | ||||
|             row++; | ||||
|         } else { | ||||
|             this._passwordEntry = null; | ||||
|         } | ||||
|  | ||||
|         if (this.prompt.confirm_visible) { | ||||
|             var label = new St.Label(({ style_class: 'prompt-dialog-password-label' })); | ||||
|             var label = new St.Label(({ style_class: 'prompt-dialog-password-label', | ||||
|                                         x_align: Clutter.ActorAlign.START, | ||||
|                                         y_align: Clutter.ActorAlign.CENTER })); | ||||
|             label.set_text(_("Type again:")); | ||||
|             layout.pack(label, 0, row); | ||||
|             layout.child_set(label, { x_expand: false, y_fill: false, | ||||
|                                       x_align: Clutter.TableAlignment.START }); | ||||
|             this._confirmEntry = new St.Entry({ style_class: 'prompt-dialog-password-entry', | ||||
|                                                 text: '', | ||||
|                                                 can_focus: true }); | ||||
|                                                 can_focus: true, | ||||
|                                                 x_expand: true }); | ||||
|             this._confirmEntry.clutter_text.set_password_char('\u25cf'); // ● U+25CF BLACK CIRCLE | ||||
|             ShellEntry.addContextMenu(this._confirmEntry, { isPassword: true }); | ||||
|             this._confirmEntry.clutter_text.connect('activate', Lang.bind(this, this._onConfirmActivate)); | ||||
|             layout.pack(this._confirmEntry, 1, row); | ||||
|             if (rtl) { | ||||
|                 layout.attach(this._confirmEntry, 0, row, 1, 1); | ||||
|                 layout.attach(label, 1, row, 1, 1); | ||||
|             } else { | ||||
|                 layout.attach(label, 0, row, 1, 1); | ||||
|                 layout.attach(this._confirmEntry, 1, row, 1, 1); | ||||
|             } | ||||
|             row++; | ||||
|         } else { | ||||
|             this._confirmEntry = null; | ||||
| @@ -130,15 +144,15 @@ const KeyringDialog = new Lang.Class({ | ||||
|             let choice = new CheckBox.CheckBox(); | ||||
|             this.prompt.bind_property('choice-label', choice.getLabelActor(), 'text', GObject.BindingFlags.SYNC_CREATE); | ||||
|             this.prompt.bind_property('choice-chosen', choice.actor, 'checked', GObject.BindingFlags.SYNC_CREATE | GObject.BindingFlags.BIDIRECTIONAL); | ||||
|             layout.pack(choice.actor, 1, row); | ||||
|             layout.attach(choice.actor, rtl ? 0 : 1, row, 1, 1); | ||||
|             row++; | ||||
|         } | ||||
|  | ||||
|         let warning = new St.Label({ style_class: 'prompt-dialog-error-label' }); | ||||
|         let warning = new St.Label({ style_class: 'prompt-dialog-error-label', | ||||
|                                      x_align: Clutter.ActorAlign.START }); | ||||
|         warning.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|         warning.clutter_text.line_wrap = true; | ||||
|         layout.pack(warning, 1, row); | ||||
|         layout.child_set(warning, { x_fill: false, x_align: Clutter.TableAlignment.START }); | ||||
|         layout.attach(warning, rtl ? 0 : 1, row, 1, 1); | ||||
|         this.prompt.bind_property('warning-visible', warning, 'visible', GObject.BindingFlags.SYNC_CREATE); | ||||
|         this.prompt.bind_property('warning', warning, 'text', GObject.BindingFlags.SYNC_CREATE); | ||||
|  | ||||
|   | ||||
| @@ -54,7 +54,7 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
|         mainContentBox.add(messageBox, | ||||
|                            { y_align: St.Align.START }); | ||||
|  | ||||
|         let subjectLabel = new St.Label({ style_class: 'prompt-dialog-headline', | ||||
|         let subjectLabel = new St.Label({ style_class: 'prompt-dialog-headline headline', | ||||
|                                             text: this._content.title }); | ||||
|         messageBox.add(subjectLabel, | ||||
|                        { y_fill:  false, | ||||
| @@ -72,24 +72,28 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
|                              expand: true }); | ||||
|         } | ||||
|  | ||||
|         let layout = new Clutter.TableLayout(); | ||||
|         let layout = new Clutter.GridLayout({ orientation: Clutter.Orientation.VERTICAL }); | ||||
|         let secretTable = new St.Widget({ style_class: 'network-dialog-secret-table', | ||||
|                                           layout_manager: layout }); | ||||
|         layout.hookup_style(secretTable); | ||||
|  | ||||
|         let rtl = secretTable.get_text_direction() == Clutter.TextDirection.RTL; | ||||
|         let initialFocusSet = false; | ||||
|         let pos = 0; | ||||
|         for (let i = 0; i < this._content.secrets.length; i++) { | ||||
|             let secret = this._content.secrets[i]; | ||||
|             let label = new St.Label({ style_class: 'prompt-dialog-password-label', | ||||
|                                        text: secret.label }); | ||||
|                                        text: secret.label, | ||||
|                                        x_align: Clutter.ActorAlign.START, | ||||
|                                        y_align: Clutter.ActorAlign.CENTER }); | ||||
|             label.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; | ||||
|  | ||||
|             let reactive = secret.key != null; | ||||
|  | ||||
|             secret.entry = new St.Entry({ style_class: 'prompt-dialog-password-entry', | ||||
|                                           text: secret.value, can_focus: reactive, | ||||
|                                           reactive: reactive }); | ||||
|                                           reactive: reactive, | ||||
|                                           x_expand: true }); | ||||
|             ShellEntry.addContextMenu(secret.entry, | ||||
|                                       { isPassword: secret.password }); | ||||
|  | ||||
| @@ -116,10 +120,13 @@ const NetworkSecretDialog = new Lang.Class({ | ||||
|             } else | ||||
|                 secret.valid = true; | ||||
|  | ||||
|             layout.pack(label, 0, pos); | ||||
|             layout.child_set(label, { x_expand: false, y_fill: false, | ||||
|                                       x_align: Clutter.TableAlignment.START }); | ||||
|             layout.pack(secret.entry, 1, pos); | ||||
|             if (rtl) { | ||||
|                 layout.attach(secret.entry, 0, pos, 1, 1); | ||||
|                 layout.attach(label, 1, pos, 1, 1); | ||||
|             } else { | ||||
|                 layout.attach(label, 0, pos, 1, 1); | ||||
|                 layout.attach(secret.entry, 1, pos, 1, 1); | ||||
|             } | ||||
|             pos++; | ||||
|  | ||||
|             if (secret.password) | ||||
| @@ -373,6 +380,12 @@ const VPNRequestHandler = new Lang.Class({ | ||||
|             argv.push('-i'); | ||||
|         if (flags & NMClient.SecretAgentGetSecretsFlags.REQUEST_NEW) | ||||
|             argv.push('-r'); | ||||
|         if (authHelper.supportsHints) { | ||||
|             for (let i = 0; i < hints.length; i++) { | ||||
|                 argv.push('-t'); | ||||
|                 argv.push(hints[i]); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this._newStylePlugin = authHelper.externalUIMode; | ||||
|  | ||||
| @@ -591,7 +604,9 @@ const NetworkAgent = new Lang.Class({ | ||||
|     Name: 'NetworkAgent', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._native = new Shell.NetworkAgent({ identifier: 'org.gnome.Shell.NetworkAgent' }); | ||||
|         this._native = new Shell.NetworkAgent({ identifier: 'org.gnome.Shell.NetworkAgent', | ||||
|                                                 capabilities: NMClient.SecretAgentCapabilities.VPN_HINTS | ||||
|                                               }); | ||||
|  | ||||
|         this._dialogs = { }; | ||||
|         this._vpnRequests = { }; | ||||
| @@ -691,16 +706,23 @@ const NetworkAgent = new Lang.Class({ | ||||
|                     let service = keyfile.get_string('VPN Connection', 'service'); | ||||
|                     let binary = keyfile.get_string('GNOME', 'auth-dialog'); | ||||
|                     let externalUIMode = false; | ||||
|                     let hints = false; | ||||
|  | ||||
|                     try { | ||||
|                         externalUIMode = keyfile.get_boolean('GNOME', 'supports-external-ui-mode'); | ||||
|                     } catch(e) { } // ignore errors if key does not exist | ||||
|  | ||||
|                     try { | ||||
|                         hints = keyfile.get_boolean('GNOME', 'supports-hints'); | ||||
|                     } catch(e) { } // ignore errors if key does not exist | ||||
|  | ||||
|                     let path = binary; | ||||
|                     if (!GLib.path_is_absolute(path)) { | ||||
|                         path = GLib.build_filenamev([Config.LIBEXECDIR, path]); | ||||
|                     } | ||||
|  | ||||
|                     if (GLib.file_test(path, GLib.FileTest.IS_EXECUTABLE)) | ||||
|                         this._vpnBinaries[service] = { fileName: path, externalUIMode: externalUIMode }; | ||||
|                         this._vpnBinaries[service] = { fileName: path, externalUIMode: externalUIMode, supportsHints: hints }; | ||||
|                     else | ||||
|                         throw new Error('VPN plugin at %s is not executable'.format(path)); | ||||
|                 } catch(e) { | ||||
|   | ||||
| @@ -50,7 +50,7 @@ const AuthenticationDialog = new Lang.Class({ | ||||
|         mainContentBox.add(messageBox, | ||||
|                            { expand: true, y_align: St.Align.START }); | ||||
|  | ||||
|         this._subjectLabel = new St.Label({ style_class: 'prompt-dialog-headline', | ||||
|         this._subjectLabel = new St.Label({ style_class: 'prompt-dialog-headline headline', | ||||
|                                             text: _("Authentication Required") }); | ||||
|  | ||||
|         messageBox.add(this._subjectLabel, | ||||
|   | ||||
| @@ -36,6 +36,8 @@ const NotificationDirection = { | ||||
|     RECEIVED: 'chat-received' | ||||
| }; | ||||
|  | ||||
| const N_ = function(s) { return s; }; | ||||
|  | ||||
| function makeMessageFromTpMessage(tpMessage, direction) { | ||||
|     let [text, flags] = tpMessage.to_text(); | ||||
|  | ||||
| @@ -783,7 +785,6 @@ const ChatNotification = new Lang.Class({ | ||||
|  | ||||
|         this._createScrollArea(); | ||||
|         this._lastGroup = null; | ||||
|         this._lastGroupActor = null; | ||||
|  | ||||
|         // Keep track of the bottom position for the current adjustment and | ||||
|         // force a scroll to the bottom if things change while we were at the | ||||
| @@ -864,13 +865,6 @@ const ChatNotification = new Lang.Class({ | ||||
|             for (let i = 0; i < expired.length; i++) | ||||
|                 expired[i].actor.destroy(); | ||||
|         } | ||||
|  | ||||
|         let groups = this._contentArea.get_children(); | ||||
|         for (let i = 0; i < groups.length; i++) { | ||||
|             let group = groups[i]; | ||||
|             if (group.get_n_children() == 0) | ||||
|                 group.destroy(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -912,16 +906,19 @@ const ChatNotification = new Lang.Class({ | ||||
|             this._lastGroup = group; | ||||
|             let emptyLine = new St.Label({ style_class: 'chat-empty-line' }); | ||||
|             this.addActor(emptyLine); | ||||
|             this._history.unshift({ actor: emptyLine, time: timestamp, | ||||
|                                     realMessage: false }); | ||||
|         } | ||||
|  | ||||
|         this._lastMessageBox = new St.BoxLayout({ vertical: false }); | ||||
|         this._lastMessageBox.add(body, props.childProps); | ||||
|         this.addActor(this._lastMessageBox); | ||||
|         let lineBox = new St.BoxLayout({ vertical: false }); | ||||
|         lineBox.add(body, props.childProps); | ||||
|         this.addActor(lineBox); | ||||
|         this._lastMessageBox = lineBox; | ||||
|  | ||||
|         this.updated(); | ||||
|  | ||||
|         let timestamp = props.timestamp; | ||||
|         this._history.unshift({ actor: body, time: timestamp, | ||||
|         this._history.unshift({ actor: lineBox, time: timestamp, | ||||
|                                 realMessage: group != 'meta' }); | ||||
|  | ||||
|         if (!props.noTimestamp) { | ||||
| @@ -947,82 +944,78 @@ const ChatNotification = new Lang.Class({ | ||||
|  | ||||
|         let format; | ||||
|  | ||||
|         let desktopSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' }); | ||||
|         let desktopSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.interface' }); | ||||
|         let clockFormat = desktopSettings.get_string(CLOCK_FORMAT_KEY); | ||||
|         let hasAmPm = date.toLocaleFormat('%p') != ''; | ||||
|  | ||||
|         switch (clockFormat) { | ||||
|             case '24h': | ||||
|                 // Show only the time if date is on today | ||||
|                 if(daysAgo < 1){ | ||||
|                     /* Translators: Time in 24h format */ | ||||
|                     format = _("%H\u2236%M"); | ||||
|                 } | ||||
|                 // Show the word "Yesterday" and time if date is on yesterday | ||||
|                 else if(daysAgo <2){ | ||||
|                     /* Translators: this is the word "Yesterday" followed by a | ||||
|                      time string in 24h format. i.e. "Yesterday, 14:30" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("Yesterday, %H\u2236%M"); | ||||
|                 } | ||||
|                 // Show a week day and time if date is in the last week | ||||
|                 else if (daysAgo < 7) { | ||||
|                     /* Translators: this is the week day name followed by a time | ||||
|                      string in 24h format. i.e. "Monday, 14:30" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%A, %H\u2236%M"); | ||||
|         if (clockFormat == '24h' || !hasAmPm) { | ||||
|             // Show only the time if date is on today | ||||
|             if(daysAgo < 1){ | ||||
|                 /* Translators: Time in 24h format */ | ||||
|                 format = N_("%H\u2236%M"); | ||||
|             } | ||||
|             // Show the word "Yesterday" and time if date is on yesterday | ||||
|             else if(daysAgo <2){ | ||||
|                 /* Translators: this is the word "Yesterday" followed by a | ||||
|                  time string in 24h format. i.e. "Yesterday, 14:30" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("Yesterday, %H\u2236%M"); | ||||
|             } | ||||
|             // Show a week day and time if date is in the last week | ||||
|             else if (daysAgo < 7) { | ||||
|                 /* Translators: this is the week day name followed by a time | ||||
|                  string in 24h format. i.e. "Monday, 14:30" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%A, %H\u2236%M"); | ||||
|  | ||||
|                 } else if (date.getYear() == now.getYear()) { | ||||
|                     /* Translators: this is the month name and day number | ||||
|                      followed by a time string in 24h format. | ||||
|                      i.e. "May 25, 14:30" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%B %d, %H\u2236%M"); | ||||
|                 } else { | ||||
|                     /* Translators: this is the month name, day number, year | ||||
|                      number followed by a time string in 24h format. | ||||
|                      i.e. "May 25 2012, 14:30" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%B %d %Y, %H\u2236%M"); | ||||
|                 } | ||||
|                 break; | ||||
|         default: | ||||
|             /* explicit fall-through */ | ||||
|             case '12h': | ||||
|                 // Show only the time if date is on today | ||||
|                 if(daysAgo < 1){ | ||||
|                     /* Translators: Time in 24h format */ | ||||
|                     format = _("%l\u2236%M %p"); | ||||
|                 } | ||||
|                 // Show the word "Yesterday" and time if date is on yesterday | ||||
|                 else if(daysAgo <2){ | ||||
|                     /* Translators: this is the word "Yesterday" followed by a | ||||
|                      time string in 12h format. i.e. "Yesterday, 2:30 pm" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("Yesterday, %l\u2236%M %p"); | ||||
|                 } | ||||
|                 // Show a week day and time if date is in the last week | ||||
|                 else if (daysAgo < 7) { | ||||
|                     /* Translators: this is the week day name followed by a time | ||||
|                      string in 12h format. i.e. "Monday, 2:30 pm" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%A, %l\u2236%M %p"); | ||||
|             } else if (date.getYear() == now.getYear()) { | ||||
|                 /* Translators: this is the month name and day number | ||||
|                  followed by a time string in 24h format. | ||||
|                  i.e. "May 25, 14:30" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%B %d, %H\u2236%M"); | ||||
|             } else { | ||||
|                 /* Translators: this is the month name, day number, year | ||||
|                  number followed by a time string in 24h format. | ||||
|                  i.e. "May 25 2012, 14:30" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%B %d %Y, %H\u2236%M"); | ||||
|             } | ||||
|         } else { | ||||
|             // Show only the time if date is on today | ||||
|             if(daysAgo < 1){ | ||||
|                 /* Translators: Time in 12h format */ | ||||
|                 format = N_("%l\u2236%M %p"); | ||||
|             } | ||||
|             // Show the word "Yesterday" and time if date is on yesterday | ||||
|             else if(daysAgo <2){ | ||||
|                 /* Translators: this is the word "Yesterday" followed by a | ||||
|                  time string in 12h format. i.e. "Yesterday, 2:30 pm" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("Yesterday, %l\u2236%M %p"); | ||||
|             } | ||||
|             // Show a week day and time if date is in the last week | ||||
|             else if (daysAgo < 7) { | ||||
|                 /* Translators: this is the week day name followed by a time | ||||
|                  string in 12h format. i.e. "Monday, 2:30 pm" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%A, %l\u2236%M %p"); | ||||
|  | ||||
|                 } else if (date.getYear() == now.getYear()) { | ||||
|                     /* Translators: this is the month name and day number | ||||
|                      followed by a time string in 12h format. | ||||
|                      i.e. "May 25, 2:30 pm" */ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%B %d, %l\u2236%M %p"); | ||||
|                 } else { | ||||
|                     /* Translators: this is the month name, day number, year | ||||
|                      number followed by a time string in 12h format. | ||||
|                      i.e. "May 25 2012, 2:30 pm"*/ | ||||
|                     // xgettext:no-c-format | ||||
|                     format = _("%B %d %Y, %l\u2236%M %p"); | ||||
|                 } | ||||
|                 break; | ||||
|             } else if (date.getYear() == now.getYear()) { | ||||
|                 /* Translators: this is the month name and day number | ||||
|                  followed by a time string in 12h format. | ||||
|                  i.e. "May 25, 2:30 pm" */ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%B %d, %l\u2236%M %p"); | ||||
|             } else { | ||||
|                 /* Translators: this is the month name, day number, year | ||||
|                  number followed by a time string in 12h format. | ||||
|                  i.e. "May 25 2012, 2:30 pm"*/ | ||||
|                 // xgettext:no-c-format | ||||
|                 format = N_("%B %d %Y, %l\u2236%M %p"); | ||||
|             } | ||||
|         } | ||||
|         return date.toLocaleFormat(format); | ||||
|         return date.toLocaleFormat(Shell.util_translate_time_string(format)); | ||||
|     }, | ||||
|  | ||||
|     appendTimestamp: function() { | ||||
| @@ -1273,9 +1266,8 @@ const SubscriptionRequestNotification = new Lang.Class({ | ||||
|         let file = contact.get_avatar_file(); | ||||
|  | ||||
|         if (file) { | ||||
|             let uri = file.get_uri(); | ||||
|             let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|             iconBox.child = textureCache.load_uri_async(uri, iconBox._size, iconBox._size, scaleFactor); | ||||
|             iconBox.child = textureCache.load_file_async(file, iconBox._size, iconBox._size, scaleFactor); | ||||
|         } | ||||
|         else { | ||||
|             iconBox.child = new St.Icon({ icon_name: 'avatar-default', | ||||
|   | ||||
| @@ -87,7 +87,7 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|         if (Main.sessionMode.hasWindows && !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 windows = display.get_tab_list(Meta.TabList.DOCKS, screen.get_active_workspace ()); | ||||
|             let windowTracker = Shell.WindowTracker.get_default(); | ||||
|             let textureCache = St.TextureCache.get_default(); | ||||
|             for (let i = 0; i < windows.length; i++) { | ||||
| @@ -100,7 +100,7 @@ const CtrlAltTabManager = new Lang.Class({ | ||||
|                     if (app) | ||||
|                         icon = app.create_icon_texture(POPUP_APPICON_SIZE); | ||||
|                     else | ||||
|                         icon = textureCache.bind_pixbuf_property(windows[i], 'icon'); | ||||
|                         icon = textureCache.bind_cairo_surface_property(windows[i], 'icon'); | ||||
|                 } | ||||
|  | ||||
|                 items.push({ name: windows[i].title, | ||||
| @@ -140,27 +140,17 @@ const CtrlAltTabPopup = new Lang.Class({ | ||||
|     Name: 'CtrlAltTabPopup', | ||||
|     Extends: SwitcherPopup.SwitcherPopup, | ||||
|  | ||||
|     _createSwitcher: function() { | ||||
|     _init: function(items) { | ||||
|         this.parent(items); | ||||
|  | ||||
|         this._switcherList = new CtrlAltTabSwitcher(this._items); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _initialSelection: function(backward, binding) { | ||||
|         if (binding == 'switch-panels') { | ||||
|             if (backward) | ||||
|                 this._selectedIndex = this._items.length - 1; | ||||
|         } else if (binding == 'switch-panels-backward') { | ||||
|             if (!backward) | ||||
|                 this._selectedIndex = this._items.length - 1; | ||||
|         } | ||||
|         this._select(this._selectedIndex); | ||||
|     }, | ||||
|  | ||||
|     _keyPressHandler: function(keysym, backwards, action) { | ||||
|     _keyPressHandler: function(keysym, action) { | ||||
|         if (action == Meta.KeyBindingAction.SWITCH_PANELS) | ||||
|             this._select(backwards ? this._previous() : this._next()); | ||||
|             this._select(this._next()); | ||||
|         else if (action == Meta.KeyBindingAction.SWITCH_PANELS_BACKWARD) | ||||
|             this._select(backwards ? this._next() : this._previous()); | ||||
|             this._select(this._previous()); | ||||
|         else if (keysym == Clutter.Left) | ||||
|             this._select(this._previous()); | ||||
|         else if (keysym == Clutter.Right) | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Signals = imports.signals; | ||||
| const Lang = imports.lang; | ||||
| @@ -269,6 +270,9 @@ const ShowAppsIcon = new Lang.Class({ | ||||
|         if (app == null) | ||||
|             return false; | ||||
|  | ||||
|         if (!this._settings.is_writable('favorite-apps')) | ||||
|             return false; | ||||
|  | ||||
|         let id = app.get_id(); | ||||
|         let isFavorite = AppFavorites.getAppFavorites().isFavorite(id); | ||||
|         return isFavorite; | ||||
| @@ -424,6 +428,8 @@ const Dash = new Lang.Class({ | ||||
|  | ||||
|         this._workId = Main.initializeDeferredWork(this._box, Lang.bind(this, this._redisplay)); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' }); | ||||
|  | ||||
|         this._appSystem = Shell.AppSystem.get_default(); | ||||
|  | ||||
|         this._appSystem.connect('installed-changed', Lang.bind(this, function() { | ||||
| @@ -529,14 +535,17 @@ const Dash = new Lang.Class({ | ||||
|         let appIcon = new AppDisplay.AppIcon(app, | ||||
|                                              { setSizeManually: true, | ||||
|                                                showLabel: false }); | ||||
|         appIcon._draggable.connect('drag-begin', | ||||
|                                    Lang.bind(this, function() { | ||||
|                                        appIcon.actor.opacity = 50; | ||||
|                                    })); | ||||
|         appIcon._draggable.connect('drag-end', | ||||
|                                    Lang.bind(this, function() { | ||||
|                                        appIcon.actor.opacity = 255; | ||||
|                                    })); | ||||
|         if (appIcon._draggable) { | ||||
|             appIcon._draggable.connect('drag-begin', | ||||
|                                        Lang.bind(this, function() { | ||||
|                                            appIcon.actor.opacity = 50; | ||||
|                                        })); | ||||
|             appIcon._draggable.connect('drag-end', | ||||
|                                        Lang.bind(this, function() { | ||||
|                                            appIcon.actor.opacity = 255; | ||||
|                                        })); | ||||
|         } | ||||
|  | ||||
|         appIcon.connect('menu-state-changed', | ||||
|                         Lang.bind(this, function(appIcon, opened) { | ||||
|                             this._itemMenuStateChanged(item, opened); | ||||
| @@ -850,6 +859,9 @@ const Dash = new Lang.Class({ | ||||
|         if (app == null || app.is_window_backed()) | ||||
|             return DND.DragMotionResult.NO_DROP; | ||||
|  | ||||
|         if (!this._settings.is_writable('favorite-apps')) | ||||
|             return DND.DragMotionResult.NO_DROP; | ||||
|  | ||||
|         let favorites = AppFavorites.getAppFavorites().getFavorites(); | ||||
|         let numFavorites = favorites.length; | ||||
|  | ||||
| @@ -926,6 +938,9 @@ const Dash = new Lang.Class({ | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         if (!this._settings.is_writable('favorite-apps')) | ||||
|             return false; | ||||
|  | ||||
|         let id = app.get_id(); | ||||
|  | ||||
|         let favorites = AppFavorites.getAppFavorites().getFavoriteMap(); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GnomeDesktop = imports.gi.GnomeDesktop; | ||||
| const GObject = imports.gi.GObject; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Cairo = imports.cairo; | ||||
| @@ -18,8 +19,7 @@ const PanelMenu = imports.ui.panelMenu; | ||||
| const PopupMenu = imports.ui.popupMenu; | ||||
| const Calendar = imports.ui.calendar; | ||||
|  | ||||
| function _onVertSepRepaint (area) | ||||
| { | ||||
| function _onVertSepRepaint(area) { | ||||
|     let cr = area.get_context(); | ||||
|     let themeNode = area.get_theme_node(); | ||||
|     let [width, height] = area.get_surface_size(); | ||||
| @@ -33,7 +33,7 @@ function _onVertSepRepaint (area) | ||||
|     cr.setLineWidth(stippleWidth); | ||||
|     cr.stroke(); | ||||
|     cr.$dispose(); | ||||
| }; | ||||
| } | ||||
|  | ||||
| const DateMenuButton = new Lang.Class({ | ||||
|     Name: 'DateMenuButton', | ||||
| @@ -80,10 +80,6 @@ const DateMenuButton = new Lang.Class({ | ||||
|  | ||||
|         this._calendar.connect('selected-date-changed', | ||||
|                                Lang.bind(this, function(calendar, date) { | ||||
|                                   // we know this._eventList is defined here, because selected-data-changed | ||||
|                                   // only gets emitted when the user clicks a date in the calendar, | ||||
|                                   // and the calender makes those dates unclickable when instantiated with | ||||
|                                   // a null event source | ||||
|                                    this._eventList.setDate(date); | ||||
|  | ||||
|                                    // Make the button reactive only if the selected date is not the current date. | ||||
| @@ -125,14 +121,19 @@ const DateMenuButton = new Lang.Class({ | ||||
|             if (isOpen) { | ||||
|                 let now = new Date(); | ||||
|                 this._calendar.setDate(now); | ||||
|  | ||||
|                 /* Translators: This is the date format to use when the calendar popup is | ||||
|                  * shown - it is shown just below the time in the shell (e.g. "Tue 9:29 AM"). | ||||
|                  */ | ||||
|                 let dateFormat = Shell.util_translate_time_string (N_("%A %B %e, %Y")); | ||||
|                 this._date.set_label(now.toLocaleFormat(dateFormat)); | ||||
|             } | ||||
|         })); | ||||
|  | ||||
|         // Done with hbox for calendar and event list | ||||
|  | ||||
|         this._clock = new GnomeDesktop.WallClock(); | ||||
|         this._clock.connect('notify::clock', Lang.bind(this, this._updateClockAndDate)); | ||||
|         this._updateClockAndDate(); | ||||
|         this._clock.bind_property('clock', this._clockDisplay, 'text', GObject.BindingFlags.SYNC_CREATE); | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|         this._sessionUpdated(); | ||||
| @@ -142,7 +143,7 @@ const DateMenuButton = new Lang.Class({ | ||||
|         let now = new Date(); | ||||
|         return now.getYear() == date.getYear() && | ||||
|                now.getMonth() == date.getMonth() && | ||||
|                now.getDay() == date.getDay(); | ||||
|                now.getDate() == date.getDate(); | ||||
|     }, | ||||
|  | ||||
|     _appInstalledChanged: function() { | ||||
| @@ -168,6 +169,10 @@ const DateMenuButton = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getEventSource: function() { | ||||
|         return new Calendar.DBusEventSource(); | ||||
|     }, | ||||
|  | ||||
|     _setEventSource: function(eventSource) { | ||||
|         if (this._eventSource) | ||||
|             this._eventSource.destroy(); | ||||
| @@ -185,7 +190,7 @@ const DateMenuButton = new Lang.Class({ | ||||
|         let eventSource; | ||||
|         let showEvents = Main.sessionMode.showCalendarEvents; | ||||
|         if (showEvents) { | ||||
|             eventSource = new Calendar.DBusEventSource(); | ||||
|             eventSource = this._getEventSource(); | ||||
|         } else { | ||||
|             eventSource = new Calendar.EmptyEventSource(); | ||||
|         } | ||||
| @@ -197,25 +202,18 @@ const DateMenuButton = new Lang.Class({ | ||||
|         this._dateAndTimeSeparator.actor.visible = Main.sessionMode.allowSettings; | ||||
|     }, | ||||
|  | ||||
|     _updateClockAndDate: function() { | ||||
|         this._clockDisplay.set_text(this._clock.clock); | ||||
|         /* Translators: This is the date format to use when the calendar popup is | ||||
|          * shown - it is shown just below the time in the shell (e.g. "Tue 9:29 AM"). | ||||
|          */ | ||||
|         let dateFormat = _("%A %B %e, %Y"); | ||||
|         let displayDate = new Date(); | ||||
|         this._date.set_label(displayDate.toLocaleFormat(dateFormat)); | ||||
|     }, | ||||
|  | ||||
|     _getCalendarApp: function() { | ||||
|         if (this._calendarApp !== undefined) | ||||
|             return this._calendarApp; | ||||
|  | ||||
|         let apps = Gio.AppInfo.get_recommended_for_type('text/calendar'); | ||||
|         if (apps && (apps.length > 0)) | ||||
|             this._calendarApp = apps[0]; | ||||
|         else | ||||
|         if (apps && (apps.length > 0)) { | ||||
|             let app = Gio.AppInfo.get_default_for_type('text/calendar', false); | ||||
|             let defaultInRecommended = apps.some(function(a) { return a.equal(app); }); | ||||
|             this._calendarApp = defaultInRecommended ? app : apps[0]; | ||||
|         } else { | ||||
|             this._calendarApp = null; | ||||
|         } | ||||
|         return this._calendarApp; | ||||
|     }, | ||||
|  | ||||
|   | ||||
							
								
								
									
										84
									
								
								js/ui/edgeDragAction.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,84 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Signals = imports.signals; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const EDGE_THRESHOLD = 20; | ||||
| const DRAG_DISTANCE = 80; | ||||
|  | ||||
| const EdgeDragAction = new Lang.Class({ | ||||
|     Name: 'EdgeDragAction', | ||||
|     Extends: Clutter.GestureAction, | ||||
|  | ||||
|     _init : function(side, allowedModes) { | ||||
|         this.parent(); | ||||
|         this._side = side; | ||||
|         this._allowedModes = allowedModes; | ||||
|         this.set_n_touch_points(1); | ||||
|  | ||||
|         global.display.connect('grab-op-begin', Lang.bind(this, function() { | ||||
|             this.cancel(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _getMonitorRect : function (x, y) { | ||||
|         let rect = new Meta.Rectangle({ x: x - 1, y: y - 1, width: 1, height: 1 }); | ||||
|         let monitorIndex = global.screen.get_monitor_index_for_rect(rect); | ||||
|  | ||||
|         return global.screen.get_monitor_geometry(monitorIndex); | ||||
|     }, | ||||
|  | ||||
|     vfunc_gesture_prepare : function(action, actor) { | ||||
|         if (this.get_n_current_points() == 0) | ||||
|             return false; | ||||
|  | ||||
|         if (!(this._allowedModes & Main.actionMode)) | ||||
|             return false; | ||||
|  | ||||
|         let [x, y] = this.get_press_coords(0); | ||||
|         let monitorRect = this._getMonitorRect(x, y); | ||||
|  | ||||
|         return ((this._side == St.Side.LEFT && x < monitorRect.x + EDGE_THRESHOLD) || | ||||
|                 (this._side == St.Side.RIGHT && x > monitorRect.x + monitorRect.width - EDGE_THRESHOLD) || | ||||
|                 (this._side == St.Side.TOP && y < monitorRect.y + EDGE_THRESHOLD) || | ||||
|                 (this._side == St.Side.BOTTOM && y > monitorRect.y + monitorRect.height - EDGE_THRESHOLD)); | ||||
|     }, | ||||
|  | ||||
|     vfunc_gesture_progress : function (action, actor) { | ||||
|         let [startX, startY] = this.get_press_coords(0); | ||||
|         let [x, y] = this.get_motion_coords(0); | ||||
|         let offsetX = Math.abs (x - startX); | ||||
|         let offsetY = Math.abs (y - startY); | ||||
|  | ||||
|         if (offsetX < EDGE_THRESHOLD && offsetY < EDGE_THRESHOLD) | ||||
|             return true; | ||||
|  | ||||
|         if ((offsetX > offsetY && | ||||
|              (this._side == St.Side.TOP || this._side == St.Side.BOTTOM)) || | ||||
|             (offsetY > offsetX && | ||||
|              (this._side == St.Side.LEFT || this._side == St.Side.RIGHT))) { | ||||
|             this.cancel(); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     vfunc_gesture_end : function (action, actor) { | ||||
|         let [startX, startY] = this.get_press_coords(0); | ||||
|         let [x, y] = this.get_motion_coords(0); | ||||
|         let monitorRect = this._getMonitorRect(startX, startY); | ||||
|  | ||||
|         if ((this._side == St.Side.TOP && y > monitorRect.y + DRAG_DISTANCE) || | ||||
|             (this._side == St.Side.BOTTOM && y < monitorRect.y + monitorRect.height - DRAG_DISTANCE) || | ||||
|             (this._side == St.Side.LEFT && x > monitorRect.x + DRAG_DISTANCE) || | ||||
|             (this._side == St.Side.RIGHT && x < monitorRect.x + monitorRect.width - DRAG_DISTANCE)) | ||||
|             this.emit('activated'); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(EdgeDragAction.prototype); | ||||
| @@ -38,8 +38,6 @@ const UserWidget = imports.ui.userWidget; | ||||
|  | ||||
| let _endSessionDialog = null; | ||||
|  | ||||
| const TRIGGER_OFFLINE_UPDATE = '/usr/libexec/pk-trigger-offline-update'; | ||||
|  | ||||
| const _ITEM_ICON_SIZE = 48; | ||||
| const _DIALOG_ICON_SIZE = 48; | ||||
|  | ||||
| @@ -134,11 +132,18 @@ const restartInstallDialogContent = { | ||||
|     showOtherSessions: true, | ||||
| }; | ||||
|  | ||||
| const DialogType = { | ||||
|   LOGOUT: 0 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_LOGOUT */, | ||||
|   SHUTDOWN: 1 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_SHUTDOWN */, | ||||
|   RESTART: 2 /* GSM_SHELL_END_SESSION_DIALOG_TYPE_RESTART */, | ||||
|   UPDATE_RESTART: 3 | ||||
| }; | ||||
|  | ||||
| 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, | ||||
|     3: restartInstallDialogContent | ||||
|     0 /* DialogType.LOGOUT */: logoutDialogContent, | ||||
|     1 /* DialogType.SHUTDOWN */: shutdownDialogContent, | ||||
|     2 /* DialogType.RESTART */: restartDialogContent, | ||||
|     3 /* DialogType.UPDATE_RESTART */: restartInstallDialogContent | ||||
| }; | ||||
|  | ||||
| const MAX_USERS_IN_SESSION_DIALOG = 5; | ||||
| @@ -155,6 +160,19 @@ const LogindSessionIface = '<node> \ | ||||
|  | ||||
| const LogindSession = Gio.DBusProxy.makeProxyWrapper(LogindSessionIface); | ||||
|  | ||||
| const PkOfflineIface = '<node> \ | ||||
| <interface name="org.freedesktop.PackageKit.Offline"> \ | ||||
|     <property name="UpdatePrepared" type="b" access="read"/> \ | ||||
|     <property name="TriggerAction" type="s" access="read"/> \ | ||||
|     <method name="Trigger"> \ | ||||
|         <arg type="s" name="action" direction="in"/> \ | ||||
|     </method> \ | ||||
|     <method name="Cancel"/> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
| const PkOfflineProxy = Gio.DBusProxy.makeProxyWrapper(PkOfflineIface); | ||||
|  | ||||
| const UPowerIface = '<node> \ | ||||
| <interface name="org.freedesktop.UPower"> \ | ||||
|     <property name="OnBattery" type="b" access="read"/> \ | ||||
| @@ -245,9 +263,14 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         this._loginManager = LoginManager.getLoginManager(); | ||||
|         this._userManager = AccountsService.UserManager.get_default(); | ||||
|         this._user = this._userManager.get_user(GLib.get_user_name()); | ||||
|         this._updatesFile = Gio.File.new_for_path('/system-update'); | ||||
|         this._preparedUpdateFile = Gio.File.new_for_path('/var/lib/PackageKit/prepared-update'); | ||||
|  | ||||
|         this._pkOfflineProxy = new PkOfflineProxy(Gio.DBus.system, | ||||
|                                                   'org.freedesktop.PackageKit', | ||||
|                                                   '/org/freedesktop/PackageKit', | ||||
|                                                   Lang.bind(this, function(proxy, error) { | ||||
|                                                       if (error) | ||||
|                                                           log(error.message); | ||||
|                                                   })); | ||||
|         this._powerProxy = new UPowerProxy(Gio.DBus.system, | ||||
|                                            'org.freedesktop.UPower', | ||||
|                                            '/org/freedesktop/UPower', | ||||
| @@ -499,31 +522,29 @@ const EndSessionDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _triggerOfflineUpdateReboot: function(callback) { | ||||
|         this._pkexecSpawn([TRIGGER_OFFLINE_UPDATE, 'reboot'], callback); | ||||
|         this._pkOfflineProxy.TriggerRemote('reboot', | ||||
|                                            function (result, error) { | ||||
|             if (error) | ||||
|                 log(error.message); | ||||
|  | ||||
|             callback(); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     _triggerOfflineUpdateShutdown: function(callback) { | ||||
|         this._pkexecSpawn([TRIGGER_OFFLINE_UPDATE, 'power-off'], callback); | ||||
|         this._pkOfflineProxy.TriggerRemote('power-off', | ||||
|                                            function (result, error) { | ||||
|             if (error) | ||||
|                 log(error.message); | ||||
|  | ||||
|             callback(); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     _triggerOfflineUpdateCancel: function(callback) { | ||||
|         this._pkexecSpawn([TRIGGER_OFFLINE_UPDATE, '--cancel'], callback); | ||||
|     }, | ||||
|  | ||||
|     _pkexecSpawn: function(argv, callback) { | ||||
|         let ret, pid; | ||||
|         try { | ||||
|             [ret, pid] = GLib.spawn_async(null, ['pkexec'].concat(argv), null, | ||||
|                                           GLib.SpawnFlags.DO_NOT_REAP_CHILD | GLib.SpawnFlags.SEARCH_PATH, | ||||
|                                           null); | ||||
|         } catch (e) { | ||||
|             log('Error spawning "pkexec %s": %s'.format(argv.join(' '), e.toString())); | ||||
|             callback(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         GLib.child_watch_add(GLib.PRIORITY_DEFAULT, pid, function(pid, status) { | ||||
|             GLib.spawn_close_pid(pid); | ||||
|         this._pkOfflineProxy.CancelRemote(function (result, error) { | ||||
|             if (error) | ||||
|                 log(error.message); | ||||
|  | ||||
|             callback(); | ||||
|         }); | ||||
| @@ -677,8 +698,9 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         this._totalSecondsToStayOpen = totalSecondsToStayOpen; | ||||
|         this._type = type; | ||||
|  | ||||
|         if (this._type == 2 && this._updatesFile.query_exists(null)) | ||||
|             this._type = 3; | ||||
|         if (this._type == DialogType.RESTART && | ||||
|             this._pkOfflineProxy.TriggerAction == 'reboot') | ||||
|             this._type = DialogType.UPDATE_RESTART; | ||||
|  | ||||
|         this._applications = []; | ||||
|         this._applicationList.destroy_all_children(); | ||||
| @@ -705,19 +727,19 @@ const EndSessionDialog = new Lang.Class({ | ||||
|         if (dialogContent.showOtherSessions) | ||||
|             this._loadSessions(); | ||||
|  | ||||
|         let preparedUpdate = this._preparedUpdateFile.query_exists(null); | ||||
|         let updateAlreadyTriggered = this._updatesFile.query_exists(null); | ||||
|         let updateAlreadyTriggered = this._pkOfflineProxy.TriggerAction == 'power-off' || this._pkOfflineProxy.TriggerAction == 'reboot'; | ||||
|         let updatePrepared = this._pkOfflineProxy.UpdatePrepared; | ||||
|         let updatesAllowed = this._updatesPermission && this._updatesPermission.allowed; | ||||
|  | ||||
|         _setCheckBoxLabel(this._checkBox, dialogContent.checkBoxText); | ||||
|         this._checkBox.actor.visible = (dialogContent.checkBoxText && preparedUpdate && updatesAllowed); | ||||
|         this._checkBox.actor.checked = (preparedUpdate && updateAlreadyTriggered); | ||||
|         this._checkBox.actor.visible = (dialogContent.checkBoxText && updatePrepared && updatesAllowed); | ||||
|         this._checkBox.actor.checked = (updatePrepared && updateAlreadyTriggered); | ||||
|  | ||||
|         // We show the warning either together with the checkbox, or when | ||||
|         // updates have already been triggered, but the user doesn't have | ||||
|         // enough permissions to cancel them. | ||||
|         this._batteryWarning.visible = (dialogContent.showBatteryWarning && | ||||
|                                         (this._checkBox.actor.visible || preparedUpdate && updateAlreadyTriggered && !updatesAllowed)); | ||||
|                                         (this._checkBox.actor.visible || updatePrepared && updateAlreadyTriggered && !updatesAllowed)); | ||||
|  | ||||
|         this._updateButtons(); | ||||
|  | ||||
|   | ||||
| @@ -77,6 +77,7 @@ function init() { | ||||
|     window._ = Gettext.gettext; | ||||
|     window.C_ = Gettext.pgettext; | ||||
|     window.ngettext = Gettext.ngettext; | ||||
|     window.N_ = function(s) { return s; }; | ||||
|  | ||||
|     // Miscellaneous monkeypatching | ||||
|     _patchContainerClass(St.BoxLayout); | ||||
|   | ||||
| @@ -203,14 +203,16 @@ const InstallExtensionDialog = new Lang.Class({ | ||||
|  | ||||
|         let message = _("Download and install “%s” from extensions.gnome.org?").format(info.name); | ||||
|  | ||||
|         let box = new St.BoxLayout(); | ||||
|         let box = new St.BoxLayout({ style_class: 'prompt-dialog-main-layout', | ||||
|                                      vertical: false }); | ||||
|         this.contentLayout.add(box); | ||||
|  | ||||
|         let gicon = new Gio.FileIcon({ file: Gio.File.new_for_uri(REPOSITORY_URL_BASE + info.icon) }) | ||||
|         let icon = new St.Icon({ gicon: gicon }); | ||||
|         box.add(icon); | ||||
|  | ||||
|         let label = new St.Label({ text: message }); | ||||
|         let label = new St.Label({ style_class: 'prompt-dialog-headline', | ||||
|                                    text: message }); | ||||
|         box.add(label); | ||||
|     }, | ||||
|  | ||||
|   | ||||
| @@ -74,7 +74,7 @@ function disableExtension(uuid) { | ||||
|  | ||||
|     if (extension.stylesheet) { | ||||
|         let theme = St.ThemeContext.get_for_stage(global.stage).get_theme(); | ||||
|         theme.unload_stylesheet(extension.stylesheet.get_path()); | ||||
|         theme.unload_stylesheet(extension.stylesheet); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
| @@ -118,7 +118,7 @@ function enableExtension(uuid) { | ||||
|         let stylesheetFile = extension.dir.get_child(stylesheetNames[i]); | ||||
|         if (stylesheetFile.query_exists(null)) { | ||||
|             let theme = St.ThemeContext.get_for_stage(global.stage).get_theme(); | ||||
|             theme.load_stylesheet(stylesheetFile.get_path()); | ||||
|             theme.load_stylesheet(stylesheetFile); | ||||
|             extension.stylesheet = stylesheetFile; | ||||
|             break; | ||||
|         } | ||||
| @@ -271,10 +271,17 @@ function onEnabledExtensionsChanged() { | ||||
| } | ||||
|  | ||||
| function _onVersionValidationChanged() { | ||||
|     // we want to reload all extensions, but only enable | ||||
|     // extensions when allowed by the sessionMode, so | ||||
|     // temporarily disable them all | ||||
|     enabledExtensions = []; | ||||
|     for (let uuid in ExtensionUtils.extensions) | ||||
|         reloadExtension(ExtensionUtils.extensions[uuid]); | ||||
|     enabledExtensions = getEnabledExtensions(); | ||||
|  | ||||
|     if (Main.sessionMode.allowExtensions) { | ||||
|         enabledExtensions.forEach(function(uuid) { | ||||
|             if (ExtensionUtils.extensions[uuid]) | ||||
|                 reloadExtension(ExtensionUtils.extensions[uuid]); | ||||
|             enableExtension(uuid); | ||||
|         }); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -32,11 +32,9 @@ const FocusCaretTracker = new Lang.Class({ | ||||
|     Name: 'FocusCaretTracker', | ||||
|  | ||||
|     _init: function() { | ||||
|         Atspi.init(); | ||||
|         Atspi.set_timeout(250, 250); | ||||
|  | ||||
|         this._atspiListener = Atspi.EventListener.new(Lang.bind(this, this._onChanged)); | ||||
|  | ||||
|         this._atspiInited = false; | ||||
|         this._focusListenerRegistered = false; | ||||
|         this._caretListenerRegistered = false; | ||||
|     }, | ||||
| @@ -48,12 +46,20 @@ const FocusCaretTracker = new Lang.Class({ | ||||
|             this.emit('caret-moved', event); | ||||
|     }, | ||||
|  | ||||
|     _initAtspi: function() { | ||||
|         if (!this._atspiInited) { | ||||
|             Atspi.init(); | ||||
|             Atspi.set_timeout(250, 250); | ||||
|             this._atspiInited = true; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     registerFocusListener: function() { | ||||
|         if (this._focusListenerRegistered) | ||||
|             return; | ||||
|  | ||||
|         // Ignore the return value, we get an exception if they fail | ||||
|         // And they should never fail | ||||
|         this._initAtspi(); | ||||
|  | ||||
|         this._atspiListener.register(STATECHANGED + ':focused'); | ||||
|         this._atspiListener.register(STATECHANGED + ':selected'); | ||||
|         this._focusListenerRegistered = true; | ||||
| @@ -63,6 +69,8 @@ const FocusCaretTracker = new Lang.Class({ | ||||
|         if (this._caretListenerRegistered) | ||||
|             return; | ||||
|  | ||||
|         this._initAtspi(); | ||||
|  | ||||
|         this._atspiListener.register(CARETMOVED); | ||||
|         this._caretListenerRegistered = true; | ||||
|     }, | ||||
|   | ||||
| @@ -56,7 +56,7 @@ const GrabHelper = new Lang.Class({ | ||||
|         this._grabStack = []; | ||||
|  | ||||
|         this._actors = []; | ||||
|         this._ignoreRelease = false; | ||||
|         this._ignoreUntilRelease = false; | ||||
|  | ||||
|         this._modalCount = 0; | ||||
|     }, | ||||
| @@ -215,7 +215,7 @@ const GrabHelper = new Lang.Class({ | ||||
|  | ||||
|         _popGrabHelper(this); | ||||
|  | ||||
|         this._ignoreRelease = false; | ||||
|         this._ignoreUntilRelease = false; | ||||
|  | ||||
|         Main.popModal(this._owner); | ||||
|         global.sync_pointer(); | ||||
| @@ -228,7 +228,7 @@ const GrabHelper = new Lang.Class({ | ||||
|     // like the ComboBoxMenu that go away on press, but need to eat | ||||
|     // the next release event. | ||||
|     ignoreRelease: function() { | ||||
|         this._ignoreRelease = true; | ||||
|         this._ignoreUntilRelease = true; | ||||
|     }, | ||||
|  | ||||
|     // ungrab: | ||||
| @@ -283,12 +283,22 @@ const GrabHelper = new Lang.Class({ | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } | ||||
|  | ||||
|         let motion = type == Clutter.EventType.MOTION; | ||||
|         let press = type == Clutter.EventType.BUTTON_PRESS; | ||||
|         let release = type == Clutter.EventType.BUTTON_RELEASE; | ||||
|         let button = press || release; | ||||
|  | ||||
|         if (release && this._ignoreRelease) { | ||||
|             this._ignoreRelease = false; | ||||
|         let touchUpdate = type == Clutter.EventType.TOUCH_UPDATE; | ||||
|         let touchBegin = type == Clutter.EventType.TOUCH_BEGIN; | ||||
|         let touchEnd = type == Clutter.EventType.TOUCH_END; | ||||
|         let touch = touchUpdate || touchBegin || touchEnd; | ||||
|  | ||||
|         if (touch && !global.display.is_pointer_emulating_sequence (event.get_event_sequence())) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         if (this._ignoreUntilRelease && (motion || release || touch)) { | ||||
|             if (release || touchEnd) | ||||
|                 this._ignoreUntilRelease = false; | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } | ||||
|  | ||||
| @@ -298,11 +308,12 @@ const GrabHelper = new Lang.Class({ | ||||
|         if (Main.keyboard.shouldTakeEvent(event)) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         if (button) { | ||||
|             // If we have a press event, ignore the next event, | ||||
|             // which should be a release event. | ||||
|             if (press) | ||||
|                 this._ignoreRelease = true; | ||||
|         if (button || touchBegin) { | ||||
|             // If we have a press event, ignore the next | ||||
|             // motion/release events. | ||||
|             if (press || touchBegin) | ||||
|                 this._ignoreUntilRelease = true; | ||||
|  | ||||
|             let i = this._actorInGrabStack(event.get_source()) + 1; | ||||
|             this.ungrab({ actor: this._grabStack[i].actor, isUser: true }); | ||||
|             return Clutter.EVENT_STOP; | ||||
|   | ||||
| @@ -11,6 +11,9 @@ const Main = imports.ui.main; | ||||
|  | ||||
| const MAX_CANDIDATES_PER_PAGE = 16; | ||||
|  | ||||
| const DEFAULT_INDEX_LABELS = [ '1', '2', '3', '4', '5', '6', '7', '8', | ||||
|                                '9', '0', 'a', 'b', 'c', 'd', 'e', 'f' ]; | ||||
|  | ||||
| const CandidateArea = new Lang.Class({ | ||||
|     Name: 'CandidateArea', | ||||
|  | ||||
| @@ -38,11 +41,11 @@ const CandidateArea = new Lang.Class({ | ||||
|  | ||||
|         this._buttonBox = new St.BoxLayout({ style_class: 'candidate-page-button-box' }); | ||||
|  | ||||
|         this._previousButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-previous' }); | ||||
|         this._previousButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-previous button' }); | ||||
|         this._previousButton.child = new St.Icon({ style_class: 'candidate-page-button-icon' }); | ||||
|         this._buttonBox.add(this._previousButton, { expand: true }); | ||||
|  | ||||
|         this._nextButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-next' }); | ||||
|         this._nextButton = new St.Button({ style_class: 'candidate-page-button candidate-page-button-next button' }); | ||||
|         this._nextButton.child = new St.Icon({ style_class: 'candidate-page-button-icon' }); | ||||
|         this._buttonBox.add(this._nextButton, { expand: true }); | ||||
|  | ||||
| @@ -89,7 +92,7 @@ const CandidateArea = new Lang.Class({ | ||||
|             if (!visible) | ||||
|                 continue; | ||||
|  | ||||
|             box._indexLabel.text = ((indexes && indexes[i]) ? indexes[i] : '%x'.format(i + 1)); | ||||
|             box._indexLabel.text = ((indexes && indexes[i]) ? indexes[i] : DEFAULT_INDEX_LABELS[i]); | ||||
|             box._candidateLabel.text = candidates[i]; | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -10,12 +10,30 @@ const St = imports.gi.St; | ||||
| const Lang = imports.lang; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const Main = imports.ui.main; | ||||
|  | ||||
| const ICON_SIZE = 96; | ||||
| const MIN_ICON_SIZE = 16; | ||||
|  | ||||
| const EXTRA_SPACE_ANIMATION_TIME = 0.25; | ||||
|  | ||||
| const ANIMATION_TIME_IN = 0.350; | ||||
| const ANIMATION_TIME_OUT = 1/2 * ANIMATION_TIME_IN; | ||||
| const ANIMATION_MAX_DELAY_FOR_ITEM = 2/3 * ANIMATION_TIME_IN; | ||||
| const ANIMATION_BASE_DELAY_FOR_ITEM = 1/4 * ANIMATION_MAX_DELAY_FOR_ITEM; | ||||
| const ANIMATION_MAX_DELAY_OUT_FOR_ITEM = 2/3 * ANIMATION_TIME_OUT; | ||||
| const ANIMATION_FADE_IN_TIME_FOR_ITEM = 1/4 * ANIMATION_TIME_IN; | ||||
|  | ||||
| const ANIMATION_BOUNCE_ICON_SCALE = 1.1; | ||||
|  | ||||
| const AnimationDirection = { | ||||
|     IN: 0, | ||||
|     OUT: 1 | ||||
| }; | ||||
|  | ||||
| const APPICON_ANIMATION_OUT_SCALE = 3; | ||||
| const APPICON_ANIMATION_OUT_TIME = 0.25; | ||||
|  | ||||
| const BaseIcon = new Lang.Class({ | ||||
|     Name: 'BaseIcon', | ||||
|  | ||||
| @@ -173,9 +191,55 @@ const BaseIcon = new Lang.Class({ | ||||
|  | ||||
|     _onIconThemeChanged: function() { | ||||
|         this._createIconTexture(this.iconSize); | ||||
|     }, | ||||
|  | ||||
|     animateZoomOut: function() { | ||||
|         // Animate only the child instead of the entire actor, so the | ||||
|         // styles like hover and running are not applied while | ||||
|         // animating. | ||||
|         zoomOutActor(this.actor.child); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function clamp(value, min, max) { | ||||
|     return Math.max(Math.min(value, max), min); | ||||
| }; | ||||
|  | ||||
| function zoomOutActor(actor) { | ||||
|     let actorClone = new Clutter.Clone({ source: actor, | ||||
|                                          reactive: false }); | ||||
|     let [width, height] = actor.get_transformed_size(); | ||||
|     let [x, y] = actor.get_transformed_position(); | ||||
|     actorClone.set_size(width, height); | ||||
|     actorClone.set_position(x, y); | ||||
|     actorClone.opacity = 255; | ||||
|     actorClone.set_pivot_point(0.5, 0.5); | ||||
|  | ||||
|     Main.uiGroup.add_actor(actorClone); | ||||
|  | ||||
|     // Avoid monitor edges to not zoom outside the current monitor | ||||
|     let monitor = Main.layoutManager.findMonitorForActor(actor); | ||||
|     let scaledWidth = width * APPICON_ANIMATION_OUT_SCALE; | ||||
|     let scaledHeight = height * APPICON_ANIMATION_OUT_SCALE; | ||||
|     let scaledX = x - (scaledWidth - width) / 2; | ||||
|     let scaledY = y - (scaledHeight - height) / 2; | ||||
|     let containedX = clamp(scaledX, monitor.x, monitor.x + monitor.width - scaledWidth); | ||||
|     let containedY = clamp(scaledY, monitor.y, monitor.y + monitor.height - scaledHeight); | ||||
|  | ||||
|     Tweener.addTween(actorClone, | ||||
|                      { time: APPICON_ANIMATION_OUT_TIME, | ||||
|                        scale_x: APPICON_ANIMATION_OUT_SCALE, | ||||
|                        scale_y: APPICON_ANIMATION_OUT_SCALE, | ||||
|                        translation_x: containedX - scaledX, | ||||
|                        translation_y: containedY - scaledY, | ||||
|                        opacity: 0, | ||||
|                        transition: 'easeOutQuad', | ||||
|                        onComplete: function() { | ||||
|                            actorClone.destroy(); | ||||
|                        } | ||||
|                     }); | ||||
| } | ||||
|  | ||||
| const IconGrid = new Lang.Class({ | ||||
|     Name: 'IconGrid', | ||||
|  | ||||
| @@ -338,15 +402,202 @@ const IconGrid = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _calculateChildBox: function(child, x, y, box) { | ||||
|         let [childMinWidth, childMinHeight, childNaturalWidth, childNaturalHeight] = | ||||
|              child.get_preferred_size(); | ||||
|     /** | ||||
|      * Intended to be override by subclasses if they need a different | ||||
|      * set of items to be animated. | ||||
|      */ | ||||
|     _getChildrenToAnimate: function() { | ||||
|         return this._getVisibleChildren(); | ||||
|     }, | ||||
|  | ||||
|     _animationDone: function() { | ||||
|         this._animating = false; | ||||
|         this.emit('animation-done'); | ||||
|     }, | ||||
|  | ||||
|     animatePulse: function(animationDirection) { | ||||
|         if (animationDirection != AnimationDirection.IN) | ||||
|             throw new Error("Pulse animation only implements 'in' animation direction"); | ||||
|  | ||||
|         if (this._animating) | ||||
|             return; | ||||
|  | ||||
|         this._animating = true; | ||||
|  | ||||
|         let actors = this._getChildrenToAnimate(); | ||||
|         if (actors.length == 0) { | ||||
|             this._animationDone(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // For few items the animation can be slow, so use a smaller | ||||
|         // delay when there are less than 4 items | ||||
|         // (ANIMATION_BASE_DELAY_FOR_ITEM = 1/4 * | ||||
|         // ANIMATION_MAX_DELAY_FOR_ITEM) | ||||
|         let maxDelay = Math.min(ANIMATION_BASE_DELAY_FOR_ITEM * actors.length, | ||||
|                                 ANIMATION_MAX_DELAY_FOR_ITEM); | ||||
|  | ||||
|         for (let index = 0; index < actors.length; index++) { | ||||
|             let actor = actors[index]; | ||||
|             actor.reactive = false; | ||||
|             actor.set_scale(0, 0); | ||||
|             actor.set_pivot_point(0.5, 0.5); | ||||
|  | ||||
|             let delay = index / actors.length * maxDelay; | ||||
|             let bounceUpTime = ANIMATION_TIME_IN / 4; | ||||
|             let isLastItem = index == actors.length - 1; | ||||
|             Tweener.addTween(actor, | ||||
|                             { time: bounceUpTime, | ||||
|                               transition: 'easeInOutQuad', | ||||
|                               delay: delay, | ||||
|                               scale_x: ANIMATION_BOUNCE_ICON_SCALE, | ||||
|                               scale_y: ANIMATION_BOUNCE_ICON_SCALE, | ||||
|                               onComplete: Lang.bind(this, function() { | ||||
|                                   Tweener.addTween(actor, | ||||
|                                                    { time: ANIMATION_TIME_IN - bounceUpTime, | ||||
|                                                      transition: 'easeInOutQuad', | ||||
|                                                      scale_x: 1, | ||||
|                                                      scale_y: 1, | ||||
|                                                      onComplete: Lang.bind(this, function() { | ||||
|                                                         if (isLastItem) | ||||
|                                                             this._animationDone(); | ||||
|                                                         actor.reactive = true; | ||||
|                                                     }) | ||||
|                                                    }); | ||||
|                               }) | ||||
|                             }); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     animateSpring: function(animationDirection, sourceActor) { | ||||
|         if (this._animating) | ||||
|             return; | ||||
|  | ||||
|         this._animating = true; | ||||
|  | ||||
|         let actors = this._getChildrenToAnimate(); | ||||
|         if (actors.length == 0) { | ||||
|             this._animationDone(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let [sourceX, sourceY] = sourceActor.get_transformed_position(); | ||||
|         let [sourceWidth, sourceHeight] = sourceActor.get_size(); | ||||
|         // Get the center | ||||
|         let [sourceCenterX, sourceCenterY] = [sourceX + sourceWidth / 2, sourceY + sourceHeight / 2]; | ||||
|         // Design decision, 1/2 of the source actor size. | ||||
|         let [sourceScaledWidth, sourceScaledHeight] = [sourceWidth / 2, sourceHeight / 2]; | ||||
|  | ||||
|         actors.forEach(function(actor) { | ||||
|             let [actorX, actorY] = actor._transformedPosition = actor.get_transformed_position(); | ||||
|             let [x, y] = [actorX - sourceX, actorY - sourceY]; | ||||
|             actor._distance = Math.sqrt(x * x + y * y); | ||||
|         }); | ||||
|         let maxDist = actors.reduce(function(prev, cur) { | ||||
|             return Math.max(prev, cur._distance); | ||||
|         }, 0); | ||||
|         let minDist = actors.reduce(function(prev, cur) { | ||||
|             return Math.min(prev, cur._distance); | ||||
|         }, Infinity); | ||||
|         let normalization = maxDist - minDist; | ||||
|  | ||||
|         for (let index = 0; index < actors.length; index++) { | ||||
|             let actor = actors[index]; | ||||
|             actor.opacity = 0; | ||||
|             actor.reactive = false; | ||||
|  | ||||
|             let actorClone = new Clutter.Clone({ source: actor }); | ||||
|             Main.uiGroup.add_actor(actorClone); | ||||
|  | ||||
|             let [width, height,,] = this._getAllocatedChildSizeAndSpacing(actor); | ||||
|             actorClone.set_size(width, height); | ||||
|             let scaleX = sourceScaledWidth / width; | ||||
|             let scaleY = sourceScaledHeight / height; | ||||
|             let [adjustedSourcePositionX, adjustedSourcePositionY] = [sourceCenterX - sourceScaledWidth / 2, sourceCenterY - sourceScaledHeight / 2]; | ||||
|  | ||||
|             let movementParams, fadeParams; | ||||
|             if (animationDirection == AnimationDirection.IN) { | ||||
|                 let isLastItem = actor._distance == minDist; | ||||
|  | ||||
|                 actorClone.opacity = 0; | ||||
|                 actorClone.set_scale(scaleX, scaleY); | ||||
|  | ||||
|                 actorClone.set_position(adjustedSourcePositionX, adjustedSourcePositionY); | ||||
|  | ||||
|                 let delay = (1 - (actor._distance - minDist) / normalization) * ANIMATION_MAX_DELAY_FOR_ITEM; | ||||
|                 let [finalX, finalY]  = actor._transformedPosition; | ||||
|                 movementParams = { time: ANIMATION_TIME_IN, | ||||
|                                    transition: 'easeInOutQuad', | ||||
|                                    delay: delay, | ||||
|                                    x: finalX, | ||||
|                                    y: finalY, | ||||
|                                    scale_x: 1, | ||||
|                                    scale_y: 1, | ||||
|                                    onComplete: Lang.bind(this, function() { | ||||
|                                        if (isLastItem) | ||||
|                                            this._animationDone(); | ||||
|  | ||||
|                                        actor.opacity = 255; | ||||
|                                        actor.reactive = true; | ||||
|                                        actorClone.destroy(); | ||||
|                                    })}; | ||||
|                 fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM, | ||||
|                                transition: 'easeInOutQuad', | ||||
|                                delay: delay, | ||||
|                                opacity: 255 }; | ||||
|             } else { | ||||
|                 let isLastItem = actor._distance == maxDist; | ||||
|  | ||||
|                 let [startX, startY]  = actor._transformedPosition; | ||||
|                 actorClone.set_position(startX, startY); | ||||
|  | ||||
|                 let delay = (actor._distance - minDist) / normalization * ANIMATION_MAX_DELAY_OUT_FOR_ITEM; | ||||
|                 movementParams = { time: ANIMATION_TIME_OUT, | ||||
|                                    transition: 'easeInOutQuad', | ||||
|                                    delay: delay, | ||||
|                                    x: adjustedSourcePositionX, | ||||
|                                    y: adjustedSourcePositionY, | ||||
|                                    scale_x: scaleX, | ||||
|                                    scale_y: scaleY, | ||||
|                                    onComplete: Lang.bind(this, function() { | ||||
|                                        if (isLastItem) { | ||||
|                                            this._animationDone(); | ||||
|                                            this._restoreItemsOpacity(); | ||||
|                                        } | ||||
|                                        actor.reactive = true; | ||||
|                                        actorClone.destroy(); | ||||
|                                    })}; | ||||
|                 fadeParams = { time: ANIMATION_FADE_IN_TIME_FOR_ITEM, | ||||
|                                transition: 'easeInOutQuad', | ||||
|                                delay: ANIMATION_TIME_OUT + delay - ANIMATION_FADE_IN_TIME_FOR_ITEM, | ||||
|                                opacity: 0 }; | ||||
|             } | ||||
|  | ||||
|  | ||||
|             Tweener.addTween(actorClone, movementParams); | ||||
|             Tweener.addTween(actorClone, fadeParams); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _restoreItemsOpacity: function() { | ||||
|         for (let index = 0; index < this._items.length; index++) { | ||||
|             this._items[index].actor.opacity = 255; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _getAllocatedChildSizeAndSpacing: function(child) { | ||||
|         let [,, natWidth, natHeight] = child.get_preferred_size(); | ||||
|         let width = Math.min(this._getHItemSize(), natWidth); | ||||
|         let xSpacing = Math.max(0, width - natWidth) / 2; | ||||
|         let height = Math.min(this._getVItemSize(), natHeight); | ||||
|         let ySpacing = Math.max(0, height - natHeight) / 2; | ||||
|         return [width, height, xSpacing, ySpacing]; | ||||
|     }, | ||||
|  | ||||
|     _calculateChildBox: function(child, x, y, box) { | ||||
|         /* Center the item in its allocation horizontally */ | ||||
|         let width = Math.min(this._getHItemSize(), childNaturalWidth); | ||||
|         let childXSpacing = Math.max(0, width - childNaturalWidth) / 2; | ||||
|         let height = Math.min(this._getVItemSize(), childNaturalHeight); | ||||
|         let childYSpacing = Math.max(0, height - childNaturalHeight) / 2; | ||||
|         let [width, height, childXSpacing, childYSpacing] = | ||||
|             this._getAllocatedChildSizeAndSpacing(child); | ||||
|  | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|         if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) { | ||||
| @@ -522,21 +773,17 @@ const IconGrid = new Lang.Class({ | ||||
|             this._fixedHItemSize = Math.max(this._hItemSize - neededSpacePerItem, MIN_ICON_SIZE); | ||||
|             this._fixedVItemSize = Math.max(this._vItemSize - neededSpacePerItem, MIN_ICON_SIZE); | ||||
|  | ||||
|             if (this._fixedHItemSize < MIN_ICON_SIZE) | ||||
|                 this._fixedHItemSize = MIN_ICON_SIZE; | ||||
|             if (this._fixedVItemSize < MIN_ICON_SIZE) | ||||
|                 this._fixedVItemSize = MIN_ICON_SIZE; | ||||
|  | ||||
|             this._updateSpacingForSize(availWidth, availHeight); | ||||
|         } | ||||
|         let scale = Math.min(this._fixedHItemSize, this._fixedVItemSize) / Math.max(this._hItemSize, this._vItemSize); | ||||
|         Meta.later_add(Meta.LaterType.BEFORE_REDRAW, Lang.bind(this, function() { this._updateChildrenScale(scale); })); | ||||
|         Meta.later_add(Meta.LaterType.BEFORE_REDRAW, | ||||
|                        Lang.bind(this, this._updateIconSizes)); | ||||
|     }, | ||||
|  | ||||
|     // Note that this is ICON_SIZE as used by BaseIcon, not elsewhere in IconGrid; it's a bit messed up | ||||
|     _updateChildrenScale: function(scale) { | ||||
|     _updateIconSizes: function() { | ||||
|         let scale = Math.min(this._fixedHItemSize, this._fixedVItemSize) / Math.max(this._hItemSize, this._vItemSize); | ||||
|         let newIconSize = Math.floor(ICON_SIZE * scale); | ||||
|         for (let i in this._items) { | ||||
|             let newIconSize = Math.floor(ICON_SIZE * scale); | ||||
|             this._items[i].icon.setIconSize(newIconSize); | ||||
|         } | ||||
|     } | ||||
| @@ -550,6 +797,7 @@ const PaginatedIconGrid = new Lang.Class({ | ||||
|     _init: function(params) { | ||||
|         this.parent(params); | ||||
|         this._nPages = 0; | ||||
|         this.currentPage = 0; | ||||
|         this._rowsPerPage = 0; | ||||
|         this._spaceBetweenPages = 0; | ||||
|         this._childrenPerPage = 0; | ||||
| @@ -613,6 +861,15 @@ const PaginatedIconGrid = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     // Overriden from IconGrid | ||||
|     _getChildrenToAnimate: function() { | ||||
|         let children = this._getVisibleChildren(); | ||||
|         let firstIndex = this._childrenPerPage * this.currentPage; | ||||
|         let lastIndex = firstIndex + this._childrenPerPage; | ||||
|  | ||||
|         return children.slice(firstIndex, lastIndex); | ||||
|     }, | ||||
|  | ||||
|     _computePages: function (availWidthPerPage, availHeightPerPage) { | ||||
|         let [nColumns, usedWidth] = this._computeLayout(availWidthPerPage); | ||||
|         let nRows; | ||||
|   | ||||
| @@ -23,6 +23,12 @@ const KEYBOARD_TYPE = 'keyboard-type'; | ||||
| const A11Y_APPLICATIONS_SCHEMA = 'org.gnome.desktop.a11y.applications'; | ||||
| const SHOW_KEYBOARD = 'screen-keyboard-enabled'; | ||||
|  | ||||
| const CURSOR_BUS_NAME = 'org.gnome.SettingsDaemon.Cursor'; | ||||
| const CURSOR_OBJECT_PATH = '/org/gnome/SettingsDaemon/Cursor'; | ||||
|  | ||||
| const CARIBOU_BUS_NAME = 'org.gnome.Caribou.Daemon'; | ||||
| const CARIBOU_OBJECT_PATH = '/org/gnome/Caribou/Daemon'; | ||||
|  | ||||
| const CaribouKeyboardIface = '<node> \ | ||||
| <interface name="org.gnome.Caribou.Keyboard"> \ | ||||
| <method name="Show"> \ | ||||
| @@ -47,13 +53,29 @@ const CaribouKeyboardIface = '<node> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
| const CaribouDaemonIface = '<node> \ | ||||
| <interface name="org.gnome.Caribou.Daemon"> \ | ||||
| <method name="Run" /> \ | ||||
| <method name="Quit" /> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
| const CursorManagerIface = '<node> \ | ||||
| <interface name="org.gnome.SettingsDaemon.Cursor"> \ | ||||
| <property name="ShowOSK" type="b" access="read" /> \ | ||||
| </interface> \ | ||||
| </node>'; | ||||
|  | ||||
| const CaribouDaemonProxy = Gio.DBusProxy.makeProxyWrapper(CaribouDaemonIface); | ||||
| const CursorManagerProxy = Gio.DBusProxy.makeProxyWrapper(CursorManagerIface); | ||||
|  | ||||
| const Key = new Lang.Class({ | ||||
|     Name: 'Key', | ||||
|  | ||||
|     _init : function(key) { | ||||
|         this._key = key; | ||||
|  | ||||
|         this.actor = this._makeKey(); | ||||
|         this.actor = this._makeKey(key, GLib.markup_escape_text(key.label, -1)); | ||||
|         this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); | ||||
|  | ||||
|         this._extended_keys = this._key.get_extended_keys(); | ||||
|         this._extended_keyboard = null; | ||||
| @@ -76,20 +98,26 @@ const Key = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _makeKey: function () { | ||||
|         let label = GLib.markup_escape_text(this._key.label, -1); | ||||
|     _onDestroy: function() { | ||||
|         if (this._boxPointer) { | ||||
|             this._boxPointer.actor.destroy(); | ||||
|             this._boxPointer = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _makeKey: function (key, label) { | ||||
|         let button = new St.Button ({ label: label, | ||||
|                                       style_class: 'keyboard-key' }); | ||||
|  | ||||
|         button.key_width = this._key.width; | ||||
|         button.connect('button-press-event', Lang.bind(this, | ||||
|             function () { | ||||
|                 this._key.press(); | ||||
|                 key.press(); | ||||
|                 return Clutter.EVENT_PROPAGATE; | ||||
|             })); | ||||
|         button.connect('button-release-event', Lang.bind(this, | ||||
|             function () { | ||||
|                 this._key.release(); | ||||
|                 key.release(); | ||||
|                 return Clutter.EVENT_PROPAGATE; | ||||
|             })); | ||||
|  | ||||
| @@ -112,18 +140,9 @@ const Key = new Lang.Class({ | ||||
|         for (let i = 0; i < this._extended_keys.length; ++i) { | ||||
|             let extended_key = this._extended_keys[i]; | ||||
|             let label = this._getUnichar(extended_key); | ||||
|             let key = new St.Button({ label: label, style_class: 'keyboard-key' }); | ||||
|             let key = this._makeKey(extended_key, label); | ||||
|  | ||||
|             key.extended_key = extended_key; | ||||
|             key.connect('button-press-event', Lang.bind(this, | ||||
|                 function () { | ||||
|                     extended_key.press(); | ||||
|                     return Clutter.EVENT_PROPAGATE; | ||||
|                 })); | ||||
|             key.connect('button-release-event', Lang.bind(this, | ||||
|                 function () { | ||||
|                     extended_key.release(); | ||||
|                     return Clutter.EVENT_PROPAGATE; | ||||
|                 })); | ||||
|             this._extended_keyboard.add(key); | ||||
|         } | ||||
|         this._boxPointer.bin.add_actor(this._extended_keyboard); | ||||
| @@ -161,11 +180,26 @@ const Keyboard = new Lang.Class({ | ||||
|  | ||||
|         this._timestamp = global.display.get_current_time_roundtrip(); | ||||
|  | ||||
|         this._keyboardSettings = new Gio.Settings({ schema: KEYBOARD_SCHEMA }); | ||||
|         this._keyboardSettings.connect('changed', Lang.bind(this, this._settingsChanged)); | ||||
|         this._a11yApplicationsSettings = new Gio.Settings({ schema: A11Y_APPLICATIONS_SCHEMA }); | ||||
|         this._a11yApplicationsSettings.connect('changed', Lang.bind(this, this._settingsChanged)); | ||||
|         this._settingsChanged(); | ||||
|         this._keyboardSettings = new Gio.Settings({ schema_id: KEYBOARD_SCHEMA }); | ||||
|         this._keyboardSettings.connect('changed', Lang.bind(this, this._sync)); | ||||
|         this._a11yApplicationsSettings = new Gio.Settings({ schema_id: A11Y_APPLICATIONS_SCHEMA }); | ||||
|         this._a11yApplicationsSettings.connect('changed', Lang.bind(this, this._sync)); | ||||
|         this._watchNameId = Gio.bus_watch_name(Gio.BusType.SESSION, CURSOR_BUS_NAME, 0, | ||||
|                                                Lang.bind(this, this._sync), | ||||
|                                                Lang.bind(this, this._sync)); | ||||
|         this._daemonProxy = null; | ||||
|         this._cursorProxy = new CursorManagerProxy(Gio.DBus.session, CURSOR_BUS_NAME, | ||||
|                                                    CURSOR_OBJECT_PATH, | ||||
|                                                    Lang.bind(this, function(proxy, error) { | ||||
|                                                        if (error) { | ||||
|                                                            log(error.message); | ||||
|                                                            return; | ||||
|                                                        } | ||||
|                                                        this._cursorProxy.connect('g-properties-changed', | ||||
|                                                                                  Lang.bind(this, this._sync)); | ||||
|                                                        this._sync(); | ||||
|                                                    })); | ||||
|         this._sync(); | ||||
|  | ||||
|         this._showIdleId = 0; | ||||
|         this._subkeysBoxPointer = null; | ||||
| @@ -183,8 +217,9 @@ const Keyboard = new Lang.Class({ | ||||
|         this._redraw(); | ||||
|     }, | ||||
|  | ||||
|     _settingsChanged: function (settings, key) { | ||||
|         this._enableKeyboard = this._a11yApplicationsSettings.get_boolean(SHOW_KEYBOARD); | ||||
|     _sync: function () { | ||||
|         this._enableKeyboard = this._a11yApplicationsSettings.get_boolean(SHOW_KEYBOARD) || | ||||
|                                this._cursorProxy.ShowOSK; | ||||
|         if (!this._enableKeyboard && !this._keyboard) | ||||
|             return; | ||||
|         if (this._enableKeyboard && this._keyboard && | ||||
| @@ -214,9 +249,35 @@ const Keyboard = new Lang.Class({ | ||||
|         this.actor = null; | ||||
|  | ||||
|         this._destroySource(); | ||||
|         if (this._daemonProxy) { | ||||
|             this._daemonProxy.QuitRemote(function (result, error) { | ||||
|                 if (error) { | ||||
|                     log(error.message); | ||||
|                     return; | ||||
|                 } | ||||
|             }); | ||||
|             this._daemonProxy = null; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _setupKeyboard: function() { | ||||
|         if (!this._daemonProxy) { | ||||
|             this._daemonProxy = new CaribouDaemonProxy(Gio.DBus.session, CARIBOU_BUS_NAME, | ||||
|                                                        CARIBOU_OBJECT_PATH, | ||||
|                                                        Lang.bind(this, function(proxy, error) { | ||||
|                                                            if (error) { | ||||
|                                                                log(error.message); | ||||
|                                                                return; | ||||
|                                                            } | ||||
|                                                        })); | ||||
|         } | ||||
|         this._daemonProxy.RunRemote(function (result, error) { | ||||
|             if (error) { | ||||
|                 log(error.message); | ||||
|                 return; | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         this.actor = new St.BoxLayout({ name: 'keyboard', vertical: true, reactive: true }); | ||||
|         Main.layoutManager.keyboardBox.add_actor(this.actor); | ||||
|         Main.layoutManager.trackChrome(this.actor); | ||||
| @@ -555,7 +616,7 @@ const Keyboard = new Lang.Class({ | ||||
|  | ||||
|     _moveTemporarily: function () { | ||||
|         let currentWindow = global.screen.get_display().focus_window; | ||||
|         let rect = currentWindow.get_outer_rect(); | ||||
|         let rect = currentWindow.get_frame_rect(); | ||||
|  | ||||
|         let newX = rect.x; | ||||
|         let newY = 3 * this.actor.height / 2; | ||||
|   | ||||
							
								
								
									
										174
									
								
								js/ui/layout.js
									
									
									
									
									
								
							
							
						
						| @@ -20,7 +20,6 @@ const Tweener = imports.ui.tweener; | ||||
| const STARTUP_ANIMATION_TIME = 0.5; | ||||
| const KEYBOARD_ANIMATION_TIME = 0.15; | ||||
| const BACKGROUND_FADE_ANIMATION_TIME = 1.0; | ||||
| const DEFAULT_BACKGROUND_COLOR = Clutter.Color.from_pixel(0x2e3436ff); | ||||
|  | ||||
| // The message tray takes this much pressure | ||||
| // in the pressure barrier at once to release it. | ||||
| @@ -160,10 +159,10 @@ const LayoutManager = new Lang.Class({ | ||||
|         this._isPopupWindowVisible = false; | ||||
|         this._startingUp = true; | ||||
|  | ||||
|         // Normally, the stage is always covered so Clutter doesn't need to clear | ||||
|         // it; however it becomes visible during the startup animation | ||||
|         // See the comment below for a longer explanation | ||||
|         global.stage.color = DEFAULT_BACKGROUND_COLOR; | ||||
|         // We don't want to paint the stage background color because either | ||||
|         // the SystemBackground we create or the MetaBackgroundActor inside | ||||
|         // global.window_group covers the entirety of the screen. | ||||
|         global.stage.no_clear_hint = true; | ||||
|  | ||||
|         // Set up stage hierarchy to group all UI actors under one container. | ||||
|         this.uiGroup = new Shell.GenericContainer({ name: 'uiGroup' }); | ||||
| @@ -224,12 +223,16 @@ const LayoutManager = new Lang.Class({ | ||||
|  | ||||
|         // A dummy actor that tracks the mouse or text cursor, based on the | ||||
|         // position and size set in setDummyCursorGeometry. | ||||
|         this.dummyCursor = new St.Widget({ width: 0, height: 0 }); | ||||
|         this.dummyCursor = new St.Widget({ width: 0, height: 0, visible: false }); | ||||
|         this.uiGroup.add_actor(this.dummyCursor); | ||||
|  | ||||
|         global.stage.remove_actor(global.top_window_group); | ||||
|         this.uiGroup.add_actor(global.top_window_group); | ||||
|  | ||||
|         let feedbackGroup = Meta.get_feedback_group_for_screen(global.screen); | ||||
|         global.stage.remove_actor(feedbackGroup); | ||||
|         this.uiGroup.add_actor(feedbackGroup); | ||||
|  | ||||
|         this._backgroundGroup = new Meta.BackgroundGroup(); | ||||
|         global.window_group.add_child(this._backgroundGroup); | ||||
|         this._backgroundGroup.lower_bottom(); | ||||
| @@ -259,7 +262,6 @@ const LayoutManager = new Lang.Class({ | ||||
|  | ||||
|         this._inOverview = true; | ||||
|         this._updateVisibility(); | ||||
|         this._updateRegions(); | ||||
|     }, | ||||
|  | ||||
|     hideOverview: function() { | ||||
| @@ -267,7 +269,6 @@ const LayoutManager = new Lang.Class({ | ||||
|  | ||||
|         this._inOverview = false; | ||||
|         this._updateVisibility(); | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
|  | ||||
|     _sessionUpdated: function() { | ||||
| @@ -361,7 +362,7 @@ const LayoutManager = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _addBackgroundMenu: function(bgManager) { | ||||
|         BackgroundMenu.addBackgroundMenu(bgManager.background.actor, this); | ||||
|         BackgroundMenu.addBackgroundMenu(bgManager.backgroundActor, this); | ||||
|     }, | ||||
|  | ||||
|     _createBackgroundManager: function(monitorIndex) { | ||||
| @@ -378,10 +379,10 @@ const LayoutManager = new Lang.Class({ | ||||
|     _showSecondaryBackgrounds: function() { | ||||
|         for (let i = 0; i < this.monitors.length; i++) { | ||||
|             if (i != this.primaryIndex) { | ||||
|                 let background = this._bgManagers[i].background; | ||||
|                 background.actor.show(); | ||||
|                 background.actor.opacity = 0; | ||||
|                 Tweener.addTween(background.actor, | ||||
|                 let backgroundActor = this._bgManagers[i].backgroundActor; | ||||
|                 backgroundActor.show(); | ||||
|                 backgroundActor.opacity = 0; | ||||
|                 Tweener.addTween(backgroundActor, | ||||
|                                  { opacity: 255, | ||||
|                                    time: BACKGROUND_FADE_ANIMATION_TIME, | ||||
|                                    transition: 'easeOutQuad' }); | ||||
| @@ -404,10 +405,16 @@ const LayoutManager = new Lang.Class({ | ||||
|             this._bgManagers.push(bgManager); | ||||
|  | ||||
|             if (i != this.primaryIndex && this._startingUp) | ||||
|                 bgManager.background.actor.hide(); | ||||
|                 bgManager.backgroundActor.hide(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateKeyboardBox: function() { | ||||
|         this.keyboardBox.set_position(this.keyboardMonitor.x, | ||||
|                                       this.keyboardMonitor.y + this.keyboardMonitor.height); | ||||
|         this.keyboardBox.set_size(this.keyboardMonitor.width, -1); | ||||
|     }, | ||||
|  | ||||
|     _updateBoxes: function() { | ||||
|         this.screenShieldGroup.set_position(0, 0); | ||||
|         this.screenShieldGroup.set_size(global.screen_width, global.screen_height); | ||||
| @@ -415,8 +422,7 @@ const LayoutManager = new Lang.Class({ | ||||
|         this.panelBox.set_position(this.primaryMonitor.x, this.primaryMonitor.y); | ||||
|         this.panelBox.set_size(this.primaryMonitor.width, -1); | ||||
|  | ||||
|         if (this.keyboardIndex < 0) | ||||
|             this.keyboardIndex = this.primaryIndex; | ||||
|         this.keyboardIndex = this.primaryIndex; | ||||
|  | ||||
|         this.trayBox.set_position(this.bottomMonitor.x, | ||||
|                                   this.bottomMonitor.y + this.bottomMonitor.height); | ||||
| @@ -452,8 +458,8 @@ const LayoutManager = new Lang.Class({ | ||||
|     _setupTrayPressure: function() { | ||||
|         this._trayPressure = new PressureBarrier(MESSAGE_TRAY_PRESSURE_THRESHOLD, | ||||
|                                                  MESSAGE_TRAY_PRESSURE_TIMEOUT, | ||||
|                                                  Shell.KeyBindingMode.NORMAL | | ||||
|                                                  Shell.KeyBindingMode.OVERVIEW); | ||||
|                                                  Shell.ActionMode.NORMAL | | ||||
|                                                  Shell.ActionMode.OVERVIEW); | ||||
|         this._trayPressure.setEventFilter(this._trayBarrierEventFilter); | ||||
|         this._trayPressure.connect('trigger', function(barrier) { | ||||
|             if (Main.layoutManager.bottomMonitor.inFullscreen) | ||||
| @@ -541,9 +547,7 @@ const LayoutManager = new Lang.Class({ | ||||
|  | ||||
|     set keyboardIndex(v) { | ||||
|         this._keyboardIndex = v; | ||||
|         this.keyboardBox.set_position(this.keyboardMonitor.x, | ||||
|                                       this.keyboardMonitor.y + this.keyboardMonitor.height); | ||||
|         this.keyboardBox.set_size(this.keyboardMonitor.width, -1); | ||||
|         this._updateKeyboardBox(); | ||||
|     }, | ||||
|  | ||||
|     get keyboardIndex() { | ||||
| @@ -583,10 +587,6 @@ const LayoutManager = new Lang.Class({ | ||||
|     // | ||||
|     // When starting a normal user session, we want to grow it out of the middle | ||||
|     // of the screen. | ||||
|     // | ||||
|     // Usually, we don't want to paint the stage background color because the | ||||
|     // MetaBackgroundActor inside global.window_group covers the entirety of the | ||||
|     // screen. So, we set no_clear_hint at the end of the animation. | ||||
|  | ||||
|     _prepareStartupAnimation: function() { | ||||
|         // During the initial transition, add a simple actor to block all events, | ||||
| @@ -597,7 +597,9 @@ const LayoutManager = new Lang.Class({ | ||||
|                                               reactive: true }); | ||||
|         this.addChrome(this._coverPane); | ||||
|  | ||||
|         if (Main.sessionMode.isGreeter) { | ||||
|         if (Meta.is_restart()) { | ||||
|             // On restart, we don't do an animation | ||||
|         } else if (Main.sessionMode.isGreeter) { | ||||
|             this.panelBox.translation_y = -this.panelBox.height; | ||||
|         } else { | ||||
|             this._updateBackgrounds(); | ||||
| @@ -636,7 +638,9 @@ const LayoutManager = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _startupAnimation: function() { | ||||
|         if (Main.sessionMode.isGreeter) | ||||
|         if (Meta.is_restart()) | ||||
|             this._startupAnimationComplete(); | ||||
|         else if (Main.sessionMode.isGreeter) | ||||
|             this._startupAnimationGreeter(); | ||||
|         else | ||||
|             this._startupAnimationSession(); | ||||
| @@ -663,10 +667,6 @@ const LayoutManager = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _startupAnimationComplete: function() { | ||||
|         // At this point, the UI group is covering everything, so | ||||
|         // we no longer need to clear the stage | ||||
|         global.stage.no_clear_hint = true; | ||||
|  | ||||
|         this._coverPane.destroy(); | ||||
|         this._coverPane = null; | ||||
|  | ||||
| @@ -905,9 +905,6 @@ const LayoutManager = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _queueUpdateRegions: function() { | ||||
|         if (Main.sessionMode.isGreeter) | ||||
|             return; | ||||
|  | ||||
|         if (this._startingUp) | ||||
|             return; | ||||
|  | ||||
| @@ -941,13 +938,16 @@ const LayoutManager = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _updateRegions: function() { | ||||
|         let rects = [], struts = [], i; | ||||
|  | ||||
|         if (this._updateRegionIdle) { | ||||
|             Meta.later_remove(this._updateRegionIdle); | ||||
|             delete this._updateRegionIdle; | ||||
|         } | ||||
|  | ||||
|         // No need to update when we have a modal. | ||||
|         if (Main.modalCount > 0) | ||||
|             return GLib.SOURCE_REMOVE; | ||||
|  | ||||
|         let rects = [], struts = [], i; | ||||
|         let isPopupMenuVisible = global.top_window_group.get_children().some(isPopupMetaWindow); | ||||
|         let wantsInputRegion = !isPopupMenuVisible; | ||||
|  | ||||
| @@ -973,87 +973,43 @@ const LayoutManager = new Lang.Class({ | ||||
|                 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 | ||||
|                 // Metacity wants to know what side of the monitor the | ||||
|                 // strut is considered to be attached to. First, we find | ||||
|                 // the monitor that contains the strut. 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 | ||||
|                 // 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. | ||||
|                 // border of that monitor, 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 monitor = this.findMonitorForActor(actorData.actor); | ||||
|                 let side; | ||||
|                 let primary = this.primaryMonitor; | ||||
|                 if (x1 <= primary.x && x2 >= primary.x + primary.width) { | ||||
|                     if (y1 <= primary.y) | ||||
|                 if (x1 <= monitor.x && x2 >= monitor.x + monitor.width) { | ||||
|                     if (y1 <= monitor.y) | ||||
|                         side = Meta.Side.TOP; | ||||
|                     else if (y2 >= primary.y + primary.height) | ||||
|                     else if (y2 >= monitor.y + monitor.height) | ||||
|                         side = Meta.Side.BOTTOM; | ||||
|                     else | ||||
|                         continue; | ||||
|                 } else if (y1 <= primary.y && y2 >= primary.y + primary.height) { | ||||
|                     if (x1 <= 0) | ||||
|                 } else if (y1 <= monitor.y && y2 >= monitor.y + monitor.height) { | ||||
|                     if (x1 <= monitor.x) | ||||
|                         side = Meta.Side.LEFT; | ||||
|                     else if (x2 >= primary.x + primary.width) | ||||
|                     else if (x2 >= monitor.x + monitor.width) | ||||
|                         side = Meta.Side.RIGHT; | ||||
|                     else | ||||
|                         continue; | ||||
|                 } else if (x1 <= 0) | ||||
|                 } else if (x1 <= monitor.x) | ||||
|                     side = Meta.Side.LEFT; | ||||
|                 else if (y1 <= 0) | ||||
|                 else if (y1 <= monitor.y) | ||||
|                     side = Meta.Side.TOP; | ||||
|                 else if (x2 >= global.screen_width) | ||||
|                 else if (x2 >= monitor.x + monitor.width) | ||||
|                     side = Meta.Side.RIGHT; | ||||
|                 else if (y2 >= global.screen_height) | ||||
|                 else if (y2 >= monitor.y + monitor.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. However skip this step | ||||
|                 // in cases where this would render an entire monitor unusable. | ||||
|                 switch (side) { | ||||
|                 case Meta.Side.TOP: | ||||
|                     let hasMonitorsAbove = this.monitors.some(Lang.bind(this, | ||||
|                         function(mon) { | ||||
|                             return this._isAboveOrBelowPrimary(mon) && | ||||
|                                    mon.y < primary.y; | ||||
|                         })); | ||||
|                     if (!hasMonitorsAbove) | ||||
|                         y1 = 0; | ||||
|                     break; | ||||
|                 case Meta.Side.BOTTOM: | ||||
|                     if (this.primaryIndex == this.bottomIndex) | ||||
|                         y2 = global.screen_height; | ||||
|                     break; | ||||
|                 case Meta.Side.LEFT: | ||||
|                     let hasMonitorsLeft = this.monitors.some(Lang.bind(this, | ||||
|                         function(mon) { | ||||
|                             return !this._isAboveOrBelowPrimary(mon) && | ||||
|                                    mon.x < primary.x; | ||||
|                         })); | ||||
|                     if (!hasMonitorsLeft) | ||||
|                         x1 = 0; | ||||
|                     break; | ||||
|                 case Meta.Side.RIGHT: | ||||
|                     let hasMonitorsRight = this.monitors.some(Lang.bind(this, | ||||
|                         function(mon) { | ||||
|                             return !this._isAboveOrBelowPrimary(mon) && | ||||
|                                    mon.x > primary.x; | ||||
|                         })); | ||||
|                     if (!hasMonitorsRight) | ||||
|                         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 }); | ||||
|                 struts.push(strut); | ||||
| @@ -1070,7 +1026,13 @@ const LayoutManager = new Lang.Class({ | ||||
|         } | ||||
|  | ||||
|         return GLib.SOURCE_REMOVE; | ||||
|     } | ||||
|     }, | ||||
|  | ||||
|     modalEnded: function() { | ||||
|         // We don't update the stage input region while in a modal, | ||||
|         // so queue an update now. | ||||
|         this._queueUpdateRegions(); | ||||
|     }, | ||||
| }); | ||||
| Signals.addSignalMethods(LayoutManager.prototype); | ||||
|  | ||||
| @@ -1098,8 +1060,8 @@ const HotCorner = new Lang.Class({ | ||||
|  | ||||
|         this._pressureBarrier = new PressureBarrier(HOT_CORNER_PRESSURE_THRESHOLD, | ||||
|                                                     HOT_CORNER_PRESSURE_TIMEOUT, | ||||
|                                                     Shell.KeyBindingMode.NORMAL | | ||||
|                                                     Shell.KeyBindingMode.OVERVIEW); | ||||
|                                                     Shell.ActionMode.NORMAL | | ||||
|                                                     Shell.ActionMode.OVERVIEW); | ||||
|         this._pressureBarrier.connect('trigger', Lang.bind(this, this._toggleOverview)); | ||||
|  | ||||
|         // Cache the three ripples instead of dynamically creating and destroying them. | ||||
| @@ -1276,10 +1238,10 @@ const HotCorner = new Lang.Class({ | ||||
| const PressureBarrier = new Lang.Class({ | ||||
|     Name: 'PressureBarrier', | ||||
|  | ||||
|     _init: function(threshold, timeout, keybindingMode) { | ||||
|     _init: function(threshold, timeout, actionMode) { | ||||
|         this._threshold = threshold; | ||||
|         this._timeout = timeout; | ||||
|         this._keybindingMode = keybindingMode; | ||||
|         this._actionMode = actionMode; | ||||
|         this._barriers = []; | ||||
|         this._eventFilter = null; | ||||
|  | ||||
| @@ -1382,7 +1344,7 @@ const PressureBarrier = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         // Throw out all events not in the proper keybinding mode | ||||
|         if (!(this._keybindingMode & Main.keybindingMode)) | ||||
|         if (!(this._actionMode & Main.actionMode)) | ||||
|             return; | ||||
|  | ||||
|         let slide = this._getDistanceAlongBarrier(barrier, event); | ||||
|   | ||||
| @@ -105,8 +105,8 @@ const Lightbox = new Lang.Class({ | ||||
|         this._container = container; | ||||
|         this._children = container.get_children(); | ||||
|         this._fadeFactor = params.fadeFactor; | ||||
|         this._radialEffect = params.radialEffect; | ||||
|         if (params.radialEffect) | ||||
|         this._radialEffect = Clutter.feature_available(Clutter.FeatureFlags.SHADERS_GLSL) && params.radialEffect; | ||||
|         if (this._radialEffect) | ||||
|             this.actor = new RadialShaderQuad({ x: 0, | ||||
|                                                 y: 0, | ||||
|                                                 reactive: params.inhibitEvents }); | ||||
|   | ||||
| @@ -797,7 +797,7 @@ const LookingGlass = new Lang.Class({ | ||||
|                                         reactive: true }); | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this._globalKeyPressEvent)); | ||||
|  | ||||
|         this._interfaceSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' }); | ||||
|         this._interfaceSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.interface' }); | ||||
|         this._interfaceSettings.connect('changed::monospace-font-name', | ||||
|                                         Lang.bind(this, this._updateFont)); | ||||
|         this._updateFont(); | ||||
| @@ -1040,7 +1040,7 @@ const LookingGlass = new Lang.Class({ | ||||
|         let availableHeight = primary.height - Main.layoutManager.keyboardBox.height; | ||||
|         let myHeight = Math.min(primary.height * 0.7, availableHeight * 0.9); | ||||
|         this.actor.x = primary.x + (primary.width - myWidth) / 2; | ||||
|         this._hiddenY = primary.y + Main.layoutManager.panelBox.height - myHeight - 4; // -4 to hide the top corners | ||||
|         this._hiddenY = primary.y + Main.layoutManager.panelBox.height - myHeight; | ||||
|         this._targetY = this._hiddenY + myHeight; | ||||
|         this.actor.y = this._hiddenY; | ||||
|         this.actor.width = myWidth; | ||||
| @@ -1086,7 +1086,7 @@ const LookingGlass = new Lang.Class({ | ||||
|         if (this._open) | ||||
|             return; | ||||
|  | ||||
|         if (!Main.pushModal(this._entry, { keybindingMode: Shell.KeyBindingMode.LOOKING_GLASS })) | ||||
|         if (!Main.pushModal(this._entry, { actionMode: Shell.ActionMode.LOOKING_GLASS })) | ||||
|             return; | ||||
|  | ||||
|         this._notebook.selectIndex(0); | ||||
|   | ||||
| @@ -441,8 +441,8 @@ const Magnifier = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _settingsInit: function(zoomRegion) { | ||||
|         this._appSettings = new Gio.Settings({ schema: APPLICATIONS_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema: MAGNIFIER_SCHEMA }); | ||||
|         this._appSettings = new Gio.Settings({ schema_id: APPLICATIONS_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: MAGNIFIER_SCHEMA }); | ||||
|  | ||||
|         if (zoomRegion) { | ||||
|             // Mag factor is accurate to two decimal places. | ||||
| @@ -1198,12 +1198,7 @@ const ZoomRegion = new Lang.Class({ | ||||
|  | ||||
|         // Add a background for when the magnified uiGroup is scrolled | ||||
|         // out of view (don't want to see desktop showing through). | ||||
|         this._background = new Clutter.Actor({ background_color: Main.DEFAULT_BACKGROUND_COLOR, | ||||
|                                                layout_manager: new Clutter.BinLayout(), | ||||
|                                                width: global.screen_width, | ||||
|                                                height: global.screen_height }); | ||||
|         let noiseTexture = (new Background.SystemBackground()).actor; | ||||
|         this._background.add_actor(noiseTexture); | ||||
|         this._background = (new Background.SystemBackground()).actor; | ||||
|         mainGroup.add_actor(this._background); | ||||
|  | ||||
|         // Clone the group that contains all of UI on the screen.  This is the | ||||
|   | ||||
							
								
								
									
										172
									
								
								js/ui/main.js
									
									
									
									
									
								
							
							
						
						| @@ -4,6 +4,7 @@ const Clutter = imports.gi.Clutter; | ||||
| const Gdk = imports.gi.Gdk; | ||||
| const Gio = imports.gi.Gio; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Lang = imports.lang; | ||||
| const Mainloop = imports.mainloop; | ||||
| const Meta = imports.gi.Meta; | ||||
| @@ -18,7 +19,9 @@ const ExtensionSystem = imports.ui.extensionSystem; | ||||
| const ExtensionDownloader = imports.ui.extensionDownloader; | ||||
| const Keyboard = imports.ui.keyboard; | ||||
| const MessageTray = imports.ui.messageTray; | ||||
| const ModalDialog = imports.ui.modalDialog; | ||||
| const OsdWindow = imports.ui.osdWindow; | ||||
| const OsdMonitorLabeler = imports.ui.osdMonitorLabeler; | ||||
| const Overview = imports.ui.overview; | ||||
| const Panel = imports.ui.panel; | ||||
| const Params = imports.misc.params; | ||||
| @@ -39,8 +42,6 @@ const Magnifier = imports.ui.magnifier; | ||||
| const XdndHandler = imports.ui.xdndHandler; | ||||
| const Util = imports.misc.util; | ||||
|  | ||||
| const DEFAULT_BACKGROUND_COLOR = Clutter.Color.from_pixel(0x2e3436ff); | ||||
|  | ||||
| const A11Y_SCHEMA = 'org.gnome.desktop.a11y.keyboard'; | ||||
| const STICKY_KEYS_ENABLE = 'stickykeys-enable'; | ||||
| const GNOMESHELL_STARTED_MESSAGE_ID = 'f3ea493c22934e26811cd62abe8e203a'; | ||||
| @@ -57,13 +58,14 @@ let notificationDaemon = null; | ||||
| let windowAttentionHandler = null; | ||||
| let ctrlAltTabManager = null; | ||||
| let osdWindowManager = null; | ||||
| let osdMonitorLabeler = null; | ||||
| let sessionMode = null; | ||||
| let shellDBusService = null; | ||||
| let shellMountOpDBusService = null; | ||||
| let screenSaverDBus = null; | ||||
| let screencastService = null; | ||||
| let modalCount = 0; | ||||
| let keybindingMode = Shell.KeyBindingMode.NONE; | ||||
| let actionMode = Shell.ActionMode.NONE; | ||||
| let modalActorFocusStack = []; | ||||
| let uiGroup = null; | ||||
| let magnifier = null; | ||||
| @@ -74,21 +76,21 @@ let _startDate; | ||||
| let _defaultCssStylesheet = null; | ||||
| let _cssStylesheet = null; | ||||
| let _a11ySettings = null; | ||||
| let dynamicWorkspacesSchema = null; | ||||
|  | ||||
| function _sessionUpdated() { | ||||
|     _loadDefaultStylesheet(); | ||||
|     if (sessionMode.isPrimary) | ||||
|         _loadDefaultStylesheet(); | ||||
|  | ||||
|     wm.setCustomKeybindingHandler('panel-main-menu', | ||||
|                                   Shell.KeyBindingMode.NORMAL | | ||||
|                                   Shell.KeyBindingMode.OVERVIEW, | ||||
|                                   Shell.ActionMode.NORMAL | | ||||
|                                   Shell.ActionMode.OVERVIEW, | ||||
|                                   sessionMode.hasOverview ? Lang.bind(overview, overview.toggle) : null); | ||||
|     wm.allowKeybinding('overlay-key', Shell.KeyBindingMode.NORMAL | | ||||
|                                       Shell.KeyBindingMode.OVERVIEW); | ||||
|     wm.allowKeybinding('overlay-key', Shell.ActionMode.NORMAL | | ||||
|                                       Shell.ActionMode.OVERVIEW); | ||||
|  | ||||
|     wm.setCustomKeybindingHandler('panel-run-dialog', | ||||
|                                   Shell.KeyBindingMode.NORMAL | | ||||
|                                   Shell.KeyBindingMode.OVERVIEW, | ||||
|                                   Shell.ActionMode.NORMAL | | ||||
|                                   Shell.ActionMode.OVERVIEW, | ||||
|                                   sessionMode.hasRunDialog ? openRunDialog : null); | ||||
|  | ||||
|     if (!sessionMode.hasRunDialog) { | ||||
| @@ -111,7 +113,8 @@ function start() { | ||||
|  | ||||
|     sessionMode = new SessionMode.SessionMode(); | ||||
|     sessionMode.connect('updated', _sessionUpdated); | ||||
|     _initializePrefs(); | ||||
|     Gtk.Settings.get_default().connect('notify::gtk-theme-name', | ||||
|                                        _loadDefaultStylesheet); | ||||
|     _initializeUI(); | ||||
|  | ||||
|     shellDBusService = new ShellDBus.GnomeShell(); | ||||
| @@ -120,17 +123,6 @@ function start() { | ||||
|     _sessionUpdated(); | ||||
| } | ||||
|  | ||||
| function _initializePrefs() { | ||||
|     let keys = new Gio.Settings({ schema: sessionMode.overridesSchema }).list_keys(); | ||||
|     for (let i = 0; i < keys.length; i++) | ||||
|         Meta.prefs_override_preference_schema(keys[i], sessionMode.overridesSchema); | ||||
|  | ||||
|     if (keys.indexOf('dynamic-workspaces') > -1) | ||||
|         dynamicWorkspacesSchema = sessionMode.overridesSchema; | ||||
|     else | ||||
|         dynamicWorkspacesSchema = 'org.gnome.mutter'; | ||||
| } | ||||
|  | ||||
| function _initializeUI() { | ||||
|     // Ensure ShellWindowTracker and ShellAppUsage are initialized; this will | ||||
|     // also initialize ShellAppSystem first.  ShellAppSystem | ||||
| @@ -143,6 +135,9 @@ function _initializeUI() { | ||||
|     Shell.WindowTracker.get_default(); | ||||
|     Shell.AppUsage.get_default(); | ||||
|  | ||||
|     let resource = Gio.Resource.load(global.datadir + '/gnome-shell-theme.gresource'); | ||||
|     resource._register(); | ||||
|  | ||||
|     _loadDefaultStylesheet(); | ||||
|  | ||||
|     // Setup the stage hierarchy early | ||||
| @@ -157,6 +152,7 @@ function _initializeUI() { | ||||
|     xdndHandler = new XdndHandler.XdndHandler(); | ||||
|     ctrlAltTabManager = new CtrlAltTab.CtrlAltTabManager(); | ||||
|     osdWindowManager = new OsdWindow.OsdWindowManager(); | ||||
|     osdMonitorLabeler = new OsdMonitorLabeler.OsdMonitorLabeler(); | ||||
|     overview = new Overview.Overview(); | ||||
|     wm = new WindowManager.WindowManager(); | ||||
|     magnifier = new Magnifier.Magnifier(); | ||||
| @@ -173,13 +169,23 @@ function _initializeUI() { | ||||
|     layoutManager.init(); | ||||
|     overview.init(); | ||||
|  | ||||
|     _a11ySettings = new Gio.Settings({ schema: A11Y_SCHEMA }); | ||||
|     _a11ySettings = new Gio.Settings({ schema_id: A11Y_SCHEMA }); | ||||
|  | ||||
|     global.display.connect('overlay-key', Lang.bind(overview, function () { | ||||
|         if (!_a11ySettings.get_boolean (STICKY_KEYS_ENABLE)) | ||||
|             overview.toggle(); | ||||
|     })); | ||||
|  | ||||
|     global.display.connect('show-restart-message', function(display, message) { | ||||
|         showRestartMessage(message); | ||||
|         return true; | ||||
|     }); | ||||
|  | ||||
|     global.display.connect('restart', function() { | ||||
|         global.reexec_self(); | ||||
|         return true; | ||||
|     }); | ||||
|  | ||||
|     // Provide the bus object for gnome-session to | ||||
|     // initiate logouts. | ||||
|     EndSessionDialog.init(); | ||||
| @@ -206,32 +212,52 @@ function _initializeUI() { | ||||
|     } | ||||
|  | ||||
|     layoutManager.connect('startup-complete', function() { | ||||
|                               if (keybindingMode == Shell.KeyBindingMode.NONE) { | ||||
|                                   keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
|                               } | ||||
|                               if (screenShield) { | ||||
|                                   screenShield.lockIfWasLocked(); | ||||
|                               } | ||||
|                               if (LoginManager.haveSystemd() && | ||||
|                                   sessionMode.currentMode != 'gdm' && | ||||
|                                   sessionMode.currentMode != 'initial-setup') { | ||||
|                                   // Do not import globally to not depend | ||||
|                                   // on systemd on non-systemd systems. | ||||
|                                   let GSystem = imports.gi.GSystem; | ||||
|                                   GSystem.log_structured_print('GNOME Shell started at ' + _startDate, | ||||
|                                                                ['MESSAGE_ID=' + GNOMESHELL_STARTED_MESSAGE_ID]); | ||||
|                               } else { | ||||
|                                   log('GNOME Shell started at ' + _startDate); | ||||
|                               } | ||||
|                           }); | ||||
|         if (actionMode == Shell.ActionMode.NONE) { | ||||
|             actionMode = Shell.ActionMode.NORMAL; | ||||
|         } | ||||
|         if (screenShield) { | ||||
|             screenShield.lockIfWasLocked(); | ||||
|         } | ||||
|         if (sessionMode.currentMode != 'gdm' && | ||||
|             sessionMode.currentMode != 'initial-setup') { | ||||
|             Shell.Global.log_structured('GNOME Shell started at ' + _startDate, | ||||
|                                         ['MESSAGE_ID=' + GNOMESHELL_STARTED_MESSAGE_ID]); | ||||
|         } | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function _getStylesheet(name) { | ||||
|     let stylesheet; | ||||
|  | ||||
|     stylesheet = Gio.File.new_for_uri('resource:///org/gnome/shell/theme/' + name); | ||||
|     if (stylesheet.query_exists(null)) | ||||
|         return stylesheet; | ||||
|  | ||||
|     stylesheet = Gio.File.new_for_path(global.datadir + '/theme/' + name); | ||||
|     if (stylesheet.query_exists(null)) | ||||
|         return stylesheet; | ||||
|  | ||||
|     return null; | ||||
| } | ||||
|  | ||||
| function _getDefaultStylesheet() { | ||||
|     let stylesheet = null; | ||||
|     let name = sessionMode.stylesheetName; | ||||
|  | ||||
|     // Look for a high-contrast variant first when using GTK+'s HighContrast | ||||
|     // theme | ||||
|     if (Gtk.Settings.get_default().gtk_theme_name == 'HighContrast') | ||||
|         stylesheet = _getStylesheet(name.replace('.css', '-high-contrast.css')); | ||||
|  | ||||
|     if (stylesheet == null) | ||||
|         stylesheet = _getStylesheet(sessionMode.stylesheetName); | ||||
|  | ||||
|     return stylesheet; | ||||
| } | ||||
|  | ||||
| function _loadDefaultStylesheet() { | ||||
|     if (!sessionMode.isPrimary) | ||||
|         return; | ||||
|  | ||||
|     let stylesheet = global.datadir + '/theme/' + sessionMode.stylesheetName; | ||||
|     if (_defaultCssStylesheet == stylesheet) | ||||
|     let stylesheet = _getDefaultStylesheet(); | ||||
|     if (_defaultCssStylesheet && _defaultCssStylesheet.equal(stylesheet)) | ||||
|         return; | ||||
|  | ||||
|     _defaultCssStylesheet = stylesheet; | ||||
| @@ -243,11 +269,10 @@ function _loadDefaultStylesheet() { | ||||
|  * | ||||
|  * Get the theme CSS file that the shell will load | ||||
|  * | ||||
|  * Returns: A file path that contains the theme CSS, | ||||
|  * Returns: A #GFile that contains the theme CSS, | ||||
|  *          null if using the default | ||||
|  */ | ||||
| function getThemeStylesheet() | ||||
| { | ||||
| function getThemeStylesheet() { | ||||
|     return _cssStylesheet; | ||||
| } | ||||
|  | ||||
| @@ -258,9 +283,8 @@ function getThemeStylesheet() | ||||
|  * | ||||
|  * Set the theme CSS file that the shell will load | ||||
|  */ | ||||
| function setThemeStylesheet(cssStylesheet) | ||||
| { | ||||
|     _cssStylesheet = cssStylesheet; | ||||
| function setThemeStylesheet(cssStylesheet) { | ||||
|     _cssStylesheet = cssStylesheet ? Gio.File.new_for_path(cssStylesheet) : null; | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -345,7 +369,7 @@ function _findModal(actor) { | ||||
|  *  - options: Meta.ModalOptions flags to indicate that the pointer is | ||||
|  *             already grabbed | ||||
|  * | ||||
|  *  - keybindingMode: used to set the current Shell.KeyBindingMode to filter | ||||
|  *  - actionMode: used to set the current Shell.ActionMode to filter | ||||
|  *                    global keybindings; the default of NONE will filter | ||||
|  *                    out all keybindings | ||||
|  * | ||||
| @@ -354,7 +378,7 @@ function _findModal(actor) { | ||||
| function pushModal(actor, params) { | ||||
|     params = Params.parse(params, { timestamp: global.get_current_time(), | ||||
|                                     options: 0, | ||||
|                                     keybindingMode: Shell.KeyBindingMode.NONE }); | ||||
|                                     actionMode: Shell.ActionMode.NONE }); | ||||
|  | ||||
|     if (modalCount == 0) { | ||||
|         if (!global.begin_modal(params.timestamp, params.options)) { | ||||
| @@ -384,9 +408,9 @@ function pushModal(actor, params) { | ||||
|                                 destroyId: actorDestroyId, | ||||
|                                 prevFocus: prevFocus, | ||||
|                                 prevFocusDestroyId: prevFocusDestroyId, | ||||
|                                 keybindingMode: keybindingMode }); | ||||
|                                 actionMode: actionMode }); | ||||
|  | ||||
|     keybindingMode = params.keybindingMode; | ||||
|     actionMode = params.actionMode; | ||||
|     global.stage.set_key_focus(actor); | ||||
|     return true; | ||||
| } | ||||
| @@ -412,7 +436,7 @@ function popModal(actor, timestamp) { | ||||
|     if (focusIndex < 0) { | ||||
|         global.stage.set_key_focus(null); | ||||
|         global.end_modal(timestamp); | ||||
|         keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
|         actionMode = Shell.ActionMode.NORMAL; | ||||
|  | ||||
|         throw new Error('incorrect pop'); | ||||
|     } | ||||
| @@ -425,7 +449,7 @@ function popModal(actor, timestamp) { | ||||
|     if (focusIndex == modalActorFocusStack.length - 1) { | ||||
|         if (record.prevFocus) | ||||
|             record.prevFocus.disconnect(record.prevFocusDestroyId); | ||||
|         keybindingMode = record.keybindingMode; | ||||
|         actionMode = record.actionMode; | ||||
|         global.stage.set_key_focus(record.prevFocus); | ||||
|     } else { | ||||
|         // If we have: | ||||
| @@ -450,7 +474,7 @@ function popModal(actor, timestamp) { | ||||
|         for (let i = modalActorFocusStack.length - 1; i > focusIndex; i--) { | ||||
|             modalActorFocusStack[i].prevFocus = modalActorFocusStack[i - 1].prevFocus; | ||||
|             modalActorFocusStack[i].prevFocusDestroyId = modalActorFocusStack[i - 1].prevFocusDestroyId; | ||||
|             modalActorFocusStack[i].keybindingMode = modalActorFocusStack[i - 1].keybindingMode; | ||||
|             modalActorFocusStack[i].actionMode = modalActorFocusStack[i - 1].actionMode; | ||||
|         } | ||||
|     } | ||||
|     modalActorFocusStack.splice(focusIndex, 1); | ||||
| @@ -458,9 +482,10 @@ function popModal(actor, timestamp) { | ||||
|     if (modalCount > 0) | ||||
|         return; | ||||
|  | ||||
|     layoutManager.modalEnded(); | ||||
|     global.end_modal(timestamp); | ||||
|     Meta.enable_unredirect_for_screen(global.screen); | ||||
|     keybindingMode = Shell.KeyBindingMode.NORMAL; | ||||
|     actionMode = Shell.ActionMode.NORMAL; | ||||
| } | ||||
|  | ||||
| function createLookingGlass() { | ||||
| @@ -621,3 +646,28 @@ function queueDeferredWork(workId) { | ||||
|         GLib.Source.set_name_by_id(_deferredTimeoutId, '[gnome-shell] _runAllDeferredWork'); | ||||
|     } | ||||
| } | ||||
|  | ||||
| const RestartMessage = new Lang.Class({ | ||||
|     Name: 'RestartMessage', | ||||
|     Extends: ModalDialog.ModalDialog, | ||||
|  | ||||
|     _init : function(message) { | ||||
|         this.parent({ shellReactive: true, | ||||
|                       styleClass: 'restart-message headline', | ||||
|                       shouldFadeIn: false, | ||||
|                       destroyOnClose: true }); | ||||
|  | ||||
|         let label = new St.Label({ text: message }); | ||||
|  | ||||
|         this.contentLayout.add(label, { x_fill: false, | ||||
|                                         y_fill: false, | ||||
|                                         x_align: St.Align.MIDDLE, | ||||
|                                         y_align: St.Align.MIDDLE }); | ||||
|         this.buttonLayout.hide(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| function showRestartMessage(message) { | ||||
|     let restartMessage = new RestartMessage(message); | ||||
|     restartMessage.open(); | ||||
| } | ||||
|   | ||||
| @@ -15,6 +15,7 @@ const Signals = imports.signals; | ||||
| const St = imports.gi.St; | ||||
| const Tp = imports.gi.TelepathyGLib; | ||||
|  | ||||
| const EdgeDragAction = imports.ui.edgeDragAction; | ||||
| const BoxPointer = imports.ui.boxpointer; | ||||
| const CtrlAltTab = imports.ui.ctrlAltTab; | ||||
| const GnomeSession = imports.misc.gnomeSession; | ||||
| @@ -26,6 +27,7 @@ const PopupMenu = imports.ui.popupMenu; | ||||
| const Params = imports.misc.params; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const Util = imports.misc.util; | ||||
| const ViewSelector = imports.ui.viewSelector; | ||||
|  | ||||
| const SHELL_KEYBINDINGS_SCHEMA = 'org.gnome.shell.keybindings'; | ||||
|  | ||||
| @@ -111,7 +113,6 @@ const FocusGrabber = new Lang.Class({ | ||||
|         if (this._focused) | ||||
|             return; | ||||
|  | ||||
|         this._prevFocusedWindow = global.display.focus_window; | ||||
|         this._prevKeyFocusActor = global.stage.get_key_focus(); | ||||
|  | ||||
|         this._focusActorChangedId = global.stage.connect('notify::key-focus', Lang.bind(this, this._focusActorChanged)); | ||||
| @@ -316,7 +317,7 @@ const NotificationGenericPolicy = new Lang.Class({ | ||||
|  | ||||
|         this.id = 'generic'; | ||||
|  | ||||
|         this._masterSettings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications' }); | ||||
|         this._masterSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.notifications' }); | ||||
|         this._masterSettings.connect('changed', Lang.bind(this, this._changed)); | ||||
|     }, | ||||
|  | ||||
| @@ -366,8 +367,8 @@ const NotificationApplicationPolicy = new Lang.Class({ | ||||
|         this.id = id; | ||||
|         this._canonicalId = this._canonicalizeId(id); | ||||
|  | ||||
|         this._masterSettings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications' }); | ||||
|         this._settings = new Gio.Settings({ schema: 'org.gnome.desktop.notifications.application', | ||||
|         this._masterSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.notifications' }); | ||||
|         this._settings = new Gio.Settings({ schema_id: 'org.gnome.desktop.notifications.application', | ||||
|                                             path: '/org/gnome/desktop/notifications/application/' + this._canonicalId + '/' }); | ||||
|  | ||||
|         this._masterSettings.connect('changed', Lang.bind(this, this._changed)); | ||||
| @@ -493,7 +494,6 @@ const Notification = new Lang.Class({ | ||||
|         this.resident = false; | ||||
|         // 'transient' is a reserved keyword in JS, so we have to use an alternate variable name | ||||
|         this.isTransient = false; | ||||
|         this.isMusic = false; | ||||
|         this.forFeedback = false; | ||||
|         this.expanded = false; | ||||
|         this.focused = false; | ||||
| @@ -504,10 +504,8 @@ const Notification = new Lang.Class({ | ||||
|         this.bannerBodyMarkup = false; | ||||
|         this._bannerBodyAdded = false; | ||||
|         this._titleFitsInBannerMode = true; | ||||
|         this._titleDirection = Clutter.TextDirection.DEFAULT; | ||||
|         this._spacing = 0; | ||||
|         this._scrollPolicy = Gtk.PolicyType.AUTOMATIC; | ||||
|         this._imageBin = null; | ||||
|         this._soundName = null; | ||||
|         this._soundFile = null; | ||||
|         this._soundPlayed = false; | ||||
| @@ -611,10 +609,7 @@ const Notification = new Lang.Class({ | ||||
|             this._actionArea = null; | ||||
|             this._buttonBox = null; | ||||
|         } | ||||
|         if (params.clear) | ||||
|             this.unsetImage(); | ||||
|  | ||||
|         if (!this._scrollArea && !this._actionArea && !this._imageBin) | ||||
|         if (!this._scrollArea && !this._actionArea) | ||||
|             this._table.remove_style_class_name('multi-line-notification'); | ||||
|  | ||||
|         if (params.gicon) { | ||||
| @@ -643,10 +638,11 @@ const Notification = new Lang.Class({ | ||||
|         title = title ? _fixMarkup(title.replace(/\n/g, ' '), false) : ''; | ||||
|         this._titleLabel.clutter_text.set_markup('<b>' + title + '</b>'); | ||||
|  | ||||
|         let titleDirection; | ||||
|         if (Pango.find_base_dir(title, -1) == Pango.Direction.RTL) | ||||
|             this._titleDirection = Clutter.TextDirection.RTL; | ||||
|             titleDirection = Clutter.TextDirection.RTL; | ||||
|         else | ||||
|             this._titleDirection = Clutter.TextDirection.LTR; | ||||
|             titleDirection = Clutter.TextDirection.LTR; | ||||
|  | ||||
|         // Let the title's text direction control the overall direction | ||||
|         // of the notification - in case where different scripts are used | ||||
| @@ -654,7 +650,7 @@ const Notification = new Lang.Class({ | ||||
|         // arguably for action buttons as well. Labels other than the title | ||||
|         // will be allocated at the available width, so that their alignment | ||||
|         // is done correctly automatically. | ||||
|         this._table.set_text_direction(this._titleDirection); | ||||
|         this._table.set_text_direction(titleDirection); | ||||
|  | ||||
|         // Unless the notification has custom content, we save this.bannerBodyText | ||||
|         // to add it to the content of the notification if the notification is | ||||
| @@ -790,45 +786,11 @@ const Notification = new Lang.Class({ | ||||
|  | ||||
|     _updateLastColumnSettings: function() { | ||||
|         if (this._scrollArea) | ||||
|             this._table.child_set(this._scrollArea, { col: this._imageBin ? 2 : 1, | ||||
|                                                       col_span: this._imageBin ? 1 : 2 }); | ||||
|             this._table.child_set(this._scrollArea, { col: 1, | ||||
|                                                       col_span: 2 }); | ||||
|         if (this._actionArea) | ||||
|             this._table.child_set(this._actionArea, { col: this._imageBin ? 2 : 1, | ||||
|                                                       col_span: this._imageBin ? 1 : 2 }); | ||||
|     }, | ||||
|  | ||||
|     setImage: function(image) { | ||||
|         this.unsetImage(); | ||||
|  | ||||
|         if (!image) | ||||
|             return; | ||||
|  | ||||
|         this._imageBin = new St.Bin({ opacity: 230, | ||||
|                                       child: image, | ||||
|                                       visible: this.expanded }); | ||||
|  | ||||
|         this._table.add_style_class_name('multi-line-notification'); | ||||
|         this._table.add_style_class_name('notification-with-image'); | ||||
|         this._addBannerBody(); | ||||
|         this._updateLastColumnSettings(); | ||||
|         this._table.add(this._imageBin, { row: 1, | ||||
|                                           col: 1, | ||||
|                                           row_span: 2, | ||||
|                                           x_expand: false, | ||||
|                                           y_expand: false, | ||||
|                                           x_fill: false, | ||||
|                                           y_fill: false }); | ||||
|     }, | ||||
|  | ||||
|     unsetImage: function() { | ||||
|         if (this._imageBin) { | ||||
|             this._table.remove_style_class_name('notification-with-image'); | ||||
|             this._table.remove_actor(this._imageBin); | ||||
|             this._imageBin = null; | ||||
|             this._updateLastColumnSettings(); | ||||
|             if (!this._scrollArea && !this._actionArea) | ||||
|                 this._table.remove_style_class_name('multi-line-notification'); | ||||
|         } | ||||
|             this._table.child_set(this._actionArea, { col: 1, | ||||
|                                                       col_span: 2 }); | ||||
|     }, | ||||
|  | ||||
|     addButton: function(button, callback) { | ||||
| @@ -923,7 +885,7 @@ const Notification = new Lang.Class({ | ||||
|         let [titleMinH, titleNatH] = this._titleLabel.get_preferred_height(availWidth); | ||||
|         let [bannerMinW, bannerNatW] = this._bannerLabel.get_preferred_width(availWidth); | ||||
|  | ||||
|         let rtl = (this._titleDirection == Clutter.TextDirection.RTL); | ||||
|         let rtl = (this._table.text_direction == Clutter.TextDirection.RTL); | ||||
|         let x = rtl ? availWidth : 0; | ||||
|  | ||||
|         if (this._secondaryIcon) { | ||||
| @@ -1060,8 +1022,6 @@ const Notification = new Lang.Class({ | ||||
|         this.actor.remove_style_class_name('notification-unexpanded'); | ||||
|  | ||||
|         // Show additional content that we keep hidden in banner mode | ||||
|         if (this._imageBin) | ||||
|             this._imageBin.show(); | ||||
|         if (this._actionArea) | ||||
|             this._actionArea.show(); | ||||
|         if (this._scrollArea) | ||||
| @@ -1100,8 +1060,6 @@ const Notification = new Lang.Class({ | ||||
|         this.expanded = false; | ||||
|  | ||||
|         // Hide additional content that we keep hidden in banner mode | ||||
|         if (this._imageBin) | ||||
|             this._imageBin.hide(); | ||||
|         if (this._actionArea) | ||||
|             this._actionArea.hide(); | ||||
|         if (this._scrollArea) | ||||
| @@ -1158,6 +1116,7 @@ const SourceActor = new Lang.Class({ | ||||
|         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, function() { | ||||
|             this._source.disconnect(this._iconUpdatedId); | ||||
|             this._actorDestroyed = true; | ||||
|         })); | ||||
|         this._actorDestroyed = false; | ||||
| @@ -1169,7 +1128,7 @@ const SourceActor = new Lang.Class({ | ||||
|  | ||||
|         this.actor.add_actor(this._iconBin); | ||||
|  | ||||
|         this._source.connect('icon-updated', Lang.bind(this, this._updateIcon)); | ||||
|         this._iconUpdatedId = this._source.connect('icon-updated', Lang.bind(this, this._updateIcon)); | ||||
|         this._updateIcon(); | ||||
|     }, | ||||
|  | ||||
| @@ -1227,8 +1186,12 @@ const SourceActorWithLabel = new Lang.Class({ | ||||
|  | ||||
|         this.actor.add_actor(this._counterBin); | ||||
|  | ||||
|         this._source.connect('count-updated', Lang.bind(this, this._updateCount)); | ||||
|         this._countUpdatedId = this._source.connect('count-updated', Lang.bind(this, this._updateCount)); | ||||
|         this._updateCount(); | ||||
|  | ||||
|         this.actor.connect('destroy', function() { | ||||
|             this._source.disconnect(this._countUpdatedId); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
| @@ -1307,7 +1270,7 @@ const Source = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     get isClearable() { | ||||
|         return !this.trayIcon && !this.isChat && !this.resident; | ||||
|         return !this.isChat && !this.resident; | ||||
|     }, | ||||
|  | ||||
|     countUpdated: function() { | ||||
| @@ -1462,16 +1425,7 @@ const Source = new Lang.Class({ | ||||
|     // Default implementation is to destroy this source, but subclasses can override | ||||
|     _lastNotificationRemoved: function() { | ||||
|         this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     getMusicNotification: function() { | ||||
|         for (let i = 0; i < this.notifications.length; i++) { | ||||
|             if (this.notifications[i].isMusic) | ||||
|                 return this.notifications[i]; | ||||
|         } | ||||
|  | ||||
|         return null; | ||||
|     }, | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Source.prototype); | ||||
|  | ||||
| @@ -1811,6 +1765,13 @@ const MessageTray = new Lang.Class({ | ||||
|                                            y_expand: true }); | ||||
|         this.actor.add_actor(this._summary); | ||||
|  | ||||
|         this._focusTrap = new ViewSelector.FocusTrap({ can_focus: true }); | ||||
|         this._focusTrap.connect('key-focus-in', Lang.bind(this, | ||||
|             function() { | ||||
|                 this._messageTrayMenuButton.actor.grab_key_focus(); | ||||
|             })); | ||||
|         this._summary.add_actor(this._focusTrap); | ||||
|  | ||||
|         this._summaryMotionId = 0; | ||||
|  | ||||
|         this._summaryBoxPointer = new BoxPointer.BoxPointer(St.Side.BOTTOM, | ||||
| @@ -1839,7 +1800,7 @@ const MessageTray = new Lang.Class({ | ||||
|         this.idleMonitor = Meta.IdleMonitor.get_core(); | ||||
|  | ||||
|         this._grabHelper = new GrabHelper.GrabHelper(this.actor, | ||||
|                                                      { keybindingMode: Shell.KeyBindingMode.MESSAGE_TRAY }); | ||||
|                                                      { actionMode: Shell.ActionMode.MESSAGE_TRAY }); | ||||
|         this._grabHelper.addActor(this._summaryBoxPointer.actor); | ||||
|         this._grabHelper.addActor(this.actor); | ||||
|  | ||||
| @@ -1901,18 +1862,18 @@ const MessageTray = new Lang.Class({ | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|  | ||||
|         Main.wm.addKeybinding('toggle-message-tray', | ||||
|                               new Gio.Settings({ schema: SHELL_KEYBINDINGS_SCHEMA }), | ||||
|                               new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }), | ||||
|                               Meta.KeyBindingFlags.NONE, | ||||
|                               Shell.KeyBindingMode.NORMAL | | ||||
|                               Shell.KeyBindingMode.MESSAGE_TRAY | | ||||
|                               Shell.KeyBindingMode.OVERVIEW, | ||||
|                               Shell.ActionMode.NORMAL | | ||||
|                               Shell.ActionMode.MESSAGE_TRAY | | ||||
|                               Shell.ActionMode.OVERVIEW, | ||||
|                               Lang.bind(this, this.toggleAndNavigate)); | ||||
|         Main.wm.addKeybinding('focus-active-notification', | ||||
|                               new Gio.Settings({ schema: SHELL_KEYBINDINGS_SCHEMA }), | ||||
|                               new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }), | ||||
|                               Meta.KeyBindingFlags.NONE, | ||||
|                               Shell.KeyBindingMode.NORMAL | | ||||
|                               Shell.KeyBindingMode.MESSAGE_TRAY | | ||||
|                               Shell.KeyBindingMode.OVERVIEW, | ||||
|                               Shell.ActionMode.NORMAL | | ||||
|                               Shell.ActionMode.MESSAGE_TRAY | | ||||
|                               Shell.ActionMode.OVERVIEW, | ||||
|                               Lang.bind(this, this._expandActiveNotification)); | ||||
|  | ||||
|         this._sources = new Map(); | ||||
| @@ -1934,6 +1895,15 @@ const MessageTray = new Lang.Class({ | ||||
|  | ||||
|         this._messageTrayMenuButton = new MessageTrayMenuButton(this); | ||||
|         this.actor.add_actor(this._messageTrayMenuButton.actor); | ||||
|  | ||||
|         this._messageTrayMenuButton.actor.connect('key-press-event', | ||||
|                                                   Lang.bind(this, this._onTrayButtonKeyPress)); | ||||
|  | ||||
|         let gesture = new EdgeDragAction.EdgeDragAction(St.Side.BOTTOM, | ||||
|                                                         Shell.ActionMode.NORMAL | | ||||
|                                                         Shell.ActionMode.OVERVIEW); | ||||
|         gesture.connect('activated', Lang.bind(this, this.toggle)); | ||||
|         global.stage.add_action(gesture); | ||||
|     }, | ||||
|  | ||||
|     close: function() { | ||||
| @@ -2026,6 +1996,16 @@ const MessageTray = new Lang.Class({ | ||||
|         return GLib.SOURCE_REMOVE; | ||||
|     }, | ||||
|  | ||||
|     _onTrayButtonKeyPress: function(actor, event) { | ||||
|         if (event.get_key_symbol() == Clutter.ISO_Left_Tab) { | ||||
|             this._focusTrap.can_focus = false; | ||||
|             this._summary.navigate_focus(null, Gtk.DirectionType.TAB_BACKWARD, false); | ||||
|             this._focusTrap.can_focus = true; | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onNotificationKeyRelease: function(actor, event) { | ||||
|         if (event.get_key_symbol() == Clutter.KEY_Escape && event.get_state() == 0) { | ||||
|             this._expireNotification(); | ||||
| @@ -2268,6 +2248,16 @@ const MessageTray = new Lang.Class({ | ||||
|             this._grabHelper.addActor(corner.actor); | ||||
|     }, | ||||
|  | ||||
|     _resetNotificationLeftTimeout: function() { | ||||
|         this._useLongerNotificationLeftTimeout = false; | ||||
|         if (this._notificationLeftTimeoutId) { | ||||
|             Mainloop.source_remove(this._notificationLeftTimeoutId); | ||||
|             this._notificationLeftTimeoutId = 0; | ||||
|             this._notificationLeftMouseX = -1; | ||||
|             this._notificationLeftMouseY = -1; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onNotificationHoverChanged: function() { | ||||
|         if (this._notificationWidget.hover == this._notificationHovered) | ||||
|             return; | ||||
| @@ -2277,14 +2267,7 @@ const MessageTray = new Lang.Class({ | ||||
|             // No dwell inside notifications at the bottom of the screen | ||||
|             this._cancelTrayDwell(); | ||||
|  | ||||
|             this._useLongerNotificationLeftTimeout = false; | ||||
|             if (this._notificationLeftTimeoutId) { | ||||
|                 Mainloop.source_remove(this._notificationLeftTimeoutId); | ||||
|                 this._notificationLeftTimeoutId = 0; | ||||
|                 this._notificationLeftMouseX = -1; | ||||
|                 this._notificationLeftMouseY = -1; | ||||
|                 return; | ||||
|             } | ||||
|             this._resetNotificationLeftTimeout(); | ||||
|  | ||||
|             if (this._showNotificationMouseX >= 0) { | ||||
|                 let actorAtShowNotificationPosition = | ||||
| @@ -2301,6 +2284,7 @@ const MessageTray = new Lang.Class({ | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             this._pointerInNotification = true; | ||||
|             this._updateState(); | ||||
|         } else { | ||||
| @@ -2398,8 +2382,22 @@ const MessageTray = new Lang.Class({ | ||||
|             if (shouldShowNotification && nextNotification) { | ||||
|                 let limited = this._busy || Main.layoutManager.bottomMonitor.inFullscreen; | ||||
|                 let showNextNotification = (!limited || nextNotification.forFeedback || nextNotification.urgency == Urgency.CRITICAL); | ||||
|                 if (showNextNotification) | ||||
|                     this._showNotification(); | ||||
|                 if (showNextNotification) { | ||||
|                     let len = this._notificationQueue.length; | ||||
|                     if (len > 1) { | ||||
|                         this._notificationQueue.length = 0; | ||||
|                         let source = new SystemNotificationSource(); | ||||
|                         this.add(source); | ||||
|                         let notification = new Notification(source, ngettext("%d new message", "%d new messages", len).format(len)); | ||||
|                         notification.setTransient(true); | ||||
|                         notification.connect('clicked', Lang.bind(this, function() { | ||||
|                             this.openTray(); | ||||
|                         })); | ||||
|                         source.notify(notification); | ||||
|                     } else { | ||||
|                         this._showNotification(); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } else if (this._notificationState == State.SHOWN) { | ||||
|             let expired = (this._userActiveWhileNotificationShown && | ||||
| @@ -2627,6 +2625,8 @@ const MessageTray = new Lang.Class({ | ||||
|         // the mouse is moving towards it or within it. | ||||
|         this._lastSeenMouseX = x; | ||||
|         this._lastSeenMouseY = y; | ||||
|  | ||||
|         this._resetNotificationLeftTimeout(); | ||||
|     }, | ||||
|  | ||||
|     _updateShowingNotification: function() { | ||||
| @@ -2719,13 +2719,7 @@ const MessageTray = new Lang.Class({ | ||||
|             this._notificationUnfocusedId = 0; | ||||
|         } | ||||
|  | ||||
|         this._useLongerNotificationLeftTimeout = false; | ||||
|         if (this._notificationLeftTimeoutId) { | ||||
|             Mainloop.source_remove(this._notificationLeftTimeoutId); | ||||
|             this._notificationLeftTimeoutId = 0; | ||||
|             this._notificationLeftMouseX = -1; | ||||
|             this._notificationLeftMouseY = -1; | ||||
|         } | ||||
|         this._resetNotificationLeftTimeout(); | ||||
|  | ||||
|         if (animate) { | ||||
|             this._tween(this._notificationWidget, '_notificationState', State.HIDDEN, | ||||
|   | ||||
| @@ -41,15 +41,17 @@ const ModalDialog = new Lang.Class({ | ||||
|     _init: function(params) { | ||||
|         params = Params.parse(params, { shellReactive: false, | ||||
|                                         styleClass: null, | ||||
|                                         keybindingMode: Shell.KeyBindingMode.SYSTEM_MODAL, | ||||
|                                         actionMode: Shell.ActionMode.SYSTEM_MODAL, | ||||
|                                         shouldFadeIn: true, | ||||
|                                         shouldFadeOut: true, | ||||
|                                         destroyOnClose: true }); | ||||
|  | ||||
|         this.state = State.CLOSED; | ||||
|         this._hasModal = false; | ||||
|         this._keybindingMode = params.keybindingMode; | ||||
|         this._actionMode = params.actionMode; | ||||
|         this._shellReactive = params.shellReactive; | ||||
|         this._shouldFadeIn = params.shouldFadeIn; | ||||
|         this._shouldFadeOut = params.shouldFadeOut; | ||||
|         this._destroyOnClose = params.destroyOnClose; | ||||
|  | ||||
|         this._group = new St.Widget({ visible: false, | ||||
| @@ -168,7 +170,7 @@ const ModalDialog = new Lang.Class({ | ||||
|         else | ||||
|             keys = []; | ||||
|  | ||||
|         let button = new St.Button({ style_class: 'modal-dialog-button', | ||||
|         let button = new St.Button({ style_class: 'modal-dialog-button button', | ||||
|                                      button_mask: St.ButtonMask.ONE | St.ButtonMask.THREE, | ||||
|                                      reactive:    true, | ||||
|                                      can_focus:   true, | ||||
| @@ -192,7 +194,7 @@ const ModalDialog = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     placeSpinner: function(layoutInfo) { | ||||
|         let spinnerIcon = global.datadir + '/theme/process-working.svg'; | ||||
|         let spinnerIcon = Gio.File.new_for_uri('resource:///org/gnome/shell/theme/process-working.svg'); | ||||
|         this._workSpinner = new Animation.AnimatedIcon(spinnerIcon, WORK_SPINNER_ICON_SIZE); | ||||
|         this._workSpinner.actor.opacity = 0; | ||||
|         this._workSpinner.actor.show(); | ||||
| @@ -307,6 +309,15 @@ const ModalDialog = new Lang.Class({ | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _closeComplete: function() { | ||||
|         this.state = State.CLOSED; | ||||
|         this._group.hide(); | ||||
|         this.emit('closed'); | ||||
|  | ||||
|         if (this._destroyOnClose) | ||||
|             this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     close: function(timestamp) { | ||||
|         if (this.state == State.CLOSED || this.state == State.CLOSING) | ||||
|             return; | ||||
| @@ -315,20 +326,16 @@ const ModalDialog = new Lang.Class({ | ||||
|         this.popModal(timestamp); | ||||
|         this._savedKeyFocus = null; | ||||
|  | ||||
|         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(); | ||||
|                                    this.emit('closed'); | ||||
|  | ||||
|                                    if (this._destroyOnClose) | ||||
|                                        this.destroy(); | ||||
|                                }) | ||||
|                          }); | ||||
|         if (this._shouldFadeOut) | ||||
|             Tweener.addTween(this._group, | ||||
|                              { opacity: 0, | ||||
|                                time: OPEN_AND_CLOSE_TIME, | ||||
|                                transition: 'easeOutQuad', | ||||
|                                onComplete: Lang.bind(this, | ||||
|                                                      this._closeComplete) | ||||
|                              }) | ||||
|         else | ||||
|             this._closeComplete(); | ||||
|     }, | ||||
|  | ||||
|     // Drop modal status without closing the dialog; this makes the | ||||
| @@ -355,7 +362,7 @@ const ModalDialog = new Lang.Class({ | ||||
|         if (this._hasModal) | ||||
|             return true; | ||||
|         if (!Main.pushModal(this._group, { timestamp: timestamp, | ||||
|                                            keybindingMode: this._keybindingMode })) | ||||
|                                            actionMode: this._actionMode })) | ||||
|             return false; | ||||
|  | ||||
|         this._hasModal = true; | ||||
|   | ||||
| @@ -91,21 +91,6 @@ const rewriteRules = { | ||||
|     ] | ||||
| }; | ||||
|  | ||||
| const STANDARD_TRAY_ICON_IMPLEMENTATIONS = { | ||||
|     'bluetooth-applet': 'bluetooth', | ||||
|     'gnome-volume-control-applet': 'volume', // renamed to gnome-sound-applet | ||||
|                                              // when moved to control center | ||||
|     'gnome-sound-applet': 'volume', | ||||
|     'nm-applet': 'network', | ||||
|     'gnome-power-manager': 'battery', | ||||
|     'keyboard': 'keyboard', | ||||
|     'a11y-keyboard': 'a11y', | ||||
|     'kbd-scrolllock': 'keyboard', | ||||
|     'kbd-numlock': 'keyboard', | ||||
|     'kbd-capslock': 'keyboard', | ||||
|     'ibus-ui-gtk': 'keyboard' | ||||
| }; | ||||
|  | ||||
| const FdoNotificationDaemon = new Lang.Class({ | ||||
|     Name: 'FdoNotificationDaemon', | ||||
|  | ||||
| @@ -120,16 +105,10 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|  | ||||
|         this._nextNotificationId = 1; | ||||
|  | ||||
|         this._trayManager = new Shell.TrayManager(); | ||||
|         this._trayIconAddedId = this._trayManager.connect('tray-icon-added', Lang.bind(this, this._onTrayIconAdded)); | ||||
|         this._trayIconRemovedId = this._trayManager.connect('tray-icon-removed', Lang.bind(this, this._onTrayIconRemoved)); | ||||
|  | ||||
|         Shell.WindowTracker.get_default().connect('notify::focus-app', | ||||
|             Lang.bind(this, this._onFocusAppChanged)); | ||||
|         Main.overview.connect('hidden', | ||||
|             Lang.bind(this, this._onFocusAppChanged)); | ||||
|  | ||||
|         this._trayManager.manage_screen(global.screen, Main.messageTray.actor); | ||||
|     }, | ||||
|  | ||||
|     _imageForNotificationData: function(hints) { | ||||
| @@ -170,11 +149,10 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     _lookupSource: function(title, pid, trayIcon) { | ||||
|     _lookupSource: function(title, pid) { | ||||
|         for (let i = 0; i < this._sources.length; i++) { | ||||
|             let source = this._sources[i]; | ||||
|             if (source.pid == pid && | ||||
|                 (source.initialTitle == title || source.trayIcon || trayIcon)) | ||||
|             if (source.pid == pid && source.initialTitle == title) | ||||
|                 return source; | ||||
|         } | ||||
|         return null; | ||||
| @@ -191,7 +169,7 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|     // | ||||
|     // Either a pid or ndata.notification is needed to retrieve or | ||||
|     // create a source. | ||||
|     _getSource: function(title, pid, ndata, sender, trayIcon) { | ||||
|     _getSource: function(title, pid, ndata, sender) { | ||||
|         if (!pid && !(ndata && ndata.notification)) | ||||
|             return null; | ||||
|  | ||||
| @@ -202,13 +180,13 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|         if (ndata && ndata.notification) | ||||
|             return ndata.notification.source; | ||||
|  | ||||
|         let source = this._lookupSource(title, pid, trayIcon); | ||||
|         let source = this._lookupSource(title, pid); | ||||
|         if (source) { | ||||
|             source.setTitle(title); | ||||
|             return source; | ||||
|         } | ||||
|  | ||||
|         let source = new FdoNotificationDaemonSource(title, pid, sender, trayIcon, ndata ? ndata.hints['desktop-entry'] : null); | ||||
|         let source = new FdoNotificationDaemonSource(title, pid, sender, ndata ? ndata.hints['desktop-entry'] : null); | ||||
|  | ||||
|         this._sources.push(source); | ||||
|         source.connect('destroy', Lang.bind(this, function() { | ||||
| @@ -333,19 +311,6 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|         return invocation.return_value(GLib.Variant.new('(u)', [id])); | ||||
|     }, | ||||
|  | ||||
|     _makeButton: function(id, label, useActionIcons) { | ||||
|         let button = new St.Button({ can_focus: true }); | ||||
|         let iconName = id.endsWith('-symbolic') ? id : id + '-symbolic'; | ||||
|         if (useActionIcons && Gtk.IconTheme.get_default().has_icon(iconName)) { | ||||
|             button.add_style_class_name('notification-icon-button'); | ||||
|             button.child = new St.Icon({ icon_name: iconName }); | ||||
|         } else { | ||||
|             button.add_style_class_name('notification-button'); | ||||
|             button.label = label; | ||||
|         } | ||||
|         return button; | ||||
|     }, | ||||
|  | ||||
|     _notifyForSource: function(source, ndata) { | ||||
|         let [id, icon, summary, body, actions, hints, notification] = | ||||
|             [ndata.id, ndata.icon, ndata.summary, ndata.body, | ||||
| @@ -373,26 +338,17 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|                 })); | ||||
|         } | ||||
|  | ||||
|         // Mark music notifications so they can be shown in the screen shield | ||||
|         notification.isMusic = (ndata.hints['category'] == 'x-gnome.music'); | ||||
|  | ||||
|         let gicon = this._iconForNotificationData(icon, hints); | ||||
|         let gimage = this._imageForNotificationData(hints); | ||||
|  | ||||
|         let image = null; | ||||
|  | ||||
|         // If an icon is not specified, we use 'image-data' or 'image-path' hint for an icon | ||||
|         // and don't show a large image. There are currently many applications that use | ||||
|         // notify_notification_set_icon_from_pixbuf() from libnotify, which in turn sets | ||||
|         // the 'image-data' hint. These applications don't typically pass in 'app_icon' | ||||
|         // argument to Notify() and actually expect the pixbuf to be shown as an icon. | ||||
|         // So the logic here does the right thing for this case. If both an icon and either | ||||
|         // one of 'image-data' or 'image-path' are specified, we show both an icon and | ||||
|         // a large image. | ||||
|         if (gicon && gimage) | ||||
|             image = new St.Icon({ gicon: gimage, | ||||
|                                   icon_size: notification.IMAGE_SIZE }); | ||||
|         else if (!gicon && gimage) | ||||
|         // one of 'image-data' or 'image-path' are specified, the icon and takes precedence. | ||||
|         if (!gicon && gimage) | ||||
|             gicon = gimage; | ||||
|         else if (!gicon) | ||||
|             gicon = this._fallbackIconForNotificationData(hints); | ||||
| @@ -402,22 +358,18 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|                                              clear: true, | ||||
|                                              soundFile: hints['sound-file'], | ||||
|                                              soundName: hints['sound-name'] }); | ||||
|         notification.setImage(image); | ||||
|  | ||||
|         let hasDefaultAction = false; | ||||
|  | ||||
|         if (actions.length) { | ||||
|             let useActionIcons = (hints['action-icons'] == true); | ||||
|  | ||||
|             for (let i = 0; i < actions.length - 1; i += 2) { | ||||
|                 let [actionId, label] = [actions[i], actions[i+1]]; | ||||
|                 if (actionId == 'default') { | ||||
|                 if (actionId == 'default') | ||||
|                     hasDefaultAction = true; | ||||
|                 } else { | ||||
|                     notification.addButton(this._makeButton(actionId, label, useActionIcons), Lang.bind(this, function() { | ||||
|                 else | ||||
|                     notification.addAction(label, Lang.bind(this, function() { | ||||
|                         this._emitActionInvoked(ndata.id, actionId); | ||||
|                     })); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -463,7 +415,7 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|     GetCapabilities: function() { | ||||
|         return [ | ||||
|             'actions', | ||||
|             'action-icons', | ||||
|             // 'action-icons', | ||||
|             'body', | ||||
|             // 'body-hyperlinks', | ||||
|             // 'body-images', | ||||
| @@ -506,20 +458,6 @@ const FdoNotificationDaemon = new Lang.Class({ | ||||
|     _emitActionInvoked: function(id, action) { | ||||
|         this._dbusImpl.emit_signal('ActionInvoked', | ||||
|                                    GLib.Variant.new('(us)', [id, action])); | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconAdded: function(o, icon) { | ||||
|         let wmClass = icon.wm_class ? icon.wm_class.toLowerCase() : ''; | ||||
|         if (STANDARD_TRAY_ICON_IMPLEMENTATIONS[wmClass] !== undefined) | ||||
|             return; | ||||
|  | ||||
|         let source = this._getSource(icon.title || icon.wm_class || C_("program", "Unknown"), icon.pid, null, null, icon); | ||||
|     }, | ||||
|  | ||||
|     _onTrayIconRemoved: function(o, icon) { | ||||
|         let source = this._lookupSource(null, icon.pid, true); | ||||
|         if (source) | ||||
|             source.destroy(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -527,10 +465,9 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|     Name: 'FdoNotificationDaemonSource', | ||||
|     Extends: MessageTray.Source, | ||||
|  | ||||
|     _init: function(title, pid, sender, trayIcon, appId) { | ||||
|     _init: function(title, pid, sender, appId) { | ||||
|         // Need to set the app before chaining up, so | ||||
|         // methods called from the parent constructor can find it | ||||
|         this.trayIcon = trayIcon; | ||||
|         this.pid = pid; | ||||
|         this.app = this._getApp(appId); | ||||
|  | ||||
| @@ -550,12 +487,6 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|                                                               Lang.bind(this, this._onNameVanished)); | ||||
|         else | ||||
|             this._nameWatcherId = 0; | ||||
|  | ||||
|         if (this.trayIcon) { | ||||
|             // Try again finding the app, using the WM_CLASS from the tray icon | ||||
|             this._setSummaryIcon(this.trayIcon); | ||||
|             this.useNotificationIcon = false; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _createPolicy: function() { | ||||
| @@ -573,15 +504,14 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|         // of which аre removed from DBus immediately. | ||||
|         // Sender being removed from DBus would normally result in a tray icon being removed, | ||||
|         // so allow the code path that handles the tray icon being removed to handle that case. | ||||
|         if (!this.trayIcon && this.app) | ||||
|         if (this.app) | ||||
|             this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     processNotification: function(notification, gicon) { | ||||
|         if (gicon) | ||||
|             this._gicon = gicon; | ||||
|         if (!this.trayIcon) | ||||
|             this.iconUpdated(); | ||||
|         this.iconUpdated(); | ||||
|  | ||||
|         let tracker = Shell.WindowTracker.get_default(); | ||||
|         if (notification.resident && this.app && tracker.focus_app == this.app) | ||||
| @@ -590,29 +520,6 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|             this.notify(notification); | ||||
|     }, | ||||
|  | ||||
|     handleSummaryClick: function(button) { | ||||
|         if (!this.trayIcon) | ||||
|             return false; | ||||
|  | ||||
|         let event = Clutter.get_current_event(); | ||||
|  | ||||
|         // Left clicks are passed through only where there aren't unacknowledged | ||||
|         // notifications, so it possible to open them in summary mode; right | ||||
|         // clicks are always forwarded, as the right click menu is not useful for | ||||
|         // tray icons | ||||
|         if (button == 1 && | ||||
|             this.notifications.length > 0) | ||||
|             return false; | ||||
|  | ||||
|         let id = global.stage.connect('deactivate', Lang.bind(this, function () { | ||||
|             global.stage.disconnect(id); | ||||
|             this.trayIcon.click(event); | ||||
|         })); | ||||
|  | ||||
|         Main.overview.hide(); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _getApp: function(appId) { | ||||
|         let app; | ||||
|  | ||||
| @@ -620,16 +527,6 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|         if (app != null) | ||||
|             return app; | ||||
|  | ||||
|         if (this.trayIcon) { | ||||
|             app = Shell.AppSystem.get_default().lookup_startup_wmclass(this.trayIcon.wm_class); | ||||
|             if (app != null) | ||||
|                 return app; | ||||
|  | ||||
|             app = Shell.AppSystem.get_default().lookup_desktop_wmclass(this.trayIcon.wm_class); | ||||
|             if (app != null) | ||||
|                 return app; | ||||
|         } | ||||
|  | ||||
|         if (appId) { | ||||
|             app = Shell.AppSystem.get_default().lookup_app(appId + '.desktop'); | ||||
|             if (app != null) | ||||
| @@ -655,8 +552,7 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _lastNotificationRemoved: function() { | ||||
|         if (!this.trayIcon) | ||||
|             this.destroy(); | ||||
|         this.destroy(); | ||||
|     }, | ||||
|  | ||||
|     openApp: function() { | ||||
| @@ -677,11 +573,7 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     createIcon: function(size) { | ||||
|         if (this.trayIcon) { | ||||
|             return new Clutter.Clone({ width: size, | ||||
|                                        height: size, | ||||
|                                        source: this.trayIcon }); | ||||
|         } else if (this.app) { | ||||
|         if (this.app) { | ||||
|             return this.app.create_icon_texture(size); | ||||
|         } else if (this._gicon) { | ||||
|             return new St.Icon({ gicon: this._gicon, | ||||
| @@ -692,6 +584,12 @@ const FdoNotificationDaemonSource = new Lang.Class({ | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const PRIORITY_URGENCY_MAP = { | ||||
|     low: MessageTray.Urgency.LOW, | ||||
|     normal: MessageTray.Urgency.NORMAL, | ||||
|     high: MessageTray.Urgency.HIGH, | ||||
|     urgent: MessageTray.Urgency.CRITICAL | ||||
| }; | ||||
|  | ||||
| const GtkNotificationDaemonNotification = new Lang.Class({ | ||||
|     Name: 'GtkNotificationDaemonNotification', | ||||
| @@ -705,12 +603,20 @@ const GtkNotificationDaemonNotification = new Lang.Class({ | ||||
|               "body": body, | ||||
|               "icon": gicon, | ||||
|               "urgent": urgent, | ||||
|               "priority": priority, | ||||
|               "buttons": buttons, | ||||
|               "default-action": defaultAction, | ||||
|               "default-action-target": defaultActionTarget } = notification; | ||||
|  | ||||
|         this.setUrgency(urgent.unpack() ? MessageTray.Urgency.CRITICAL | ||||
|                                         : MessageTray.Urgency.NORMAL); | ||||
|         if (priority) { | ||||
|             let urgency = PRIORITY_URGENCY_MAP[priority.unpack()]; | ||||
|             this.setUrgency(urgency != undefined ? urgency : MessageTray.Urgency.NORMAL); | ||||
|         } else if (urgent) { | ||||
|             this.setUrgency(urgent.unpack() ? MessageTray.Urgency.CRITICAL | ||||
|                             : MessageTray.Urgency.NORMAL); | ||||
|         } else { | ||||
|             this.setUrgency(MessageTray.Urgency.NORMAL); | ||||
|         } | ||||
|  | ||||
|         if (buttons) { | ||||
|             buttons.deep_unpack().forEach(Lang.bind(this, function(button) { | ||||
|   | ||||
							
								
								
									
										141
									
								
								js/ui/osdMonitorLabeler.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,141 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Gio = imports.gi.Gio; | ||||
| const St = imports.gi.St; | ||||
|  | ||||
| const Lang = imports.lang; | ||||
| const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
| const Meta = imports.gi.Meta; | ||||
|  | ||||
| const FADE_TIME = 0.1; | ||||
|  | ||||
| const OsdMonitorLabel = new Lang.Class({ | ||||
|     Name: 'OsdMonitorLabel', | ||||
|  | ||||
|     _init: function(monitor, label) { | ||||
|         this._actor = new St.Widget({ opacity: 0, | ||||
|                                       x_expand: true, | ||||
|                                       y_expand: true }); | ||||
|  | ||||
|         this._monitor = monitor; | ||||
|  | ||||
|         this._box = new St.BoxLayout({ style_class: 'osd-window', | ||||
|                                        vertical: true }); | ||||
|         this._actor.add_actor(this._box); | ||||
|  | ||||
|         this._label = new St.Label({ style_class: 'osd-monitor-label', | ||||
|                                      text: label }); | ||||
|         this._box.add(this._label); | ||||
|  | ||||
|         Main.uiGroup.add_child(this._actor); | ||||
|         Main.uiGroup.set_child_above_sibling(this._actor, null); | ||||
|         this._position(); | ||||
|  | ||||
|         Meta.disable_unredirect_for_screen(global.screen); | ||||
|         Tweener.addTween(this._actor, | ||||
|                          { opacity: 255, | ||||
|                            time: FADE_TIME, | ||||
|                            transition: 'easeOutQuad' }); | ||||
|     }, | ||||
|  | ||||
|     _position: function() { | ||||
|         let workArea = Main.layoutManager.getWorkAreaForMonitor(this._monitor); | ||||
|  | ||||
|         if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL) | ||||
|             this._box.x = workArea.x + (workArea.width - this._box.width); | ||||
|         else | ||||
|             this._box.x = workArea.x; | ||||
|  | ||||
|         this._box.y = workArea.y; | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         Tweener.addTween(this._actor, | ||||
|                          { opacity: 0, | ||||
|                            time: FADE_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, function() { | ||||
|                                this._actor.destroy(); | ||||
|                                Meta.enable_unredirect_for_screen(global.screen); | ||||
|                            }) | ||||
|                          }); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| const OsdMonitorLabeler = new Lang.Class({ | ||||
|     Name: 'OsdMonitorLabeler', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._monitorManager = Meta.MonitorManager.get(); | ||||
|         this._client = null; | ||||
|         this._clientWatchId = 0; | ||||
|         this._osdLabels = []; | ||||
|         this._monitorLabels = null; | ||||
|         Main.layoutManager.connect('monitors-changed', | ||||
|                                     Lang.bind(this, this._reset)); | ||||
|         this._reset(); | ||||
|     }, | ||||
|  | ||||
|     _reset: function() { | ||||
|         for (let i in this._osdLabels) | ||||
|             this._osdLabels[i].destroy(); | ||||
|         this._osdLabels = []; | ||||
|         this._monitorLabels = new Map(); | ||||
|         let monitors = Main.layoutManager.monitors; | ||||
|         for (let i in monitors) | ||||
|             this._monitorLabels.set(monitors[i].index, []); | ||||
|     }, | ||||
|  | ||||
|     _trackClient: function(client) { | ||||
|         if (this._client) | ||||
|             return (this._client == client); | ||||
|  | ||||
|         this._client = client; | ||||
|         this._clientWatchId = Gio.bus_watch_name(Gio.BusType.SESSION, client, 0, null, | ||||
|                                                  Lang.bind(this, function(c, name) { | ||||
|                                                      this.hide(name); | ||||
|                                                  })); | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _untrackClient: function(client) { | ||||
|         if (!this._client || this._client != client) | ||||
|             return false; | ||||
|  | ||||
|         Gio.bus_unwatch_name(this._clientWatchId); | ||||
|         this._clientWatchId = 0; | ||||
|         this._client = null; | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     show: function(client, params) { | ||||
|         if (!this._trackClient(client)) | ||||
|             return; | ||||
|  | ||||
|         this._reset(); | ||||
|  | ||||
|         for (let id in params) { | ||||
|             let monitor = this._monitorManager.get_monitor_for_output(id); | ||||
|             if (monitor == -1) | ||||
|                 continue; | ||||
|             this._monitorLabels.get(monitor).push(params[id].deep_unpack()); | ||||
|         } | ||||
|  | ||||
|         // In mirrored display setups, more than one physical outputs | ||||
|         // might be showing the same logical monitor. In that case, we | ||||
|         // join each output's labels on the same OSD widget. | ||||
|         for (let [monitor, labels] of this._monitorLabels.entries()) { | ||||
|             labels.sort(); | ||||
|             this._osdLabels.push(new OsdMonitorLabel(monitor, labels.join(' '))); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     hide: function(client) { | ||||
|         if (!this._untrackClient(client)) | ||||
|             return; | ||||
|  | ||||
|         this._reset(); | ||||
|     } | ||||
| }); | ||||
| @@ -185,7 +185,7 @@ const Overview = new Lang.Class({ | ||||
|         for (let i = 0; i < Main.layoutManager.monitors.length; i++) { | ||||
|             let bgManager = new Background.BackgroundManager({ container: this._backgroundGroup, | ||||
|                                                                monitorIndex: i, | ||||
|                                                                effects: Meta.BackgroundEffects.VIGNETTE }); | ||||
|                                                                vignette: true }); | ||||
|             this._bgManagers.push(bgManager); | ||||
|         } | ||||
|     }, | ||||
| @@ -193,11 +193,9 @@ const Overview = new Lang.Class({ | ||||
|     _unshadeBackgrounds: function() { | ||||
|         let backgrounds = this._backgroundGroup.get_children(); | ||||
|         for (let i = 0; i < backgrounds.length; i++) { | ||||
|             let background = backgrounds[i]._delegate; | ||||
|  | ||||
|             Tweener.addTween(background, | ||||
|             Tweener.addTween(backgrounds[i], | ||||
|                              { brightness: 1.0, | ||||
|                                vignetteSharpness: 0.0, | ||||
|                                vignette_sharpness: 0.0, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
| @@ -207,11 +205,9 @@ const Overview = new Lang.Class({ | ||||
|     _shadeBackgrounds: function() { | ||||
|         let backgrounds = this._backgroundGroup.get_children(); | ||||
|         for (let i = 0; i < backgrounds.length; i++) { | ||||
|             let background = backgrounds[i]._delegate; | ||||
|  | ||||
|             Tweener.addTween(background, | ||||
|             Tweener.addTween(backgrounds[i], | ||||
|                              { brightness: Lightbox.VIGNETTE_BRIGHTNESS, | ||||
|                                vignetteSharpness: Lightbox.VIGNETTE_SHARPNESS, | ||||
|                                vignette_sharpness: Lightbox.VIGNETTE_SHARPNESS, | ||||
|                                time: SHADE_ANIMATION_TIME, | ||||
|                                transition: 'easeOutQuad' | ||||
|                              }); | ||||
| @@ -242,7 +238,7 @@ const Overview = new Lang.Class({ | ||||
|                                         opacity: 0 }); | ||||
|         this._overview.add_actor(this._panelGhost); | ||||
|  | ||||
|         this._searchEntry = new St.Entry({ name: 'searchEntry', | ||||
|         this._searchEntry = new St.Entry({ style_class: 'search-entry', | ||||
|                                            /* Translators: this is the text displayed | ||||
|                                               in the search entry when no search is | ||||
|                                               active; it should not exceed ~30 | ||||
| @@ -425,8 +421,6 @@ const Overview = new Lang.Class({ | ||||
|         this.emit('windows-restacked', stackIndices); | ||||
|     }, | ||||
|  | ||||
|     //// Public methods //// | ||||
|  | ||||
|     beginItemDrag: function(source) { | ||||
|         this.emit('item-drag-begin'); | ||||
|         this._inDrag = true; | ||||
| @@ -441,37 +435,20 @@ const Overview = new Lang.Class({ | ||||
|         this._inDrag = false; | ||||
|     }, | ||||
|  | ||||
|     beginWindowDrag: function(clone) { | ||||
|         this.emit('window-drag-begin', clone); | ||||
|     beginWindowDrag: function(window) { | ||||
|         this.emit('window-drag-begin', window); | ||||
|         this._inDrag = true; | ||||
|     }, | ||||
|  | ||||
|     cancelledWindowDrag: function(clone) { | ||||
|         this.emit('window-drag-cancelled', clone); | ||||
|     cancelledWindowDrag: function(window) { | ||||
|         this.emit('window-drag-cancelled', window); | ||||
|     }, | ||||
|  | ||||
|     endWindowDrag: function(clone) { | ||||
|         this.emit('window-drag-end', clone); | ||||
|     endWindowDrag: function(window) { | ||||
|         this.emit('window-drag-end', window); | ||||
|         this._inDrag = false; | ||||
|     }, | ||||
|  | ||||
|     // show: | ||||
|     // | ||||
|     // Animates the overview visible and grabs mouse and keyboard input | ||||
|     show: function() { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|         if (this._shown) | ||||
|             return; | ||||
|         this._shown = true; | ||||
|  | ||||
|         if (!this._syncGrab()) | ||||
|             return; | ||||
|  | ||||
|         Main.layoutManager.showOverview(); | ||||
|         this._animateVisible(); | ||||
|     }, | ||||
|  | ||||
|     focusSearch: function() { | ||||
|         this.show(); | ||||
|         this._searchEntry.grab_key_focus(); | ||||
| @@ -504,6 +481,69 @@ const Overview = new Lang.Class({ | ||||
|                          }); | ||||
|     }, | ||||
|  | ||||
|     // Checks if the Activities button is currently sensitive to | ||||
|     // clicks. The first call to this function within the | ||||
|     // OVERVIEW_ACTIVATION_TIMEOUT time of the hot corner being | ||||
|     // triggered will return false. This avoids opening and closing | ||||
|     // the overview if the user both triggered the hot corner and | ||||
|     // clicked the Activities button. | ||||
|     shouldToggleByCornerOrButton: function() { | ||||
|         if (this.animationInProgress) | ||||
|             return false; | ||||
|         if (this._inDrag) | ||||
|             return false; | ||||
|         if (this._activationTime == 0 || Date.now() / 1000 - this._activationTime > OVERVIEW_ACTIVATION_TIMEOUT) | ||||
|             return true; | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     _syncGrab: function() { | ||||
|         // We delay grab changes during animation so that when removing the | ||||
|         // overview we don't have a problem with the release of a press/release | ||||
|         // going to an application. | ||||
|         if (this.animationInProgress) | ||||
|             return true; | ||||
|  | ||||
|         if (this._shown) { | ||||
|             let shouldBeModal = !this._inXdndDrag; | ||||
|             if (shouldBeModal) { | ||||
|                 if (!this._modal) { | ||||
|                     if (Main.pushModal(this._overview, | ||||
|                                        { actionMode: Shell.ActionMode.OVERVIEW })) { | ||||
|                         this._modal = true; | ||||
|                     } else { | ||||
|                         this.hide(); | ||||
|                         return false; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             if (this._modal) { | ||||
|                 Main.popModal(this._overview); | ||||
|                 this._modal = false; | ||||
|             } | ||||
|         } | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     // show: | ||||
|     // | ||||
|     // Animates the overview visible and grabs mouse and keyboard input | ||||
|     show: function() { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|         if (this._shown) | ||||
|             return; | ||||
|         this._shown = true; | ||||
|  | ||||
|         if (!this._syncGrab()) | ||||
|             return; | ||||
|  | ||||
|         Main.layoutManager.showOverview(); | ||||
|         this._animateVisible(); | ||||
|     }, | ||||
|  | ||||
|  | ||||
|     _animateVisible: function() { | ||||
|         if (this.visible || this.animationInProgress) | ||||
|             return; | ||||
| @@ -531,6 +571,20 @@ const Overview = new Lang.Class({ | ||||
|         this.emit('showing'); | ||||
|     }, | ||||
|  | ||||
|     _showDone: function() { | ||||
|         this.animationInProgress = false; | ||||
|         this._desktopFade.hide(); | ||||
|         this._coverPane.hide(); | ||||
|  | ||||
|         this.emit('shown'); | ||||
|         // Handle any calls to hide* while we were showing | ||||
|         if (!this._shown) | ||||
|             this._animateNotVisible(); | ||||
|  | ||||
|         this._syncGrab(); | ||||
|         global.sync_pointer(); | ||||
|     }, | ||||
|  | ||||
|     // hide: | ||||
|     // | ||||
|     // Reverses the effect of show() | ||||
| @@ -551,68 +605,12 @@ const Overview = new Lang.Class({ | ||||
|                 return; | ||||
|         } | ||||
|  | ||||
|         this._animateNotVisible(); | ||||
|  | ||||
|         this._shown = false; | ||||
|  | ||||
|         this._animateNotVisible(); | ||||
|         this._syncGrab(); | ||||
|     }, | ||||
|  | ||||
|     toggle: function() { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         if (this.visible) | ||||
|             this.hide(); | ||||
|         else | ||||
|             this.show(); | ||||
|     }, | ||||
|  | ||||
|     // Checks if the Activities button is currently sensitive to | ||||
|     // clicks. The first call to this function within the | ||||
|     // OVERVIEW_ACTIVATION_TIMEOUT time of the hot corner being | ||||
|     // triggered will return false. This avoids opening and closing | ||||
|     // the overview if the user both triggered the hot corner and | ||||
|     // clicked the Activities button. | ||||
|     shouldToggleByCornerOrButton: function() { | ||||
|         if (this.animationInProgress) | ||||
|             return false; | ||||
|         if (this._inDrag) | ||||
|             return false; | ||||
|         if (this._activationTime == 0 || Date.now() / 1000 - this._activationTime > OVERVIEW_ACTIVATION_TIMEOUT) | ||||
|             return true; | ||||
|         return false; | ||||
|     }, | ||||
|  | ||||
|     //// Private methods //// | ||||
|  | ||||
|     _syncGrab: function() { | ||||
|         // We delay grab changes during animation so that when removing the | ||||
|         // overview we don't have a problem with the release of a press/release | ||||
|         // going to an application. | ||||
|         if (this.animationInProgress) | ||||
|             return true; | ||||
|  | ||||
|         if (this._shown) { | ||||
|             let shouldBeModal = !this._inXdndDrag; | ||||
|             if (shouldBeModal) { | ||||
|                 if (!this._modal) { | ||||
|                     if (Main.pushModal(this._overview, | ||||
|                                        { keybindingMode: Shell.KeyBindingMode.OVERVIEW })) { | ||||
|                         this._modal = true; | ||||
|                     } else { | ||||
|                         this.hide(); | ||||
|                         return false; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             if (this._modal) { | ||||
|                 Main.popModal(this._overview); | ||||
|                 this._modal = false; | ||||
|             } | ||||
|         } | ||||
|         return true; | ||||
|     }, | ||||
|  | ||||
|     _animateNotVisible: function() { | ||||
|         if (!this.visible || this.animationInProgress) | ||||
| @@ -621,7 +619,7 @@ const Overview = new Lang.Class({ | ||||
|         this.animationInProgress = true; | ||||
|         this.visibleTarget = false; | ||||
|  | ||||
|         this.viewSelector.zoomFromOverview(); | ||||
|         this.viewSelector.animateFromOverview(); | ||||
|  | ||||
|         // Make other elements fade out. | ||||
|         Tweener.addTween(this._stack, | ||||
| @@ -638,20 +636,6 @@ const Overview = new Lang.Class({ | ||||
|         this.emit('hiding'); | ||||
|     }, | ||||
|  | ||||
|     _showDone: function() { | ||||
|         this.animationInProgress = false; | ||||
|         this._desktopFade.hide(); | ||||
|         this._coverPane.hide(); | ||||
|  | ||||
|         this.emit('shown'); | ||||
|         // Handle any calls to hide* while we were showing | ||||
|         if (!this._shown) | ||||
|             this._animateNotVisible(); | ||||
|  | ||||
|         this._syncGrab(); | ||||
|         global.sync_pointer(); | ||||
|     }, | ||||
|  | ||||
|     _hideDone: function() { | ||||
|         // Re-enable unredirection | ||||
|         Meta.enable_unredirect_for_screen(global.screen); | ||||
| @@ -677,6 +661,20 @@ const Overview = new Lang.Class({ | ||||
|             this._fakePointerEvent(); | ||||
|             this._needsFakePointerEvent = false; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     toggle: function() { | ||||
|         if (this.isDummy) | ||||
|             return; | ||||
|  | ||||
|         if (this.visible) | ||||
|             this.hide(); | ||||
|         else | ||||
|             this.show(); | ||||
|     }, | ||||
|  | ||||
|     getShowAppsButton: function() { | ||||
|         return this._dash.showAppsButton; | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(Overview.prototype); | ||||
|   | ||||
| @@ -36,7 +36,7 @@ const SlideLayout = new Lang.Class({ | ||||
|  | ||||
|     _init: function(params) { | ||||
|         this._slideX = 1; | ||||
|         this._translationX = 0; | ||||
|         this._translationX = undefined; | ||||
|         this._direction = SlideDirection.LEFT; | ||||
|  | ||||
|         this.parent(params); | ||||
| @@ -64,7 +64,8 @@ const SlideLayout = new Lang.Class({ | ||||
|         // flags only determine what to do if the allocated box is bigger | ||||
|         // than the actor's box. | ||||
|         let realDirection = getRtlSlideDirection(this._direction, child); | ||||
|         let alignX = (realDirection == SlideDirection.LEFT) ? (availWidth - natWidth) : 0; | ||||
|         let alignX = (realDirection == SlideDirection.LEFT) ? (availWidth - natWidth) | ||||
|                                                             : (availWidth - natWidth * this._slideX); | ||||
|  | ||||
|         let actorBox = new Clutter.ActorBox(); | ||||
|         actorBox.x1 = box.x1 + alignX + this._translationX; | ||||
| @@ -118,7 +119,6 @@ const SlidingControl = new Lang.Class({ | ||||
|                                      style_class: 'overview-controls', | ||||
|                                      clip_to_allocation: true }); | ||||
|  | ||||
|         Main.overview.connect('showing', Lang.bind(this, this._onOverviewShowing)); | ||||
|         Main.overview.connect('hiding', Lang.bind(this, this._onOverviewHiding)); | ||||
|  | ||||
|         Main.overview.connect('item-drag-begin', Lang.bind(this, this._onDragBegin)); | ||||
| @@ -162,7 +162,8 @@ const SlidingControl = new Lang.Class({ | ||||
|         let translationEnd = 0; | ||||
|         let translation = this._getTranslation(); | ||||
|  | ||||
|         if (this._visible) { | ||||
|         let shouldShow = (this._getSlide() > 0); | ||||
|         if (shouldShow) { | ||||
|             translationStart = translation; | ||||
|         } else { | ||||
|             translationEnd = translation; | ||||
| @@ -177,14 +178,9 @@ const SlidingControl = new Lang.Class({ | ||||
|                                         transition: 'easeOutQuad' }); | ||||
|     }, | ||||
|  | ||||
|     _onOverviewShowing: function() { | ||||
|         this._visible = true; | ||||
|         this.layout.slideX = this._getSlide(); | ||||
|         this.layout.translationX = this._getTranslation(); | ||||
|         this.slideIn(); | ||||
|     }, | ||||
|  | ||||
|     _onOverviewHiding: function() { | ||||
|         // We need to explicitly slideOut since showing pages | ||||
|         // doesn't imply sliding out, instead, hiding the overview does. | ||||
|         this.slideOut(); | ||||
|     }, | ||||
|  | ||||
| @@ -198,7 +194,7 @@ const SlidingControl = new Lang.Class({ | ||||
|  | ||||
|     _onDragBegin: function() { | ||||
|         this._inDrag = true; | ||||
|         this.layout.translationX = 0; | ||||
|         this._updateTranslation(); | ||||
|         this._updateSlide(); | ||||
|     }, | ||||
|  | ||||
| @@ -223,7 +219,6 @@ const SlidingControl = new Lang.Class({ | ||||
|  | ||||
|     slideIn: function() { | ||||
|         this._visible = true; | ||||
|         this._updateTranslation(); | ||||
|         // we will update slideX and the translation from pageEmpty | ||||
|     }, | ||||
|  | ||||
| @@ -462,9 +457,6 @@ const MessagesIndicator = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onSourceAdded: function(tray, source) { | ||||
|         if (source.trayIcon) | ||||
|             return; | ||||
|  | ||||
|         source.connect('count-updated', Lang.bind(this, this._updateCount)); | ||||
|         this._sources.push(source); | ||||
|         this._updateCount(); | ||||
|   | ||||
							
								
								
									
										217
									
								
								js/ui/panel.js
									
									
									
									
									
								
							
							
						
						| @@ -14,7 +14,6 @@ const St = imports.gi.St; | ||||
| const Signals = imports.signals; | ||||
| const Atk = imports.gi.Atk; | ||||
|  | ||||
|  | ||||
| const Animation = imports.ui.animation; | ||||
| const Config = imports.misc.config; | ||||
| const CtrlAltTab = imports.ui.ctrlAltTab; | ||||
| @@ -27,6 +26,7 @@ const Main = imports.ui.main; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const PANEL_ICON_SIZE = 24; | ||||
| const APP_MENU_ICON_MARGIN = 2; | ||||
|  | ||||
| const BUTTON_DND_ACTIVATION_TIMEOUT = 250; | ||||
|  | ||||
| @@ -75,90 +75,6 @@ function _unpremultiply(color) { | ||||
|                                blue: blue, alpha: color.alpha }); | ||||
| }; | ||||
|  | ||||
| const TextShadower = new Lang.Class({ | ||||
|     Name: 'TextShadower', | ||||
|  | ||||
|     _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._label = new St.Label(); | ||||
|         this.actor.add_actor(this._label); | ||||
|         for (let i = 0; i < 4; i++) { | ||||
|             let actor = new St.Label({ style_class: 'label-shadow' }); | ||||
|             actor.clutter_text.ellipsize = Pango.EllipsizeMode.END; | ||||
|             this.actor.add_actor(actor); | ||||
|         } | ||||
|         this._label.raise_top(); | ||||
|     }, | ||||
|  | ||||
|     setText: function(text) { | ||||
|         let children = this.actor.get_children(); | ||||
|         for (let i = 0; i < children.length; i++) | ||||
|             children[i].set_text(text); | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         let [minWidth, natWidth] = this._label.get_preferred_width(forHeight); | ||||
|         alloc.min_size = minWidth + 2; | ||||
|         alloc.natural_size = natWidth + 2; | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let [minHeight, natHeight] = this._label.get_preferred_height(forWidth); | ||||
|         alloc.min_size = minHeight + 2; | ||||
|         alloc.natural_size = natHeight + 2; | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
|         let children = this.actor.get_children(); | ||||
|  | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
|  | ||||
|         let [minChildWidth, minChildHeight, natChildWidth, natChildHeight] = | ||||
|             this._label.get_preferred_size(); | ||||
|  | ||||
|         let childWidth = Math.min(natChildWidth, availWidth - 2); | ||||
|         let childHeight = Math.min(natChildHeight, availHeight - 2); | ||||
|  | ||||
|         for (let i = 0; i < children.length; i++) { | ||||
|             let child = children[i]; | ||||
|             let childBox = new Clutter.ActorBox(); | ||||
|             // The order of the labels here is arbitrary, except | ||||
|             // we know the "real" label is at the end because Clutter.Actor | ||||
|             // sorts by Z order | ||||
|             switch (i) { | ||||
|                 case 0: // top | ||||
|                     childBox.x1 = 1; | ||||
|                     childBox.y1 = 0; | ||||
|                     break; | ||||
|                 case 1: // right | ||||
|                     childBox.x1 = 2; | ||||
|                     childBox.y1 = 1; | ||||
|                     break; | ||||
|                 case 2: // bottom | ||||
|                     childBox.x1 = 1; | ||||
|                     childBox.y1 = 2; | ||||
|                     break; | ||||
|                 case 3: // left | ||||
|                     childBox.x1 = 0; | ||||
|                     childBox.y1 = 1; | ||||
|                     break; | ||||
|                 case 4: // center | ||||
|                     childBox.x1 = 1; | ||||
|                     childBox.y1 = 1; | ||||
|                     break; | ||||
|             } | ||||
|             childBox.x2 = childBox.x1 + childWidth; | ||||
|             childBox.y2 = childBox.y1 + childHeight; | ||||
|             child.allocate(childBox, flags); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|  | ||||
| /** | ||||
|  * AppMenuButton: | ||||
|  * | ||||
| @@ -182,6 +98,7 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._targetApp = null; | ||||
|         this._appMenuNotifyId = 0; | ||||
|         this._actionGroupNotifyId = 0; | ||||
|         this._busyNotifyId = 0; | ||||
|  | ||||
|         let bin = new St.Bin({ name: 'appMenu' }); | ||||
|         bin.connect('style-changed', Lang.bind(this, this._onStyleChanged)); | ||||
| @@ -190,33 +107,20 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this.actor.bind_property("reactive", this.actor, "can-focus", 0); | ||||
|         this.actor.reactive = false; | ||||
|  | ||||
|         this._container = new Shell.GenericContainer(); | ||||
|         this._container = new St.BoxLayout({ style_class: 'panel-status-menu-box' }); | ||||
|         bin.set_child(this._container); | ||||
|         this._container.connect('get-preferred-width', Lang.bind(this, this._getContentPreferredWidth)); | ||||
|         this._container.connect('get-preferred-height', Lang.bind(this, this._getContentPreferredHeight)); | ||||
|         this._container.connect('allocate', Lang.bind(this, this._contentAllocate)); | ||||
|  | ||||
|         let textureCache = St.TextureCache.get_default(); | ||||
|         textureCache.connect('icon-theme-changed', | ||||
|                              Lang.bind(this, this._onIconThemeChanged)); | ||||
|  | ||||
|         this._iconBox = new Shell.Slicer({ name: 'appMenuIcon' }); | ||||
|         this._iconBox.connect('style-changed', | ||||
|                               Lang.bind(this, this._onIconBoxStyleChanged)); | ||||
|         this._iconBox.connect('notify::allocation', | ||||
|                               Lang.bind(this, this._updateIconBoxClip)); | ||||
|         this._iconBox = new St.Bin({ style_class: 'app-menu-icon' }); | ||||
|         this._container.add_actor(this._iconBox); | ||||
|  | ||||
|         this._hbox = new St.BoxLayout({ style_class: 'panel-status-menu-box' }); | ||||
|         this._container.add_actor(this._hbox); | ||||
|  | ||||
|         this._label = new TextShadower(); | ||||
|         this._label.actor.y_align = Clutter.ActorAlign.CENTER; | ||||
|         this._hbox.add_actor(this._label.actor); | ||||
|         this._label = new St.Label(); | ||||
|         this._container.add_actor(this._label); | ||||
|         this._arrow = PopupMenu.arrowIcon(St.Side.BOTTOM); | ||||
|         this._hbox.add_actor(this._arrow); | ||||
|  | ||||
|         this._iconBottomClip = 0; | ||||
|         this._container.add_actor(this._arrow); | ||||
|  | ||||
|         this._visible = !Main.overview.visible; | ||||
|         if (!this._visible) | ||||
| @@ -274,25 +178,19 @@ const AppMenuButton = new Lang.Class({ | ||||
|     _onStyleChanged: function(actor) { | ||||
|         let node = actor.get_theme_node(); | ||||
|         let [success, icon] = node.lookup_url('spinner-image', false); | ||||
|         if (!success || this._spinnerIcon == icon) | ||||
|         if (!success || (this._spinnerIcon && this._spinnerIcon.equal(icon))) | ||||
|             return; | ||||
|         this._spinnerIcon = icon; | ||||
|         this._spinner = new Animation.AnimatedIcon(this._spinnerIcon, PANEL_ICON_SIZE); | ||||
|         this._hbox.add_actor(this._spinner.actor); | ||||
|         this._container.add_actor(this._spinner.actor); | ||||
|         this._spinner.actor.hide(); | ||||
|     }, | ||||
|  | ||||
|     _onIconBoxStyleChanged: function() { | ||||
|         let node = this._iconBox.get_theme_node(); | ||||
|         this._iconBottomClip = node.get_length('app-icon-bottom-clip'); | ||||
|         this._updateIconBoxClip(); | ||||
|     }, | ||||
|  | ||||
|     _syncIcon: function() { | ||||
|         if (!this._targetApp) | ||||
|             return; | ||||
|  | ||||
|         let icon = this._targetApp.get_faded_icon(2 * PANEL_ICON_SIZE, this._iconBox.text_direction); | ||||
|         let icon = this._targetApp.create_icon_texture(PANEL_ICON_SIZE - APP_MENU_ICON_MARGIN); | ||||
|         this._iconBox.set_child(icon); | ||||
|     }, | ||||
|  | ||||
| @@ -303,16 +201,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._syncIcon(); | ||||
|     }, | ||||
|  | ||||
|     _updateIconBoxClip: function() { | ||||
|         let allocation = this._iconBox.allocation; | ||||
|         if (this._iconBottomClip > 0) | ||||
|             this._iconBox.set_clip(0, 0, | ||||
|                                    allocation.x2 - allocation.x1, | ||||
|                                    allocation.y2 - allocation.y1 - this._iconBottomClip); | ||||
|         else | ||||
|             this._iconBox.remove_clip(); | ||||
|     }, | ||||
|  | ||||
|     stopAnimation: function() { | ||||
|         if (this._stop) | ||||
|             return; | ||||
| @@ -345,64 +233,6 @@ const AppMenuButton = new Lang.Class({ | ||||
|         this._spinner.actor.show(); | ||||
|     }, | ||||
|  | ||||
|     _getContentPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         let [minSize, naturalSize] = this._iconBox.get_preferred_width(forHeight); | ||||
|         alloc.min_size = minSize; | ||||
|         alloc.natural_size = naturalSize; | ||||
|         [minSize, naturalSize] = this._hbox.get_preferred_width(forHeight); | ||||
|         alloc.min_size = alloc.min_size + Math.max(0, minSize - Math.floor(alloc.min_size / 2)); | ||||
|         alloc.natural_size = alloc.natural_size + Math.max(0, naturalSize - Math.floor(alloc.natural_size / 2)); | ||||
|     }, | ||||
|  | ||||
|     _getContentPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let [minSize, naturalSize] = this._iconBox.get_preferred_height(forWidth); | ||||
|         alloc.min_size = minSize; | ||||
|         alloc.natural_size = naturalSize; | ||||
|         [minSize, naturalSize] = this._hbox.get_preferred_height(forWidth); | ||||
|         if (minSize > alloc.min_size) | ||||
|             alloc.min_size = minSize; | ||||
|         if (naturalSize > alloc.natural_size) | ||||
|             alloc.natural_size = naturalSize; | ||||
|     }, | ||||
|  | ||||
|     _contentAllocate: function(actor, box, flags) { | ||||
|         let allocWidth = box.x2 - box.x1; | ||||
|         let allocHeight = box.y2 - box.y1; | ||||
|         let childBox = new Clutter.ActorBox(); | ||||
|  | ||||
|         let [minWidth, minHeight, naturalWidth, naturalHeight] = this._iconBox.get_preferred_size(); | ||||
|  | ||||
|         let direction = this.actor.get_text_direction(); | ||||
|  | ||||
|         let yPadding = Math.floor(Math.max(0, allocHeight - naturalHeight) / 2); | ||||
|         childBox.y1 = yPadding; | ||||
|         childBox.y2 = childBox.y1 + Math.min(naturalHeight, allocHeight); | ||||
|         if (direction == Clutter.TextDirection.LTR) { | ||||
|             childBox.x1 = 0; | ||||
|             childBox.x2 = childBox.x1 + Math.min(naturalWidth, allocWidth); | ||||
|         } else { | ||||
|             childBox.x1 = Math.max(0, allocWidth - naturalWidth); | ||||
|             childBox.x2 = allocWidth; | ||||
|         } | ||||
|         this._iconBox.allocate(childBox, flags); | ||||
|  | ||||
|         let iconWidth = childBox.x2 - childBox.x1; | ||||
|  | ||||
|         [minWidth, naturalWidth] = this._hbox.get_preferred_width(-1); | ||||
|  | ||||
|         childBox.y1 = 0; | ||||
|         childBox.y2 = allocHeight; | ||||
|  | ||||
|         if (direction == Clutter.TextDirection.LTR) { | ||||
|             childBox.x1 = Math.floor(iconWidth / 2); | ||||
|             childBox.x2 = Math.min(childBox.x1 + naturalWidth, allocWidth); | ||||
|         } else { | ||||
|             childBox.x2 = allocWidth - Math.floor(iconWidth / 2); | ||||
|             childBox.x1 = Math.max(0, childBox.x2 - naturalWidth); | ||||
|         } | ||||
|         this._hbox.allocate(childBox, flags); | ||||
|     }, | ||||
|  | ||||
|     _onAppStateChanged: function(appSys, app) { | ||||
|         let state = app.state; | ||||
|         if (state != Shell.AppState.STARTING) { | ||||
| @@ -458,13 +288,18 @@ const AppMenuButton = new Lang.Class({ | ||||
|                 this._targetApp.disconnect(this._actionGroupNotifyId); | ||||
|                 this._actionGroupNotifyId = 0; | ||||
|             } | ||||
|             if (this._busyNotifyId) { | ||||
|                 this._targetApp.disconnect(this._busyNotifyId); | ||||
|                 this._busyNotifyId = 0; | ||||
|             } | ||||
|  | ||||
|             this._targetApp = targetApp; | ||||
|  | ||||
|             if (this._targetApp) { | ||||
|                 this._appMenuNotifyId = this._targetApp.connect('notify::menu', Lang.bind(this, this._sync)); | ||||
|                 this._actionGroupNotifyId = this._targetApp.connect('notify::action-group', Lang.bind(this, this._sync)); | ||||
|                 this._label.setText(this._targetApp.get_name()); | ||||
|                 this._busyNotifyId = this._targetApp.connect('notify::busy', Lang.bind(this, this._sync)); | ||||
|                 this._label.set_text(this._targetApp.get_name()); | ||||
|                 this.actor.set_accessible_name(this._targetApp.get_name()); | ||||
|             } | ||||
|         } | ||||
| @@ -477,7 +312,7 @@ const AppMenuButton = new Lang.Class({ | ||||
|  | ||||
|         let isBusy = (this._targetApp != null && | ||||
|                       (this._targetApp.get_state() == Shell.AppState.STARTING || | ||||
|                        this._targetApp.get_state() == Shell.AppState.BUSY)); | ||||
|                        this._targetApp.get_busy())); | ||||
|         if (isBusy) | ||||
|             this.startAnimation(); | ||||
|         else | ||||
| @@ -572,7 +407,6 @@ const ActivitiesButton = new Lang.Class({ | ||||
|         this.actor.label_actor = this._label; | ||||
|  | ||||
|         this.actor.connect('captured-event', Lang.bind(this, this._onCapturedEvent)); | ||||
|         this.actor.connect_after('button-release-event', Lang.bind(this, this._onButtonRelease)); | ||||
|         this.actor.connect_after('key-release-event', Lang.bind(this, this._onKeyRelease)); | ||||
|  | ||||
|         Main.overview.connect('showing', Lang.bind(this, function() { | ||||
| @@ -601,16 +435,21 @@ const ActivitiesButton = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onCapturedEvent: function(actor, event) { | ||||
|         if (event.type() == Clutter.EventType.BUTTON_PRESS) { | ||||
|         if (event.type() == Clutter.EventType.BUTTON_PRESS || | ||||
|             event.type() == Clutter.EventType.TOUCH_BEGIN) { | ||||
|             if (!Main.overview.shouldToggleByCornerOrButton()) | ||||
|                 return Clutter.EVENT_STOP; | ||||
|         } | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onButtonRelease: function() { | ||||
|         Main.overview.toggle(); | ||||
|         this.menu.close(); | ||||
|     _onEvent: function(actor, event) { | ||||
|         this.parent(actor, event); | ||||
|  | ||||
|         if (event.type() == Clutter.EventType.TOUCH_END || | ||||
|             event.type() == Clutter.EventType.BUTTON_RELEASE) | ||||
|             Main.overview.toggle(); | ||||
|  | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
| @@ -884,7 +723,7 @@ const Panel = new Lang.Class({ | ||||
|  | ||||
|         this.statusArea = {}; | ||||
|  | ||||
|         this.menuManager = new PopupMenu.PopupMenuManager(this, { keybindingMode: Shell.KeyBindingMode.TOPBAR_POPUP }); | ||||
|         this.menuManager = new PopupMenu.PopupMenuManager(this, { actionMode: Shell.ActionMode.TOPBAR_POPUP }); | ||||
|  | ||||
|         this._leftBox = new St.BoxLayout({ name: 'panelLeft' }); | ||||
|         this.actor.add_actor(this._leftBox); | ||||
| @@ -1016,7 +855,7 @@ const Panel = new Lang.Class({ | ||||
|         if (!dragWindow) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         let rect = dragWindow.get_outer_rect(); | ||||
|         let rect = dragWindow.get_frame_rect(); | ||||
|         let [stageX, stageY] = event.get_coords(); | ||||
|  | ||||
|         let allowDrag = dragWindow.maximized_vertically && | ||||
|   | ||||
| @@ -41,8 +41,7 @@ const ButtonBox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _getPreferredWidth: function(actor, forHeight, alloc) { | ||||
|         let children = actor.get_children(); | ||||
|         let child = children.length > 0 ? children[0] : null; | ||||
|         let child = actor.get_first_child(); | ||||
|  | ||||
|         if (child) { | ||||
|             [alloc.min_size, alloc.natural_size] = child.get_preferred_width(-1); | ||||
| @@ -55,8 +54,7 @@ const ButtonBox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _getPreferredHeight: function(actor, forWidth, alloc) { | ||||
|         let children = actor.get_children(); | ||||
|         let child = children.length > 0 ? children[0] : null; | ||||
|         let child = actor.get_first_child(); | ||||
|  | ||||
|         if (child) { | ||||
|             [alloc.min_size, alloc.natural_size] = child.get_preferred_height(-1); | ||||
| @@ -66,13 +64,11 @@ const ButtonBox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _allocate: function(actor, box, flags) { | ||||
|         let children = actor.get_children(); | ||||
|         if (children.length == 0) | ||||
|         let child = actor.get_first_child(); | ||||
|         if (!child) | ||||
|             return; | ||||
|  | ||||
|         let child = children[0]; | ||||
|         let [minWidth, natWidth] = child.get_preferred_width(-1); | ||||
|         let [minHeight, natHeight] = child.get_preferred_height(-1); | ||||
|  | ||||
|         let availWidth = box.x2 - box.x1; | ||||
|         let availHeight = box.y2 - box.y1; | ||||
| @@ -104,8 +100,7 @@ const Button = new Lang.Class({ | ||||
|                       accessible_name: nameText ? nameText : "", | ||||
|                       accessible_role: Atk.Role.MENU }); | ||||
|  | ||||
|         this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress)); | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this._onSourceKeyPress)); | ||||
|         this.actor.connect('event', Lang.bind(this, this._onEvent)); | ||||
|         this.actor.connect('notify::visible', Lang.bind(this, this._onVisibilityChanged)); | ||||
|  | ||||
|         if (dontCreateMenu) | ||||
| @@ -135,32 +130,13 @@ const Button = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _onButtonPress: function(actor, event) { | ||||
|         if (!this.menu) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         this.menu.toggle(); | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onSourceKeyPress: function(actor, event) { | ||||
|         if (!this.menu) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_space || symbol == Clutter.KEY_Return) { | ||||
|     _onEvent: function(actor, event) { | ||||
|         if (this.menu && | ||||
|             (event.type() == Clutter.EventType.TOUCH_BEGIN || | ||||
|              event.type() == Clutter.EventType.BUTTON_PRESS)) | ||||
|             this.menu.toggle(); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (symbol == Clutter.KEY_Escape && this.menu.isOpen) { | ||||
|             this.menu.close(); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } 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 Clutter.EVENT_STOP; | ||||
|         } else | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onVisibilityChanged: function() { | ||||
| @@ -197,7 +173,8 @@ const Button = new Lang.Class({ | ||||
|         // menu is higher then the screen; it's useful if part of the menu is | ||||
|         // scrollable so the minimum height is smaller than the natural height | ||||
|         let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex); | ||||
|         this.menu.actor.style = ('max-height: ' + Math.round(workArea.height) + 'px;'); | ||||
|         let verticalMargins = this.menu.actor.margin_top + this.menu.actor.margin_bottom; | ||||
|         this.menu.actor.style = ('max-height: ' + Math.round(workArea.height - verticalMargins) + 'px;'); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|   | ||||
| @@ -1,9 +1,7 @@ | ||||
| // -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Lang = imports.lang; | ||||
| const Shell = imports.gi.Shell; | ||||
| const Signals = imports.signals; | ||||
| @@ -15,7 +13,6 @@ const GrabHelper = imports.ui.grabHelper; | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
| const Separator = imports.ui.separator; | ||||
| const Slider = imports.ui.slider; | ||||
| const Tweener = imports.ui.tweener; | ||||
|  | ||||
| const Ornament = { | ||||
| @@ -24,17 +21,6 @@ const Ornament = { | ||||
|     CHECK: 2, | ||||
| }; | ||||
|  | ||||
| function _ensureStyle(actor) { | ||||
|     if (actor.get_children) { | ||||
|         let children = actor.get_children(); | ||||
|         for (let i = 0; i < children.length; i++) | ||||
|             _ensureStyle(children[i]); | ||||
|     } | ||||
|  | ||||
|     if (actor instanceof St.Widget) | ||||
|         actor.ensure_style(); | ||||
| } | ||||
|  | ||||
| function isPopupMenuItemVisible(child) { | ||||
|     if (child._delegate instanceof PopupMenuSection) | ||||
|         if (child._delegate.isEmpty()) | ||||
| @@ -46,33 +32,28 @@ function isPopupMenuItemVisible(child) { | ||||
|  * @side Side to which the arrow points. | ||||
|  */ | ||||
| function arrowIcon(side) { | ||||
|     let rotation; | ||||
|     let iconName; | ||||
|     switch (side) { | ||||
|         case St.Side.TOP: | ||||
|             rotation = 180; | ||||
|             iconName = 'pan-up-symbolic'; | ||||
|             break; | ||||
|         case St.Side.RIGHT: | ||||
|             rotation = - 90; | ||||
|             iconName = 'pan-end-symbolic'; | ||||
|             break; | ||||
|         case St.Side.BOTTOM: | ||||
|             rotation = 0; | ||||
|             iconName = 'pan-down-symbolic'; | ||||
|             break; | ||||
|         case St.Side.LEFT: | ||||
|             rotation = 90; | ||||
|             iconName = 'pan-start-symbolic'; | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     let gicon = new Gio.FileIcon({ file: Gio.File.new_for_path(global.datadir + | ||||
|                                              '/theme/menu-arrow-symbolic.svg') }); | ||||
|  | ||||
|     let arrow = new St.Icon({ style_class: 'popup-menu-arrow', | ||||
|                               gicon: gicon, | ||||
|                               icon_name: iconName, | ||||
|                               accessible_role: Atk.Role.ARROW, | ||||
|                               y_expand: true, | ||||
|                               y_align: Clutter.ActorAlign.CENTER }); | ||||
|  | ||||
|     arrow.rotation_angle_z = rotation; | ||||
|  | ||||
|     return arrow; | ||||
| } | ||||
|  | ||||
| @@ -110,7 +91,9 @@ const PopupBaseMenuItem = new Lang.Class({ | ||||
|             this.actor.add_style_class_name(params.style_class); | ||||
|  | ||||
|         if (this._activatable) { | ||||
|             this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPressEvent)); | ||||
|             this.actor.connect('button-release-event', Lang.bind(this, this._onButtonReleaseEvent)); | ||||
|             this.actor.connect('touch-event', Lang.bind(this, this._onTouchEvent)); | ||||
|             this.actor.connect('key-press-event', Lang.bind(this, this._onKeyPressEvent)); | ||||
|         } | ||||
|         if (params.reactive && params.hover) | ||||
| @@ -132,11 +115,30 @@ const PopupBaseMenuItem = new Lang.Class({ | ||||
|         this._parent = parent; | ||||
|     }, | ||||
|  | ||||
|     _onButtonPressEvent: function (actor, event) { | ||||
|         // This is the CSS active state | ||||
|         this.actor.add_style_pseudo_class ('active'); | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onButtonReleaseEvent: function (actor, event) { | ||||
|         this.actor.remove_style_pseudo_class ('active'); | ||||
|         this.activate(event); | ||||
|         return Clutter.EVENT_STOP; | ||||
|     }, | ||||
|  | ||||
|     _onTouchEvent: function (actor, event) { | ||||
|         if (event.type() == Clutter.EventType.TOUCH_END) { | ||||
|             this.actor.remove_style_pseudo_class ('active'); | ||||
|             this.activate(event); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (event.type() == Clutter.EventType.TOUCH_BEGIN) { | ||||
|             // This is the CSS active state | ||||
|             this.actor.add_style_pseudo_class ('active'); | ||||
|         } | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     _onKeyPressEvent: function (actor, event) { | ||||
|         let symbol = event.get_key_symbol(); | ||||
|  | ||||
| @@ -168,10 +170,17 @@ const PopupBaseMenuItem = new Lang.Class({ | ||||
|         if (activeChanged) { | ||||
|             this.active = active; | ||||
|             if (active) { | ||||
|                 this.actor.add_style_pseudo_class('active'); | ||||
|                 this.actor.add_style_class_name('active'); | ||||
|                 this.actor.grab_key_focus(); | ||||
|             } else { | ||||
|                 this.actor.remove_style_pseudo_class('active'); | ||||
|                 this.actor.remove_style_class_name('active'); | ||||
|                 // Remove the CSS active state if the user press the button and | ||||
|                 // while holding moves to another menu item, so we don't paint all items. | ||||
|                 // The correct behaviour would be to set the new item with the CSS | ||||
|                 // active state as well, but button-press-event is not trigered, | ||||
|                 // so we should track it in our own, which would involve some work | ||||
|                 // in the container | ||||
|                 this.actor.remove_style_pseudo_class ('active'); | ||||
|             } | ||||
|             this.emit('active-changed', active); | ||||
|         } | ||||
| @@ -255,8 +264,8 @@ const PopupSeparatorMenuItem = new Lang.Class({ | ||||
|                            Lang.bind(this, this._syncVisibility)); | ||||
|         this._syncVisibility(); | ||||
|  | ||||
|         this._separator = new Separator.HorizontalSeparator({ style_class: 'popup-separator-menu-item' }); | ||||
|         this.actor.add(this._separator.actor, { expand: true }); | ||||
|         this._separator = new St.Widget({ style_class: 'popup-separator-menu-item' }); | ||||
|         this.actor.add(this._separator, { expand: true }); | ||||
|     }, | ||||
|  | ||||
|     _syncVisibility: function() { | ||||
| @@ -383,9 +392,9 @@ const PopupImageMenuItem = new Lang.Class({ | ||||
|         this.parent(params); | ||||
|  | ||||
|         this.label = new St.Label({ text: text }); | ||||
|         this.addActor(this.label); | ||||
|         this.actor.add_child(this.label); | ||||
|         this._icon = new St.Icon({ style_class: 'popup-menu-icon' }); | ||||
|         this.addActor(this._icon, { align: St.Align.END }); | ||||
|         this.actor.add_child(this._icon, { align: St.Align.END }); | ||||
|  | ||||
|         this.setIcon(iconName); | ||||
|     }, | ||||
| @@ -741,6 +750,10 @@ const PopupMenu = new Lang.Class({ | ||||
|         global.focus_manager.add_group(this.actor); | ||||
|         this.actor.reactive = true; | ||||
|  | ||||
|         if (this.sourceActor) | ||||
|             this._keyPressId = this.sourceActor.connect('key-press-event', | ||||
|                                                         Lang.bind(this, this._onKeyPress)); | ||||
|  | ||||
|         this._openedSubMenu = null; | ||||
|     }, | ||||
|  | ||||
| @@ -751,6 +764,40 @@ const PopupMenu = new Lang.Class({ | ||||
|         this._openedSubMenu = submenu; | ||||
|     }, | ||||
|  | ||||
|     _onKeyPress: function(actor, event) { | ||||
|         let navKey; | ||||
|         switch (this._boxPointer.arrowSide) { | ||||
|             case St.Side.TOP: | ||||
|                 navKey = Clutter.KEY_Down; | ||||
|                 break; | ||||
|             case St.Side.BOTTOM: | ||||
|                 navKey = Clutter.KEY_Up; | ||||
|                 break; | ||||
|             case St.Side.LEFT: | ||||
|                 navKey = Clutter.KEY_Right; | ||||
|                 break; | ||||
|             case St.Side.RIGHT: | ||||
|                 navKey = Clutter.KEY_Left; | ||||
|                 break; | ||||
|         } | ||||
|  | ||||
|         let symbol = event.get_key_symbol(); | ||||
|         if (symbol == Clutter.KEY_space || symbol == Clutter.KEY_Return) { | ||||
|             this.toggle(); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (symbol == Clutter.KEY_Escape && this.isOpen) { | ||||
|             this.close(); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (symbol == navKey) { | ||||
|             if (!this.isOpen) | ||||
|                 this.toggle(); | ||||
|             this.actor.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|  | ||||
|     setArrowOrigin: function(origin) { | ||||
|         this._boxPointer.setArrowOrigin(origin); | ||||
|     }, | ||||
| @@ -791,6 +838,12 @@ const PopupMenu = new Lang.Class({ | ||||
|  | ||||
|         this.isOpen = false; | ||||
|         this.emit('open-state-changed', false); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
|         if (this._keyPressId) | ||||
|             this.sourceActor.disconnect(this._keyPressId); | ||||
|         this.parent(); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -884,12 +937,14 @@ const PopupSubMenu = new Lang.Class({ | ||||
|         if (animate && needsScrollbar) | ||||
|             animate = false; | ||||
|  | ||||
|         let targetAngle = this.actor.text_direction == Clutter.TextDirection.RTL ? -90 : 90; | ||||
|  | ||||
|         if (animate) { | ||||
|             let [minHeight, naturalHeight] = this.actor.get_preferred_height(-1); | ||||
|             this.actor.height = 0; | ||||
|             this.actor._arrowRotation = this._arrow.rotation_angle_z; | ||||
|             Tweener.addTween(this.actor, | ||||
|                              { _arrowRotation: this.actor._arrowRotation + 90, | ||||
|                              { _arrowRotation: targetAngle, | ||||
|                                height: naturalHeight, | ||||
|                                time: 0.25, | ||||
|                                onUpdateScope: this, | ||||
| @@ -902,7 +957,7 @@ const PopupSubMenu = new Lang.Class({ | ||||
|                                } | ||||
|                              }); | ||||
|         } else { | ||||
|             this._arrow.rotation_angle_z = this.actor._arrowRotation + 90; | ||||
|             this._arrow.rotation_angle_z = targetAngle; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
| @@ -922,7 +977,7 @@ const PopupSubMenu = new Lang.Class({ | ||||
|         if (animate) { | ||||
|             this.actor._arrowRotation = this._arrow.rotation_angle_z; | ||||
|             Tweener.addTween(this.actor, | ||||
|                              { _arrowRotation: this.actor._arrowRotation - 90, | ||||
|                              { _arrowRotation: 0, | ||||
|                                height: 0, | ||||
|                                time: 0.25, | ||||
|                                onUpdateScope: this, | ||||
| @@ -936,7 +991,7 @@ const PopupSubMenu = new Lang.Class({ | ||||
|                                }, | ||||
|                              }); | ||||
|         } else { | ||||
|             this._arrow.rotation_angle_z = this.actor._arrowRotation - 90; | ||||
|             this._arrow.rotation_angle_z = 0; | ||||
|             this.actor.hide(); | ||||
|         } | ||||
|     }, | ||||
| @@ -1014,8 +1069,6 @@ const PopupSubMenuMenuItem = new Lang.Class({ | ||||
|         this._triangleBin = new St.Widget({ y_expand: true, | ||||
|                                             y_align: Clutter.ActorAlign.CENTER }); | ||||
|         this._triangleBin.add_child(this._triangle); | ||||
|         if (this._triangleBin.get_text_direction() == Clutter.TextDirection.RTL) | ||||
|             this._triangleBin.set_scale(-1.0, 1.0); | ||||
|  | ||||
|         this.actor.add_child(this._triangleBin); | ||||
|         this.actor.add_accessible_state (Atk.StateType.EXPANDABLE); | ||||
| @@ -1041,10 +1094,12 @@ const PopupSubMenuMenuItem = new Lang.Class({ | ||||
|             this.actor.add_style_pseudo_class('open'); | ||||
|             this._getTopMenu()._setOpenedSubMenu(this.menu); | ||||
|             this.actor.add_accessible_state (Atk.StateType.EXPANDED); | ||||
|             this.actor.add_style_pseudo_class('checked'); | ||||
|         } else { | ||||
|             this.actor.remove_style_pseudo_class('open'); | ||||
|             this._getTopMenu()._setOpenedSubMenu(null); | ||||
|             this.actor.remove_accessible_state (Atk.StateType.EXPANDED); | ||||
|             this.actor.remove_style_pseudo_class('checked'); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
| @@ -1089,6 +1144,9 @@ const PopupSubMenuMenuItem = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _onButtonReleaseEvent: function(actor) { | ||||
|         // Since we override the parent, we need to manage what the parent does | ||||
|         // with the active style class | ||||
|         this.actor.remove_style_pseudo_class ('active'); | ||||
|         this._setOpenState(!this._getOpenState()); | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     } | ||||
|   | ||||
| @@ -110,6 +110,13 @@ function loadRemoteSearchProviders(callback) { | ||||
|             else | ||||
|                 remoteProvider = new RemoteSearchProvider(appInfo, busName, objectPath); | ||||
|  | ||||
|             remoteProvider.defaultEnabled = true; | ||||
|             try { | ||||
|                 remoteProvider.defaultEnabled = !keyfile.get_boolean(group, 'DefaultDisabled'); | ||||
|             } catch(e) { | ||||
|                 // ignore error | ||||
|             } | ||||
|  | ||||
|             objectPaths[objectPath] = remoteProvider; | ||||
|             loadedProviders.push(remoteProvider); | ||||
|         } catch(e) { | ||||
| @@ -117,7 +124,7 @@ function loadRemoteSearchProviders(callback) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let searchSettings = new Gio.Settings({ schema: Search.SEARCH_PROVIDERS_SCHEMA }); | ||||
|     let searchSettings = new Gio.Settings({ schema_id: Search.SEARCH_PROVIDERS_SCHEMA }); | ||||
|     if (searchSettings.get_boolean('disable-external')) { | ||||
|         callback([]); | ||||
|         return; | ||||
| @@ -132,8 +139,14 @@ function loadRemoteSearchProviders(callback) { | ||||
|  | ||||
|     loadedProviders = loadedProviders.filter(function(provider) { | ||||
|         let appId = provider.appInfo.get_id(); | ||||
|         let disabled = searchSettings.get_strv('disabled'); | ||||
|         return disabled.indexOf(appId) == -1; | ||||
|  | ||||
|         if (provider.defaultEnabled) { | ||||
|             let disabled = searchSettings.get_strv('disabled'); | ||||
|             return disabled.indexOf(appId) == -1; | ||||
|         } else { | ||||
|             let enabled = searchSettings.get_strv('enabled'); | ||||
|             return enabled.indexOf(appId) != -1; | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     loadedProviders.sort(function(providerA, providerB) { | ||||
|   | ||||
| @@ -38,8 +38,8 @@ const RunDialog = new Lang.Class({ | ||||
|         this.parent({ styleClass: 'run-dialog', | ||||
|                       destroyOnClose: false }); | ||||
|  | ||||
|         this._lockdownSettings = new Gio.Settings({ schema: LOCKDOWN_SCHEMA }); | ||||
|         this._terminalSettings = new Gio.Settings({ schema: TERMINAL_SCHEMA }); | ||||
|         this._lockdownSettings = new Gio.Settings({ schema_id: LOCKDOWN_SCHEMA }); | ||||
|         this._terminalSettings = new Gio.Settings({ schema_id: TERMINAL_SCHEMA }); | ||||
|         global.settings.connect('changed::development-tools', Lang.bind(this, function () { | ||||
|             this._enableInternalCommands = global.settings.get_boolean('development-tools'); | ||||
|         })); | ||||
| @@ -50,14 +50,10 @@ const RunDialog = new Lang.Class({ | ||||
|                                        Main.createLookingGlass().open(); | ||||
|                                    }), | ||||
|  | ||||
|                                    'r': Lang.bind(this, function() { | ||||
|                                        global.reexec_self(); | ||||
|                                    }), | ||||
|                                    'r': Lang.bind(this, this._restart), | ||||
|  | ||||
|                                    // Developer brain backwards compatibility | ||||
|                                    'restart': Lang.bind(this, function() { | ||||
|                                        global.reexec_self(); | ||||
|                                    }), | ||||
|                                    'restart': Lang.bind(this, this._restart), | ||||
|  | ||||
|                                    'debugexit': Lang.bind(this, function() { | ||||
|                                        Meta.quit(Meta.ExitCode.ERROR); | ||||
| @@ -186,6 +182,10 @@ const RunDialog = new Lang.Class({ | ||||
|         let results = someResults.reduce(function(a, b) { | ||||
|             return a.concat(b); | ||||
|         }, []); | ||||
|  | ||||
|         if (!results.length) | ||||
|             return null; | ||||
|  | ||||
|         let common = results.reduce(_getCommon, null); | ||||
|         return common.substr(text.length); | ||||
|     }, | ||||
| @@ -271,6 +271,12 @@ const RunDialog = new Lang.Class({ | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _restart: function() { | ||||
|         this._shouldFadeOut = false; | ||||
|         this.close(); | ||||
|         Meta.restart(_("Restarting…")); | ||||
|     }, | ||||
|  | ||||
|     open: function() { | ||||
|         this._history.lastItem(); | ||||
|         this._errorBox.hide(); | ||||
|   | ||||
| @@ -85,7 +85,8 @@ const Clock = new Lang.Class({ | ||||
|         let date = new Date(); | ||||
|         /* Translators: This is a time format for a date in | ||||
|            long format */ | ||||
|         this._date.text = date.toLocaleFormat(_("%A, %B %d")); | ||||
|         let dateFormat = Shell.util_translate_time_string(N_("%A, %B %d")); | ||||
|         this._date.text = date.toLocaleFormat(dateFormat); | ||||
|     }, | ||||
|  | ||||
|     destroy: function() { | ||||
| @@ -100,18 +101,14 @@ const NotificationsBox = new Lang.Class({ | ||||
|     _init: function() { | ||||
|         this.actor = new St.BoxLayout({ vertical: true, | ||||
|                                         name: 'screenShieldNotifications', | ||||
|                                         style_class: 'screen-shield-notifications-box' }); | ||||
|  | ||||
|         this._musicBin = new St.Bin({ style_class: 'screen-shield-notifications-box', | ||||
|                                       visible: false }); | ||||
|                                         style_class: 'screen-shield-notifications-container' }); | ||||
|  | ||||
|         this._scrollView = new St.ScrollView({ x_fill: false, x_align: St.Align.START, | ||||
|                                                hscrollbar_policy: Gtk.PolicyType.NEVER }); | ||||
|         this._notificationBox = new St.BoxLayout({ vertical: true, | ||||
|                                                    style_class: 'screen-shield-notifications-box' }); | ||||
|                                                    style_class: 'screen-shield-notifications-container' }); | ||||
|         this._scrollView.add_actor(this._notificationBox); | ||||
|  | ||||
|         this.actor.add(this._musicBin); | ||||
|         this.actor.add(this._scrollView, { x_fill: true, x_align: St.Align.START }); | ||||
|  | ||||
|         this._sources = new Map(); | ||||
| @@ -138,12 +135,11 @@ const NotificationsBox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _updateVisibility: function() { | ||||
|         this._musicBin.visible = this._musicBin.child != null && this._musicBin.child.visible; | ||||
|         this._notificationBox.visible = this._notificationBox.get_children().some(function(a) { | ||||
|             return a.visible; | ||||
|         }); | ||||
|  | ||||
|         this.actor.visible = this._musicBin.visible || this._notificationBox.visible; | ||||
|         this.actor.visible = this._notificationBox.visible; | ||||
|     }, | ||||
|  | ||||
|     _makeNotificationCountText: function(count, isChat) { | ||||
| @@ -191,7 +187,7 @@ const NotificationsBox = new Lang.Class({ | ||||
|         for (let i = 0; i < source.notifications.length; i++) { | ||||
|             let n = source.notifications[i]; | ||||
|  | ||||
|             if (n.acknowledged || n.isMusic) | ||||
|             if (n.acknowledged) | ||||
|                 continue; | ||||
|  | ||||
|             let body = ''; | ||||
| @@ -212,29 +208,13 @@ const NotificationsBox = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _showSource: function(source, obj, box) { | ||||
|         let musicNotification = source.getMusicNotification(); | ||||
|  | ||||
|         if (musicNotification != null && | ||||
|             this._musicBin.child == null) { | ||||
|             musicNotification.acknowledged = true; | ||||
|             if (musicNotification.actor.get_parent() != null) | ||||
|                 musicNotification.actor.get_parent().remove_actor(musicNotification.actor); | ||||
|             this._musicBin.child = musicNotification.actor; | ||||
|             this._musicBin.child.visible = obj.visible; | ||||
|  | ||||
|             musicNotification.expand(false /* animate */); | ||||
|  | ||||
|             obj.musicNotification = musicNotification; | ||||
|         } | ||||
|  | ||||
|         if (obj.detailed) { | ||||
|             [obj.titleLabel, obj.countLabel] = this._makeNotificationDetailedSource(source, box); | ||||
|         } else { | ||||
|             [obj.titleLabel, obj.countLabel] = this._makeNotificationSource(source, box); | ||||
|         } | ||||
|  | ||||
|         box.visible = obj.visible && | ||||
|             (source.unseenCount > (musicNotification ? 1 : 0)); | ||||
|         box.visible = obj.visible && (source.unseenCount > 0); | ||||
|     }, | ||||
|  | ||||
|     _sourceAdded: function(tray, source, initial) { | ||||
| @@ -245,23 +225,16 @@ const NotificationsBox = new Lang.Class({ | ||||
|             sourceCountChangedId: 0, | ||||
|             sourceTitleChangedId: 0, | ||||
|             sourceUpdatedId: 0, | ||||
|             sourceNotifyId: 0, | ||||
|             musicNotification: null, | ||||
|             sourceBox: null, | ||||
|             titleLabel: null, | ||||
|             countLabel: null, | ||||
|         }; | ||||
|  | ||||
|         obj.sourceBox = new St.BoxLayout({ style_class: 'screen-shield-notification-source' }); | ||||
|         obj.sourceBox = new St.BoxLayout({ style_class: 'screen-shield-notification-source', | ||||
|                                            x_expand: true }); | ||||
|         this._showSource(source, obj, obj.sourceBox); | ||||
|         this._notificationBox.add(obj.sourceBox, { x_fill: false, x_align: St.Align.START }); | ||||
|  | ||||
|         if (obj.musicNotification) { | ||||
|             obj.sourceNotifyId = source.connect('notify', Lang.bind(this, function(source, notification) { | ||||
|                 notification.acknowledged = true; | ||||
|             })); | ||||
|         } | ||||
|  | ||||
|         obj.sourceCountChangedId = source.connect('count-updated', Lang.bind(this, function(source) { | ||||
|             this._countChanged(source, obj); | ||||
|         })); | ||||
| @@ -322,8 +295,7 @@ const NotificationsBox = new Lang.Class({ | ||||
|             obj.countLabel.text = this._makeNotificationCountText(count, source.isChat); | ||||
|         } | ||||
|  | ||||
|         obj.sourceBox.visible = obj.visible && | ||||
|             (source.unseenCount > (obj.musicNotification ? 1 : 0)); | ||||
|         obj.sourceBox.visible = obj.visible && (source.unseenCount > 0); | ||||
|  | ||||
|         this._updateVisibility(); | ||||
|         if (obj.sourceBox.visible) | ||||
| @@ -335,10 +307,7 @@ const NotificationsBox = new Lang.Class({ | ||||
|             return; | ||||
|  | ||||
|         obj.visible = source.policy.showInLockScreen; | ||||
|         if (obj.musicNotification) | ||||
|             obj.musicNotification.actor.visible = obj.visible; | ||||
|         obj.sourceBox.visible = obj.visible && | ||||
|             source.unseenCount > (obj.musicNotification ? 1 : 0); | ||||
|         obj.sourceBox.visible = obj.visible && source.unseenCount > 0; | ||||
|  | ||||
|         this._updateVisibility(); | ||||
|         if (obj.sourceBox.visible) | ||||
| @@ -365,13 +334,6 @@ const NotificationsBox = new Lang.Class({ | ||||
|         obj.sourceBox.destroy(); | ||||
|         obj.sourceBox = obj.titleLabel = obj.countLabel = null; | ||||
|  | ||||
|         if (obj.musicNotification) { | ||||
|             this._musicBin.child = null; | ||||
|             obj.musicNotification = null; | ||||
|  | ||||
|             source.disconnect(obj.sourceNotifyId); | ||||
|         } | ||||
|  | ||||
|         source.disconnect(obj.sourceDestroyId); | ||||
|         source.disconnect(obj.sourceCountChangedId); | ||||
|         source.disconnect(obj.sourceTitleChangedId); | ||||
| @@ -444,14 +406,12 @@ function clamp(value, min, max) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * To test screen shield, make sure to kill gnome-screensaver. | ||||
|  * | ||||
|  * If you are setting org.gnome.desktop.session.idle-delay directly in dconf, | ||||
|  * rather than through System Settings, you also need to set | ||||
|  * org.gnome.settings-daemon.plugins.power.sleep-display-ac and | ||||
|  * org.gnome.settings-daemon.plugins.power.sleep-display-battery to the same value. | ||||
|  * This will ensure that the screen blanks at the right time when it fades out. | ||||
|  * https://bugzilla.gnome.org/show_bug.cgi?id=668703 explains the dependance. | ||||
|  * https://bugzilla.gnome.org/show_bug.cgi?id=668703 explains the dependency. | ||||
|  */ | ||||
| const ScreenShield = new Lang.Class({ | ||||
|     Name: 'ScreenShield', | ||||
| @@ -563,7 +523,7 @@ const ScreenShield = new Lang.Class({ | ||||
|                 this._loginSession.connectSignal('Unlock', Lang.bind(this, function() { this.deactivate(false); })); | ||||
|             })); | ||||
|  | ||||
|         this._settings = new Gio.Settings({ schema: SCREENSAVER_SCHEMA }); | ||||
|         this._settings = new Gio.Settings({ schema_id: SCREENSAVER_SCHEMA }); | ||||
|  | ||||
|         this._isModal = false; | ||||
|         this._hasLockScreen = false; | ||||
| @@ -648,14 +608,14 @@ const ScreenShield = new Lang.Class({ | ||||
|         if (this._isModal) | ||||
|             return true; | ||||
|  | ||||
|         this._isModal = Main.pushModal(this.actor, { keybindingMode: Shell.KeyBindingMode.LOCK_SCREEN }); | ||||
|         this._isModal = Main.pushModal(this.actor, { actionMode: Shell.ActionMode.LOCK_SCREEN }); | ||||
|         if (this._isModal) | ||||
|             return true; | ||||
|  | ||||
|         // We failed to get a pointer grab, it means that | ||||
|         // something else has it. Try with a keyboard grab only | ||||
|         this._isModal = Main.pushModal(this.actor, { options: Meta.ModalOptions.POINTER_ALREADY_GRABBED, | ||||
|                                                      keybindingMode: Shell.KeyBindingMode.LOCK_SCREEN }); | ||||
|                                                      actionMode: Shell.ActionMode.LOCK_SCREEN }); | ||||
|         return this._isModal; | ||||
|     }, | ||||
|  | ||||
| @@ -671,7 +631,9 @@ const ScreenShield = new Lang.Class({ | ||||
|         if (this._lockScreenState != MessageTray.State.SHOWN) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         let isEnter = (symbol == Clutter.KEY_Return || symbol == Clutter.KEY_KP_Enter); | ||||
|         let isEnter = (symbol == Clutter.KEY_Return || | ||||
|                        symbol == Clutter.KEY_KP_Enter || | ||||
|                        symbol == Clutter.KEY_ISO_Enter); | ||||
|         if (!isEnter && !(GLib.unichar_isprint(unichar) || symbol == Clutter.KEY_Escape)) | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
| @@ -689,10 +651,10 @@ const ScreenShield = new Lang.Class({ | ||||
|             return Clutter.EVENT_PROPAGATE; | ||||
|  | ||||
|         let delta = 0; | ||||
|         if (event.get_scroll_direction() == Clutter.ScrollDirection.UP) | ||||
|         if (event.get_scroll_direction() == Clutter.ScrollDirection.SMOOTH) | ||||
|             delta = Math.abs(event.get_scroll_delta()[0]); | ||||
|         else | ||||
|             delta = 5; | ||||
|         else if (event.get_scroll_direction() == Clutter.ScrollDirection.SMOOTH) | ||||
|             delta = Math.max(0, event.get_scroll_delta()[0]); | ||||
|  | ||||
|         this._lockScreenScrollCounter += delta; | ||||
|  | ||||
| @@ -902,17 +864,11 @@ const ScreenShield = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     showDialog: function() { | ||||
|         // Ensure that the stage window is mapped, before taking a grab | ||||
|         // otherwise X errors out | ||||
|         Meta.later_add(Meta.LaterType.BEFORE_REDRAW, Lang.bind(this, function() { | ||||
|             if (!this._becomeModal()) { | ||||
|                 // In the login screen, this is a hard error. Fail-whale | ||||
|                 log('Could not acquire modal grab for the login screen. Aborting login process.'); | ||||
|                 Meta.quit(Meta.ExitCode.ERROR); | ||||
|             } | ||||
|  | ||||
|             return false; | ||||
|         })); | ||||
|         if (!this._becomeModal()) { | ||||
|             // In the login screen, this is a hard error. Fail-whale | ||||
|             log('Could not acquire modal grab for the login screen. Aborting login process.'); | ||||
|             Meta.quit(Meta.ExitCode.ERROR); | ||||
|         } | ||||
|  | ||||
|         this.actor.show(); | ||||
|         this._isGreeter = Main.sessionMode.isGreeter; | ||||
| @@ -927,6 +883,11 @@ const ScreenShield = new Lang.Class({ | ||||
|  | ||||
|         this._lockScreenState = MessageTray.State.HIDDEN; | ||||
|         this._lockScreenGroup.hide(); | ||||
|  | ||||
|         if (this._dialog) { | ||||
|             this._dialog.actor.grab_key_focus(); | ||||
|             this._dialog.actor.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _hideLockScreen: function(animate, velocity) { | ||||
| @@ -1335,6 +1296,8 @@ const ScreenShield = new Lang.Class({ | ||||
|  | ||||
|     // If the previous shell crashed, and gnome-session restarted us, then re-lock | ||||
|     lockIfWasLocked: function() { | ||||
|         if (!this._settings.get_boolean(LOCK_ENABLED_KEY)) | ||||
|             return; | ||||
|         let wasLocked = global.get_runtime_state('b', LOCKED_STATE_STR); | ||||
|         if (wasLocked === null) | ||||
|             return; | ||||
|   | ||||
| @@ -43,6 +43,8 @@ const ScreencastService = new Lang.Class({ | ||||
|  | ||||
|         this._recorders = new Map(); | ||||
|  | ||||
|         this._lockdownSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.lockdown' }); | ||||
|  | ||||
|         Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated)); | ||||
|     }, | ||||
|  | ||||
| @@ -68,8 +70,8 @@ const ScreencastService = new Lang.Class({ | ||||
|         if (Main.sessionMode.allowScreencast) | ||||
|             return; | ||||
|  | ||||
|         this._recorders.clear(); | ||||
|         this.emit('updated'); | ||||
|         for (let sender of this._recorders.keys()) | ||||
|             this._stopRecordingForSender(sender); | ||||
|     }, | ||||
|  | ||||
|     _onNameVanished: function(connection, name) { | ||||
| @@ -97,13 +99,14 @@ const ScreencastService = new Lang.Class({ | ||||
|             recorder.set_pipeline(options['pipeline']); | ||||
|         if (options['framerate']) | ||||
|             recorder.set_framerate(options['framerate']); | ||||
|         if (options['draw-cursor']) | ||||
|         if ('draw-cursor' in options) | ||||
|             recorder.set_draw_cursor(options['draw-cursor']); | ||||
|     }, | ||||
|  | ||||
|     ScreencastAsync: function(params, invocation) { | ||||
|         let returnValue = [false, '']; | ||||
|         if (!Main.sessionMode.allowScreencast) { | ||||
|         if (!Main.sessionMode.allowScreencast || | ||||
|             this._lockdownSettings.get_boolean('disable-save-to-disk')) { | ||||
|             invocation.return_value(GLib.Variant.new('(bs)', returnValue)); | ||||
|             return; | ||||
|         } | ||||
| @@ -117,6 +120,8 @@ const ScreencastService = new Lang.Class({ | ||||
|             this._applyOptionalParameters(recorder, options); | ||||
|             let [success, fileName] = recorder.record(); | ||||
|             returnValue = [success, fileName ? fileName : '']; | ||||
|             if (!success) | ||||
|                 this._stopRecordingForSender(sender); | ||||
|         } | ||||
|  | ||||
|         invocation.return_value(GLib.Variant.new('(bs)', returnValue)); | ||||
| @@ -124,7 +129,8 @@ const ScreencastService = new Lang.Class({ | ||||
|  | ||||
|     ScreencastAreaAsync: function(params, invocation) { | ||||
|         let returnValue = [false, '']; | ||||
|         if (!Main.sessionMode.allowScreencast) { | ||||
|         if (!Main.sessionMode.allowScreencast || | ||||
|             this._lockdownSettings.get_boolean('disable-save-to-disk')) { | ||||
|             invocation.return_value(GLib.Variant.new('(bs)', returnValue)); | ||||
|             return; | ||||
|         } | ||||
| @@ -150,6 +156,8 @@ const ScreencastService = new Lang.Class({ | ||||
|             this._applyOptionalParameters(recorder, options); | ||||
|             let [success, fileName] = recorder.record(); | ||||
|             returnValue = [success, fileName ? fileName : '']; | ||||
|             if (!success) | ||||
|                 this._stopRecordingForSender(sender); | ||||
|         } | ||||
|  | ||||
|         invocation.return_value(GLib.Variant.new('(bs)', returnValue)); | ||||
|   | ||||
| @@ -65,29 +65,97 @@ const ScreenshotService = new Lang.Class({ | ||||
|         this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(ScreenshotIface, this); | ||||
|         this._dbusImpl.export(Gio.DBus.session, '/org/gnome/Shell/Screenshot'); | ||||
|  | ||||
|         this._screenShooter = new Map(); | ||||
|  | ||||
|         this._lockdownSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.lockdown' }); | ||||
|  | ||||
|         Gio.DBus.session.own_name('org.gnome.Shell.Screenshot', Gio.BusNameOwnerFlags.REPLACE, null, null); | ||||
|     }, | ||||
|  | ||||
|     _createScreenshot: function(invocation) { | ||||
|         let sender = invocation.get_sender(); | ||||
|         if (this._screenShooter.has(sender) || | ||||
|             this._lockdownSettings.get_boolean('disable-save-to-disk')) { | ||||
|             invocation.return_value(GLib.Variant.new('(bs)', [false, ''])); | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         let shooter = new Shell.Screenshot(); | ||||
|         shooter._watchNameId = | ||||
|                         Gio.bus_watch_name(Gio.BusType.SESSION, sender, 0, null, | ||||
|                                            Lang.bind(this, this._onNameVanished)); | ||||
|  | ||||
|         this._screenShooter.set(sender, shooter); | ||||
|  | ||||
|         return shooter; | ||||
|     }, | ||||
|  | ||||
|     _onNameVanished: function(connection, name) { | ||||
|         this._removeShooterForSender(name); | ||||
|     }, | ||||
|  | ||||
|     _removeShooterForSender: function(sender) { | ||||
|         let shooter = this._screenShooter.get(sender); | ||||
|         if (!shooter) | ||||
|             return; | ||||
|  | ||||
|         Gio.bus_unwatch_name(shooter._watchNameId); | ||||
|         this._screenShooter.delete(sender); | ||||
|     }, | ||||
|  | ||||
|     _checkArea: function(x, y, width, height) { | ||||
|         return x >= 0 && y >= 0 && | ||||
|                width > 0 && height > 0 && | ||||
|                x + width <= global.screen_width && | ||||
|                y + height <= global.screen_height; | ||||
|     }, | ||||
|  | ||||
|     _onScreenshotComplete: function(obj, result, area, filenameUsed, flash, invocation) { | ||||
|         if (flash && result) { | ||||
|             let flashspot = new Flashspot(area); | ||||
|             flashspot.fire(); | ||||
|         if (result) { | ||||
|             if (flash) { | ||||
|                 let flashspot = new Flashspot(area); | ||||
|                 flashspot.fire(Lang.bind(this, function() { | ||||
|                     this._removeShooterForSender(invocation.get_sender()); | ||||
|                 })); | ||||
|             } | ||||
|             else | ||||
|                 this._removeShooterForSender(invocation.get_sender()); | ||||
|         } | ||||
|  | ||||
|         let retval = GLib.Variant.new('(bs)', [result, filenameUsed]); | ||||
|         invocation.return_value(retval); | ||||
|     }, | ||||
|  | ||||
|     _scaleArea: function(x, y, width, height) { | ||||
|         let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|         x *= scaleFactor; | ||||
|         y *= scaleFactor; | ||||
|         width *= scaleFactor; | ||||
|         height *= scaleFactor; | ||||
|         return [x, y, width, height]; | ||||
|     }, | ||||
|  | ||||
|     _unscaleArea: function(x, y, width, height) { | ||||
|         let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; | ||||
|         x /= scaleFactor; | ||||
|         y /= scaleFactor; | ||||
|         width /= scaleFactor; | ||||
|         height /= scaleFactor; | ||||
|         return [x, y, width, height]; | ||||
|     }, | ||||
|  | ||||
|     ScreenshotAreaAsync : function (params, invocation) { | ||||
|         let [x, y, width, height, flash, filename, callback] = params; | ||||
|         if (x < 0 || y < 0 || | ||||
|             width <= 0 || height <= 0 || | ||||
|             x + width > global.screen_width || y + height > global.screen_height) { | ||||
|             invocation.return_error_literal(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED, | ||||
|                         "Invalid params"); | ||||
|         [x, y, width, height] = this._scaleArea(x, y, width, height); | ||||
|         if (!this._checkArea(x, y, width, height)) { | ||||
|             invocation.return_error_literal(Gio.IOErrorEnum, | ||||
|                                             Gio.IOErrorEnum.CANCELLED, | ||||
|                                             "Invalid params"); | ||||
|             return; | ||||
|         } | ||||
|         let screenshot = new Shell.Screenshot(); | ||||
|         let screenshot = this._createScreenshot(invocation); | ||||
|         if (!screenshot) | ||||
|             return; | ||||
|         screenshot.screenshot_area (x, y, width, height, filename, | ||||
|                                 Lang.bind(this, this._onScreenshotComplete, | ||||
|                                           flash, invocation)); | ||||
| @@ -95,7 +163,9 @@ const ScreenshotService = new Lang.Class({ | ||||
|  | ||||
|     ScreenshotWindowAsync : function (params, invocation) { | ||||
|         let [include_frame, include_cursor, flash, filename] = params; | ||||
|         let screenshot = new Shell.Screenshot(); | ||||
|         let screenshot = this._createScreenshot(invocation); | ||||
|         if (!screenshot) | ||||
|             return; | ||||
|         screenshot.screenshot_window (include_frame, include_cursor, filename, | ||||
|                                   Lang.bind(this, this._onScreenshotComplete, | ||||
|                                             flash, invocation)); | ||||
| @@ -103,7 +173,9 @@ const ScreenshotService = new Lang.Class({ | ||||
|  | ||||
|     ScreenshotAsync : function (params, invocation) { | ||||
|         let [include_cursor, flash, filename] = params; | ||||
|         let screenshot = new Shell.Screenshot(); | ||||
|         let screenshot = this._createScreenshot(invocation); | ||||
|         if (!screenshot) | ||||
|             return; | ||||
|         screenshot.screenshot(include_cursor, filename, | ||||
|                           Lang.bind(this, this._onScreenshotComplete, | ||||
|                                     flash, invocation)); | ||||
| @@ -115,9 +187,9 @@ const ScreenshotService = new Lang.Class({ | ||||
|         selectArea.connect('finished', Lang.bind(this, | ||||
|             function(selectArea, areaRectangle) { | ||||
|                 if (areaRectangle) { | ||||
|                     let retval = GLib.Variant.new('(iiii)', | ||||
|                         [areaRectangle.x, areaRectangle.y, | ||||
|                          areaRectangle.width, areaRectangle.height]); | ||||
|                     let retRectangle = this._unscaleArea(areaRectangle.x, areaRectangle.y, | ||||
|                         areaRectangle.width, areaRectangle.height); | ||||
|                     let retval = GLib.Variant.new('(iiii)', retRectangle); | ||||
|                     invocation.return_value(retval); | ||||
|                 } else { | ||||
|                     invocation.return_error_literal(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED, | ||||
| @@ -126,9 +198,18 @@ const ScreenshotService = new Lang.Class({ | ||||
|             })); | ||||
|     }, | ||||
|  | ||||
|     FlashArea: function(x, y, width, height) { | ||||
|     FlashAreaAsync: function(params, invocation) { | ||||
|         let [x, y, width, height] = params; | ||||
|         [x, y, width, height] = this._scaleArea(x, y, width, height); | ||||
|         if (!this._checkArea(x, y, width, height)) { | ||||
|             invocation.return_error_literal(Gio.IOErrorEnum, | ||||
|                                             Gio.IOErrorEnum.CANCELLED, | ||||
|                                             "Invalid params"); | ||||
|             return; | ||||
|         } | ||||
|         let flashspot = new Flashspot({ x : x, y : y, width: width, height: height}); | ||||
|         flashspot.fire(); | ||||
|         invocation.return_value(null); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -268,7 +349,7 @@ const Flashspot = new Lang.Class({ | ||||
|         this.actor.set_position(area.x, area.y); | ||||
|     }, | ||||
|  | ||||
|     fire: function() { | ||||
|     fire: function(doneCallback) { | ||||
|         this.actor.show(); | ||||
|         this.actor.opacity = 255; | ||||
|         Tweener.addTween(this.actor, | ||||
| @@ -276,6 +357,8 @@ const Flashspot = new Lang.Class({ | ||||
|                            time: FLASHSPOT_ANIMATION_OUT_TIME, | ||||
|                            transition: 'easeOutQuad', | ||||
|                            onComplete: Lang.bind(this, function() { | ||||
|                                if (doneCallback) | ||||
|                                    doneCallback(); | ||||
|                                this.destroy(); | ||||
|                            }) | ||||
|                          }); | ||||
|   | ||||
| @@ -7,6 +7,7 @@ const Meta = imports.gi.Meta; | ||||
| const Shell = imports.gi.Shell; | ||||
|  | ||||
| const Main = imports.ui.main; | ||||
| const Params = imports.misc.params; | ||||
|  | ||||
| // This module provides functionality for driving the shell user interface | ||||
| // in an automated fashion. The primary current use case for this is | ||||
| @@ -78,6 +79,7 @@ const PerfHelperIface = '<node> \ | ||||
|     <arg type="i" direction="in" /> \ | ||||
|     <arg type="b" direction="in" /> \ | ||||
|     <arg type="b" direction="in" /> \ | ||||
|     <arg type="b" direction="in" /> \ | ||||
| </method> \ | ||||
| <method name="WaitWindows" /> \ | ||||
| <method name="DestroyWindows" /> \ | ||||
| @@ -97,11 +99,36 @@ function _getPerfHelper() { | ||||
|     return _perfHelper; | ||||
| } | ||||
|  | ||||
| function _callRemote(obj, method, ...args) { | ||||
|     let cb; | ||||
|     let errcb; | ||||
|  | ||||
|     args.push(function(result, excp) { | ||||
|                   if (excp) { | ||||
|                       if (errcb) | ||||
|                           errcb(excp); | ||||
|                   } else { | ||||
|                       if (cb) | ||||
|                           cb(); | ||||
|                   } | ||||
|              }); | ||||
|  | ||||
|     method.apply(obj, args); | ||||
|  | ||||
|     return function(callback, error_callback) { | ||||
|         cb = callback; | ||||
|         errcb = error_callback; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * createTestWindow: | ||||
|  * @width: width of window, in pixels | ||||
|  * @height: height of window, in pixels | ||||
|  * @alpha: whether the window should be alpha transparent | ||||
|  * @params: options for window creation. | ||||
|  *   width - width of window, in pixels (default 640) | ||||
|  *   height - height of window, in pixels (default 480) | ||||
|  *   alpha - whether the window should have an alpha channel (default false) | ||||
|  *   maximized - whether the window should be created maximized (default false) | ||||
|  *   redraws - whether the window should continually redraw itself (default false) | ||||
|  * @maximized: whethe the window should be created maximized | ||||
|  * | ||||
|  * Creates a window using gnome-shell-perf-helper for testing purposes. | ||||
| @@ -110,19 +137,17 @@ function _getPerfHelper() { | ||||
|  * because of the normal X asynchronous mapping process, to actually wait | ||||
|  * until the window has been mapped and exposed, use waitTestWindows(). | ||||
|  */ | ||||
| function createTestWindow(width, height, alpha, maximized) { | ||||
|     let cb; | ||||
| function createTestWindow(width, height, params) { | ||||
|     params = Params.parse(params, { width: 640, | ||||
|                                     height: 480, | ||||
|                                     alpha: false, | ||||
|                                     maximized: false, | ||||
|                                     redraws: false }); | ||||
|  | ||||
|     let perfHelper = _getPerfHelper(); | ||||
|  | ||||
|     perfHelper.CreateWindowRemote(width, height, alpha, maximized, | ||||
|                                   function(result, excp) { | ||||
|                                       if (cb) | ||||
|                                           cb(); | ||||
|                                   }); | ||||
|  | ||||
|     return function(callback) { | ||||
|         cb = callback; | ||||
|     }; | ||||
|     return _callRemote(perfHelper, perfHelper.CreateWindowRemote, | ||||
|                        params.width, params.height, | ||||
|                        params.alpha, params.maximized, params.redraws); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -132,17 +157,8 @@ function createTestWindow(width, height, alpha, maximized) { | ||||
|  * created with createTestWindow have been mapped and exposed. | ||||
|  */ | ||||
| function waitTestWindows() { | ||||
|     let cb; | ||||
|     let perfHelper = _getPerfHelper(); | ||||
|  | ||||
|     perfHelper.WaitWindowsRemote(function(result, excp) { | ||||
|                                      if (cb) | ||||
|                                          cb(); | ||||
|                                  }); | ||||
|  | ||||
|     return function(callback) { | ||||
|         cb = callback; | ||||
|     }; | ||||
|     return _callRemote(perfHelper, perfHelper.WaitWindowsRemote); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -155,17 +171,8 @@ function waitTestWindows() { | ||||
|  * process because of normal X asynchronicity. | ||||
|  */ | ||||
| function destroyTestWindows() { | ||||
|     let cb; | ||||
|     let perfHelper = _getPerfHelper(); | ||||
|  | ||||
|     perfHelper.DestroyWindowsRemote(function(result, excp) { | ||||
|                                         if (cb) | ||||
|                                             cb(); | ||||
|                                     }); | ||||
|  | ||||
|     return function(callback) { | ||||
|         cb = callback; | ||||
|     }; | ||||
|     return _callRemote(perfHelper, perfHelper.DestroyWindowsRemote); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -208,6 +215,10 @@ function _step(g, finish, onError) { | ||||
|         let waitFunction = g.next(); | ||||
|         waitFunction(function() { | ||||
|                          _step(g, finish, onError); | ||||
|                      }, | ||||
|                      function(err) { | ||||
|                          if (onError) | ||||
|                              onError(err); | ||||
|                      }); | ||||
|     } catch (err if err instanceof StopIteration) { | ||||
|         if (finish) | ||||
| @@ -306,8 +317,8 @@ function _collect(scriptModule, outputFile) { | ||||
|         print ('------------------------------------------------------------'); | ||||
|         for (let i = 0; i < metrics.length; i++) { | ||||
|             let metric = metrics[i]; | ||||
|             print ('# ' + scriptModule.METRIC_DESCRIPTIONS[metric]); | ||||
|             print (metric + ': ' +  scriptModule.METRICS[metric]); | ||||
|             print ('# ' + scriptModule.METRICS[metric].description); | ||||
|             print (metric + ': ' +  scriptModule.METRICS[metric].value + scriptModule.METRICS[metric].units); | ||||
|         } | ||||
|         print ('------------------------------------------------------------'); | ||||
|     } | ||||
| @@ -360,7 +371,12 @@ function runPerfScript(scriptModule, outputFile) { | ||||
|  | ||||
|     _step(g, | ||||
|           function() { | ||||
|               _collect(scriptModule, outputFile); | ||||
|               try { | ||||
|                   _collect(scriptModule, outputFile); | ||||
|               } catch (err) { | ||||
|                   log("Script failed: " + err + "\n" + err.stack); | ||||
|                   Meta.exit(Meta.ExitCode.ERROR); | ||||
|               } | ||||
|               Meta.exit(Meta.ExitCode.SUCCESS); | ||||
|           }, | ||||
|          function(err) { | ||||
|   | ||||
							
								
								
									
										444
									
								
								js/ui/search.js
									
									
									
									
									
								
							
							
						
						| @@ -2,10 +2,12 @@ | ||||
|  | ||||
| const Clutter = imports.gi.Clutter; | ||||
| const Lang = imports.lang; | ||||
| const GLib = imports.gi.GLib; | ||||
| const Gio = imports.gi.Gio; | ||||
| const Gtk = imports.gi.Gtk; | ||||
| const Meta = imports.gi.Meta; | ||||
| const Signals = imports.signals; | ||||
| const Shell = imports.gi.Shell; | ||||
| const St = imports.gi.St; | ||||
| const Atk = imports.gi.Atk; | ||||
|  | ||||
| @@ -23,107 +25,6 @@ const SEARCH_PROVIDERS_SCHEMA = 'org.gnome.desktop.search-providers'; | ||||
| const MAX_LIST_SEARCH_RESULTS_ROWS = 3; | ||||
| const MAX_GRID_SEARCH_RESULTS_ROWS = 1; | ||||
|  | ||||
| const SearchSystem = new Lang.Class({ | ||||
|     Name: 'SearchSystem', | ||||
|  | ||||
|     _init: function() { | ||||
|         this._providers = []; | ||||
|  | ||||
|         this._registerProvider(new AppDisplay.AppSearchProvider()); | ||||
|  | ||||
|         this._searchSettings = new Gio.Settings({ schema: SEARCH_PROVIDERS_SCHEMA }); | ||||
|         this._searchSettings.connect('changed::disabled', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|         this._searchSettings.connect('changed::disable-external', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|         this._searchSettings.connect('changed::sort-order', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|  | ||||
|         this._reloadRemoteProviders(); | ||||
|  | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|     }, | ||||
|  | ||||
|     addProvider: function(provider) { | ||||
|         this._providers.push(provider); | ||||
|         this.emit('providers-changed'); | ||||
|     }, | ||||
|  | ||||
|     _reloadRemoteProviders: function() { | ||||
|         let remoteProviders = this._providers.filter(function(provider) { | ||||
|             return provider.isRemoteProvider; | ||||
|         }); | ||||
|         remoteProviders.forEach(Lang.bind(this, function(provider) { | ||||
|             this._unregisterProvider(provider); | ||||
|         })); | ||||
|  | ||||
|         RemoteSearch.loadRemoteSearchProviders(Lang.bind(this, function(providers) { | ||||
|             providers.forEach(Lang.bind(this, this._registerProvider)); | ||||
|         })); | ||||
|  | ||||
|         this.emit('providers-changed'); | ||||
|     }, | ||||
|  | ||||
|     _registerProvider: function (provider) { | ||||
|         this._providers.push(provider); | ||||
|     }, | ||||
|  | ||||
|     _unregisterProvider: function (provider) { | ||||
|         let index = this._providers.indexOf(provider); | ||||
|         this._providers.splice(index, 1); | ||||
|  | ||||
|         if (provider.display) | ||||
|             provider.display.destroy(); | ||||
|     }, | ||||
|  | ||||
|     getProviders: function() { | ||||
|         return this._providers; | ||||
|     }, | ||||
|  | ||||
|     getTerms: function() { | ||||
|         return this._terms; | ||||
|     }, | ||||
|  | ||||
|     reset: function() { | ||||
|         this._terms = []; | ||||
|         this._results = {}; | ||||
|     }, | ||||
|  | ||||
|     _gotResults: function(results, provider) { | ||||
|         this._results[provider.id] = results; | ||||
|         this.emit('search-updated', provider, results); | ||||
|     }, | ||||
|  | ||||
|     setTerms: function(terms) { | ||||
|         this._cancellable.cancel(); | ||||
|         this._cancellable.reset(); | ||||
|  | ||||
|         let previousResults = this._results; | ||||
|         let previousTerms = this._terms; | ||||
|         this.reset(); | ||||
|  | ||||
|         if (!terms) | ||||
|             return; | ||||
|  | ||||
|         let searchString = terms.join(' '); | ||||
|         let previousSearchString = previousTerms.join(' '); | ||||
|         if (searchString == previousSearchString) | ||||
|             return; | ||||
|  | ||||
|         let isSubSearch = false; | ||||
|         if (previousTerms.length > 0) | ||||
|             isSubSearch = searchString.indexOf(previousSearchString) == 0; | ||||
|  | ||||
|         this._terms = terms; | ||||
|  | ||||
|         this._providers.forEach(Lang.bind(this, function(provider) { | ||||
|             let previousProviderResults = previousResults[provider.id]; | ||||
|             if (isSubSearch && previousProviderResults) | ||||
|                 provider.getSubsearchResultSet(previousProviderResults, terms, Lang.bind(this, this._gotResults, provider), this._cancellable); | ||||
|             else | ||||
|                 provider.getInitialResultSet(terms, Lang.bind(this, this._gotResults, provider), this._cancellable); | ||||
|         })); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(SearchSystem.prototype); | ||||
|  | ||||
| const MaxWidthBin = new Lang.Class({ | ||||
|     Name: 'MaxWidthBin', | ||||
|     Extends: St.Bin, | ||||
| @@ -163,13 +64,6 @@ const SearchResult = new Lang.Class({ | ||||
|  | ||||
|     activate: function() { | ||||
|         this.emit('activate', this.metaInfo.id); | ||||
|     }, | ||||
|  | ||||
|     setSelected: function(selected) { | ||||
|         if (selected) | ||||
|             this.actor.add_style_pseudo_class('selected'); | ||||
|         else | ||||
|             this.actor.remove_style_pseudo_class('selected'); | ||||
|     } | ||||
| }); | ||||
| Signals.addSignalMethods(SearchResult.prototype); | ||||
| @@ -230,59 +124,11 @@ const GridSearchResult = new Lang.Class({ | ||||
|  | ||||
|         this.actor.style_class = 'grid-search-result'; | ||||
|  | ||||
|         let content = provider.createResultObject(metaInfo); | ||||
|         let dragSource = null; | ||||
|  | ||||
|         if (content == null) { | ||||
|             let actor = new St.Bin(); | ||||
|             let icon = new IconGrid.BaseIcon(this.metaInfo['name'], | ||||
|                                              { createIcon: this.metaInfo['createIcon'] }); | ||||
|             actor.set_child(icon.actor); | ||||
|             actor.label_actor = icon.label; | ||||
|             dragSource = icon.icon; | ||||
|             content = { actor: actor, icon: icon }; | ||||
|         } else { | ||||
|             if (content._delegate && content._delegate.getDragActorSource) | ||||
|                 dragSource = content._delegate.getDragActorSource(); | ||||
|         } | ||||
|  | ||||
|         this.actor.set_child(content.actor); | ||||
|         this.actor.label_actor = content.actor.label_actor; | ||||
|         this.icon = content.icon; | ||||
|  | ||||
|         let draggable = DND.makeDraggable(this.actor); | ||||
|         draggable.connect('drag-begin', | ||||
|                           Lang.bind(this, function() { | ||||
|                               Main.overview.beginItemDrag(this); | ||||
|                           })); | ||||
|         draggable.connect('drag-cancelled', | ||||
|                           Lang.bind(this, function() { | ||||
|                               Main.overview.cancelledItemDrag(this); | ||||
|                           })); | ||||
|         draggable.connect('drag-end', | ||||
|                           Lang.bind(this, function() { | ||||
|                               Main.overview.endItemDrag(this); | ||||
|                           })); | ||||
|  | ||||
|         if (!dragSource) | ||||
|             // not exactly right, but alignment problems are hard to notice | ||||
|             dragSource = content; | ||||
|         this._dragActorSource = dragSource; | ||||
|     }, | ||||
|  | ||||
|     getDragActorSource: function() { | ||||
|         return this._dragActorSource; | ||||
|     }, | ||||
|  | ||||
|     getDragActor: function() { | ||||
|         return this.metaInfo['createIcon'](Main.overview.dashIconSize); | ||||
|     }, | ||||
|  | ||||
|     shellWorkspaceLaunch: function(params) { | ||||
|         if (this.provider.dragActivateResult) | ||||
|             this.provider.dragActivateResult(this.metaInfo.id, params); | ||||
|         else | ||||
|             this.provider.activateResult(this.metaInfo.id, this.terms); | ||||
|         this.icon = new IconGrid.BaseIcon(this.metaInfo['name'], | ||||
|                                           { createIcon: this.metaInfo['createIcon'] }); | ||||
|         let content = new St.Bin({ child: this.icon.actor }); | ||||
|         this.actor.set_child(content); | ||||
|         this.actor.label_actor = this.icon.label; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -314,7 +160,11 @@ const SearchResultsBase = new Lang.Class({ | ||||
|         this._terms = []; | ||||
|     }, | ||||
|  | ||||
|     _clearResultDisplay: function() { | ||||
|     _createResultDisplay: function(meta) { | ||||
|         if (this.provider.createResultObject) | ||||
|             return this.provider.createResultObject(meta); | ||||
|  | ||||
|         return null; | ||||
|     }, | ||||
|  | ||||
|     clear: function() { | ||||
| @@ -349,12 +199,9 @@ const SearchResultsBase = new Lang.Class({ | ||||
|             this._cancellable.reset(); | ||||
|  | ||||
|             this.provider.getResultMetas(metasNeeded, Lang.bind(this, function(metas) { | ||||
|                 if (metas.length == 0) { | ||||
|                     callback(false); | ||||
|                     return; | ||||
|                 } | ||||
|                 if (metas.length != metasNeeded.length) { | ||||
|                     log('Wrong number of result metas returned by search provider'); | ||||
|                     log('Wrong number of result metas returned by search provider ' + this.provider.id + | ||||
|                         ': expected ' + metasNeeded.length + ' but got ' + metas.length); | ||||
|                     callback(false); | ||||
|                     return; | ||||
|                 } | ||||
| @@ -384,9 +231,11 @@ const SearchResultsBase = new Lang.Class({ | ||||
|             let hasMoreResults = results.length < providerResults.length; | ||||
|  | ||||
|             this._ensureResultActors(results, Lang.bind(this, function(successful) { | ||||
|                 this._clearResultDisplay(); | ||||
|                 if (!successful) | ||||
|                 if (!successful) { | ||||
|                     this._clearResultDisplay(); | ||||
|                     callback(); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 // To avoid CSS transitions causing flickering when | ||||
|                 // the first search result stays the same, we hide the | ||||
| @@ -416,6 +265,7 @@ const ListSearchResults = new Lang.Class({ | ||||
|         this.providerIcon.connect('key-focus-in', Lang.bind(this, this._keyFocusIn)); | ||||
|         this.providerIcon.connect('clicked', Lang.bind(this, | ||||
|             function() { | ||||
|                 this.providerIcon.animateLaunch(); | ||||
|                 provider.launchSearch(this._terms); | ||||
|                 Main.overview.toggle(); | ||||
|             })); | ||||
| @@ -433,7 +283,7 @@ const ListSearchResults = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _setMoreIconVisible: function(visible) { | ||||
|         this.providerIcon.moreIcon.visible = true; | ||||
|         this.providerIcon.moreIcon.visible = visible; | ||||
|     }, | ||||
|  | ||||
|     _getMaxDisplayedResults: function() { | ||||
| @@ -445,7 +295,7 @@ const ListSearchResults = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _createResultDisplay: function(meta) { | ||||
|         return new ListSearchResult(this.provider, meta); | ||||
|         return this.parent(meta) || new ListSearchResult(this.provider, meta); | ||||
|     }, | ||||
|  | ||||
|     _addItem: function(display) { | ||||
| @@ -465,8 +315,14 @@ const GridSearchResults = new Lang.Class({ | ||||
|     Name: 'GridSearchResults', | ||||
|     Extends: SearchResultsBase, | ||||
|  | ||||
|     _init: function(provider) { | ||||
|     _init: function(provider, parentContainer) { | ||||
|         this.parent(provider); | ||||
|         // We need to use the parent container to know how much results we can show. | ||||
|         // None of the actors in this class can be used for that, since the main actor | ||||
|         // goes hidden when no results are displayed, and then it lost its allocation. | ||||
|         // Then on the next use of _getMaxDisplayedResults allocation is 0, en therefore | ||||
|         // it doesn't show any result although we have some. | ||||
|         this._parentContainer = parentContainer; | ||||
|  | ||||
|         this._grid = new IconGrid.IconGrid({ rowLimit: MAX_GRID_SEARCH_RESULTS_ROWS, | ||||
|                                              xAlign: St.Align.START }); | ||||
| @@ -477,16 +333,9 @@ const GridSearchResults = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _getMaxDisplayedResults: function() { | ||||
|         return this._grid.columnsForWidth(this._bin.width) * this._grid.getRowLimit(); | ||||
|     }, | ||||
|  | ||||
|     _renderResults: function(metas) { | ||||
|         for (let i = 0; i < metas.length; i++) { | ||||
|             let display = new GridSearchResult(this.provider, metas[i]); | ||||
|             display.connect('activate', Lang.bind(this, this._activateResult)); | ||||
|             display.actor.connect('key-focus-in', Lang.bind(this, this._keyFocusIn)); | ||||
|             this._grid.addItem(display); | ||||
|         } | ||||
|         let parentThemeNode = this._parentContainer.get_theme_node(); | ||||
|         let availableWidth = parentThemeNode.adjust_for_width(this._parentContainer.width); | ||||
|         return this._grid.columnsForWidth(availableWidth) * this._grid.getRowLimit(); | ||||
|     }, | ||||
|  | ||||
|     _clearResultDisplay: function () { | ||||
| @@ -494,7 +343,7 @@ const GridSearchResults = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _createResultDisplay: function(meta) { | ||||
|         return new GridSearchResult(this.provider, meta); | ||||
|         return this.parent(meta) || new GridSearchResult(this.provider, meta); | ||||
|     }, | ||||
|  | ||||
|     _addItem: function(display) { | ||||
| @@ -546,16 +395,136 @@ const SearchResults = new Lang.Class({ | ||||
|         this._statusText = new St.Label({ style_class: 'search-statustext' }); | ||||
|         this._statusBin = new St.Bin({ x_align: St.Align.MIDDLE, | ||||
|                                        y_align: St.Align.MIDDLE }); | ||||
|         this._content.add(this._statusBin, { expand: true }); | ||||
|         this.actor.add(this._statusBin, { expand: true }); | ||||
|         this._statusBin.add_actor(this._statusText); | ||||
|  | ||||
|         this._highlightDefault = false; | ||||
|         this._defaultResult = null; | ||||
|         this._startingSearch = false; | ||||
|  | ||||
|         this._searchSystem = new SearchSystem(); | ||||
|         this._searchSystem.connect('search-updated', Lang.bind(this, this._updateResults)); | ||||
|         this._searchSystem.connect('providers-changed', Lang.bind(this, this._updateProviderDisplays)); | ||||
|         this._updateProviderDisplays(); | ||||
|         this._terms = []; | ||||
|         this._results = {}; | ||||
|  | ||||
|         this._providers = []; | ||||
|  | ||||
|         this._searchSettings = new Gio.Settings({ schema_id: SEARCH_PROVIDERS_SCHEMA }); | ||||
|         this._searchSettings.connect('changed::disabled', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|         this._searchSettings.connect('changed::disable-external', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|         this._searchSettings.connect('changed::sort-order', Lang.bind(this, this._reloadRemoteProviders)); | ||||
|  | ||||
|         this._searchTimeoutId = 0; | ||||
|         this._cancellable = new Gio.Cancellable(); | ||||
|  | ||||
|         this._registerProvider(new AppDisplay.AppSearchProvider()); | ||||
|         this._reloadRemoteProviders(); | ||||
|     }, | ||||
|  | ||||
|     _reloadRemoteProviders: function() { | ||||
|         let remoteProviders = this._providers.filter(function(provider) { | ||||
|             return provider.isRemoteProvider; | ||||
|         }); | ||||
|         remoteProviders.forEach(Lang.bind(this, function(provider) { | ||||
|             this._unregisterProvider(provider); | ||||
|         })); | ||||
|  | ||||
|         RemoteSearch.loadRemoteSearchProviders(Lang.bind(this, function(providers) { | ||||
|             providers.forEach(Lang.bind(this, this._registerProvider)); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     _registerProvider: function (provider) { | ||||
|         this._providers.push(provider); | ||||
|         this._ensureProviderDisplay(provider); | ||||
|     }, | ||||
|  | ||||
|     _unregisterProvider: function (provider) { | ||||
|         let index = this._providers.indexOf(provider); | ||||
|         this._providers.splice(index, 1); | ||||
|  | ||||
|         if (provider.display) | ||||
|             provider.display.destroy(); | ||||
|     }, | ||||
|  | ||||
|     _gotResults: function(results, provider) { | ||||
|         this._results[provider.id] = results; | ||||
|         this._updateResults(provider, results); | ||||
|     }, | ||||
|  | ||||
|     _clearSearchTimeout: function() { | ||||
|         if (this._searchTimeoutId > 0) { | ||||
|             GLib.source_remove(this._searchTimeoutId); | ||||
|             this._searchTimeoutId = 0; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _reset: function() { | ||||
|         this._terms = []; | ||||
|         this._results = {}; | ||||
|         this._clearDisplay(); | ||||
|         this._clearSearchTimeout(); | ||||
|         this._defaultResult = null; | ||||
|         this._startingSearch = false; | ||||
|  | ||||
|         this._updateSearchProgress(); | ||||
|     }, | ||||
|  | ||||
|     _doSearch: function() { | ||||
|         this._startingSearch = false; | ||||
|  | ||||
|         let previousResults = this._results; | ||||
|         this._results = {}; | ||||
|  | ||||
|         this._providers.forEach(Lang.bind(this, function(provider) { | ||||
|             provider.searchInProgress = true; | ||||
|  | ||||
|             let previousProviderResults = previousResults[provider.id]; | ||||
|             if (this._isSubSearch && previousProviderResults) | ||||
|                 provider.getSubsearchResultSet(previousProviderResults, this._terms, Lang.bind(this, this._gotResults, provider), this._cancellable); | ||||
|             else | ||||
|                 provider.getInitialResultSet(this._terms, Lang.bind(this, this._gotResults, provider), this._cancellable); | ||||
|         })); | ||||
|  | ||||
|         this._updateSearchProgress(); | ||||
|  | ||||
|         this._clearSearchTimeout(); | ||||
|     }, | ||||
|  | ||||
|     _onSearchTimeout: function() { | ||||
|         this._searchTimeoutId = 0; | ||||
|         this._doSearch(); | ||||
|         return GLib.SOURCE_REMOVE; | ||||
|     }, | ||||
|  | ||||
|     setTerms: function(terms) { | ||||
|         // Check for the case of making a duplicate previous search before | ||||
|         // setting state of the current search or cancelling the search. | ||||
|         // This will prevent incorrect state being as a result of a duplicate | ||||
|         // search while the previous search is still active. | ||||
|         let searchString = terms.join(' '); | ||||
|         let previousSearchString = this._terms.join(' '); | ||||
|         if (searchString == previousSearchString) | ||||
|             return; | ||||
|  | ||||
|         this._startingSearch = true; | ||||
|  | ||||
|         this._cancellable.cancel(); | ||||
|         this._cancellable.reset(); | ||||
|  | ||||
|         if (terms.length == 0) { | ||||
|             this._reset(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         let isSubSearch = false; | ||||
|         if (this._terms.length > 0) | ||||
|             isSubSearch = searchString.indexOf(previousSearchString) == 0; | ||||
|  | ||||
|         this._terms = terms; | ||||
|         this._isSubSearch = isSubSearch; | ||||
|         this._updateSearchProgress(); | ||||
|  | ||||
|         if (this._searchTimeoutId == 0) | ||||
|             this._searchTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, 150, Lang.bind(this, this._onSearchTimeout)); | ||||
|     }, | ||||
|  | ||||
|     _onPan: function(action) { | ||||
| @@ -577,44 +546,24 @@ const SearchResults = new Lang.Class({ | ||||
|         if (provider.appInfo) | ||||
|             providerDisplay = new ListSearchResults(provider); | ||||
|         else | ||||
|             providerDisplay = new GridSearchResults(provider); | ||||
|             providerDisplay = new GridSearchResults(provider, this.actor); | ||||
|  | ||||
|         providerDisplay.connect('key-focus-in', Lang.bind(this, this._keyFocusIn)); | ||||
|         providerDisplay.actor.hide(); | ||||
|         this._content.add(providerDisplay.actor); | ||||
|         provider.display = providerDisplay; | ||||
|     }, | ||||
|  | ||||
|     _updateProviderDisplays: function() { | ||||
|         this._searchSystem.getProviders().forEach(Lang.bind(this, this._ensureProviderDisplay)); | ||||
|     }, | ||||
|  | ||||
|     _clearDisplay: function() { | ||||
|         this._searchSystem.getProviders().forEach(function(provider) { | ||||
|         this._providers.forEach(function(provider) { | ||||
|             provider.display.clear(); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     reset: function() { | ||||
|         this._searchSystem.reset(); | ||||
|         this._statusBin.hide(); | ||||
|         this._clearDisplay(); | ||||
|         this._defaultResult = null; | ||||
|     }, | ||||
|  | ||||
|     startingSearch: function() { | ||||
|         this.reset(); | ||||
|         this._statusText.set_text(_("Searching…")); | ||||
|         this._statusBin.show(); | ||||
|     }, | ||||
|  | ||||
|     setTerms: function(terms) { | ||||
|         this._searchSystem.setTerms(terms); | ||||
|     }, | ||||
|  | ||||
|     _maybeSetInitialSelection: function() { | ||||
|         let newDefaultResult = null; | ||||
|  | ||||
|         let providers = this._searchSystem.getProviders(); | ||||
|         let providers = this._providers; | ||||
|         for (let i = 0; i < providers.length; i++) { | ||||
|             let provider = providers[i]; | ||||
|             let display = provider.display; | ||||
| @@ -630,54 +579,64 @@ const SearchResults = new Lang.Class({ | ||||
|         } | ||||
|  | ||||
|         if (newDefaultResult != this._defaultResult) { | ||||
|             if (this._defaultResult) | ||||
|                 this._defaultResult.setSelected(false); | ||||
|             if (newDefaultResult) { | ||||
|                 newDefaultResult.setSelected(this._highlightDefault); | ||||
|                 if (this._highlightDefault) | ||||
|                     Util.ensureActorVisibleInScrollView(this._scrollView, newDefaultResult.actor); | ||||
|             } | ||||
|             this._setSelected(this._defaultResult, false); | ||||
|             this._setSelected(newDefaultResult, this._highlightDefault); | ||||
|  | ||||
|             this._defaultResult = newDefaultResult; | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateStatusText: function () { | ||||
|         let haveResults = this._searchSystem.getProviders().some(function(provider) { | ||||
|     get searchInProgress() { | ||||
|         if (this._startingSearch) | ||||
|             return true; | ||||
|  | ||||
|         return this._providers.some(function(provider) { | ||||
|             return provider.searchInProgress; | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|     _updateSearchProgress: function () { | ||||
|         let haveResults = this._providers.some(function(provider) { | ||||
|             let display = provider.display; | ||||
|             return (display.getFirstResult() != null); | ||||
|         }); | ||||
|  | ||||
|         this._scrollView.visible = haveResults; | ||||
|         this._statusBin.visible = !haveResults; | ||||
|  | ||||
|         if (!haveResults) { | ||||
|             this._statusText.set_text(_("No results.")); | ||||
|             this._statusBin.show(); | ||||
|         } else { | ||||
|             this._statusBin.hide(); | ||||
|             if (this.searchInProgress) { | ||||
|                 this._statusText.set_text(_("Searching…")); | ||||
|             } else { | ||||
|                 this._statusText.set_text(_("No results.")); | ||||
|             } | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     _updateResults: function(searchSystem, provider, results) { | ||||
|         let terms = searchSystem.getTerms(); | ||||
|     _updateResults: function(provider, results) { | ||||
|         let terms = this._terms; | ||||
|         let display = provider.display; | ||||
|  | ||||
|         display.updateSearch(results, terms, Lang.bind(this, function() { | ||||
|             provider.searchInProgress = false; | ||||
|  | ||||
|             this._maybeSetInitialSelection(); | ||||
|             this._updateStatusText(); | ||||
|             this._updateSearchProgress(); | ||||
|         })); | ||||
|     }, | ||||
|  | ||||
|     activateDefault: function() { | ||||
|         // If we have a search queued up, force the search now. | ||||
|         if (this._searchTimeoutId > 0) | ||||
|             this._doSearch(); | ||||
|  | ||||
|         if (this._defaultResult) | ||||
|             this._defaultResult.activate(); | ||||
|     }, | ||||
|  | ||||
|     highlightDefault: function(highlight) { | ||||
|         this._highlightDefault = highlight; | ||||
|         if (this._defaultResult) { | ||||
|             this._defaultResult.setSelected(highlight); | ||||
|             if (highlight) | ||||
|                 Util.ensureActorVisibleInScrollView(this._scrollView, this._defaultResult.actor); | ||||
|         } | ||||
|         this._setSelected(this._defaultResult, highlight); | ||||
|     }, | ||||
|  | ||||
|     navigateFocus: function(direction) { | ||||
| @@ -692,6 +651,18 @@ const SearchResults = new Lang.Class({ | ||||
|  | ||||
|         let from = this._defaultResult ? this._defaultResult.actor : null; | ||||
|         this.actor.navigate_focus(from, direction, false); | ||||
|     }, | ||||
|  | ||||
|     _setSelected: function(result, selected) { | ||||
|         if (!result) | ||||
|             return; | ||||
|  | ||||
|         if (selected) { | ||||
|             result.actor.add_style_pseudo_class('selected'); | ||||
|             Util.ensureActorVisibleInScrollView(this._scrollView, result.actor); | ||||
|         } else { | ||||
|             result.actor.remove_style_pseudo_class('selected'); | ||||
|         } | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -725,5 +696,12 @@ const ProviderIcon = new Lang.Class({ | ||||
|                                  gicon: provider.appInfo.get_icon() }); | ||||
|         this._content.add_actor(icon); | ||||
|         this._content.add_actor(this.moreIcon); | ||||
|     }, | ||||
|  | ||||
|     animateLaunch: function() { | ||||
|         let appSys = Shell.AppSystem.get_default(); | ||||
|         let app = appSys.lookup_app(this.provider.appInfo.get_id()); | ||||
|         if (app.state == Shell.AppState.STOPPED) | ||||
|             IconGrid.zoomOutActor(this._content); | ||||
|     } | ||||
| }); | ||||
|   | ||||
| @@ -18,7 +18,6 @@ const _modes = { | ||||
|     'restrictive': { | ||||
|         parentMode: null, | ||||
|         stylesheetName: 'gnome-shell.css', | ||||
|         overridesSchema: 'org.gnome.shell.overrides', | ||||
|         hasOverview: false, | ||||
|         showCalendarEvents: false, | ||||
|         allowSettings: false, | ||||
|   | ||||
| @@ -25,6 +25,10 @@ const GnomeShellIface = '<node> \ | ||||
| <method name="ShowOSD"> \ | ||||
|     <arg type="a{sv}" direction="in" name="params"/> \ | ||||
| </method> \ | ||||
| <method name="ShowMonitorLabels"> \ | ||||
|     <arg type="a{uv}" direction="in" name="params" /> \ | ||||
| </method> \ | ||||
| <method name="HideMonitorLabels" /> \ | ||||
| <method name="FocusApp"> \ | ||||
|     <arg type="s" direction="in" name="id"/> \ | ||||
| </method> \ | ||||
| @@ -44,8 +48,7 @@ const GnomeShellIface = '<node> \ | ||||
| </method> \ | ||||
| <signal name="AcceleratorActivated"> \ | ||||
|     <arg name="action" type="u" /> \ | ||||
|     <arg name="deviceid" type="u" /> \ | ||||
|     <arg name="timestamp" type="u" /> \ | ||||
|     <arg name="parameters" type="a{sv}" /> \ | ||||
| </signal> \ | ||||
| <property name="Mode" type="s" access="read" /> \ | ||||
| <property name="OverviewActive" type="b" access="readwrite" /> \ | ||||
| @@ -90,6 +93,12 @@ const GnomeShell = new Lang.Class({ | ||||
|             function(display, action, deviceid, timestamp) { | ||||
|                 this._emitAcceleratorActivated(action, deviceid, timestamp); | ||||
|             })); | ||||
|  | ||||
|         this._cachedOverviewVisible = false; | ||||
|         Main.overview.connect('showing', | ||||
|                               Lang.bind(this, this._checkOverviewVisibleChanged)); | ||||
|         Main.overview.connect('hidden', | ||||
|                               Lang.bind(this, this._checkOverviewVisibleChanged)); | ||||
|     }, | ||||
|  | ||||
|     /** | ||||
| @@ -134,7 +143,7 @@ const GnomeShell = new Lang.Class({ | ||||
|             params[param] = params[param].deep_unpack(); | ||||
|  | ||||
|         let monitorIndex = -1; | ||||
|         if (params['monitor']) | ||||
|         if (params['monitor'] >= 0) | ||||
|             monitorIndex = params['monitor']; | ||||
|  | ||||
|         let icon = null; | ||||
| @@ -190,11 +199,14 @@ const GnomeShell = new Lang.Class({ | ||||
|  | ||||
|         let connection = this._dbusImpl.get_connection(); | ||||
|         let info = this._dbusImpl.get_info(); | ||||
|         let params = { 'device-id': GLib.Variant.new('u', deviceid), | ||||
|                        'timestamp': GLib.Variant.new('u', timestamp), | ||||
|                        'action-mode': GLib.Variant.new('u', Main.actionMode) }; | ||||
|         connection.emit_signal(destination, | ||||
|                                this._dbusImpl.get_object_path(), | ||||
|                                info ? info.name : null, | ||||
|                                'AcceleratorActivated', | ||||
|                                GLib.Variant.new('(uuu)', [action, deviceid, timestamp])); | ||||
|                                GLib.Variant.new('(ua{sv})', [action, params])); | ||||
|     }, | ||||
|  | ||||
|     _grabAcceleratorForSender: function(accelerator, flags, sender) { | ||||
| @@ -232,11 +244,29 @@ const GnomeShell = new Lang.Class({ | ||||
|         this._grabbers.delete(name); | ||||
|     }, | ||||
|  | ||||
|     ShowMonitorLabelsAsync: function(params, invocation) { | ||||
|         let sender = invocation.get_sender(); | ||||
|         let [dict] = params; | ||||
|         Main.osdMonitorLabeler.show(sender, dict); | ||||
|     }, | ||||
|  | ||||
|     HideMonitorLabelsAsync: function(params, invocation) { | ||||
|         let sender = invocation.get_sender(); | ||||
|         Main.osdMonitorLabeler.hide(sender); | ||||
|     }, | ||||
|  | ||||
|  | ||||
|     Mode: global.session_mode, | ||||
|  | ||||
|     _checkOverviewVisibleChanged: function() { | ||||
|         if (Main.overview.visible !== this._cachedOverviewVisible) { | ||||
|             this._cachedOverviewVisible = Main.overview.visible; | ||||
|             this._dbusImpl.emit_property_changed('OverviewActive', new GLib.Variant('b', this._cachedOverviewVisible)); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     get OverviewActive() { | ||||
|         return Main.overview.visible; | ||||
|         return this._cachedOverviewVisible; | ||||
|     }, | ||||
|  | ||||
|     set OverviewActive(visible) { | ||||
|   | ||||
| @@ -17,8 +17,6 @@ const EntryMenu = new Lang.Class({ | ||||
|     _init: function(entry) { | ||||
|         this.parent(entry, 0, St.Side.TOP); | ||||
|  | ||||
|         this.actor.add_style_class_name('entry-context-menu'); | ||||
|  | ||||
|         this._entry = entry; | ||||
|         this._clipboard = St.Clipboard.get_default(); | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ const Lang = imports.lang; | ||||
| const St = imports.gi.St; | ||||
| const Signals = imports.signals; | ||||
|  | ||||
| const SLIDER_SCROLL_STEP = 0.05; /* Slider scrolling step in % */ | ||||
| const SLIDER_SCROLL_STEP = 0.02; /* Slider scrolling step in % */ | ||||
|  | ||||
| const Slider = new Lang.Class({ | ||||
|     Name: "Slider", | ||||
| @@ -24,6 +24,7 @@ const Slider = new Lang.Class({ | ||||
|                                           accessible_role: Atk.Role.SLIDER }); | ||||
|         this.actor.connect('repaint', Lang.bind(this, this._sliderRepaint)); | ||||
|         this.actor.connect('button-press-event', Lang.bind(this, this._startDragging)); | ||||
|         this.actor.connect('touch-event', Lang.bind(this, this._touchDragging)); | ||||
|         this.actor.connect('scroll-event', Lang.bind(this, this._onScrollEvent)); | ||||
|         this.actor.connect('key-press-event', Lang.bind(this, this.onKeyPressEvent)); | ||||
|  | ||||
| @@ -121,11 +122,21 @@ const Slider = new Lang.Class({ | ||||
|         this._dragging = true; | ||||
|  | ||||
|         let device = event.get_device(); | ||||
|         device.grab(this.actor); | ||||
|         this._grabbedDevice = device; | ||||
|         let sequence = event.get_event_sequence(); | ||||
|  | ||||
|         if (sequence != null) | ||||
|             device.sequence_grab(sequence, this.actor); | ||||
|         else | ||||
|             device.grab(this.actor); | ||||
|  | ||||
|         this._grabbedDevice = device; | ||||
|         this._grabbedSequence = sequence; | ||||
|  | ||||
|         if (sequence == null) { | ||||
|             this._releaseId = this.actor.connect('button-release-event', Lang.bind(this, this._endDragging)); | ||||
|             this._motionId = this.actor.connect('motion-event', Lang.bind(this, this._motionEvent)); | ||||
|         } | ||||
|  | ||||
|         this._releaseId = this.actor.connect('button-release-event', Lang.bind(this, this._endDragging)); | ||||
|         this._motionId = this.actor.connect('motion-event', Lang.bind(this, this._motionEvent)); | ||||
|         let absX, absY; | ||||
|         [absX, absY] = event.get_coords(); | ||||
|         this._moveHandle(absX, absY); | ||||
| @@ -134,10 +145,17 @@ const Slider = new Lang.Class({ | ||||
|  | ||||
|     _endDragging: function() { | ||||
|         if (this._dragging) { | ||||
|             this.actor.disconnect(this._releaseId); | ||||
|             this.actor.disconnect(this._motionId); | ||||
|             if (this._releaseId) | ||||
|                 this.actor.disconnect(this._releaseId); | ||||
|             if (this._motionId) | ||||
|                 this.actor.disconnect(this._motionId); | ||||
|  | ||||
|             this._grabbedDevice.ungrab(); | ||||
|             if (this._grabbedSequence != null) | ||||
|                 this._grabbedDevice.sequence_ungrab(this._grabbedSequence); | ||||
|             else | ||||
|                 this._grabbedDevice.ungrab(); | ||||
|  | ||||
|             this._grabbedSequence = null; | ||||
|             this._grabbedDevice = null; | ||||
|             this._dragging = false; | ||||
|  | ||||
| @@ -146,6 +164,24 @@ const Slider = new Lang.Class({ | ||||
|         return Clutter.EVENT_STOP; | ||||
|     }, | ||||
|  | ||||
|     _touchDragging: function(actor, event) { | ||||
|         let device = event.get_device(); | ||||
|         let sequence = event.get_event_sequence(); | ||||
|  | ||||
|         if (!this._dragging && | ||||
|             event.type() == Clutter.EventType.TOUCH_BEGIN) { | ||||
|             this.startDragging(event); | ||||
|             return Clutter.EVENT_STOP; | ||||
|         } else if (device.sequence_get_grabbed_actor(sequence) == actor) { | ||||
|             if (event.type() == Clutter.EventType.TOUCH_UPDATE) | ||||
|                 return this._motionEvent(actor, event); | ||||
|             else if (event.type() == Clutter.EventType.TOUCH_END) | ||||
|                 return this._endDragging(); | ||||
|         } | ||||
|  | ||||
|         return Clutter.EVENT_PROPAGATE; | ||||
|     }, | ||||
|  | ||||
|     scroll: function(event) { | ||||
|         let direction = event.get_scroll_direction(); | ||||
|         let delta; | ||||
| @@ -161,7 +197,7 @@ const Slider = new Lang.Class({ | ||||
|             let [dx, dy] = event.get_scroll_delta(); | ||||
|             // Even though the slider is horizontal, use dy to match | ||||
|             // the UP/DOWN above. | ||||
|             delta = -dy / 10; | ||||
|             delta = -dy * SLIDER_SCROLL_STEP; | ||||
|         } | ||||
|  | ||||
|         this._value = Math.min(Math.max(0, this._value + delta), 1); | ||||
|   | ||||
| @@ -48,7 +48,7 @@ const ATIndicator = new Lang.Class({ | ||||
|  | ||||
|         this.actor.add_child(this._hbox); | ||||
|  | ||||
|         this._a11ySettings = new Gio.Settings({ schema: A11Y_SCHEMA }); | ||||
|         this._a11ySettings = new Gio.Settings({ schema_id: A11Y_SCHEMA }); | ||||
|         this._a11ySettings.connect('changed::' + KEY_ALWAYS_SHOW, Lang.bind(this, this._queueSyncMenuVisibility)); | ||||
|  | ||||
|         let highContrast = this._buildHCItem(); | ||||
| @@ -118,7 +118,7 @@ const ATIndicator = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _buildItem: function(string, schema, key) { | ||||
|         let settings = new Gio.Settings({ schema: schema }); | ||||
|         let settings = new Gio.Settings({ schema_id: schema }); | ||||
|         let widget = this._buildItemExtended(string, | ||||
|             settings.get_boolean(key), | ||||
|             settings.is_writable(key), | ||||
| @@ -134,8 +134,8 @@ const ATIndicator = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _buildHCItem: function() { | ||||
|         let interfaceSettings = new Gio.Settings({ schema: DESKTOP_INTERFACE_SCHEMA }); | ||||
|         let wmSettings = new Gio.Settings({ schema: WM_SCHEMA }); | ||||
|         let interfaceSettings = new Gio.Settings({ schema_id: DESKTOP_INTERFACE_SCHEMA }); | ||||
|         let wmSettings = new Gio.Settings({ schema_id: WM_SCHEMA }); | ||||
|         let gtkTheme = interfaceSettings.get_string(KEY_GTK_THEME); | ||||
|         let iconTheme = interfaceSettings.get_string(KEY_ICON_THEME); | ||||
|         let wmTheme = wmSettings.get_string(KEY_WM_THEME); | ||||
| @@ -186,7 +186,7 @@ const ATIndicator = new Lang.Class({ | ||||
|     }, | ||||
|  | ||||
|     _buildFontItem: function() { | ||||
|         let settings = new Gio.Settings({ schema: DESKTOP_INTERFACE_SCHEMA }); | ||||
|         let settings = new Gio.Settings({ schema_id: DESKTOP_INTERFACE_SCHEMA }); | ||||
|  | ||||
|         let factor = settings.get_double(KEY_TEXT_SCALING_FACTOR); | ||||
|         let initial_setting = (factor > 1.0); | ||||
|   | ||||