/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "nsEditorCommands.h" #include "mozFlushType.h" #include "mozilla/ArrayUtils.h" #include "mozilla/Assertions.h" #include "nsCOMPtr.h" #include "nsCRT.h" #include "nsDebug.h" #include "nsError.h" #include "nsIClipboard.h" #include "nsICommandParams.h" #include "nsID.h" #include "nsIDOMDocument.h" #include "nsIDocument.h" #include "nsIEditor.h" #include "nsIEditorMailSupport.h" #include "nsIPlaintextEditor.h" #include "nsISelection.h" #include "nsISelectionController.h" #include "nsITransferable.h" #include "nsString.h" #include "nsAString.h" class nsISupports; #define STATE_ENABLED "state_enabled" #define STATE_DATA "state_data" nsBaseEditorCommand::nsBaseEditorCommand() {} NS_IMPL_ISUPPORTS(nsBaseEditorCommand, nsIControllerCommand) NS_IMETHODIMP nsUndoCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { bool isEnabled, isEditable = false; nsresult rv = editor->GetIsSelectionEditable(&isEditable); NS_ENSURE_SUCCESS(rv, rv); if (isEditable) return editor->CanUndo(&isEnabled, outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsUndoCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->Undo(1); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsUndoCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsUndoCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsRedoCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { bool isEnabled, isEditable = false; nsresult rv = editor->GetIsSelectionEditable(&isEditable); NS_ENSURE_SUCCESS(rv, rv); if (isEditable) return editor->CanRedo(&isEnabled, outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsRedoCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->Redo(1); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsRedoCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsRedoCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsClearUndoCommand::IsCommandEnabled(const char *aCommandName, nsISupports *refCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(refCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsClearUndoCommand::DoCommand(const char *aCommandName, nsISupports *refCon) { nsCOMPtr editor = do_QueryInterface(refCon); NS_ENSURE_TRUE(editor, NS_ERROR_NOT_IMPLEMENTED); editor->EnableUndo(false); // Turning off undo clears undo/redo stacks. editor->EnableUndo(true); // This re-enables undo/redo. return NS_OK; } NS_IMETHODIMP nsClearUndoCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { return DoCommand(aCommandName, refCon); } NS_IMETHODIMP nsClearUndoCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { NS_ENSURE_ARG_POINTER(aParams); bool enabled; nsresult rv = IsCommandEnabled(aCommandName, refCon, &enabled); NS_ENSURE_SUCCESS(rv, rv); return aParams->SetBooleanValue(STATE_ENABLED, enabled); } NS_IMETHODIMP nsCutCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { bool isEditable = false; nsresult rv = editor->GetIsSelectionEditable(&isEditable); NS_ENSURE_SUCCESS(rv, rv); if (isEditable) return editor->CanCut(outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsCutCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->Cut(); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsCutCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsCutCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsCutOrDeleteCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsCutOrDeleteCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { nsCOMPtr selection; nsresult rv = editor->GetSelection(getter_AddRefs(selection)); if (NS_SUCCEEDED(rv) && selection && selection->Collapsed()) { return editor->DeleteSelection(nsIEditor::eNext, nsIEditor::eStrip); } return editor->Cut(); } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsCutOrDeleteCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsCutOrDeleteCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsCopyCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->CanCopy(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsCopyCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->Copy(); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsCopyCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsCopyCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsCopyOrDeleteCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsCopyOrDeleteCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { nsCOMPtr selection; nsresult rv = editor->GetSelection(getter_AddRefs(selection)); if (NS_SUCCEEDED(rv) && selection && selection->Collapsed()) { return editor->DeleteSelection(nsIEditor::eNextWord, nsIEditor::eStrip); } return editor->Copy(); } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsCopyOrDeleteCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsCopyOrDeleteCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsCopyAndCollapseToEndCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->CanCopy(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsCopyAndCollapseToEndCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { nsresult rv = editor->Copy(); if (NS_FAILED(rv)) { return rv; } nsCOMPtr selection; rv = editor->GetSelection(getter_AddRefs(selection)); if (NS_SUCCEEDED(rv) && selection) { selection->CollapseToEnd(); } return rv; } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsCopyAndCollapseToEndCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsCopyAndCollapseToEndCommand::GetCommandStateParams( const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsPasteCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { bool isEditable = false; nsresult rv = editor->GetIsSelectionEditable(&isEditable); NS_ENSURE_SUCCESS(rv, rv); if (isEditable) return editor->CanPaste(nsIClipboard::kGlobalClipboard, outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsPasteCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); return editor->Paste(nsIClipboard::kGlobalClipboard); } NS_IMETHODIMP nsPasteCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsPasteCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsPasteTransferableCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { bool isEditable = false; nsresult rv = editor->GetIsSelectionEditable(&isEditable); NS_ENSURE_SUCCESS(rv, rv); if (isEditable) return editor->CanPasteTransferable(nullptr, outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsPasteTransferableCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { return NS_ERROR_FAILURE; } NS_IMETHODIMP nsPasteTransferableCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); nsCOMPtr supports; aParams->GetISupportsValue("transferable", getter_AddRefs(supports)); NS_ENSURE_TRUE(supports, NS_ERROR_FAILURE); nsCOMPtr trans = do_QueryInterface(supports); NS_ENSURE_TRUE(trans, NS_ERROR_FAILURE); return editor->PasteTransferable(trans); } NS_IMETHODIMP nsPasteTransferableCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); nsCOMPtr trans; nsCOMPtr supports; aParams->GetISupportsValue("transferable", getter_AddRefs(supports)); if (supports) { trans = do_QueryInterface(supports); NS_ENSURE_TRUE(trans, NS_ERROR_FAILURE); } bool canPaste; nsresult rv = editor->CanPasteTransferable(trans, &canPaste); NS_ENSURE_SUCCESS(rv, rv); return aParams->SetBooleanValue(STATE_ENABLED, canPaste); } NS_IMETHODIMP nsSwitchTextDirectionCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsSwitchTextDirectionCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); return editor->SwitchTextDirection(); } NS_IMETHODIMP nsSwitchTextDirectionCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsSwitchTextDirectionCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canSwitchTextDirection = true; IsCommandEnabled(aCommandName, aCommandRefCon, &canSwitchTextDirection); return aParams->SetBooleanValue(STATE_ENABLED, canSwitchTextDirection); } NS_IMETHODIMP nsDeleteCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); *outCmdEnabled = false; NS_ENSURE_TRUE(editor, NS_OK); // We can generally delete whenever the selection is editable. However, // cmd_delete doesn't make sense if the selection is collapsed because it's // directionless, which is the same condition under which we can't cut. nsresult rv = editor->GetIsSelectionEditable(outCmdEnabled); NS_ENSURE_SUCCESS(rv, rv); if (!nsCRT::strcmp("cmd_delete", aCommandName) && *outCmdEnabled) { rv = editor->CanDelete(outCmdEnabled); NS_ENSURE_SUCCESS(rv, rv); } return NS_OK; } NS_IMETHODIMP nsDeleteCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); nsIEditor::EDirection deleteDir = nsIEditor::eNone; if (!nsCRT::strcmp("cmd_delete", aCommandName)) { // Really this should probably be eNone, but it only makes a difference if // the selection is collapsed, and then this command is disabled. So let's // keep it as it always was to avoid breaking things. deleteDir = nsIEditor::ePrevious; } else if (!nsCRT::strcmp("cmd_deleteCharForward", aCommandName)) { deleteDir = nsIEditor::eNext; } else if (!nsCRT::strcmp("cmd_deleteCharBackward", aCommandName)) { deleteDir = nsIEditor::ePrevious; } else if (!nsCRT::strcmp("cmd_deleteWordBackward", aCommandName)) { deleteDir = nsIEditor::ePreviousWord; } else if (!nsCRT::strcmp("cmd_deleteWordForward", aCommandName)) { deleteDir = nsIEditor::eNextWord; } else if (!nsCRT::strcmp("cmd_deleteToBeginningOfLine", aCommandName)) { deleteDir = nsIEditor::eToBeginningOfLine; } else if (!nsCRT::strcmp("cmd_deleteToEndOfLine", aCommandName)) { deleteDir = nsIEditor::eToEndOfLine; } else { MOZ_CRASH("Unrecognized nsDeleteCommand"); } return editor->DeleteSelection(deleteDir, nsIEditor::eStrip); } NS_IMETHODIMP nsDeleteCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsDeleteCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsSelectAllCommand::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsresult rv = NS_OK; // You can always select all, unless the selection is editable, // and the editable region is empty! *outCmdEnabled = true; bool docIsEmpty; // you can select all if there is an editor which is non-empty nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) { rv = editor->GetDocumentIsEmpty(&docIsEmpty); NS_ENSURE_SUCCESS(rv, rv); *outCmdEnabled = !docIsEmpty; } return rv; } NS_IMETHODIMP nsSelectAllCommand::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->SelectAll(); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsSelectAllCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsSelectAllCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsSelectionMoveCommands::IsCommandEnabled(const char *aCommandName, nsISupports *aCommandRefCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(aCommandRefCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_OK; } static const struct ScrollCommand { const char *reverseScroll; const char *forwardScroll; nsresult (NS_STDCALL nsISelectionController::*scroll)(bool); } scrollCommands[] = { { "cmd_scrollTop", "cmd_scrollBottom", &nsISelectionController::CompleteScroll }, { "cmd_scrollPageUp", "cmd_scrollPageDown", &nsISelectionController::ScrollPage }, { "cmd_scrollLineUp", "cmd_scrollLineDown", &nsISelectionController::ScrollLine } }; static const struct MoveCommand { const char *reverseMove; const char *forwardMove; const char *reverseSelect; const char *forwardSelect; nsresult (NS_STDCALL nsISelectionController::*move)(bool, bool); } moveCommands[] = { { "cmd_charPrevious", "cmd_charNext", "cmd_selectCharPrevious", "cmd_selectCharNext", &nsISelectionController::CharacterMove }, { "cmd_linePrevious", "cmd_lineNext", "cmd_selectLinePrevious", "cmd_selectLineNext", &nsISelectionController::LineMove }, { "cmd_wordPrevious", "cmd_wordNext", "cmd_selectWordPrevious", "cmd_selectWordNext", &nsISelectionController::WordMove }, { "cmd_beginLine", "cmd_endLine", "cmd_selectBeginLine", "cmd_selectEndLine", &nsISelectionController::IntraLineMove }, { "cmd_movePageUp", "cmd_movePageDown", "cmd_selectPageUp", "cmd_selectPageDown", &nsISelectionController::PageMove }, { "cmd_moveTop", "cmd_moveBottom", "cmd_selectTop", "cmd_selectBottom", &nsISelectionController::CompleteMove } }; static const struct PhysicalCommand { const char *move; const char *select; int16_t direction; int16_t amount; } physicalCommands[] = { { "cmd_moveLeft", "cmd_selectLeft", nsISelectionController::MOVE_LEFT, 0 }, { "cmd_moveRight", "cmd_selectRight", nsISelectionController::MOVE_RIGHT, 0 }, { "cmd_moveUp", "cmd_selectUp", nsISelectionController::MOVE_UP, 0 }, { "cmd_moveDown", "cmd_selectDown", nsISelectionController::MOVE_DOWN, 0 }, { "cmd_moveLeft2", "cmd_selectLeft2", nsISelectionController::MOVE_LEFT, 1 }, { "cmd_moveRight2", "cmd_selectRight2", nsISelectionController::MOVE_RIGHT, 1 }, { "cmd_moveUp2", "cmd_selectUp2", nsISelectionController::MOVE_UP, 1 }, { "cmd_moveDown2", "cmd_selectDown2", nsISelectionController::MOVE_DOWN, 1 } }; NS_IMETHODIMP nsSelectionMoveCommands::DoCommand(const char *aCommandName, nsISupports *aCommandRefCon) { nsCOMPtr editor = do_QueryInterface(aCommandRefCon); NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE); nsCOMPtr domDoc; editor->GetDocument(getter_AddRefs(domDoc)); nsCOMPtr doc = do_QueryInterface(domDoc); if (doc) { // Most of the commands below (possibly all of them) need layout to // be up to date. doc->FlushPendingNotifications(Flush_Layout); } nsCOMPtr selCont; nsresult rv = editor->GetSelectionController(getter_AddRefs(selCont)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(selCont, NS_ERROR_FAILURE); // scroll commands for (size_t i = 0; i < mozilla::ArrayLength(scrollCommands); i++) { const ScrollCommand &cmd = scrollCommands[i]; if (!nsCRT::strcmp(aCommandName, cmd.reverseScroll)) { return (selCont->*(cmd.scroll))(false); } else if (!nsCRT::strcmp(aCommandName, cmd.forwardScroll)) { return (selCont->*(cmd.scroll))(true); } } // caret movement/selection commands for (size_t i = 0; i < mozilla::ArrayLength(moveCommands); i++) { const MoveCommand &cmd = moveCommands[i]; if (!nsCRT::strcmp(aCommandName, cmd.reverseMove)) { return (selCont->*(cmd.move))(false, false); } else if (!nsCRT::strcmp(aCommandName, cmd.forwardMove)) { return (selCont->*(cmd.move))(true, false); } else if (!nsCRT::strcmp(aCommandName, cmd.reverseSelect)) { return (selCont->*(cmd.move))(false, true); } else if (!nsCRT::strcmp(aCommandName, cmd.forwardSelect)) { return (selCont->*(cmd.move))(true, true); } } // physical-direction movement/selection for (size_t i = 0; i < mozilla::ArrayLength(physicalCommands); i++) { const PhysicalCommand &cmd = physicalCommands[i]; if (!nsCRT::strcmp(aCommandName, cmd.move)) { return selCont->PhysicalMove(cmd.direction, cmd.amount, false); } else if (!nsCRT::strcmp(aCommandName, cmd.select)) { return selCont->PhysicalMove(cmd.direction, cmd.amount, true); } } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsSelectionMoveCommands::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { return DoCommand(aCommandName, aCommandRefCon); } NS_IMETHODIMP nsSelectionMoveCommands::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *aCommandRefCon) { bool canUndo; IsCommandEnabled(aCommandName, aCommandRefCon, &canUndo); return aParams->SetBooleanValue(STATE_ENABLED, canUndo); } NS_IMETHODIMP nsInsertPlaintextCommand::IsCommandEnabled(const char *aCommandName, nsISupports *refCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(refCon); if (editor) return editor->GetIsSelectionEditable(outCmdEnabled); *outCmdEnabled = false; return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsInsertPlaintextCommand::DoCommand(const char *aCommandName, nsISupports *refCon) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsInsertPlaintextCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { NS_ENSURE_ARG_POINTER(aParams); nsCOMPtr editor = do_QueryInterface(refCon); NS_ENSURE_TRUE(editor, NS_ERROR_NOT_IMPLEMENTED); // Get text to insert from command params nsAutoString text; nsresult rv = aParams->GetStringValue(STATE_DATA, text); NS_ENSURE_SUCCESS(rv, rv); if (!text.IsEmpty()) return editor->InsertText(text); return NS_OK; } NS_IMETHODIMP nsInsertPlaintextCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { NS_ENSURE_ARG_POINTER(aParams); bool outCmdEnabled = false; IsCommandEnabled(aCommandName, refCon, &outCmdEnabled); return aParams->SetBooleanValue(STATE_ENABLED, outCmdEnabled); } NS_IMETHODIMP nsPasteQuotationCommand::IsCommandEnabled(const char *aCommandName, nsISupports *refCon, bool *outCmdEnabled) { NS_ENSURE_ARG_POINTER(outCmdEnabled); nsCOMPtr editor = do_QueryInterface(refCon); nsCOMPtr mailEditor = do_QueryInterface(refCon); if (editor && mailEditor) { uint32_t flags; editor->GetFlags(&flags); if (!(flags & nsIPlaintextEditor::eEditorSingleLineMask)) return editor->CanPaste(nsIClipboard::kGlobalClipboard, outCmdEnabled); } *outCmdEnabled = false; return NS_OK; } NS_IMETHODIMP nsPasteQuotationCommand::DoCommand(const char *aCommandName, nsISupports *refCon) { nsCOMPtr mailEditor = do_QueryInterface(refCon); if (mailEditor) return mailEditor->PasteAsQuotation(nsIClipboard::kGlobalClipboard); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsPasteQuotationCommand::DoCommandParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { nsCOMPtr mailEditor = do_QueryInterface(refCon); if (mailEditor) return mailEditor->PasteAsQuotation(nsIClipboard::kGlobalClipboard); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsPasteQuotationCommand::GetCommandStateParams(const char *aCommandName, nsICommandParams *aParams, nsISupports *refCon) { nsCOMPtr editor = do_QueryInterface(refCon); if (editor) { bool enabled = false; editor->CanPaste(nsIClipboard::kGlobalClipboard, &enabled); aParams->SetBooleanValue(STATE_ENABLED, enabled); } return NS_OK; }