/* ***** 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 Session Store. * * The Initial Developer of the Original Code is Mozilla Foundation. * Portions created by the Initial Developer are Copyright (C) 2009 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Mark Finkle * * 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 ***** */ const Cc = Components.classes; const Ci = Components.interfaces; const Cu = Components.utils; const Cr = Components.results; Cu.import("resource://gre/modules/XPCOMUtils.jsm"); Cu.import("resource://gre/modules/Services.jsm"); #ifdef MOZ_CRASH_REPORTER XPCOMUtils.defineLazyServiceGetter(this, "CrashReporter", "@mozilla.org/xre/app-info;1", "nsICrashReporter"); #endif XPCOMUtils.defineLazyGetter(this, "NetUtil", function() { Cu.import("resource://gre/modules/NetUtil.jsm"); return NetUtil; }); // ----------------------------------------------------------------------- // Session Store // ----------------------------------------------------------------------- const STATE_STOPPED = 0; const STATE_RUNNING = 1; const STATE_QUITTING = -1; function SessionStore() { } SessionStore.prototype = { classID: Components.ID("{8c1f07d6-cba3-4226-a315-8bd43d67d032}"), QueryInterface: XPCOMUtils.generateQI([Ci.nsISessionStore, Ci.nsIDOMEventListener, Ci.nsIObserver, Ci.nsISupportsWeakReference]), _windows: {}, _lastSaveTime: 0, _interval: 15000, _maxTabsUndo: 5, init: function ss_init() { // Get file references this._sessionFile = Services.dirsvc.get("ProfD", Ci.nsILocalFile); this._sessionFileBackup = this._sessionFile.clone(); this._sessionFile.append("sessionstore.js"); this._sessionFileBackup.append("sessionstore.bak"); this._loadState = STATE_STOPPED; try { if (this._sessionFileBackup.exists()) this._sessionFileBackup.remove(false); if (this._sessionFile.exists()) this._sessionFile.copyTo(null, this._sessionFileBackup.leafName); } catch (ex) { Cu.reportError(ex); // file was write-locked? } try { this._interval = Services.prefs.getIntPref("sessionstore.interval"); this._maxTabsUndo = Services.prefs.getIntPref("sessionstore.max_tabs_undo"); } catch (e) {} }, observe: function ss_observe(aSubject, aTopic, aData) { let self = this; let observerService = Services.obs; switch (aTopic) { case "app-startup": observerService.addObserver(this, "final-ui-startup", true); observerService.addObserver(this, "domwindowopened", true); observerService.addObserver(this, "domwindowclosed", true); observerService.addObserver(this, "browser-lastwindow-close-granted", true); observerService.addObserver(this, "quit-application-requested", true); observerService.addObserver(this, "quit-application-granted", true); observerService.addObserver(this, "quit-application", true); break; case "final-ui-startup": observerService.removeObserver(this, "final-ui-startup"); this.init(); break; case "domwindowopened": let window = aSubject; window.addEventListener("load", function() { self.onWindowOpen(window); window.removeEventListener("load", arguments.callee, false); }, false); break; case "domwindowclosed": // catch closed windows this.onWindowClose(aSubject); break; case "browser-lastwindow-close-granted": // Force and open timer to save state if (this._saveTimer) { this._saveTimer.cancel(); this._saveTimer = null; this.saveState(); } // Freeze the data at what we've got (ignoring closing windows) this._loadState = STATE_QUITTING; break; case "quit-application-requested": // Get a current snapshot of all windows this._forEachBrowserWindow(function(aWindow) { self._collectWindowData(aWindow); }); break; case "quit-application-granted": case "quit-application": // Freeze the data at what we've got (ignoring closing windows) this._loadState = STATE_QUITTING; observerService.removeObserver(this, "domwindowopened"); observerService.removeObserver(this, "domwindowclosed"); observerService.removeObserver(this, "browser-lastwindow-close-granted"); observerService.removeObserver(this, "quit-application-requested"); observerService.removeObserver(this, "quit-application-granted"); observerService.removeObserver(this, "quit-application"); // Make sure to break our cycle with the save timer if (this._saveTimer) { this._saveTimer.cancel(); this._saveTimer = null; } break; case "timer-callback": // Timer call back for delayed saving this._saveTimer = null; this.saveState(); break; } }, handleEvent: function ss_handleEvent(aEvent) { let window = aEvent.currentTarget.ownerDocument.defaultView; switch (aEvent.type) { case "load": case "pageshow": this.onTabLoad(window, aEvent.currentTarget, aEvent); break; case "TabOpen": case "TabClose": { let browser = aEvent.originalTarget.linkedBrowser; if (aEvent.type == "TabOpen") { this.onTabAdd(window, browser); } else { this.onTabClose(window, browser); this.onTabRemove(window, browser); } break; } case "TabSelect": { let browser = aEvent.originalTarget.linkedBrowser; this.onTabSelect(window, browser); break; } } }, receiveMessage: function ss_receiveMessage(aMessage) { let window = aMessage.target.ownerDocument.defaultView; switch (aMessage.name) { case "pageshow": this.onTabLoad(window, aMessage.target, aMessage); break; } }, onWindowOpen: function ss_onWindowOpen(aWindow) { // Return if window has already been initialized if (aWindow && aWindow.__SSID && this._windows[aWindow.__SSID]) return; // Ignore non-browser windows and windows opened while shutting down if (aWindow.document.documentElement.getAttribute("windowtype") != "navigator:browser" || this._loadState == STATE_QUITTING) return; // Assign it a unique identifier (timestamp) and create its data object aWindow.__SSID = "window" + Date.now(); this._windows[aWindow.__SSID] = { tabs: [], selected: 0, closedTabs: [] }; // Perform additional initialization when the first window is loading if (this._loadState == STATE_STOPPED) { this._loadState = STATE_RUNNING; this._lastSaveTime = Date.now(); } // Add tab change listeners to all already existing tabs let tabs = aWindow.Browser.tabs; for (let i = 0; i < tabs.length; i++) this.onTabAdd(aWindow, tabs[i].browser, true); // Notification of tab add/remove/selection let tabContainer = aWindow.document.getElementById("tabs"); tabContainer.addEventListener("TabOpen", this, true); tabContainer.addEventListener("TabClose", this, true); tabContainer.addEventListener("TabSelect", this, true); }, onWindowClose: function ss_onWindowClose(aWindow) { // Ignore windows not tracked by SessionStore if (!aWindow.__SSID || !this._windows[aWindow.__SSID]) return; let tabContainer = aWindow.document.getElementById("tabs"); tabContainer.removeEventListener("TabOpen", this, true); tabContainer.removeEventListener("TabClose", this, true); tabContainer.removeEventListener("TabSelect", this, true); if (this._loadState == STATE_RUNNING) { // Update all window data for a last time this._collectWindowData(aWindow); // Clear this window from the list delete this._windows[aWindow.__SSID]; // Save the state without this window to disk this.saveStateDelayed(); } let tabs = aWindow.Browser.tabs; for (let i = 0; i < tabs.length; i++) this.onTabRemove(aWindow, tabs[i].browser, true); delete aWindow.__SSID; }, onTabAdd: function ss_onTabAdd(aWindow, aBrowser, aNoNotification) { aBrowser.messageManager.addMessageListener("pageshow", this, true); if (!aNoNotification) this.saveStateDelayed(); this._updateCrashReportURL(aWindow); }, onTabRemove: function ss_onTabRemove(aWindow, aBrowser, aNoNotification) { aBrowser.messageManager.removeMessageListener("pageshow", this, true); delete aBrowser.__SS_data; if (!aNoNotification) this.saveStateDelayed(); }, onTabClose: function ss_onTabClose(aWindow, aBrowser) { if (this._maxTabsUndo == 0) return; if (aWindow.Browser.tabs.length > 0) { // Bundle this browser's data and extra data and save in the closedTabs // window property let data = aBrowser.__SS_data; data.extraData = aBrowser.__SS_extdata; this._windows[aWindow.__SSID].closedTabs.unshift(data); let length = this._windows[aWindow.__SSID].closedTabs.length; if (length > this._maxTabsUndo) this._windows[aWindow.__SSID].closedTabs.splice(this._maxTabsUndo, length - this._maxTabsUndo); } }, onTabLoad: function ss_onTabLoad(aWindow, aBrowser, aMessage) { delete aBrowser.__SS_data; this._collectTabData(aBrowser); this.saveStateDelayed(); this._updateCrashReportURL(aWindow); }, onTabSelect: function ss_onTabSelect(aWindow, aBrowser) { if (this._loadState != STATE_RUNNING) return; let index = 0; let browser = aWindow.Browser; let tabs = browser.tabs; for (let i = 0; i < tabs.length; i++) { if (tabs[i].browser == aBrowser) { index = i; break; } } this._windows[aWindow.__SSID].selected = index + 1; // 1-based this._updateCrashReportURL(aWindow); }, saveStateDelayed: function ss_saveStateDelayed() { if (!this._saveTimer) { // Interval until the next disk operation is allowed let minimalDelay = this._lastSaveTime + this._interval - Date.now(); // If we have to wait, set a timer, otherwise saveState directly let delay = Math.max(minimalDelay, 2000); if (delay > 0) { this._saveTimer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer); this._saveTimer.init(this, delay, Ci.nsITimer.TYPE_ONE_SHOT); } else { this.saveState(); } } }, saveState: function ss_saveState() { let data = this._getCurrentState(); this._writeFile(this._sessionFile, JSON.stringify(data)); this._lastSaveTime = Date.now(); }, _getCurrentState: function ss_getCurrentState() { let self = this; this._forEachBrowserWindow(function(aWindow) { self._collectWindowData(aWindow); }); let data = { windows: [] }; let index; for (index in this._windows) data.windows.push(this._windows[index]); return data; }, _collectTabData: function ss__collectTabData(aBrowser) { let tabData = { entries: [{}] }; tabData.entries[0] = { url: aBrowser.currentURI.spec, title: aBrowser.contentTitle }; tabData.index = 1; tabData.attributes = { image: aBrowser.mIconURL }; aBrowser.__SS_data = tabData; }, _collectWindowData: function ss__collectWindowData(aWindow) { // Ignore windows not tracked by SessionStore if (!aWindow.__SSID || !this._windows[aWindow.__SSID]) return; let winData = this._windows[aWindow.__SSID]; winData.tabs = []; let tabs = aWindow.Browser.tabs; for (let i = 0; i < tabs.length; i++) { if (tabs[i].browser.__SS_data) { let browser = tabs[i].browser; let tabData = browser.__SS_data; if (browser.__SS_extdata) tabData.extData = browser.__SS_extdata; winData.tabs.push(tabData); } } }, _forEachBrowserWindow: function ss_forEachBrowserWindow(aFunc) { let windowsEnum = Services.wm.getEnumerator("navigator:browser"); while (windowsEnum.hasMoreElements()) { let window = windowsEnum.getNext(); if (window.__SSID && !window.closed) aFunc.call(this, window); } }, _writeFile: function ss_writeFile(aFile, aData) { let stateString = Cc["@mozilla.org/supports-string;1"].createInstance(Ci.nsISupportsString); stateString.data = aData; Services.obs.notifyObservers(stateString, "sessionstore-state-write", ""); // Don't touch the file if an observer has deleted all state data if (!stateString.data) return; // Initialize the file output stream. let ostream = Cc["@mozilla.org/network/safe-file-output-stream;1"].createInstance(Ci.nsIFileOutputStream); ostream.init(aFile, 0x02 | 0x08 | 0x20, 0600, 0); // Obtain a converter to convert our data to a UTF-8 encoded input stream. let converter = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter); converter.charset = "UTF-8"; // Asynchronously copy the data to the file. let istream = converter.convertToInputStream(aData); NetUtil.asyncCopy(istream, ostream, function(rc) { if (Components.isSuccessCode(rc)) { Services.obs.notifyObservers(null, "sessionstore-state-write-complete", ""); } }); }, _updateCrashReportURL: function ss_updateCrashReportURL(aWindow) { #ifdef MOZ_CRASH_REPORTER try { let currentURI = aWindow.Browser.selectedBrowser.currentURI.clone(); // if the current URI contains a username/password, remove it try { currentURI.userPass = ""; } catch (ex) { } // ignore failures on about: URIs CrashReporter.annotateCrashReport("URL", currentURI.spec); } catch (ex) { // don't make noise when crashreporter is built but not enabled if (ex.result != Components.results.NS_ERROR_NOT_INITIALIZED) Components.utils.reportError("SessionStore:" + ex); } #endif }, getBrowserState: function ss_getBrowserState() { let data = this._getCurrentState(); return JSON.stringify(data); }, getClosedTabCount: function ss_getClosedTabCount(aWindow) { if (!aWindow || !aWindow.__SSID) return 0; // not a browser window, or not otherwise tracked by SS. return this._windows[aWindow.__SSID].closedTabs.length; }, getClosedTabData: function ss_getClosedTabData(aWindow) { if (!aWindow.__SSID) throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); return JSON.stringify(this._windows[aWindow.__SSID].closedTabs); }, undoCloseTab: function ss_undoCloseTab(aWindow, aIndex) { if (!aWindow.__SSID) throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); let closedTabs = this._windows[aWindow.__SSID].closedTabs; if (!closedTabs) return null; // default to the most-recently closed tab aIndex = aIndex || 0; if (!(aIndex in closedTabs)) throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); // fetch the data of closed tab, while removing it from the array let closedTab = closedTabs.splice(aIndex, 1).shift(); // create a new tab and bring to front let tab = aWindow.Browser.addTab(closedTab.entries[0].url, true); // Put back the extra data tab.browser.__SS_extdata = closedTab.extraData; // TODO: save and restore more data (position, field values, etc) return tab.chromeTab; }, forgetClosedTab: function ss_forgetClosedTab(aWindow, aIndex) { if (!aWindow.__SSID) throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); let closedTabs = this._windows[aWindow.__SSID].closedTabs; // default to the most-recently closed tab aIndex = aIndex || 0; if (!(aIndex in closedTabs)) throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); // remove closed tab from the array closedTabs.splice(aIndex, 1); }, getTabValue: function ss_getTabValue(aTab, aKey) { let browser = aTab.linkedBrowser; let data = browser.__SS_extdata || {}; return data[aKey] || ""; }, setTabValue: function ss_setTabValue(aTab, aKey, aStringValue) { let browser = aTab.linkedBrowser; if (!browser.__SS_extdata) browser.__SS_extdata = {}; browser.__SS_extdata[aKey] = aStringValue; this.saveStateDelayed(); }, deleteTabValue: function ss_deleteTabValue(aTab, aKey) { let browser = aTab.linkedBrowser; if (browser.__SS_extdata && browser.__SS_extdata[aKey]) delete browser.__SS_extdata[aKey]; else throw (Components.returnCode = Cr.NS_ERROR_INVALID_ARG); } }; const NSGetFactory = XPCOMUtils.generateNSGetFactory([SessionStore]);