mirror of
https://github.com/rn10950/RetroZilla.git
synced 2024-11-11 02:10:17 +01:00
370 lines
12 KiB
C++
370 lines
12 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is TransforMiiX XSLT processor code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Peter Van der Beken <peterv@propagandism.org>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "txHTMLOutput.h"
|
|
#include "nsCOMArray.h"
|
|
#include "nsStaticNameTable.h"
|
|
#include "txAtoms.h"
|
|
#include "txOutputFormat.h"
|
|
#include "txStringUtils.h"
|
|
#include "XMLUtils.h"
|
|
|
|
#define EMPTY_ELEMENTS_COUNT 13
|
|
const char* const kHTMLEmptyTags[] =
|
|
{
|
|
"area",
|
|
"base",
|
|
"basefont",
|
|
"br",
|
|
"col",
|
|
"frame",
|
|
"hr",
|
|
"img",
|
|
"input",
|
|
"isindex",
|
|
"link",
|
|
"meta",
|
|
"param"
|
|
};
|
|
|
|
#define SHORTHAND_ATTR_COUNT 12
|
|
const char* const kHTMLEmptyAttributes[] =
|
|
{
|
|
"checked",
|
|
"compact",
|
|
"declare",
|
|
"defer",
|
|
"disabled",
|
|
"ismap",
|
|
"multiple",
|
|
"noresize",
|
|
"noshade",
|
|
"nowrap",
|
|
"readonly",
|
|
"selected"
|
|
};
|
|
|
|
struct txEmptyAttributesMaps
|
|
{
|
|
typedef nsCOMArray<nsIAtom> EmptyAttrBag;
|
|
EmptyAttrBag mMaps[SHORTHAND_ATTR_COUNT];
|
|
};
|
|
|
|
static PRInt32 gTableRefCount;
|
|
static nsStaticCaseInsensitiveNameTable* gHTMLEmptyTagsTable;
|
|
static nsStaticCaseInsensitiveNameTable* gHTMLEmptyAttributesTable;
|
|
static txEmptyAttributesMaps* gHTMLEmptyAttributesMaps;
|
|
|
|
/* static */
|
|
nsresult
|
|
txHTMLOutput::init()
|
|
{
|
|
if (0 == gTableRefCount++) {
|
|
NS_ASSERTION(!gHTMLEmptyTagsTable, "pre existing array!");
|
|
gHTMLEmptyTagsTable = new nsStaticCaseInsensitiveNameTable();
|
|
if (!gHTMLEmptyTagsTable) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
gHTMLEmptyTagsTable->Init(kHTMLEmptyTags, EMPTY_ELEMENTS_COUNT);
|
|
|
|
NS_ASSERTION(!gHTMLEmptyAttributesTable, "pre existing array!");
|
|
gHTMLEmptyAttributesTable = new nsStaticCaseInsensitiveNameTable();
|
|
if (!gHTMLEmptyAttributesTable) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
gHTMLEmptyAttributesTable->Init(kHTMLEmptyAttributes,
|
|
SHORTHAND_ATTR_COUNT);
|
|
|
|
NS_ASSERTION(!gHTMLEmptyAttributesMaps, "pre existing map!");
|
|
gHTMLEmptyAttributesMaps = new txEmptyAttributesMaps();
|
|
if (!gHTMLEmptyAttributesMaps) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
// checked
|
|
gHTMLEmptyAttributesMaps->mMaps[0].AppendObject(txHTMLAtoms::input);
|
|
|
|
// compact
|
|
gHTMLEmptyAttributesMaps->mMaps[1].AppendObject(txHTMLAtoms::dir);
|
|
gHTMLEmptyAttributesMaps->mMaps[1].AppendObject(txHTMLAtoms::dl);
|
|
gHTMLEmptyAttributesMaps->mMaps[1].AppendObject(txHTMLAtoms::menu);
|
|
gHTMLEmptyAttributesMaps->mMaps[1].AppendObject(txHTMLAtoms::ol);
|
|
gHTMLEmptyAttributesMaps->mMaps[1].AppendObject(txHTMLAtoms::ul);
|
|
|
|
// declare
|
|
gHTMLEmptyAttributesMaps->mMaps[2].AppendObject(txHTMLAtoms::object);
|
|
|
|
// defer
|
|
gHTMLEmptyAttributesMaps->mMaps[3].AppendObject(txHTMLAtoms::script);
|
|
|
|
// disabled
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::button);
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::input);
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::optgroup);
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::option);
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::select);
|
|
gHTMLEmptyAttributesMaps->mMaps[4].AppendObject(txHTMLAtoms::textarea);
|
|
|
|
// ismap
|
|
gHTMLEmptyAttributesMaps->mMaps[5].AppendObject(txHTMLAtoms::img);
|
|
gHTMLEmptyAttributesMaps->mMaps[5].AppendObject(txHTMLAtoms::input);
|
|
|
|
// multiple
|
|
gHTMLEmptyAttributesMaps->mMaps[6].AppendObject(txHTMLAtoms::select);
|
|
|
|
// noresize
|
|
gHTMLEmptyAttributesMaps->mMaps[7].AppendObject(txHTMLAtoms::frame);
|
|
|
|
// noshade
|
|
gHTMLEmptyAttributesMaps->mMaps[8].AppendObject(txHTMLAtoms::hr);
|
|
|
|
// nowrap
|
|
gHTMLEmptyAttributesMaps->mMaps[9].AppendObject(txHTMLAtoms::td);
|
|
gHTMLEmptyAttributesMaps->mMaps[9].AppendObject(txHTMLAtoms::th);
|
|
|
|
// readonly
|
|
gHTMLEmptyAttributesMaps->mMaps[10].AppendObject(txHTMLAtoms::input);
|
|
gHTMLEmptyAttributesMaps->mMaps[10].AppendObject(txHTMLAtoms::textarea);
|
|
|
|
// selected
|
|
gHTMLEmptyAttributesMaps->mMaps[11].AppendObject(txHTMLAtoms::option);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
/* static */
|
|
void
|
|
txHTMLOutput::shutdown()
|
|
{
|
|
if (0 == --gTableRefCount) {
|
|
if (gHTMLEmptyTagsTable) {
|
|
delete gHTMLEmptyTagsTable;
|
|
gHTMLEmptyTagsTable = nsnull;
|
|
}
|
|
if (gHTMLEmptyAttributesTable) {
|
|
delete gHTMLEmptyAttributesTable;
|
|
gHTMLEmptyAttributesTable = nsnull;
|
|
}
|
|
if (gHTMLEmptyAttributesMaps) {
|
|
delete gHTMLEmptyAttributesMaps;
|
|
gHTMLEmptyAttributesMaps = nsnull;
|
|
}
|
|
}
|
|
}
|
|
|
|
txHTMLOutput::txHTMLOutput(txOutputFormat* aFormat, ostream* aOut)
|
|
: txXMLOutput(aFormat, aOut)
|
|
{
|
|
mUseEmptyElementShorthand = PR_FALSE;
|
|
}
|
|
|
|
txHTMLOutput::~txHTMLOutput()
|
|
{
|
|
}
|
|
|
|
void txHTMLOutput::attribute(const nsAString& aName,
|
|
const PRInt32 aNsID,
|
|
const nsAString& aValue)
|
|
{
|
|
if (!mStartTagOpen)
|
|
// XXX Signal this? (can't add attributes after element closed)
|
|
return;
|
|
|
|
MBool shortHand = MB_FALSE;
|
|
if (aNsID == kNameSpaceID_None) {
|
|
const nsAString& localPart = XMLUtils::getLocalPart(aName);
|
|
shortHand = isShorthandAttribute(localPart);
|
|
if (shortHand &&
|
|
localPart.Equals(aValue, txCaseInsensitiveStringComparator())) {
|
|
txListIterator iter(&mAttributes);
|
|
txOutAttr* setAtt = 0;
|
|
nsCOMPtr<nsIAtom> localName = do_GetAtom(localPart);
|
|
txExpandedName att(aNsID, localName);
|
|
while ((setAtt = (txOutAttr*)iter.next())) {
|
|
if (setAtt->mName == att) {
|
|
setAtt->mShorthand = MB_TRUE;
|
|
break;
|
|
}
|
|
}
|
|
if (!setAtt) {
|
|
setAtt = new txOutAttr(aNsID, localName, EmptyString());
|
|
setAtt->mShorthand = MB_TRUE;
|
|
mAttributes.add(setAtt);
|
|
}
|
|
}
|
|
}
|
|
if (!shortHand)
|
|
txXMLOutput::attribute(aName, aNsID, aValue);
|
|
}
|
|
|
|
void txHTMLOutput::characters(const nsAString& aData, PRBool aDOE)
|
|
{
|
|
if (aDOE) {
|
|
closeStartTag(MB_FALSE);
|
|
printUTF8Chars(aData);
|
|
|
|
return;
|
|
}
|
|
|
|
// Special-case script and style
|
|
if (!mCurrentElements.isEmpty()) {
|
|
txExpandedName* currentElement = (txExpandedName*)mCurrentElements.peek();
|
|
if (currentElement->mNamespaceID == kNameSpaceID_None &&
|
|
(currentElement->mLocalName == txHTMLAtoms::script ||
|
|
currentElement->mLocalName == txHTMLAtoms::style)) {
|
|
closeStartTag(MB_FALSE);
|
|
printUTF8Chars(aData);
|
|
return;
|
|
}
|
|
}
|
|
txXMLOutput::characters(aData, aDOE);
|
|
}
|
|
|
|
void txHTMLOutput::endElement(const nsAString& aName,
|
|
const PRInt32 aNsID)
|
|
{
|
|
const nsAString& localPart = XMLUtils::getLocalPart(aName);
|
|
if ((aNsID == kNameSpaceID_None) && isShorthandElement(localPart) &&
|
|
mStartTagOpen) {
|
|
MBool newLine = (mOutputFormat.mIndent == eTrue) &&
|
|
mAfterEndTag;
|
|
closeStartTag(MB_FALSE);
|
|
if (newLine)
|
|
*mOut << endl;
|
|
if (mOutputFormat.mIndent == eTrue)
|
|
mIndentLevel -= DEFAULT_INDENT;
|
|
mAfterEndTag = MB_TRUE;
|
|
}
|
|
else {
|
|
txXMLOutput::endElement(aName, aNsID);
|
|
}
|
|
delete (txExpandedName*)mCurrentElements.pop();
|
|
}
|
|
|
|
void txHTMLOutput::processingInstruction(const nsAString& aTarget,
|
|
const nsAString& aData)
|
|
{
|
|
closeStartTag(MB_FALSE);
|
|
if (mOutputFormat.mIndent == eTrue) {
|
|
for (PRUint32 i = 0; i < mIndentLevel; i++)
|
|
*mOut << ' ';
|
|
}
|
|
*mOut << PI_START;
|
|
printUTF8Chars(aTarget);
|
|
*mOut << SPACE;
|
|
printUTF8Chars(aData);
|
|
*mOut << R_ANGLE_BRACKET;
|
|
if (mOutputFormat.mIndent == eTrue)
|
|
*mOut << endl;
|
|
}
|
|
|
|
void txHTMLOutput::startDocument()
|
|
{
|
|
// XXX Should be using mOutputFormat.getVersion
|
|
*mOut << DOCTYPE_START << "html " << PUBLIC;
|
|
*mOut << " \"-//W3C//DTD HTML 4.0 Transitional//EN\"";
|
|
*mOut << " \"http://www.w3.org/TR/REC-html40/loose.dtd\"";
|
|
*mOut << DOCTYPE_END << endl;
|
|
}
|
|
|
|
void txHTMLOutput::startElement(const nsAString& aName,
|
|
const PRInt32 aNsID)
|
|
{
|
|
txXMLOutput::startElement(aName, aNsID);
|
|
|
|
nsCOMPtr<nsIAtom> localAtom;
|
|
if (aNsID == kNameSpaceID_None) {
|
|
nsAutoString localName;
|
|
TX_ToLowerCase(aName, localName);
|
|
localAtom = do_GetAtom(localName);
|
|
}
|
|
else {
|
|
localAtom = do_GetAtom(aName);
|
|
}
|
|
NS_ASSERTION(localAtom, "Can't get atom");
|
|
txExpandedName* currentElement = new txExpandedName(aNsID, localAtom);
|
|
NS_ASSERTION(currentElement, "Can't create currentElement");
|
|
if (currentElement)
|
|
mCurrentElements.push(currentElement);
|
|
}
|
|
|
|
void txHTMLOutput::closeStartTag(MBool aUseEmptyElementShorthand)
|
|
{
|
|
txExpandedName* currentElement = mCurrentElements.isEmpty() ?
|
|
nsnull : (txExpandedName*)mCurrentElements.peek();
|
|
if (mStartTagOpen && currentElement &&
|
|
(currentElement->mNamespaceID == kNameSpaceID_None) &&
|
|
(currentElement->mLocalName == txHTMLAtoms::head)) {
|
|
txXMLOutput::closeStartTag(MB_FALSE);
|
|
if (mOutputFormat.mIndent == eTrue) {
|
|
*mOut << endl;
|
|
for (PRUint32 i = 0; i < mIndentLevel; i++)
|
|
*mOut << ' ';
|
|
}
|
|
*mOut << LT << "meta http-equiv=" << QUOTE << "Content-Type" << QUOTE;
|
|
*mOut << " content=" << QUOTE;
|
|
printUTF8Chars(mOutputFormat.mMediaType);
|
|
*mOut << "; charset=";
|
|
printUTF8Chars(mOutputFormat.mEncoding);
|
|
*mOut << QUOTE << GT;
|
|
}
|
|
else {
|
|
txXMLOutput::closeStartTag(aUseEmptyElementShorthand);
|
|
}
|
|
}
|
|
|
|
MBool txHTMLOutput::isShorthandElement(const nsAString& aLocalName)
|
|
{
|
|
return (gHTMLEmptyTagsTable->Lookup(aLocalName) !=
|
|
nsStaticCaseInsensitiveNameTable::NOT_FOUND);
|
|
}
|
|
|
|
MBool txHTMLOutput::isShorthandAttribute(const nsAString& aLocalName)
|
|
{
|
|
PRInt32 index = gHTMLEmptyTagsTable->Lookup(aLocalName);
|
|
if (index == nsStaticCaseInsensitiveNameTable::NOT_FOUND) {
|
|
return PR_FALSE;
|
|
}
|
|
|
|
txExpandedName* currentElement = (txExpandedName*)mCurrentElements.peek();
|
|
return (gHTMLEmptyAttributesMaps->mMaps[index].IndexOf(currentElement->mLocalName) > -1);
|
|
}
|