154 строки
4.9 KiB
JavaScript
154 строки
4.9 KiB
JavaScript
ChromeUtils.import("resource://gre/modules/Services.jsm");
|
|
ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
XPCOMUtils.defineLazyGlobalGetters(this, ["URL"]);
|
|
|
|
const {actionTypes: at} = ChromeUtils.import("resource://activity-stream/common/Actions.jsm", {});
|
|
|
|
XPCOMUtils.defineLazyModuleGetters(this, {
|
|
DownloadsCommon: "resource:///modules/DownloadsCommon.jsm",
|
|
DownloadsViewUI: "resource:///modules/DownloadsViewUI.jsm",
|
|
FileUtils: "resource://gre/modules/FileUtils.jsm",
|
|
});
|
|
|
|
const DOWNLOAD_CHANGED_DELAY_TIME = 1000; // time in ms to delay timer for downloads changed events
|
|
|
|
this.DownloadsManager = class DownloadsManager {
|
|
constructor(store) {
|
|
this._downloadData = null;
|
|
this._store = null;
|
|
this._downloadItems = new Map();
|
|
this._downloadTimer = null;
|
|
}
|
|
|
|
setTimeout(callback, delay) {
|
|
let timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
|
|
timer.initWithCallback(callback, delay, Ci.nsITimer.TYPE_ONE_SHOT);
|
|
return timer;
|
|
}
|
|
|
|
formatDownload(download) {
|
|
return {
|
|
hostname: new URL(download.source.url).hostname,
|
|
url: download.source.url,
|
|
path: download.target.path,
|
|
title: DownloadsViewUI.getDisplayName(download),
|
|
description: DownloadsViewUI.getSizeWithUnits(download) ||
|
|
DownloadsCommon.strings.sizeUnknown,
|
|
referrer: download.source.referrer,
|
|
date_added: download.endTime,
|
|
};
|
|
}
|
|
|
|
init(store) {
|
|
this._store = store;
|
|
this._browser = Services.appShell.hiddenDOMWindow;
|
|
this._downloadData = DownloadsCommon.getData(this._browser.ownerGlobal, true, false, true);
|
|
this._downloadData.addView(this);
|
|
}
|
|
|
|
onDownloadAdded(download) {
|
|
if (!this._downloadItems.has(download.source.url)) {
|
|
this._downloadItems.set(download.source.url, download);
|
|
|
|
// On startup, all existing downloads fire this notification, so debounce them
|
|
if (this._downloadTimer) {
|
|
this._downloadTimer.delay = DOWNLOAD_CHANGED_DELAY_TIME;
|
|
} else {
|
|
this._downloadTimer = this.setTimeout(() => {
|
|
this._downloadTimer = null;
|
|
this._store.dispatch({type: at.DOWNLOAD_CHANGED});
|
|
}, DOWNLOAD_CHANGED_DELAY_TIME);
|
|
}
|
|
}
|
|
}
|
|
|
|
onDownloadRemoved(download) {
|
|
if (this._downloadItems.has(download.source.url)) {
|
|
this._downloadItems.delete(download.source.url);
|
|
this._store.dispatch({type: at.DOWNLOAD_CHANGED});
|
|
}
|
|
}
|
|
|
|
async getDownloads(threshold, {numItems = this._downloadItems.size, onlySucceeded = false, onlyExists = false}) {
|
|
if (!threshold) {
|
|
return [];
|
|
}
|
|
let results = [];
|
|
|
|
// Only get downloads within the time threshold specified and sort by recency
|
|
const downloadThreshold = Date.now() - threshold;
|
|
let downloads = [...this._downloadItems.values()]
|
|
.filter(download => download.endTime > downloadThreshold)
|
|
.sort((download1, download2) => download1.endTime < download2.endTime);
|
|
|
|
for (const download of downloads) {
|
|
// Only include downloads where the file still exists
|
|
if (onlyExists) {
|
|
// Refresh download to ensure the 'exists' attribute is up to date
|
|
await download.refresh();
|
|
if (!download.target.exists) { continue; }
|
|
}
|
|
// Only include downloads that were completed successfully
|
|
if (onlySucceeded) {
|
|
if (!download.succeeded) { continue; }
|
|
}
|
|
const formattedDownloadForHighlights = this.formatDownload(download);
|
|
results.push(formattedDownloadForHighlights);
|
|
if (results.length === numItems) {
|
|
break;
|
|
}
|
|
}
|
|
return results;
|
|
}
|
|
|
|
uninit() {
|
|
if (this._downloadData) {
|
|
this._downloadData.removeView(this);
|
|
this._downloadData = null;
|
|
}
|
|
if (this._downloadTimer) {
|
|
this._downloadTimer.cancel();
|
|
this._downloadTimer = null;
|
|
}
|
|
}
|
|
|
|
onAction(action) {
|
|
let doDownloadAction = callback => {
|
|
let download = this._downloadItems.get(action.data.url);
|
|
if (download) {
|
|
callback(download);
|
|
}
|
|
};
|
|
|
|
switch (action.type) {
|
|
case at.COPY_DOWNLOAD_LINK:
|
|
doDownloadAction(download => {
|
|
DownloadsCommon.copyDownloadLink(download);
|
|
});
|
|
break;
|
|
case at.REMOVE_DOWNLOAD_FILE:
|
|
doDownloadAction(download => {
|
|
DownloadsCommon.deleteDownload(download).catch(Cu.reportError);
|
|
});
|
|
break;
|
|
case at.SHOW_DOWNLOAD_FILE:
|
|
doDownloadAction(download => {
|
|
DownloadsCommon.showDownloadedFile(
|
|
new FileUtils.File(download.target.path));
|
|
});
|
|
break;
|
|
case at.OPEN_DOWNLOAD_FILE:
|
|
doDownloadAction(download => {
|
|
DownloadsCommon.openDownloadedFile(
|
|
new FileUtils.File(download.target.path), null,
|
|
this._browser.ownerGlobal);
|
|
});
|
|
break;
|
|
case at.UNINIT:
|
|
this.uninit();
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
this.EXPORTED_SYMBOLS = ["DownloadsManager"];
|