summaryrefslogtreecommitdiffstats
path: root/widget/nsHTMLFormatConverter.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'widget/nsHTMLFormatConverter.cpp')
-rw-r--r--widget/nsHTMLFormatConverter.cpp247
1 files changed, 247 insertions, 0 deletions
diff --git a/widget/nsHTMLFormatConverter.cpp b/widget/nsHTMLFormatConverter.cpp
new file mode 100644
index 000000000..6c8191927
--- /dev/null
+++ b/widget/nsHTMLFormatConverter.cpp
@@ -0,0 +1,247 @@
+/* -*- 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 "nsHTMLFormatConverter.h"
+
+#include "nsArray.h"
+#include "nsCRT.h"
+#include "nsIComponentManager.h"
+#include "nsCOMPtr.h"
+#include "nsXPCOM.h"
+#include "nsISupportsPrimitives.h"
+
+#include "nsITransferable.h" // for mime defs, this is BAD
+
+// HTML convertor stuff
+#include "nsPrimitiveHelpers.h"
+#include "nsIDocumentEncoder.h"
+#include "nsContentUtils.h"
+
+nsHTMLFormatConverter::nsHTMLFormatConverter()
+{
+}
+
+nsHTMLFormatConverter::~nsHTMLFormatConverter()
+{
+}
+
+NS_IMPL_ISUPPORTS(nsHTMLFormatConverter, nsIFormatConverter)
+
+//
+// GetInputDataFlavors
+//
+// Creates a new list and returns the list of all the flavors this converter
+// knows how to import. In this case, it's just HTML.
+//
+// Flavors (strings) are wrapped in a primitive object so that JavaScript can
+// access them easily via XPConnect.
+//
+NS_IMETHODIMP
+nsHTMLFormatConverter::GetInputDataFlavors(nsIArray **_retval)
+{
+ if ( !_retval )
+ return NS_ERROR_INVALID_ARG;
+
+ nsCOMPtr<nsIMutableArray> array = nsArray::Create();
+ nsresult rv = AddFlavorToList ( array, kHTMLMime );
+
+ array.forget(_retval);
+ return rv;
+
+} // GetInputDataFlavors
+
+
+//
+// GetOutputDataFlavors
+//
+// Creates a new list and returns the list of all the flavors this converter
+// knows how to export (convert). In this case, it's all sorts of things that HTML can be
+// converted to.
+//
+// Flavors (strings) are wrapped in a primitive object so that JavaScript can
+// access them easily via XPConnect.
+//
+NS_IMETHODIMP
+nsHTMLFormatConverter::GetOutputDataFlavors(nsIArray **_retval)
+{
+ if ( !_retval )
+ return NS_ERROR_INVALID_ARG;
+
+ nsCOMPtr<nsIMutableArray> array = nsArray::Create();
+ nsresult rv = AddFlavorToList ( array, kHTMLMime );
+ if ( NS_FAILED(rv) )
+ return rv;
+ rv = AddFlavorToList ( array, kUnicodeMime );
+ if ( NS_FAILED(rv) )
+ return rv;
+
+ array.forget(_retval);
+ return rv;
+
+} // GetOutputDataFlavors
+
+
+//
+// AddFlavorToList
+//
+// Convenience routine for adding a flavor wrapped in an nsISupportsCString object
+// to a list
+//
+nsresult
+nsHTMLFormatConverter :: AddFlavorToList ( nsCOMPtr<nsIMutableArray>& inList, const char* inFlavor )
+{
+ nsresult rv;
+
+ nsCOMPtr<nsISupportsCString> dataFlavor =
+ do_CreateInstance(NS_SUPPORTS_CSTRING_CONTRACTID, &rv);
+ if ( dataFlavor ) {
+ dataFlavor->SetData ( nsDependentCString(inFlavor) );
+ // add to list as an nsISupports so the correct interface gets the addref
+ // in AppendElement()
+ nsCOMPtr<nsISupports> genericFlavor ( do_QueryInterface(dataFlavor) );
+ inList->AppendElement ( genericFlavor, /*weak =*/ false);
+ }
+ return rv;
+
+} // AddFlavorToList
+
+
+//
+// CanConvert
+//
+// Determines if we support the given conversion. Currently, this method only
+// converts from HTML to others.
+//
+NS_IMETHODIMP
+nsHTMLFormatConverter::CanConvert(const char *aFromDataFlavor, const char *aToDataFlavor, bool *_retval)
+{
+ if ( !_retval )
+ return NS_ERROR_INVALID_ARG;
+
+ *_retval = false;
+ if ( !nsCRT::strcmp(aFromDataFlavor, kHTMLMime) ) {
+ if ( !nsCRT::strcmp(aToDataFlavor, kHTMLMime) )
+ *_retval = true;
+ else if ( !nsCRT::strcmp(aToDataFlavor, kUnicodeMime) )
+ *_retval = true;
+#if NOT_NOW
+// pinkerton
+// no one uses this flavor right now, so it's just slowing things down. If anyone cares I
+// can put it back in.
+ else if ( toFlavor.Equals(kAOLMailMime) )
+ *_retval = true;
+#endif
+ }
+ return NS_OK;
+
+} // CanConvert
+
+
+
+//
+// Convert
+//
+// Convert data from one flavor to another. The data is wrapped in primitive objects so that it is
+// accessible from JS. Currently, this only accepts HTML input, so anything else is invalid.
+//
+//XXX This method copies the data WAAAAY too many time for my liking. Grrrrrr. Mostly it's because
+//XXX we _must_ put things into nsStrings so that the parser will accept it. Lame lame lame lame. We
+//XXX also can't just get raw unicode out of the nsString, so we have to allocate heap to get
+//XXX unicode out of the string. Lame lame lame.
+//
+NS_IMETHODIMP
+nsHTMLFormatConverter::Convert(const char *aFromDataFlavor, nsISupports *aFromData, uint32_t aDataLen,
+ const char *aToDataFlavor, nsISupports **aToData, uint32_t *aDataToLen)
+{
+ if ( !aToData || !aDataToLen )
+ return NS_ERROR_INVALID_ARG;
+
+ nsresult rv = NS_OK;
+ *aToData = nullptr;
+ *aDataToLen = 0;
+
+ if ( !nsCRT::strcmp(aFromDataFlavor, kHTMLMime) ) {
+ nsAutoCString toFlavor ( aToDataFlavor );
+
+ // HTML on clipboard is going to always be double byte so it will be in a primitive
+ // class of nsISupportsString. Also, since the data is in two byte chunks the
+ // length represents the length in 1-byte chars, so we need to divide by two.
+ nsCOMPtr<nsISupportsString> dataWrapper0 ( do_QueryInterface(aFromData) );
+ if (!dataWrapper0) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ nsAutoString dataStr;
+ dataWrapper0->GetData ( dataStr ); // COPY #1
+ // note: conversion to text/plain is done inside the clipboard. we do not need to worry
+ // about it here.
+ if ( toFlavor.Equals(kHTMLMime) || toFlavor.Equals(kUnicodeMime) ) {
+ nsresult res;
+ if (toFlavor.Equals(kHTMLMime)) {
+ int32_t dataLen = dataStr.Length() * 2;
+ nsPrimitiveHelpers::CreatePrimitiveForData ( toFlavor.get(), dataStr.get(), dataLen, aToData );
+ if ( *aToData )
+ *aDataToLen = dataLen;
+ } else {
+ nsAutoString outStr;
+ res = ConvertFromHTMLToUnicode(dataStr, outStr);
+ if (NS_SUCCEEDED(res)) {
+ int32_t dataLen = outStr.Length() * 2;
+ nsPrimitiveHelpers::CreatePrimitiveForData ( toFlavor.get(), outStr.get(), dataLen, aToData );
+ if ( *aToData )
+ *aDataToLen = dataLen;
+ }
+ }
+ } // else if HTML or Unicode
+ else if ( toFlavor.Equals(kAOLMailMime) ) {
+ nsAutoString outStr;
+ if ( NS_SUCCEEDED(ConvertFromHTMLToAOLMail(dataStr, outStr)) ) {
+ int32_t dataLen = outStr.Length() * 2;
+ nsPrimitiveHelpers::CreatePrimitiveForData ( toFlavor.get(), outStr.get(), dataLen, aToData );
+ if ( *aToData )
+ *aDataToLen = dataLen;
+ }
+ } // else if AOL mail
+ else {
+ rv = NS_ERROR_FAILURE;
+ }
+ } // if we got html mime
+ else
+ rv = NS_ERROR_FAILURE;
+
+ return rv;
+
+} // Convert
+
+
+//
+// ConvertFromHTMLToUnicode
+//
+// Takes HTML and converts it to plain text but in unicode.
+//
+NS_IMETHODIMP
+nsHTMLFormatConverter::ConvertFromHTMLToUnicode(const nsAutoString & aFromStr, nsAutoString & aToStr)
+{
+ return nsContentUtils::ConvertToPlainText(aFromStr,
+ aToStr,
+ nsIDocumentEncoder::OutputSelectionOnly |
+ nsIDocumentEncoder::OutputAbsoluteLinks |
+ nsIDocumentEncoder::OutputNoScriptContent |
+ nsIDocumentEncoder::OutputNoFramesContent,
+ 0);
+} // ConvertFromHTMLToUnicode
+
+
+NS_IMETHODIMP
+nsHTMLFormatConverter::ConvertFromHTMLToAOLMail(const nsAutoString & aFromStr,
+ nsAutoString & aToStr)
+{
+ aToStr.AssignLiteral("<HTML>");
+ aToStr.Append(aFromStr);
+ aToStr.AppendLiteral("</HTML>");
+
+ return NS_OK;
+}
+