summaryrefslogtreecommitdiffstats
path: root/editor/libeditor/HTMLEditorDocumentCommands.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'editor/libeditor/HTMLEditorDocumentCommands.cpp')
-rw-r--r--editor/libeditor/HTMLEditorDocumentCommands.cpp412
1 files changed, 412 insertions, 0 deletions
diff --git a/editor/libeditor/HTMLEditorDocumentCommands.cpp b/editor/libeditor/HTMLEditorDocumentCommands.cpp
new file mode 100644
index 0000000000..111c4fc1b0
--- /dev/null
+++ b/editor/libeditor/HTMLEditorDocumentCommands.cpp
@@ -0,0 +1,412 @@
+/* -*- 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 "mozilla/EditorCommands.h"
+
+#include "mozilla/HTMLEditor.h" // for HTMLEditor
+#include "mozilla/TextEditor.h" // for TextEditor
+#include "mozilla/dom/Document.h" // for Document
+#include "nsCommandParams.h" // for nsCommandParams
+#include "nsIEditingSession.h" // for nsIEditingSession, etc
+#include "nsIPrincipal.h" // for nsIPrincipal
+#include "nsISupportsImpl.h" // for nsPresContext::Release
+#include "nsISupportsUtils.h" // for NS_IF_ADDREF
+#include "nsIURI.h" // for nsIURI
+#include "nsPresContext.h" // for nsPresContext
+
+// defines
+#define STATE_ENABLED "state_enabled"
+#define STATE_ALL "state_all"
+#define STATE_ATTRIBUTE "state_attribute"
+#define STATE_DATA "state_data"
+
+namespace mozilla {
+
+/*****************************************************************************
+ * mozilla::SetDocumentStateCommand
+ *
+ * Commands for document state that may be changed via doCommandParams
+ * As of 11/11/02, this is just "cmd_setDocumentModified"
+ * Note that you can use the same command class, SetDocumentStateCommand,
+ * for more than one of this type of command
+ * We check the input command param for different behavior
+ *****************************************************************************/
+
+StaticRefPtr<SetDocumentStateCommand> SetDocumentStateCommand::sInstance;
+
+bool SetDocumentStateCommand::IsCommandEnabled(Command aCommand,
+ TextEditor* aTextEditor) const {
+ // These commands are always enabled if given editor is an HTMLEditor.
+ return aTextEditor && aTextEditor->AsHTMLEditor();
+}
+
+nsresult SetDocumentStateCommand::DoCommand(Command aCommand,
+ TextEditor& aTextEditor,
+ nsIPrincipal* aPrincipal) const {
+ return NS_ERROR_NOT_IMPLEMENTED;
+}
+
+nsresult SetDocumentStateCommand::DoCommandParam(
+ Command aCommand, const Maybe<bool>& aBoolParam, TextEditor& aTextEditor,
+ nsIPrincipal* aPrincipal) const {
+ if (NS_WARN_IF(aBoolParam.isNothing())) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (NS_WARN_IF(!aTextEditor.AsHTMLEditor())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ switch (aCommand) {
+ case Command::SetDocumentModified: {
+ if (aBoolParam.value()) {
+ nsresult rv = aTextEditor.IncrementModificationCount(1);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::IncrementModificationCount() failed");
+ return rv;
+ }
+ nsresult rv = aTextEditor.ResetModificationCount();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::ResetModificationCount() failed");
+ return rv;
+ }
+ case Command::SetDocumentReadOnly: {
+ ErrorResult error;
+ if (aBoolParam.value()) {
+ nsresult rv = aTextEditor.AddFlags(nsIEditor::eEditorReadonlyMask);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::AddFlags(nsIEditor::eEditorReadonlyMask) failed");
+ return rv;
+ }
+ nsresult rv = aTextEditor.RemoveFlags(nsIEditor::eEditorReadonlyMask);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::RemoveFlags(nsIEditor::eEditorReadonlyMask) failed");
+ return rv;
+ }
+ case Command::SetDocumentUseCSS: {
+ nsresult rv = MOZ_KnownLive(aTextEditor.AsHTMLEditor())
+ ->SetIsCSSEnabled(aBoolParam.value());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "HTMLEditor::SetIsCSSEnabled() failed");
+ return rv;
+ }
+ case Command::SetDocumentInsertBROnEnterKeyPress: {
+ nsresult rv =
+ aTextEditor.AsHTMLEditor()->SetReturnInParagraphCreatesNewParagraph(
+ !aBoolParam.value());
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "HTMLEditor::SetReturnInParagraphCreatesNewParagraph() failed");
+ return rv;
+ }
+ case Command::ToggleObjectResizers: {
+ MOZ_KnownLive(aTextEditor.AsHTMLEditor())
+ ->EnableObjectResizer(aBoolParam.value());
+ return NS_OK;
+ }
+ case Command::ToggleInlineTableEditor: {
+ MOZ_KnownLive(aTextEditor.AsHTMLEditor())
+ ->EnableInlineTableEditor(aBoolParam.value());
+ return NS_OK;
+ }
+ case Command::ToggleAbsolutePositionEditor: {
+ MOZ_KnownLive(aTextEditor.AsHTMLEditor())
+ ->EnableAbsolutePositionEditor(aBoolParam.value());
+ return NS_OK;
+ }
+ default:
+ return NS_ERROR_NOT_IMPLEMENTED;
+ }
+}
+
+nsresult SetDocumentStateCommand::DoCommandParam(
+ Command aCommand, const nsACString& aCStringParam, TextEditor& aTextEditor,
+ nsIPrincipal* aPrincipal) const {
+ if (NS_WARN_IF(aCStringParam.IsVoid())) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (NS_WARN_IF(!aTextEditor.AsHTMLEditor())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ switch (aCommand) {
+ case Command::SetDocumentDefaultParagraphSeparator: {
+ if (aCStringParam.LowerCaseEqualsLiteral("div")) {
+ aTextEditor.AsHTMLEditor()->SetDefaultParagraphSeparator(
+ ParagraphSeparator::div);
+ return NS_OK;
+ }
+ if (aCStringParam.LowerCaseEqualsLiteral("p")) {
+ aTextEditor.AsHTMLEditor()->SetDefaultParagraphSeparator(
+ ParagraphSeparator::p);
+ return NS_OK;
+ }
+ if (aCStringParam.LowerCaseEqualsLiteral("br")) {
+ // Mozilla extension for backwards compatibility
+ aTextEditor.AsHTMLEditor()->SetDefaultParagraphSeparator(
+ ParagraphSeparator::br);
+ return NS_OK;
+ }
+
+ // This should not be reachable from nsHTMLDocument::ExecCommand
+ // XXX Shouldn't return error in this case because Chrome does not throw
+ // exception in this case.
+ NS_WARNING("Invalid default paragraph separator");
+ return NS_ERROR_UNEXPECTED;
+ }
+ default:
+ return NS_ERROR_NOT_IMPLEMENTED;
+ }
+}
+
+nsresult SetDocumentStateCommand::GetCommandStateParams(
+ Command aCommand, nsCommandParams& aParams, TextEditor* aTextEditor,
+ nsIEditingSession* aEditingSession) const {
+ // If the result is set to STATE_ATTRIBUTE as CString value,
+ // queryCommandValue() returns the string value.
+ // Otherwise, ignored.
+
+ // The base editor owns most state info
+ if (NS_WARN_IF(!aTextEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (NS_WARN_IF(!aTextEditor->AsHTMLEditor())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Always get the enabled state
+ nsresult rv =
+ aParams.SetBool(STATE_ENABLED, IsCommandEnabled(aCommand, aTextEditor));
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+
+ switch (aCommand) {
+ case Command::SetDocumentModified: {
+ bool modified;
+ rv = aTextEditor->GetDocumentModified(&modified);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::GetDocumentModified() failed");
+ return rv;
+ }
+ // XXX Nobody refers this result due to wrong type.
+ rv = aParams.SetBool(STATE_ATTRIBUTE, modified);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ATTRIBUTE) failed");
+ return rv;
+ }
+ case Command::SetDocumentReadOnly: {
+ // XXX Nobody refers this result due to wrong type.
+ rv = aParams.SetBool(STATE_ATTRIBUTE, aTextEditor->IsReadonly());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ATTRIBUTE) failed");
+ return rv;
+ }
+ case Command::SetDocumentUseCSS: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ rv = aParams.SetBool(STATE_ALL, htmlEditor->IsCSSEnabled());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ALL) failed");
+ return rv;
+ }
+ case Command::SetDocumentInsertBROnEnterKeyPress: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ bool createPOnReturn;
+ DebugOnly<nsresult> rvIgnored =
+ htmlEditor->GetReturnInParagraphCreatesNewParagraph(&createPOnReturn);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "HTMLEditor::GetReturnInParagraphCreatesNewParagraph() failed");
+ // XXX Nobody refers this result due to wrong type.
+ rv = aParams.SetBool(STATE_ATTRIBUTE, !createPOnReturn);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ATTRIBUTE) failed");
+ return rv;
+ }
+ case Command::SetDocumentDefaultParagraphSeparator: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ switch (htmlEditor->GetDefaultParagraphSeparator()) {
+ case ParagraphSeparator::div: {
+ DebugOnly<nsresult> rv =
+ aParams.SetCString(STATE_ATTRIBUTE, "div"_ns);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "Failed to set command params to return \"div\"");
+ return NS_OK;
+ }
+ case ParagraphSeparator::p: {
+ DebugOnly<nsresult> rv = aParams.SetCString(STATE_ATTRIBUTE, "p"_ns);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "Failed to set command params to return \"p\"");
+ return NS_OK;
+ }
+ case ParagraphSeparator::br: {
+ DebugOnly<nsresult> rv = aParams.SetCString(STATE_ATTRIBUTE, "br"_ns);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "Failed to set command params to return \"br\"");
+ return NS_OK;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Invalid paragraph separator value");
+ return NS_ERROR_UNEXPECTED;
+ }
+ }
+ case Command::ToggleObjectResizers: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ // We returned the result as STATE_ATTRIBUTE with bool value 60 or
+ // earlier. So, the result was ignored by both
+ // nsHTMLDocument::QueryCommandValue() and
+ // nsHTMLDocument::QueryCommandState().
+ rv = aParams.SetBool(STATE_ALL, htmlEditor->IsObjectResizerEnabled());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ALL) failed");
+ return rv;
+ }
+ case Command::ToggleInlineTableEditor: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ // We returned the result as STATE_ATTRIBUTE with bool value 60 or
+ // earlier. So, the result was ignored by both
+ // nsHTMLDocument::QueryCommandValue() and
+ // nsHTMLDocument::QueryCommandState().
+ rv = aParams.SetBool(STATE_ALL, htmlEditor->IsInlineTableEditorEnabled());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCommandParams::SetBool(STATE_ALL) failed");
+ return rv;
+ }
+ case Command::ToggleAbsolutePositionEditor: {
+ HTMLEditor* htmlEditor = aTextEditor->AsHTMLEditor();
+ if (NS_WARN_IF(!htmlEditor)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ return aParams.SetBool(STATE_ALL,
+ htmlEditor->IsAbsolutePositionEditorEnabled());
+ }
+ default:
+ return NS_ERROR_NOT_IMPLEMENTED;
+ }
+}
+
+/*****************************************************************************
+ * mozilla::DocumentStateCommand
+ *
+ * Commands just for state notification
+ * As of 11/21/02, possible commands are:
+ * "obs_documentCreated"
+ * "obs_documentWillBeDestroyed"
+ * "obs_documentLocationChanged"
+ * Note that you can use the same command class, DocumentStateCommand
+ * for these or future observer commands.
+ * We check the input command param for different behavior
+ *
+ * How to use:
+ * 1. Get the nsCommandManager for the current editor
+ * 2. Implement an nsIObserve object, e.g:
+ *
+ * void Observe(
+ * in nsISupports aSubject, // The nsCommandManager calling this
+ * // Observer
+ * in string aTopic, // command name, e.g.:"obs_documentCreated"
+ * // or "obs_documentWillBeDestroyed"
+ in wstring aData ); // ignored (set to "command_status_changed")
+ *
+ * 3. Add the observer by:
+ * commandManager.addObserver(observeobject, obs_documentCreated);
+ * 4. In the appropriate location in editorSession, editor, or commands code,
+ * trigger the notification of this observer by something like:
+ *
+ * RefPtr<nsCommandManager> commandManager = mDocShell->GetCommandManager();
+ * commandManager->CommandStatusChanged(obs_documentCreated);
+ *
+ * 5. Use GetCommandStateParams() to obtain state information
+ * e.g., any creation state codes when creating an editor are
+ * supplied for "obs_documentCreated" command in the
+ * "state_data" param's value
+ *****************************************************************************/
+
+StaticRefPtr<DocumentStateCommand> DocumentStateCommand::sInstance;
+
+bool DocumentStateCommand::IsCommandEnabled(Command aCommand,
+ TextEditor* aTextEditor) const {
+ // Always return false to discourage callers from using DoCommand()
+ return false;
+}
+
+nsresult DocumentStateCommand::DoCommand(Command aCommand,
+ TextEditor& aTextEditor,
+ nsIPrincipal* aPrincipal) const {
+ return NS_ERROR_NOT_IMPLEMENTED;
+}
+
+nsresult DocumentStateCommand::GetCommandStateParams(
+ Command aCommand, nsCommandParams& aParams, TextEditor* aTextEditor,
+ nsIEditingSession* aEditingSession) const {
+ switch (aCommand) {
+ case Command::EditorObserverDocumentCreated: {
+ uint32_t editorStatus = nsIEditingSession::eEditorErrorUnknown;
+ if (aEditingSession) {
+ // Current context is initially set to nsIEditingSession until editor is
+ // successfully created and source doc is loaded. Embedder gets error
+ // status if this fails. If called before startup is finished,
+ // status will be eEditorCreationInProgress.
+ nsresult rv = aEditingSession->GetEditorStatus(&editorStatus);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("nsIEditingSession::GetEditorStatus() failed");
+ return rv;
+ }
+ } else if (aTextEditor) {
+ // If current context is an editor, then everything started up OK!
+ editorStatus = nsIEditingSession::eEditorOK;
+ }
+
+ // Note that if refCon is not-null, but is neither
+ // an nsIEditingSession or nsIEditor, we return "eEditorErrorUnknown"
+ DebugOnly<nsresult> rvIgnored = aParams.SetInt(STATE_DATA, editorStatus);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "Failed to set editor status");
+ return NS_OK;
+ }
+ case Command::EditorObserverDocumentLocationChanged: {
+ if (!aTextEditor) {
+ return NS_OK;
+ }
+ dom::Document* document = aTextEditor->GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return NS_ERROR_FAILURE;
+ }
+ nsIURI* uri = document->GetDocumentURI();
+ if (NS_WARN_IF(!uri)) {
+ return NS_ERROR_FAILURE;
+ }
+ nsresult rv = aParams.SetISupports(STATE_DATA, uri);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsCOmmandParms::SetISupports(STATE_DATA) failed");
+ return rv;
+ }
+ default:
+ return NS_ERROR_NOT_IMPLEMENTED;
+ }
+}
+
+} // namespace mozilla