зеркало из https://github.com/mozilla/gecko-dev.git
969 строки
30 KiB
C++
969 строки
30 KiB
C++
/* -*- 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
if (editor) {
|
|
nsCOMPtr<nsISelection> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
if (editor) {
|
|
nsCOMPtr<nsISelection> 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<nsIEditor> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
if (editor) {
|
|
nsresult rv = editor->Copy();
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
nsCOMPtr<nsISelection> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE);
|
|
|
|
nsCOMPtr<nsISupports> supports;
|
|
aParams->GetISupportsValue("transferable", getter_AddRefs(supports));
|
|
NS_ENSURE_TRUE(supports, NS_ERROR_FAILURE);
|
|
|
|
nsCOMPtr<nsITransferable> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE);
|
|
|
|
nsCOMPtr<nsITransferable> trans;
|
|
|
|
nsCOMPtr<nsISupports> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> 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<nsIEditor> editor = do_QueryInterface(aCommandRefCon);
|
|
NS_ENSURE_TRUE(editor, NS_ERROR_FAILURE);
|
|
|
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
|
editor->GetDocument(getter_AddRefs(domDoc));
|
|
nsCOMPtr<nsIDocument> 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<nsISelectionController> 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<nsIEditor> 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<nsIPlaintextEditor> 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<nsIEditor> editor = do_QueryInterface(refCon);
|
|
nsCOMPtr<nsIEditorMailSupport> 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<nsIEditorMailSupport> 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<nsIEditorMailSupport> 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<nsIEditor> editor = do_QueryInterface(refCon);
|
|
if (editor) {
|
|
bool enabled = false;
|
|
editor->CanPaste(nsIClipboard::kGlobalClipboard, &enabled);
|
|
aParams->SetBooleanValue(STATE_ENABLED, enabled);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|