2 * Copyright (C) 2006, 2007 Apple Inc. All rights reserved.
3 * Copyright (C) 2010 Igalia S.L
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
15 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
18 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 #include "ContextMenuController.h"
30 #if ENABLE(CONTEXT_MENUS)
32 #include "BackForwardController.h"
34 #include "ContextMenu.h"
35 #include "ContextMenuClient.h"
36 #include "ContextMenuItem.h"
37 #include "ContextMenuProvider.h"
39 #include "DocumentFragment.h"
40 #include "DocumentLoader.h"
42 #include "EditorClient.h"
44 #include "EventHandler.h"
45 #include "EventNames.h"
46 #include "FormState.h"
48 #include "FrameLoadRequest.h"
49 #include "FrameLoader.h"
50 #include "FrameLoaderClient.h"
51 #include "FrameSelection.h"
52 #include "HTMLFormElement.h"
53 #include "HitTestRequest.h"
54 #include "HitTestResult.h"
55 #include "InspectorController.h"
56 #include "LocalizedStrings.h"
57 #include "MouseEvent.h"
58 #include "NavigationAction.h"
61 #include "PlatformEvent.h"
62 #include "RenderObject.h"
63 #include "ReplaceSelectionCommand.h"
64 #include "ResourceRequest.h"
66 #include "TextIterator.h"
67 #include "TypingCommand.h"
68 #include "UserTypingGestureIndicator.h"
69 #include "WindowFeatures.h"
71 #include <wtf/unicode/Unicode.h>
74 #include <wtf/gobject/GOwnPtr.h>
78 using namespace Unicode;
82 ContextMenuController::ContextMenuController(Page* page, ContextMenuClient* client)
86 ASSERT_ARG(page, page);
87 ASSERT_ARG(client, client);
90 ContextMenuController::~ContextMenuController()
92 #if ENABLE(TIZEN_CONTEXT_MENU)
93 //this fixes a bug which occured when WebKit was shut down if a context menu was active
94 //then ContextMenuController destructor was calling:
95 //- first m_client->contextMenuDestroyed which destroyed the m_client object explicitly and then
96 //- and then m_contextMenu implicitily
97 //m_contextMenu destructor was then trying to call freePlatformDescription on m_contextMenuClient
98 //which was already destroyed
99 if (contextMenu() && contextMenu()->platformDescription()) {
100 m_client->freePlatformDescription(contextMenu()->platformDescription());
101 contextMenu()->setPlatformDescription(0);
104 m_client->contextMenuDestroyed();
107 PassOwnPtr<ContextMenuController> ContextMenuController::create(Page* page, ContextMenuClient* client)
109 return adoptPtr(new ContextMenuController(page, client));
112 void ContextMenuController::clearContextMenu()
114 m_contextMenu.clear();
116 m_menuProvider->contextMenuCleared();
120 void ContextMenuController::handleContextMenuEvent(Event* event)
122 m_contextMenu = createContextMenu(event);
128 showContextMenu(event);
131 static PassOwnPtr<ContextMenuItem> separatorItem()
133 return adoptPtr(new ContextMenuItem(SeparatorType, ContextMenuItemTagNoAction, String()));
136 void ContextMenuController::showContextMenu(Event* event, PassRefPtr<ContextMenuProvider> menuProvider)
138 m_menuProvider = menuProvider;
140 m_contextMenu = createContextMenu(event);
141 if (!m_contextMenu) {
146 m_menuProvider->populateContextMenu(m_contextMenu.get());
147 if (m_hitTestResult.isSelected()) {
148 appendItem(*separatorItem(), m_contextMenu.get());
151 showContextMenu(event);
154 PassOwnPtr<ContextMenu> ContextMenuController::createContextMenu(Event* event)
156 if (!event->isMouseEvent())
159 MouseEvent* mouseEvent = static_cast<MouseEvent*>(event);
160 HitTestResult result(mouseEvent->absoluteLocation());
162 if (Frame* frame = event->target()->toNode()->document()->frame())
163 result = frame->eventHandler()->hitTestResultAtPoint(mouseEvent->absoluteLocation(), false);
165 if (!result.innerNonSharedNode())
168 m_hitTestResult = result;
170 #if ENABLE(TIZEN_CONTEXT_MENU)
171 return adoptPtr(new ContextMenu(result));
173 return adoptPtr(new ContextMenu);
177 void ContextMenuController::showContextMenu(Event* event)
179 #if ENABLE(INSPECTOR) && !ENABLE(TIZEN_CONTEXT_MENU)
180 if (m_page->inspectorController()->enabled())
181 addInspectElementItem();
184 #if USE(CROSS_PLATFORM_CONTEXT_MENUS)
185 m_contextMenu = m_client->customizeMenu(m_contextMenu.release());
187 PlatformMenuDescription customMenu = m_client->getCustomMenuFromDefaultItems(m_contextMenu.get());
188 m_contextMenu->setPlatformDescription(customMenu);
190 event->setDefaultHandled();
193 static void openNewWindow(const KURL& urlToLoad, Frame* frame)
195 if (Page* oldPage = frame->page()) {
196 FrameLoadRequest request(frame->document()->securityOrigin(), ResourceRequest(urlToLoad, frame->loader()->outgoingReferrer()));
197 if (Page* newPage = oldPage->chrome()->createWindow(frame, request, WindowFeatures(), NavigationAction(request.resourceRequest()))) {
198 newPage->mainFrame()->loader()->loadFrameRequest(request, false, false, 0, 0, MaybeSendReferrer);
199 newPage->chrome()->show();
205 static void insertUnicodeCharacter(UChar character, Frame* frame)
207 String text(&character, 1);
208 if (!frame->editor()->shouldInsertText(text, frame->selection()->toNormalizedRange().get(), EditorInsertActionTyped))
211 TypingCommand::insertText(frame->document(), text, 0, TypingCommand::TextCompositionNone);
215 void ContextMenuController::contextMenuItemSelected(ContextMenuItem* item)
217 ASSERT(item->type() == ActionType || item->type() == CheckableActionType);
219 if (item->action() >= ContextMenuItemBaseApplicationTag) {
220 m_client->contextMenuItemSelected(item, m_contextMenu.get());
224 if (item->action() >= ContextMenuItemBaseCustomTag) {
225 ASSERT(m_menuProvider);
226 m_menuProvider->contextMenuItemSelected(item);
230 Frame* frame = m_hitTestResult.innerNonSharedNode()->document()->frame();
234 #if ENABLE(TIZEN_WEBKIT_PASTEBOARD)
235 if (Page* page = frame->page()) {
236 if (item->action() == ContextMenuItemTagCopyLinkToClipboard
237 || item->action() == ContextMenuItemTagCopyImageToClipboard
238 || item->action() == ContextMenuItemTagCopyImageUrlToClipboard
239 || item->action() == ContextMenuItemTagCopyMediaLinkToClipboard
240 || item->action() == ContextMenuItemTagCopy)
241 Pasteboard::generalPasteboard()->setPage(page);
245 switch (item->action()) {
246 case ContextMenuItemTagOpenLinkInNewWindow:
247 openNewWindow(m_hitTestResult.absoluteLinkURL(), frame);
249 case ContextMenuItemTagDownloadLinkToDisk:
250 // FIXME: Some day we should be able to do this from within WebCore.
251 m_client->downloadURL(m_hitTestResult.absoluteLinkURL());
253 case ContextMenuItemTagCopyLinkToClipboard:
254 frame->editor()->copyURL(m_hitTestResult.absoluteLinkURL(), m_hitTestResult.textContent());
256 case ContextMenuItemTagOpenImageInNewWindow:
257 openNewWindow(m_hitTestResult.absoluteImageURL(), frame);
259 case ContextMenuItemTagDownloadImageToDisk:
260 // FIXME: Some day we should be able to do this from within WebCore.
261 #if ENABLE(TIZEN_INSTALL_CONTENT)
262 m_client->saveAsImage(m_contextMenu.get(), frame, m_hitTestResult.absoluteImageURL());
264 m_client->downloadURL(m_hitTestResult.absoluteImageURL());
267 #if ENABLE(TIZEN_INSTALL_CONTENT)
268 case ContextMenuItemTagSendImageViaEmail:
269 m_client->sendImageViaEmail(m_contextMenu.get(), frame, m_hitTestResult.absoluteImageURL());
271 case ContextMenuItemTagSendImageViaMessage:
272 m_client->sendImageViaMessage(m_contextMenu.get(), frame, m_hitTestResult.absoluteImageURL());
275 case ContextMenuItemTagCopyImageToClipboard:
276 // FIXME: The Pasteboard class is not written yet
277 // For now, call into the client. This is temporary!
278 frame->editor()->copyImage(m_hitTestResult);
280 #if PLATFORM(QT) || PLATFORM(GTK) || PLATFORM(EFL)
281 case ContextMenuItemTagCopyImageUrlToClipboard:
282 frame->editor()->copyURL(m_hitTestResult.absoluteImageURL(), m_hitTestResult.textContent());
285 case ContextMenuItemTagOpenMediaInNewWindow:
286 openNewWindow(m_hitTestResult.absoluteMediaURL(), frame);
288 case ContextMenuItemTagCopyMediaLinkToClipboard:
289 frame->editor()->copyURL(m_hitTestResult.absoluteMediaURL(), m_hitTestResult.textContent());
291 case ContextMenuItemTagToggleMediaControls:
292 m_hitTestResult.toggleMediaControlsDisplay();
294 case ContextMenuItemTagToggleMediaLoop:
295 m_hitTestResult.toggleMediaLoopPlayback();
297 case ContextMenuItemTagEnterVideoFullscreen:
298 m_hitTestResult.enterFullscreenForVideo();
300 case ContextMenuItemTagMediaPlayPause:
301 m_hitTestResult.toggleMediaPlayState();
303 case ContextMenuItemTagMediaMute:
304 m_hitTestResult.toggleMediaMuteState();
306 case ContextMenuItemTagOpenFrameInNewWindow: {
307 DocumentLoader* loader = frame->loader()->documentLoader();
308 if (!loader->unreachableURL().isEmpty())
309 openNewWindow(loader->unreachableURL(), frame);
311 openNewWindow(loader->url(), frame);
314 case ContextMenuItemTagCopy:
315 frame->editor()->copy();
317 case ContextMenuItemTagGoBack:
318 if (Page* page = frame->page())
319 page->backForward()->goBackOrForward(-1);
321 case ContextMenuItemTagGoForward:
322 if (Page* page = frame->page())
323 page->backForward()->goBackOrForward(1);
325 case ContextMenuItemTagStop:
326 frame->loader()->stop();
328 case ContextMenuItemTagReload:
329 frame->loader()->reload();
331 case ContextMenuItemTagCut:
332 frame->editor()->command("Cut").execute();
334 case ContextMenuItemTagPaste:
335 frame->editor()->command("Paste").execute();
338 case ContextMenuItemTagDelete:
339 frame->editor()->performDelete();
341 case ContextMenuItemTagUnicodeInsertLRMMark:
342 insertUnicodeCharacter(leftToRightMark, frame);
344 case ContextMenuItemTagUnicodeInsertRLMMark:
345 insertUnicodeCharacter(rightToLeftMark, frame);
347 case ContextMenuItemTagUnicodeInsertLREMark:
348 insertUnicodeCharacter(leftToRightEmbed, frame);
350 case ContextMenuItemTagUnicodeInsertRLEMark:
351 insertUnicodeCharacter(rightToLeftEmbed, frame);
353 case ContextMenuItemTagUnicodeInsertLROMark:
354 insertUnicodeCharacter(leftToRightOverride, frame);
356 case ContextMenuItemTagUnicodeInsertRLOMark:
357 insertUnicodeCharacter(rightToLeftOverride, frame);
359 case ContextMenuItemTagUnicodeInsertPDFMark:
360 insertUnicodeCharacter(popDirectionalFormatting, frame);
362 case ContextMenuItemTagUnicodeInsertZWSMark:
363 insertUnicodeCharacter(zeroWidthSpace, frame);
365 case ContextMenuItemTagUnicodeInsertZWJMark:
366 insertUnicodeCharacter(zeroWidthJoiner, frame);
368 case ContextMenuItemTagUnicodeInsertZWNJMark:
369 insertUnicodeCharacter(zeroWidthNonJoiner, frame);
372 #if PLATFORM(GTK) || PLATFORM(QT) || PLATFORM(EFL)
373 case ContextMenuItemTagSelectAll:
374 frame->editor()->command("SelectAll").execute();
377 #if ENABLE(TIZEN_CONTEXT_MENU_SELECT)
378 case ContextMenuItemTagSelectWord:
379 frame->editor()->command("SelectWord").execute();
382 case ContextMenuItemTagSpellingGuess:
383 ASSERT(frame->editor()->selectedText().length());
384 if (frame->editor()->shouldInsertText(item->title(), frame->selection()->toNormalizedRange().get(), EditorInsertActionPasted)) {
385 Document* document = frame->document();
386 RefPtr<ReplaceSelectionCommand> command = ReplaceSelectionCommand::create(document, createFragmentFromMarkup(document, item->title(), ""), ReplaceSelectionCommand::SelectReplacement | ReplaceSelectionCommand::MatchStyle | ReplaceSelectionCommand::PreventNesting);
387 applyCommand(command);
388 frame->selection()->revealSelection(ScrollAlignment::alignToEdgeIfNeeded);
391 case ContextMenuItemTagIgnoreSpelling:
392 frame->editor()->ignoreSpelling();
394 case ContextMenuItemTagLearnSpelling:
395 frame->editor()->learnSpelling();
397 case ContextMenuItemTagSearchWeb:
398 m_client->searchWithGoogle(frame);
400 case ContextMenuItemTagLookUpInDictionary:
401 // FIXME: Some day we may be able to do this from within WebCore.
402 m_client->lookUpInDictionary(frame);
404 case ContextMenuItemTagOpenLink:
405 if (Frame* targetFrame = m_hitTestResult.targetFrame())
406 targetFrame->loader()->loadFrameRequest(FrameLoadRequest(frame->document()->securityOrigin(), ResourceRequest(m_hitTestResult.absoluteLinkURL(), frame->loader()->outgoingReferrer())), false, false, 0, 0, MaybeSendReferrer);
408 openNewWindow(m_hitTestResult.absoluteLinkURL(), frame);
410 case ContextMenuItemTagBold:
411 frame->editor()->command("ToggleBold").execute();
413 case ContextMenuItemTagItalic:
414 frame->editor()->command("ToggleItalic").execute();
416 case ContextMenuItemTagUnderline:
417 frame->editor()->toggleUnderline();
419 case ContextMenuItemTagOutline:
420 // We actually never enable this because CSS does not have a way to specify an outline font,
421 // which may make this difficult to implement. Maybe a special case of text-shadow?
423 case ContextMenuItemTagStartSpeaking: {
425 RefPtr<Range> selectedRange = frame->selection()->toNormalizedRange();
426 if (!selectedRange || selectedRange->collapsed(ec)) {
427 Document* document = m_hitTestResult.innerNonSharedNode()->document();
428 selectedRange = document->createRange();
429 selectedRange->selectNode(document->documentElement(), ec);
431 m_client->speak(plainText(selectedRange.get()));
434 case ContextMenuItemTagStopSpeaking:
435 m_client->stopSpeaking();
437 case ContextMenuItemTagDefaultDirection:
438 frame->editor()->setBaseWritingDirection(NaturalWritingDirection);
440 case ContextMenuItemTagLeftToRight:
441 frame->editor()->setBaseWritingDirection(LeftToRightWritingDirection);
443 case ContextMenuItemTagRightToLeft:
444 frame->editor()->setBaseWritingDirection(RightToLeftWritingDirection);
446 case ContextMenuItemTagTextDirectionDefault:
447 frame->editor()->command("MakeTextWritingDirectionNatural").execute();
449 case ContextMenuItemTagTextDirectionLeftToRight:
450 frame->editor()->command("MakeTextWritingDirectionLeftToRight").execute();
452 case ContextMenuItemTagTextDirectionRightToLeft:
453 frame->editor()->command("MakeTextWritingDirectionRightToLeft").execute();
456 case ContextMenuItemTagSearchInSpotlight:
457 m_client->searchWithSpotlight();
460 case ContextMenuItemTagShowSpellingPanel:
461 frame->editor()->showSpellingGuessPanel();
463 case ContextMenuItemTagCheckSpelling:
464 frame->editor()->advanceToNextMisspelling();
466 case ContextMenuItemTagCheckSpellingWhileTyping:
467 frame->editor()->toggleContinuousSpellChecking();
469 case ContextMenuItemTagCheckGrammarWithSpelling:
470 frame->editor()->toggleGrammarChecking();
473 case ContextMenuItemTagShowFonts:
474 frame->editor()->showFontPanel();
476 case ContextMenuItemTagStyles:
477 frame->editor()->showStylesPanel();
479 case ContextMenuItemTagShowColors:
480 frame->editor()->showColorPanel();
483 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
484 case ContextMenuItemTagMakeUpperCase:
485 frame->editor()->uppercaseWord();
487 case ContextMenuItemTagMakeLowerCase:
488 frame->editor()->lowercaseWord();
490 case ContextMenuItemTagCapitalize:
491 frame->editor()->capitalizeWord();
493 case ContextMenuItemTagShowSubstitutions:
494 frame->editor()->showSubstitutionsPanel();
496 case ContextMenuItemTagSmartCopyPaste:
497 frame->editor()->toggleSmartInsertDelete();
499 case ContextMenuItemTagSmartQuotes:
500 frame->editor()->toggleAutomaticQuoteSubstitution();
502 case ContextMenuItemTagSmartDashes:
503 frame->editor()->toggleAutomaticDashSubstitution();
505 case ContextMenuItemTagSmartLinks:
506 frame->editor()->toggleAutomaticLinkDetection();
508 case ContextMenuItemTagTextReplacement:
509 frame->editor()->toggleAutomaticTextReplacement();
511 case ContextMenuItemTagCorrectSpellingAutomatically:
512 frame->editor()->toggleAutomaticSpellingCorrection();
514 case ContextMenuItemTagChangeBack:
515 frame->editor()->changeBackToReplacedString(m_hitTestResult.replacedString());
518 #if ENABLE(INSPECTOR)
519 case ContextMenuItemTagInspectElement:
520 if (Page* page = frame->page())
521 page->inspectorController()->inspect(m_hitTestResult.innerNonSharedNode());
529 void ContextMenuController::appendItem(ContextMenuItem& menuItem, ContextMenu* parentMenu)
531 checkOrEnableIfNeeded(menuItem);
533 parentMenu->appendItem(menuItem);
536 void ContextMenuController::createAndAppendFontSubMenu(ContextMenuItem& fontMenuItem)
538 ContextMenu fontMenu;
541 ContextMenuItem showFonts(ActionType, ContextMenuItemTagShowFonts, contextMenuItemTagShowFonts());
543 ContextMenuItem bold(CheckableActionType, ContextMenuItemTagBold, contextMenuItemTagBold());
544 ContextMenuItem italic(CheckableActionType, ContextMenuItemTagItalic, contextMenuItemTagItalic());
545 ContextMenuItem underline(CheckableActionType, ContextMenuItemTagUnderline, contextMenuItemTagUnderline());
546 ContextMenuItem outline(ActionType, ContextMenuItemTagOutline, contextMenuItemTagOutline());
548 ContextMenuItem styles(ActionType, ContextMenuItemTagStyles, contextMenuItemTagStyles());
549 ContextMenuItem showColors(ActionType, ContextMenuItemTagShowColors, contextMenuItemTagShowColors());
553 appendItem(showFonts, &fontMenu);
555 appendItem(bold, &fontMenu);
556 appendItem(italic, &fontMenu);
557 appendItem(underline, &fontMenu);
558 appendItem(outline, &fontMenu);
560 appendItem(styles, &fontMenu);
561 appendItem(*separatorItem(), &fontMenu);
562 appendItem(showColors, &fontMenu);
565 fontMenuItem.setSubMenu(&fontMenu);
571 void ContextMenuController::createAndAppendSpellingAndGrammarSubMenu(ContextMenuItem& spellingAndGrammarMenuItem)
573 ContextMenu spellingAndGrammarMenu;
575 ContextMenuItem showSpellingPanel(ActionType, ContextMenuItemTagShowSpellingPanel,
576 contextMenuItemTagShowSpellingPanel(true));
577 ContextMenuItem checkSpelling(ActionType, ContextMenuItemTagCheckSpelling,
578 contextMenuItemTagCheckSpelling());
579 ContextMenuItem checkAsYouType(CheckableActionType, ContextMenuItemTagCheckSpellingWhileTyping,
580 contextMenuItemTagCheckSpellingWhileTyping());
581 ContextMenuItem grammarWithSpelling(CheckableActionType, ContextMenuItemTagCheckGrammarWithSpelling,
582 contextMenuItemTagCheckGrammarWithSpelling());
583 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
584 ContextMenuItem correctSpelling(CheckableActionType, ContextMenuItemTagCorrectSpellingAutomatically,
585 contextMenuItemTagCorrectSpellingAutomatically());
588 appendItem(showSpellingPanel, &spellingAndGrammarMenu);
589 appendItem(checkSpelling, &spellingAndGrammarMenu);
590 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
591 appendItem(*separatorItem(), &spellingAndGrammarMenu);
593 appendItem(checkAsYouType, &spellingAndGrammarMenu);
594 appendItem(grammarWithSpelling, &spellingAndGrammarMenu);
595 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
596 appendItem(correctSpelling, &spellingAndGrammarMenu);
599 spellingAndGrammarMenuItem.setSubMenu(&spellingAndGrammarMenu);
602 #endif // !PLATFORM(GTK)
607 void ContextMenuController::createAndAppendSpeechSubMenu(ContextMenuItem& speechMenuItem)
609 ContextMenu speechMenu;
611 ContextMenuItem start(ActionType, ContextMenuItemTagStartSpeaking, contextMenuItemTagStartSpeaking());
612 ContextMenuItem stop(ActionType, ContextMenuItemTagStopSpeaking, contextMenuItemTagStopSpeaking());
614 appendItem(start, &speechMenu);
615 appendItem(stop, &speechMenu);
617 speechMenuItem.setSubMenu(&speechMenu);
624 void ContextMenuController::createAndAppendUnicodeSubMenu(ContextMenuItem& unicodeMenuItem)
626 ContextMenu unicodeMenu;
628 ContextMenuItem leftToRightMarkMenuItem(ActionType, ContextMenuItemTagUnicodeInsertLRMMark, contextMenuItemTagUnicodeInsertLRMMark());
629 ContextMenuItem rightToLeftMarkMenuItem(ActionType, ContextMenuItemTagUnicodeInsertRLMMark, contextMenuItemTagUnicodeInsertRLMMark());
630 ContextMenuItem leftToRightEmbedMenuItem(ActionType, ContextMenuItemTagUnicodeInsertLREMark, contextMenuItemTagUnicodeInsertLREMark());
631 ContextMenuItem rightToLeftEmbedMenuItem(ActionType, ContextMenuItemTagUnicodeInsertRLEMark, contextMenuItemTagUnicodeInsertRLEMark());
632 ContextMenuItem leftToRightOverrideMenuItem(ActionType, ContextMenuItemTagUnicodeInsertLROMark, contextMenuItemTagUnicodeInsertLROMark());
633 ContextMenuItem rightToLeftOverrideMenuItem(ActionType, ContextMenuItemTagUnicodeInsertRLOMark, contextMenuItemTagUnicodeInsertRLOMark());
634 ContextMenuItem popDirectionalFormattingMenuItem(ActionType, ContextMenuItemTagUnicodeInsertPDFMark, contextMenuItemTagUnicodeInsertPDFMark());
635 ContextMenuItem zeroWidthSpaceMenuItem(ActionType, ContextMenuItemTagUnicodeInsertZWSMark, contextMenuItemTagUnicodeInsertZWSMark());
636 ContextMenuItem zeroWidthJoinerMenuItem(ActionType, ContextMenuItemTagUnicodeInsertZWJMark, contextMenuItemTagUnicodeInsertZWJMark());
637 ContextMenuItem zeroWidthNonJoinerMenuItem(ActionType, ContextMenuItemTagUnicodeInsertZWNJMark, contextMenuItemTagUnicodeInsertZWNJMark());
639 appendItem(leftToRightMarkMenuItem, &unicodeMenu);
640 appendItem(rightToLeftMarkMenuItem, &unicodeMenu);
641 appendItem(leftToRightEmbedMenuItem, &unicodeMenu);
642 appendItem(rightToLeftEmbedMenuItem, &unicodeMenu);
643 appendItem(leftToRightOverrideMenuItem, &unicodeMenu);
644 appendItem(rightToLeftOverrideMenuItem, &unicodeMenu);
645 appendItem(popDirectionalFormattingMenuItem, &unicodeMenu);
646 appendItem(zeroWidthSpaceMenuItem, &unicodeMenu);
647 appendItem(zeroWidthJoinerMenuItem, &unicodeMenu);
648 appendItem(zeroWidthNonJoinerMenuItem, &unicodeMenu);
650 unicodeMenuItem.setSubMenu(&unicodeMenu);
655 void ContextMenuController::createAndAppendWritingDirectionSubMenu(ContextMenuItem& writingDirectionMenuItem)
657 ContextMenu writingDirectionMenu;
659 ContextMenuItem defaultItem(ActionType, ContextMenuItemTagDefaultDirection,
660 contextMenuItemTagDefaultDirection());
661 ContextMenuItem ltr(CheckableActionType, ContextMenuItemTagLeftToRight, contextMenuItemTagLeftToRight());
662 ContextMenuItem rtl(CheckableActionType, ContextMenuItemTagRightToLeft, contextMenuItemTagRightToLeft());
664 appendItem(defaultItem, &writingDirectionMenu);
665 appendItem(ltr, &writingDirectionMenu);
666 appendItem(rtl, &writingDirectionMenu);
668 writingDirectionMenuItem.setSubMenu(&writingDirectionMenu);
671 void ContextMenuController::createAndAppendTextDirectionSubMenu(ContextMenuItem& textDirectionMenuItem)
673 ContextMenu textDirectionMenu;
675 ContextMenuItem defaultItem(ActionType, ContextMenuItemTagTextDirectionDefault, contextMenuItemTagDefaultDirection());
676 ContextMenuItem ltr(CheckableActionType, ContextMenuItemTagTextDirectionLeftToRight, contextMenuItemTagLeftToRight());
677 ContextMenuItem rtl(CheckableActionType, ContextMenuItemTagTextDirectionRightToLeft, contextMenuItemTagRightToLeft());
679 appendItem(defaultItem, &textDirectionMenu);
680 appendItem(ltr, &textDirectionMenu);
681 appendItem(rtl, &textDirectionMenu);
683 textDirectionMenuItem.setSubMenu(&textDirectionMenu);
688 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
690 void ContextMenuController::createAndAppendSubstitutionsSubMenu(ContextMenuItem& substitutionsMenuItem)
692 ContextMenu substitutionsMenu;
694 ContextMenuItem showSubstitutions(ActionType, ContextMenuItemTagShowSubstitutions, contextMenuItemTagShowSubstitutions(true));
695 ContextMenuItem smartCopyPaste(CheckableActionType, ContextMenuItemTagSmartCopyPaste, contextMenuItemTagSmartCopyPaste());
696 ContextMenuItem smartQuotes(CheckableActionType, ContextMenuItemTagSmartQuotes, contextMenuItemTagSmartQuotes());
697 ContextMenuItem smartDashes(CheckableActionType, ContextMenuItemTagSmartDashes, contextMenuItemTagSmartDashes());
698 ContextMenuItem smartLinks(CheckableActionType, ContextMenuItemTagSmartLinks, contextMenuItemTagSmartLinks());
699 ContextMenuItem textReplacement(CheckableActionType, ContextMenuItemTagTextReplacement, contextMenuItemTagTextReplacement());
701 appendItem(showSubstitutions, &substitutionsMenu);
702 appendItem(*separatorItem(), &substitutionsMenu);
703 appendItem(smartCopyPaste, &substitutionsMenu);
704 appendItem(smartQuotes, &substitutionsMenu);
705 appendItem(smartDashes, &substitutionsMenu);
706 appendItem(smartLinks, &substitutionsMenu);
707 appendItem(textReplacement, &substitutionsMenu);
709 substitutionsMenuItem.setSubMenu(&substitutionsMenu);
712 void ContextMenuController::createAndAppendTransformationsSubMenu(ContextMenuItem& transformationsMenuItem)
714 ContextMenu transformationsMenu;
716 ContextMenuItem makeUpperCase(ActionType, ContextMenuItemTagMakeUpperCase, contextMenuItemTagMakeUpperCase());
717 ContextMenuItem makeLowerCase(ActionType, ContextMenuItemTagMakeLowerCase, contextMenuItemTagMakeLowerCase());
718 ContextMenuItem capitalize(ActionType, ContextMenuItemTagCapitalize, contextMenuItemTagCapitalize());
720 appendItem(makeUpperCase, &transformationsMenu);
721 appendItem(makeLowerCase, &transformationsMenu);
722 appendItem(capitalize, &transformationsMenu);
724 transformationsMenuItem.setSubMenu(&transformationsMenu);
729 static bool selectionContainsPossibleWord(Frame* frame)
731 // Current algorithm: look for a character that's not just a separator.
732 for (TextIterator it(frame->selection()->toNormalizedRange().get()); !it.atEnd(); it.advance()) {
733 int length = it.length();
734 const UChar* characters = it.characters();
735 for (int i = 0; i < length; ++i)
736 if (!(category(characters[i]) & (Separator_Space | Separator_Line | Separator_Paragraph)))
743 #if defined(BUILDING_ON_LEOPARD) || defined(BUILDING_ON_SNOW_LEOPARD)
744 #define INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM 1
746 #define INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM 0
750 void ContextMenuController::populate()
752 ContextMenuItem OpenLinkItem(ActionType, ContextMenuItemTagOpenLink, contextMenuItemTagOpenLink());
753 ContextMenuItem OpenLinkInNewWindowItem(ActionType, ContextMenuItemTagOpenLinkInNewWindow,
754 contextMenuItemTagOpenLinkInNewWindow());
755 ContextMenuItem DownloadFileItem(ActionType, ContextMenuItemTagDownloadLinkToDisk,
756 contextMenuItemTagDownloadLinkToDisk());
757 ContextMenuItem CopyLinkItem(ActionType, ContextMenuItemTagCopyLinkToClipboard,
758 contextMenuItemTagCopyLinkToClipboard());
759 ContextMenuItem OpenImageInNewWindowItem(ActionType, ContextMenuItemTagOpenImageInNewWindow,
760 contextMenuItemTagOpenImageInNewWindow());
761 ContextMenuItem DownloadImageItem(ActionType, ContextMenuItemTagDownloadImageToDisk,
762 contextMenuItemTagDownloadImageToDisk());
763 ContextMenuItem CopyImageItem(ActionType, ContextMenuItemTagCopyImageToClipboard,
764 contextMenuItemTagCopyImageToClipboard());
765 #if PLATFORM(QT) || PLATFORM(GTK) || PLATFORM(EFL)
766 ContextMenuItem CopyImageUrlItem(ActionType, ContextMenuItemTagCopyImageUrlToClipboard,
767 contextMenuItemTagCopyImageUrlToClipboard());
769 ContextMenuItem OpenMediaInNewWindowItem(ActionType, ContextMenuItemTagOpenMediaInNewWindow, String());
770 ContextMenuItem CopyMediaLinkItem(ActionType, ContextMenuItemTagCopyMediaLinkToClipboard,
772 ContextMenuItem MediaPlayPause(ActionType, ContextMenuItemTagMediaPlayPause,
773 contextMenuItemTagMediaPlay());
774 ContextMenuItem MediaMute(ActionType, ContextMenuItemTagMediaMute,
775 contextMenuItemTagMediaMute());
776 ContextMenuItem ToggleMediaControls(CheckableActionType, ContextMenuItemTagToggleMediaControls,
777 contextMenuItemTagToggleMediaControls());
778 ContextMenuItem ToggleMediaLoop(CheckableActionType, ContextMenuItemTagToggleMediaLoop,
779 contextMenuItemTagToggleMediaLoop());
780 ContextMenuItem EnterVideoFullscreen(ActionType, ContextMenuItemTagEnterVideoFullscreen,
781 contextMenuItemTagEnterVideoFullscreen());
783 ContextMenuItem SearchSpotlightItem(ActionType, ContextMenuItemTagSearchInSpotlight,
784 contextMenuItemTagSearchInSpotlight());
787 ContextMenuItem SearchWebItem(ActionType, ContextMenuItemTagSearchWeb, contextMenuItemTagSearchWeb());
789 ContextMenuItem CopyItem(ActionType, ContextMenuItemTagCopy, contextMenuItemTagCopy());
790 ContextMenuItem BackItem(ActionType, ContextMenuItemTagGoBack, contextMenuItemTagGoBack());
791 ContextMenuItem ForwardItem(ActionType, ContextMenuItemTagGoForward, contextMenuItemTagGoForward());
792 ContextMenuItem StopItem(ActionType, ContextMenuItemTagStop, contextMenuItemTagStop());
793 ContextMenuItem ReloadItem(ActionType, ContextMenuItemTagReload, contextMenuItemTagReload());
794 ContextMenuItem OpenFrameItem(ActionType, ContextMenuItemTagOpenFrameInNewWindow,
795 contextMenuItemTagOpenFrameInNewWindow());
796 ContextMenuItem NoGuessesItem(ActionType, ContextMenuItemTagNoGuessesFound,
797 contextMenuItemTagNoGuessesFound());
798 ContextMenuItem IgnoreSpellingItem(ActionType, ContextMenuItemTagIgnoreSpelling,
799 contextMenuItemTagIgnoreSpelling());
800 ContextMenuItem LearnSpellingItem(ActionType, ContextMenuItemTagLearnSpelling,
801 contextMenuItemTagLearnSpelling());
802 ContextMenuItem IgnoreGrammarItem(ActionType, ContextMenuItemTagIgnoreGrammar,
803 contextMenuItemTagIgnoreGrammar());
804 ContextMenuItem CutItem(ActionType, ContextMenuItemTagCut, contextMenuItemTagCut());
805 ContextMenuItem PasteItem(ActionType, ContextMenuItemTagPaste, contextMenuItemTagPaste());
807 ContextMenuItem DeleteItem(ActionType, ContextMenuItemTagDelete, contextMenuItemTagDelete());
809 #if PLATFORM(GTK) || PLATFORM(QT) || PLATFORM(EFL)
810 ContextMenuItem SelectAllItem(ActionType, ContextMenuItemTagSelectAll, contextMenuItemTagSelectAll());
812 #if ENABLE(TIZEN_CONTEXT_MENU_SELECT)
813 ContextMenuItem SelectWordItem(ActionType, ContextMenuItemTagSelectWord, contextMenuItemTagSelectWord());
816 Node* node = m_hitTestResult.innerNonSharedNode();
820 if (!m_hitTestResult.isContentEditable() && (node->isElementNode() && static_cast<Element*>(node)->isFormControlElement()))
823 Frame* frame = node->document()->frame();
827 if (!m_hitTestResult.isContentEditable()) {
828 FrameLoader* loader = frame->loader();
829 KURL linkURL = m_hitTestResult.absoluteLinkURL();
830 if (!linkURL.isEmpty()) {
831 #if ENABLE(TIZEN_WEBKIT2_TEXT_SELECTION)
832 if (m_hitTestResult.isSelected()) {
833 if (selectionContainsPossibleWord(frame)) {
834 appendItem(SearchWebItem, m_contextMenu.get());
836 appendItem(CopyItem, m_contextMenu.get());
840 #if ENABLE(TIZEN_DOWNLOAD_LINK_FILTER)
841 if (loader->client()->canHandleRequest(ResourceRequest(linkURL)) &&
842 (linkURL.protocolIs("http") || linkURL.protocolIs("https") || linkURL.protocolIs("ftp") || linkURL.protocolIs("ftps"))) {
844 if (loader->client()->canHandleRequest(ResourceRequest(linkURL))) {
846 appendItem(OpenLinkItem, m_contextMenu.get());
847 appendItem(OpenLinkInNewWindowItem, m_contextMenu.get());
848 appendItem(DownloadFileItem, m_contextMenu.get());
851 if (m_hitTestResult.isSelected())
852 appendItem(CopyItem, m_contextMenu.get());
854 appendItem(CopyLinkItem, m_contextMenu.get());
857 KURL imageURL = m_hitTestResult.absoluteImageURL();
858 if (!imageURL.isEmpty()) {
859 if (!linkURL.isEmpty())
860 appendItem(*separatorItem(), m_contextMenu.get());
862 appendItem(OpenImageInNewWindowItem, m_contextMenu.get());
863 appendItem(DownloadImageItem, m_contextMenu.get());
864 if (imageURL.isLocalFile() || m_hitTestResult.image())
865 appendItem(CopyImageItem, m_contextMenu.get());
866 #if PLATFORM(QT) || PLATFORM(GTK) || PLATFORM(EFL)
867 appendItem(CopyImageUrlItem, m_contextMenu.get());
871 KURL mediaURL = m_hitTestResult.absoluteMediaURL();
872 if (!mediaURL.isEmpty()) {
873 if (!linkURL.isEmpty() || !imageURL.isEmpty())
874 appendItem(*separatorItem(), m_contextMenu.get());
876 appendItem(MediaPlayPause, m_contextMenu.get());
877 appendItem(MediaMute, m_contextMenu.get());
878 appendItem(ToggleMediaControls, m_contextMenu.get());
879 appendItem(ToggleMediaLoop, m_contextMenu.get());
880 appendItem(EnterVideoFullscreen, m_contextMenu.get());
882 appendItem(*separatorItem(), m_contextMenu.get());
883 appendItem(CopyMediaLinkItem, m_contextMenu.get());
884 appendItem(OpenMediaInNewWindowItem, m_contextMenu.get());
887 if (imageURL.isEmpty() && linkURL.isEmpty() && mediaURL.isEmpty()) {
888 if (m_hitTestResult.isSelected()) {
889 if (selectionContainsPossibleWord(frame)) {
891 String selectedString = frame->displayStringModifiedByEncoding(frame->editor()->selectedText());
892 ContextMenuItem LookUpInDictionaryItem(ActionType, ContextMenuItemTagLookUpInDictionary, contextMenuItemTagLookUpInDictionary(selectedString));
894 #if INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM
895 appendItem(SearchSpotlightItem, m_contextMenu.get());
897 appendItem(LookUpInDictionaryItem, m_contextMenu.get());
902 appendItem(SearchWebItem, m_contextMenu.get());
903 appendItem(*separatorItem(), m_contextMenu.get());
906 #if PLATFORM(MAC) && INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM
907 appendItem(LookUpInDictionaryItem, m_contextMenu.get());
908 appendItem(*separatorItem(), m_contextMenu.get());
912 appendItem(CopyItem, m_contextMenu.get());
914 appendItem(*separatorItem(), m_contextMenu.get());
916 ContextMenuItem SpeechMenuItem(SubmenuType, ContextMenuItemTagSpeechMenu, contextMenuItemTagSpeechMenu());
917 createAndAppendSpeechSubMenu(SpeechMenuItem);
918 appendItem(SpeechMenuItem, m_contextMenu.get());
921 #if ENABLE(INSPECTOR)
922 if (!(frame->page() && frame->page()->inspectorController()->hasInspectorFrontendClient())) {
925 // In GTK+ unavailable items are not hidden but insensitive
927 appendItem(BackItem, m_contextMenu.get());
928 appendItem(ForwardItem, m_contextMenu.get());
929 appendItem(StopItem, m_contextMenu.get());
930 appendItem(ReloadItem, m_contextMenu.get());
932 if (frame->page() && frame->page()->backForward()->canGoBackOrForward(-1))
933 appendItem(BackItem, m_contextMenu.get());
935 if (frame->page() && frame->page()->backForward()->canGoBackOrForward(1))
936 appendItem(ForwardItem, m_contextMenu.get());
938 // use isLoadingInAPISense rather than isLoading because Stop/Reload are
939 // intended to match WebKit's API, not WebCore's internal notion of loading status
940 if (loader->documentLoader()->isLoadingInAPISense())
941 appendItem(StopItem, m_contextMenu.get());
943 appendItem(ReloadItem, m_contextMenu.get());
945 #if ENABLE(INSPECTOR)
949 if (frame->page() && frame != frame->page()->mainFrame())
950 appendItem(OpenFrameItem, m_contextMenu.get());
953 } else { // Make an editing context menu
954 FrameSelection* selection = frame->selection();
955 bool inPasswordField = selection->isInPasswordField();
956 bool spellCheckingEnabled = frame->editor()->isSpellCheckingEnabledFor(node);
958 if (!inPasswordField && spellCheckingEnabled) {
959 // Consider adding spelling-related or grammar-related context menu items (never both, since a single selected range
960 // is never considered a misspelling and bad grammar at the same time)
963 Vector<String> guesses = frame->editor()->guessesForMisspelledOrUngrammaticalSelection(misspelling, badGrammar);
964 if (misspelling || badGrammar) {
965 size_t size = guesses.size();
967 // If there's bad grammar but no suggestions (e.g., repeated word), just leave off the suggestions
968 // list and trailing separator rather than adding a "No Guesses Found" item (matches AppKit)
970 appendItem(NoGuessesItem, m_contextMenu.get());
971 appendItem(*separatorItem(), m_contextMenu.get());
974 for (unsigned i = 0; i < size; i++) {
975 const String &guess = guesses[i];
976 if (!guess.isEmpty()) {
977 ContextMenuItem item(ActionType, ContextMenuItemTagSpellingGuess, guess);
978 appendItem(item, m_contextMenu.get());
981 appendItem(*separatorItem(), m_contextMenu.get());
985 appendItem(IgnoreSpellingItem, m_contextMenu.get());
986 appendItem(LearnSpellingItem, m_contextMenu.get());
988 appendItem(IgnoreGrammarItem, m_contextMenu.get());
989 appendItem(*separatorItem(), m_contextMenu.get());
990 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
992 // If the string was autocorrected, generate a contextual menu item allowing it to be changed back.
993 String replacedString = m_hitTestResult.replacedString();
994 if (!replacedString.isEmpty()) {
995 ContextMenuItem item(ActionType, ContextMenuItemTagChangeBack, contextMenuItemTagChangeBack(replacedString));
996 appendItem(item, m_contextMenu.get());
997 appendItem(*separatorItem(), m_contextMenu.get());
1003 FrameLoader* loader = frame->loader();
1004 KURL linkURL = m_hitTestResult.absoluteLinkURL();
1005 if (!linkURL.isEmpty()) {
1006 if (loader->client()->canHandleRequest(ResourceRequest(linkURL))) {
1007 appendItem(OpenLinkItem, m_contextMenu.get());
1008 appendItem(OpenLinkInNewWindowItem, m_contextMenu.get());
1009 appendItem(DownloadFileItem, m_contextMenu.get());
1011 appendItem(CopyLinkItem, m_contextMenu.get());
1012 appendItem(*separatorItem(), m_contextMenu.get());
1015 if (m_hitTestResult.isSelected() && !inPasswordField && selectionContainsPossibleWord(frame)) {
1017 String selectedString = frame->displayStringModifiedByEncoding(frame->editor()->selectedText());
1018 ContextMenuItem LookUpInDictionaryItem(ActionType, ContextMenuItemTagLookUpInDictionary, contextMenuItemTagLookUpInDictionary(selectedString));
1020 #if INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM
1021 appendItem(SearchSpotlightItem, m_contextMenu.get());
1023 appendItem(LookUpInDictionaryItem, m_contextMenu.get());
1028 appendItem(SearchWebItem, m_contextMenu.get());
1029 appendItem(*separatorItem(), m_contextMenu.get());
1032 #if PLATFORM(MAC) && INCLUDE_SPOTLIGHT_CONTEXT_MENU_ITEM
1033 appendItem(LookUpInDictionaryItem, m_contextMenu.get());
1034 appendItem(*separatorItem(), m_contextMenu.get());
1038 #if ENABLE(TIZEN_CONTEXT_MENU_WEBKIT_2)
1039 if (m_hitTestResult.isSelected()) {
1040 appendItem(CutItem, m_contextMenu.get());
1041 appendItem(CopyItem, m_contextMenu.get());
1044 appendItem(CutItem, m_contextMenu.get());
1045 appendItem(CopyItem, m_contextMenu.get());
1047 appendItem(PasteItem, m_contextMenu.get());
1049 appendItem(DeleteItem, m_contextMenu.get());
1050 appendItem(*separatorItem(), m_contextMenu.get());
1052 #if PLATFORM(GTK) || PLATFORM(QT) || PLATFORM(EFL)
1053 appendItem(SelectAllItem, m_contextMenu.get());
1055 #if ENABLE(TIZEN_CONTEXT_MENU_SELECT)
1056 appendItem(SelectWordItem, m_contextMenu.get());
1059 if (!inPasswordField) {
1061 #if !ENABLE(TIZEN_CONTEXT_MENU_TEMPORARY_FIX)
1062 appendItem(*separatorItem(), m_contextMenu.get());
1063 ContextMenuItem SpellingAndGrammarMenuItem(SubmenuType, ContextMenuItemTagSpellingMenu,
1064 contextMenuItemTagSpellingMenu());
1065 createAndAppendSpellingAndGrammarSubMenu(SpellingAndGrammarMenuItem);
1066 appendItem(SpellingAndGrammarMenuItem, m_contextMenu.get());
1067 #endif //TIZEN_CONTEXT_MENU_TEMPORARY_FIX
1069 #if PLATFORM(MAC) && !defined(BUILDING_ON_LEOPARD)
1070 ContextMenuItem substitutionsMenuItem(SubmenuType, ContextMenuItemTagSubstitutionsMenu,
1071 contextMenuItemTagSubstitutionsMenu());
1072 createAndAppendSubstitutionsSubMenu(substitutionsMenuItem);
1073 appendItem(substitutionsMenuItem, m_contextMenu.get());
1074 ContextMenuItem transformationsMenuItem(SubmenuType, ContextMenuItemTagTransformationsMenu,
1075 contextMenuItemTagTransformationsMenu());
1076 createAndAppendTransformationsSubMenu(transformationsMenuItem);
1077 appendItem(transformationsMenuItem, m_contextMenu.get());
1080 bool shouldShowFontMenu = frame->editor()->canEditRichly();
1082 #if ENABLE(TIZEN_CONTEXT_MENU_TEMPORARY_FIX)
1083 bool shouldShowFontMenu = false;
1085 bool shouldShowFontMenu = true;
1086 #endif //TIZEN_CONTEXT_MENU_TEMPORARY_FIX
1088 if (shouldShowFontMenu) {
1089 ContextMenuItem FontMenuItem(SubmenuType, ContextMenuItemTagFontMenu,
1090 contextMenuItemTagFontMenu());
1091 createAndAppendFontSubMenu(FontMenuItem);
1092 appendItem(FontMenuItem, m_contextMenu.get());
1095 ContextMenuItem SpeechMenuItem(SubmenuType, ContextMenuItemTagSpeechMenu, contextMenuItemTagSpeechMenu());
1096 createAndAppendSpeechSubMenu(SpeechMenuItem);
1097 appendItem(SpeechMenuItem, m_contextMenu.get());
1100 EditorClient* client = frame->editor()->client();
1101 if (client && client->shouldShowUnicodeMenu()) {
1102 ContextMenuItem UnicodeMenuItem(SubmenuType, ContextMenuItemTagUnicode, contextMenuItemTagUnicode());
1103 createAndAppendUnicodeSubMenu(UnicodeMenuItem);
1104 appendItem(*separatorItem(), m_contextMenu.get());
1105 appendItem(UnicodeMenuItem, m_contextMenu.get());
1108 ContextMenuItem WritingDirectionMenuItem(SubmenuType, ContextMenuItemTagWritingDirectionMenu,
1109 contextMenuItemTagWritingDirectionMenu());
1110 #if !ENABLE(TIZEN_CONTEXT_MENU_TEMPORARY_FIX)
1111 createAndAppendWritingDirectionSubMenu(WritingDirectionMenuItem);
1112 appendItem(WritingDirectionMenuItem, m_contextMenu.get());
1113 if (Page* page = frame->page()) {
1114 if (Settings* settings = page->settings()) {
1115 bool includeTextDirectionSubmenu = settings->textDirectionSubmenuInclusionBehavior() == TextDirectionSubmenuAlwaysIncluded
1116 || (settings->textDirectionSubmenuInclusionBehavior() == TextDirectionSubmenuAutomaticallyIncluded && frame->editor()->hasBidiSelection());
1117 if (includeTextDirectionSubmenu) {
1118 ContextMenuItem TextDirectionMenuItem(SubmenuType, ContextMenuItemTagTextDirectionMenu,
1119 contextMenuItemTagTextDirectionMenu());
1120 createAndAppendTextDirectionSubMenu(TextDirectionMenuItem);
1121 appendItem(TextDirectionMenuItem, m_contextMenu.get());
1125 #endif //TIZEN_CONTEXT_MENU_TEMPORARY_FIX
1131 #if ENABLE(INSPECTOR)
1132 void ContextMenuController::addInspectElementItem()
1134 Node* node = m_hitTestResult.innerNonSharedNode();
1138 Frame* frame = node->document()->frame();
1142 Page* page = frame->page();
1146 if (!page->inspectorController())
1149 ContextMenuItem InspectElementItem(ActionType, ContextMenuItemTagInspectElement, contextMenuItemTagInspectElement());
1150 #if USE(CROSS_PLATFORM_CONTEXT_MENUS)
1151 if (!m_contextMenu->items().isEmpty())
1153 if (m_contextMenu->itemCount())
1155 appendItem(*separatorItem(), m_contextMenu.get());
1156 appendItem(InspectElementItem, m_contextMenu.get());
1158 #endif // ENABLE(INSPECTOR)
1160 void ContextMenuController::checkOrEnableIfNeeded(ContextMenuItem& item) const
1162 if (item.type() == SeparatorType)
1165 Frame* frame = m_hitTestResult.innerNonSharedNode()->document()->frame();
1169 // Custom items already have proper checked and enabled values.
1170 if (ContextMenuItemBaseCustomTag <= item.action() && item.action() <= ContextMenuItemLastCustomTag)
1173 bool shouldEnable = true;
1174 bool shouldCheck = false;
1176 switch (item.action()) {
1177 case ContextMenuItemTagCheckSpelling:
1178 shouldEnable = frame->editor()->canEdit();
1180 case ContextMenuItemTagDefaultDirection:
1181 shouldCheck = false;
1182 shouldEnable = false;
1184 case ContextMenuItemTagLeftToRight:
1185 case ContextMenuItemTagRightToLeft: {
1186 String direction = item.action() == ContextMenuItemTagLeftToRight ? "ltr" : "rtl";
1187 shouldCheck = frame->editor()->selectionHasStyle(CSSPropertyDirection, direction) != FalseTriState;
1188 shouldEnable = true;
1191 case ContextMenuItemTagTextDirectionDefault: {
1192 Editor::Command command = frame->editor()->command("MakeTextWritingDirectionNatural");
1193 shouldCheck = command.state() == TrueTriState;
1194 shouldEnable = command.isEnabled();
1197 case ContextMenuItemTagTextDirectionLeftToRight: {
1198 Editor::Command command = frame->editor()->command("MakeTextWritingDirectionLeftToRight");
1199 shouldCheck = command.state() == TrueTriState;
1200 shouldEnable = command.isEnabled();
1203 case ContextMenuItemTagTextDirectionRightToLeft: {
1204 Editor::Command command = frame->editor()->command("MakeTextWritingDirectionRightToLeft");
1205 shouldCheck = command.state() == TrueTriState;
1206 shouldEnable = command.isEnabled();
1209 case ContextMenuItemTagCopy:
1210 shouldEnable = frame->editor()->canDHTMLCopy() || frame->editor()->canCopy();
1212 case ContextMenuItemTagCut:
1213 shouldEnable = frame->editor()->canDHTMLCut() || frame->editor()->canCut();
1215 case ContextMenuItemTagIgnoreSpelling:
1216 case ContextMenuItemTagLearnSpelling:
1217 shouldEnable = frame->selection()->isRange();
1219 case ContextMenuItemTagPaste:
1220 shouldEnable = frame->editor()->canDHTMLPaste() || frame->editor()->canPaste();
1222 #if ENABLE(TIZEN_CONTEXT_MENU_SELECT)
1223 case ContextMenuItemTagSelectWord:
1224 shouldEnable = frame->editor()->canSelectRange();
1228 case ContextMenuItemTagDelete:
1229 shouldEnable = frame->editor()->canDelete();
1231 case ContextMenuItemTagInputMethods:
1232 case ContextMenuItemTagUnicode:
1233 case ContextMenuItemTagUnicodeInsertLRMMark:
1234 case ContextMenuItemTagUnicodeInsertRLMMark:
1235 case ContextMenuItemTagUnicodeInsertLREMark:
1236 case ContextMenuItemTagUnicodeInsertRLEMark:
1237 case ContextMenuItemTagUnicodeInsertLROMark:
1238 case ContextMenuItemTagUnicodeInsertRLOMark:
1239 case ContextMenuItemTagUnicodeInsertPDFMark:
1240 case ContextMenuItemTagUnicodeInsertZWSMark:
1241 case ContextMenuItemTagUnicodeInsertZWJMark:
1242 case ContextMenuItemTagUnicodeInsertZWNJMark:
1243 shouldEnable = true;
1246 #if PLATFORM(GTK) || PLATFORM(EFL)
1247 case ContextMenuItemTagSelectAll:
1248 shouldEnable = true;
1251 case ContextMenuItemTagUnderline: {
1252 shouldCheck = frame->editor()->selectionHasStyle(CSSPropertyWebkitTextDecorationsInEffect, "underline") != FalseTriState;
1253 shouldEnable = frame->editor()->canEditRichly();
1256 case ContextMenuItemTagLookUpInDictionary:
1257 shouldEnable = frame->selection()->isRange();
1259 case ContextMenuItemTagCheckGrammarWithSpelling:
1260 if (frame->editor()->isGrammarCheckingEnabled())
1262 shouldEnable = true;
1264 case ContextMenuItemTagItalic: {
1265 shouldCheck = frame->editor()->selectionHasStyle(CSSPropertyFontStyle, "italic") != FalseTriState;
1266 shouldEnable = frame->editor()->canEditRichly();
1269 case ContextMenuItemTagBold: {
1270 shouldCheck = frame->editor()->selectionHasStyle(CSSPropertyFontWeight, "bold") != FalseTriState;
1271 shouldEnable = frame->editor()->canEditRichly();
1274 case ContextMenuItemTagOutline:
1275 shouldEnable = false;
1277 case ContextMenuItemTagShowSpellingPanel:
1278 if (frame->editor()->spellingPanelIsShowing())
1279 item.setTitle(contextMenuItemTagShowSpellingPanel(false));
1281 item.setTitle(contextMenuItemTagShowSpellingPanel(true));
1282 shouldEnable = frame->editor()->canEdit();
1284 case ContextMenuItemTagNoGuessesFound:
1285 shouldEnable = false;
1287 case ContextMenuItemTagCheckSpellingWhileTyping:
1288 shouldCheck = frame->editor()->isContinuousSpellCheckingEnabled();
1291 case ContextMenuItemTagSubstitutionsMenu:
1292 case ContextMenuItemTagTransformationsMenu:
1294 case ContextMenuItemTagShowSubstitutions:
1295 #ifndef BUILDING_ON_LEOPARD
1296 if (frame->editor()->substitutionsPanelIsShowing())
1297 item.setTitle(contextMenuItemTagShowSubstitutions(false));
1299 item.setTitle(contextMenuItemTagShowSubstitutions(true));
1300 shouldEnable = frame->editor()->canEdit();
1303 case ContextMenuItemTagMakeUpperCase:
1304 case ContextMenuItemTagMakeLowerCase:
1305 case ContextMenuItemTagCapitalize:
1306 case ContextMenuItemTagChangeBack:
1307 shouldEnable = frame->editor()->canEdit();
1309 case ContextMenuItemTagCorrectSpellingAutomatically:
1310 #ifndef BUILDING_ON_LEOPARD
1311 shouldCheck = frame->editor()->isAutomaticSpellingCorrectionEnabled();
1314 case ContextMenuItemTagSmartCopyPaste:
1315 #ifndef BUILDING_ON_LEOPARD
1316 shouldCheck = frame->editor()->smartInsertDeleteEnabled();
1319 case ContextMenuItemTagSmartQuotes:
1320 #ifndef BUILDING_ON_LEOPARD
1321 shouldCheck = frame->editor()->isAutomaticQuoteSubstitutionEnabled();
1324 case ContextMenuItemTagSmartDashes:
1325 #ifndef BUILDING_ON_LEOPARD
1326 shouldCheck = frame->editor()->isAutomaticDashSubstitutionEnabled();
1329 case ContextMenuItemTagSmartLinks:
1330 #ifndef BUILDING_ON_LEOPARD
1331 shouldCheck = frame->editor()->isAutomaticLinkDetectionEnabled();
1334 case ContextMenuItemTagTextReplacement:
1335 #ifndef BUILDING_ON_LEOPARD
1336 shouldCheck = frame->editor()->isAutomaticTextReplacementEnabled();
1339 case ContextMenuItemTagStopSpeaking:
1340 shouldEnable = client() && client()->isSpeaking();
1342 #else // PLATFORM(MAC) ends here
1343 case ContextMenuItemTagStopSpeaking:
1347 case ContextMenuItemTagGoBack:
1348 shouldEnable = frame->page() && frame->page()->backForward()->canGoBackOrForward(-1);
1350 case ContextMenuItemTagGoForward:
1351 shouldEnable = frame->page() && frame->page()->backForward()->canGoBackOrForward(1);
1353 case ContextMenuItemTagStop:
1354 shouldEnable = frame->loader()->documentLoader()->isLoadingInAPISense();
1356 case ContextMenuItemTagReload:
1357 shouldEnable = !frame->loader()->documentLoader()->isLoadingInAPISense();
1359 case ContextMenuItemTagFontMenu:
1360 shouldEnable = frame->editor()->canEditRichly();
1363 case ContextMenuItemTagGoBack:
1364 case ContextMenuItemTagGoForward:
1365 case ContextMenuItemTagStop:
1366 case ContextMenuItemTagReload:
1367 case ContextMenuItemTagFontMenu:
1369 case ContextMenuItemTagNoAction:
1370 case ContextMenuItemTagOpenLinkInNewWindow:
1371 case ContextMenuItemTagDownloadLinkToDisk:
1372 case ContextMenuItemTagCopyLinkToClipboard:
1373 case ContextMenuItemTagOpenImageInNewWindow:
1374 case ContextMenuItemTagDownloadImageToDisk:
1375 case ContextMenuItemTagCopyImageToClipboard:
1376 #if PLATFORM(QT) || PLATFORM(GTK) || PLATFORM(EFL)
1377 case ContextMenuItemTagCopyImageUrlToClipboard:
1380 case ContextMenuItemTagOpenMediaInNewWindow:
1381 if (m_hitTestResult.mediaIsVideo())
1382 item.setTitle(contextMenuItemTagOpenVideoInNewWindow());
1384 item.setTitle(contextMenuItemTagOpenAudioInNewWindow());
1386 case ContextMenuItemTagCopyMediaLinkToClipboard:
1387 if (m_hitTestResult.mediaIsVideo())
1388 item.setTitle(contextMenuItemTagCopyVideoLinkToClipboard());
1390 item.setTitle(contextMenuItemTagCopyAudioLinkToClipboard());
1392 case ContextMenuItemTagToggleMediaControls:
1393 shouldCheck = m_hitTestResult.mediaControlsEnabled();
1395 case ContextMenuItemTagToggleMediaLoop:
1396 shouldCheck = m_hitTestResult.mediaLoopEnabled();
1398 case ContextMenuItemTagEnterVideoFullscreen:
1399 shouldEnable = m_hitTestResult.mediaSupportsFullscreen();
1401 case ContextMenuItemTagOpenFrameInNewWindow:
1402 case ContextMenuItemTagSpellingGuess:
1403 case ContextMenuItemTagOther:
1404 case ContextMenuItemTagSearchInSpotlight:
1405 case ContextMenuItemTagSearchWeb:
1406 case ContextMenuItemTagOpenWithDefaultApplication:
1407 case ContextMenuItemPDFActualSize:
1408 case ContextMenuItemPDFZoomIn:
1409 case ContextMenuItemPDFZoomOut:
1410 case ContextMenuItemPDFAutoSize:
1411 case ContextMenuItemPDFSinglePage:
1412 case ContextMenuItemPDFFacingPages:
1413 case ContextMenuItemPDFContinuous:
1414 case ContextMenuItemPDFNextPage:
1415 case ContextMenuItemPDFPreviousPage:
1416 case ContextMenuItemTagOpenLink:
1417 case ContextMenuItemTagIgnoreGrammar:
1418 case ContextMenuItemTagSpellingMenu:
1419 case ContextMenuItemTagShowFonts:
1420 case ContextMenuItemTagStyles:
1421 case ContextMenuItemTagShowColors:
1422 case ContextMenuItemTagSpeechMenu:
1423 case ContextMenuItemTagStartSpeaking:
1424 case ContextMenuItemTagWritingDirectionMenu:
1425 case ContextMenuItemTagTextDirectionMenu:
1426 case ContextMenuItemTagPDFSinglePageScrolling:
1427 case ContextMenuItemTagPDFFacingPagesScrolling:
1428 #if ENABLE(INSPECTOR)
1429 case ContextMenuItemTagInspectElement:
1431 case ContextMenuItemBaseCustomTag:
1432 case ContextMenuItemCustomTagNoAction:
1433 case ContextMenuItemLastCustomTag:
1434 case ContextMenuItemBaseApplicationTag:
1436 case ContextMenuItemTagMediaPlayPause:
1437 if (m_hitTestResult.mediaPlaying())
1438 item.setTitle(contextMenuItemTagMediaPause());
1440 item.setTitle(contextMenuItemTagMediaPlay());
1442 case ContextMenuItemTagMediaMute:
1443 shouldEnable = m_hitTestResult.mediaHasAudio();
1444 shouldCheck = shouldEnable && m_hitTestResult.mediaMuted();
1448 item.setChecked(shouldCheck);
1449 item.setEnabled(shouldEnable);
1452 #if USE(ACCESSIBILITY_CONTEXT_MENUS)
1453 void ContextMenuController::showContextMenuAt(Frame* frame, const IntPoint& clickPoint)
1455 // Simulate a click in the middle of the accessibility object.
1456 PlatformMouseEvent mouseEvent(clickPoint, clickPoint, RightButton, PlatformEvent::MousePressed, 1, false, false, false, false, currentTime());
1457 bool handled = frame->eventHandler()->sendContextMenuEvent(mouseEvent);
1458 if (handled && client())
1459 client()->showContextMenu();
1463 } // namespace WebCore
1465 #endif // ENABLE(CONTEXT_MENUS)