chromium/components/ui_devtools/views/dom_agent_mac.mm

// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "components/ui_devtools/views/dom_agent_mac.h"

#import <AppKit/AppKit.h>

#include "base/functional/bind.h"
#include "base/ranges/algorithm.h"
#include "components/ui_devtools/views/widget_element.h"
#include "ui/views/widget/native_widget_mac.h"

namespace ui_devtools {

DOMAgentMac::DOMAgentMac() = default;

DOMAgentMac::~DOMAgentMac() {
  CHECK(!IsInObserverList());
}

protocol::Response DOMAgentMac::enable() {
  init_native_widget_subscription_ =
      views::NativeWidgetMac::RegisterInitNativeWidgetCallback(
          base::BindRepeating(&DOMAgentMac::OnNativeWidgetAdded,
                              base::Unretained(this)));
  return DOMAgent::enable();
}

protocol::Response DOMAgentMac::disable() {
  init_native_widget_subscription_ = {};
  for (views::Widget* widget : roots_)
    widget->RemoveObserver(this);
  roots_.clear();
  return DOMAgent::disable();
}

std::vector<UIElement*> DOMAgentMac::CreateChildrenForRoot() {
  if (roots_.size() == 0)
    InitializeRootsFromOpenWindows();

  std::vector<UIElement*> children;
  for (views::Widget* widget : roots_) {
    UIElement* widget_element = new WidgetElement(widget, this, element_root());
    children.push_back(widget_element);
  }
  return children;
}

void DOMAgentMac::OnWidgetDestroying(views::Widget* widget) {
  widget->RemoveObserver(this);
  roots_.erase(base::ranges::find(roots_, widget), roots_.end());
}

void DOMAgentMac::OnNativeWidgetAdded(views::NativeWidgetMac* native_widget) {
  views::Widget* widget = native_widget->GetWidget();
  DCHECK(widget);
  roots_.push_back(widget);
  UIElement* widget_element = new WidgetElement(widget, this, element_root());
  element_root()->AddChild(widget_element);
  widget->AddObserver(this);
}

std::unique_ptr<protocol::DOM::Node> DOMAgentMac::BuildTreeForWindow(
    UIElement* window_element_root) {
  // Window elements aren't supported on Mac.
  NOTREACHED_IN_MIGRATION();
  return nullptr;
}

void DOMAgentMac::InitializeRootsFromOpenWindows() {
  for (NSWindow* window in NSApp.windows) {
    if (views::Widget* widget =
            views::Widget::GetWidgetForNativeWindow(window)) {
      // When in immersive fullscreen mode, an overlay widget has two associated
      // NSWindows:
      // 1. An invisible one created by Chrome, which serves as an anchor
      //    for child widgets.
      // 2. A visible AppKit-owned NSToolbarFullScreenWindow.
      // We ensures here that a widget is only observed once.
      if (!widget->HasObserver(this)) {
        widget->AddObserver(this);
        roots_.push_back(widget);
      }
    }
  }
}

// static
std::unique_ptr<DOMAgentViews> DOMAgentViews::Create() {
  return std::make_unique<DOMAgentMac>();
}

}  // namespace ui_devtools