home *** CD-ROM | disk | FTP | other *** search
/ PC World 2007 March / PCWorld_2007-03_cd.bin / komunikace / nvu / nvu-1.0-cs-CZ.win32.installer.exe / components / nsHelperAppDlg.js < prev    next >
Text File  |  2005-06-17  |  34KB  |  872 lines

  1. /*
  2. */
  3.  
  4. /* This file implements the nsIHelperAppLauncherDialog interface.
  5.  *
  6.  * The implementation consists of a JavaScript "class" named nsUnknownContentTypeDialog,
  7.  * comprised of:
  8.  *   - a JS constructor function
  9.  *   - a prototype providing all the interface methods and implementation stuff
  10.  *
  11.  * In addition, this file implements an nsIModule object that registers the
  12.  * nsUnknownContentTypeDialog component.
  13.  */
  14.  
  15.  
  16. /* ctor
  17.  */
  18. function nsUnknownContentTypeDialog() {
  19.     // Initialize data properties.
  20.     this.mLauncher = null;
  21.     this.mContext  = null;
  22.     this.mSourcePath = null;
  23.     this.chosenApp = null;
  24.     this.givenDefaultApp = false;
  25.     this.updateSelf = true;
  26.     this.mTitle    = "";
  27. }
  28.  
  29. nsUnknownContentTypeDialog.prototype = {
  30.     nsIMIMEInfo  : Components.interfaces.nsIMIMEInfo,
  31.  
  32.     // This "class" supports nsIHelperAppLauncherDialog, and nsISupports.
  33.     QueryInterface: function (iid) {
  34.         if (!iid.equals(Components.interfaces.nsIHelperAppLauncherDialog) &&
  35.             !iid.equals(Components.interfaces.nsISupports)) {
  36.             throw Components.results.NS_ERROR_NO_INTERFACE;
  37.         }
  38.         return this;
  39.     },
  40.  
  41.     // ---------- nsIHelperAppLauncherDialog methods ----------
  42.  
  43.     // show: Open XUL dialog using window watcher.  Since the dialog is not
  44.     //       modal, it needs to be a top level window and the way to open
  45.     //       one of those is via that route).
  46.     show: function(aLauncher, aContext)  {  
  47.       this.mLauncher = aLauncher;
  48.       this.mContext  = aContext;
  49.       // Display the dialog using the Window Watcher interface.
  50.       
  51.       var ir = aContext.QueryInterface(Components.interfaces.nsIInterfaceRequestor);
  52.       var dwi = ir.getInterface(Components.interfaces.nsIDOMWindowInternal);
  53.       var ww = Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
  54.                 .getService(Components.interfaces.nsIWindowWatcher);
  55.       this.mDialog = ww.openWindow(dwi,
  56.                                    "chrome://mozapps/content/downloads/unknownContentType.xul",
  57.                                    null,
  58.                                    "chrome,centerscreen,titlebar,dialog=yes,dependent",
  59.                                    null);
  60.       // Hook this object to the dialog.
  61.       this.mDialog.dialog = this;
  62.       
  63.       // Hook up utility functions. 
  64.       this.getSpecialFolderKey = this.mDialog.getSpecialFolderKey;
  65.       
  66.       // Watch for error notifications.
  67.       this.progressListener.helperAppDlg = this;
  68.       this.mLauncher.setWebProgressListener(this.progressListener);
  69.     },
  70.  
  71.     // promptForSaveToFile:  Display file picker dialog and return selected file.
  72.     //                       This is called by the External Helper App Service
  73.     //                       after the ucth dialog calls |saveToDisk| with a null
  74.     //                       target filename (no target, therefore user must pick).
  75.     //
  76.     //                       Alternatively, if the user has selected to have all
  77.     //                       files download to a specific location, return that
  78.     //                       location and don't ask via the dialog. 
  79.     //
  80.     // Note - this function is called without a dialog, so it cannot access any part
  81.     // of the dialog XUL as other functions on this object do. 
  82.     promptForSaveToFile: function(aLauncher, aContext, aDefaultFile, aSuggestedFileExtension) {
  83.       var result = "";
  84.       
  85.       this.mLauncher = aLauncher;
  86.  
  87.       // If the user is always downloading to the same location, the default download
  88.       // folder is stored in preferences. If a value is found stored, use that 
  89.       // automatically and don't ask via a dialog. 
  90.       const kDownloadFolderPref = "browser.download.defaultFolder";
  91.       var prefs = Components.classes["@mozilla.org/preferences-service;1"].getService(Components.interfaces.nsIPrefBranch);
  92.       try {
  93.         result = prefs.getComplexValue(kDownloadFolderPref, Components.interfaces.nsILocalFile);
  94.         result = this.validateLeafName(result, aDefaultFile, aSuggestedFileExtension);
  95.       }
  96.       catch (e) { 
  97.         // If we get here, it's because we have a new profile and the user has never configured download
  98.         // options, so "browser.download.defaultFolder" is not set yet. If the default is autodownload, 
  99.         // we need to discover the default save location. 
  100.         var autodownload = prefs.getBoolPref("browser.download.useDownloadDir");
  101.         if (autodownload) {
  102.           function getSpecialFolderKey(aFolderType) 
  103.           {
  104.             return aFolderType == "Desktop" ? "DeskP" : "Pers";
  105.             return "Home";
  106.           }
  107.           
  108.           function getDownloadsFolder(aFolder)
  109.           {
  110.             var fileLocator = Components.classes["@mozilla.org/file/directory_service;1"].getService(Components.interfaces.nsIProperties);
  111.  
  112.             var dir = fileLocator.get(getSpecialFolderKey(aFolder), Components.interfaces.nsILocalFile);
  113.             
  114.             var bundle = Components.classes["@mozilla.org/intl/stringbundle;1"].getService(Components.interfaces.nsIStringBundleService);
  115.             bundle = bundle.createBundle("chrome://mozapps/locale/downloads/unknownContentType.properties");
  116.  
  117.             var description = bundle.GetStringFromName("myDownloads");
  118.             if (aFolder != "Desktop")
  119.               dir.append(description);
  120.               
  121.             return dir;
  122.           }
  123.  
  124.           var defaultFolder = null;
  125.           switch (prefs.getIntPref("browser.download.folderList")) {
  126.           case 0:
  127.             defaultFolder = getDownloadsFolder("Desktop")
  128.             break;
  129.           case 1:
  130.             defaultFolder = getDownloadsFolder("Downloads");
  131.             break;
  132.           case 2:
  133.             defaultFolder = prefs.getComplexValue("browser.download.dir", Components.interfaces.nsILocalFile);
  134.             break;
  135.           }
  136.           
  137.           // While we're here, set the pref too so that we don't keep coming back into this less efficient
  138.           // code block. 
  139.           prefs.setComplexValue("browser.download.defaultFolder", Components.interfaces.nsILocalFile, defaultFolder);
  140.           
  141.           result = this.validateLeafName(defaultFolder, aDefaultFile, aSuggestedFileExtension);
  142.         }
  143.       }
  144.       
  145.       if (!result) {
  146.         // Use file picker to show dialog.
  147.         var nsIFilePicker = Components.interfaces.nsIFilePicker;
  148.         var picker = Components.classes["@mozilla.org/filepicker;1"].createInstance(nsIFilePicker);
  149.  
  150.         var bundle = Components.classes["@mozilla.org/intl/stringbundle;1"].getService(Components.interfaces.nsIStringBundleService);
  151.         bundle = bundle.createBundle("chrome://mozapps/locale/downloads/unknownContentType.properties");
  152.  
  153.         var windowTitle = bundle.GetStringFromName("saveDialogTitle");
  154.         var parent = aContext.QueryInterface(Components.interfaces.nsIInterfaceRequestor).getInterface(Components.interfaces.nsIDOMWindowInternal);
  155.         picker.init(parent, windowTitle, nsIFilePicker.modeSave);
  156.         picker.defaultString = aDefaultFile;
  157.  
  158.         if (aSuggestedFileExtension) {
  159.           // aSuggestedFileExtension includes the period, so strip it
  160.           picker.defaultExtension = aSuggestedFileExtension.substring(1);
  161.         } 
  162.         else {
  163.           try {
  164.             picker.defaultExtension = this.mLauncher.MIMEInfo.primaryExtension;
  165.           } 
  166.           catch (ex) { }
  167.         }
  168.  
  169.         var wildCardExtension = "*";
  170.         if (aSuggestedFileExtension) {
  171.           wildCardExtension += aSuggestedFileExtension;
  172.           picker.appendFilter(this.mLauncher.MIMEInfo.Description, wildCardExtension);
  173.         }
  174.  
  175.         picker.appendFilters( nsIFilePicker.filterAll );
  176.  
  177.         // Pull in the user's preferences and get the default download directory.
  178.         var prefs = Components.classes["@mozilla.org/preferences-service;1"].getService(Components.interfaces.nsIPrefBranch);
  179.         try {
  180.           var startDir = prefs.getComplexValue("browser.download.dir", Components.interfaces.nsILocalFile);
  181.           if (startDir.exists()) {
  182.             picker.displayDirectory = startDir;
  183.           }
  184.         } 
  185.         catch(exception) { }
  186.  
  187.         var dlgResult = picker.show();
  188.  
  189.         if (dlgResult == nsIFilePicker.returnCancel) {
  190.           // null result means user cancelled.
  191.           return null;
  192.         }
  193.  
  194.  
  195.         // Be sure to save the directory the user chose through the Save As... 
  196.         // dialog  as the new browser.download.dir
  197.         result = picker.file;
  198.  
  199.         if (result) {
  200.           var newDir = result.parent;
  201.           prefs.setComplexValue("browser.download.dir", Components.interfaces.nsILocalFile, newDir);
  202.         }
  203.       }
  204.       return result;
  205.     },
  206.     
  207.     validateLeafName: function (aLocalFile, aLeafName, aFileExt)
  208.     {
  209.       if (aLeafName == "")
  210.         aLeafName = "unnamed" + (aFileExt ? "." + aFileExt : "");
  211.       aLocalFile.append(aLeafName);
  212.  
  213.       this.makeFileUnique(aLocalFile);
  214.  
  215.       if (aLocalFile.isExecutable() && !this.mLauncher.targetFile.isExecutable()) {
  216.         var f = aLocalFile.clone();
  217.         aLocalFile.leafName = aLocalFile.leafName + "." + this.mLauncher.MIMEInfo.primaryExtension; 
  218.  
  219.         f.remove(false);
  220.         this.makeFileUnique(aLocalFile);
  221.       }
  222.       return aLocalFile;
  223.     },
  224.  
  225.     makeFileUnique: function (aLocalFile)
  226.     {
  227.       try {
  228.         // Since we're automatically downloading, we don't get the file picker's 
  229.         // logic to check for existing files, so we need to do that here.
  230.         //
  231.         // Note - this code is identical to that in 
  232.         //   browser/base/content/contentAreaUtils.js. 
  233.         // If you are updating this code, update that code too! We can't share code
  234.         // here since this is called in a js component. 
  235.         while (aLocalFile.exists()) {
  236.           var parts = /.+-(\d+)(\..*)?$/.exec(aLocalFile.leafName);
  237.           if (parts) {
  238.             aLocalFile.leafName = aLocalFile.leafName.replace(/((\d+)\.)|((\d+)$)/,
  239.                                                               function (str, dot, dotNum, noDot, noDotNum, pos, s) {
  240.                                                                 return (parseInt(str) + 1) + (dot ? "." : "");
  241.                                                               });
  242.           }
  243.           else {
  244.             aLocalFile.leafName = aLocalFile.leafName.replace(/\.|$/, "-1$&");
  245.           }
  246.         }
  247.         aLocalFile.create(Components.interfaces.nsIFile.NORMAL_FILE_TYPE, 0600);
  248.       }
  249.       catch (e) {
  250.         dump("*** exception in validateLeafName: " + e + "\n");
  251.         if (aLocalFile.leafName == "" || aLocalFile.isDirectory()) {
  252.           aLocalFile.append("unnamed");
  253.           if (aLocalFile.exists())
  254.             aLocalFile.createUnique(Components.interfaces.nsIFile.NORMAL_FILE_TYPE, 0600);
  255.         }
  256.       }
  257.     },
  258.     
  259.     // ---------- implementation methods ----------
  260.  
  261.     // Web progress listener so we can detect errors while mLauncher is
  262.     // streaming the data to a temporary file.
  263.     progressListener: {
  264.         // Implementation properties.
  265.         helperAppDlg: null,
  266.  
  267.         // nsIWebProgressListener methods.
  268.         // Look for error notifications and display alert to user.
  269.         onStatusChange: function( aWebProgress, aRequest, aStatus, aMessage ) {
  270.             if ( aStatus != Components.results.NS_OK ) {
  271.                 // Get prompt service.
  272.                 var prompter = Components.classes[ "@mozilla.org/embedcomp/prompt-service;1" ]
  273.                                    .getService( Components.interfaces.nsIPromptService );
  274.                 // Display error alert (using text supplied by back-end).
  275.                 prompter.alert( this.dialog, this.helperAppDlg.mTitle, aMessage );
  276.  
  277.                 // Close the dialog.
  278.                 this.helperAppDlg.onCancel();
  279.                 if ( this.helperAppDlg.mDialog ) {
  280.                     this.helperAppDlg.mDialog.close();
  281.                 }
  282.             }
  283.         },
  284.  
  285.         // Ignore onProgressChange, onStateChange, onLocationChange, and onSecurityChange notifications.
  286.         onProgressChange: function( aWebProgress,
  287.                                     aRequest,
  288.                                     aCurSelfProgress,
  289.                                     aMaxSelfProgress,
  290.                                     aCurTotalProgress,
  291.                                     aMaxTotalProgress ) {
  292.         },
  293.  
  294.         onStateChange: function( aWebProgress, aRequest, aStateFlags, aStatus ) {
  295.         },
  296.  
  297.         onLocationChange: function( aWebProgress, aRequest, aLocation ) {
  298.         },
  299.  
  300.         onSecurityChange: function( aWebProgress, aRequest, state ) {
  301.         }
  302.     },
  303.  
  304.     // initDialog:  Fill various dialog fields with initial content.
  305.     initDialog : function() {
  306.       // Put file name in window title.
  307.       var win   = this.dialogElement( "unknownContentType" );
  308.       var suggestedFileName = this.mLauncher.suggestedFileName;
  309.  
  310.       // Some URIs do not implement nsIURL, so we can't just QI.
  311.       var url   = this.mLauncher.source;
  312.       var fname = "";
  313.       this.mSourcePath = url.prePath;
  314.       try {
  315.           url = url.QueryInterface( Components.interfaces.nsIURL );
  316.           // A url, use file name from it.
  317.           fname = url.fileName;
  318.           this.mSourcePath += url.directory;
  319.       } catch (ex) {
  320.           // A generic uri, use path.
  321.           fname = url.path;
  322.           this.mSourcePath += url.path;
  323.       }
  324.  
  325.       if (suggestedFileName)
  326.         fname = suggestedFileName;
  327.       
  328.       var displayName = fname.replace(/ +/g, " ");
  329.  
  330.       this.mTitle = this.dialogElement("strings").getFormattedString("title", [displayName]);
  331.       win.setAttribute( "title", this.mTitle );
  332.  
  333.       // Put content type, filename and location into intro.
  334.       this.initIntro(url, fname, displayName);
  335.  
  336.       var iconString = "moz-icon://" + fname + "?size=16&contentType=" + this.mLauncher.MIMEInfo.MIMEType;
  337.       this.dialogElement("contentTypeImage").setAttribute("src", iconString);
  338.  
  339.       this.initAppAndSaveToDiskValues();
  340.  
  341.       // Initialize "always ask me" box. This should always be disabled
  342.       // and set to true for the ambiguous type application/octet-stream.
  343.       // We don't also check for application/x-msdownload here since we
  344.       // want users to be able to autodownload .exe files. 
  345.       var rememberChoice = this.dialogElement("rememberChoice");
  346.  
  347.       var mimeType = this.mLauncher.MIMEInfo.MIMEType;
  348.       if (mimeType == "application/octet-stream" || 
  349.           mimeType == "application/x-msdownload" ||
  350.           this.mLauncher.targetFile.isExecutable()) {
  351.         rememberChoice.checked = false;
  352.         rememberChoice.disabled = true;
  353.       }
  354.       else {
  355.         rememberChoice.checked = !this.mLauncher.MIMEInfo.alwaysAskBeforeHandling;
  356.       }
  357.       this.toggleRememberChoice(rememberChoice);
  358.  
  359.       // XXXben - menulist won't init properly, hack. 
  360.       var openHandler = this.dialogElement("openHandler");
  361.       openHandler.parentNode.removeChild(openHandler);
  362.       var openHandlerBox = this.dialogElement("openHandlerBox");
  363.       openHandlerBox.appendChild(openHandler);
  364.  
  365.       this.mDialog.setTimeout("dialog.postShowCallback()", 0);
  366.     },
  367.     
  368.     postShowCallback: function () {
  369.       this.mDialog.sizeToContent();
  370.  
  371.       // Set initial focus
  372.       this.dialogElement("mode").focus();
  373.     },
  374.  
  375.     // initIntro:
  376.     initIntro: function(url, filename, displayname) {
  377.         this.dialogElement( "location" ).value = displayname;
  378.         this.dialogElement( "location" ).setAttribute("realname", filename);
  379.  
  380.         // if mSourcePath is a local file, then let's use the pretty path name instead of an ugly
  381.         // url...
  382.         var pathString = this.mSourcePath;
  383.         try 
  384.         {
  385.           var fileURL = url.QueryInterface(Components.interfaces.nsIFileURL);
  386.           if (fileURL)
  387.           {
  388.             var fileObject = fileURL.file;
  389.             if (fileObject)
  390.             {
  391.               var parentObject = fileObject.parent;
  392.               if (parentObject)
  393.               {
  394.                 pathString = parentObject.path;
  395.               }
  396.             }
  397.           }
  398.         } catch(ex) {}
  399.  
  400.         // Set the location text, which is separate from the intro text so it can be cropped
  401.         var location = this.dialogElement( "source" );
  402.         location.value = pathString;
  403.         
  404.         // Show the type of file. 
  405.         var type = this.dialogElement("type");
  406.         var mimeInfo = this.mLauncher.MIMEInfo;
  407.         
  408.         // 1. Try to use the pretty description of the type, if one is available.
  409.         var typeString = mimeInfo.Description;
  410.         
  411.         if (typeString == "") {
  412.           // 2. If there is none, use the extension to identify the file, e.g. "ZIP file"
  413.           var primaryExtension = "";
  414.           try {
  415.             primaryExtension = mimeInfo.primaryExtension;
  416.           }
  417.           catch (ex) {
  418.           }
  419.           if (primaryExtension != "")
  420.             typeString = primaryExtension.toUpperCase() + " file";
  421.           // 3. If we can't even do that, just give up and show the MIME type. 
  422.           else
  423.             typeString = mimeInfo.MIMEType;
  424.         }
  425.         
  426.         type.value = typeString;
  427.     },
  428.  
  429.     // Returns true if opening the default application makes sense.
  430.     openWithDefaultOK: function() {
  431.         var result;
  432.  
  433.         // The checking is different on Windows...
  434.         // Windows presents some special cases.
  435.         // We need to prevent use of "system default" when the file is
  436.         // executable (so the user doesn't launch nasty programs downloaded
  437.         // from the web), and, enable use of "system default" if it isn't
  438.         // executable (because we will prompt the user for the default app
  439.         // in that case).
  440.         
  441.         // Need to get temporary file and check for executable-ness.
  442.         var tmpFile = this.mLauncher.targetFile;
  443.         
  444.         //  Default is Ok if the file isn't executable (and vice-versa).
  445.         return !tmpFile.isExecutable();
  446.     },
  447.     
  448.     // Set "default" application description field.
  449.     initDefaultApp: function() {
  450.       // Use description, if we can get one.
  451.       var desc = this.mLauncher.MIMEInfo.defaultDescription;
  452.       if (desc) {
  453.         var defaultApp = this.dialogElement("strings").getFormattedString("defaultApp", [desc]);
  454.         this.dialogElement("defaultHandler").label = defaultApp;
  455.       }
  456.       else {
  457.         this.dialogElement("modeDeck").setAttribute("selectedIndex", "1");
  458.         // Hide the default handler item too, in case the user picks a 
  459.         // custom handler at a later date which triggers the menulist to show.
  460.         this.dialogElement("defaultHandler").hidden = true;
  461.       }
  462.     },
  463.  
  464.     // getPath:
  465.     getPath: function (aFile) {
  466.       return aFile.path;
  467.     },
  468.  
  469.     // initAppAndSaveToDiskValues:
  470.     initAppAndSaveToDiskValues: function() {
  471.       var modeGroup = this.dialogElement("mode");
  472.  
  473.       // We don't let users open .exe files or random binary data directly 
  474.       // from the browser at the moment because of security concerns. 
  475.       var openWithDefaultOK = this.openWithDefaultOK();
  476.       var mimeType = this.mLauncher.MIMEInfo.MIMEType;
  477.       if (this.mLauncher.targetFile.isExecutable() || (
  478.           (mimeType == "application/octet-stream" ||
  479.            mimeType == "application/x-msdownload") && 
  480.            !openWithDefaultOK)) {
  481.         this.dialogElement("open").disabled = true;
  482.         var openHandler = this.dialogElement("openHandler");
  483.         openHandler.disabled = true;
  484.         openHandler.label = "";
  485.         modeGroup.selectedItem = this.dialogElement("save");
  486.         return;
  487.       }
  488.     
  489.       // Fill in helper app info, if there is any.
  490.       this.chosenApp = this.mLauncher.MIMEInfo.preferredApplicationHandler;
  491.       // Initialize "default application" field.
  492.       this.initDefaultApp();
  493.  
  494.       var otherHandler = this.dialogElement("otherHandler");
  495.               
  496.       // Fill application name textbox.
  497.       if (this.chosenApp && this.chosenApp.path) {
  498.         otherHandler.setAttribute("path", this.getPath(this.chosenApp));
  499.         otherHandler.label = this.chosenApp.leafName;
  500.         otherHandler.hidden = false;
  501.       }
  502.  
  503.       var useDefault = this.dialogElement("useSystemDefault");
  504.       var openHandler = this.dialogElement("openHandler");
  505.       openHandler.selectedIndex = 0;
  506.  
  507.       if (this.mLauncher.MIMEInfo.preferredAction == this.nsIMIMEInfo.useSystemDefault) {
  508.         // Open (using system default).
  509.         modeGroup.selectedItem = this.dialogElement("open");
  510.       } else if (this.mLauncher.MIMEInfo.preferredAction == this.nsIMIMEInfo.useHelperApp) {
  511.         // Open with given helper app.
  512.         modeGroup.selectedItem = this.dialogElement("open");
  513.         openHandler.selectedIndex = 1;
  514.       } else {
  515.         // Save to disk.
  516.         modeGroup.selectedItem = this.dialogElement("save");
  517.       }
  518.       
  519.       // If we don't have a "default app" then disable that choice.
  520.       if (!openWithDefaultOK) {
  521.         var useDefault = this.dialogElement("defaultHandler");
  522.         var isSelected = useDefault.selected;
  523.         
  524.         // Disable that choice.
  525.         useDefault.hidden = true;
  526.         // If that's the default, then switch to "save to disk."
  527.         if (isSelected) {
  528.           openHandler.selectedIndex = 1;
  529.           modeGroup.selectedItem = this.dialogElement("save");
  530.         }
  531.       }
  532.       
  533.       // otherHandler is always disabled on Mac
  534.       otherHandler.nextSibling.hidden = otherHandler.nextSibling.nextSibling.hidden = false;
  535.       this.updateOKButton();
  536.     },
  537.  
  538.     // Returns the user-selected application
  539.     helperAppChoice: function() {
  540.       return this.chosenApp;
  541.     },
  542.     
  543.     get saveToDisk() {
  544.       return this.dialogElement("save").selected;
  545.     },
  546.     
  547.     get useOtherHandler() {
  548.       return this.dialogElement("open").selected && this.dialogElement("openHandler").selectedIndex == 1;
  549.     },
  550.     
  551.     get useSystemDefault() {
  552.       return this.dialogElement("open").selected && this.dialogElement("openHandler").selectedIndex == 0;
  553.     },
  554.     
  555.     toggleRememberChoice: function (aCheckbox) {
  556.         this.dialogElement("settingsChange").hidden = !aCheckbox.checked;
  557.         this.mDialog.sizeToContent();
  558.     },
  559.     
  560.     openHandlerCommand: function () {
  561.       var openHandler = this.dialogElement("openHandler");
  562.       if (openHandler.selectedItem.id == "choose")
  563.         this.chooseApp();
  564.       else
  565.         openHandler.setAttribute("lastSelectedItemID", openHandler.selectedItem.id);
  566.     },
  567.  
  568.     updateOKButton: function() {
  569.       var ok = false;
  570.       if (this.dialogElement("save").selected) {
  571.         // This is always OK.
  572.         ok = true;
  573.       } 
  574.       else if (this.dialogElement("open").selected) {
  575.         switch (this.dialogElement("openHandler").selectedIndex) {
  576.         case 0:
  577.           // No app need be specified in this case.
  578.           ok = true;
  579.           break;
  580.         case 1:
  581.           // only enable the OK button if we have a default app to use or if 
  582.           // the user chose an app....
  583.           ok = this.chosenApp || /\S/.test(this.dialogElement("otherHandler").getAttribute("path")); 
  584.         break;
  585.         }
  586.       }
  587.  
  588.       // Enable Ok button if ok to press.
  589.       this.mDialog.document.documentElement.getButton("accept").disabled = !ok;
  590.     },
  591.     
  592.     // Returns true iff the user-specified helper app has been modified.
  593.     appChanged: function() {
  594.       return this.helperAppChoice() != this.mLauncher.MIMEInfo.preferredApplicationHandler;
  595.     },
  596.  
  597.     updateMIMEInfo: function() {
  598.       var needUpdate = false;
  599.       // If current selection differs from what's in the mime info object,
  600.       // then we need to update.
  601.       if (this.saveToDisk) {
  602.         needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.saveToDisk;
  603.         if (needUpdate)
  604.           this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.saveToDisk;
  605.       } 
  606.       else if (this.useSystemDefault) {
  607.         needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.useSystemDefault;
  608.         if (needUpdate)
  609.           this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.useSystemDefault;
  610.       } 
  611.       else {
  612.         // For "open with", we need to check both preferred action and whether the user chose
  613.         // a new app.
  614.         needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.useHelperApp || this.appChanged();
  615.         if (needUpdate) {
  616.           this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.useHelperApp;
  617.           // App may have changed - Update application and description
  618.           var app = this.helperAppChoice();
  619.           this.mLauncher.MIMEInfo.preferredApplicationHandler = app;
  620.           this.mLauncher.MIMEInfo.applicationDescription = "";
  621.         }
  622.       }
  623.       // We will also need to update if the "always ask" flag has changed.
  624.       needUpdate = needUpdate || this.mLauncher.MIMEInfo.alwaysAskBeforeHandling != (!this.dialogElement("rememberChoice").checked);
  625.  
  626.       // One last special case: If the input "always ask" flag was false, then we always
  627.       // update.  In that case we are displaying the helper app dialog for the first
  628.       // time for this mime type and we need to store the user's action in the mimeTypes.rdf
  629.       // data source (whether that action has changed or not; if it didn't change, then we need
  630.       // to store the "always ask" flag so the helper app dialog will or won't display
  631.       // next time, per the user's selection).
  632.       needUpdate = needUpdate || !this.mLauncher.MIMEInfo.alwaysAskBeforeHandling;
  633.  
  634.       // Make sure mime info has updated setting for the "always ask" flag.
  635.       this.mLauncher.MIMEInfo.alwaysAskBeforeHandling = !this.dialogElement("rememberChoice").checked;
  636.  
  637.       return needUpdate;        
  638.     },
  639.     
  640.     // See if the user changed things, and if so, update the
  641.     // mimeTypes.rdf entry for this mime type.
  642.     updateHelperAppPref: function() {
  643.       var ha = new this.mDialog.HelperApps();
  644.       ha.updateTypeInfo(this.mLauncher.MIMEInfo);
  645.     },
  646.     
  647.     // onOK:
  648.     onOK: function() {
  649.       // Verify typed app path, if necessary.
  650.       if (this.useOtherHandler) {
  651.         var helperApp = this.helperAppChoice();
  652.         if (!helperApp || !helperApp.exists()) {
  653.           // Show alert and try again.        
  654.           var bundle = this.dialogElement("strings");                    
  655.           var msg = bundle.getFormattedString("badApp", [this.dialogElement("otherHandler").path]);
  656.           var svc = Components.classes["@mozilla.org/embedcomp/prompt-service;1"].getService(Components.interfaces.nsIPromptService);
  657.           svc.alert(this.mDialog, bundle.getString("badApp.title"), msg);
  658.  
  659.           // Disable the OK button.
  660.           this.mDialog.document.documentElement.getButton("accept").disabled = true;
  661.           this.dialogElement("mode").focus();          
  662.  
  663.           // Clear chosen application.
  664.           this.chosenApp = null;
  665.  
  666.           // Leave dialog up.
  667.           return false;
  668.         }
  669.       }
  670.         
  671.       // Remove our web progress listener (a progress dialog will be
  672.       // taking over).
  673.       this.mLauncher.setWebProgressListener(null);
  674.       
  675.       // saveToDisk and launchWithApplication can return errors in 
  676.       // certain circumstances (e.g. The user clicks cancel in the
  677.       // "Save to Disk" dialog. In those cases, we don't want to
  678.       // update the helper application preferences in the RDF file.
  679.       try {
  680.         var needUpdate = this.updateMIMEInfo();
  681.         
  682.         if (this.dialogElement("save").selected) {
  683.           // If we're using a default download location, create a path
  684.           // for the file to be saved to to pass to |saveToDisk| - otherwise
  685.           // we must ask the user to pick a save name.
  686.  
  687.           this.mLauncher.saveToDisk(null, false);
  688.         }
  689.         else
  690.           this.mLauncher.launchWithApplication(null, false);
  691.  
  692.         // Update user pref for this mime type (if necessary). We do not
  693.         // store anything in the mime type preferences for the ambiguous
  694.         // type application/octet-stream. We do NOT do this for 
  695.         // application/x-msdownload since we want users to be able to 
  696.         // autodownload these to disk. 
  697.         if (needUpdate && this.mLauncher.MIMEInfo.MIMEType != "application/octet-stream")
  698.           this.updateHelperAppPref();
  699.       } catch(e) { }
  700.  
  701.       // Unhook dialog from this object.
  702.       this.mDialog.dialog = null;
  703.  
  704.       // Close up dialog by returning true.
  705.       return true;
  706.     },
  707.  
  708.     // onCancel:
  709.     onCancel: function() {
  710.       // Remove our web progress listener.
  711.       this.mLauncher.setWebProgressListener(null);
  712.  
  713.       // Cancel app launcher.
  714.       try {
  715.         this.mLauncher.Cancel();
  716.       } catch(exception) {
  717.       }
  718.  
  719.       // Unhook dialog from this object.
  720.       this.mDialog.dialog = null;
  721.  
  722.       // Close up dialog by returning true.
  723.       return true;
  724.     },
  725.  
  726.     // dialogElement:  Convenience. 
  727.     dialogElement: function(id) {
  728.       return this.mDialog.document.getElementById(id);
  729.     },
  730.  
  731.     // chooseApp:  Open file picker and prompt user for application.
  732.     chooseApp: function() {
  733.       var nsIFilePicker = Components.interfaces.nsIFilePicker;
  734.       var fp = Components.classes["@mozilla.org/filepicker;1"].createInstance(nsIFilePicker);
  735.       fp.init(this.mDialog,
  736.               this.dialogElement("strings").getString("chooseAppFilePickerTitle"),
  737.               nsIFilePicker.modeOpen);
  738.  
  739.       fp.appendFilters(nsIFilePicker.filterApps);
  740.  
  741.       if (fp.show() == nsIFilePicker.returnOK && fp.file) {
  742.         // Show the "handler" menulist since we have a (user-specified) 
  743.         // application now.
  744.         this.dialogElement("modeDeck").setAttribute("selectedIndex", "0");
  745.         
  746.         // Remember the file they chose to run.
  747.         this.chosenApp = fp.file;
  748.         // Update dialog.
  749.         var otherHandler = this.dialogElement("otherHandler");
  750.         otherHandler.removeAttribute("hidden");
  751.         otherHandler.setAttribute("path", this.getPath(this.chosenApp));
  752.         otherHandler.label = this.chosenApp.leafName;
  753.         this.dialogElement("openHandler").selectedIndex = 1;
  754.         this.dialogElement("openHandler").setAttribute("lastSelectedItemID", "otherHandler");
  755.         
  756.         this.dialogElement("mode").selectedItem = this.dialogElement("open");
  757.       }
  758.       else {
  759.         var openHandler = this.dialogElement("openHandler");
  760.         var lastSelectedID = openHandler.getAttribute("lastSelectedItemID");
  761.         if (!lastSelectedID)
  762.           lastSelectedID = "defaultHandler";
  763.         openHandler.selectedItem = this.dialogElement(lastSelectedID);
  764.       }
  765.     },
  766.  
  767.     // Turn this on to get debugging messages.
  768.     debug: false,
  769.  
  770.     // Dump text (if debug is on).
  771.     dump: function( text ) {
  772.         if ( this.debug ) {
  773.             dump( text ); 
  774.         }
  775.     },
  776.  
  777.     // dumpInfo:
  778.     doDebug: function() {
  779.         const nsIProgressDialog = Components.interfaces.nsIProgressDialog;
  780.         // Open new progress dialog.
  781.         var progress = Components.classes[ "@mozilla.org/progressdialog;1" ]
  782.                          .createInstance( nsIProgressDialog );
  783.         // Show it.
  784.         progress.open( this.mDialog );
  785.     },
  786.  
  787.     // dumpObj:
  788.     dumpObj: function( spec ) {
  789.          var val = "<undefined>";
  790.          try {
  791.              val = eval( "this."+spec ).toString();
  792.          } catch( exception ) {
  793.          }
  794.          this.dump( spec + "=" + val + "\n" );
  795.     },
  796.  
  797.     // dumpObjectProperties
  798.     dumpObjectProperties: function( desc, obj ) {
  799.          for( prop in obj ) {
  800.              this.dump( desc + "." + prop + "=" );
  801.              var val = "<undefined>";
  802.              try {
  803.                  val = obj[ prop ];
  804.              } catch ( exception ) {
  805.              }
  806.              this.dump( val + "\n" );
  807.          }
  808.     }
  809. }
  810.  
  811. // This Component's module implementation.  All the code below is used to get this
  812. // component registered and accessible via XPCOM.
  813. var module = {
  814.     firstTime: true,
  815.  
  816.     // registerSelf: Register this component.
  817.     registerSelf: function (compMgr, fileSpec, location, type) {
  818.         if (this.firstTime) {
  819.             this.firstTime = false;
  820.             throw Components.results.NS_ERROR_FACTORY_REGISTER_AGAIN;
  821.         }
  822.         compMgr = compMgr.QueryInterface(Components.interfaces.nsIComponentRegistrar);
  823.  
  824.         compMgr.registerFactoryLocation( this.cid,
  825.                                          "Unknown Content Type Dialog",
  826.                                          this.contractId,
  827.                                          fileSpec,
  828.                                          location,
  829.                                          type );
  830.     },
  831.  
  832.     // getClassObject: Return this component's factory object.
  833.     getClassObject: function (compMgr, cid, iid) {
  834.         if (!cid.equals(this.cid)) {
  835.             throw Components.results.NS_ERROR_NO_INTERFACE;
  836.         }
  837.  
  838.         if (!iid.equals(Components.interfaces.nsIFactory)) {
  839.             throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
  840.         }
  841.  
  842.         return this.factory;
  843.     },
  844.  
  845.     /* CID for this class */
  846.     cid: Components.ID("{F68578EB-6EC2-4169-AE19-8C6243F0ABE1}"),
  847.  
  848.     /* Contract ID for this class */
  849.     contractId: "@mozilla.org/helperapplauncherdialog;1",
  850.  
  851.     /* factory object */
  852.     factory: {
  853.         // createInstance: Return a new nsProgressDialog object.
  854.         createInstance: function (outer, iid) {
  855.             if (outer != null)
  856.                 throw Components.results.NS_ERROR_NO_AGGREGATION;
  857.  
  858.             return (new nsUnknownContentTypeDialog()).QueryInterface(iid);
  859.         }
  860.     },
  861.  
  862.     // canUnload: n/a (returns true)
  863.     canUnload: function(compMgr) {
  864.         return true;
  865.     }
  866. };
  867.  
  868. // NSGetModule: Return the nsIModule object.
  869. function NSGetModule(compMgr, fileSpec) {
  870.     return module;
  871. }
  872.