mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
5b68485189
--HG-- extra : rebase_source : 7f3d38e4ad52f6e54512ea899466224fe836f762
882 lines
35 KiB
JavaScript
882 lines
35 KiB
JavaScript
/* ***** BEGIN LICENSE BLOCK *****
|
||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||
*
|
||
* The contents of this file are subject to the Mozilla Public License Version
|
||
* 1.1 (the "License"); you may not use this file except in compliance with
|
||
* the License. You may obtain a copy of the License at
|
||
* http://www.mozilla.org/MPL/
|
||
*
|
||
* Software distributed under the License is distributed on an "AS IS" basis,
|
||
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||
* for the specific language governing rights and limitations under the
|
||
* License.
|
||
*
|
||
* The Original Code is sessionstore test code.
|
||
*
|
||
* The Initial Developer of the Original Code is
|
||
* Mozilla Foundation.
|
||
* Portions created by the Initial Developer are Copyright (C) 2010
|
||
* the Initial Developer. All Rights Reserved.
|
||
*
|
||
* Contributor(s):
|
||
* Paul O’Shannessy <paul@oshannessy.com>
|
||
*
|
||
* Alternatively, the contents of this file may be used under the terms of
|
||
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||
* of those above. If you wish to allow use of your version of this file only
|
||
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||
* use your version of this file under the terms of the MPL, indicate your
|
||
* decision by deleting the provisions above and replace them with the notice
|
||
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||
* the provisions above, a recipient may use your version of this file under
|
||
* the terms of any one of the MPL, the GPL or the LGPL.
|
||
*
|
||
* ***** END LICENSE BLOCK ***** */
|
||
|
||
let stateBackup = ss.getBrowserState();
|
||
|
||
const TAB_STATE_NEEDS_RESTORE = 1;
|
||
const TAB_STATE_RESTORING = 2;
|
||
|
||
function test() {
|
||
/** Test for Bug 586068 - Cascade page loads when restoring **/
|
||
waitForExplicitFinish();
|
||
// This test does a lot of window opening / closing and waiting for loads.
|
||
// In order to prevent timeouts, we'll extend the default that mochitest uses.
|
||
requestLongerTimeout(4);
|
||
runNextTest();
|
||
}
|
||
|
||
// test_reloadCascade, test_reloadReload are generated tests that are run out
|
||
// of cycle (since they depend on current state). They're listed in [tests] here
|
||
// so that it is obvious when they run in respect to the other tests.
|
||
let tests = [test_cascade, test_select, test_multiWindowState,
|
||
test_setWindowStateNoOverwrite, test_setWindowStateOverwrite,
|
||
test_setBrowserStateInterrupted, test_reload,
|
||
/* test_reloadReload, */ test_reloadCascadeSetup,
|
||
/* test_reloadCascade, */ test_apptabs_only,
|
||
test_restore_apptabs_ondemand];
|
||
function runNextTest() {
|
||
// Reset the pref
|
||
try {
|
||
Services.prefs.clearUserPref("browser.sessionstore.restore_on_demand");
|
||
Services.prefs.clearUserPref("browser.sessionstore.restore_pinned_tabs_on_demand");
|
||
} catch (e) {}
|
||
|
||
// set an empty state & run the next test, or finish
|
||
if (tests.length) {
|
||
// Enumerate windows and close everything but our primary window. We can't
|
||
// use waitForFocus() because apparently it's buggy. See bug 599253.
|
||
var windowsEnum = Services.wm.getEnumerator("navigator:browser");
|
||
while (windowsEnum.hasMoreElements()) {
|
||
var currentWindow = windowsEnum.getNext();
|
||
if (currentWindow != window) {
|
||
currentWindow.close();
|
||
}
|
||
}
|
||
|
||
ss.setBrowserState(JSON.stringify({ windows: [{ tabs: [{ url: 'about:blank' }] }] }));
|
||
let currentTest = tests.shift();
|
||
info("running " + currentTest.name);
|
||
executeSoon(currentTest);
|
||
}
|
||
else {
|
||
ss.setBrowserState(stateBackup);
|
||
executeSoon(finish);
|
||
}
|
||
}
|
||
|
||
|
||
function test_cascade() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
dump("\n\nload: " + aBrowser.currentURI.spec + "\n" + JSON.stringify(countTabs()) + "\n\n");
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_cascade_progressCallback();
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com" }], extData: { "uniq": r() } }
|
||
] }] };
|
||
|
||
let loadCount = 0;
|
||
// Since our progress listener is fired before the one in sessionstore, our
|
||
// expected counts look a little weird. This is because we inspect the state
|
||
// before sessionstore has marked the tab as finished restoring and before it
|
||
// starts restoring the next tab
|
||
let expectedCounts = [
|
||
[3, 3, 0],
|
||
[2, 3, 1],
|
||
[1, 3, 2],
|
||
[0, 3, 3],
|
||
[0, 2, 4],
|
||
[0, 1, 5]
|
||
];
|
||
|
||
function test_cascade_progressCallback() {
|
||
loadCount++;
|
||
let counts = countTabs();
|
||
let expected = expectedCounts[loadCount - 1];
|
||
|
||
is(counts[0], expected[0], "test_cascade: load " + loadCount + " - # tabs that need to be restored");
|
||
is(counts[1], expected[1], "test_cascade: load " + loadCount + " - # tabs that are restoring");
|
||
is(counts[2], expected[2], "test_cascade: load " + loadCount + " - # tabs that has been restored");
|
||
|
||
if (loadCount < state.windows[0].tabs.length)
|
||
return;
|
||
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
runNextTest();
|
||
}
|
||
|
||
// This progress listener will get attached before the listener in session store.
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
function test_select() {
|
||
// Set the pref to true so we know exactly how many tabs should be restoring at
|
||
// any given time. This guarantees that a finishing load won't start another.
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", true);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_select_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org" }], extData: { "uniq": r() } }
|
||
], selected: 1 }] };
|
||
|
||
let loadCount = 0;
|
||
// expectedCounts looks a little wierd for the test case, but it works. See
|
||
// comment in test_cascade for an explanation
|
||
let expectedCounts = [
|
||
[5, 1, 0],
|
||
[4, 1, 1],
|
||
[3, 1, 2],
|
||
[2, 1, 3],
|
||
[1, 1, 4],
|
||
[0, 1, 5]
|
||
];
|
||
let tabOrder = [0, 5, 1, 4, 3, 2];
|
||
|
||
function test_select_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
let counts = countTabs();
|
||
let expected = expectedCounts[loadCount - 1];
|
||
|
||
is(counts[0], expected[0], "test_select: load " + loadCount + " - # tabs that need to be restored");
|
||
is(counts[1], expected[1], "test_select: load " + loadCount + " - # tabs that are restoring");
|
||
is(counts[2], expected[2], "test_select: load " + loadCount + " - # tabs that has been restored");
|
||
|
||
if (loadCount < state.windows[0].tabs.length) {
|
||
// double check that this tab was the right one
|
||
let expectedData = state.windows[0].tabs[tabOrder[loadCount - 1]].extData.uniq;
|
||
let tab;
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
if (!tab && window.gBrowser.tabs[i].linkedBrowser == aBrowser)
|
||
tab = window.gBrowser.tabs[i];
|
||
}
|
||
is(ss.getTabValue(tab, "uniq"), expectedData, "test_select: load " + loadCount + " - correct tab was restored");
|
||
|
||
// select the next tab
|
||
window.gBrowser.selectTabAtIndex(tabOrder[loadCount]);
|
||
return;
|
||
}
|
||
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
runNextTest();
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
function test_multiWindowState() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
// We only care about load events when the tab still has
|
||
// __SS_restoreState == TAB_STATE_RESTORING on it.
|
||
// Since our listener is attached before the sessionstore one, this works out.
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_multiWindowState_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
// The first window will be put into the already open window and the second
|
||
// window will be opened with _openWindowWithState, which is the source of the problem.
|
||
let state = { windows: [
|
||
{
|
||
tabs: [
|
||
{ entries: [{ url: "http://example.org#0" }], extData: { "uniq": r() } }
|
||
],
|
||
selected: 1
|
||
},
|
||
{
|
||
tabs: [
|
||
{ entries: [{ url: "http://example.com#1" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#2" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#3" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#4" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#6" }], extData: { "uniq": r() } }
|
||
],
|
||
selected: 4
|
||
}
|
||
] };
|
||
let numTabs = state.windows[0].tabs.length + state.windows[1].tabs.length;
|
||
|
||
let loadCount = 0;
|
||
function test_multiWindowState_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
if (loadCount < numTabs)
|
||
return;
|
||
|
||
// We don't actually care about load order in this test, just that they all
|
||
// do load.
|
||
is(loadCount, numTabs, "test_multiWindowState: all tabs were restored");
|
||
let count = countTabs();
|
||
is(count[0], 0,
|
||
"test_multiWindowState: there are no tabs left needing restore");
|
||
|
||
// Remove the progress listener from this window, it will be removed from
|
||
// theWin when that window is closed (in setBrowserState).
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
runNextTest();
|
||
}
|
||
|
||
// We also want to catch the 2nd window, so we need to observe domwindowopened
|
||
function windowObserver(aSubject, aTopic, aData) {
|
||
let theWin = aSubject.QueryInterface(Ci.nsIDOMWindow);
|
||
if (aTopic == "domwindowopened") {
|
||
theWin.addEventListener("load", function() {
|
||
theWin.removeEventListener("load", arguments.callee, false);
|
||
|
||
Services.ww.unregisterNotification(windowObserver);
|
||
theWin.gBrowser.addTabsProgressListener(progressListener);
|
||
}, false);
|
||
}
|
||
}
|
||
Services.ww.registerNotification(windowObserver);
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
function test_setWindowStateNoOverwrite() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
// We only care about load events when the tab still has
|
||
// __SS_restoreState == TAB_STATE_RESTORING on it.
|
||
// Since our listener is attached before the sessionstore one, this works out.
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_setWindowStateNoOverwrite_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
// We'll use 2 states so that we can make sure calling setWindowState doesn't
|
||
// wipe out currently restoring data.
|
||
let state1 = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.com#1" }] },
|
||
{ entries: [{ url: "http://example.com#2" }] },
|
||
{ entries: [{ url: "http://example.com#3" }] },
|
||
{ entries: [{ url: "http://example.com#4" }] },
|
||
{ entries: [{ url: "http://example.com#5" }] },
|
||
] }] };
|
||
let state2 = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org#1" }] },
|
||
{ entries: [{ url: "http://example.org#2" }] },
|
||
{ entries: [{ url: "http://example.org#3" }] },
|
||
{ entries: [{ url: "http://example.org#4" }] },
|
||
{ entries: [{ url: "http://example.org#5" }] }
|
||
] }] };
|
||
|
||
let numTabs = state1.windows[0].tabs.length + state2.windows[0].tabs.length;
|
||
|
||
let loadCount = 0;
|
||
function test_setWindowStateNoOverwrite_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
// When loadCount == 2, we'll also restore state2 into the window
|
||
if (loadCount == 2)
|
||
ss.setWindowState(window, JSON.stringify(state2), false);
|
||
|
||
if (loadCount < numTabs)
|
||
return;
|
||
|
||
// We don't actually care about load order in this test, just that they all
|
||
// do load.
|
||
is(loadCount, numTabs, "test_setWindowStateNoOverwrite: all tabs were restored");
|
||
// window.__SS_tabsToRestore isn't decremented until after the progress
|
||
// listener is called. Since we get in here before that, we still expect
|
||
// the count to be 1.
|
||
is(window.__SS_tabsToRestore, 1,
|
||
"test_setWindowStateNoOverwrite: window doesn't think there are more tabs to restore");
|
||
let count = countTabs();
|
||
is(count[0], 0,
|
||
"test_setWindowStateNoOverwrite: there are no tabs left needing restore");
|
||
|
||
// Remove the progress listener from this window, it will be removed from
|
||
// theWin when that window is closed (in setBrowserState).
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
|
||
runNextTest();
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setWindowState(window, JSON.stringify(state1), true);
|
||
}
|
||
|
||
|
||
function test_setWindowStateOverwrite() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
// We only care about load events when the tab still has
|
||
// __SS_restoreState == TAB_STATE_RESTORING on it.
|
||
// Since our listener is attached before the sessionstore one, this works out.
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_setWindowStateOverwrite_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
// We'll use 2 states so that we can make sure calling setWindowState doesn't
|
||
// wipe out currently restoring data.
|
||
let state1 = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.com#1" }] },
|
||
{ entries: [{ url: "http://example.com#2" }] },
|
||
{ entries: [{ url: "http://example.com#3" }] },
|
||
{ entries: [{ url: "http://example.com#4" }] },
|
||
{ entries: [{ url: "http://example.com#5" }] },
|
||
] }] };
|
||
let state2 = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org#1" }] },
|
||
{ entries: [{ url: "http://example.org#2" }] },
|
||
{ entries: [{ url: "http://example.org#3" }] },
|
||
{ entries: [{ url: "http://example.org#4" }] },
|
||
{ entries: [{ url: "http://example.org#5" }] }
|
||
] }] };
|
||
|
||
let numTabs = 2 + state2.windows[0].tabs.length;
|
||
|
||
let loadCount = 0;
|
||
function test_setWindowStateOverwrite_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
// When loadCount == 2, we'll also restore state2 into the window
|
||
if (loadCount == 2)
|
||
ss.setWindowState(window, JSON.stringify(state2), true);
|
||
|
||
if (loadCount < numTabs)
|
||
return;
|
||
|
||
// We don't actually care about load order in this test, just that they all
|
||
// do load.
|
||
is(loadCount, numTabs, "test_setWindowStateOverwrite: all tabs were restored");
|
||
// window.__SS_tabsToRestore isn't decremented until after the progress
|
||
// listener is called. Since we get in here before that, we still expect
|
||
// the count to be 1.
|
||
is(window.__SS_tabsToRestore, 1,
|
||
"test_setWindowStateOverwrite: window doesn't think there are more tabs to restore");
|
||
let count = countTabs();
|
||
is(count[0], 0,
|
||
"test_setWindowStateOverwrite: there are no tabs left needing restore");
|
||
|
||
// Remove the progress listener from this window, it will be removed from
|
||
// theWin when that window is closed (in setBrowserState).
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
|
||
runNextTest();
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setWindowState(window, JSON.stringify(state1), true);
|
||
}
|
||
|
||
|
||
function test_setBrowserStateInterrupted() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
// We only care about load events when the tab still has
|
||
// __SS_restoreState == TAB_STATE_RESTORING on it.
|
||
// Since our listener is attached before the sessionstore one, this works out.
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_setBrowserStateInterrupted_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
// 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" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#2" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#3" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#4" }], extData: { "uniq": r() } }
|
||
],
|
||
selected: 1
|
||
},
|
||
{
|
||
tabs: [
|
||
{ entries: [{ url: "http://example.com#1" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#2" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#3" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#4" }], extData: { "uniq": r() } },
|
||
],
|
||
selected: 3
|
||
}
|
||
] };
|
||
let state2 = { windows: [
|
||
{
|
||
tabs: [
|
||
{ entries: [{ url: "http://example.org#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#6" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#7" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org#8" }], extData: { "uniq": r() } }
|
||
],
|
||
selected: 3
|
||
},
|
||
{
|
||
tabs: [
|
||
{ entries: [{ url: "http://example.com#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#6" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#7" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com#8" }], 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;
|
||
function test_setBrowserStateInterrupted_progressCallback(aBrowser) {
|
||
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,
|
||
"test_setBrowserStateInterrupted: all tabs were restored");
|
||
let count = countTabs();
|
||
is(count[0], 0,
|
||
"test_setBrowserStateInterrupted: there are no tabs left needing restore");
|
||
|
||
// Remove the progress listener from this window, it will be removed from
|
||
// theWin when that window is closed (in setBrowserState).
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
Services.ww.unregisterNotification(windowObserver);
|
||
runNextTest();
|
||
}
|
||
|
||
// We also want to catch the extra windows (there should be 2), so we need to observe domwindowopened
|
||
function windowObserver(aSubject, aTopic, aData) {
|
||
let theWin = aSubject.QueryInterface(Ci.nsIDOMWindow);
|
||
if (aTopic == "domwindowopened") {
|
||
theWin.addEventListener("load", function() {
|
||
theWin.removeEventListener("load", arguments.callee, false);
|
||
|
||
Services.ww.unregisterNotification(windowObserver);
|
||
theWin.gBrowser.addTabsProgressListener(progressListener);
|
||
}, false);
|
||
}
|
||
}
|
||
Services.ww.registerNotification(windowObserver);
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state1));
|
||
}
|
||
|
||
|
||
function test_reload() {
|
||
// Set the pref to true so we know exactly how many tabs should be restoring at
|
||
// any given time. This guarantees that a finishing load won't start another.
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", true);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_reload_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org/#1" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#2" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#3" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#4" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#6" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#7" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#8" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#9" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#10" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#11" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#12" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#13" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#14" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#15" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#16" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#17" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#18" }], extData: { "uniq": r() } }
|
||
], selected: 1 }] };
|
||
|
||
let loadCount = 0;
|
||
function test_reload_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
is(aBrowser.currentURI.spec, state.windows[0].tabs[loadCount - 1].entries[0].url,
|
||
"test_reload: load " + loadCount + " - browser loaded correct url");
|
||
|
||
if (loadCount <= state.windows[0].tabs.length) {
|
||
// double check that this tab was the right one
|
||
let expectedData = state.windows[0].tabs[loadCount - 1].extData.uniq;
|
||
let tab;
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
if (!tab && window.gBrowser.tabs[i].linkedBrowser == aBrowser)
|
||
tab = window.gBrowser.tabs[i];
|
||
}
|
||
is(ss.getTabValue(tab, "uniq"), expectedData,
|
||
"test_reload: load " + loadCount + " - correct tab was restored");
|
||
|
||
if (loadCount == state.windows[0].tabs.length) {
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
executeSoon(function() {
|
||
_test_reloadAfter("test_reloadReload", state, runNextTest);
|
||
});
|
||
}
|
||
else {
|
||
// reload the next tab
|
||
window.gBrowser.reloadTab(window.gBrowser.tabs[loadCount]);
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
// This doesn't actually test anything, just does a cascaded restore with default
|
||
// settings. This really just sets up to test that reloads work.
|
||
function test_reloadCascadeSetup() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", false);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_cascadeReloadSetup_progressCallback();
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.com/#1" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com/#2" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com/#3" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com/#4" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com/#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.com/#6" }], extData: { "uniq": r() } }
|
||
] }] };
|
||
|
||
let loadCount = 0;
|
||
function test_cascadeReloadSetup_progressCallback() {
|
||
loadCount++;
|
||
if (loadCount < state.windows[0].tabs.length)
|
||
return;
|
||
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
executeSoon(function() {
|
||
_test_reloadAfter("test_reloadCascade", state, runNextTest);
|
||
});
|
||
}
|
||
|
||
// This progress listener will get attached before the listener in session store.
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
// This is a generic function that will attempt to reload each test. We do this
|
||
// a couple times, so make it utilitarian.
|
||
// This test expects that aState contains a single window and that each tab has
|
||
// a unique extData value eg. { "uniq": value }.
|
||
function _test_reloadAfter(aTestName, aState, aCallback) {
|
||
info("starting " + aTestName);
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_reloadAfter_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
// Simulate a left mouse button click with no modifiers, which is what
|
||
// Command-R, or clicking reload does.
|
||
let fakeEvent = {
|
||
button: 0,
|
||
metaKey: false,
|
||
altKey: false,
|
||
ctrlKey: false,
|
||
shiftKey: false,
|
||
}
|
||
|
||
let loadCount = 0;
|
||
function test_reloadAfter_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
if (loadCount <= aState.windows[0].tabs.length) {
|
||
// double check that this tab was the right one
|
||
let expectedData = aState.windows[0].tabs[loadCount - 1].extData.uniq;
|
||
let tab;
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
if (!tab && window.gBrowser.tabs[i].linkedBrowser == aBrowser)
|
||
tab = window.gBrowser.tabs[i];
|
||
}
|
||
is(ss.getTabValue(tab, "uniq"), expectedData,
|
||
aTestName + ": load " + loadCount + " - correct tab was reloaded");
|
||
|
||
if (loadCount == aState.windows[0].tabs.length) {
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
aCallback();
|
||
}
|
||
else {
|
||
// reload the next tab
|
||
window.gBrowser.selectTabAtIndex(loadCount);
|
||
BrowserReloadOrDuplicate(fakeEvent);
|
||
}
|
||
}
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
BrowserReloadOrDuplicate(fakeEvent);
|
||
}
|
||
|
||
|
||
// This test ensures that app tabs are restored regardless of restore_on_demand
|
||
function test_apptabs_only() {
|
||
// Set the pref to true so we know exactly how many tabs should be restoring at
|
||
// any given time. This guarantees that a finishing load won't start another.
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", true);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_apptabs_only_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org/#1" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#2" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#3" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#4" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#6" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#7" }], extData: { "uniq": r() } },
|
||
], selected: 5 }] };
|
||
|
||
let loadCount = 0;
|
||
function test_apptabs_only_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
// We'll make sure that the loads we get come from pinned tabs or the
|
||
// the selected tab.
|
||
|
||
// get the tab
|
||
let tab;
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
if (!tab && window.gBrowser.tabs[i].linkedBrowser == aBrowser)
|
||
tab = window.gBrowser.tabs[i];
|
||
}
|
||
|
||
ok(tab.pinned || gBrowser.selectedTab == tab,
|
||
"test_apptabs_only: load came from pinned or selected tab");
|
||
|
||
// We should get 4 loads: 3 app tabs + 1 normal selected tab
|
||
if (loadCount < 4)
|
||
return;
|
||
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
runNextTest();
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
// This test ensures that app tabs are not restored when restore_pinned_tabs_on_demand is set
|
||
function test_restore_apptabs_ondemand() {
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_on_demand", true);
|
||
Services.prefs.setBoolPref("browser.sessionstore.restore_pinned_tabs_on_demand", true);
|
||
|
||
// We have our own progress listener for this test, which we'll attach before our state is set
|
||
let progressListener = {
|
||
onStateChange: function (aBrowser, aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
if (aBrowser.__SS_restoreState == TAB_STATE_RESTORING &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_STOP &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
|
||
aStateFlags & Ci.nsIWebProgressListener.STATE_IS_WINDOW)
|
||
test_restore_apptabs_ondemand_progressCallback(aBrowser);
|
||
}
|
||
}
|
||
|
||
let state = { windows: [{ tabs: [
|
||
{ entries: [{ url: "http://example.org/#1" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#2" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#3" }], extData: { "uniq": r() }, pinned: true },
|
||
{ entries: [{ url: "http://example.org/#4" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#5" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#6" }], extData: { "uniq": r() } },
|
||
{ entries: [{ url: "http://example.org/#7" }], extData: { "uniq": r() } },
|
||
], selected: 5 }] };
|
||
|
||
let loadCount = 0;
|
||
let nextTestTimer;
|
||
function test_restore_apptabs_ondemand_progressCallback(aBrowser) {
|
||
loadCount++;
|
||
|
||
// get the tab
|
||
let tab;
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
if (!tab && window.gBrowser.tabs[i].linkedBrowser == aBrowser)
|
||
tab = window.gBrowser.tabs[i];
|
||
}
|
||
|
||
// Check that the load only comes from the selected tab.
|
||
ok(gBrowser.selectedTab == tab,
|
||
"test_restore_apptabs_ondemand: load came from selected tab");
|
||
|
||
// We should get only 1 load: the selected tab
|
||
if (loadCount == 1) {
|
||
nextTestTimer = setTimeout(nextTest, 1000);
|
||
return;
|
||
}
|
||
else if (loadCount > 1) {
|
||
clearTimeout(nextTestTimer);
|
||
}
|
||
|
||
function nextTest() {
|
||
window.gBrowser.removeTabsProgressListener(progressListener);
|
||
runNextTest();
|
||
}
|
||
nextTest();
|
||
}
|
||
|
||
window.gBrowser.addTabsProgressListener(progressListener);
|
||
ss.setBrowserState(JSON.stringify(state));
|
||
}
|
||
|
||
|
||
function countTabs() {
|
||
let needsRestore = 0,
|
||
isRestoring = 0,
|
||
wasRestored = 0;
|
||
|
||
let windowsEnum = Services.wm.getEnumerator("navigator:browser");
|
||
|
||
while (windowsEnum.hasMoreElements()) {
|
||
let window = windowsEnum.getNext();
|
||
if (window.closed)
|
||
continue;
|
||
|
||
for (let i = 0; i < window.gBrowser.tabs.length; i++) {
|
||
let browser = window.gBrowser.tabs[i].linkedBrowser;
|
||
if (browser.__SS_restoreState == TAB_STATE_RESTORING)
|
||
isRestoring++;
|
||
else if (browser.__SS_restoreState == TAB_STATE_NEEDS_RESTORE)
|
||
needsRestore++;
|
||
else
|
||
wasRestored++;
|
||
}
|
||
}
|
||
return [needsRestore, isRestoring, wasRestored];
|
||
}
|
||
|