forked from mirrors/gecko-dev
		
	 a259026c9d
			
		
	
	
		a259026c9d
		
	
	
	
	
		
			
			This also removes any redundant Ci.nsISupports elements in the interface
lists.
This was done using the following script:
acecb401b7/processors/chromeutils-generateQI.jsm
MozReview-Commit-ID: AIx10P8GpZY
--HG--
extra : rebase_source : a29c07530586dc18ba040f19215475ac20fcfb3b
		
	
			
		
			
				
	
	
		
			384 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			384 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
 | |
| /* 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/. */
 | |
| 
 | |
| /* globals ContentAreaUtils */
 | |
| 
 | |
| const APK_MIME_TYPE = "application/vnd.android.package-archive";
 | |
| 
 | |
| const OMA_DOWNLOAD_DESCRIPTOR_MIME_TYPE = "application/vnd.oma.dd+xml";
 | |
| const OMA_DRM_MESSAGE_MIME = "application/vnd.oma.drm.message";
 | |
| const OMA_DRM_CONTENT_MIME = "application/vnd.oma.drm.content";
 | |
| const OMA_DRM_RIGHTS_MIME = "application/vnd.oma.drm.rights+wbxml";
 | |
| 
 | |
| const PREF_BD_USEDOWNLOADDIR = "browser.download.useDownloadDir";
 | |
| const URI_GENERIC_ICON_DOWNLOAD = "drawable://alert_download";
 | |
| 
 | |
| ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm");
 | |
| 
 | |
| XPCOMUtils.defineLazyModuleGetters(this, {
 | |
|   Downloads: "resource://gre/modules/Downloads.jsm",
 | |
|   EventDispatcher: "resource://gre/modules/Messaging.jsm",
 | |
|   FileUtils: "resource://gre/modules/FileUtils.jsm",
 | |
|   HelperApps: "resource://gre/modules/HelperApps.jsm",
 | |
|   NetUtil: "resource://gre/modules/NetUtil.jsm",
 | |
|   RuntimePermissions: "resource://gre/modules/RuntimePermissions.jsm",
 | |
|   Services: "resource://gre/modules/Services.jsm",
 | |
|   Snackbars: "resource://gre/modules/Snackbars.jsm",
 | |
|   Task: "resource://gre/modules/Task.jsm",
 | |
| });
 | |
| 
 | |
| // -----------------------------------------------------------------------
 | |
| // HelperApp Launcher Dialog
 | |
| // -----------------------------------------------------------------------
 | |
| 
 | |
| XPCOMUtils.defineLazyGetter(this, "ContentAreaUtils", function() {
 | |
|   let ContentAreaUtils = {};
 | |
|   Services.scriptloader.loadSubScript("chrome://global/content/contentAreaUtils.js", ContentAreaUtils);
 | |
|   return ContentAreaUtils;
 | |
| });
 | |
| 
 | |
| function HelperAppLauncherDialog() { }
 | |
| 
 | |
