blob: fbfc11276b79ea0992ce7cb78b4d219d89c4198e [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// 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_controller_list.h"
#include <algorithm>
#include "base/containers/contains.h"
#include "base/observer_list.h"
#include "chrome/browser/extensions/api/tabs/windows_util.h"
#include "chrome/browser/extensions/window_controller_list_observer.h"
#include "chrome/common/extensions/api/windows.h"
#include "components/sessions/core/session_id.h"
#include "extensions/browser/extension_function.h"
#include "ui/base/base_window.h"
namespace extensions {
///////////////////////////////////////////////////////////////////////////////
// WindowControllerList
// static
WindowControllerList* WindowControllerList::GetInstance() {
return base::Singleton<WindowControllerList>::get();
}
WindowControllerList::WindowControllerList() = default;
WindowControllerList::~WindowControllerList() = default;
void WindowControllerList::AddExtensionWindow(WindowController* window) {
windows_.push_back(window);
for (auto& observer : observers_)
observer.OnWindowControllerAdded(window);
}
void WindowControllerList::RemoveExtensionWindow(WindowController* window) {
auto iter = std::ranges::find(windows_, window);
if (iter != windows_.end()) {
windows_.erase(iter);
for (auto& observer : observers_)
observer.OnWindowControllerRemoved(window);
}
}
void WindowControllerList::NotifyWindowBoundsChanged(WindowController* window) {
if (base::Contains(windows_, window)) {
for (auto& observer : observers_)
observer.OnWindowBoundsChanged(window);
}
}
void WindowControllerList::AddObserver(WindowControllerListObserver* observer) {
observers_.AddObserver(observer);
}
void WindowControllerList::RemoveObserver(
WindowControllerListObserver* observer) {
observers_.RemoveObserver(observer);
}
WindowController* WindowControllerList::FindWindowForFunctionByIdWithFilter(
const ExtensionFunction* function,
int id,
WindowController::TypeFilter filter) const {
for (auto iter = windows_.begin(); iter != windows_.end(); ++iter) {
if ((*iter)->GetWindowId() == id) {
if (windows_util::CanOperateOnWindow(function, *iter, filter))
return *iter;
return nullptr;
}
}
return nullptr;
}
WindowController* WindowControllerList::CurrentWindowForFunction(
ExtensionFunction* function) const {
return CurrentWindowForFunctionWithFilter(function,
WindowController::kNoWindowFilter);
}
WindowController* WindowControllerList::CurrentWindowForFunctionWithFilter(
ExtensionFunction* function,
WindowController::TypeFilter filter) const {
// Always prefer the focused window if available. If there is no focused
// window, prefer the window to which the sender window is logically parented.
// Since the browser window is not "focused" when an extension popup is open
// (because popup is hosted in a separate window, which is focused instead),
// we need to check for the logical parent window here. If neither of these
// are available, return the last window.
WindowController* last_window = nullptr;
WindowController* parent_window = nullptr;
for (const auto& controller : windows_) {
if (!windows_util::CanOperateOnWindow(function, controller, filter)) {
continue;
}
if (controller->window()->IsActive()) {
// If the window is focused, return it immediately.
return controller;
}
#if !BUILDFLAG(IS_ANDROID)
// TODO(crbug.com/371432155): Support on Android.
// windows_util::CalledFromChildWindow() checks native widgets for parents.
if (windows_util::CalledFromChildWindow(function, controller)) {
parent_window = controller;
}
#endif
last_window = controller;
}
return parent_window ? parent_window : last_window;
}
} // namespace extensions