File: reload_page_dialog_controller.cc

package info (click to toggle)
chromium 140.0.7339.127-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 6,192,880 kB
  • sloc: cpp: 35,093,808; ansic: 7,161,670; javascript: 4,199,694; python: 1,441,797; asm: 949,904; xml: 747,503; pascal: 187,748; perl: 88,691; sh: 88,248; objc: 79,953; sql: 52,714; cs: 44,599; fortran: 24,137; makefile: 22,114; tcl: 15,277; php: 13,980; yacc: 9,000; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (162 lines) | stat: -rw-r--r-- 6,268 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
// Copyright 2025 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/ui/extensions/reload_page_dialog_controller.h"

#include "base/barrier_closure.h"
#include "base/feature_list.h"
#include "base/functional/bind.h"
#include "base/functional/callback_forward.h"
#include "base/functional/callback_helpers.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/extensions/extension_util.h"
#include "chrome/browser/ui/extensions/extension_dialog_utils.h"
#include "chrome/browser/ui/toolbar/toolbar_action_view_controller.h"
#include "chrome/grit/generated_resources.h"
#include "content/public/browser/browser_context.h"
#include "extensions/browser/extension_icon_placeholder.h"
#include "extensions/browser/image_loader.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_features.h"
#include "extensions/common/manifest_handlers/icons_handler.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/base/models/dialog_model.h"

namespace {

std::u16string GetTitle(
    const std::vector<extensions::ReloadPageDialogController::ExtensionInfo>&
        extensions_info) {
  if (!base::FeatureList::IsEnabled(
          extensions_features::kExtensionsMenuAccessControl) ||
      extensions_info.empty()) {
    return l10n_util::GetStringUTF16(
        IDS_EXTENSION_SITE_RELOAD_PAGE_BUBBLE_HEADING);
  }

  if (extensions_info.size() == 1) {
    std::u16string extension_name =
        extensions::util::GetFixupExtensionNameForUIDisplay(
            base::UTF8ToUTF16(extensions_info[0].name));
    return l10n_util::GetStringFUTF16(
        IDS_EXTENSION_RELOAD_PAGE_BUBBLE_ALLOW_SINGLE_EXTENSION_TITLE,
        extension_name);
  }

  return l10n_util::GetStringUTF16(
      IDS_EXTENSION_RELOAD_PAGE_BUBBLE_ALLOW_MULTIPLE_EXTENSIONS_TITLE);
}

}  // namespace

namespace extensions {

DEFINE_ELEMENT_IDENTIFIER_VALUE(kReloadPageDialogOkButtonElementId);
DEFINE_ELEMENT_IDENTIFIER_VALUE(kReloadPageDialogCancelButtonElementId);

ReloadPageDialogController::ReloadPageDialogController(
    gfx::NativeWindow parent,
    content::BrowserContext* browser_context,
    base::OnceClosure callback)
    : parent_(parent),
      browser_context_(browser_context),
      on_dialog_accepted_(std::move(callback)) {}
ReloadPageDialogController::~ReloadPageDialogController() = default;

void ReloadPageDialogController::TriggerShow(
    const std::vector<const Extension*>& extensions) {
  if (!base::FeatureList::IsEnabled(
          extensions_features::kExtensionsMenuAccessControl)) {
    for (const Extension* extension : extensions) {
      ExtensionInfo extension_info;
      extension_info.id = extension->id();
      extensions_info_.push_back(extension_info);
    }
    Show();
    return;
  }

  // We need to load the icon for each extension before showing the dialog.
  // Since icon loading is asynchronous, we use a BarrierClosure. It acts as
  // a counter and will call this->Show() only after all icon-loading callbacks
  // have completed.
  int extensions_count = extensions.size();
  base::RepeatingClosure barrier_closure = base::BarrierClosure(
      extensions_count, base::BindOnce(&ReloadPageDialogController::Show,
                                       weak_ptr_factory_.GetWeakPtr()));
  const int icon_size = extension_misc::EXTENSION_ICON_SMALLISH;
  auto* image_loader = ImageLoader::Get(browser_context_);

  for (const Extension* extension : extensions) {
    ExtensionResource icon = IconsInfo::GetIconResource(
        extension, icon_size, ExtensionIconSet::Match::kBigger);
    if (icon.empty()) {
      gfx::Image placeholder_icon =
          ExtensionIconPlaceholder::CreateImage(icon_size, extension->name());
      OnExtensionIconLoaded(extension->id(), extension->name(), barrier_closure,
                            placeholder_icon);
    } else {
      gfx::Size max_size(icon_size, icon_size);
      image_loader->LoadImageAsync(
          extension, icon, max_size,
          base::BindOnce(&ReloadPageDialogController::OnExtensionIconLoaded,
                         weak_ptr_factory_.GetWeakPtr(), extension->id(),
                         extension->name(), barrier_closure));
    }
  }
}

void ReloadPageDialogController::Show() {
  ui::DialogModel::Builder dialog_builder;
  dialog_builder.SetTitle(GetTitle(extensions_info_))
      .AddOkButton(base::BindOnce(std::move(on_dialog_accepted_)),
                   ui::DialogModel::Button::Params()
                       .SetLabel(l10n_util::GetStringUTF16(
                           IDS_EXTENSION_RELOAD_PAGE_BUBBLE_OK_BUTTON))
                       .SetId(kReloadPageDialogOkButtonElementId))
      .AddCancelButton(base::DoNothing(),
                       ui::DialogModel::Button::Params().SetId(
                           kReloadPageDialogCancelButtonElementId));

  int extensions_count = extensions_info_.size();

  if (base::FeatureList::IsEnabled(
          extensions_features::kExtensionsMenuAccessControl)) {
    if (extensions_count == 1) {
      dialog_builder.SetIcon(
          ui::ImageModel::FromImage(extensions_info_[0].icon));
    } else if (extensions_count > 1) {
      for (auto extension_info : extensions_info_) {
        dialog_builder.AddMenuItem(
            ui::ImageModel::FromImage(extension_info.icon),
            util::GetFixupExtensionNameForUIDisplay(extension_info.name),
            base::DoNothing(),
            ui::DialogModelMenuItem::Params().SetIsEnabled(false));
      }
    }
  }

  std::vector<ExtensionId> extension_ids;
  extension_ids.reserve(extensions_info_.size());
  for (const auto& info : extensions_info_) {
    extension_ids.push_back(info.id);
  }

  ShowDialog(parent_, extension_ids, dialog_builder.Build());
}

void ReloadPageDialogController::OnExtensionIconLoaded(
    const ExtensionId& extension_id,
    const std::string& extension_name,
    base::OnceClosure done_callback,
    const gfx::Image& icon) {
  ExtensionInfo extension_info;
  extension_info.id = extension_id;
  extension_info.name = extension_name;
  extension_info.icon = icon;
  extensions_info_.push_back(extension_info);
  std::move(done_callback).Run();
}

}  // namespace extensions