forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			212 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			212 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| const PREF_RESTORE_ON_DEMAND = "browser.sessionstore.restore_on_demand";
 | |
| 
 | |
| requestLongerTimeout(2);
 | |
| 
 | |
| add_task(async function test() {
 | |
|   Services.prefs.setBoolPref(PREF_RESTORE_ON_DEMAND, false);
 | |
|   registerCleanupFunction(function () {
 | |
|     Services.prefs.clearUserPref(PREF_RESTORE_ON_DEMAND);
 | |
|   });
 | |
| 
 | |
|   // The first state will be loaded using setBrowserState, followed by the 2nd
 | |
|   // state also being loaded using setBrowserState, interrupting the first restore.
 | |
|   let state1 = {
 | |
|     windows: [
 | |
|       {
 | |
|         tabs: [
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#1", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#2", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#3", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#4", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|         ],
 | |
|         selected: 1,
 | |
|       },
 | |
|       {
 | |
|         tabs: [
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#1", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#2", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#3", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#4", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|         ],
 | |
|         selected: 3,
 | |
|       },
 | |
|     ],
 | |
|   };
 | |
|   let state2 = {
 | |
|     windows: [
 | |
|       {
 | |
|         tabs: [
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#5", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#6", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#7", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.org#8", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|         ],
 | |
|         selected: 3,
 | |
|       },
 | |
|       {
 | |
|         tabs: [
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#5", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#6", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#7", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|           {
 | |
|             entries: [
 | |
|               { url: "http://example.com#8", triggeringPrincipal_base64 },
 | |
|             ],
 | |
|             extData: { uniq: r() },
 | |
|           },
 | |
|         ],
 | |
|         selected: 1,
 | |
|       },
 | |
|     ],
 | |
|   };
 | |
| 
 | |
|   // interruptedAfter will be set after the selected tab from each window have loaded.
 | |
|   let interruptedAfter = 0;
 | |
|   let loadedWindow1 = false;
 | |
|   let loadedWindow2 = false;
 | |
|   let numTabs = state2.windows[0].tabs.length + state2.windows[1].tabs.length;
 | |
| 
 | |
|   let loadCount = 0;
 | |
|   let promiseRestoringTabs = new Promise(resolve => {
 | |
|     gProgressListener.setCallback(function (
 | |
|       aBrowser,
 | |
|       aNeedRestore,
 | |
|       aRestoring,
 | |
|       aRestored
 | |
|     ) {
 | |
|       loadCount++;
 | |
| 
 | |
|       if (
 | |
|         aBrowser.currentURI.spec == state1.windows[0].tabs[2].entries[0].url
 | |
|       ) {
 | |
|         loadedWindow1 = true;
 | |
|       }
 | |
|       if (
 | |
|         aBrowser.currentURI.spec == state1.windows[1].tabs[0].entries[0].url
 | |
|       ) {
 | |
|         loadedWindow2 = true;
 | |
|       }
 | |
| 
 | |
|       if (!interruptedAfter && loadedWindow1 && loadedWindow2) {
 | |
|         interruptedAfter = loadCount;
 | |
|         ss.setBrowserState(JSON.stringify(state2));
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (loadCount < numTabs + interruptedAfter) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // We don't actually care about load order in this test, just that they all
 | |
|       // do load.
 | |
|       is(loadCount, numTabs + interruptedAfter, "all tabs were restored");
 | |
|       is(aNeedRestore, 0, "there are no tabs left needing restore");
 | |
| 
 | |
|       // Remove the progress listener.
 | |
|       gProgressListener.unsetCallback();
 | |
|       resolve();
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   // We also want to catch the extra windows (there should be 2), so we need to observe domwindowopened
 | |
|   Services.ww.registerNotification(function observer(aSubject, aTopic, aData) {
 | |
|     if (aTopic == "domwindowopened") {
 | |
|       let win = aSubject;
 | |
|       win.addEventListener(
 | |
|         "load",
 | |
|         function () {
 | |
|           Services.ww.unregisterNotification(observer);
 | |
|           win.gBrowser.addTabsProgressListener(gProgressListener);
 | |
|         },
 | |
|         { once: true }
 | |
|       );
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   let backupState = ss.getBrowserState();
 | |
|   ss.setBrowserState(JSON.stringify(state1));
 | |
|   await promiseRestoringTabs;
 | |
| 
 | |
|   // Cleanup.
 | |
|   await promiseAllButPrimaryWindowClosed();
 | |
|   await promiseBrowserState(backupState);
 | |
| });
 | 
