summaryrefslogtreecommitdiffstats
path: root/application/basilisk/components/extensions/ext-sessions.js
blob: 3052957d61f30e0a3b220e7dee9706d8b35e4f06 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
/* -*- Mode: indent-tabs-mode: nil; js-indent-level: 2 -*- */
/* vim: set sts=2 sw=2 et tw=80: */
"use strict";

Cu.import("resource://gre/modules/ExtensionUtils.jsm");
var {
  promiseObserved,
  SingletonEventManager,
} = ExtensionUtils;

XPCOMUtils.defineLazyModuleGetter(this, "SessionStore",
                                  "resource:///modules/sessionstore/SessionStore.jsm");

const SS_ON_CLOSED_OBJECTS_CHANGED = "sessionstore-closed-objects-changed";

function getRecentlyClosed(maxResults, extension) {
  let recentlyClosed = [];

  // Get closed windows
  let closedWindowData = SessionStore.getClosedWindowData(false);
  for (let window of closedWindowData) {
    recentlyClosed.push({
      lastModified: window.closedAt,
      window: WindowManager.convertFromSessionStoreClosedData(window, extension)});
  }

  // Get closed tabs
  for (let window of WindowListManager.browserWindows()) {
    let closedTabData = SessionStore.getClosedTabData(window, false);
    for (let tab of closedTabData) {
      recentlyClosed.push({
        lastModified: tab.closedAt,
        tab: TabManager.for(extension).convertFromSessionStoreClosedData(tab, window)});
    }
  }

  // Sort windows and tabs
  recentlyClosed.sort((a, b) => b.lastModified - a.lastModified);
  return recentlyClosed.slice(0, maxResults);
}

function createSession(restored, extension, sessionId) {
  if (!restored) {
    return Promise.reject({message: `Could not restore object using sessionId ${sessionId}.`});
  }
  let sessionObj = {lastModified: Date.now()};
  if (restored instanceof Ci.nsIDOMChromeWindow) {
    return promiseObserved("sessionstore-single-window-restored", subject => subject == restored).then(() => {
      sessionObj.window = WindowManager.convert(extension, restored, {populate: true});
      return Promise.resolve([sessionObj]);
    });
  }
  sessionObj.tab = TabManager.for(extension).convert(restored);
  return Promise.resolve([sessionObj]);
}

extensions.registerSchemaAPI("sessions", "addon_parent", context => {
  let {extension} = context;
  return {
    sessions: {
      getRecentlyClosed: function(filter) {
        let maxResults = filter.maxResults == undefined ? this.MAX_SESSION_RESULTS : filter.maxResults;
        return Promise.resolve(getRecentlyClosed(maxResults, extension));
      },

      restore: function(sessionId) {
        let session, closedId;
        if (sessionId) {
          closedId = sessionId;
          session = SessionStore.undoCloseById(closedId);
        } else if (SessionStore.lastClosedObjectType == "window") {
          // If the most recently closed object is a window, just undo closing the most recent window.
          session = SessionStore.undoCloseWindow(0);
        } else {
          // It is a tab, and we cannot call SessionStore.undoCloseTab without a window,
          // so we must find the tab in which case we can just use its closedId.
          let recentlyClosedTabs = [];
          for (let window of WindowListManager.browserWindows()) {
            let closedTabData = SessionStore.getClosedTabData(window, false);
            for (let tab of closedTabData) {
              recentlyClosedTabs.push(tab);
            }
          }

          // Sort the tabs.
          recentlyClosedTabs.sort((a, b) => b.closedAt - a.closedAt);

          // Use the closedId of the most recently closed tab to restore it.
          closedId = recentlyClosedTabs[0].closedId;
          session = SessionStore.undoCloseById(closedId);
        }
        return createSession(session, extension, closedId);
      },

      onChanged: new SingletonEventManager(context, "sessions.onChanged", fire => {
        let observer = () => {
          context.runSafe(fire);
        };

        Services.obs.addObserver(observer, SS_ON_CLOSED_OBJECTS_CHANGED, false);
        return () => {
          Services.obs.removeObserver(observer, SS_ON_CLOSED_OBJECTS_CHANGED);
        };
      }).api(),
    },
  };
});