File: context_menu_runner.mm

package info (click to toggle)
chromium 138.0.7204.183-1~deb12u1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm-proposed-updates
  • size: 6,080,960 kB
  • sloc: cpp: 34,937,079; ansic: 7,176,967; javascript: 4,110,704; python: 1,419,954; asm: 946,768; xml: 739,971; pascal: 187,324; sh: 89,623; perl: 88,663; objc: 79,944; sql: 50,304; cs: 41,786; fortran: 24,137; makefile: 21,811; php: 13,980; tcl: 13,166; yacc: 8,925; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (102 lines) | stat: -rw-r--r-- 3,344 bytes parent folder | download | duplicates (4)
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
// Copyright 2024 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/remote_cocoa/app_shim/context_menu_runner.h"

#include "base/strings/sys_string_conversions.h"
#include "components/remote_cocoa/app_shim/mojo_menu_model.h"
#include "ui/base/cocoa/menu_utils.h"

namespace remote_cocoa {

namespace {

// Retrieves an NSMenuItem which has the specified command_id. This function
// traverses the given `model` in the depth-first order. When this function
// finds an item whose command_id is the same as the given `command_id`, it
// returns the NSMenuItem associated with the item. This function emulates
// views::MenuItemViews::GetMenuItemByID() for Mac.
NSMenuItem* GetMenuItemById(ui::MenuModel* model,
                            NSMenu* menu,
                            int command_id) {
  for (size_t i = 0; i < model->GetItemCount(); ++i) {
    NSMenuItem* item = [menu itemAtIndex:i];
    if (model->GetCommandIdAt(i) == command_id) {
      return item;
    }

    ui::MenuModel* submenu = model->GetSubmenuModelAt(i);
    if (submenu && [item hasSubmenu]) {
      NSMenuItem* subitem =
          GetMenuItemById(submenu, [item submenu], command_id);
      if (subitem) {
        return subitem;
      }
    }
  }
  return nil;
}

}  // namespace

ContextMenuRunner::ContextMenuRunner(
    mojo::PendingRemote<mojom::MenuHost> host,
    mojo::PendingReceiver<mojom::Menu> receiver)
    : receiver_(this, std::move(receiver)), menu_host_(std::move(host)) {}

ContextMenuRunner::~ContextMenuRunner() {
  if (menu_controller_) {
    CHECK(!menu_controller_.menuOpen);
  }
}

void ContextMenuRunner::ShowMenu(mojom::ContextMenuPtr menu,
                                 NSWindow* window,
                                 NSView* target_view) {
  menu_model_ =
      std::make_unique<MojoMenuModel>(std::move(menu->items), menu_host_.get());
  menu_delegate_ = [[MenuControllerCocoaDelegateImpl alloc]
      initWithParams:std::move(menu->params)];
  menu_controller_ =
      [[MenuControllerCocoa alloc] initWithModel:menu_model_.get()
                                        delegate:menu_delegate_];

  if (!target_view) {
    target_view = window.contentView;
  }

  NSEvent* clickEvent =
      ui::EventForPositioningContextMenu(menu->anchor, window);

  ui::ShowContextMenu(menu_controller_.menu, clickEvent, target_view,
                      /*allow_nested_tasks=*/true);

  menu_host_->MenuClosed();
}

void ContextMenuRunner::Cancel() {
  if (menu_controller_) {
    [menu_controller_ cancel];
  }
}

void ContextMenuRunner::UpdateMenuItem(int32_t command_id,
                                       bool enabled,
                                       bool visible,
                                       const std::u16string& label) {
  NSMenuItem* item =
      GetMenuItemById(menu_model_.get(), menu_controller_.menu, command_id);
  if (!item) {
    return;
  }

  // Update the returned NSMenuItem directly so we can update it immediately.
  // There is no need to update the MenuModel as well, since the model is only
  // read from to create the initial NSMenu and never touched again later.
  item.enabled = enabled;
  item.title = base::SysUTF16ToNSString(label);
  item.hidden = !visible;
}

}  // namespace remote_cocoa