Chromium Code Reviews| 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 |