| HelperAppLauncherDialog.prototype = {
 | |
|   classID: Components.ID("{e9d277a0-268a-4ec2-bb8c-10fdf3e44611}"),
 | |
|   QueryInterface: ChromeUtils.generateQI([Ci.nsIHelperAppLauncherDialog]),
 | |
| 
 | |
|   /**
 | |
|    * Returns false if `url` represents a local or special URL that we don't
 | |
|    * wish to ever download.
 | |
|    *
 | |
|    * Returns true otherwise.
 | |
|    */
 | |
|   _canDownload: function(url, alreadyResolved = false) {
 | |
|     // The common case.
 | |
|     if (url.schemeIs("http") ||
 | |
|         url.schemeIs("https") ||
 | |
|         url.schemeIs("ftp")) {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     // The less-common opposite case.
 | |
|     if (url.schemeIs("chrome") ||
 | |
|         url.schemeIs("jar") ||
 | |
|         url.schemeIs("resource") ||
 | |
|         url.schemeIs("wyciwyg") ||
 | |
|         url.schemeIs("file")) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     // For all other URIs, try to resolve them to an inner URI, and check that.
 | |
|     if (!alreadyResolved) {
 | |
|       let innerURI = NetUtil.newChannel({
 | |
|         uri: url,
 | |
|         loadUsingSystemPrincipal: true
 | |
|       }).URI;
 | |
| 
 | |
|       if (!url.equals(innerURI)) {
 | |
|         return this._canDownload(innerURI, true);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Anything else is fine to download.
 | |
|     return true;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns true if `launcher` represents a download for which we wish
 | |
|    * to prompt.
 | |
|    */
 | |
|   _shouldPrompt: function(launcher) {
 | |
|     let mimeType = this._getMimeTypeFromLauncher(launcher);
 | |
| 
 | |
|     // Straight equality: nsIMIMEInfo normalizes.
 | |
|     return APK_MIME_TYPE == mimeType || OMA_DOWNLOAD_DESCRIPTOR_MIME_TYPE == mimeType;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns true if `launcher` represents a download for which we wish to
 | |
|    * offer a "Save to disk" option.
 | |
|    */
 | |
|   _shouldAddSaveToDiskIntent: function(launcher) {
 | |
|       let mimeType = this._getMimeTypeFromLauncher(launcher);
 | |
| 
 | |
|       // We can't handle OMA downloads. So don't even try. (Bug 1219078)
 | |
|       return mimeType != OMA_DOWNLOAD_DESCRIPTOR_MIME_TYPE;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns true if `launcher`represents a download that should not be handled by Firefox
 | |
|    * or a third-party app and instead be forwarded to Android's download manager.
 | |
|    */
 | |
|   _shouldForwardToAndroidDownloadManager: function(aLauncher) {
 | |
|     let forwardDownload = Services.prefs.getBoolPref("browser.download.forward_oma_android_download_manager");
 | |
|     if (!forwardDownload) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     let mimeType = aLauncher.MIMEInfo.MIMEType;
 | |
|     if (!mimeType) {
 | |
|       mimeType = ContentAreaUtils.getMIMETypeForURI(aLauncher.source) || "";
 | |
|     }
 | |
| 
 | |
|     return [
 | |
|       OMA_DOWNLOAD_DESCRIPTOR_MIME_TYPE,
 | |
|       OMA_DRM_MESSAGE_MIME,
 | |
|       OMA_DRM_CONTENT_MIME,
 | |
|       OMA_DRM_RIGHTS_MIME
 | |
|     ].includes(mimeType);
 | |
|   },
 | |
| 
 | |
|   show: function hald_show(aLauncher, aContext, aReason) {
 | |
|     if (!this._canDownload(aLauncher.source)) {
 | |
|       this._refuseDownload(aLauncher);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (this._shouldForwardToAndroidDownloadManager(aLauncher)) {
 | |
|       Task.spawn(function* () {
 | |
|         try {
 | |
|           let hasPermission = yield RuntimePermissions.waitForPermissions(RuntimePermissions.WRITE_EXTERNAL_STORAGE);
 | |
|           if (hasPermission) {
 | |
|             this._downloadWithAndroidDownloadManager(aLauncher);
 | |
|           }
 | |
|         } finally {
 | |
|           aLauncher.cancel(Cr.NS_BINDING_ABORTED);
 | |
|         }
 | |
|       }.bind(this)).catch(Cu.reportError);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let bundle = Services.strings.createBundle("chrome://browser/locale/browser.properties");
 | |
| 
 | |
|     let apps = HelperApps.getAppsForUri(aLauncher.source, {
 | |
|       mimeType: aLauncher.MIMEInfo.MIMEType,
 | |
|     });
 | |
| 
 | |
|     if (this._shouldAddSaveToDiskIntent(aLauncher)) {
 | |
|       // Add a fake intent for save to disk at the top of the list.
 | |
|       apps.unshift({
 | |
|         name: bundle.GetStringFromName("helperapps.saveToDisk"),
 | |
|         packageName: "org.mozilla.gecko.Download",
 | |
|         iconUri: "drawable://icon",
 | |
|         selected: true, // Default to download for files
 | |
|         launch: function() {
 | |
|           // Reset the preferredAction here.
 | |
|           aLauncher.MIMEInfo.preferredAction = Ci.nsIMIMEInfo.saveToDisk;
 | |
|           aLauncher.saveToDisk(null, false);
 | |
|           return true;
 | |
|         }
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     // We do not handle this download and there are no apps that want to do it
 | |
|     if (apps.length === 0) {
 | |
|       this._refuseDownload(aLauncher);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let callback = function(app) {
 | |
|       aLauncher.MIMEInfo.preferredAction = Ci.nsIMIMEInfo.useHelperApp;
 | |
|       if (!app.launch(aLauncher.source)) {
 | |
|         // Once the app is done we need to get rid of the temp file. This shouldn't
 | |
|         // get run in the saveToDisk case.
 | |
|         aLauncher.cancel(Cr.NS_BINDING_ABORTED);
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     // See if the user already marked something as the default for this mimetype,
 | |
|     // and if that app is still installed.
 | |
|     let preferredApp = this._getPreferredApp(aLauncher);
 | |
|     if (preferredApp) {
 | |
|       let pref = apps.filter(function(app) {
 | |
|         return app.packageName === preferredApp;
 | |
|       });
 | |
| 
 | |
|       if (pref.length > 0) {
 | |
|         callback(pref[0]);
 | |
|         return;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // If there's only one choice, and we don't want to prompt, go right ahead
 | |
|     // and choose that app automatically.
 | |
|     if (!this._shouldPrompt(aLauncher) && (apps.length === 1)) {
 | |
|       callback(apps[0]);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Otherwise, let's go through the prompt.
 | |
|     let alwaysUse = bundle.GetStringFromName("helperapps.alwaysUse");
 | |
|     let justOnce = bundle.GetStringFromName("helperapps.useJustOnce");
 | |
|     let newButtonOrder = this._useNewButtonOrder();
 | |
| 
 | |
|     HelperApps.prompt(apps, {
 | |
|       window: aContext,
 | |
|       title: bundle.GetStringFromName("helperapps.pick"),
 | |
|       buttons: [
 | |
|         newButtonOrder ? alwaysUse : justOnce,
 | |
|         newButtonOrder ? justOnce : alwaysUse
 | |
|       ],
 | |
|       // Tapping an app twice should choose "Just once".
 | |
|       doubleTapButton: newButtonOrder ? 1 : 0
 | |
|     }, (data) => {
 | |
|       if (data.button < 0) {
 | |
|         aLauncher.cancel(Cr.NS_BINDING_ABORTED);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       callback(apps[data.icongrid0]);
 | |
| 
 | |
|       if (data.button === (newButtonOrder ? 0 : 1)) {
 | |
|         this._setPreferredApp(aLauncher, apps[data.icongrid0]);
 | |
|       }
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * In the system app chooser, the order of the "Always" and "Just once" buttons has been swapped
 | |
|    * around starting from Lollipop.
 | |
|    */
 | |
|   _useNewButtonOrder: function() {
 | |
|     return Services.sysinfo.getPropertyAsUint32("version") >= 21;
 | |
|   },
 | |
| 
 | |
|   _refuseDownload: function(aLauncher) {
 | |
|     aLauncher.cancel(Cr.NS_BINDING_ABORTED);
 | |
| 
 | |
|     Services.console.logStringMessage("Refusing download of non-downloadable file.");
 | |
| 
 | |
|     let bundle = Services.strings.createBundle("chrome://browser/locale/browser.properties");
 | |
|     let failedText = bundle.GetStringFromName("download.blocked");
 | |
| 
 | |
|     Snackbars.show(failedText, Snackbars.LENGTH_LONG);
 | |
|   },
 | |
| 
 | |
|   _downloadWithAndroidDownloadManager(aLauncher) {
 | |
|     let mimeType = aLauncher.MIMEInfo.MIMEType;
 | |
|     if (!mimeType) {
 | |
|       mimeType = ContentAreaUtils.getMIMETypeForURI(aLauncher.source) || "";
 | |
|     }
 | |
| 
 | |
|     EventDispatcher.instance.sendRequest({
 | |
|       "type": "Download:AndroidDownloadManager",
 | |
|       "uri": aLauncher.source.spec,
 | |
|       "mimeType": mimeType,
 | |
|       "filename": aLauncher.suggestedFileName
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   _getPrefName: function getPrefName(mimetype) {
 | |
|     return "browser.download.preferred." + mimetype.replace("\\", ".");
 | |
|   },
 | |
| 
 | |
|   _getMimeTypeFromLauncher: function(launcher) {
 | |
|     let mime = launcher.MIMEInfo.MIMEType;
 | |
|     if (!mime)
 | |
|       mime = ContentAreaUtils.getMIMETypeForURI(launcher.source) || "";
 | |
|     return mime;
 | |
|   },
 | |
| 
 | |
|   _getPreferredApp: function getPreferredApp(launcher) {
 | |
|     let mime = this._getMimeTypeFromLauncher(launcher);
 | |
|     if (!mime)
 | |
|       return;
 | |
| 
 | |
|     try {
 | |
|       return Services.prefs.getCharPref(this._getPrefName(mime));
 | |
|     } catch (ex) {
 | |
|       Services.console.logStringMessage("Error getting pref for " + mime + ".");
 | |
|     }
 | |
|     return null;
 | |
|   },
 | |
| 
 | |
|   _setPreferredApp: function setPreferredApp(launcher, app) {
 | |
|     let mime = this._getMimeTypeFromLauncher(launcher);
 | |
|     if (!mime)
 | |
|       return;
 | |
| 
 | |
|     if (app)
 | |
|       Services.prefs.setCharPref(this._getPrefName(mime), app.packageName);
 | |
|     else
 | |
|       Services.prefs.clearUserPref(this._getPrefName(mime));
 | |
|   },
 | |
| 
 | |
|   promptForSaveToFileAsync: function(aLauncher, aContext, aDefaultFile,
 | |
|                                       aSuggestedFileExt, aForcePrompt) {
 | |
|     Task.spawn(function* () {
 | |
|       let file = null;
 | |
|       try {
 | |
|         let hasPermission = yield RuntimePermissions.waitForPermissions(RuntimePermissions.WRITE_EXTERNAL_STORAGE);
 | |
|         if (hasPermission) {
 | |
|           // If we do have the STORAGE permission then pick the public downloads directory as destination
 | |
|           // for this file. Without the permission saveDestinationAvailable(null) will be called which
 | |
|           // will effectively cancel the download.
 | |
|           let preferredDir = yield Downloads.getPreferredDownloadsDirectory();
 | |
|           file = this.validateLeafName(new FileUtils.File(preferredDir),
 | |
|                                        aDefaultFile, aSuggestedFileExt);
 | |
|         }
 | |
|       } finally {
 | |
|         // The file argument will be null in case any exception occurred.
 | |
|         aLauncher.saveDestinationAvailable(file);
 | |
|       }
 | |
|     }.bind(this)).catch(Cu.reportError);
 | |
|   },
 | |
| 
 | |
|   validateLeafName: function hald_validateLeafName(aLocalFile, aLeafName, aFileExt) {
 | |
|     if (!(aLocalFile && this.isUsableDirectory(aLocalFile)))
 | |
|       return null;
 | |
| 
 | |
|     // Remove any leading periods, since we don't want to save hidden files
 | |
|     // automatically.
 | |
|     aLeafName = aLeafName.replace(/^\.+/, "");
 | |
| 
 | |
|     if (aLeafName == "")
 | |
|       aLeafName = "unnamed" + (aFileExt ? "." + aFileExt : "");
 | |
|     aLocalFile.append(aLeafName);
 | |
| 
 | |
|     this.makeFileUnique(aLocalFile);
 | |
|     return aLocalFile;
 | |
|   },
 | |
| 
 | |
|   makeFileUnique: function hald_makeFileUnique(aLocalFile) {
 | |
|     try {
 | |
|       // Note - this code is identical to that in
 | |
|       //   toolkit/content/contentAreaUtils.js.
 | |
|       // If you are updating this code, update that code too! We can't share code
 | |
|       // here since this is called in a js component.
 | |
|       let collisionCount = 0;
 | |
|       while (aLocalFile.exists()) {
 | |
|         collisionCount++;
 | |
|         if (collisionCount == 1) {
 | |
|           // Append "(2)" before the last dot in (or at the end of) the filename
 | |
|           // special case .ext.gz etc files so we don't wind up with .tar(2).gz
 | |
|           if (aLocalFile.leafName.match(/\.[^\.]{1,3}\.(gz|bz2|Z)$/i))
 | |
|             aLocalFile.leafName = aLocalFile.leafName.replace(/\.[^\.]{1,3}\.(gz|bz2|Z)$/i, "(2)$&");
 | |
|           else
 | |
|             aLocalFile.leafName = aLocalFile.leafName.replace(/(\.[^\.]*)?$/, "(2)$&");
 | |
|         } else {
 | |
|           // replace the last (n) in the filename with (n+1)
 | |
|           aLocalFile.leafName = aLocalFile.leafName.replace(/^(.*\()\d+\)/, "$1" + (collisionCount + 1) + ")");
 | |
|         }
 | |
|       }
 | |
|       aLocalFile.create(Ci.nsIFile.NORMAL_FILE_TYPE, 0o600);
 | |
|     } catch (e) {
 | |
|       dump("*** exception in validateLeafName: " + e + "\n");
 | |
| 
 | |
|       if (e.result == Cr.NS_ERROR_FILE_ACCESS_DENIED)
 | |
|         throw e;
 | |
| 
 | |
|       if (aLocalFile.leafName == "" || aLocalFile.isDirectory()) {
 | |
|         aLocalFile.append("unnamed");
 | |
|         if (aLocalFile.exists())
 | |
|           aLocalFile.createUnique(Ci.nsIFile.NORMAL_FILE_TYPE, 0o600);
 | |
|       }
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   isUsableDirectory: function hald_isUsableDirectory(aDirectory) {
 | |
|     return aDirectory.exists() && aDirectory.isDirectory() && aDirectory.isWritable();
 | |
|   },
 | |
| };
 | |
| 
 | |
| this.NSGetFactory = XPCOMUtils.generateNSGetFactory([HelperAppLauncherDialog]);
 |