chromium/android_webview/renderer/aw_render_view_ext.cc

// 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 "android_webview/renderer/aw_render_view_ext.h"

#include <map>

#include "android_webview/common/mojom/frame.mojom.h"
#include "base/containers/contains.h"
#include "base/no_destructor.h"
#include "base/not_fatal_until.h"
#include "content/public/renderer/render_frame.h"
#include "third_party/blink/public/common/associated_interfaces/associated_interface_provider.h"
#include "third_party/blink/public/web/web_local_frame.h"
#include "third_party/blink/public/web/web_view.h"

namespace android_webview {

// Registry for blink::WebView => AwRenderViewExt lookups
typedef std::map<blink::WebView*, AwRenderViewExt*> ViewExtMap;
ViewExtMap* GetViewExtMap() {
  static base::NoDestructor<ViewExtMap> map;
  return map.get();
}

AwRenderViewExt::AwRenderViewExt(blink::WebView* web_view,
                                 bool created_by_renderer)
    : blink::WebViewObserver(web_view),
      created_by_renderer_(created_by_renderer) {
  DCHECK(web_view != nullptr);
  DCHECK(!base::Contains(*GetViewExtMap(), web_view));
  GetViewExtMap()->emplace(web_view, this);
}

AwRenderViewExt::~AwRenderViewExt() {
  // Remove myself from the blink::WebView => AwRenderViewExt register. Ideally,
  // we'd just use GetWebView() and erase by key. However, by this time the
  // GetWebView has already been cleared so we have to iterate over all
  // WebViews in the map and wipe the one(s) that point to this
  // AwRenderViewExt

  auto* map = GetViewExtMap();
  auto it = map->begin();
  while (it != map->end()) {
    if (it->second == this) {
      it = map->erase(it);
    } else {
      ++it;
    }
  }
}

// static
void AwRenderViewExt::WebViewCreated(blink::WebView* web_view,
                                     bool created_by_renderer) {
  new AwRenderViewExt(web_view,
                      created_by_renderer);  // |web_view| takes ownership.
}

// static
AwRenderViewExt* AwRenderViewExt::FromWebView(blink::WebView* web_view) {
  DCHECK(web_view != nullptr);
  auto iter = GetViewExtMap()->find(web_view);
  CHECK(GetViewExtMap()->end() != iter, base::NotFatalUntil::M130)
      << "AwRenderViewExt should always exist for a WebView";
  AwRenderViewExt* render_view_ext = iter->second;
  return render_view_ext;
}

void AwRenderViewExt::DidCommitCompositorFrame() {
  UpdateContentsSize();
}

void AwRenderViewExt::DidUpdateMainFrameLayout() {
  // The size may have changed.
  needs_contents_size_update_ = true;
}

void AwRenderViewExt::OnDestruct() {
  delete this;
}

void AwRenderViewExt::UpdateContentsSize() {
  blink::WebView* webview = GetWebView();
  blink::WebFrame* main_frame = webview->MainFrame();

  // Even without out-of-process iframes, we now create RemoteFrames for the
  // main frame when you navigate cross-process, to create a placeholder in the
  // old process. This is necessary to support things like postMessage across
  // windows that have references to each other. The RemoteFrame will
  // immediately go away if there aren't any active frames left in the old
  // process.
  if (!main_frame->IsWebLocalFrame())
    return;

  if (!needs_contents_size_update_)
    return;
  needs_contents_size_update_ = false;

  gfx::Size contents_size = main_frame->ToWebLocalFrame()->DocumentSize();

  // Fall back to contentsPreferredMinimumSize if the mainFrame is reporting a
  // 0x0 size (this happens during initial load).
  if (contents_size.IsEmpty()) {
    contents_size = webview->ContentsPreferredMinimumSize();
  }

  if (contents_size == last_sent_contents_size_)
    return;

  last_sent_contents_size_ = contents_size;

  mojo::AssociatedRemote<mojom::FrameHost> frame_host_remote;
  content::RenderFrame::FromWebFrame(main_frame->ToWebLocalFrame())
      ->GetRemoteAssociatedInterfaces()
      ->GetInterface(&frame_host_remote);
  frame_host_remote->ContentsSizeChanged(contents_size);
}

}  // namespace android_webview