0002-Bug-1419426-Implement-browserSettings.contextMenuSho.patch 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. From f19a0f3bc5e1e87d3c663cc2b147c5c0831519c5 Mon Sep 17 00:00:00 2001
  2. Message-Id: <f19a0f3bc5e1e87d3c663cc2b147c5c0831519c5.1512038840.git.jan.steffens@gmail.com>
  3. In-Reply-To: <05ec1aa0d5e8806dd0c5c6d08c82846a1389b599.1512038840.git.jan.steffens@gmail.com>
  4. References: <05ec1aa0d5e8806dd0c5c6d08c82846a1389b599.1512038840.git.jan.steffens@gmail.com>
  5. From: Bob Silverberg <bsilverberg@mozilla.com>
  6. Date: Fri, 24 Nov 2017 07:45:03 -0500
  7. Subject: [PATCH 2/2] Bug 1419426 - Implement
  8. browserSettings.contextMenuShowEvent, r=kmag a=gchang
  9. Uplift for 58.
  10. ---
  11. .../components/extensions/ext-browserSettings.js | 45 ++++++++++++++++
  12. .../extensions/schemas/browser_settings.json | 10 ++++
  13. .../test/xpcshell/test_ext_browserSettings.js | 62 ++++++++++++++++++++--
  14. 3 files changed, 114 insertions(+), 3 deletions(-)
  15. diff --git a/toolkit/components/extensions/ext-browserSettings.js b/toolkit/components/extensions/ext-browserSettings.js
  16. index f3212f351baf6975..2b24bcc1d09091f2 100644
  17. --- a/toolkit/components/extensions/ext-browserSettings.js
  18. +++ b/toolkit/components/extensions/ext-browserSettings.js
  19. @@ -2,17 +2,23 @@
  20. /* vim: set sts=2 sw=2 et tw=80: */
  21. "use strict";
  22. +XPCOMUtils.defineLazyModuleGetter(this, "AppConstants",
  23. + "resource://gre/modules/AppConstants.jsm");
  24. XPCOMUtils.defineLazyModuleGetter(this, "ExtensionSettingsStore",
  25. "resource://gre/modules/ExtensionSettingsStore.jsm");
  26. XPCOMUtils.defineLazyModuleGetter(this, "Services",
  27. "resource://gre/modules/Services.jsm");
  28. XPCOMUtils.defineLazyServiceGetter(this, "aboutNewTabService",
  29. "@mozilla.org/browser/aboutnewtab-service;1",
  30. "nsIAboutNewTabService");
  31. Cu.import("resource://gre/modules/ExtensionPreferencesManager.jsm");
  32. +var {
  33. + ExtensionError,
  34. +} = ExtensionUtils;
  35. +
  36. const HOMEPAGE_OVERRIDE_SETTING = "homepage_override";
  37. const HOMEPAGE_URL_PREF = "browser.startup.homepage";
  38. const URL_STORE_TYPE = "url_overrides";
  39. @@ -82,6 +88,16 @@ ExtensionPreferencesManager.addSetting("imageAnimationBehavior", {
  40. },
  41. });
  42. +ExtensionPreferencesManager.addSetting("contextMenuShowEvent", {
  43. + prefNames: [
  44. + "ui.context_menus.after_mouseup",
  45. + ],
  46. +
  47. + setCallback(value) {
  48. + return {[this.prefNames[0]]: value === "mouseup"};
  49. + },
  50. +});
  51. +
  52. this.browserSettings = class extends ExtensionAPI {
  53. getAPI(context) {
  54. let {extension} = context;
  55. @@ -114,6 +130,35 @@ this.browserSettings = class extends ExtensionAPI {
  56. () => {
  57. return aboutNewTabService.newTabURL;
  58. }, URL_STORE_TYPE, true),
  59. + contextMenuShowEvent: Object.assign(
  60. + getSettingsAPI(
  61. + extension,
  62. + "contextMenuShowEvent",
  63. + () => {
  64. + if (AppConstants.platform === "win") {
  65. + return "mouseup";
  66. + }
  67. + let prefValue = Services.prefs.getBoolPref(
  68. + "ui.context_menus.after_mouseup", null);
  69. + return prefValue ? "mouseup" : "mousedown";
  70. + }
  71. + ),
  72. + {
  73. + set: details => {
  74. + if (!["mouseup", "mousedown"].includes(details.value)) {
  75. + throw new ExtensionError(
  76. + `${details.value} is not a valid value for contextMenuShowEvent.`);
  77. + }
  78. + if (AppConstants.platform === "android" ||
  79. + (AppConstants.platform === "win" &&
  80. + details.value === "mousedown")) {
  81. + return false;
  82. + }
  83. + return ExtensionPreferencesManager.setSetting(
  84. + extension, "contextMenuShowEvent", details.value);
  85. + },
  86. + }
  87. + ),
  88. },
  89. };
  90. }
  91. diff --git a/toolkit/components/extensions/schemas/browser_settings.json b/toolkit/components/extensions/schemas/browser_settings.json
  92. index af073d933723cbd5..4f354e69dfedaf96 100644
  93. --- a/toolkit/components/extensions/schemas/browser_settings.json
  94. +++ b/toolkit/components/extensions/schemas/browser_settings.json
  95. @@ -27,28 +27,38 @@
  96. "type": "string",
  97. "enum": ["normal", "none", "once"],
  98. "description": "How images should be animated in the browser."
  99. + },
  100. + {
  101. + "id": "ContextMenuMouseEvent",
  102. + "type": "string",
  103. + "enum": ["mouseup", "mousedown"],
  104. + "description": "After which mouse event context menus should popup."
  105. }
  106. ],
  107. "properties": {
  108. "allowPopupsForUserEvents": {
  109. "$ref": "types.Setting",
  110. "description": "Allows or disallows pop-up windows from opening in response to user events."
  111. },
  112. "cacheEnabled": {
  113. "$ref": "types.Setting",
  114. "description": "Enables or disables the browser cache."
  115. },
  116. "homepageOverride": {
  117. "$ref": "types.Setting",
  118. "description": "Returns the value of the overridden home page. Read-only."
  119. },
  120. "imageAnimationBehavior": {
  121. "$ref": "types.Setting",
  122. "description": "Controls the behaviour of image animation in the browser. This setting's value is of type ImageAnimationBehavior, defaulting to <code>normal</code>."
  123. },
  124. "newTabPageOverride": {
  125. "$ref": "types.Setting",
  126. "description": "Returns the value of the overridden new tab page. Read-only."
  127. + },
  128. + "contextMenuShowEvent": {
  129. + "$ref": "types.Setting",
  130. + "description": "Controls after which mouse event context menus popup. This setting's value is of type ContextMenuMouseEvent, which has possible values of <code>mouseup</code> and <code>mousedown</code>."
  131. }
  132. }
  133. }
  134. diff --git a/toolkit/components/extensions/test/xpcshell/test_ext_browserSettings.js b/toolkit/components/extensions/test/xpcshell/test_ext_browserSettings.js
  135. index 5c441df3e4198671..7e9c1576a723dfc6 100644
  136. --- a/toolkit/components/extensions/test/xpcshell/test_ext_browserSettings.js
  137. +++ b/toolkit/components/extensions/test/xpcshell/test_ext_browserSettings.js
  138. @@ -24,13 +24,20 @@ add_task(async function test_browser_settings() {
  139. "browser.cache.memory.enable": true,
  140. "dom.popup_allowed_events": Preferences.get("dom.popup_allowed_events"),
  141. "image.animation_mode": "none",
  142. + "ui.context_menus.after_mouseup": false,
  143. };
  144. async function background() {
  145. browser.test.onMessage.addListener(async (msg, apiName, value) => {
  146. let apiObj = browser.browserSettings[apiName];
  147. - await apiObj.set({value});
  148. - browser.test.sendMessage("settingData", await apiObj.get({}));
  149. + let result = await apiObj.set({value});
  150. + if (msg === "set") {
  151. + browser.test.assertTrue(result, "set returns true.");
  152. + browser.test.sendMessage("settingData", await apiObj.get({}));
  153. + } else {
  154. + browser.test.assertFalse(result, "set returns false for a no-op.");
  155. + browser.test.sendMessage("no-op set");
  156. + }
  157. });
  158. }
  159. @@ -69,33 +76,82 @@ add_task(async function test_browser_settings() {
  160. }
  161. }
  162. + async function testNoOpSetting(setting, value, expected) {
  163. + extension.sendMessage("setNoOp", setting, value);
  164. + await extension.awaitMessage("no-op set");
  165. + for (let pref in expected) {
  166. + equal(Preferences.get(pref), expected[pref], `${pref} set correctly for ${value}`);
  167. + }
  168. + }
  169. +
  170. await testSetting(
  171. "cacheEnabled", false,
  172. {
  173. "browser.cache.disk.enable": false,
  174. "browser.cache.memory.enable": false,
  175. });
  176. await testSetting(
  177. "cacheEnabled", true,
  178. {
  179. "browser.cache.disk.enable": true,
  180. "browser.cache.memory.enable": true,
  181. });
  182. await testSetting(
  183. "allowPopupsForUserEvents", false,
  184. {"dom.popup_allowed_events": ""});
  185. await testSetting(
  186. "allowPopupsForUserEvents", true,
  187. {"dom.popup_allowed_events": PREFS["dom.popup_allowed_events"]});
  188. for (let value of ["normal", "none", "once"]) {
  189. await testSetting(
  190. "imageAnimationBehavior", value,
  191. {"image.animation_mode": value});
  192. }
  193. - await extension.unload();
  194. + // This setting is a no-op on Android.
  195. + if (AppConstants.platform === "android") {
  196. + await testNoOpSetting("contextMenuShowEvent", "mouseup",
  197. + {"ui.context_menus.after_mouseup": false});
  198. + } else {
  199. + await testSetting(
  200. + "contextMenuShowEvent", "mouseup",
  201. + {"ui.context_menus.after_mouseup": true});
  202. + }
  203. + // "mousedown" is also a no-op on Windows.
  204. + if (["android", "win"].includes(AppConstants.platform)) {
  205. + await testNoOpSetting("contextMenuShowEvent", "mousedown",
  206. + {"ui.context_menus.after_mouseup": AppConstants.platform === "win"});
  207. + } else {
  208. + await testSetting(
  209. + "contextMenuShowEvent", "mousedown",
  210. + {"ui.context_menus.after_mouseup": false});
  211. + }
  212. +
  213. + await extension.unload();
  214. await promiseShutdownManager();
  215. });
  216. +
  217. +add_task(async function test_bad_value() {
  218. + async function background() {
  219. + await browser.test.assertRejects(
  220. + browser.browserSettings.contextMenuShowEvent.set({value: "bad"}),
  221. + /bad is not a valid value for contextMenuShowEvent/,
  222. + "contextMenuShowEvent.set rejects with an invalid value.");
  223. +
  224. + browser.test.sendMessage("done");
  225. + }
  226. +
  227. + let extension = ExtensionTestUtils.loadExtension({
  228. + background,
  229. + manifest: {
  230. + permissions: ["browserSettings"],
  231. + },
  232. + });
  233. +
  234. + await extension.startup();
  235. + await extension.awaitMessage("done");
  236. + await extension.unload();
  237. +});
  238. --
  239. 2.15.1