// Copyright (c) 2012 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/extensions/window_event_router.h" #include "base/values.h" #include "chrome/browser/extensions/event_names.h" #include "chrome/browser/extensions/event_router.h" #include "chrome/browser/extensions/extension_system.h" #include "chrome/browser/extensions/window_controller.h" #include "chrome/browser/extensions/window_controller_list.h" #include "chrome/browser/profiles/profile.h" #include "chrome/common/chrome_notification_types.h" #include "chrome/common/extensions/extension_constants.h" #include "content/public/browser/notification_service.h" #if defined(TOOLKIT_GTK) #include "ui/base/x/active_window_watcher_x.h" #endif namespace event_names = extensions::event_names; namespace extensions { WindowEventRouter::WindowEventRouter(Profile* profile) : profile_(profile), focused_profile_(NULL), focused_window_id_(extension_misc::kUnknownWindowId) { DCHECK(!profile->IsOffTheRecord()); WindowControllerList::GetInstance()->AddObserver(this); #if defined(TOOLKIT_VIEWS) views::WidgetFocusManager::GetInstance()->AddFocusChangeListener(this); #elif defined(TOOLKIT_GTK) ui::ActiveWindowWatcherX::AddObserver(this); #elif defined(OS_MACOSX) // Needed for when no suitable window can be passed to an extension as the // currently focused window. registrar_.Add(this, chrome::NOTIFICATION_NO_KEY_WINDOW, content::NotificationService::AllSources()); #endif } WindowEventRouter::~WindowEventRouter() { WindowControllerList::GetInstance()->RemoveObserver(this); #if defined(TOOLKIT_VIEWS) views::WidgetFocusManager::GetInstance()->RemoveFocusChangeListener(this); #elif defined(TOOLKIT_GTK) ui::ActiveWindowWatcherX::RemoveObserver(this); #endif } void WindowEventRouter::OnWindowControllerAdded( WindowController* window_controller) { if (!profile_->IsSameProfile(window_controller->profile())) return; scoped_ptr args(new ListValue()); DictionaryValue* window_dictionary = window_controller->CreateWindowValue(); args->Append(window_dictionary); DispatchEvent(event_names::kOnWindowCreated, window_controller->profile(), args.Pass()); } void WindowEventRouter::OnWindowControllerRemoved( WindowController* window_controller) { if (!profile_->IsSameProfile(window_controller->profile())) return; int window_id = window_controller->GetWindowId(); scoped_ptr args(new ListValue()); args->Append(Value::CreateIntegerValue(window_id)); DispatchEvent(event_names::kOnWindowRemoved, window_controller->profile(), args.Pass()); } #if defined(TOOLKIT_VIEWS) void WindowEventRouter::OnNativeFocusChange( gfx::NativeView focused_before, gfx::NativeView focused_now) { if (!focused_now) OnActiveWindowChanged(NULL); } #elif defined(TOOLKIT_GTK) void WindowEventRouter::ActiveWindowChanged( GdkWindow* active_window) { if (!active_window) OnActiveWindowChanged(NULL); } #endif void WindowEventRouter::Observe( int type, const content::NotificationSource& source, const content::NotificationDetails& details) { #if defined(OS_MACOSX) if (chrome::NOTIFICATION_NO_KEY_WINDOW == type) { OnActiveWindowChanged(NULL); return; } #endif } void WindowEventRouter::OnActiveWindowChanged( WindowController* window_controller) { Profile* window_profile = NULL; int window_id = extension_misc::kUnknownWindowId; if (window_controller && profile_->IsSameProfile(window_controller->profile())) { window_profile = window_controller->profile(); window_id = window_controller->GetWindowId(); } if (focused_window_id_ == window_id) return; // window_profile is either the default profile for the active window, its // incognito profile, or NULL iff the previous profile is losing focus. // Note that |previous_focused_profile| may already be destroyed! Profile* previous_focused_profile = focused_profile_; focused_profile_ = window_profile; focused_window_id_ = window_id; scoped_ptr real_args(new ListValue()); real_args->Append(Value::CreateIntegerValue(window_id)); // When switching between windows in the default and incognito profiles, // dispatch WINDOW_ID_NONE to extensions whose profile lost focus that // can't see the new focused window across the incognito boundary. // See crbug.com/46610. scoped_ptr none_args(NULL); if (focused_profile_ != NULL && previous_focused_profile != NULL && focused_profile_ != previous_focused_profile) { none_args.reset(new ListValue()); none_args->Append(Value::CreateIntegerValue( extension_misc::kUnknownWindowId)); } // Note that we may pass a NULL |window_profile| for the |restrict_to_profile| // argument. ExtensionSystem::Get(profile_)->event_router()-> DispatchEventsToRenderersAcrossIncognito( event_names::kOnWindowFocusedChanged, real_args.Pass(), window_profile, none_args.Pass(), GURL()); } void WindowEventRouter::DispatchEvent(const char* event_name, Profile* profile, scoped_ptr args) { ExtensionSystem::Get(profile)->event_router()-> DispatchEventToRenderers(event_name, args.Pass(), profile, GURL()); } } // namespace extensions