forked from mirrors/gecko-dev
		
	Bug 1808932: Take abandonment event even when closing the result view while the input does not have focus. r=mak
Differential Revision: https://phabricator.services.mozilla.com/D167641
This commit is contained in:
		
							parent
							
								
									021a6d01e4
								
							
						
					
					
						commit
						025ed7282a
					
				
					 3 changed files with 142 additions and 0 deletions
				
			
		|  | @ -3210,11 +3210,24 @@ export class UrlbarInput { | |||
|         if (event.target.closest("tab")) { | ||||
|           break; | ||||
|         } | ||||
| 
 | ||||
|         // Close the view when clicking on toolbars and other UI pieces that
 | ||||
|         // might not automatically remove focus from the input.
 | ||||
|         // Respect the autohide preference for easier inspecting/debugging via
 | ||||
|         // the browser toolbox.
 | ||||
|         if (!lazy.UrlbarPrefs.get("ui.popup.disable_autohide")) { | ||||
|           if (this.view.isOpen && !this.hasAttribute("focused")) { | ||||
|             // In this case, as blur event never happen from the inputField, we
 | ||||
|             // record abandonment event explicitly.
 | ||||
|             let blurEvent = new FocusEvent("blur", { | ||||
|               relatedTarget: this.inputField, | ||||
|             }); | ||||
|             this.controller.engagementEvent.record(blurEvent, { | ||||
|               searchString: this._lastSearchString, | ||||
|               searchSource: this.getSearchSource(blurEvent), | ||||
|             }); | ||||
|           } | ||||
| 
 | ||||
|           this.view.close(); | ||||
|         } | ||||
|         break; | ||||
|  |  | |||
|  | @ -6,6 +6,7 @@ | |||
| support-files = | ||||
|   head.js | ||||
| 
 | ||||
| [browser_glean_telemetry_abandonment.js] | ||||
| [browser_glean_telemetry_engagement.js] | ||||
| [browser_interventions.js] | ||||
| [browser_picks.js] | ||||
|  |  | |||
|  | @ -0,0 +1,128 @@ | |||
| /* Any copyright is dedicated to the Public Domain. | ||||
|  * http://creativecommons.org/publicdomain/zero/1.0/ */
 | ||||
| 
 | ||||
| "use strict"; | ||||
| 
 | ||||
| // Test for abandonment telemetry for tips using Glean.
 | ||||
| 
 | ||||
| add_setup(async function() { | ||||
|   await SpecialPowers.pushPrefEnv({ | ||||
|     set: [ | ||||
|       ["browser.urlbar.searchEngagementTelemetry.enabled", true], | ||||
|       ["browser.urlbar.searchTips.test.ignoreShowLimits", true], | ||||
|       ["browser.urlbar.showSearchTerms.featureGate", true], | ||||
|     ], | ||||
|   }); | ||||
|   const engine = await SearchTestUtils.promiseNewSearchEngine({ | ||||
|     url: | ||||
|       "chrome://mochitests/content/browser/browser/components/urlbar/tests/browser/searchSuggestionEngine.xml", | ||||
|   }); | ||||
|   const originalDefaultEngine = await Services.search.getDefault(); | ||||
|   await Services.search.setDefault( | ||||
|     engine, | ||||
|     Ci.nsISearchService.CHANGE_REASON_UNKNOWN | ||||
|   ); | ||||
|   await Services.search.moveEngine(engine, 0); | ||||
| 
 | ||||
|   registerCleanupFunction(async function() { | ||||
|     await SpecialPowers.popPrefEnv(); | ||||
|     await Services.search.setDefault( | ||||
|       originalDefaultEngine, | ||||
|       Ci.nsISearchService.CHANGE_REASON_UNKNOWN | ||||
|     ); | ||||
|     resetSearchTipsProvider(); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| add_task(async function tip_persist() { | ||||
|   await doTest(async browser => { | ||||
|     await showPersistSearchTip("test"); | ||||
|     gURLBar.focus(); | ||||
|     await UrlbarTestUtils.promisePopupClose(window, () => { | ||||
|       gURLBar.blur(); | ||||
|     }); | ||||
| 
 | ||||
|     assertGleanTelemetry([{ results: "tip_persist" }]); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| add_task(async function mouse_down_with_tip() { | ||||
|   await doTest(async browser => { | ||||
|     await showPersistSearchTip("test"); | ||||
|     await UrlbarTestUtils.promisePopupClose(window, () => { | ||||
|       EventUtils.synthesizeMouseAtCenter(browser, {}); | ||||
|     }); | ||||
| 
 | ||||
|     assertGleanTelemetry([{ results: "tip_persist" }]); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| add_task(async function mouse_down_without_tip() { | ||||
|   await doTest(async browser => { | ||||
|     EventUtils.synthesizeMouseAtCenter(browser, {}); | ||||
| 
 | ||||
|     assertGleanTelemetry([]); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| function assertGleanTelemetry(expectedExtraList) { | ||||
|   const telemetries = Glean.urlbar.abandonment.testGetValue() ?? []; | ||||
|   Assert.equal(telemetries.length, expectedExtraList.length); | ||||
| 
 | ||||
|   for (let i = 0; i < telemetries.length; i++) { | ||||
|     const telemetry = telemetries[i]; | ||||
|     Assert.equal(telemetry.category, "urlbar"); | ||||
|     Assert.equal(telemetry.name, "abandonment"); | ||||
| 
 | ||||
|     const expectedExtra = expectedExtraList[i]; | ||||
|     for (const key of Object.keys(expectedExtra)) { | ||||
|       Assert.equal( | ||||
|         telemetry.extra[key], | ||||
|         expectedExtra[key], | ||||
|         `${key} is correct` | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| async function doEnter() { | ||||
|   const onLoad = BrowserTestUtils.browserLoaded(gBrowser.selectedBrowser); | ||||
|   EventUtils.synthesizeKey("KEY_Enter"); | ||||
|   await onLoad; | ||||
| } | ||||
| 
 | ||||
| async function doTest(testFn) { | ||||
|   await Services.fog.testFlushAllChildren(); | ||||
|   Services.fog.testResetFOG(); | ||||
|   gURLBar.controller.engagementEvent.reset(); | ||||
| 
 | ||||
|   await BrowserTestUtils.withNewTab(gBrowser, testFn); | ||||
| } | ||||
| 
 | ||||
| async function openPopup(input) { | ||||
|   await UrlbarTestUtils.promisePopupOpen(window, async () => { | ||||
|     EventUtils.synthesizeMouseAtCenter(gURLBar.inputField, {}); | ||||
|     await BrowserTestUtils.waitForCondition( | ||||
|       () => | ||||
|         gURLBar.inputField.ownerDocument.activeElement === gURLBar.inputField | ||||
|     ); | ||||
|     for (let i = 0; i < input.length; i++) { | ||||
|       EventUtils.synthesizeKey(input.charAt(i)); | ||||
|     } | ||||
|   }); | ||||
|   await UrlbarTestUtils.promiseSearchComplete(window); | ||||
| } | ||||
| 
 | ||||
| async function showPersistSearchTip(word) { | ||||
|   await openPopup(word); | ||||
|   await doEnter(); | ||||
|   await BrowserTestUtils.waitForCondition(async () => { | ||||
|     for (let i = 0; i < UrlbarTestUtils.getResultCount(window); i++) { | ||||
|       const detail = await UrlbarTestUtils.getDetailsOfResultAt(window, i); | ||||
|       if (detail.result.payload?.type === "searchTip_persist") { | ||||
|         return true; | ||||
|       } | ||||
|     } | ||||
|     return false; | ||||
|   }); | ||||
| } | ||||
		Loading…
	
		Reference in a new issue
	
	 Daisuke Akatsuka
						Daisuke Akatsuka