// Copyright (c) 2011 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. #include "chrome/browser/chromeos/wm_overview_controller.h" #include <algorithm> #include <vector> #include "base/memory/linked_ptr.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/chromeos/wm_ipc.h" #include "chrome/browser/chromeos/wm_overview_favicon.h" #include "chrome/browser/chromeos/wm_overview_snapshot.h" #include "chrome/browser/chromeos/wm_overview_title.h" #include "chrome/browser/tab_contents/thumbnail_generator.h" #include "chrome/browser/tabs/tab_strip_model.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/tab_contents/tab_contents_wrapper.h" #include "chrome/browser/ui/views/frame/browser_view.h" #include "content/browser/renderer_host/render_view_host.h" #include "content/browser/renderer_host/render_widget_host.h" #include "content/browser/renderer_host/render_widget_host_view.h" #include "content/browser/tab_contents/tab_contents.h" #include "content/browser/tab_contents/tab_contents_view.h" #include "content/common/notification_service.h" #include "views/widget/root_view.h" #include "views/widget/widget_gtk.h" #include "views/window/window.h" using std::vector; #if !defined(OS_CHROMEOS) #error This file is only meant to be compiled for ChromeOS #endif namespace chromeos { // Use this timer to delay consecutive snapshots during the updating process. // We will start the timer upon successfully retrieving a new snapshot, or if // for some reason the current snapshot is still pending (while Chrome is // still loading the page.) static const int kDelayTimeMs = 10; // This is the size of the web page when we lay it out for a snapshot. static const int kSnapshotWebPageWidth = 1024; static const int kSnapshotWebPageHeight = 1280; static const double kSnapshotWebPageRatio = static_cast<double>(kSnapshotWebPageWidth) / kSnapshotWebPageHeight; // This is the maximum percentage of the original browser client area // that a snapshot can take up. static const double kSnapshotMaxSizeRatio = 0.77; // This is the height of the title in pixels. static const int kTitleHeight = 32; // The number of additional padding pixels to remove from the title width. static const int kFaviconPadding = 5; class BrowserListener : public TabStripModelObserver { public: BrowserListener(Browser* browser, WmOverviewController* parent); ~BrowserListener(); // Begin TabStripModelObserver methods virtual void TabInsertedAt(TabContentsWrapper* contents, int index, bool foreground); virtual void TabClosingAt(TabStripModel* tab_strip_model, TabContentsWrapper* contents, int index) {} virtual void TabDetachedAt(TabContentsWrapper* contents, int index); virtual void TabMoved(TabContentsWrapper* contents, int from_index, int to_index); virtual void TabChangedAt(TabContentsWrapper* contents, int index, TabStripModelObserver::TabChangeType change_type); virtual void TabStripEmpty(); virtual void TabDeselected(TabContentsWrapper* contents) {} virtual void TabSelectedAt(TabContentsWrapper* old_contents, TabContentsWrapper* new_contents, int index, bool user_gesture); // End TabStripModelObserver methods // Returns the number of tabs in this child. int count() const { return browser_->tabstrip_model()->count(); } // Returns the browser that this child gets its data from. Browser* browser() const { return browser_; } // Removes all the snapshots and re-populates them from the browser. void RecreateSnapshots(); // Mark the given snapshot as dirty, and start the delay timer. void MarkSnapshotAsDirty(int index); // Updates the selected index and tab count on the toplevel window. void UpdateSelectedIndex(int index); // Update the first "dirty" snapshot, which is ordered after (excluding) // the snapshot whose index is given by |start_from|; When |start_from| is // -1, search start at the begining of the list. // Return the index of the snapshot which is actually updated; -1 if there // are no more tab contents (after |start_from|) to configure on this // listener. int ConfigureNextUnconfiguredSnapshot(int start_from); // Saves the currently selected tab. void SaveCurrentTab() { original_selected_tab_ = browser_->active_index(); } // Reverts the selected browser tab to the tab that was selected // when This BrowserListener was created, or the last time // SaveCurrentTab was called. void RestoreOriginalSelectedTab(); // Selects the tab at the given index. void SelectTab(int index, uint32 timestamp); // Shows any snapshots that are not visible. void ShowSnapshots(); // Callback for |AskForSnapshot|, start delay timer for next round. void OnSnapshotReady(const SkBitmap& sk_bitmap); // Returns the tab contents from the tab model for this child at index. TabContents* GetTabContentsAt(int index) const { return browser_->tabstrip_model()->GetTabContentsAt(index)->tab_contents(); } private: // Calculate the size of a cell based on the browser window's size. gfx::Size CalculateCellSize(); // Configures a cell from the tab contents. void ConfigureCell(WmOverviewSnapshot* cell, TabContents* contents); // Configures a cell from the model. void ConfigureCell(WmOverviewSnapshot* cell, int index) { ConfigureCell(cell, GetTabContentsAt(index)); } // Inserts a new snapshot, initialized from the model, at the given // index, and renumbers any following snapshots. void InsertSnapshot(int index); // Removes the snapshot at index. void ClearSnapshot(int index); // Renumbers the index atom in the snapshots starting at the given // index. void RenumberSnapshots(int start_index); Browser* browser_; // Not owned WmOverviewController* controller_; // Not owned // Which renderer host we are working on. RenderWidgetHost* current_renderer_host_; // Not owned // Widgets containing snapshot images for this browser. Note that // these are all subclasses of WidgetGtk, and they are all added to // parents, so they will be deleted by the parents when they are // closed. struct SnapshotNode { WmOverviewSnapshot* snapshot; // Not owned WmOverviewTitle* title; // Not owned WmOverviewFavicon* favicon; // Not owned }; typedef std::vector<SnapshotNode> SnapshotVector; SnapshotVector snapshots_; // Non-zero if we are currently setting the tab from within SelectTab. // This is used to make sure we use the right timestamp when sending // property changes that originated from the window manager. uint32 select_tab_timestamp_; // The tab selected the last time SaveCurrentTab is called. int original_selected_tab_; DISALLOW_COPY_AND_ASSIGN(BrowserListener); }; BrowserListener::BrowserListener(Browser* browser, WmOverviewController* controller) : browser_(browser), controller_(controller), current_renderer_host_(NULL), select_tab_timestamp_(0), original_selected_tab_(-1) { CHECK(browser_); CHECK(controller_); browser_->tabstrip_model()->AddObserver(this); // This browser didn't already exist, and so we haven't been // watching it for tab insertions, so we need to create the // snapshots associated with it. RecreateSnapshots(); } BrowserListener::~BrowserListener() { browser_->tabstrip_model()->RemoveObserver(this); } void BrowserListener::TabInsertedAt(TabContentsWrapper* contents, int index, bool foreground) { InsertSnapshot(index); RenumberSnapshots(index); UpdateSelectedIndex(browser_->active_index()); } void BrowserListener::TabDetachedAt(TabContentsWrapper* contents, int index) { ClearSnapshot(index); UpdateSelectedIndex(browser_->active_index()); RenumberSnapshots(index); } void BrowserListener::TabMoved(TabContentsWrapper* contents, int from_index, int to_index) { // Need to reorder tab in the snapshots list, and reset the window // type atom on the affected snapshots (the one moved, and all the // ones after it), so that their indices are correct. SnapshotNode node = snapshots_[from_index]; snapshots_.erase(snapshots_.begin() + from_index); snapshots_.insert(snapshots_.begin() + to_index, node); RenumberSnapshots(std::min(to_index, from_index)); UpdateSelectedIndex(browser_->active_index()); } void BrowserListener::TabChangedAt( TabContentsWrapper* contents, int index, TabStripModelObserver::TabChangeType change_type) { if (change_type != TabStripModelObserver::LOADING_ONLY) { snapshots_[index].title->SetTitle(contents->tab_contents()->GetTitle()); snapshots_[index].title->SetUrl(contents->tab_contents()->GetURL()); snapshots_[index].favicon->SetFavicon( contents->tab_contents()->GetFavicon()); if (change_type != TabStripModelObserver::TITLE_NOT_LOADING) MarkSnapshotAsDirty(index); } } void BrowserListener::TabStripEmpty() { snapshots_.clear(); } void BrowserListener::TabSelectedAt(TabContentsWrapper* old_contents, TabContentsWrapper* new_contents, int index, bool user_gesture) { if (old_contents == new_contents) return; UpdateSelectedIndex(index); } void BrowserListener::MarkSnapshotAsDirty(int index) { snapshots_[index].snapshot->reload_snapshot(); controller_->UpdateSnapshots(); } void BrowserListener::RecreateSnapshots() { snapshots_.clear(); for (int i = 0; i < count(); ++i) InsertSnapshot(i); RenumberSnapshots(0); } void BrowserListener::UpdateSelectedIndex(int index) { WmIpcWindowType type = WmIpc::instance()->GetWindowType( GTK_WIDGET(browser_->window()->GetNativeHandle()), NULL); // Make sure we only operate on toplevel windows. if (type == WM_IPC_WINDOW_CHROME_TOPLEVEL) { std::vector<int> params; params.push_back(browser_->tab_count()); params.push_back(index); params.push_back(select_tab_timestamp_ ? select_tab_timestamp_ : gtk_get_current_event_time()); WmIpc::instance()->SetWindowType( GTK_WIDGET(browser_->window()->GetNativeHandle()), WM_IPC_WINDOW_CHROME_TOPLEVEL, ¶ms); } } int BrowserListener::ConfigureNextUnconfiguredSnapshot(int start_from) { for (SnapshotVector::size_type i = start_from + 1; i < snapshots_.size(); ++i) { WmOverviewSnapshot* cell = snapshots_[i].snapshot; if (!cell->configured_snapshot()) { ConfigureCell(cell, i); return i; } } return -1; } void BrowserListener::RestoreOriginalSelectedTab() { if (original_selected_tab_ >= 0) { browser_->ActivateTabAt(original_selected_tab_, false); UpdateSelectedIndex(browser_->active_index()); } } void BrowserListener::ShowSnapshots() { for (SnapshotVector::size_type i = 0; i < snapshots_.size(); ++i) { const SnapshotNode& node = snapshots_[i]; if (!node.snapshot->IsVisible()) node.snapshot->Show(); if (!snapshots_[i].title->IsVisible()) node.title->Show(); if (!snapshots_[i].favicon->IsVisible()) node.favicon->Show(); } } void BrowserListener::SelectTab(int index, uint32 timestamp) { // Ignore requests to switch to non-existent tabs (the window manager gets // notified asynchronously about the number of tabs in each window, so there's // no guarantee that the messages that it sends us will make sense). if (index < 0 || index >= browser_->tab_count()) return; uint32 old_value = select_tab_timestamp_; select_tab_timestamp_ = timestamp; browser_->ActivateTabAt(index, true); select_tab_timestamp_ = old_value; } gfx::Size BrowserListener::CalculateCellSize() { // Make the page size and the cell size a fixed size for overview // mode. The cell size is calculated based on the desired maximum // size on the screen, so it's related to the width and height of // the browser client area. In this way, when this snapshot gets // to the window manager, it will already have the correct size, // and will be scaled by 1.0, meaning that it won't be resampled // and will not be blurry. gfx::Rect bounds = static_cast<BrowserView*>(browser_->window())-> GetClientAreaBounds(); const gfx::Size max_size = gfx::Size( bounds.width() * kSnapshotMaxSizeRatio, bounds.height() * kSnapshotMaxSizeRatio); const double max_size_ratio = static_cast<double>(max_size.width()) / max_size.height(); gfx::Size cell_size; if (kSnapshotWebPageRatio > max_size_ratio) { const double scale_factor = static_cast<double>(max_size.width())/ kSnapshotWebPageWidth; cell_size = gfx::Size(max_size.width(), kSnapshotWebPageHeight * scale_factor + 0.5); } else { const double scale_factor = static_cast<double>(max_size.height())/ kSnapshotWebPageHeight; cell_size = gfx::Size(kSnapshotWebPageWidth * scale_factor + 0.5, max_size.height()); } return cell_size; } void BrowserListener::OnSnapshotReady(const SkBitmap& sk_bitmap) { for (int i = 0; i < count(); i++) { RenderWidgetHostView* view = GetTabContentsAt(i)->GetRenderWidgetHostView(); if (view && view->GetRenderWidgetHost() == current_renderer_host_) { snapshots_[i].snapshot->SetImage(sk_bitmap); current_renderer_host_ = NULL; // Start timer for next round of snapshot updating. controller_->StartDelayTimer(); return; } } DCHECK(current_renderer_host_ == NULL); } void BrowserListener::ConfigureCell(WmOverviewSnapshot* cell, TabContents* contents) { if (contents) { ThumbnailGenerator* generator = g_browser_process->GetThumbnailGenerator(); // TODO: Make sure that if the cell gets deleted before the // callback is called that it sticks around until it gets // called. (some kind of "in flight" list that uses linked_ptr // to make sure they don't actually get deleted?) Also, make // sure that any request for a thumbnail eventually returns // (even if it has bogus data), so we don't leak orphaned cells, // which could happen if a tab is closed while it is being // rendered. ThumbnailGenerator::ThumbnailReadyCallback* callback = NewCallback(this, &BrowserListener::OnSnapshotReady); current_renderer_host_ = contents->render_view_host(); generator->AskForSnapshot(contents->render_view_host(), false, callback, gfx::Size(kSnapshotWebPageWidth, kSnapshotWebPageHeight), CalculateCellSize()); } else { // This happens because the contents haven't been loaded yet. // Make sure we set the snapshot image to something, otherwise // configured_snapshot remains false and // ConfigureNextUnconfiguredSnapshot would get stuck. current_renderer_host_ = NULL; cell->SetImage(SkBitmap()); cell->reload_snapshot(); controller_->StartDelayTimer(); } } void BrowserListener::InsertSnapshot(int index) { SnapshotNode node; node.snapshot = new WmOverviewSnapshot; gfx::Size cell_size = CalculateCellSize(); node.snapshot->Init(cell_size, browser_, index); node.favicon = new WmOverviewFavicon; node.favicon->Init(node.snapshot); node.favicon->SetFavicon(browser_->GetTabContentsAt(index)->GetFavicon()); node.title = new WmOverviewTitle; node.title->Init(gfx::Size(std::max(0, cell_size.width() - WmOverviewFavicon::kIconSize - kFaviconPadding), kTitleHeight), node.snapshot); node.title->SetTitle(browser_->GetTabContentsAt(index)->GetTitle()); snapshots_.insert(snapshots_.begin() + index, node); node.snapshot->reload_snapshot(); controller_->UpdateSnapshots(); } // Removes the snapshot at index. void BrowserListener::ClearSnapshot(int index) { snapshots_[index].snapshot->CloseNow(); snapshots_[index].title->CloseNow(); snapshots_[index].favicon->CloseNow(); snapshots_.erase(snapshots_.begin() + index); } void BrowserListener::RenumberSnapshots(int start_index) { for (SnapshotVector::size_type i = start_index; i < snapshots_.size(); ++i) { if (snapshots_[i].snapshot->index() != static_cast<int>(i)) snapshots_[i].snapshot->UpdateIndex(browser_, i); } } /////////////////////////////////// // WmOverviewController methods // static WmOverviewController* WmOverviewController::GetInstance() { static WmOverviewController* instance = NULL; if (!instance) { instance = Singleton<WmOverviewController>::get(); } return instance; } WmOverviewController::WmOverviewController() : layout_mode_(ACTIVE_MODE), updating_snapshots_(false), browser_listener_index_(0), tab_contents_index_(-1) { AddAllBrowsers(); if (registrar_.IsEmpty()) { // Make sure we get notifications for when the tab contents are // connected, so we know when a new browser has been created. registrar_.Add(this, NotificationType::TAB_CONTENTS_CONNECTED, NotificationService::AllSources()); // Ask for notification when the snapshot source image has changed // and needs to be refreshed. registrar_.Add(this, NotificationType::THUMBNAIL_GENERATOR_SNAPSHOT_CHANGED, NotificationService::AllSources()); } BrowserList::AddObserver(this); WmMessageListener::GetInstance()->AddObserver(this); } WmOverviewController::~WmOverviewController() { WmMessageListener::GetInstance()->RemoveObserver(this); BrowserList::RemoveObserver(this); listeners_.clear(); } void WmOverviewController::Observe(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { switch (type.value) { // Now that the tab contents are ready, we create the listeners // and snapshots for any new browsers out there. This actually // results in us traversing the list of browsers more often than // necessary (whenever a tab is connected, as opposed to only when // a new browser is created), but other notifications aren't // sufficient to know when the first tab of a new browser has its // dimensions set. The implementation of AddAllBrowsers avoids // doing anything to already-existing browsers, so it's not a huge // problem, but still, it would be nice if there were a more // appropriate (browser-level) notification. case NotificationType::TAB_CONTENTS_CONNECTED: AddAllBrowsers(); break; case NotificationType::THUMBNAIL_GENERATOR_SNAPSHOT_CHANGED: { // It's OK to do this in active mode too -- nothing will happen // except invalidation of the snapshot, because the delay timer // won't start until we're in overview mode. RenderWidgetHost* renderer = Details<RenderViewHost>(details).ptr(); SnapshotImageChanged(renderer); break; } default: // Do nothing. break; } } void WmOverviewController::SnapshotImageChanged(RenderWidgetHost* renderer) { // Find out which TabContents this renderer is attached to, and then // invalidate the associated snapshot so it'll update. BrowserListenerVector::iterator iter = listeners_.begin(); while (iter != listeners_.end()) { for (int i = 0; i < (*iter)->count(); i++) { RenderWidgetHostView* view = (*iter)->GetTabContentsAt(i)->GetRenderWidgetHostView(); if (view && view->GetRenderWidgetHost() == renderer) { (*iter)->MarkSnapshotAsDirty(i); return; } } ++iter; } DLOG(ERROR) << "SnapshotImageChanged, but we do not know which it is?"; } void WmOverviewController::OnBrowserRemoved(const Browser* browser) { for (BrowserListenerVector::iterator i = listeners_.begin(); i != listeners_.end(); ++i) { if ((*i)->browser() == browser) { listeners_.erase(i); return; } } } BrowserView* GetBrowserViewForGdkWindow(GdkWindow* gdk_window) { gpointer data = NULL; gdk_window_get_user_data(gdk_window, &data); GtkWidget* widget = reinterpret_cast<GtkWidget*>(data); if (widget) { GtkWindow* gtk_window = GTK_WINDOW(widget); return BrowserView::GetBrowserViewForNativeWindow(gtk_window); } else { return NULL; } } void WmOverviewController::ProcessWmMessage(const WmIpc::Message& message, GdkWindow* window) { switch (message.type()) { case WM_IPC_MESSAGE_CHROME_NOTIFY_LAYOUT_MODE: { layout_mode_ = message.param(0) == 0 ? ACTIVE_MODE : OVERVIEW_MODE; if (layout_mode_ == ACTIVE_MODE || BrowserList::size() == 0) { Hide(message.param(1) != 0); } else { Show(); } break; } case WM_IPC_MESSAGE_CHROME_NOTIFY_TAB_SELECT: { BrowserView* browser_window = GetBrowserViewForGdkWindow(window); // Find out which listener this goes to, and send it there. for (BrowserListenerVector::iterator i = listeners_.begin(); i != listeners_.end(); ++i) { if ((*i)->browser()->window() == browser_window) { // param(0): index of the tab to select. // param(1): timestamp of the event. (*i)->SelectTab(message.param(0), message.param(1)); break; } } break; } default: break; } } void WmOverviewController::StartDelayTimer() { // We're rate limiting the number of times we can reconfigure the // snapshots. If we were to restart the delay timer, it could // result in a very long delay until they get configured if tabs // keep changing. updating_snapshots_ = false; if (layout_mode_ == OVERVIEW_MODE) { delay_timer_.Start( base::TimeDelta::FromMilliseconds(kDelayTimeMs), this, &WmOverviewController::UpdateSnapshots); } } void WmOverviewController::RestoreTabSelections() { for (BrowserListenerVector::iterator i = listeners_.begin(); i != listeners_.end(); ++i) { (*i)->RestoreOriginalSelectedTab(); } } void WmOverviewController::SaveTabSelections() { for (BrowserListenerVector::iterator i = listeners_.begin(); i != listeners_.end(); ++i) { (*i)->SaveCurrentTab(); } } void WmOverviewController::Show() { SaveTabSelections(); for (BrowserListenerVector::iterator i = listeners_.begin(); i != listeners_.end(); ++i) { (*i)->ShowSnapshots(); } // TODO(jiesun): Make the focused tab as the start point. browser_listener_index_ = 0; tab_contents_index_ = -1; UpdateSnapshots(); } void WmOverviewController::Hide(bool cancelled) { delay_timer_.Stop(); updating_snapshots_ = false; if (cancelled) { RestoreTabSelections(); } } void WmOverviewController::UpdateSnapshots() { // Only updating snapshots during overview mode. if (layout_mode_ != OVERVIEW_MODE) return; // Only reloading snapshots when not already started. if (updating_snapshots_ || delay_timer_.IsRunning()) return; // Only update one snapshot in round-robin mode. // Start delay timer after each update unless all snapshots had been updated. if (!listeners_.size()) return; if (int(listeners_.size()) <= browser_listener_index_) { DLOG(INFO) << "Browser listener(s) have disappeared since last update"; browser_listener_index_ = 0; tab_contents_index_ = -1; } else { BrowserListener* listener = listeners_[browser_listener_index_].get(); if (listener->count() <= tab_contents_index_) { DLOG(INFO) << "Tab content(s) have disappeared since last update"; tab_contents_index_ = -1; } } int browser_listener_index = browser_listener_index_; int tab_contents_index = tab_contents_index_; bool loop_back = false; while (1) { BrowserListener* listener = listeners_[browser_listener_index].get(); tab_contents_index = listener->ConfigureNextUnconfiguredSnapshot(tab_contents_index); if (tab_contents_index >= 0) { updating_snapshots_ = true; // Prevent future parallel updates. browser_listener_index_ = browser_listener_index; tab_contents_index_ = tab_contents_index; return; } // Found next one; browser_listener_index++; browser_listener_index = browser_listener_index % listeners_.size(); tab_contents_index = -1; if (loop_back) break; loop_back = browser_listener_index == browser_listener_index_; } // All snapshots have been fully updated. updating_snapshots_ = false; } void WmOverviewController::AddAllBrowsers() { // Make a copy so the old ones aren't deleted yet. BrowserListenerVector old_listeners; listeners_.swap(old_listeners); // Iterator through the browser list, adding all the browsers in the // new order. If they were in the old list of listeners, just copy // that linked pointer, instead of making a new listener, so that we // can avoid lots of spurious destroy/create messages. BrowserList::const_iterator iterator = BrowserList::begin(); while (iterator != BrowserList::end()) { // Don't add a browser to the list if that type of browser doesn't // have snapshots in overview mode. if ((*iterator)->type() != Browser::TYPE_NORMAL && (*iterator)->type() != Browser::TYPE_APP) { ++iterator; continue; } BrowserListenerVector::value_type item( BrowserListenerVector::value_type(NULL)); for (BrowserListenerVector::iterator old_iter = old_listeners.begin(); old_iter != old_listeners.end(); ++old_iter) { if ((*old_iter)->browser() == *iterator) { item = *old_iter; break; } } // This browser isn't tracked by any listener, so create it. if (item.get() == NULL) { item = BrowserListenerVector::value_type( new BrowserListener(*iterator, this)); } listeners_.push_back(item); ++iterator; } } } // namespace chromeos