2012-02-23 06:37:04 -08:00
# - * - indent - tabs - mode : nil - * -
2012-05-21 04:12:37 -07:00
# 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/.
2007-03-22 10:30:00 -07:00
2007-06-16 14:23:17 -07:00
const Ci = Components . interfaces ;
const Cc = Components . classes ;
const Cr = Components . results ;
2007-09-11 08:52:04 -07:00
const Cu = Components . utils ;
2009-04-15 14:49:11 -07:00
const XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" ;
2014-10-23 14:43:51 -07:00
const POLARIS _ENABLED = "browser.polaris.enabled" ;
2009-04-15 14:49:11 -07:00
2007-09-11 08:52:04 -07:00
Cu . import ( "resource://gre/modules/XPCOMUtils.jsm" ) ;
2010-04-12 07:33:10 -07:00
Cu . import ( "resource://gre/modules/Services.jsm" ) ;
2007-09-05 20:02:22 -07:00
2013-07-05 13:55:00 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "AboutHome" ,
"resource:///modules/AboutHome.jsm" ) ;
2014-10-24 08:19:00 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "UITour" ,
"resource:///modules/UITour.jsm" ) ;
2012-02-15 15:44:29 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "AddonManager" ,
"resource://gre/modules/AddonManager.jsm" ) ;
2010-08-10 15:15:26 -07:00
2013-08-08 17:16:47 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "ContentClick" ,
"resource:///modules/ContentClick.jsm" ) ;
2014-03-31 01:51:21 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "DirectoryLinksProvider" ,
2014-09-22 00:53:19 -07:00
"resource:///modules/DirectoryLinksProvider.jsm" ) ;
2014-03-31 01:51:21 -07:00
2012-02-15 15:44:29 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "NetUtil" ,
"resource://gre/modules/NetUtil.jsm" ) ;
2012-12-08 11:50:54 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "FileUtils" ,
"resource://gre/modules/FileUtils.jsm" ) ;
2012-02-15 15:44:29 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "PlacesUtils" ,
"resource://gre/modules/PlacesUtils.jsm" ) ;
2010-11-23 05:04:07 -08:00
2012-03-23 00:15:36 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "BookmarkHTMLUtils" ,
"resource://gre/modules/BookmarkHTMLUtils.jsm" ) ;
2013-04-16 22:36:02 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "BookmarkJSONUtils" ,
"resource://gre/modules/BookmarkJSONUtils.jsm" ) ;
2014-03-15 14:37:37 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "WebappManager" ,
"resource:///modules/WebappManager.jsm" ) ;
2012-03-12 17:33:10 -07:00
2012-05-02 13:11:19 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "PageThumbs" ,
2013-02-16 20:15:41 -08:00
"resource://gre/modules/PageThumbs.jsm" ) ;
2012-05-02 13:11:19 -07:00
2012-08-21 11:17:18 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "NewTabUtils" ,
2013-02-16 20:16:16 -08:00
"resource://gre/modules/NewTabUtils.jsm" ) ;
2012-08-21 11:17:18 -07:00
2014-02-25 09:51:41 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "CustomizationTabPreloader" ,
"resource:///modules/CustomizationTabPreloader.jsm" ) ;
2012-06-02 09:56:24 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "PdfJs" ,
"resource://pdf.js/PdfJs.jsm" ) ;
2013-10-21 18:39:43 -07:00
# ifdef NIGHTLY _BUILD
2013-09-30 16:23:41 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "ShumwayUtils" ,
"resource://shumway/ShumwayUtils.jsm" ) ;
2013-10-21 18:39:43 -07:00
# endif
2013-09-30 16:23:41 -07:00
2012-10-05 02:37:00 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "webrtcUI" ,
"resource:///modules/webrtcUI.jsm" ) ;
2012-10-05 16:25:52 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "PrivateBrowsingUtils" ,
"resource://gre/modules/PrivateBrowsingUtils.jsm" ) ;
2012-12-05 17:59:26 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "RecentWindow" ,
2013-03-18 08:46:17 -07:00
"resource:///modules/RecentWindow.jsm" ) ;
2012-12-05 17:59:26 -08:00
2013-04-09 01:23:40 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "Task" ,
"resource://gre/modules/Task.jsm" ) ;
2013-04-16 23:08:11 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "PlacesBackups" ,
"resource://gre/modules/PlacesBackups.jsm" ) ;
2013-07-03 14:41:25 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "OS" ,
"resource://gre/modules/osfile.jsm" ) ;
2013-04-16 23:08:11 -07:00
2014-08-11 11:13:36 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "RemotePrompt" ,
"resource:///modules/RemotePrompt.jsm" ) ;
XPCOMUtils . defineLazyModuleGetter ( this , "ContentPrefServiceParent" ,
"resource://gre/modules/ContentPrefServiceParent.jsm" ) ;
2014-04-04 14:50:42 -07:00
2013-07-26 22:40:07 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "SessionStore" ,
"resource:///modules/sessionstore/SessionStore.jsm" ) ;
2013-12-10 11:49:02 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "BrowserUITelemetry" ,
"resource:///modules/BrowserUITelemetry.jsm" ) ;
2014-02-04 05:43:20 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "AsyncShutdown" ,
2014-02-22 13:37:00 -08:00
"resource://gre/modules/AsyncShutdown.jsm" ) ;
2014-02-04 05:43:20 -08:00
2014-06-28 11:09:45 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "LoginManagerParent" ,
"resource://gre/modules/LoginManagerParent.jsm" ) ;
2014-10-17 15:55:14 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "SimpleServiceDiscovery" ,
"resource://gre/modules/SimpleServiceDiscovery.jsm" ) ;
2014-03-22 20:46:18 -07:00
# ifdef NIGHTLY _BUILD
XPCOMUtils . defineLazyModuleGetter ( this , "SignInToWebsiteUX" ,
"resource:///modules/SignInToWebsite.jsm" ) ;
# endif
2014-04-24 19:09:20 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "ContentSearch" ,
"resource:///modules/ContentSearch.jsm" ) ;
2014-10-08 18:56:26 -07:00
# ifdef E10S _TESTING _ONLY
XPCOMUtils . defineLazyModuleGetter ( this , "UpdateChannel" ,
"resource://gre/modules/UpdateChannel.jsm" ) ;
# endif
2014-08-12 00:59:11 -07:00
XPCOMUtils . defineLazyGetter ( this , "ShellService" , function ( ) {
try {
return Cc [ "@mozilla.org/browser/shell-service;1" ] .
getService ( Ci . nsIShellService ) ;
}
catch ( ex ) {
return null ;
}
} ) ;
2014-08-21 08:42:42 -07:00
XPCOMUtils . defineLazyModuleGetter ( this , "FormValidationHandler" ,
"resource:///modules/FormValidationHandler.jsm" ) ;
2014-11-06 23:12:26 -08:00
XPCOMUtils . defineLazyModuleGetter ( this , "WebChannel" ,
"resource://gre/modules/WebChannel.jsm" ) ;
2009-10-02 04:26:04 -07:00
const PREF _PLUGINS _NOTIFYUSER = "plugins.update.notifyUser" ;
const PREF _PLUGINS _UPDATEURL = "plugins.update.url" ;
2008-03-07 16:16:38 -08:00
2014-02-04 05:43:20 -08:00
// Seconds of idle before trying to create a bookmarks backup.
2014-05-27 22:10:00 -07:00
const BOOKMARKS _BACKUP _IDLE _TIME _SEC = 8 * 60 ;
2014-02-04 05:43:20 -08:00
// Minimum interval between backups. We try to not create more than one backup
// per interval.
const BOOKMARKS _BACKUP _MIN _INTERVAL _DAYS = 1 ;
// Maximum interval between backups. If the last backup is older than these
// days we will try to create a new one more aggressively.
2014-05-27 22:10:00 -07:00
const BOOKMARKS _BACKUP _MAX _INTERVAL _DAYS = 3 ;
2008-03-13 12:25:49 -07:00
2014-12-16 09:10:28 -08:00
// Record the current default search engine in Telemetry.
function recordDefaultSearchEngine ( ) {
let engine ;
try {
engine = Services . search . defaultEngine ;
} catch ( e ) { }
let name ;
if ( ! engine ) {
name = "NONE" ;
} else if ( engine . identifier ) {
name = engine . identifier ;
} else if ( engine . name ) {
name = "other-" + engine . name ;
} else {
name = "UNDEFINED" ;
}
let engines = Services . telemetry . getKeyedHistogramById ( "SEARCH_DEFAULT_ENGINE" ) ;
engines . add ( name , true )
}
2007-09-05 20:02:22 -07:00
// Factory object
const BrowserGlueServiceFactory = {
_instance : null ,
2009-12-17 13:28:08 -08:00
createInstance : function BGSF _createInstance ( outer , iid ) {
2007-09-05 20:02:22 -07:00
if ( outer != null )
throw Components . results . NS _ERROR _NO _AGGREGATION ;
return this . _instance == null ?
this . _instance = new BrowserGlue ( ) : this . _instance ;
}
} ;
2007-03-22 10:30:00 -07:00
// Constructor
function BrowserGlue ( ) {
2009-09-29 03:44:18 -07:00
XPCOMUtils . defineLazyServiceGetter ( this , "_idleService" ,
"@mozilla.org/widget/idleservice;1" ,
"nsIIdleService" ) ;
2009-10-05 03:13:04 -07:00
XPCOMUtils . defineLazyGetter ( this , "_distributionCustomizer" , function ( ) {
2010-01-21 15:49:11 -08:00
Cu . import ( "resource:///modules/distribution.js" ) ;
2009-10-05 03:13:04 -07:00
return new DistributionCustomizer ( ) ;
} ) ;
2009-10-14 02:53:13 -07:00
XPCOMUtils . defineLazyGetter ( this , "_sanitizer" ,
function ( ) {
let sanitizerScope = { } ;
2010-04-12 07:33:10 -07:00
Services . scriptloader . loadSubScript ( "chrome://browser/content/sanitize.js" , sanitizerScope ) ;
2009-10-14 02:53:13 -07:00
return sanitizerScope . Sanitizer ;
} ) ;
2007-03-22 10:30:00 -07:00
this . _init ( ) ;
}
2009-07-30 21:57:57 -07:00
# ifndef XP _MACOSX
# OS X has the concept of zero - window sessions and therefore ignores the
# browser - lastwindow - close - * topics .
# define OBSERVE _LASTWINDOW _CLOSE _TOPICS 1
# endif
2007-03-22 10:30:00 -07:00
BrowserGlue . prototype = {
2007-06-15 11:01:25 -07:00
_saveSession : false ,
2009-09-29 03:44:18 -07:00
_isPlacesInitObserver : false ,
_isPlacesLockedObserver : false ,
2010-04-22 05:53:54 -07:00
_isPlacesShutdownObserver : false ,
2009-09-29 03:44:18 -07:00
_isPlacesDatabaseLocked : false ,
2012-05-28 02:43:59 -07:00
_migrationImportsDefaultBookmarks : false ,
2007-06-15 11:01:25 -07:00
2009-12-17 13:28:08 -08:00
_setPrefToSaveSession : function BG _ _setPrefToSaveSession ( aForce ) {
2009-10-14 02:53:13 -07:00
if ( ! this . _saveSession && ! aForce )
return ;
2010-04-12 07:33:10 -07:00
Services . prefs . setBoolPref ( "browser.sessionstore.resume_session_once" , true ) ;
2009-06-11 13:17:33 -07:00
// This method can be called via [NSApplication terminate:] on Mac, which
// ends up causing prefs not to be flushed to disk, so we need to do that
// explicitly here. See bug 497652.
2010-04-12 07:33:10 -07:00
Services . prefs . savePrefFile ( null ) ;
2008-02-07 01:28:43 -08:00
} ,
2010-08-11 18:28:27 -07:00
# ifdef MOZ _SERVICES _SYNC
_setSyncAutoconnectDelay : function BG _ _setSyncAutoconnectDelay ( ) {
// Assume that a non-zero value for services.sync.autoconnectDelay should override
if ( Services . prefs . prefHasUserValue ( "services.sync.autoconnectDelay" ) ) {
let prefDelay = Services . prefs . getIntPref ( "services.sync.autoconnectDelay" ) ;
if ( prefDelay > 0 )
return ;
}
// delays are in seconds
const MAX _DELAY = 300 ;
let delay = 3 ;
2011-01-22 17:46:22 -08:00
let browserEnum = Services . wm . getEnumerator ( "navigator:browser" ) ;
while ( browserEnum . hasMoreElements ( ) ) {
delay += browserEnum . getNext ( ) . gBrowser . tabs . length ;
2010-08-11 18:28:27 -07:00
}
delay = delay <= MAX _DELAY ? delay : MAX _DELAY ;
2011-07-21 12:20:55 -07:00
Cu . import ( "resource://services-sync/main.js" ) ;
2012-08-29 14:43:41 -07:00
Weave . Service . scheduler . delayedAutoConnect ( delay ) ;
2010-08-11 18:28:27 -07:00
} ,
# endif
2014-11-12 05:16:00 -08:00
// nsIObserver implementation
2009-12-17 13:28:08 -08:00
observe : function BG _observe ( subject , topic , data ) {
2010-02-06 02:40:06 -08:00
switch ( topic ) {
2007-09-11 08:52:04 -07:00
case "prefservice:after-app-defaults" :
this . _onAppDefaults ( ) ;
break ;
2007-03-22 10:30:00 -07:00
case "final-ui-startup" :
2013-05-13 09:25:39 -07:00
this . _finalUIStartup ( ) ;
2007-03-22 10:30:00 -07:00
break ;
2010-10-02 08:53:37 -07:00
case "browser-delayed-startup-finished" :
2013-07-26 22:40:07 -07:00
this . _onFirstWindowLoaded ( subject ) ;
2010-10-02 08:53:37 -07:00
Services . obs . removeObserver ( this , "browser-delayed-startup-finished" ) ;
break ;
2008-03-07 16:16:38 -08:00
case "sessionstore-windows-restored" :
2012-01-18 15:25:05 -08:00
this . _onWindowsRestored ( ) ;
2008-03-07 16:16:38 -08:00
break ;
2007-03-22 10:30:00 -07:00
case "browser:purge-session-history" :
// reset the console service's error buffer
2010-04-12 07:33:10 -07:00
Services . console . logStringMessage ( null ) ; // clear the console (in case it's open)
Services . console . reset ( ) ;
2007-03-22 10:30:00 -07:00
break ;
2007-06-15 11:01:25 -07:00
case "quit-application-requested" :
2007-07-25 14:43:17 -07:00
this . _onQuitRequest ( subject , data ) ;
2007-06-15 11:01:25 -07:00
break ;
case "quit-application-granted" :
2014-02-07 09:47:35 -08:00
this . _onQuitApplicationGranted ( ) ;
2007-06-15 11:01:25 -07:00
break ;
2009-07-30 21:57:57 -07:00
# ifdef OBSERVE _LASTWINDOW _CLOSE _TOPICS
case "browser-lastwindow-close-requested" :
// The application is not actually quitting, but the last full browser
// window is about to be closed.
this . _onQuitRequest ( subject , "lastwindow" ) ;
break ;
case "browser-lastwindow-close-granted" :
2009-10-14 02:53:13 -07:00
this . _setPrefToSaveSession ( ) ;
2009-07-30 21:57:57 -07:00
break ;
2010-08-11 18:28:27 -07:00
# endif
# ifdef MOZ _SERVICES _SYNC
case "weave:service:ready" :
this . _setSyncAutoconnectDelay ( ) ;
break ;
2012-06-29 15:30:45 -07:00
case "weave:engine:clients:display-uri" :
this . _onDisplaySyncURI ( subject ) ;
break ;
2009-07-30 21:57:57 -07:00
# endif
2008-02-07 01:28:43 -08:00
case "session-save" :
2009-10-14 02:53:13 -07:00
this . _setPrefToSaveSession ( true ) ;
2008-02-07 01:28:43 -08:00
subject . QueryInterface ( Ci . nsISupportsPRBool ) ;
subject . data = true ;
break ;
2008-11-12 12:45:53 -08:00
case "places-init-complete" :
2012-05-28 02:43:59 -07:00
if ( ! this . _migrationImportsDefaultBookmarks )
this . _initPlaces ( false ) ;
2010-04-12 07:33:10 -07:00
Services . obs . removeObserver ( this , "places-init-complete" ) ;
2009-09-29 03:44:18 -07:00
this . _isPlacesInitObserver = false ;
2008-12-30 04:44:55 -08:00
// no longer needed, since history was initialized completely.
2010-04-12 07:33:10 -07:00
Services . obs . removeObserver ( this , "places-database-locked" ) ;
2009-09-29 03:44:18 -07:00
this . _isPlacesLockedObserver = false ;
2008-12-30 04:44:55 -08:00
break ;
case "places-database-locked" :
this . _isPlacesDatabaseLocked = true ;
2009-10-14 02:53:13 -07:00
// Stop observing, so further attempts to load history service
// will not show the prompt.
2010-04-12 07:33:10 -07:00
Services . obs . removeObserver ( this , "places-database-locked" ) ;
2009-09-29 03:44:18 -07:00
this . _isPlacesLockedObserver = false ;
2008-11-12 12:45:53 -08:00
break ;
2010-04-22 05:53:54 -07:00
case "places-shutdown" :
if ( this . _isPlacesShutdownObserver ) {
Services . obs . removeObserver ( this , "places-shutdown" ) ;
this . _isPlacesShutdownObserver = false ;
}
2010-08-27 18:24:54 -07:00
// places-shutdown is fired when the profile is about to disappear.
2013-01-29 09:06:03 -08:00
this . _onPlacesShutdown ( ) ;
2010-04-22 05:53:54 -07:00
break ;
2008-03-13 12:25:49 -07:00
case "idle" :
2014-02-04 05:43:20 -08:00
this . _backupBookmarks ( ) ;
2008-03-13 12:25:49 -07:00
break ;
2009-10-05 03:13:04 -07:00
case "distribution-customization-complete" :
2010-04-12 07:33:10 -07:00
Services . obs . removeObserver ( this , "distribution-customization-complete" ) ;
2009-10-05 03:13:04 -07:00
// Customization has finished, we don't need the customizer anymore.
delete this . _distributionCustomizer ;
break ;
2010-04-06 19:49:23 -07:00
case "browser-glue-test" : // used by tests
if ( data == "post-update-notification" ) {
2010-04-12 07:33:10 -07:00
if ( Services . prefs . prefHasUserValue ( "app.update.postupdate" ) )
2010-04-06 19:49:23 -07:00
this . _showUpdateNotification ( ) ;
2010-12-22 10:26:18 -08:00
}
else if ( data == "force-ui-migration" ) {
this . _migrateUI ( ) ;
2010-04-06 19:49:23 -07:00
}
2011-04-12 07:32:20 -07:00
else if ( data == "force-distribution-customization" ) {
2011-04-12 18:15:39 -07:00
this . _distributionCustomizer . applyPrefDefaults ( ) ;
2011-04-12 07:32:20 -07:00
this . _distributionCustomizer . applyCustomizations ( ) ;
2011-04-12 18:15:39 -07:00
// To apply distribution bookmarks use "places-init-complete".
2011-04-12 07:32:20 -07:00
}
2011-04-12 17:43:28 -07:00
else if ( data == "force-places-init" ) {
2012-05-28 02:43:59 -07:00
this . _initPlaces ( false ) ;
2011-04-12 17:43:28 -07:00
}
2010-04-06 19:49:23 -07:00
break ;
2012-05-28 02:43:59 -07:00
case "initial-migration-will-import-default-bookmarks" :
this . _migrationImportsDefaultBookmarks = true ;
break ;
case "initial-migration-did-import-default-bookmarks" :
this . _initPlaces ( true ) ;
2012-03-23 00:15:36 -07:00
break ;
2012-12-13 16:09:03 -08:00
case "handle-xul-text-link" :
let linkHandled = subject . QueryInterface ( Ci . nsISupportsPRBool ) ;
if ( ! linkHandled . data ) {
let win = this . getMostRecentBrowserWindow ( ) ;
if ( win ) {
win . openUILinkIn ( data , "tab" ) ;
linkHandled . data = true ;
}
}
break ;
2013-01-29 09:06:03 -08:00
case "profile-before-change" :
2014-02-07 09:47:35 -08:00
// Any component depending on Places should be finalized in
// _onPlacesShutdown. Any component that doesn't need to act after
// the UI has gone should be finalized in _onQuitApplicationGranted.
this . _dispose ( ) ;
2013-01-29 09:06:03 -08:00
break ;
2013-02-14 14:47:49 -08:00
case "keyword-search" :
// This is very similar to code in
// browser.js:BrowserSearch.recordSearchInHealthReport(). The code could
// be consolidated if there is will. We need the observer in
// nsBrowserGlue to prevent double counting.
2014-07-17 08:37:22 -07:00
let win = this . getMostRecentBrowserWindow ( ) ;
BrowserUITelemetry . countSearchEvent ( "urlbar" , win . gURLBar . value ) ;
2014-10-31 13:47:13 -07:00
let engine = null ;
try {
engine = subject . QueryInterface ( Ci . nsISearchEngine ) ;
} catch ( ex ) {
Cu . reportError ( ex ) ;
}
win . BrowserSearch . recordSearchInTelemetry ( engine , "urlbar" ) ;
2014-07-17 08:37:22 -07:00
# ifdef MOZ _SERVICES _HEALTHREPORT
2013-02-14 14:47:49 -08:00
let reporter = Cc [ "@mozilla.org/datareporting/service;1" ]
. getService ( )
. wrappedJSObject
. healthReporter ;
if ( ! reporter ) {
return ;
}
reporter . onInit ( ) . then ( function record ( ) {
try {
2013-10-18 12:31:39 -07:00
reporter . getProvider ( "org.mozilla.searches" ) . recordSearch ( engine , "urlbar" ) ;
2013-02-14 14:47:49 -08:00
} catch ( ex ) {
Cu . reportError ( ex ) ;
}
} ) ;
# endif
2014-07-17 08:37:22 -07:00
break ;
2013-05-27 08:21:47 -07:00
case "browser-search-engine-modified" :
if ( data != "engine-default" && data != "engine-current" ) {
break ;
}
// Enforce that the search service's defaultEngine is always equal to
// its currentEngine. The search service will notify us any time either
// of them are changed (either by directly setting the relevant prefs,
// i.e. if add-ons try to change this directly, or if the
// nsIBrowserSearchService setters are called).
2013-07-01 16:46:07 -07:00
// No need to initialize the search service, since it's guaranteed to be
// initialized already when this notification fires.
2013-05-27 08:21:47 -07:00
let ss = Services . search ;
if ( ss . currentEngine . name == ss . defaultEngine . name )
return ;
if ( data == "engine-current" )
ss . defaultEngine = ss . currentEngine ;
else
ss . currentEngine = ss . defaultEngine ;
2014-12-16 09:10:28 -08:00
recordDefaultSearchEngine ( ) ;
2013-05-27 08:21:47 -07:00
break ;
2013-07-01 16:46:07 -07:00
case "browser-search-service" :
if ( data != "init-complete" )
return ;
Services . obs . removeObserver ( this , "browser-search-service" ) ;
this . _syncSearchEngines ( ) ;
2014-12-16 09:10:28 -08:00
recordDefaultSearchEngine ( ) ;
2013-07-01 16:46:07 -07:00
break ;
2014-10-28 09:55:14 -07:00
# ifdef NIGHTLY _BUILD
2014-10-23 14:43:51 -07:00
case "nsPref:changed" :
if ( data == POLARIS _ENABLED ) {
let enabled = Services . prefs . getBoolPref ( POLARIS _ENABLED ) ;
2014-11-03 15:21:47 -08:00
if ( enabled ) {
2014-11-11 15:05:55 -08:00
Services . prefs . setBoolPref ( "privacy.donottrackheader.enabled" , enabled ) ;
Services . prefs . setBoolPref ( "privacy.trackingprotection.enabled" , enabled ) ;
Services . prefs . setBoolPref ( "privacy.trackingprotection.ui.enabled" , enabled ) ;
2014-11-03 15:21:47 -08:00
} else {
// Don't reset DNT because its visible pref is independent of
// Polaris and may have been previously set.
Services . prefs . clearUserPref ( "privacy.trackingprotection.enabled" ) ;
Services . prefs . clearUserPref ( "privacy.trackingprotection.ui.enabled" ) ;
}
2014-10-23 14:43:51 -07:00
}
2014-10-28 09:55:14 -07:00
# endif
2013-07-01 16:46:07 -07:00
}
} ,
_syncSearchEngines : function ( ) {
// Only do this if the search service is already initialized. This function
// gets called in finalUIStartup and from a browser-search-service observer,
// to catch both cases (search service initialization occurring before and
// after final-ui-startup)
if ( Services . search . isInitialized ) {
Services . search . defaultEngine = Services . search . currentEngine ;
2007-03-22 10:30:00 -07:00
}
2013-02-14 14:47:49 -08:00
} ,
2008-03-13 12:25:49 -07:00
2007-03-22 10:30:00 -07:00
// initialization (called on application startup)
2009-12-17 13:28:08 -08:00
_init : function BG _ _init ( ) {
2010-04-22 07:11:21 -07:00
let os = Services . obs ;
os . addObserver ( this , "prefservice:after-app-defaults" , false ) ;
os . addObserver ( this , "final-ui-startup" , false ) ;
2010-10-02 08:53:37 -07:00
os . addObserver ( this , "browser-delayed-startup-finished" , false ) ;
2010-04-22 07:11:21 -07:00
os . addObserver ( this , "sessionstore-windows-restored" , false ) ;
os . addObserver ( this , "browser:purge-session-history" , false ) ;
os . addObserver ( this , "quit-application-requested" , false ) ;
os . addObserver ( this , "quit-application-granted" , false ) ;
2009-07-30 21:57:57 -07:00
# ifdef OBSERVE _LASTWINDOW _CLOSE _TOPICS
2010-04-22 07:11:21 -07:00
os . addObserver ( this , "browser-lastwindow-close-requested" , false ) ;
os . addObserver ( this , "browser-lastwindow-close-granted" , false ) ;
2010-08-11 18:28:27 -07:00
# endif
# ifdef MOZ _SERVICES _SYNC
os . addObserver ( this , "weave:service:ready" , false ) ;
2012-06-29 15:30:45 -07:00
os . addObserver ( this , "weave:engine:clients:display-uri" , false ) ;
2009-07-30 21:57:57 -07:00
# endif
2010-04-22 07:11:21 -07:00
os . addObserver ( this , "session-save" , false ) ;
os . addObserver ( this , "places-init-complete" , false ) ;
2009-09-29 03:44:18 -07:00
this . _isPlacesInitObserver = true ;
2010-04-22 07:11:21 -07:00
os . addObserver ( this , "places-database-locked" , false ) ;
2009-09-29 03:44:18 -07:00
this . _isPlacesLockedObserver = true ;
2010-04-22 07:11:21 -07:00
os . addObserver ( this , "distribution-customization-complete" , false ) ;
os . addObserver ( this , "places-shutdown" , false ) ;
2010-04-22 05:53:54 -07:00
this . _isPlacesShutdownObserver = true ;
2012-12-13 16:09:03 -08:00
os . addObserver ( this , "handle-xul-text-link" , false ) ;
2013-01-29 09:06:03 -08:00
os . addObserver ( this , "profile-before-change" , false ) ;
2013-02-14 14:47:49 -08:00
# ifdef MOZ _SERVICES _HEALTHREPORT
os . addObserver ( this , "keyword-search" , false ) ;
# endif
2013-05-27 08:21:47 -07:00
os . addObserver ( this , "browser-search-engine-modified" , false ) ;
2013-07-01 16:46:07 -07:00
os . addObserver ( this , "browser-search-service" , false ) ;
2007-03-22 10:30:00 -07:00
} ,
// cleanup (called on application shutdown)
2009-12-17 13:28:08 -08:00
_dispose : function BG _ _dispose ( ) {
2010-04-22 07:11:21 -07:00
let os = Services . obs ;
os . removeObserver ( this , "prefservice:after-app-defaults" ) ;
os . removeObserver ( this , "final-ui-startup" ) ;
os . removeObserver ( this , "sessionstore-windows-restored" ) ;
os . removeObserver ( this , "browser:purge-session-history" ) ;
os . removeObserver ( this , "quit-application-requested" ) ;
os . removeObserver ( this , "quit-application-granted" ) ;
2009-07-30 21:57:57 -07:00
# ifdef OBSERVE _LASTWINDOW _CLOSE _TOPICS
2010-04-22 07:11:21 -07:00
os . removeObserver ( this , "browser-lastwindow-close-requested" ) ;
os . removeObserver ( this , "browser-lastwindow-close-granted" ) ;
2010-08-11 18:28:27 -07:00
# endif
# ifdef MOZ _SERVICES _SYNC
2013-04-06 11:46:35 -07:00
os . removeObserver ( this , "weave:service:ready" ) ;
os . removeObserver ( this , "weave:engine:clients:display-uri" ) ;
2009-07-30 21:57:57 -07:00
# endif
2010-04-22 07:11:21 -07:00
os . removeObserver ( this , "session-save" ) ;
2014-02-04 05:43:20 -08:00
if ( this . _bookmarksBackupIdleTime ) {
this . _idleService . removeIdleObserver ( this , this . _bookmarksBackupIdleTime ) ;
delete this . _bookmarksBackupIdleTime ;
}
2009-09-29 03:44:18 -07:00
if ( this . _isPlacesInitObserver )
2010-04-22 07:11:21 -07:00
os . removeObserver ( this , "places-init-complete" ) ;
2009-09-29 03:44:18 -07:00
if ( this . _isPlacesLockedObserver )
2010-04-22 07:11:21 -07:00
os . removeObserver ( this , "places-database-locked" ) ;
2010-04-22 05:53:54 -07:00
if ( this . _isPlacesShutdownObserver )
2010-04-22 07:11:21 -07:00
os . removeObserver ( this , "places-shutdown" ) ;
2012-12-13 16:09:03 -08:00
os . removeObserver ( this , "handle-xul-text-link" ) ;
2013-01-29 09:06:03 -08:00
os . removeObserver ( this , "profile-before-change" ) ;
2013-02-14 14:47:49 -08:00
# ifdef MOZ _SERVICES _HEALTHREPORT
os . removeObserver ( this , "keyword-search" ) ;
# endif
2013-05-27 08:21:47 -07:00
os . removeObserver ( this , "browser-search-engine-modified" ) ;
2013-07-01 16:46:07 -07:00
try {
os . removeObserver ( this , "browser-search-service" ) ;
// may have already been removed by the observer
} catch ( ex ) { }
2014-10-23 14:43:51 -07:00
# ifdef NIGHTLY _BUILD
Services . prefs . removeObserver ( POLARIS _ENABLED , this ) ;
# endif
2007-03-22 10:30:00 -07:00
} ,
2009-12-17 13:28:08 -08:00
_onAppDefaults : function BG _ _onAppDefaults ( ) {
2007-09-11 08:52:04 -07:00
// apply distribution customizations (prefs)
2013-05-13 09:25:39 -07:00
// other customizations are applied in _finalUIStartup()
2009-10-05 03:13:04 -07:00
this . _distributionCustomizer . applyPrefDefaults ( ) ;
2007-09-11 08:52:04 -07:00
} ,
2013-05-13 09:25:39 -07:00
// runs on startup, before the first command line handler is invoked
// (i.e. before the first window is opened)
_finalUIStartup : function BG _ _finalUIStartup ( ) {
2009-10-14 02:53:13 -07:00
this . _sanitizer . onStartup ( ) ;
2007-03-22 10:30:00 -07:00
// check if we're in safe mode
2010-04-12 07:33:10 -07:00
if ( Services . appinfo . inSafeMode ) {
Services . ww . openWindow ( null , "chrome://browser/content/safeMode.xul" ,
"_blank" , "chrome,centerscreen,modal,resizable=no" , null ) ;
2007-03-22 10:30:00 -07:00
}
2007-05-03 01:56:41 -07:00
2007-09-11 08:52:04 -07:00
// apply distribution customizations
// prefs are applied in _onAppDefaults()
2009-10-05 03:13:04 -07:00
this . _distributionCustomizer . applyCustomizations ( ) ;
2007-09-11 08:52:04 -07:00
2008-02-04 11:37:26 -08:00
// handle any UI migration
2008-03-20 11:57:05 -07:00
this . _migrateUI ( ) ;
2008-09-30 22:36:21 -07:00
2013-07-01 16:46:07 -07:00
this . _syncSearchEngines ( ) ;
2014-03-15 14:37:37 -07:00
WebappManager . init ( ) ;
2012-05-02 13:11:19 -07:00
PageThumbs . init ( ) ;
2012-08-21 11:17:18 -07:00
NewTabUtils . init ( ) ;
2014-03-31 01:51:21 -07:00
DirectoryLinksProvider . init ( ) ;
NewTabUtils . links . addProvider ( DirectoryLinksProvider ) ;
2014-03-22 20:46:18 -07:00
# ifdef NIGHTLY _BUILD
if ( Services . prefs . getBoolPref ( "dom.identity.enabled" ) ) {
SignInToWebsiteUX . init ( ) ;
}
# endif
2013-10-21 18:39:43 -07:00
# ifdef NIGHTLY _BUILD
2013-09-30 16:23:41 -07:00
ShumwayUtils . init ( ) ;
2013-10-21 18:39:43 -07:00
# endif
2012-10-05 02:37:00 -07:00
webrtcUI . init ( ) ;
2013-07-05 13:55:00 -07:00
AboutHome . init ( ) ;
2013-10-17 11:26:25 -07:00
SessionStore . init ( ) ;
2013-12-10 11:49:02 -08:00
BrowserUITelemetry . init ( ) ;
2014-04-24 19:09:20 -07:00
ContentSearch . init ( ) ;
2014-08-21 08:42:42 -07:00
FormValidationHandler . init ( ) ;
2012-05-02 13:11:19 -07:00
2014-08-11 11:13:36 -07:00
ContentClick . init ( ) ;
RemotePrompt . init ( ) ;
ContentPrefServiceParent . init ( ) ;
2013-08-08 17:16:47 -07:00
2014-06-28 11:09:45 -07:00
LoginManagerParent . init ( ) ;
2014-11-03 14:42:10 -08:00
# ifdef NIGHTLY _BUILD
Services . prefs . addObserver ( POLARIS _ENABLED , this , false ) ;
# endif
2010-04-12 07:33:10 -07:00
Services . obs . notifyObservers ( null , "browser-ui-startup-complete" , "" ) ;
2007-05-10 17:42:22 -07:00
} ,
2007-05-03 01:56:41 -07:00
2013-07-30 20:19:58 -07:00
_checkForOldBuildUpdates : function ( ) {
// check for update if our build is old
if ( Services . prefs . getBoolPref ( "app.update.enabled" ) &&
Services . prefs . getBoolPref ( "app.update.checkInstallTime" ) ) {
let buildID = Services . appinfo . appBuildID ;
let today = new Date ( ) . getTime ( ) ;
let buildDate = new Date ( buildID . slice ( 0 , 4 ) , // year
buildID . slice ( 4 , 6 ) - 1 , // months are zero-based.
buildID . slice ( 6 , 8 ) , // day
buildID . slice ( 8 , 10 ) , // hour
buildID . slice ( 10 , 12 ) , // min
buildID . slice ( 12 , 14 ) ) // ms
. getTime ( ) ;
const millisecondsIn24Hours = 86400000 ;
let acceptableAge = Services . prefs . getIntPref ( "app.update.checkInstallTime.days" ) * millisecondsIn24Hours ;
if ( buildDate + acceptableAge < today ) {
Cc [ "@mozilla.org/updates/update-service;1" ] . getService ( Ci . nsIApplicationUpdateService ) . checkForBackgroundUpdates ( ) ;
}
}
} ,
2013-02-01 13:47:07 -08:00
_trackSlowStartup : function ( ) {
2013-02-19 15:15:38 -08:00
if ( Services . startup . interrupted ||
Services . prefs . getBoolPref ( "browser.slowStartup.notificationDisabled" ) )
2013-02-01 13:47:07 -08:00
return ;
let currentTime = Date . now ( ) - Services . startup . getStartupInfo ( ) . process ;
let averageTime = 0 ;
let samples = 0 ;
try {
averageTime = Services . prefs . getIntPref ( "browser.slowStartup.averageTime" ) ;
samples = Services . prefs . getIntPref ( "browser.slowStartup.samples" ) ;
} catch ( e ) { }
2013-08-12 16:55:05 -07:00
let totalTime = ( averageTime * samples ) + currentTime ;
samples ++ ;
averageTime = totalTime / samples ;
2013-02-01 13:47:07 -08:00
if ( samples >= Services . prefs . getIntPref ( "browser.slowStartup.maxSamples" ) ) {
if ( averageTime > Services . prefs . getIntPref ( "browser.slowStartup.timeThreshold" ) )
this . _showSlowStartupNotification ( ) ;
averageTime = 0 ;
samples = 0 ;
}
Services . prefs . setIntPref ( "browser.slowStartup.averageTime" , averageTime ) ;
Services . prefs . setIntPref ( "browser.slowStartup.samples" , samples ) ;
} ,
_showSlowStartupNotification : function ( ) {
let win = this . getMostRecentBrowserWindow ( ) ;
if ( ! win )
return ;
let productName = Services . strings
. createBundle ( "chrome://branding/locale/brand.properties" )
. GetStringFromName ( "brandFullName" ) ;
let message = win . gNavigatorBundle . getFormattedString ( "slowStartup.message" , [ productName ] ) ;
let buttons = [
{
label : win . gNavigatorBundle . getString ( "slowStartup.helpButton.label" ) ,
accessKey : win . gNavigatorBundle . getString ( "slowStartup.helpButton.accesskey" ) ,
callback : function ( ) {
2013-08-23 18:30:07 -07:00
win . openUILinkIn ( "https://support.mozilla.org/kb/reset-firefox-easily-fix-most-problems" , "tab" ) ;
2013-02-01 13:47:07 -08:00
}
} ,
{
label : win . gNavigatorBundle . getString ( "slowStartup.disableNotificationButton.label" ) ,
accessKey : win . gNavigatorBundle . getString ( "slowStartup.disableNotificationButton.accesskey" ) ,
callback : function ( ) {
Services . prefs . setBoolPref ( "browser.slowStartup.notificationDisabled" , true ) ;
}
}
] ;
let nb = win . document . getElementById ( "global-notificationbox" ) ;
nb . appendNotification ( message , "slow-startup" ,
"chrome://browser/skin/slowStartup-16.png" ,
nb . PRIORITY _INFO _LOW , buttons ) ;
} ,
2013-06-28 20:25:08 -07:00
/ * *
* Show a notification bar offering a reset if the profile has been unused for some time .
* /
_resetUnusedProfileNotification : function ( ) {
let win = this . getMostRecentBrowserWindow ( ) ;
if ( ! win )
return ;
Cu . import ( "resource://gre/modules/ResetProfile.jsm" ) ;
if ( ! ResetProfile . resetSupported ( ) )
return ;
let productName = Services . strings
. createBundle ( "chrome://branding/locale/brand.properties" )
. GetStringFromName ( "brandShortName" ) ;
let resetBundle = Services . strings
. createBundle ( "chrome://global/locale/resetProfile.properties" ) ;
let message = resetBundle . formatStringFromName ( "resetUnusedProfile.message" , [ productName ] , 1 ) ;
let buttons = [
{
2014-10-10 18:37:16 -07:00
label : resetBundle . formatStringFromName ( "refreshProfile.resetButton.label" , [ productName ] , 1 ) ,
accessKey : resetBundle . GetStringFromName ( "refreshProfile.resetButton.accesskey" ) ,
2013-06-28 20:25:08 -07:00
callback : function ( ) {
ResetProfile . openConfirmationDialog ( win ) ;
}
} ,
] ;
let nb = win . document . getElementById ( "global-notificationbox" ) ;
nb . appendNotification ( message , "reset-unused-profile" ,
"chrome://global/skin/icons/question-16.png" ,
nb . PRIORITY _INFO _LOW , buttons ) ;
} ,
2014-06-27 11:20:02 -07:00
_firstWindowTelemetry : function ( aWindow ) {
# ifdef XP _WIN
let SCALING _PROBE _NAME = "DISPLAY_SCALING_MSWIN" ;
# elifdef XP _MACOSX
let SCALING _PROBE _NAME = "DISPLAY_SCALING_OSX" ;
# elifdef XP _LINUX
let SCALING _PROBE _NAME = "DISPLAY_SCALING_LINUX" ;
# else
let SCALING _PROBE _NAME = "" ;
# endif
if ( SCALING _PROBE _NAME ) {
let scaling = aWindow . devicePixelRatio * 100 ;
Services . telemetry . getHistogramById ( SCALING _PROBE _NAME ) . add ( scaling ) ;
}
} ,
2010-10-02 08:53:37 -07:00
// the first browser window has finished initializing
2013-07-26 22:40:07 -07:00
_onFirstWindowLoaded : function BG _ _onFirstWindowLoaded ( aWindow ) {
2014-09-22 09:33:29 -07:00
// Initialize PdfJs when running in-process and remote. This only
// happens once since PdfJs registers global hooks. If the PdfJs
// extension is installed the init method below will be overridden
// leaving initialization to the extension.
// parent only: configure default prefs, set up pref observers, register
// pdf content handler, and initializes parent side message manager
// shim for privileged api access.
PdfJs . init ( true ) ;
// child only: similar to the call above for parent - register content
// handler and init message manager child shim for privileged api access.
// With older versions of the extension installed, this load will fail
// passively.
aWindow . messageManager . loadFrameScript ( "resource://pdf.js/pdfjschildbootstrap.js" , true ) ;
2010-10-02 08:53:37 -07:00
# ifdef XP _WIN
// For windows seven, initialize the jump list module.
const WINTASKBAR _CONTRACTID = "@mozilla.org/windows-taskbar;1" ;
if ( WINTASKBAR _CONTRACTID in Cc &&
Cc [ WINTASKBAR _CONTRACTID ] . getService ( Ci . nsIWinTaskbar ) . available ) {
let temp = { } ;
2013-05-28 19:21:53 -07:00
Cu . import ( "resource:///modules/WindowsJumpLists.jsm" , temp ) ;
2010-10-02 08:53:37 -07:00
temp . WinTaskbarJumpList . startup ( ) ;
}
# endif
2013-02-01 13:47:07 -08:00
2014-11-06 23:12:26 -08:00
// A channel for "remote troubleshooting" code...
let channel = new WebChannel ( "remote-troubleshooting" , "remote-troubleshooting" ) ;
channel . listen ( ( id , data , target ) => {
if ( data . command == "request" ) {
let { Troubleshoot } = Cu . import ( "resource://gre/modules/Troubleshoot.jsm" , { } ) ;
Troubleshoot . snapshot ( data => {
// for privacy we remove crash IDs and all preferences (but bug 1091944
// exists to expose prefs once we are confident of privacy implications)
delete data . crashes ;
delete data . modifiedPreferences ;
channel . send ( data , target ) ;
} ) ;
}
} ) ;
2013-02-01 13:47:07 -08:00
this . _trackSlowStartup ( ) ;
2013-06-28 20:25:08 -07:00
// Offer to reset a user's profile if it hasn't been used for 60 days.
const OFFER _PROFILE _RESET _INTERVAL _MS = 60 * 24 * 60 * 60 * 1000 ;
let lastUse = Services . appinfo . replacedLockTime ;
2014-03-23 17:41:06 -07:00
let disableResetPrompt = false ;
try {
disableResetPrompt = Services . prefs . getBoolPref ( "browser.disableResetPrompt" ) ;
} catch ( e ) { }
if ( ! disableResetPrompt && lastUse &&
2013-07-27 08:22:56 -07:00
Date . now ( ) - lastUse >= OFFER _PROFILE _RESET _INTERVAL _MS ) {
2013-06-28 20:25:08 -07:00
this . _resetUnusedProfileNotification ( ) ;
}
2013-07-30 20:19:58 -07:00
this . _checkForOldBuildUpdates ( ) ;
2014-06-27 11:20:02 -07:00
this . _firstWindowTelemetry ( aWindow ) ;
2010-10-02 08:53:37 -07:00
} ,
2013-01-29 09:06:03 -08:00
/ * *
2014-02-07 09:47:35 -08:00
* Application shutdown handler .
2013-01-29 09:06:03 -08:00
* /
2014-02-07 09:47:35 -08:00
_onQuitApplicationGranted : function ( ) {
// This pref must be set here because SessionStore will use its value
// on quit-application.
this . _setPrefToSaveSession ( ) ;
2014-02-08 13:04:31 -08:00
// Call trackStartupCrashEnd here in case the delayed call on startup hasn't
// yet occurred (see trackStartupCrashEnd caller in browser.js).
2014-02-07 09:47:35 -08:00
try {
let appStartup = Cc [ "@mozilla.org/toolkit/app-startup;1" ]
. getService ( Ci . nsIAppStartup ) ;
appStartup . trackStartupCrashEnd ( ) ;
} catch ( e ) {
Cu . reportError ( "Could not end startup crash tracking in quit-application-granted: " + e ) ;
}
2014-02-25 09:51:41 -08:00
CustomizationTabPreloader . uninit ( ) ;
2014-03-15 14:37:37 -07:00
WebappManager . uninit ( ) ;
2014-03-22 20:46:18 -07:00
# ifdef NIGHTLY _BUILD
if ( Services . prefs . getBoolPref ( "dom.identity.enabled" ) ) {
SignInToWebsiteUX . uninit ( ) ;
}
# endif
2013-01-29 09:06:03 -08:00
webrtcUI . uninit ( ) ;
2014-08-21 08:42:42 -07:00
FormValidationHandler . uninit ( ) ;
2007-03-22 10:30:00 -07:00
} ,
2014-10-17 15:55:14 -07:00
_initServiceDiscovery : function ( ) {
var rokuDevice = {
id : "roku:ecp" ,
target : "roku:ecp" ,
factory : function ( aService ) {
Cu . import ( "resource://gre/modules/RokuApp.jsm" ) ;
return new RokuApp ( aService ) ;
} ,
2014-12-15 15:52:55 -08:00
mirror : true ,
2014-10-17 15:55:14 -07:00
types : [ "video/mp4" ] ,
extensions : [ "mp4" ]
} ;
// Register targets
SimpleServiceDiscovery . registerDevice ( rokuDevice ) ;
// Search for devices continuously every 120 seconds
SimpleServiceDiscovery . search ( 120 * 1000 ) ;
} ,
2012-01-18 15:25:05 -08:00
// All initial windows have opened.
_onWindowsRestored : function BG _ _onWindowsRestored ( ) {
2014-11-05 10:21:42 -08:00
# ifdef MOZ _DEV _EDITION
this . _createExtraDefaultProfile ( ) ;
# endif
2014-10-17 15:55:14 -07:00
this . _initServiceDiscovery ( ) ;
2010-04-06 19:49:23 -07:00
// Show update notification, if needed.
2010-04-12 07:33:10 -07:00
if ( Services . prefs . prefHasUserValue ( "app.update.postupdate" ) )
2010-04-06 19:49:23 -07:00
this . _showUpdateNotification ( ) ;
2008-12-30 04:44:55 -08:00
// Load the "more info" page for a locked places.sqlite
2010-08-09 08:59:59 -07:00
// This property is set earlier by places-database-locked topic.
2008-12-30 04:44:55 -08:00
if ( this . _isPlacesDatabaseLocked ) {
this . _showPlacesLockedNotificationBox ( ) ;
}
2009-10-02 04:26:04 -07:00
// If there are plugins installed that are outdated, and the user hasn't
// been warned about them yet, open the plugins update page.
2010-04-12 07:33:10 -07:00
if ( Services . prefs . getBoolPref ( PREF _PLUGINS _NOTIFYUSER ) )
2009-10-02 04:26:04 -07:00
this . _showPluginUpdatePage ( ) ;
2011-08-12 13:50:47 -07:00
// For any add-ons that were installed disabled and can be enabled offer
2012-02-15 15:44:29 -08:00
// them to the user.
let changedIDs = AddonManager . getStartupChanges ( AddonManager . STARTUP _CHANGE _INSTALLED ) ;
if ( changedIDs . length > 0 ) {
2012-11-30 20:11:44 -08:00
let win = this . getMostRecentBrowserWindow ( ) ;
2012-02-15 15:44:29 -08:00
AddonManager . getAddonsByIDs ( changedIDs , function ( aAddons ) {
aAddons . forEach ( function ( aAddon ) {
// If the add-on isn't user disabled or can't be enabled then skip it.
if ( ! aAddon . userDisabled || ! ( aAddon . permissions & AddonManager . PERM _CAN _ENABLE ) )
return ;
2012-11-30 20:11:44 -08:00
win . openUILinkIn ( "about:newaddon?id=" + aAddon . id , "tab" ) ;
2012-02-15 15:44:29 -08:00
} )
} ) ;
}
2012-02-06 17:39:07 -08:00
2012-03-06 14:28:53 -08:00
// Perform default browser checking.
2014-08-12 00:59:11 -07:00
if ( ShellService ) {
2012-03-06 14:28:53 -08:00
# ifdef DEBUG
2012-03-21 12:05:23 -07:00
let shouldCheck = false ;
2012-03-06 14:28:53 -08:00
# else
2014-08-12 00:59:11 -07:00
let shouldCheck = ShellService . shouldCheckDefaultBrowser ;
2012-03-06 14:28:53 -08:00
# endif
2012-03-21 12:05:23 -07:00
let willRecoverSession = false ;
2012-03-06 14:28:53 -08:00
try {
2012-03-21 12:05:23 -07:00
let ss = Cc [ "@mozilla.org/browser/sessionstartup;1" ] .
2012-03-06 14:28:53 -08:00
getService ( Ci . nsISessionStartup ) ;
willRecoverSession =
( ss . sessionType == Ci . nsISessionStartup . RECOVER _SESSION ) ;
}
catch ( ex ) { /* never mind; suppose SessionStore is broken */ }
2013-05-30 10:32:17 -07:00
2014-08-12 00:59:11 -07:00
// startup check, check all assoc
let isDefault = ShellService . isDefaultBrowser ( true , false ) ;
2013-05-30 10:32:17 -07:00
try {
// Report default browser status on startup to telemetry
// so we can track whether we are the default.
Services . telemetry . getHistogramById ( "BROWSER_IS_USER_DEFAULT" )
. add ( isDefault ) ;
}
catch ( ex ) { /* Don't break the default prompt if telemetry is broken. */ }
if ( shouldCheck && ! isDefault && ! willRecoverSession ) {
2012-03-06 14:28:53 -08:00
Services . tm . mainThread . dispatch ( function ( ) {
2014-08-12 00:59:11 -07:00
DefaultBrowserCheck . prompt ( this . getMostRecentBrowserWindow ( ) ) ;
2012-03-21 12:05:23 -07:00
} . bind ( this ) , Ci . nsIThread . DISPATCH _NORMAL ) ;
2012-03-06 14:28:53 -08:00
}
}
2014-09-10 15:06:19 -07:00
# ifdef E10S _TESTING _ONLY
E10SUINotification . checkStatus ( ) ;
# endif
2008-03-07 16:16:38 -08:00
} ,
2014-11-05 10:21:42 -08:00
# ifdef MOZ _DEV _EDITION
_createExtraDefaultProfile : function ( ) {
// If Developer Edition is the only installed Firefox version and no other
// profiles are present, create a second one for use by other versions.
// This helps Firefox versions earlier than 35 avoid accidentally using the
// unsuitable Developer Edition profile.
let profileService = Cc [ "@mozilla.org/toolkit/profile-service;1" ]
. getService ( Ci . nsIToolkitProfileService ) ;
let profileCount = profileService . profileCount ;
if ( profileCount == 1 && profileService . selectedProfile . name != "default" ) {
let newProfile ;
try {
newProfile = profileService . createProfile ( null , "default" ) ;
profileService . defaultProfile = newProfile ;
profileService . flush ( ) ;
} catch ( e ) {
Cu . reportError ( "Could not create profile 'default': " + e ) ;
}
if ( newProfile ) {
// We don't want a default profile with Developer Edition settings, an
// empty profile directory will do. The profile service of the other
// Firefox will populate it with its own stuff.
let newProfilePath = newProfile . rootDir . path ;
OS . File . removeDir ( newProfilePath ) . then ( ( ) => {
return OS . File . makeDir ( newProfilePath ) ;
} ) . then ( null , e => {
Cu . reportError ( "Could not empty profile 'default': " + e ) ;
} ) ;
}
}
} ,
# endif
2009-12-17 13:28:08 -08:00
_onQuitRequest : function BG _ _onQuitRequest ( aCancelQuit , aQuitType ) {
2008-01-29 13:16:02 -08:00
// If user has already dismissed quit request, then do nothing
if ( ( aCancelQuit instanceof Ci . nsISupportsPRBool ) && aCancelQuit . data )
return ;
2011-02-07 17:50:37 -08:00
// There are several cases where we won't show a dialog here:
// 1. There is only 1 tab open in 1 window
// 2. The session will be restored at startup, indicated by
// browser.startup.page == 3 or browser.sessionstore.resume_session_once == true
// 3. browser.warnOnQuit == false
// 4. The browser is currently in Private Browsing mode
2012-12-28 11:41:13 -08:00
// 5. The browser will be restarted.
2011-02-07 17:50:37 -08:00
//
// Otherwise these are the conditions and the associated dialogs that will be shown:
// 1. aQuitType == "lastwindow" or "quit" and browser.showQuitWarning == true
// - The quit dialog will be shown
2012-12-28 11:41:13 -08:00
// 2. aQuitType == "lastwindow" && browser.tabs.warnOnClose == true
2011-02-07 17:50:37 -08:00
// - The "closing multiple tabs" dialog will be shown
//
// aQuitType == "lastwindow" is overloaded. "lastwindow" is used to indicate
// "the last window is closing but we're not quitting (a non-browser window is open)"
// and also "we're quitting by closing the last window".
2012-12-28 11:41:13 -08:00
if ( aQuitType == "restart" )
return ;
2007-06-15 11:01:25 -07:00
var windowcount = 0 ;
2008-05-10 08:52:32 -07:00
var pagecount = 0 ;
2010-04-12 07:33:10 -07:00
var browserEnum = Services . wm . getEnumerator ( "navigator:browser" ) ;
2012-06-28 14:22:41 -07:00
let allWindowsPrivate = true ;
2008-05-10 08:52:32 -07:00
while ( browserEnum . hasMoreElements ( ) ) {
2013-09-13 14:29:51 -07:00
// XXXbz should we skip closed windows here?
2008-05-10 08:52:32 -07:00
windowcount ++ ;
var browser = browserEnum . getNext ( ) ;
2012-10-05 16:25:52 -07:00
if ( ! PrivateBrowsingUtils . isWindowPrivate ( browser ) )
2012-06-28 14:22:41 -07:00
allWindowsPrivate = false ;
2008-05-10 08:52:32 -07:00
var tabbrowser = browser . document . getElementById ( "content" ) ;
if ( tabbrowser )
2010-08-13 05:40:19 -07:00
pagecount += tabbrowser . browsers . length - tabbrowser . _numPinnedTabs ;
2008-05-10 08:52:32 -07:00
}
2007-06-15 11:01:25 -07:00
this . _saveSession = false ;
2008-05-10 08:52:32 -07:00
if ( pagecount < 2 )
return ;
2007-06-15 11:01:25 -07:00
2011-02-07 17:50:37 -08:00
if ( ! aQuitType )
2007-07-25 14:43:17 -07:00
aQuitType = "quit" ;
2011-02-07 17:50:37 -08:00
// browser.warnOnQuit is a hidden global boolean to override all quit prompts
// browser.showQuitWarning specifically covers quitting
// browser.tabs.warnOnClose is the global "warn when closing multiple tabs" pref
var sessionWillBeRestored = Services . prefs . getIntPref ( "browser.startup.page" ) == 3 ||
Services . prefs . getBoolPref ( "browser.sessionstore.resume_session_once" ) ;
if ( sessionWillBeRestored || ! Services . prefs . getBoolPref ( "browser.warnOnQuit" ) )
return ;
2014-08-08 07:04:45 -07:00
let win = Services . wm . getMostRecentWindow ( "navigator:browser" ) ;
2011-02-07 17:50:37 -08:00
// On last window close or quit && showQuitWarning, we want to show the
// quit warning.
2012-12-28 11:41:13 -08:00
if ( ! Services . prefs . getBoolPref ( "browser.showQuitWarning" ) ) {
if ( aQuitType == "lastwindow" ) {
// If aQuitType is "lastwindow" and we aren't showing the quit warning,
// we should show the window closing warning instead. warnAboutClosing
// tabs checks browser.tabs.warnOnClose and returns if it's ok to close
// the window. It doesn't actually close the window.
2014-08-08 07:04:45 -07:00
aCancelQuit . data =
! win . gBrowser . warnAboutClosingTabs ( win . gBrowser . closingTabsEnum . ALL ) ;
2012-12-28 11:41:13 -08:00
}
2011-02-07 17:50:37 -08:00
return ;
}
2014-08-08 07:04:45 -07:00
let prompt = Services . prompt ;
2014-08-08 07:04:45 -07:00
let quitBundle = Services . strings . createBundle ( "chrome://browser/locale/quitDialog.properties" ) ;
let brandBundle = Services . strings . createBundle ( "chrome://branding/locale/brand.properties" ) ;
2008-05-08 08:08:40 -07:00
2014-08-08 07:04:45 -07:00
let appName = brandBundle . GetStringFromName ( "brandShortName" ) ;
let quitDialogTitle = quitBundle . formatStringFromName ( "quitDialogTitle" ,
[ appName ] , 1 ) ;
2014-08-08 07:04:45 -07:00
let neverAskText = quitBundle . GetStringFromName ( "neverAsk2" ) ;
let neverAsk = { value : false } ;
let choice ;
if ( allWindowsPrivate ) {
let text = quitBundle . formatStringFromName ( "messagePrivate" , [ appName ] , 1 ) ;
let flags = prompt . BUTTON _TITLE _IS _STRING * prompt . BUTTON _POS _0 +
prompt . BUTTON _TITLE _IS _STRING * prompt . BUTTON _POS _1 +
prompt . BUTTON _POS _0 _DEFAULT ;
choice = prompt . confirmEx ( win , quitDialogTitle , text , flags ,
quitBundle . GetStringFromName ( "quitTitle" ) ,
quitBundle . GetStringFromName ( "cancelTitle" ) ,
null ,
neverAskText , neverAsk ) ;
// The order of the buttons differs between the prompt.confirmEx calls
// here so we need to fix this for proper handling below.
if ( choice == 0 ) {
choice = 2 ;
}
} else {
let text = quitBundle . formatStringFromName (
windowcount == 1 ? "messageNoWindows" : "message" , [ appName ] , 1 ) ;
let flags = prompt . BUTTON _TITLE _IS _STRING * prompt . BUTTON _POS _0 +
prompt . BUTTON _TITLE _IS _STRING * prompt . BUTTON _POS _1 +
prompt . BUTTON _TITLE _IS _STRING * prompt . BUTTON _POS _2 +
prompt . BUTTON _POS _0 _DEFAULT ;
choice = prompt . confirmEx ( win , quitDialogTitle , text , flags ,
quitBundle . GetStringFromName ( "saveTitle" ) ,
quitBundle . GetStringFromName ( "cancelTitle" ) ,
quitBundle . GetStringFromName ( "quitTitle" ) ,
neverAskText , neverAsk ) ;
}
2008-05-08 08:08:40 -07:00
2014-08-08 07:04:45 -07:00
switch ( choice ) {
2008-05-08 08:08:40 -07:00
case 2 : // Quit
if ( neverAsk . value )
2011-02-07 17:50:37 -08:00
Services . prefs . setBoolPref ( "browser.showQuitWarning" , false ) ;
2008-05-08 08:08:40 -07:00
break ;
case 1 : // Cancel
aCancelQuit . QueryInterface ( Ci . nsISupportsPRBool ) ;
aCancelQuit . data = true ;
break ;
case 0 : // Save & Quit
this . _saveSession = true ;
if ( neverAsk . value ) {
2012-12-28 11:41:13 -08:00
// always save state when shutting down
Services . prefs . setIntPref ( "browser.startup.page" , 3 ) ;
2007-06-15 11:01:25 -07:00
}
2008-05-08 08:08:40 -07:00
break ;
2007-06-15 11:01:25 -07:00
}
} ,
2010-04-06 19:49:23 -07:00
_showUpdateNotification : function BG _ _showUpdateNotification ( ) {
2010-04-12 07:33:10 -07:00
Services . prefs . clearUserPref ( "app.update.postupdate" ) ;
2010-04-06 19:49:23 -07:00
var um = Cc [ "@mozilla.org/updates/update-manager;1" ] .
getService ( Ci . nsIUpdateManager ) ;
try {
// If the updates.xml file is deleted then getUpdateAt will throw.
var update = um . getUpdateAt ( 0 ) . QueryInterface ( Ci . nsIPropertyBag ) ;
}
catch ( e ) {
// This should never happen.
Cu . reportError ( "Unable to find update: " + e ) ;
return ;
}
var actions = update . getProperty ( "actions" ) ;
if ( ! actions || actions . indexOf ( "silent" ) != - 1 )
return ;
var formatter = Cc [ "@mozilla.org/toolkit/URLFormatterService;1" ] .
getService ( Ci . nsIURLFormatter ) ;
2010-04-12 07:33:10 -07:00
var browserBundle = Services . strings . createBundle ( "chrome://browser/locale/browser.properties" ) ;
var brandBundle = Services . strings . createBundle ( "chrome://branding/locale/brand.properties" ) ;
2010-04-06 19:49:23 -07:00
var appName = brandBundle . GetStringFromName ( "brandShortName" ) ;
function getNotifyString ( aPropData ) {
var propValue = update . getProperty ( aPropData . propName ) ;
if ( ! propValue ) {
if ( aPropData . prefName )
propValue = formatter . formatURLPref ( aPropData . prefName ) ;
else if ( aPropData . stringParams )
propValue = browserBundle . formatStringFromName ( aPropData . stringName ,
aPropData . stringParams ,
aPropData . stringParams . length ) ;
else
propValue = browserBundle . GetStringFromName ( aPropData . stringName ) ;
}
return propValue ;
}
if ( actions . indexOf ( "showNotification" ) != - 1 ) {
let text = getNotifyString ( { propName : "notificationText" ,
stringName : "puNotifyText" ,
stringParams : [ appName ] } ) ;
let url = getNotifyString ( { propName : "notificationURL" ,
prefName : "startup.homepage_override_url" } ) ;
let label = getNotifyString ( { propName : "notificationButtonLabel" ,
stringName : "pu.notifyButton.label" } ) ;
let key = getNotifyString ( { propName : "notificationButtonAccessKey" ,
stringName : "pu.notifyButton.accesskey" } ) ;
let win = this . getMostRecentBrowserWindow ( ) ;
2012-11-30 20:11:44 -08:00
let notifyBox = win . gBrowser . getNotificationBox ( ) ;
2010-04-06 19:49:23 -07:00
let buttons = [
{
label : label ,
accessKey : key ,
popup : null ,
callback : function ( aNotificationBar , aButton ) {
2012-11-30 20:11:44 -08:00
win . openUILinkIn ( url , "tab" ) ;
2010-04-06 19:49:23 -07:00
}
}
] ;
2011-11-04 14:45:01 -07:00
let notification = notifyBox . appendNotification ( text , "post-update-notification" ,
null , notifyBox . PRIORITY _INFO _LOW ,
buttons ) ;
notification . persistence = - 1 ; // Until user closes it
2010-04-06 19:49:23 -07:00
}
if ( actions . indexOf ( "showAlert" ) == - 1 )
return ;
let notifier ;
try {
notifier = Cc [ "@mozilla.org/alerts-service;1" ] .
getService ( Ci . nsIAlertsService ) ;
}
catch ( e ) {
// nsIAlertsService is not available for this platform
return ;
}
let title = getNotifyString ( { propName : "alertTitle" ,
stringName : "puAlertTitle" ,
stringParams : [ appName ] } ) ;
let text = getNotifyString ( { propName : "alertText" ,
stringName : "puAlertText" ,
stringParams : [ appName ] } ) ;
let url = getNotifyString ( { propName : "alertURL" ,
prefName : "startup.homepage_override_url" } ) ;
var self = this ;
function clickCallback ( subject , topic , data ) {
// This callback will be called twice but only once with this topic
if ( topic != "alertclickcallback" )
return ;
let win = self . getMostRecentBrowserWindow ( ) ;
2012-11-30 20:11:44 -08:00
win . openUILinkIn ( data , "tab" ) ;
2010-04-06 19:49:23 -07:00
}
try {
// This will throw NS_ERROR_NOT_AVAILABLE if the notification cannot
// be displayed per the idl.
2012-05-04 11:23:55 -07:00
notifier . showAlertNotification ( null , title , text ,
2010-04-06 19:49:23 -07:00
true , url , clickCallback ) ;
}
catch ( e ) {
}
} ,
2009-12-17 13:28:08 -08:00
_showPluginUpdatePage : function BG _ _showPluginUpdatePage ( ) {
2010-04-12 07:33:10 -07:00
Services . prefs . setBoolPref ( PREF _PLUGINS _NOTIFYUSER , false ) ;
2009-10-02 04:26:04 -07:00
var formatter = Cc [ "@mozilla.org/toolkit/URLFormatterService;1" ] .
getService ( Ci . nsIURLFormatter ) ;
var updateUrl = formatter . formatURLPref ( PREF _PLUGINS _UPDATEURL ) ;
var win = this . getMostRecentBrowserWindow ( ) ;
2012-11-30 20:11:44 -08:00
win . openUILinkIn ( updateUrl , "tab" ) ;
2009-10-02 04:26:04 -07:00
} ,
2008-10-30 23:42:02 -07:00
2007-05-03 01:56:41 -07:00
/ * *
* Initialize Places
2008-11-12 12:45:53 -08:00
* - imports the bookmarks html file if bookmarks database is empty , try to
* restore bookmarks from a JSON backup if the backend indicates that the
* database was corrupt .
*
* These prefs can be set up by the frontend :
*
* WARNING : setting these preferences to true will overwite existing bookmarks
2008-03-28 09:43:17 -07:00
*
* - browser . places . importBookmarksHTML
2008-11-12 12:45:53 -08:00
* Set to true will import the bookmarks . html file from the profile folder .
2008-04-08 11:42:53 -07:00
* - browser . places . smartBookmarksVersion
* Set during HTML import to indicate that Smart Bookmarks were created .
* Set to - 1 to disable Smart Bookmarks creation .
* Set to 0 to restore current Smart Bookmarks .
2008-04-02 12:33:01 -07:00
* - browser . bookmarks . restore _default _bookmarks
* Set to true by safe - mode dialog to indicate we must restore default
* bookmarks .
2007-05-03 01:56:41 -07:00
* /
2012-05-28 02:43:59 -07:00
_initPlaces : function BG _ _initPlaces ( aInitialMigrationPerformed ) {
2008-11-12 12:45:53 -08:00
// We must instantiate the history service since it will tell us if we
// need to import or restore bookmarks due to first-run, corruption or
// forced migration (due to a major schema change).
// If the database is corrupt or has been newly created we should
// import bookmarks.
2014-03-03 08:51:42 -08:00
let dbStatus = PlacesUtils . history . databaseStatus ;
let importBookmarks = ! aInitialMigrationPerformed &&
2012-03-23 00:15:36 -07:00
( dbStatus == PlacesUtils . history . DATABASE _STATUS _CREATE ||
dbStatus == PlacesUtils . history . DATABASE _STATUS _CORRUPT ) ;
2009-02-09 10:16:02 -08:00
2008-11-12 12:45:53 -08:00
// Check if user or an extension has required to import bookmarks.html
2014-03-03 08:51:42 -08:00
let importBookmarksHTML = false ;
2008-11-12 12:45:53 -08:00
try {
importBookmarksHTML =
2010-04-12 07:33:10 -07:00
Services . prefs . getBoolPref ( "browser.places.importBookmarksHTML" ) ;
2008-11-12 12:45:53 -08:00
if ( importBookmarksHTML )
importBookmarks = true ;
} catch ( ex ) { }
2007-12-04 00:08:33 -08:00
2014-03-25 09:25:58 -07:00
// Support legacy bookmarks.html format for apps that depend on that format.
let autoExportHTML = false ;
try {
autoExportHTML = Services . prefs . getBoolPref ( "browser.bookmarks.autoExportHTML" ) ;
} catch ( ex ) { } // Do not export.
if ( autoExportHTML ) {
// Sqlite.jsm and Places shutdown happen at profile-before-change, thus,
// to be on the safe side, this should run earlier.
AsyncShutdown . profileChangeTeardown . addBlocker (
"Places: export bookmarks.html" ,
( ) => BookmarkHTMLUtils . exportToFile ( BookmarkHTMLUtils . defaultPath ) ) ;
}
2013-04-09 01:23:40 -07:00
Task . spawn ( function ( ) {
// Check if Safe Mode or the user has required to restore bookmarks from
// default profile's bookmarks.html
2014-03-03 08:51:42 -08:00
let restoreDefaultBookmarks = false ;
2013-04-09 01:23:40 -07:00
try {
restoreDefaultBookmarks =
Services . prefs . getBoolPref ( "browser.bookmarks.restore_default_bookmarks" ) ;
if ( restoreDefaultBookmarks ) {
// Ensure that we already have a bookmarks backup for today.
2013-10-06 23:52:08 -07:00
yield this . _backupBookmarks ( ) ;
2013-04-09 01:23:40 -07:00
importBookmarks = true ;
}
} catch ( ex ) { }
2014-02-04 05:43:20 -08:00
// This may be reused later, check for "=== undefined" to see if it has
// been populated already.
let lastBackupFile ;
2013-04-09 01:23:40 -07:00
// If the user did not require to restore default bookmarks, or import
// from bookmarks.html, we will try to restore from JSON
if ( importBookmarks && ! restoreDefaultBookmarks && ! importBookmarksHTML ) {
// get latest JSON backup
2014-05-20 23:39:00 -07:00
lastBackupFile = yield PlacesBackups . getMostRecentBackup ( ) ;
2014-02-04 05:43:20 -08:00
if ( lastBackupFile ) {
2013-04-09 01:23:40 -07:00
// restore from JSON backup
2014-02-04 05:43:20 -08:00
yield BookmarkJSONUtils . importFromFile ( lastBackupFile , true ) ;
2013-04-09 01:23:40 -07:00
importBookmarks = false ;
}
else {
// We have created a new database but we don't have any backup available
importBookmarks = true ;
2014-03-03 08:51:42 -08:00
if ( yield OS . File . exists ( BookmarkHTMLUtils . defaultPath ) ) {
2013-04-09 01:23:40 -07:00
// If bookmarks.html is available in current profile import it...
importBookmarksHTML = true ;
}
else {
// ...otherwise we will restore defaults
restoreDefaultBookmarks = true ;
}
}
2008-11-12 12:45:53 -08:00
}
2007-12-04 03:22:14 -08:00
2013-04-09 01:23:40 -07:00
// If bookmarks are not imported, then initialize smart bookmarks. This
// happens during a common startup.
// Otherwise, if any kind of import runs, smart bookmarks creation should be
// delayed till the import operations has finished. Not doing so would
// cause them to be overwritten by the newly imported bookmarks.
if ( ! importBookmarks ) {
// Now apply distribution customized bookmarks.
// This should always run after Places initialization.
this . _distributionCustomizer . applyBookmarks ( ) ;
this . ensurePlacesDefaultQueriesInitialized ( ) ;
2008-11-12 12:45:53 -08:00
}
else {
2013-04-09 01:23:40 -07:00
// An import operation is about to run.
// Don't try to recreate smart bookmarks if autoExportHTML is true or
// smart bookmarks are disabled.
2014-03-03 08:51:42 -08:00
let smartBookmarksVersion = 0 ;
2013-04-09 01:23:40 -07:00
try {
smartBookmarksVersion = Services . prefs . getIntPref ( "browser.places.smartBookmarksVersion" ) ;
} catch ( ex ) { }
if ( ! autoExportHTML && smartBookmarksVersion != - 1 )
Services . prefs . setIntPref ( "browser.places.smartBookmarksVersion" , 0 ) ;
2014-03-03 08:51:42 -08:00
let bookmarksUrl = null ;
2013-04-09 01:23:40 -07:00
if ( restoreDefaultBookmarks ) {
// User wants to restore bookmarks.html file from default profile folder
2014-03-03 08:51:42 -08:00
bookmarksUrl = "resource:///defaults/profile/bookmarks.html" ;
2008-12-09 12:39:32 -08:00
}
2014-03-03 08:51:42 -08:00
else if ( yield OS . File . exists ( BookmarkHTMLUtils . defaultPath ) ) {
bookmarksUrl = OS . Path . toFileURI ( BookmarkHTMLUtils . defaultPath ) ;
2008-12-09 12:39:32 -08:00
}
2008-11-12 12:45:53 -08:00
2014-03-03 08:51:42 -08:00
if ( bookmarksUrl ) {
2013-04-09 01:23:40 -07:00
// Import from bookmarks.html file.
try {
2014-03-03 08:51:42 -08:00
BookmarkHTMLUtils . importFromURL ( bookmarksUrl , true ) . then ( null ,
2013-04-09 01:23:40 -07:00
function onFailure ( ) {
Cu . reportError ( "Bookmarks.html file could be corrupt." ) ;
}
) . then (
function onComplete ( ) {
// Now apply distribution customized bookmarks.
// This should always run after Places initialization.
this . _distributionCustomizer . applyBookmarks ( ) ;
// Ensure that smart bookmarks are created once the operation is
// complete.
this . ensurePlacesDefaultQueriesInitialized ( ) ;
} . bind ( this )
) ;
} catch ( err ) {
Cu . reportError ( "Bookmarks.html file could be corrupt. " + err ) ;
}
}
else {
Cu . reportError ( "Unable to find bookmarks.html file." ) ;
}
2008-11-12 12:45:53 -08:00
2013-04-09 01:23:40 -07:00
// Reset preferences, so we won't try to import again at next run
if ( importBookmarksHTML )
Services . prefs . setBoolPref ( "browser.places.importBookmarksHTML" , false ) ;
if ( restoreDefaultBookmarks )
Services . prefs . setBoolPref ( "browser.bookmarks.restore_default_bookmarks" ,
false ) ;
2007-05-03 01:56:41 -07:00
}
2008-11-18 09:25:39 -08:00
2013-04-09 01:23:40 -07:00
// Initialize bookmark archiving on idle.
2014-02-04 05:43:20 -08:00
if ( ! this . _bookmarksBackupIdleTime ) {
this . _bookmarksBackupIdleTime = BOOKMARKS _BACKUP _IDLE _TIME _SEC ;
// If there is no backup, or the last bookmarks backup is too old, use
// a more aggressive idle observer.
if ( lastBackupFile === undefined )
lastBackupFile = yield PlacesBackups . getMostRecentBackup ( ) ;
if ( ! lastBackupFile ) {
this . _bookmarksBackupIdleTime /= 2 ;
}
else {
let lastBackupTime = PlacesBackups . getDateForFile ( lastBackupFile ) ;
let profileLastUse = Services . appinfo . replacedLockTime || Date . now ( ) ;
// If there is a backup after the last profile usage date it's fine,
// regardless its age. Otherwise check how old is the last
// available backup compared to that session.
if ( profileLastUse > lastBackupTime ) {
let backupAge = Math . round ( ( profileLastUse - lastBackupTime ) / 86400000 ) ;
// Report the age of the last available backup.
try {
Services . telemetry
. getHistogramById ( "PLACES_BACKUPS_DAYSFROMLAST" )
. add ( backupAge ) ;
} catch ( ex ) {
Components . utils . reportError ( "Unable to report telemetry." ) ;
}
if ( backupAge > BOOKMARKS _BACKUP _MAX _INTERVAL _DAYS )
this . _bookmarksBackupIdleTime /= 2 ;
}
}
this . _idleService . addIdleObserver ( this , this . _bookmarksBackupIdleTime ) ;
2012-03-23 00:15:36 -07:00
}
2013-04-16 22:36:02 -07:00
Services . obs . notifyObservers ( null , "places-browser-init-complete" , "" ) ;
2013-04-09 01:23:40 -07:00
} . bind ( this ) ) ;
2007-05-03 01:56:41 -07:00
} ,
/ * *
* Places shut - down tasks
2013-01-29 09:06:03 -08:00
* - finalize components depending on Places .
2014-02-04 05:43:20 -08:00
* - export bookmarks as HTML , if so configured .
2007-05-03 01:56:41 -07:00
* /
2013-01-29 09:06:03 -08:00
_onPlacesShutdown : function BG _ _onPlacesShutdown ( ) {
this . _sanitizer . onShutdown ( ) ;
PageThumbs . uninit ( ) ;
2014-02-04 05:43:20 -08:00
if ( this . _bookmarksBackupIdleTime ) {
this . _idleService . removeIdleObserver ( this , this . _bookmarksBackupIdleTime ) ;
delete this . _bookmarksBackupIdleTime ;
2010-04-22 05:53:54 -07:00
}
2008-03-13 12:25:49 -07:00
} ,
2013-04-09 01:23:40 -07:00
/ * *
2014-02-04 05:43:20 -08:00
* If a backup for today doesn ' t exist , this creates one .
2013-04-09 01:23:40 -07:00
* /
_backupBookmarks : function BG _ _backupBookmarks ( ) {
return Task . spawn ( function ( ) {
2013-10-06 23:52:08 -07:00
let lastBackupFile = yield PlacesBackups . getMostRecentBackup ( ) ;
// Should backup bookmarks if there are no backups or the maximum
// interval between backups elapsed.
if ( ! lastBackupFile ||
2014-02-04 05:43:20 -08:00
new Date ( ) - PlacesBackups . getDateForFile ( lastBackupFile ) > BOOKMARKS _BACKUP _MIN _INTERVAL _DAYS * 86400000 ) {
let maxBackups = Services . prefs . getIntPref ( "browser.bookmarks.max_backups" ) ;
yield PlacesBackups . create ( maxBackups ) ;
2013-10-06 23:52:08 -07:00
}
2013-04-09 01:23:40 -07:00
} ) ;
2007-05-03 01:56:41 -07:00
} ,
2007-11-26 16:08:50 -08:00
2008-12-30 04:44:55 -08:00
/ * *
* Show the notificationBox for a locked places database .
* /
2009-12-17 13:28:08 -08:00
_showPlacesLockedNotificationBox : function BG _ _showPlacesLockedNotificationBox ( ) {
2010-04-12 07:33:10 -07:00
var brandBundle = Services . strings . createBundle ( "chrome://branding/locale/brand.properties" ) ;
2008-12-30 04:44:55 -08:00
var applicationName = brandBundle . GetStringFromName ( "brandShortName" ) ;
2010-04-12 07:33:10 -07:00
var placesBundle = Services . strings . createBundle ( "chrome://browser/locale/places/places.properties" ) ;
2008-12-30 04:44:55 -08:00
var title = placesBundle . GetStringFromName ( "lockPrompt.title" ) ;
var text = placesBundle . formatStringFromName ( "lockPrompt.text" , [ applicationName ] , 1 ) ;
var buttonText = placesBundle . GetStringFromName ( "lockPromptInfoButton.label" ) ;
var accessKey = placesBundle . GetStringFromName ( "lockPromptInfoButton.accessKey" ) ;
var helpTopic = "places-locked" ;
var url = Cc [ "@mozilla.org/toolkit/URLFormatterService;1" ] .
getService ( Components . interfaces . nsIURLFormatter ) .
formatURLPref ( "app.support.baseURL" ) ;
url += helpTopic ;
2012-11-30 20:11:44 -08:00
var win = this . getMostRecentBrowserWindow ( ) ;
2008-12-30 04:44:55 -08:00
var buttons = [
{
label : buttonText ,
accessKey : accessKey ,
popup : null ,
callback : function ( aNotificationBar , aButton ) {
2012-11-30 20:11:44 -08:00
win . openUILinkIn ( url , "tab" ) ;
2008-12-30 04:44:55 -08:00
}
}
] ;
2012-11-30 20:11:44 -08:00
var notifyBox = win . gBrowser . getNotificationBox ( ) ;
2011-11-04 14:45:01 -07:00
var notification = notifyBox . appendNotification ( text , title , null ,
notifyBox . PRIORITY _CRITICAL _MEDIUM ,
buttons ) ;
notification . persistence = - 1 ; // Until user closes it
2008-12-30 04:44:55 -08:00
} ,
2009-12-17 13:28:08 -08:00
_migrateUI : function BG _ _migrateUI ( ) {
2014-11-26 15:13:57 -08:00
const UI _VERSION = 27 ;
2014-08-19 07:31:00 -07:00
const BROWSER _DOCURL = "chrome://browser/content/browser.xul" ;
2013-11-04 06:11:44 -08:00
let currentUIVersion = 0 ;
try {
currentUIVersion = Services . prefs . getIntPref ( "browser.migration.version" ) ;
} catch ( ex ) { }
2013-11-05 01:23:40 -08:00
if ( currentUIVersion >= UI _VERSION )
2013-11-04 06:11:44 -08:00
return ;
2014-08-19 07:31:00 -07:00
let xulStore = Cc [ "@mozilla.org/xul/xulstore;1" ] . getService ( Ci . nsIXULStore ) ;
2008-02-04 11:37:26 -08:00
2010-07-09 12:44:01 -07:00
if ( currentUIVersion < 2 ) {
// This code adds the customizable bookmarks button.
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2010-07-09 12:44:01 -07:00
// Need to migrate only if toolbar is customized and the element is not found.
if ( currentset &&
currentset . indexOf ( "bookmarks-menu-button-container" ) == - 1 ) {
2012-03-28 01:33:17 -07:00
currentset += ",bookmarks-menu-button-container" ;
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2010-07-09 12:44:01 -07:00
}
}
2008-02-04 11:37:26 -08:00
2010-09-02 19:20:12 -07:00
if ( currentUIVersion < 3 ) {
// This code merges the reload/stop/go button into the url bar.
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2010-09-02 19:20:12 -07:00
// Need to migrate only if toolbar is customized and all 3 elements are found.
if ( currentset &&
currentset . indexOf ( "reload-button" ) != - 1 &&
currentset . indexOf ( "stop-button" ) != - 1 &&
currentset . indexOf ( "urlbar-container" ) != - 1 &&
currentset . indexOf ( "urlbar-container,reload-button,stop-button" ) == - 1 ) {
2010-12-08 21:24:46 -08:00
currentset = currentset . replace ( /(^|,)reload-button($|,)/ , "$1$2" )
. replace ( /(^|,)stop-button($|,)/ , "$1$2" )
. replace ( /(^|,)urlbar-container($|,)/ ,
2010-09-02 19:20:12 -07:00
"$1urlbar-container,reload-button,stop-button$2" ) ;
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2010-09-02 19:20:12 -07:00
}
}
2010-12-08 21:24:46 -08:00
if ( currentUIVersion < 4 ) {
// This code moves the home button to the immediate left of the bookmarks menu button.
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2010-12-08 21:24:46 -08:00
// Need to migrate only if toolbar is customized and the elements are found.
if ( currentset &&
currentset . indexOf ( "home-button" ) != - 1 &&
currentset . indexOf ( "bookmarks-menu-button-container" ) != - 1 ) {
currentset = currentset . replace ( /(^|,)home-button($|,)/ , "$1$2" )
. replace ( /(^|,)bookmarks-menu-button-container($|,)/ ,
"$1home-button,bookmarks-menu-button-container$2" ) ;
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2010-12-08 21:24:46 -08:00
}
}
2010-12-22 10:26:18 -08:00
if ( currentUIVersion < 5 ) {
// This code uncollapses PersonalToolbar if its collapsed status is not
// persisted, and user customized it or changed default bookmarks.
2014-08-19 07:31:00 -07:00
//
2010-12-22 10:26:18 -08:00
// If the user does not have a persisted value for the toolbar's
// "collapsed" attribute, try to determine whether it's customized.
2014-08-19 07:31:00 -07:00
if ( ! xulStore . hasValue ( BROWSER _DOCURL , "PersonalToolbar" , "collapsed" ) ) {
2010-12-22 10:26:18 -08:00
// We consider the toolbar customized if it has more than
// 3 children, or if it has a persisted currentset value.
2014-08-19 07:31:00 -07:00
let toolbarIsCustomized = xulStore . hasValue ( BROWSER _DOCURL ,
"PersonalToolbar" , "currentset" ) ;
2014-08-10 11:16:32 -07:00
let getToolbarFolderCount = function ( ) {
2010-12-22 10:26:18 -08:00
let toolbarFolder =
PlacesUtils . getFolderContents ( PlacesUtils . toolbarFolderId ) . root ;
let toolbarChildCount = toolbarFolder . childCount ;
toolbarFolder . containerOpen = false ;
return toolbarChildCount ;
2014-08-10 11:16:32 -07:00
} ;
2010-12-22 10:26:18 -08:00
if ( toolbarIsCustomized || getToolbarFolderCount ( ) > 3 ) {
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "PersonalToolbar" , "collapsed" , "false" ) ;
2010-12-22 10:26:18 -08:00
}
}
}
2013-01-15 07:47:58 -08:00
if ( currentUIVersion < 9 ) {
2012-08-25 02:37:55 -07:00
// This code adds the customizable downloads buttons.
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2012-08-25 02:37:55 -07:00
// Since the Downloads button is located in the navigation bar by default,
// migration needs to happen only if the toolbar was customized using a
// previous UI version, and the button was not already placed on the
// toolbar manually.
if ( currentset &&
currentset . indexOf ( "downloads-button" ) == - 1 ) {
// The element is added either after the search bar or before the home
// button. As a last resort, the element is added just before the
// non-customizable window controls.
if ( currentset . indexOf ( "search-container" ) != - 1 ) {
currentset = currentset . replace ( /(^|,)search-container($|,)/ ,
"$1search-container,downloads-button$2" )
} else if ( currentset . indexOf ( "home-button" ) != - 1 ) {
currentset = currentset . replace ( /(^|,)home-button($|,)/ ,
"$1downloads-button,home-button$2" )
} else {
currentset = currentset . replace ( /(^|,)window-controls($|,)/ ,
"$1downloads-button,window-controls$2" )
}
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2012-08-25 02:37:55 -07:00
}
2012-09-27 05:57:39 -07:00
}
2013-04-09 14:07:02 -07:00
# ifdef XP _WIN
if ( currentUIVersion < 10 ) {
// For Windows systems with display set to > 96dpi (i.e. systemDefaultScale
// will return a value > 1.0), we want to discard any saved full-zoom settings,
// as we'll now be scaling the content according to the system resolution
// scale factor (Windows "logical DPI" setting)
let sm = Cc [ "@mozilla.org/gfx/screenmanager;1" ] . getService ( Ci . nsIScreenManager ) ;
if ( sm . systemDefaultScale > 1.0 ) {
let cps2 = Cc [ "@mozilla.org/content-pref/service;1" ] .
getService ( Ci . nsIContentPrefService2 ) ;
cps2 . removeByName ( "browser.content.full-zoom" , null ) ;
}
}
# endif
2013-04-08 10:09:51 -07:00
if ( currentUIVersion < 11 ) {
Services . prefs . clearUserPref ( "dom.disable_window_move_resize" ) ;
Services . prefs . clearUserPref ( "dom.disable_window_flip" ) ;
Services . prefs . clearUserPref ( "dom.event.contextmenu.enabled" ) ;
Services . prefs . clearUserPref ( "javascript.enabled" ) ;
Services . prefs . clearUserPref ( "permissions.default.image" ) ;
}
2013-04-23 07:06:17 -07:00
if ( currentUIVersion < 12 ) {
// Remove bookmarks-menu-button-container, then place
// bookmarks-menu-button into its position.
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2013-04-23 07:06:17 -07:00
// Need to migrate only if toolbar is customized.
if ( currentset ) {
2013-05-10 08:19:48 -07:00
if ( currentset . contains ( "bookmarks-menu-button-container" ) ) {
currentset = currentset . replace ( /(^|,)bookmarks-menu-button-container($|,)/ ,
"$1bookmarks-menu-button$2" ) ;
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2013-04-23 07:06:17 -07:00
}
}
}
2013-07-03 14:41:25 -07:00
if ( currentUIVersion < 14 ) {
// DOM Storage doesn't specially handle about: pages anymore.
let path = OS . Path . join ( OS . Constants . Path . profileDir ,
"chromeappsstore.sqlite" ) ;
OS . File . remove ( path ) ;
}
2013-07-04 15:49:58 -07:00
if ( currentUIVersion < 16 ) {
2014-08-19 07:31:00 -07:00
let isCollapsed = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "collapsed" ) ;
2013-05-14 09:15:36 -07:00
if ( isCollapsed == "true" ) {
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "collapsed" , "false" ) ;
2013-05-14 09:15:36 -07:00
}
}
2013-06-07 11:46:53 -07:00
// Insert the bookmarks-menu-button into the nav-bar if it isn't already
// there.
2013-07-04 15:49:58 -07:00
if ( currentUIVersion < 17 ) {
2014-08-19 07:31:00 -07:00
let currentset = xulStore . getValue ( BROWSER _DOCURL , "nav-bar" , "currentset" ) ;
2013-06-07 11:46:53 -07:00
// Need to migrate only if toolbar is customized.
if ( currentset ) {
if ( ! currentset . contains ( "bookmarks-menu-button" ) ) {
// The button isn't in the nav-bar, so let's look for an appropriate
// place to put it.
if ( currentset . contains ( "downloads-button" ) ) {
currentset = currentset . replace ( /(^|,)downloads-button($|,)/ ,
"$1bookmarks-menu-button,downloads-button$2" ) ;
} else if ( currentset . contains ( "home-button" ) ) {
currentset = currentset . replace ( /(^|,)home-button($|,)/ ,
"$1bookmarks-menu-button,home-button$2" ) ;
} else {
// Just append.
currentset = currentset . replace ( /(^|,)window-controls($|,)/ ,
"$1bookmarks-menu-button,window-controls$2" )
}
2014-08-19 07:31:00 -07:00
xulStore . setValue ( BROWSER _DOCURL , "nav-bar" , "currentset" , currentset ) ;
2013-06-07 11:46:53 -07:00
}
}
}
2013-11-22 11:59:37 -08:00
if ( currentUIVersion < 18 ) {
// Remove iconsize and mode from all the toolbars
let toolbars = [ "navigator-toolbox" , "nav-bar" , "PersonalToolbar" ,
"addon-bar" , "TabsToolbar" , "toolbar-menubar" ] ;
for ( let resourceName of [ "mode" , "iconsize" ] ) {
for ( let toolbarId of toolbars ) {
2014-08-19 07:31:00 -07:00
xulStore . removeValue ( BROWSER _DOCURL , toolbarId , resourceName ) ;
2013-11-22 11:59:37 -08:00
}
}
}
2013-12-19 08:01:50 -08:00
if ( currentUIVersion < 19 ) {
let detector = null ;
try {
detector = Services . prefs . getComplexValue ( "intl.charset.detector" ,
Ci . nsIPrefLocalizedString ) . data ;
} catch ( ex ) { }
if ( ! ( detector == "" ||
detector == "ja_parallel_state_machine" ||
detector == "ruprob" ||
detector == "ukprob" ) ) {
// If the encoding detector pref value is not reachable from the UI,
// reset to default (varies by localization).
Services . prefs . clearUserPref ( "intl.charset.detector" ) ;
}
}
2014-02-21 03:51:55 -08:00
if ( currentUIVersion < 20 ) {
// Remove persisted collapsed state from TabsToolbar.
2014-08-19 07:31:00 -07:00
xulStore . removeValue ( BROWSER _DOCURL , "TabsToolbar" , "collapsed" ) ;
2014-02-21 03:51:55 -08:00
}
2014-03-06 08:29:12 -08:00
if ( currentUIVersion < 21 ) {
// Make sure the 'toolbarbutton-1' class will always be present from here
// on out.
2014-08-19 07:31:00 -07:00
xulStore . removeValue ( BROWSER _DOCURL , "bookmarks-menu-button" , "class" ) ;
2014-03-06 08:29:12 -08:00
}
2014-03-12 15:30:41 -07:00
if ( currentUIVersion < 22 ) {
// Reset the Sync promobox count to promote the new FxAccount-based Sync.
Services . prefs . clearUserPref ( "browser.syncPromoViewsLeft" ) ;
Services . prefs . clearUserPref ( "browser.syncPromoViewsLeftMap" ) ;
}
2014-08-04 16:02:19 -07:00
if ( currentUIVersion < 23 ) {
const kSelectedEnginePref = "browser.search.selectedEngine" ;
if ( Services . prefs . prefHasUserValue ( kSelectedEnginePref ) ) {
try {
let name = Services . prefs . getComplexValue ( kSelectedEnginePref ,
Ci . nsIPrefLocalizedString ) . data ;
Services . search . currentEngine = Services . search . getEngineByName ( name ) ;
} catch ( ex ) { }
}
}
2014-10-18 05:39:34 -07:00
if ( currentUIVersion < 24 ) {
// Reset homepage pref for users who have it set to start.mozilla.org
// or google.com/firefox.
const HOMEPAGE _PREF = "browser.startup.homepage" ;
2014-10-24 03:56:08 -07:00
if ( Services . prefs . prefHasUserValue ( HOMEPAGE _PREF ) ) {
const DEFAULT =
Services . prefs . getDefaultBranch ( HOMEPAGE _PREF )
. getComplexValue ( "" , Ci . nsIPrefLocalizedString ) . data ;
let value =
Services . prefs . getComplexValue ( HOMEPAGE _PREF , Ci . nsISupportsString ) ;
let updated =
value . data . replace ( /https?:\/\/start\.mozilla\.org[^|]*/i , DEFAULT )
. replace ( /https?:\/\/(www\.)?google\.[a-z.]+\/firefox[^|]*/i ,
DEFAULT ) ;
if ( updated != value . data ) {
if ( updated == DEFAULT ) {
Services . prefs . clearUserPref ( HOMEPAGE _PREF ) ;
} else {
value . data = updated ;
Services . prefs . setComplexValue ( HOMEPAGE _PREF ,
Ci . nsISupportsString , value ) ;
}
}
2014-10-18 05:39:34 -07:00
}
}
2014-01-28 09:26:16 -08:00
if ( currentUIVersion < 25 ) {
// Make sure the doNotTrack value conforms to the conversion from
// three-state to two-state. (This reverts a setting of "please track me"
// to the default "don't say anything").
try {
if ( Services . prefs . getBoolPref ( "privacy.donottrackheader.enabled" ) &&
Services . prefs . getIntPref ( "privacy.donottrackheader.value" ) != 1 ) {
Services . prefs . clearUserPref ( "privacy.donottrackheader.enabled" ) ;
Services . prefs . clearUserPref ( "privacy.donottrackheader.value" ) ;
}
}
catch ( ex ) { }
}
2014-11-12 05:16:00 -08:00
if ( currentUIVersion < 26 ) {
// Refactor urlbar suggestion preferences to make it extendable and
// allow new suggestion types (e.g: search suggestions).
let types = [ "history" , "bookmark" , "openpage" ] ;
let defaultBehavior = 0 ;
try {
defaultBehavior = Services . prefs . getIntPref ( "browser.urlbar.default.behavior" ) ;
} catch ( ex ) { }
try {
let autocompleteEnabled = Services . prefs . getBoolPref ( "browser.urlbar.autocomplete.enabled" ) ;
if ( ! autocompleteEnabled ) {
defaultBehavior = - 1 ;
}
} catch ( ex ) { }
// If the default behavior is:
// -1 - all new "...suggest.*" preferences will be false
// 0 - all new "...suggest.*" preferences will use the default values
// > 0 - all new "...suggest.*" preferences will be inherited
for ( let type of types ) {
let prefValue = defaultBehavior == 0 ;
if ( defaultBehavior > 0 ) {
prefValue = ! ! ( defaultBehavior & Ci . mozIPlacesAutoComplete [ "BEHAVIOR_" + type . toUpperCase ( ) ] ) ;
}
Services . prefs . setBoolPref ( "browser.urlbar.suggest." + type , prefValue ) ;
}
// Typed behavior will be used only for results from history.
if ( defaultBehavior != - 1 &&
! ! ( defaultBehavior & Ci . mozIPlacesAutoComplete [ "BEHAVIOR_TYPED" ] ) ) {
Services . prefs . setBoolPref ( "browser.urlbar.suggest.history.onlyTyped" , true ) ;
}
}
2014-11-26 15:13:57 -08:00
if ( currentUIVersion < 27 ) {
// Fix up document color use:
const kOldColorPref = "browser.display.use_document_colors" ;
if ( Services . prefs . prefHasUserValue ( kOldColorPref ) &&
! Services . prefs . getBoolPref ( kOldColorPref ) ) {
Services . prefs . setIntPref ( "browser.display.document_color_use" , 2 ) ;
}
}
2010-07-09 12:44:01 -07:00
// Update the migration version.
Services . prefs . setIntPref ( "browser.migration.version" , UI _VERSION ) ;
2008-02-04 11:37:26 -08:00
} ,
2007-03-22 10:30:00 -07:00
// ------------------------------
// public nsIBrowserGlue members
// ------------------------------
2010-09-02 19:20:12 -07:00
2009-12-17 13:28:08 -08:00
sanitize : function BG _sanitize ( aParentWindow ) {
2009-10-14 02:53:13 -07:00
this . _sanitizer . sanitize ( aParentWindow ) ;
2007-09-05 20:02:22 -07:00
} ,
2007-03-22 10:30:00 -07:00
2009-12-17 13:28:08 -08:00
ensurePlacesDefaultQueriesInitialized :
function BG _ensurePlacesDefaultQueriesInitialized ( ) {
2009-03-26 05:58:32 -07:00
// This is actual version of the smart bookmarks, must be increased every
// time smart bookmarks change.
// When adding a new smart bookmark below, its newInVersion property must
// be set to the version it has been added in, we will compare its value
// to users' smartBookmarksVersion and add new smart bookmarks without
// recreating old deleted ones.
2014-03-10 09:53:19 -07:00
const SMART _BOOKMARKS _VERSION = 7 ;
2008-04-08 11:42:53 -07:00
const SMART _BOOKMARKS _ANNO = "Places/SmartBookmark" ;
const SMART _BOOKMARKS _PREF = "browser.places.smartBookmarksVersion" ;
2009-10-14 02:53:13 -07:00
// TODO bug 399268: should this be a pref?
2008-04-08 11:42:53 -07:00
const MAX _RESULTS = 10 ;
2010-11-23 05:04:07 -08:00
// Get current smart bookmarks version. If not set, create them.
let smartBookmarksCurrentVersion = 0 ;
2007-11-26 16:08:50 -08:00
try {
2010-04-12 07:33:10 -07:00
smartBookmarksCurrentVersion = Services . prefs . getIntPref ( SMART _BOOKMARKS _PREF ) ;
2010-11-23 05:04:07 -08:00
} catch ( ex ) { }
2007-11-26 16:08:50 -08:00
2010-11-23 05:04:07 -08:00
// If version is current or smart bookmarks are disabled, just bail out.
2008-04-08 11:42:53 -07:00
if ( smartBookmarksCurrentVersion == - 1 ||
2010-11-23 05:04:07 -08:00
smartBookmarksCurrentVersion >= SMART _BOOKMARKS _VERSION ) {
2007-11-26 16:08:50 -08:00
return ;
2010-11-23 05:04:07 -08:00
}
2007-11-26 16:08:50 -08:00
2010-11-23 05:04:07 -08:00
let batch = {
2009-12-17 13:28:08 -08:00
runBatched : function BG _EPDQI _runBatched ( ) {
2010-11-23 05:04:07 -08:00
let menuIndex = 0 ;
let toolbarIndex = 0 ;
let bundle = Services . strings . createBundle ( "chrome://browser/locale/places/places.properties" ) ;
let smartBookmarks = {
MostVisited : {
title : bundle . GetStringFromName ( "mostVisitedTitle" ) ,
2012-04-06 16:17:53 -07:00
uri : NetUtil . newURI ( "place:sort=" +
2010-11-23 05:04:07 -08:00
Ci . nsINavHistoryQueryOptions . SORT _BY _VISITCOUNT _DESCENDING +
"&maxResults=" + MAX _RESULTS ) ,
parent : PlacesUtils . toolbarFolderId ,
2014-03-10 09:53:19 -07:00
get position ( ) { return toolbarIndex ++ ; } ,
2010-11-23 05:04:07 -08:00
newInVersion : 1
} ,
RecentlyBookmarked : {
title : bundle . GetStringFromName ( "recentlyBookmarkedTitle" ) ,
uri : NetUtil . newURI ( "place:folder=BOOKMARKS_MENU" +
2008-04-08 11:42:53 -07:00
"&folder=UNFILED_BOOKMARKS" +
"&folder=TOOLBAR" +
"&queryType=" +
Ci . nsINavHistoryQueryOptions . QUERY _TYPE _BOOKMARKS +
"&sort=" +
Ci . nsINavHistoryQueryOptions . SORT _BY _DATEADDED _DESCENDING +
"&maxResults=" + MAX _RESULTS +
"&excludeQueries=1" ) ,
2010-11-23 05:04:07 -08:00
parent : PlacesUtils . bookmarksMenuFolderId ,
2014-03-10 09:53:19 -07:00
get position ( ) { return menuIndex ++ ; } ,
2010-11-23 05:04:07 -08:00
newInVersion : 1
} ,
RecentTags : {
title : bundle . GetStringFromName ( "recentTagsTitle" ) ,
uri : NetUtil . newURI ( "place:" +
"type=" +
Ci . nsINavHistoryQueryOptions . RESULTS _AS _TAG _QUERY +
"&sort=" +
Ci . nsINavHistoryQueryOptions . SORT _BY _LASTMODIFIED _DESCENDING +
"&maxResults=" + MAX _RESULTS ) ,
parent : PlacesUtils . bookmarksMenuFolderId ,
2014-03-10 09:53:19 -07:00
get position ( ) { return menuIndex ++ ; } ,
2010-11-23 05:04:07 -08:00
newInVersion : 1
2013-11-23 12:03:34 -08:00
} ,
2013-12-10 10:42:16 -08:00
} ;
2014-02-28 05:37:31 -08:00
if ( Services . metro && Services . metro . supported ) {
2013-12-10 10:42:16 -08:00
smartBookmarks . Windows8Touch = {
2014-03-10 09:53:19 -07:00
title : PlacesUtils . getString ( "windows8TouchTitle" ) ,
get uri ( ) {
let metroBookmarksRoot = PlacesUtils . annotations . getItemsWithAnnotation ( 'metro/bookmarksRoot' , { } ) ;
if ( metroBookmarksRoot . length > 0 ) {
return NetUtil . newURI ( "place:folder=" +
metroBookmarksRoot [ 0 ] +
"&queryType=" +
Ci . nsINavHistoryQueryOptions . QUERY _TYPE _BOOKMARKS +
"&sort=" +
Ci . nsINavHistoryQueryOptions . SORT _BY _DATEADDED _DESCENDING +
"&maxResults=" + MAX _RESULTS +
"&excludeQueries=1" )
}
return null ;
} ,
2013-11-23 12:03:34 -08:00
parent : PlacesUtils . bookmarksMenuFolderId ,
2014-03-10 09:53:19 -07:00
get position ( ) { return menuIndex ++ ; } ,
newInVersion : 7
2013-12-10 10:42:16 -08:00
} ;
}
2010-11-23 05:04:07 -08:00
2009-03-26 05:58:32 -07:00
// Set current itemId, parent and position if Smart Bookmark exists,
// we will use these informations to create the new version at the same
// position.
2010-11-23 05:04:07 -08:00
let smartBookmarkItemIds = PlacesUtils . annotations . getItemsWithAnnotation ( SMART _BOOKMARKS _ANNO ) ;
smartBookmarkItemIds . forEach ( function ( itemId ) {
let queryId = PlacesUtils . annotations . getItemAnnotation ( itemId , SMART _BOOKMARKS _ANNO ) ;
if ( queryId in smartBookmarks ) {
let smartBookmark = smartBookmarks [ queryId ] ;
2014-03-10 09:53:19 -07:00
if ( ! smartBookmark . uri ) {
PlacesUtils . bookmarks . removeItem ( itemId ) ;
return ;
}
2012-04-19 08:15:30 -07:00
smartBookmark . itemId = itemId ;
smartBookmark . parent = PlacesUtils . bookmarks . getFolderIdForItem ( itemId ) ;
2014-03-10 09:53:19 -07:00
smartBookmark . updatedPosition = PlacesUtils . bookmarks . getItemIndex ( itemId ) ;
2010-11-23 05:04:07 -08:00
}
else {
2008-04-08 11:42:53 -07:00
// We don't remove old Smart Bookmarks because user could still
// find them useful, or could have personalized them.
// Instead we remove the Smart Bookmark annotation.
2010-11-23 05:04:07 -08:00
PlacesUtils . annotations . removeItemAnnotation ( itemId , SMART _BOOKMARKS _ANNO ) ;
2008-04-08 11:42:53 -07:00
}
2010-11-23 05:04:07 -08:00
} ) ;
for ( let queryId in smartBookmarks ) {
let smartBookmark = smartBookmarks [ queryId ] ;
2008-04-08 11:42:53 -07:00
2009-03-26 05:58:32 -07:00
// We update or create only changed or new smart bookmarks.
// Also we respect user choices, so we won't try to create a smart
// bookmark if it has been removed.
if ( smartBookmarksCurrentVersion > 0 &&
smartBookmark . newInVersion <= smartBookmarksCurrentVersion &&
2014-03-10 09:53:19 -07:00
! smartBookmark . itemId || ! smartBookmark . uri )
2009-03-26 05:58:32 -07:00
continue ;
2010-11-23 05:04:07 -08:00
// Remove old version of the smart bookmark if it exists, since it
// will be replaced in place.
if ( smartBookmark . itemId ) {
PlacesUtils . bookmarks . removeItem ( smartBookmark . itemId ) ;
}
// Create the new smart bookmark and store its updated itemId.
smartBookmark . itemId =
PlacesUtils . bookmarks . insertBookmark ( smartBookmark . parent ,
smartBookmark . uri ,
2014-03-10 09:53:19 -07:00
smartBookmark . updatedPosition || smartBookmark . position ,
2010-11-23 05:04:07 -08:00
smartBookmark . title ) ;
PlacesUtils . annotations . setItemAnnotation ( smartBookmark . itemId ,
SMART _BOOKMARKS _ANNO ,
queryId , 0 ,
PlacesUtils . annotations . EXPIRE _NEVER ) ;
2008-04-08 11:42:53 -07:00
}
2010-09-02 19:20:12 -07:00
2008-04-08 11:42:53 -07:00
// If we are creating all Smart Bookmarks from ground up, add a
// separator below them in the bookmarks menu.
2009-03-26 05:58:32 -07:00
if ( smartBookmarksCurrentVersion == 0 &&
2010-04-28 00:17:09 -07:00
smartBookmarkItemIds . length == 0 ) {
2010-11-23 05:04:07 -08:00
let id = PlacesUtils . bookmarks . getIdForItemAt ( PlacesUtils . bookmarksMenuFolderId ,
menuIndex ) ;
2010-04-28 00:17:09 -07:00
// Don't add a separator if the menu was empty or there is one already.
2010-11-23 05:04:07 -08:00
if ( id != - 1 &&
PlacesUtils . bookmarks . getItemType ( id ) != PlacesUtils . bookmarks . TYPE _SEPARATOR ) {
PlacesUtils . bookmarks . insertSeparator ( PlacesUtils . bookmarksMenuFolderId ,
menuIndex ) ;
}
2010-04-28 00:17:09 -07:00
}
2007-11-26 16:08:50 -08:00
}
} ;
try {
2010-11-23 05:04:07 -08:00
PlacesUtils . bookmarks . runInBatchMode ( batch , null ) ;
2007-11-26 16:08:50 -08:00
}
catch ( ex ) {
Components . utils . reportError ( ex ) ;
}
finally {
2010-04-12 07:33:10 -07:00
Services . prefs . setIntPref ( SMART _BOOKMARKS _PREF , SMART _BOOKMARKS _VERSION ) ;
Services . prefs . savePrefFile ( null ) ;
2007-11-26 16:08:50 -08:00
}
} ,
2008-10-30 23:42:02 -07:00
// this returns the most recent non-popup browser window
2009-12-17 13:28:08 -08:00
getMostRecentBrowserWindow : function BG _getMostRecentBrowserWindow ( ) {
2012-12-05 17:59:26 -08:00
return RecentWindow . getMostRecentBrowserWindow ( ) ;
2008-10-30 23:42:02 -07:00
} ,
2012-06-29 15:30:45 -07:00
# ifdef MOZ _SERVICES _SYNC
/ * *
* Called as an observer when Sync ' s "display URI" notification is fired .
*
* We open the received URI in a background tab .
*
* Eventually , this will likely be replaced by a more robust tab syncing
* feature . This functionality is considered somewhat evil by UX because it
* opens a new tab automatically without any prompting . However , it is a
* lesser evil than sending a tab to a specific device ( from e . g . Fennec )
* and having nothing happen on the receiving end .
* /
_onDisplaySyncURI : function _onDisplaySyncURI ( data ) {
try {
2013-03-01 06:14:49 -08:00
let tabbrowser = RecentWindow . getMostRecentBrowserWindow ( { private : false } ) . gBrowser ;
2012-06-29 15:30:45 -07:00
// The payload is wrapped weirdly because of how Sync does notifications.
tabbrowser . addTab ( data . wrappedJSObject . object . uri ) ;
} catch ( ex ) {
Cu . reportError ( "Error displaying tab received by Sync: " + ex ) ;
}
} ,
# endif
2008-10-30 23:42:02 -07:00
2007-09-05 20:02:22 -07:00
// for XPCOM
classID : Components . ID ( "{eab9012e-5f74-4cbc-b2b5-a590235513cc}" ) ,
2007-03-22 10:30:00 -07:00
2007-09-27 12:37:47 -07:00
QueryInterface : XPCOMUtils . generateQI ( [ Ci . nsIObserver ,
Ci . nsISupportsWeakReference ,
Ci . nsIBrowserGlue ] ) ,
2007-03-22 10:30:00 -07:00
2007-09-05 20:02:22 -07:00
// redefine the default factory for XPCOMUtils
_xpcom _factory : BrowserGlueServiceFactory ,
2007-03-22 10:30:00 -07:00
}
2010-09-09 21:59:51 -07:00
function ContentPermissionPrompt ( ) { }
2008-10-23 11:01:32 -07:00
2010-09-09 21:59:51 -07:00
ContentPermissionPrompt . prototype = {
classID : Components . ID ( "{d8903bf6-68d5-4e97-bcd1-e4d3012f721a}" ) ,
2008-10-23 11:01:32 -07:00
2010-09-09 21:59:51 -07:00
QueryInterface : XPCOMUtils . generateQI ( [ Ci . nsIContentPermissionPrompt ] ) ,
2010-04-09 10:45:25 -07:00
2013-08-23 16:04:19 -07:00
_getBrowserForRequest : function ( aRequest ) {
2013-09-04 08:40:16 -07:00
// "element" is only defined in e10s mode.
let browser = aRequest . element ;
2013-08-23 16:04:19 -07:00
if ( ! browser ) {
2013-09-04 08:40:16 -07:00
// Find the requesting browser.
browser = aRequest . window . QueryInterface ( Ci . nsIInterfaceRequestor )
2013-08-23 16:04:19 -07:00
. getInterface ( Ci . nsIWebNavigation )
. QueryInterface ( Ci . nsIDocShell )
. chromeEventHandler ;
}
return browser ;
2012-07-12 08:45:38 -07:00
} ,
2010-09-09 21:59:51 -07:00
2012-07-12 08:45:38 -07:00
/ * *
* Show a permission prompt .
*
* @ param aRequest The permission request .
* @ param aMessage The message to display on the prompt .
* @ param aPermission The type of permission to prompt .
* @ param aActions An array of actions of the form :
* [ main action , secondary actions , ... ]
* Actions are of the form { stringId , action , expireType , callback }
* Permission is granted if action is null or ALLOW _ACTION .
* @ param aNotificationId The id of the PopupNotification .
* @ param aAnchorId The id for the PopupNotification anchor .
2013-04-01 10:38:36 -07:00
* @ param aOptions Options for the PopupNotification
2012-07-12 08:45:38 -07:00
* /
_showPrompt : function CPP _showPrompt ( aRequest , aMessage , aPermission , aActions ,
2013-04-01 10:38:36 -07:00
aNotificationId , aAnchorId , aOptions ) {
2013-03-24 03:32:44 -07:00
function onFullScreen ( ) {
popup . remove ( ) ;
}
2012-07-12 08:45:38 -07:00
var browserBundle = Services . strings . createBundle ( "chrome://browser/locale/browser.properties" ) ;
2010-09-09 21:59:51 -07:00
2013-08-23 16:04:19 -07:00
var browser = this . _getBrowserForRequest ( aRequest ) ;
2013-07-22 17:24:11 -07:00
var chromeWin = browser . ownerDocument . defaultView ;
2012-07-12 08:45:38 -07:00
var requestPrincipal = aRequest . principal ;
// Transform the prompt actions into PopupNotification actions.
var popupNotificationActions = [ ] ;
for ( var i = 0 ; i < aActions . length ; i ++ ) {
let promptAction = aActions [ i ] ;
// Don't offer action in PB mode if the action remembers permission for more than a session.
if ( PrivateBrowsingUtils . isWindowPrivate ( chromeWin ) &&
promptAction . expireType != Ci . nsIPermissionManager . EXPIRE _SESSION &&
promptAction . action ) {
continue ;
}
2010-04-09 10:45:25 -07:00
2012-07-12 08:45:38 -07:00
var action = {
label : browserBundle . GetStringFromName ( promptAction . stringId ) ,
accessKey : browserBundle . GetStringFromName ( promptAction . stringId + ".accesskey" ) ,
callback : function ( ) {
if ( promptAction . callback ) {
promptAction . callback ( ) ;
}
2010-04-09 10:45:25 -07:00
2012-07-12 08:45:38 -07:00
// Remember permissions.
if ( promptAction . action ) {
Services . perms . addFromPrincipal ( requestPrincipal , aPermission ,
promptAction . action , promptAction . expireType ) ;
}
2009-04-15 14:49:11 -07:00
2012-07-12 08:45:38 -07:00
// Grant permission if action is null or ALLOW_ACTION.
if ( ! promptAction . action || promptAction . action == Ci . nsIPermissionManager . ALLOW _ACTION ) {
aRequest . allow ( ) ;
} else {
aRequest . cancel ( ) ;
}
} ,
} ;
2010-09-02 19:20:12 -07:00
2012-07-12 08:45:38 -07:00
popupNotificationActions . push ( action ) ;
2009-04-15 14:49:11 -07:00
}
2013-03-24 03:32:44 -07:00
var mainAction = popupNotificationActions . length ?
popupNotificationActions [ 0 ] : null ;
2012-07-12 08:45:38 -07:00
var secondaryActions = popupNotificationActions . splice ( 1 ) ;
2013-03-24 03:32:44 -07:00
2014-02-09 12:34:40 -08:00
// Only allow exactly one permission rquest here.
let types = aRequest . types . QueryInterface ( Ci . nsIArray ) ;
if ( types . length != 1 ) {
aRequest . cancel ( ) ;
return ;
}
let perm = types . queryElementAt ( 0 , Ci . nsIContentPermissionType ) ;
if ( perm . type == "pointerLock" ) {
2013-03-24 03:32:44 -07:00
// If there's no mainAction, this is the autoAllow warning prompt.
let autoAllow = ! mainAction ;
2014-02-18 19:05:34 -08:00
if ( ! aOptions )
aOptions = { } ;
aOptions . removeOnDismissal = autoAllow ;
aOptions . eventCallback = type => {
if ( type == "removed" ) {
browser . removeEventListener ( "mozfullscreenchange" , onFullScreen , true ) ;
if ( autoAllow ) {
aRequest . allow ( ) ;
2013-04-01 10:38:36 -07:00
}
2014-02-18 19:05:34 -08:00
}
}
2013-03-24 03:32:44 -07:00
}
var popup = chromeWin . PopupNotifications . show ( browser , aNotificationId , aMessage , aAnchorId ,
2013-04-01 10:38:36 -07:00
mainAction , secondaryActions , aOptions ) ;
2014-02-09 12:34:40 -08:00
if ( perm . type == "pointerLock" ) {
2013-03-24 03:32:44 -07:00
// pointerLock is automatically allowed in fullscreen mode (and revoked
// upon exit), so if the page enters fullscreen mode after requesting
// pointerLock (but before the user has granted permission), we should
// remove the now-impotent notification.
2013-03-24 05:45:38 -07:00
browser . addEventListener ( "mozfullscreenchange" , onFullScreen , true ) ;
2013-03-24 03:32:44 -07:00
}
2012-07-12 08:45:38 -07:00
} ,
2008-10-23 11:01:32 -07:00
2012-07-12 08:45:38 -07:00
_promptGeo : function ( aRequest ) {
var secHistogram = Services . telemetry . getHistogramById ( "SECURITY_UI" ) ;
2010-04-09 10:45:25 -07:00
var browserBundle = Services . strings . createBundle ( "chrome://browser/locale/browser.properties" ) ;
2012-07-12 08:45:38 -07:00
var requestingURI = aRequest . principal . URI ;
2009-04-15 14:49:11 -07:00
2012-07-12 08:45:38 -07:00
var message ;
// Share location action.
var actions = [ {
stringId : "geolocation.shareLocation" ,
action : null ,
expireType : null ,
2012-06-03 22:25:55 -07:00
callback : function ( ) {
2012-09-08 09:20:51 -07:00
secHistogram . add ( Ci . nsISecurityUITelemetry . WARNING _GEOLOCATION _REQUEST _SHARE _LOCATION ) ;
2010-04-09 10:45:25 -07:00
} ,
2012-07-12 08:45:38 -07:00
} ] ;
2010-04-09 10:45:25 -07:00
if ( requestingURI . schemeIs ( "file" ) ) {
2011-11-28 13:02:45 -08:00
message = browserBundle . formatStringFromName ( "geolocation.shareWithFile" ,
2010-09-09 21:59:51 -07:00
[ requestingURI . path ] , 1 ) ;
2010-04-09 10:45:25 -07:00
} else {
2011-11-28 13:02:45 -08:00
message = browserBundle . formatStringFromName ( "geolocation.shareWithSite" ,
2010-04-09 10:45:25 -07:00
[ requestingURI . host ] , 1 ) ;
2012-07-12 08:45:38 -07:00
// Always share location action.
actions . push ( {
stringId : "geolocation.alwaysShareLocation" ,
action : Ci . nsIPermissionManager . ALLOW _ACTION ,
expireType : null ,
callback : function ( ) {
secHistogram . add ( Ci . nsISecurityUITelemetry . WARNING _GEOLOCATION _REQUEST _ALWAYS _SHARE ) ;
} ,
} ) ;
2010-04-09 10:45:25 -07:00
2012-07-12 08:45:38 -07:00
// Never share location action.
actions . push ( {
stringId : "geolocation.neverShareLocation" ,
action : Ci . nsIPermissionManager . DENY _ACTION ,
expireType : null ,
callback : function ( ) {
secHistogram . add ( Ci . nsISecurityUITelemetry . WARNING _GEOLOCATION _REQUEST _NEVER _SHARE ) ;
} ,
} ) ;
2010-04-09 10:45:25 -07:00
}
2009-04-15 14:49:11 -07:00
2014-02-18 19:05:34 -08:00
var options = {
learnMoreURL : Services . urlFormatter . formatURLPref ( "browser.geolocation.warning.infoURL" ) ,
} ;
2012-12-24 16:08:13 -08:00
2012-09-08 09:20:51 -07:00
secHistogram . add ( Ci . nsISecurityUITelemetry . WARNING _GEOLOCATION _REQUEST ) ;
2012-07-12 08:45:38 -07:00
2013-04-01 10:38:36 -07:00
this . _showPrompt ( aRequest , message , "geo" , actions , "geolocation" ,
2014-02-18 19:05:34 -08:00
"geo-notification-icon" , options ) ;
2012-07-12 08:45:38 -07:00
} ,
_promptWebNotifications : function ( aRequest ) {
var browserBundle = Services . strings . createBundle ( "chrome://browser/locale/browser.properties" ) ;
var requestingURI = aRequest . principal . URI ;
var message = browserBundle . formatStringFromName ( "webNotifications.showFromSite" ,
[ requestingURI . host ] , 1 ) ;
var actions = [
{
stringId : "webNotifications.showForSession" ,
action : Ci . nsIPermissionManager . ALLOW _ACTION ,
expireType : Ci . nsIPermissionManager . EXPIRE _SESSION ,
callback : function ( ) { } ,
} ,
{
stringId : "webNotifications.alwaysShow" ,
action : Ci . nsIPermissionManager . ALLOW _ACTION ,
expireType : null ,
callback : function ( ) { } ,
} ,
{
stringId : "webNotifications.neverShow" ,
action : Ci . nsIPermissionManager . DENY _ACTION ,
expireType : null ,
callback : function ( ) { } ,
} ,
] ;
this . _showPrompt ( aRequest , message , "desktop-notification" , actions ,
"web-notifications" ,
2013-04-01 10:38:36 -07:00
"web-notifications-notification-icon" , null ) ;
2012-07-12 08:45:38 -07:00
} ,
2013-03-24 03:32:44 -07:00
_promptPointerLock : function CPP _promtPointerLock ( aRequest , autoAllow ) {
let browserBundle = Services . strings . createBundle ( "chrome://browser/locale/browser.properties" ) ;
let requestingURI = aRequest . principal . URI ;
let originString = requestingURI . schemeIs ( "file" ) ? requestingURI . path : requestingURI . host ;
let message = browserBundle . formatStringFromName ( autoAllow ?
2013-04-03 01:22:05 -07:00
"pointerLock.autoLock.title2" : "pointerLock.title2" ,
2013-03-24 03:32:44 -07:00
[ originString ] , 1 ) ;
// If this is an autoAllow info prompt, offer no actions.
// _showPrompt() will allow the request when it's dismissed.
let actions = [ ] ;
if ( ! autoAllow ) {
actions = [
{
2013-04-03 01:22:05 -07:00
stringId : "pointerLock.allow2" ,
2013-03-24 03:32:44 -07:00
action : null ,
expireType : null ,
callback : function ( ) { } ,
} ,
{
stringId : "pointerLock.alwaysAllow" ,
action : Ci . nsIPermissionManager . ALLOW _ACTION ,
expireType : null ,
callback : function ( ) { } ,
} ,
{
stringId : "pointerLock.neverAllow" ,
action : Ci . nsIPermissionManager . DENY _ACTION ,
expireType : null ,
callback : function ( ) { } ,
} ,
] ;
}
2013-04-01 10:38:36 -07:00
this . _showPrompt ( aRequest , message , "pointerLock" , actions , "pointerLock" ,
"pointerLock-notification-icon" , null ) ;
2013-03-24 03:32:44 -07:00
} ,
2012-07-12 08:45:38 -07:00
prompt : function CPP _prompt ( request ) {
2013-03-24 03:32:44 -07:00
2014-02-09 12:34:40 -08:00
// Only allow exactly one permission rquest here.
let types = request . types . QueryInterface ( Ci . nsIArray ) ;
if ( types . length != 1 ) {
request . cancel ( ) ;
return ;
}
let perm = types . queryElementAt ( 0 , Ci . nsIContentPermissionType ) ;
2012-07-12 08:45:38 -07:00
const kFeatureKeys = { "geolocation" : "geo" ,
2013-03-24 03:32:44 -07:00
"desktop-notification" : "desktop-notification" ,
"pointerLock" : "pointerLock" ,
} ;
2012-07-12 08:45:38 -07:00
// Make sure that we support the request.
2014-02-09 12:34:40 -08:00
if ( ! ( perm . type in kFeatureKeys ) ) {
2012-07-12 08:45:38 -07:00
return ;
}
var requestingPrincipal = request . principal ;
var requestingURI = requestingPrincipal . URI ;
// Ignore requests from non-nsIStandardURLs
if ( ! ( requestingURI instanceof Ci . nsIStandardURL ) )
return ;
2013-03-24 03:32:44 -07:00
var autoAllow = false ;
2014-02-09 12:34:40 -08:00
var permissionKey = kFeatureKeys [ perm . type ] ;
2012-07-12 08:45:38 -07:00
var result = Services . perms . testExactPermissionFromPrincipal ( requestingPrincipal , permissionKey ) ;
if ( result == Ci . nsIPermissionManager . DENY _ACTION ) {
request . cancel ( ) ;
return ;
}
2013-03-24 03:32:44 -07:00
if ( result == Ci . nsIPermissionManager . ALLOW _ACTION ) {
autoAllow = true ;
// For pointerLock, we still want to show a warning prompt.
2014-02-09 12:34:40 -08:00
if ( perm . type != "pointerLock" ) {
2013-03-24 03:32:44 -07:00
request . allow ( ) ;
return ;
}
}
2013-11-11 15:36:54 -08:00
var browser = this . _getBrowserForRequest ( request ) ;
var chromeWin = browser . ownerDocument . defaultView ;
if ( ! chromeWin . PopupNotifications )
// Ignore requests from browsers hosted in windows that don't support
// PopupNotifications.
return ;
2012-07-12 08:45:38 -07:00
// Show the prompt.
2014-02-09 12:34:40 -08:00
switch ( perm . type ) {
2012-07-12 08:45:38 -07:00
case "geolocation" :
this . _promptGeo ( request ) ;
break ;
case "desktop-notification" :
this . _promptWebNotifications ( request ) ;
break ;
2013-03-24 03:32:44 -07:00
case "pointerLock" :
this . _promptPointerLock ( request , autoAllow ) ;
break ;
2012-07-12 08:45:38 -07:00
}
2013-03-24 03:32:44 -07:00
} ,
2008-10-23 11:01:32 -07:00
} ;
2014-08-12 00:59:11 -07:00
let DefaultBrowserCheck = {
get OPTIONPOPUP ( ) { return "defaultBrowserNotificationPopup" } ,
closePrompt : function ( aNode ) {
if ( this . _notification ) {
this . _notification . close ( ) ;
}
} ,
setAsDefault : function ( ) {
let claimAllTypes = true ;
# ifdef XP _WIN
try {
// In Windows 8, the UI for selecting default protocol is much
// nicer than the UI for setting file type associations. So we
// only show the protocol association screen on Windows 8.
// Windows 8 is version 6.2.
let version = Services . sysinfo . getProperty ( "version" ) ;
claimAllTypes = ( parseFloat ( version ) < 6.2 ) ;
} catch ( ex ) { }
# endif
2014-09-24 08:18:45 -07:00
try {
ShellService . setDefaultBrowser ( claimAllTypes , false ) ;
} catch ( ex ) {
Cu . reportError ( ex ) ;
}
2014-08-12 00:59:11 -07:00
} ,
2014-11-02 16:08:58 -08:00
_createPopup : function ( win , notNowStrings , neverStrings ) {
2014-08-12 00:59:11 -07:00
let doc = win . document ;
let popup = doc . createElement ( "menupopup" ) ;
popup . id = this . OPTIONPOPUP ;
let notNowItem = doc . createElement ( "menuitem" ) ;
notNowItem . id = "defaultBrowserNotNow" ;
2014-11-02 16:08:58 -08:00
notNowItem . setAttribute ( "label" , notNowStrings . label ) ;
notNowItem . setAttribute ( "accesskey" , notNowStrings . accesskey ) ;
2014-08-12 00:59:11 -07:00
popup . appendChild ( notNowItem ) ;
let neverItem = doc . createElement ( "menuitem" ) ;
neverItem . id = "defaultBrowserNever" ;
2014-11-02 16:08:58 -08:00
neverItem . setAttribute ( "label" , neverStrings . label ) ;
neverItem . setAttribute ( "accesskey" , neverStrings . accesskey ) ;
2014-08-12 00:59:11 -07:00
popup . appendChild ( neverItem ) ;
popup . addEventListener ( "command" , this ) ;
let popupset = doc . getElementById ( "mainPopupSet" ) ;
popupset . appendChild ( popup ) ;
} ,
handleEvent : function ( event ) {
if ( event . type == "command" ) {
if ( event . target . id == "defaultBrowserNever" ) {
ShellService . shouldCheckDefaultBrowser = false ;
}
this . closePrompt ( ) ;
}
} ,
prompt : function ( win ) {
let brandBundle = win . document . getElementById ( "bundle_brand" ) ;
let shellBundle = win . document . getElementById ( "bundle_shell" ) ;
let brandShortName = brandBundle . getString ( "brandShortName" ) ;
let promptMessage = shellBundle . getFormattedString ( "setDefaultBrowserMessage2" ,
[ brandShortName ] ) ;
2014-11-02 16:08:58 -08:00
let yesButton = shellBundle . getFormattedString ( "setDefaultBrowserConfirm.label" ,
[ brandShortName ] ) ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
let notNowButton = shellBundle . getString ( "setDefaultBrowserNotNow.label" ) ;
let notNowButtonKey = shellBundle . getString ( "setDefaultBrowserNotNow.accesskey" ) ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
let neverLabel = shellBundle . getString ( "setDefaultBrowserNever.label" ) ;
let neverKey = shellBundle . getString ( "setDefaultBrowserNever.accesskey" ) ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
let useNotificationBar = Services . prefs . getBoolPref ( "browser.defaultbrowser.notificationbar" ) ;
if ( useNotificationBar ) {
let optionsMessage = shellBundle . getString ( "setDefaultBrowserOptions.label" ) ;
let optionsKey = shellBundle . getString ( "setDefaultBrowserOptions.accesskey" ) ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
let yesButtonKey = shellBundle . getString ( "setDefaultBrowserConfirm.accesskey" ) ;
let notificationBox = win . document . getElementById ( "high-priority-global-notificationbox" ) ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
this . _createPopup ( win , {
label : notNowButton ,
accesskey : notNowButtonKey
} , {
label : neverLabel ,
accesskey : neverKey
} ) ;
let buttons = [
{
label : yesButton ,
accessKey : yesButtonKey ,
callback : ( ) => {
this . setAsDefault ( ) ;
this . closePrompt ( ) ;
}
} ,
{
label : optionsMessage ,
accessKey : optionsKey ,
popup : this . OPTIONPOPUP
}
] ;
2014-08-12 00:59:11 -07:00
2014-11-02 16:08:58 -08:00
let iconPixels = win . devicePixelRatio > 1 ? "32" : "16" ;
let iconURL = "chrome://branding/content/icon" + iconPixels + ".png" ;
const priority = notificationBox . PRIORITY _WARNING _HIGH ;
let callback = this . _onNotificationEvent . bind ( this ) ;
this . _notification = notificationBox . appendNotification ( promptMessage , "default-browser" ,
iconURL , priority , buttons ,
callback ) ;
} else {
// Modal prompt
let promptTitle = shellBundle . getString ( "setDefaultBrowserTitle" ) ;
let ps = Services . prompt ;
let dontAsk = { value : false } ;
let buttonFlags = ( ps . BUTTON _TITLE _IS _STRING * ps . BUTTON _POS _0 ) +
( ps . BUTTON _TITLE _IS _STRING * ps . BUTTON _POS _1 ) +
ps . BUTTON _POS _0 _DEFAULT ;
let rv = ps . confirmEx ( win , promptTitle , promptMessage , buttonFlags ,
yesButton , notNowButton , null , neverLabel , dontAsk ) ;
if ( rv == 0 ) {
this . setAsDefault ( ) ;
} else if ( dontAsk . value ) {
ShellService . shouldCheckDefaultBrowser = false ;
}
}
2014-08-12 00:59:11 -07:00
} ,
_onNotificationEvent : function ( eventType ) {
if ( eventType == "removed" ) {
let doc = this . _notification . ownerDocument ;
let popup = doc . getElementById ( this . OPTIONPOPUP ) ;
popup . removeEventListener ( "command" , this ) ;
popup . remove ( ) ;
delete this . _notification ;
}
} ,
} ;
2014-09-10 15:06:19 -07:00
# ifdef E10S _TESTING _ONLY
let E10SUINotification = {
// Increase this number each time we want to roll out an
// e10s testing period to Nightly users.
2014-11-28 07:59:29 -08:00
CURRENT _NOTICE _COUNT : 3 ,
2014-10-08 18:56:26 -07:00
CURRENT _PROMPT _PREF : "browser.displayedE10SPrompt.1" ,
PREVIOUS _PROMPT _PREF : "browser.displayedE10SPrompt" ,
2014-09-10 15:06:19 -07:00
checkStatus : function ( ) {
2014-09-13 11:20:16 -07:00
let skipE10sChecks = false ;
try {
2014-10-08 18:56:26 -07:00
skipE10sChecks = ( UpdateChannel . get ( ) != "nightly" ) ||
Services . prefs . getBoolPref ( "browser.tabs.remote.autostart.disabled-because-using-a11y" ) ;
2014-09-13 11:20:16 -07:00
} catch ( e ) { }
if ( skipE10sChecks ) {
return ;
}
2014-09-10 15:06:19 -07:00
if ( Services . appinfo . browserTabsRemoteAutostart ) {
let notice = 0 ;
try {
notice = Services . prefs . getIntPref ( "browser.displayedE10SNotice" ) ;
} catch ( e ) { }
let activationNoticeShown = notice >= this . CURRENT _NOTICE _COUNT ;
if ( ! activationNoticeShown ) {
this . _showE10sActivatedNotice ( ) ;
}
2014-09-13 11:20:16 -07:00
// e10s doesn't work with accessibility, so we prompt to disable
// e10s if a11y is enabled, now or in the future.
Services . obs . addObserver ( this , "a11y-init-or-shutdown" , true ) ;
if ( Services . appinfo . accessibilityEnabled ) {
this . _showE10sAccessibilityWarning ( ) ;
}
2014-09-10 04:53:10 -07:00
} else {
2014-09-22 11:55:40 -07:00
let displayFeedbackRequest = false ;
try {
displayFeedbackRequest = Services . prefs . getBoolPref ( "browser.requestE10sFeedback" ) ;
} catch ( e ) { }
if ( displayFeedbackRequest ) {
let win = RecentWindow . getMostRecentBrowserWindow ( ) ;
if ( ! win ) {
return ;
}
Services . prefs . clearUserPref ( "browser.requestE10sFeedback" ) ;
let url = Services . urlFormatter . formatURLPref ( "app.feedback.baseURL" ) ;
url += "?utm_source=tab&utm_campaign=e10sfeedback" ;
win . openUILinkIn ( url , "tab" ) ;
return ;
}
2014-09-10 04:53:10 -07:00
let e10sPromptShownCount = 0 ;
try {
2014-10-08 18:56:26 -07:00
e10sPromptShownCount = Services . prefs . getIntPref ( this . CURRENT _PROMPT _PREF ) ;
2014-09-10 04:53:10 -07:00
} catch ( e ) { }
2014-10-08 18:56:26 -07:00
let isHardwareAccelerated = true ;
2014-11-14 08:51:19 -08:00
// Linux and Windows are currently ok, mac not so much.
# ifdef XP _MACOSX
2014-10-08 18:56:26 -07:00
try {
let win = RecentWindow . getMostRecentBrowserWindow ( ) ;
let winutils = win . QueryInterface ( Ci . nsIInterfaceRequestor ) . getInterface ( Ci . nsIDOMWindowUtils ) ;
isHardwareAccelerated = winutils . layerManagerType != "Basic" ;
} catch ( e ) { }
2014-11-14 08:51:19 -08:00
# endif
2014-10-08 18:56:26 -07:00
2014-09-13 11:20:16 -07:00
if ( ! Services . appinfo . inSafeMode &&
! Services . appinfo . accessibilityEnabled &&
2014-09-18 10:26:33 -07:00
! Services . appinfo . keyboardMayHaveIME &&
2014-10-08 18:56:26 -07:00
isHardwareAccelerated &&
2014-09-13 11:20:16 -07:00
e10sPromptShownCount < 5 ) {
2014-09-10 04:53:10 -07:00
Services . tm . mainThread . dispatch ( ( ) => {
try {
this . _showE10SPrompt ( ) ;
2014-10-08 18:56:26 -07:00
Services . prefs . setIntPref ( this . CURRENT _PROMPT _PREF , e10sPromptShownCount + 1 ) ;
Services . prefs . clearUserPref ( this . PREVIOUS _PROMPT _PREF ) ;
2014-09-10 04:53:10 -07:00
} catch ( ex ) {
Cu . reportError ( "Failed to show e10s prompt: " + ex ) ;
}
} , Ci . nsIThread . DISPATCH _NORMAL ) ;
}
2014-09-10 15:06:19 -07:00
}
} ,
2014-09-13 11:20:16 -07:00
QueryInterface : XPCOMUtils . generateQI ( [ Ci . nsIObserver , Ci . nsISupportsWeakReference ] ) ,
observe : function ( subject , topic , data ) {
if ( topic == "a11y-init-or-shutdown" && data == "1" ) {
this . _showE10sAccessibilityWarning ( ) ;
}
} ,
2014-09-10 15:06:19 -07:00
_showE10sActivatedNotice : function ( ) {
let win = RecentWindow . getMostRecentBrowserWindow ( ) ;
if ( ! win )
return ;
Services . prefs . setIntPref ( "browser.displayedE10SNotice" , this . CURRENT _NOTICE _COUNT ) ;
let nb = win . document . getElementById ( "high-priority-global-notificationbox" ) ;
2014-11-06 11:23:43 -08:00
let message = "You're now helping to test Process Separation (e10s)! Please report problems you find." ;
2014-09-10 15:06:19 -07:00
let buttons = [
{
label : "Learn More" ,
accessKey : "L" ,
callback : function ( ) {
win . openUILinkIn ( "https://wiki.mozilla.org/Electrolysis" , "tab" ) ;
}
}
] ;
nb . appendNotification ( message , "e10s-activated-noticed" ,
null , nb . PRIORITY _WARNING _MEDIUM , buttons ) ;
2014-09-10 04:53:10 -07:00
} ,
_showE10SPrompt : function BG _ _showE10SPrompt ( ) {
let win = RecentWindow . getMostRecentBrowserWindow ( ) ;
if ( ! win )
return ;
let browser = win . gBrowser . selectedBrowser ;
2014-10-08 18:56:26 -07:00
let promptMessage = "Would you like to help us test multiprocess Nightly (e10s)? You can also enable e10s in Nightly preferences. Notable fixes:" ;
2014-09-10 04:53:10 -07:00
let mainAction = {
label : "Enable and Restart" ,
accessKey : "E" ,
callback : function ( ) {
Services . prefs . setBoolPref ( "browser.tabs.remote.autostart" , true ) ;
Services . prefs . setBoolPref ( "browser.enabledE10SFromPrompt" , true ) ;
// Restart the app
let cancelQuit = Cc [ "@mozilla.org/supports-PRBool;1" ] . createInstance ( Ci . nsISupportsPRBool ) ;
Services . obs . notifyObservers ( cancelQuit , "quit-application-requested" , "restart" ) ;
if ( cancelQuit . data )
return ; // somebody canceled our quit request
Services . startup . quit ( Services . startup . eAttemptQuit | Services . startup . eRestart ) ;
}
} ;
let secondaryActions = [
{
label : "No thanks" ,
accessKey : "N" ,
callback : function ( ) {
2014-10-08 18:56:26 -07:00
Services . prefs . setIntPref ( E10SUINotification . CURRENT _PROMPT _PREF , 5 ) ;
2014-09-10 04:53:10 -07:00
}
}
] ;
let options = {
popupIconURL : "chrome://browser/skin/e10s-64@2x.png" ,
learnMoreURL : "https://wiki.mozilla.org/Electrolysis" ,
persistWhileVisible : true
} ;
2014-10-08 18:56:26 -07:00
win . PopupNotifications . show ( browser , "enable-e10s" , promptMessage , null , mainAction , secondaryActions , options ) ;
let highlights = [
"Less crashing!" ,
"Improved add-on compatibility and DevTools" ,
"PDF.js, Web Console, Spellchecking, WebRTC now work"
] ;
let doorhangerExtraContent = win . document . getElementById ( "enable-e10s-notification" )
. querySelector ( "popupnotificationcontent" ) ;
for ( let highlight of highlights ) {
let highlightLabel = win . document . createElement ( "label" ) ;
highlightLabel . setAttribute ( "value" , highlight ) ;
doorhangerExtraContent . appendChild ( highlightLabel ) ;
}
2014-09-10 04:53:10 -07:00
} ,
2014-09-13 11:20:16 -07:00
_warnedAboutAccessibility : false ,
_showE10sAccessibilityWarning : function ( ) {
Services . prefs . setBoolPref ( "browser.tabs.remote.autostart.disabled-because-using-a11y" , true ) ;
if ( this . _warnedAboutAccessibility ) {
return ;
}
this . _warnedAboutAccessibility = true ;
let win = RecentWindow . getMostRecentBrowserWindow ( ) ;
if ( ! win ) {
// Just restart immediately.
Services . startup . quit ( Services . startup . eAttemptQuit | Services . startup . eRestart ) ;
return ;
}
let browser = win . gBrowser . selectedBrowser ;
let promptMessage = "Multiprocess Nightly (e10s) does not yet support accessibility features. Multiprocessing will be disabled if you restart Firefox. Would you like to restart?" ;
let mainAction = {
label : "Disable and Restart" ,
accessKey : "R" ,
callback : function ( ) {
// Restart the app
let cancelQuit = Cc [ "@mozilla.org/supports-PRBool;1" ] . createInstance ( Ci . nsISupportsPRBool ) ;
Services . obs . notifyObservers ( cancelQuit , "quit-application-requested" , "restart" ) ;
if ( cancelQuit . data )
return ; // somebody canceled our quit request
Services . startup . quit ( Services . startup . eAttemptQuit | Services . startup . eRestart ) ;
}
} ;
let secondaryActions = [
{
label : "Don't Disable" ,
accessKey : "D" ,
callback : function ( ) {
Services . prefs . setBoolPref ( "browser.tabs.remote.autostart.disabled-because-using-a11y" , false ) ;
}
}
] ;
let options = {
popupIconURL : "chrome://browser/skin/e10s-64@2x.png" ,
learnMoreURL : "https://wiki.mozilla.org/Electrolysis" ,
persistWhileVisible : true
} ;
win . PopupNotifications . show ( browser , "a11y_enabled_with_e10s" , promptMessage , null , mainAction , secondaryActions , options ) ;
} ,
2014-09-10 15:06:19 -07:00
} ;
# endif
2010-09-09 21:59:51 -07:00
var components = [ BrowserGlue , ContentPermissionPrompt ] ;
2012-10-31 09:13:28 -07:00
this . NSGetFactory = XPCOMUtils . generateNSGetFactory ( components ) ;
2014-10-24 08:19:00 -07:00
// Listen for UITour messages.
// Do it here instead of the UITour module itself so that the UITour module is lazy loaded
// when the first message is received.
let globalMM = Cc [ "@mozilla.org/globalmessagemanager;1" ] . getService ( Ci . nsIMessageListenerManager ) ;
globalMM . addMessageListener ( "UITour:onPageEvent" , function ( aMessage ) {
UITour . onPageEvent ( aMessage , aMessage . data ) ;
} ) ;