OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "config.h" | |
6 #include "core/page/CustomContextMenuProvider.h" | |
7 | |
8 #include "core/dom/Document.h" | |
9 #include "core/dom/ElementTraversal.h" | |
10 #include "core/events/EventDispatcher.h" | |
11 #include "core/events/MouseEvent.h" | |
12 #include "core/html/HTMLMenuElement.h" | |
13 #include "core/html/HTMLMenuItemElement.h" | |
14 #include "core/page/ContextMenuController.h" | |
15 #include "core/page/Page.h" | |
16 #include "platform/ContextMenu.h" | |
17 | |
18 namespace blink { | |
19 | |
20 using namespace HTMLNames; | |
21 | |
22 CustomContextMenuProvider::CustomContextMenuProvider(HTMLMenuElement& menu, HTML Element& subject) | |
23 : m_menu(menu) | |
24 , m_subjectElement(subject) | |
25 { | |
26 } | |
27 | |
28 CustomContextMenuProvider::~CustomContextMenuProvider() | |
29 { | |
30 } | |
31 | |
32 void CustomContextMenuProvider::populateContextMenu(ContextMenu* menu) | |
33 { | |
34 populateContextMenuItems(*m_menu, *menu); | |
35 } | |
36 | |
37 void CustomContextMenuProvider::contextMenuItemSelected(const ContextMenuItem* i tem) | |
38 { | |
39 if (HTMLElement* element = menuItemAt(item->action())) { | |
40 RefPtrWillBeRawPtr<SimulatedMouseEvent> click = SimulatedMouseEvent::cre ate(EventTypeNames::click, m_menu->document().domWindow(), Event::create()); | |
41 click->setRelatedTarget(m_subjectElement.get()); | |
42 element->dispatchEvent(click.release()); | |
43 } | |
44 } | |
45 | |
46 void CustomContextMenuProvider::contextMenuCleared() | |
47 { | |
48 m_menuItems.clear(); | |
49 m_subjectElement = nullptr; | |
50 } | |
51 | |
52 void CustomContextMenuProvider::appendSeparator(ContextMenu& contextMenu) | |
53 { | |
54 // Avoid separators at the start of any menu and submenu. | |
55 if (!contextMenu.items().size()) | |
56 return; | |
57 | |
58 // Collapse all sequences of two or more adjacent separators in the menu or any submenus to a single separator. | |
tkent
2014/08/25 23:07:42
I recommend to wrap code comments in 80 columns.
pals
2014/08/26 07:13:23
Done.
| |
59 ContextMenuItem lastItem = contextMenu.items().last(); | |
60 if (lastItem.type() == SeparatorType) | |
61 return; | |
62 | |
63 contextMenu.appendItem(ContextMenuItem(SeparatorType, ContextMenuItemCustomT agNoAction, String())); | |
64 } | |
65 | |
66 void CustomContextMenuProvider::appendMenuItem(HTMLMenuItemElement* menuItem, Co ntextMenu& contextMenu) | |
67 { | |
68 // Avoid menuitems with no label. | |
69 String labelString = menuItem->fastGetAttribute(labelAttr); | |
70 if (labelString.isNull() || labelString.isEmpty()) | |
tkent
2014/08/25 23:07:42
isEmpty() is enough. isEmpty() contains isNull().
pals
2014/08/26 07:13:23
Done.
| |
71 return; | |
72 | |
73 m_menuItems.append(menuItem); | |
74 contextMenu.appendItem(ContextMenuItem(ActionType, static_cast<ContextMenuAc tion>(ContextMenuItemBaseCustomTag + m_menuItems.size() - 1), labelString)); | |
75 } | |
76 | |
77 void CustomContextMenuProvider::populateContextMenuItems(const HTMLMenuElement& menu, ContextMenu& contextMenu) | |
78 { | |
79 HTMLElement* nextElement = Traversal<HTMLElement>::firstWithin(menu); | |
80 while (nextElement) { | |
81 if (isHTMLHRElement(*nextElement)) { | |
82 appendSeparator(contextMenu); | |
83 nextElement = Traversal<HTMLElement>::next(*nextElement, &menu); | |
84 } else if (isHTMLMenuElement(*nextElement)) { | |
85 ContextMenu subMenu; | |
86 String labelString = nextElement->fastGetAttribute(labelAttr); | |
87 if (labelString.isNull() || labelString.isEmpty()) { | |
tkent
2014/08/25 23:07:42
This looks incorrect.
The following code is for t
pals
2014/08/26 07:13:23
My understanding from the above two statements is
tkent
2014/08/26 08:11:26
ok, let's revisit here after the clarification.
| |
88 appendSeparator(contextMenu); | |
89 populateContextMenuItems(*toHTMLMenuElement(nextElement), contex tMenu); | |
90 appendSeparator(contextMenu); | |
91 } else { | |
92 populateContextMenuItems(*toHTMLMenuElement(nextElement), subMen u); | |
93 contextMenu.appendItem(ContextMenuItem(SubmenuType, ContextMenuI temCustomTagNoAction, labelString, &subMenu)); | |
94 } | |
95 nextElement = Traversal<HTMLElement>::nextSibling(*nextElement); | |
96 } else if (isHTMLMenuItemElement(*nextElement)) { | |
97 appendMenuItem(toHTMLMenuItemElement(nextElement), contextMenu); | |
98 if (ContextMenuItemBaseCustomTag + m_menuItems.size() >= ContextMenu ItemLastCustomTag) | |
99 break; | |
100 nextElement = Traversal<HTMLElement>::next(*nextElement, &menu); | |
101 } else { | |
102 nextElement = Traversal<HTMLElement>::next(*nextElement, &menu); | |
103 } | |
104 } | |
105 | |
106 // Remove separators at the end of the menu and any submenus. | |
107 while (contextMenu.items().last().type() == SeparatorType) | |
tkent
2014/08/25 23:07:42
Need a test in a case of <menu type=popup> without
pals
2014/08/26 07:13:23
Done.
| |
108 contextMenu.removeLastItem(); | |
109 } | |
110 | |
111 HTMLElement* CustomContextMenuProvider::menuItemAt(unsigned menuId) | |
112 { | |
113 int itemIndex = menuId - ContextMenuItemBaseCustomTag; | |
114 if (itemIndex < 0 || static_cast<unsigned long>(itemIndex) >= m_menuItems.si ze()) | |
115 return 0; | |
116 return m_menuItems[itemIndex].get(); | |
117 } | |
118 | |
119 } // namespace blink | |
OLD | NEW |