// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef CHROME_BROWSER_BROWSER_LIST_H__ #define CHROME_BROWSER_BROWSER_LIST_H__ #include #include #include "chrome/browser/browser.h" namespace views { class Window; }; class WebContents; // Stores a list of all Browser objects. class BrowserList { public: typedef std::vector list_type; typedef list_type::iterator iterator; typedef list_type::const_iterator const_iterator; typedef list_type::const_reverse_iterator const_reverse_iterator; // It is not allowed to change the global window list (add or remove any // browser windows while handling observer callbacks. class Observer { public: // Called immediately after a browser is added to the list virtual void OnBrowserAdded(const Browser* browser) = 0; // Called immediately before a browser is removed from the list virtual void OnBrowserRemoving(const Browser* browser) = 0; }; // Adds and removes browsers from the global list. The browser object should // be valid BEFORE these calls (for the benefit of observers), so notify and // THEN delete the object. static void AddBrowser(Browser* browser); static void RemoveBrowser(Browser* browser); // Adds and removes non-browser dependent windows. These are windows that are // top level, but whose lifetime is associated wtih the existence of at least // one active Browser. When the last Browser is destroyed, all open dependent // windows are closed. static void AddDependentWindow(views::Window* window); static void RemoveDependentWindow(views::Window* window); static void AddObserver(Observer* observer); static void RemoveObserver(Observer* observer); // Called by Browser objects when their window is activated (focused). This // allows us to determine what the last active Browser was. static void SetLastActive(Browser* browser); // Returns the Browser object whose window was most recently active. If the // most recently open Browser's window was closed, returns the first Browser // in the list. If no Browsers exist, returns NULL. static Browser* GetLastActive(); // Find an existing browser window with the provided type. If the last active // has the right type, it is returned. Otherwise, the next available browser // is returned. Returns NULL if no such browser currently exists. static Browser* FindBrowserWithType(Profile* p, BrowserType::Type t); // Closes all browsers. If use_post is true the windows are closed by way of // posting a WM_CLOSE message, otherwise the windows are closed directly. In // almost all cases you'll want to use true, the one exception is ending // the session. use_post should only be false when invoked from end session. static void CloseAllBrowsers(bool use_post); // Returns true if there is at least one Browser with the specified profile. static bool HasBrowserWithProfile(Profile* profile); // Set whether the last active browser should be modal or not, if // |is_app_modal| is true, the last active browser window will be activated // and brought to the front whenever the user attempts to activate any other // browser window. If |is_app_modal| is false all window activation works as // normal. SetIsShowingAppModalDialog should not be called with |is_app_modal| // set to true if the last active browser is already modal. // // TODO(devint): http://b/issue?id=1123402 Application modal dialogs aren't // selected, just the last active browser. Therefore, to properly use this // function we have to set the modal dialog as a child of a browser, // activate that browser window, call SetIsShowingAppModalDialog(true), and // implement the modal dialog as window modal to its parent. This still isn't // perfect,however, because it just assures that the browser is activated, and // the dialog will be on top of that browser, but inactive. It will activate // if the users attempts to interact with its parent window (the browser). // Ideally we should activate the modal dialog, not just its parent browser. // // There is probably a less clunky way overall to implement application // modality. Currently, if IsShowingAppModalDialog returns true, we handle // messages right before the browser activates, and activate whatever // GetLastActive() returns instead of whatever was trying to be activated. // It'd be better if we could use built in OS modality handling to deal with // this, but Windows only supports system modal or parent window modal. static void SetIsShowingAppModalDialog(bool is_app_modal); // True if the last active browser is application modal, false otherwise. See // SetIsShowingAppModalDialog for more details. static bool IsShowingAppModalDialog(); static const_iterator begin() { return browsers_.begin(); } static const_iterator end() { return browsers_.end(); } static size_t size() { return browsers_.size(); } // Returns iterated access to list of open browsers ordered by when // they were last active. The underlying data structure is a vector // and we push_back on recent access so a reverse iterator gives the // latest accessed browser first. static const_reverse_iterator begin_last_active() { return last_active_browsers_.rbegin(); } static const_reverse_iterator end_last_active() { return last_active_browsers_.rend(); } // Return the number of browsers with the following profile which are // currently open. static size_t GetBrowserCount(Profile* p); // Return the number of browsers with the following profile and type which are // currently open. static size_t GetBrowserCountForType(Profile* p, BrowserType::Type type); // Returns true if at least one off the record session is active. static bool IsOffTheRecordSessionActive(); private: // Closes all registered dependent windows. static void CloseAllDependentWindows(); // Helper method to remove a browser instance from a list of browsers static void RemoveBrowserFrom(Browser* browser, list_type* browser_list); static list_type browsers_; static std::vector observers_; static list_type last_active_browsers_; typedef std::vector DependentWindowList; static DependentWindowList dependent_windows_; // True if last_active_ is app modal, false otherwise. static bool is_app_modal_; }; // Iterates through all web view hosts in all browser windows. Because the // renderers act asynchronously, getting a host through this interface does // not guarantee that the renderer is ready to go. Doing anything to affect // browser windows or tabs while iterating may cause incorrect behavior. // // Example: // for (WebContentsIterator iterator; !iterator.done(); iterator++) { // WebContents* cur = *iterator; // -or- // iterator->operationOnWebContents(); // ... // } class WebContentsIterator { public: WebContentsIterator(); // Returns true if we are past the last Browser. bool done() const { return cur_ == NULL; } // Returns the current WebContents, valid as long as !Done() WebContents* operator->() const { return cur_; } WebContents* operator*() const { return cur_; } // Incrementing operators, valid as long as !Done() WebContents* operator++() { // ++preincrement Advance(); return cur_; } WebContents* operator++(int) { // postincrement++ WebContents* tmp = cur_; Advance(); return tmp; } private: // Loads the next host into Cur. This is designed so that for the initial // call when browser_iterator_ points to the first browser and // web_view_index_ is -1, it will fill the first host. void Advance(); // iterator over all the Browser objects BrowserList::const_iterator browser_iterator_; // tab index into the current Browser of the current web view int web_view_index_; // Current WebContents, or NULL if we're at the end of the list. This can // be extracted given the browser iterator and index, but it's nice to cache // this since the caller may access the current host many times. WebContents* cur_; }; #endif // CHROME_BROWSER_BROWSER_LIST_H__