blob: 532a48430488ec2cf5de74ab0144cc8e19797195 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2002, 2006 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
* Jens Lukowski/Innoopract - initial renaming/restructuring
*
*******************************************************************************/
package org.eclipse.wst.xml.core.internal.contentmodel.util;
import java.util.Hashtable;
import java.util.List;
import java.util.Stack;
import java.util.Vector;
import org.eclipse.wst.xml.core.internal.contentmodel.CMAnyElement;
import org.eclipse.wst.xml.core.internal.contentmodel.CMAttributeDeclaration;
import org.eclipse.wst.xml.core.internal.contentmodel.CMContent;
import org.eclipse.wst.xml.core.internal.contentmodel.CMDataType;
import org.eclipse.wst.xml.core.internal.contentmodel.CMDocument;
import org.eclipse.wst.xml.core.internal.contentmodel.CMElementDeclaration;
import org.eclipse.wst.xml.core.internal.contentmodel.CMGroup;
import org.eclipse.wst.xml.core.internal.contentmodel.CMNamedNodeMap;
import org.eclipse.wst.xml.core.internal.contentmodel.CMNode;
import org.eclipse.wst.xml.core.internal.contentmodel.CMNodeList;
import org.eclipse.wst.xml.core.internal.contentmodel.ContentModelManager;
import org.eclipse.wst.xml.core.internal.contentmodel.internal.util.CMDataTypeValueHelper;
import org.w3c.dom.Attr;
import org.w3c.dom.DOMImplementation;
import org.w3c.dom.Document;
import org.w3c.dom.DocumentType;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.ProcessingInstruction;
import org.w3c.dom.Text;
/**
* todo... common up this code with 'ContentBuilder'
*/
public class DOMContentBuilderImpl extends CMVisitor implements DOMContentBuilder {
protected int buildPolicy = BUILD_ALL_CONTENT;
protected Hashtable propertyTable = new Hashtable();
protected boolean alwaysVisit = false;
protected List resultList;
protected Document document;
protected Node currentParent;
protected Node topParent;
protected Vector visitedCMElementDeclarationList = new Vector();
protected boolean attachNodesToParent = true;
protected NamespaceTable namespaceTable;
protected List namespaceInfoList;
protected Element rootElement; // this is used only teporarily via
// createDefaultRootContent
protected ExternalCMDocumentSupport externalCMDocumentSupport;
public boolean supressCreationOfDoctypeAndXMLDeclaration;
protected CMDataTypeValueHelper valueHelper = new CMDataTypeValueHelper();
protected int numOfRepeatableElements = 1;
protected Stack cmGroupStack = new Stack();
public interface ExternalCMDocumentSupport {
public CMDocument getCMDocument(Element element, String uri);
}
public void setExternalCMDocumentSupport(ExternalCMDocumentSupport externalCMDocumentSupport) {
this.externalCMDocumentSupport = externalCMDocumentSupport;
}
public DOMContentBuilderImpl(Document document) {
this.document = document;
namespaceTable = new NamespaceTable(document);
}
public void setBuildPolicy(int buildPolicy) {
this.buildPolicy = buildPolicy;
}
public int getBuildPolicy() {
return buildPolicy;
}
protected boolean buildAllContent(int policy) {
return (policy & BUILD_ALL_CONTENT) == BUILD_ALL_CONTENT;
}
protected boolean buildOptionalElements(int policy) {
return (policy & BUILD_OPTIONAL_ELEMENTS) == BUILD_OPTIONAL_ELEMENTS;
}
protected boolean buildOptionalAttributes(int policy) {
return (policy & BUILD_OPTIONAL_ATTRIBUTES) == BUILD_OPTIONAL_ATTRIBUTES;
}
protected boolean buildFirstChoice(int policy) {
return (policy & BUILD_FIRST_CHOICE) == BUILD_FIRST_CHOICE;
}
protected boolean buildTextNodes(int policy) {
return (policy & BUILD_TEXT_NODES) == BUILD_TEXT_NODES;
}
protected boolean buildFirstSubstitution(int policy) {
return (policy & BUILD_FIRST_SUBSTITUTION) == BUILD_FIRST_SUBSTITUTION;
}
public List getResult() {
return resultList;
}
public void setProperty(String propertyName, Object value) {
propertyTable.put(propertyName, value);
}
public Object getProperty(String propertyName) {
return propertyTable.get(propertyName);
}
public void build(Node parent, CMNode child) {
resultList = new Vector();
topParent = parent;
currentParent = parent;
if (parent instanceof Element) {
namespaceTable.addElementLineage((Element) parent);
}
attachNodesToParent = false;
alwaysVisit = true;
visitCMNode(child);
}
public void createDefaultRootContent(CMDocument cmDocument, CMElementDeclaration rootCMElementDeclaration, List namespaceInfoList) throws Exception {
this.namespaceInfoList = namespaceInfoList;
createDefaultRootContent(cmDocument, rootCMElementDeclaration);
}
public void createDefaultRootContent(CMDocument cmDocument, CMElementDeclaration rootCMElementDeclaration) throws Exception {
String grammarFileName = cmDocument.getNodeName();
if (!supressCreationOfDoctypeAndXMLDeclaration) {
// TODO cs... investigate to see if this code path is ever used,
// doesn't seem to be
// for now I'm setting the encoding to UTF-8 just incase this code
// path is used somewhere
//
String piValue = "version=\"1.0\""; //$NON-NLS-1$
String encoding = "UTF-8"; //$NON-NLS-1$
piValue += " encoding=\"" + encoding + "\""; //$NON-NLS-1$ //$NON-NLS-2$
ProcessingInstruction pi = document.createProcessingInstruction("xml", piValue); //$NON-NLS-1$
document.appendChild(pi);
// if we have a 'dtd' then add a DOCTYPE tag
//
if (grammarFileName != null && grammarFileName.endsWith("dtd")) //$NON-NLS-1$
{
DOMImplementation domImpl = document.getImplementation();
DocumentType documentType = domImpl.createDocumentType(rootCMElementDeclaration.getElementName(), grammarFileName, grammarFileName);
document.appendChild(documentType);
}
}
// if we have a schema add an xsi:schemaLocation attribute
//
if (grammarFileName != null && grammarFileName.endsWith("xsd") && namespaceInfoList != null) //$NON-NLS-1$
{
DOMNamespaceInfoManager manager = new DOMNamespaceInfoManager();
String name = rootCMElementDeclaration.getNodeName();
if (namespaceInfoList.size() > 0) {
NamespaceInfo info = (NamespaceInfo) namespaceInfoList.get(0);
if (info.prefix != null && info.prefix.length() > 0) {
name = info.prefix + ":" + name; //$NON-NLS-1$
}
}
rootElement = createElement(rootCMElementDeclaration, name, document);
manager.addNamespaceInfo(rootElement, namespaceInfoList, true);
}
createDefaultContent(document, rootCMElementDeclaration);
}
public void createDefaultContent(Node parent, CMElementDeclaration ed) {
currentParent = parent;
alwaysVisit = true;
visitCMElementDeclaration(ed);
}
public String computeName(CMNode cmNode, Node parent) {
String prefix = null;
return DOMNamespaceHelper.computeName(cmNode, parent, prefix, namespaceTable);
}
// overide the following 'create' methods to control how nodes are created
//
protected Element createElement(CMElementDeclaration ed, String name, Node parent) {
return document.createElement(name);
}
protected Attr createAttribute(CMAttributeDeclaration ad, String name, Node parent) {
return document.createAttribute(name);
}
protected Text createTextNode(CMDataType dataType, String value, Node parent) {
return document.createTextNode(value);
}
protected void handlePushParent(Element parent, CMElementDeclaration ed) {
}
protected void handlePopParent(Element element, CMElementDeclaration ed) {
}
// The range must be between 1 and 99.
public void setNumOfRepeatableElements(int i) {
numOfRepeatableElements = i;
}
protected int getNumOfRepeatableElements() {
return numOfRepeatableElements;
}
public void visitCMElementDeclaration(CMElementDeclaration ed) {
int forcedMin = (buildOptionalElements(buildPolicy) || alwaysVisit) ? 1 : 0;
int min = Math.max(ed.getMinOccur(), forcedMin);
// Correct the min value if the element is contained in
// a group.
if (!cmGroupStack.isEmpty()) {
CMGroup group = (CMGroup) cmGroupStack.peek();
int gmin = group.getMinOccur();
if (gmin == 0)
if (buildOptionalElements(buildPolicy)) {
/* do nothing: min = min */
}
else {
min = min * gmin; // min = 0
}
else {
min = min * gmin;
}
}
int max = Math.min(ed.getMaxOccur(), getNumOfRepeatableElements());
if (max < min)
max = min;
alwaysVisit = false;
// Note - ed may not be abstract but has substitutionGroups
// involved.
if (buildFirstSubstitution(buildPolicy) || isAbstract(ed)) // leave
// this
// for
// backward
// compatibility
// for now
{
// Note - To change so that if ed is optional, we do not
// generate anything here.
ed = getSubstitution(ed);
// Note - the returned ed may be an abstract element in
// which case the xml will be invalid.
}
if (min > 0 && !visitedCMElementDeclarationList.contains(ed)) {
visitedCMElementDeclarationList.add(ed);
for (int i = 1; i <= max; i++) {
// create an Element for each
Element element = null;
if (rootElement != null) {
element = rootElement;
rootElement = null;
}
else {
element = createElement(ed, computeName(ed, currentParent), currentParent);
}
// visit the children of the GrammarElement
Node oldParent = currentParent;
currentParent = element;
handlePushParent(element, ed);
namespaceTable.addElement(element);
boolean oldAttachNodesToParent = attachNodesToParent;
attachNodesToParent = true;
// instead of calling super.visitCMElementDeclaration()
// we duplicate the code with some minor modifications
CMNamedNodeMap nodeMap = ed.getAttributes();
int size = nodeMap.getLength();
for (int j = 0; j < size; j++) {
visitCMNode(nodeMap.item(j));
}
CMContent content = ed.getContent();
if (content != null) {
visitCMNode(content);
}
if (ed.getContentType() == CMElementDeclaration.PCDATA) {
CMDataType dataType = ed.getDataType();
if (dataType != null) {
visitCMDataType(dataType);
}
}
// end duplication
attachNodesToParent = oldAttachNodesToParent;
handlePopParent(element, ed);
currentParent = oldParent;
linkNode(element);
}
int size = visitedCMElementDeclarationList.size();
visitedCMElementDeclarationList.remove(size - 1);
}
}
public void visitCMDataType(CMDataType dataType) {
Text text = null;
String value = null;
// For backward compatibility:
// Previous code uses a property value but new one uses
// buildPolicy.
if (getProperty(PROPERTY_BUILD_BLANK_TEXT_NODES) != null && getProperty(PROPERTY_BUILD_BLANK_TEXT_NODES).equals("true")) //$NON-NLS-1$
buildPolicy = buildPolicy ^ BUILD_TEXT_NODES;
if (buildTextNodes(buildPolicy)) {
value = valueHelper.getValue(dataType);
if (value == null) {
if (currentParent != null && currentParent.getNodeType() == Node.ELEMENT_NODE) {
value = currentParent.getNodeName();
}
else {
value = "pcdata"; //$NON-NLS-1$
}
}
}
else {
value = ""; //$NON-NLS-1$
}
text = createTextNode(dataType, value, currentParent);
linkNode(text);
}
public void visitCMGroup(CMGroup e) {
cmGroupStack.push(e);
int forcedMin = (buildOptionalElements(buildPolicy) || alwaysVisit) ? 1 : 0;
int min = Math.max(e.getMinOccur(), forcedMin);
int max = 0;
if (e.getMaxOccur() == -1) // unbounded
max = getNumOfRepeatableElements();
else
max = Math.min(e.getMaxOccur(), getNumOfRepeatableElements());
if (max < min)
max = min;
alwaysVisit = false;
for (int i = 1; i <= max; i++) {
if (e.getOperator() == CMGroup.CHOICE && buildFirstChoice(buildPolicy)) {
CMNode hintNode = null;
// todo... the CMGroup should specify the hint... but it seems
// as though
// the Yamato guys are making the CMElement specify the hint.
// I do it that way for now until... we should fix this post
// GA
//
int listSize = visitedCMElementDeclarationList.size();
if (listSize > 0) {
CMElementDeclaration ed = (CMElementDeclaration) visitedCMElementDeclarationList.get(listSize - 1);
Object contentHint = ed.getProperty("contentHint"); //$NON-NLS-1$
if (contentHint instanceof CMNode) {
hintNode = (CMNode) contentHint;
}
}
// see if this hint corresponds to a valid choice
//
CMNode cmNode = null;
if (hintNode != null) {
CMNodeList nodeList = e.getChildNodes();
int nodeListLength = nodeList.getLength();
for (int j = 0; j < nodeListLength; j++) {
if (hintNode == nodeList.item(j)) {
cmNode = hintNode;
}
}
}
// if no cmNode has been determined from the hint, just use
// the first choice
//
if (cmNode == null) {
CMNodeList nodeList = e.getChildNodes();
if (nodeList.getLength() > 0) {
cmNode = nodeList.item(0);
}
}
if (cmNode != null) {
visitCMNode(cmNode);
}
}
else if (e.getOperator() == CMGroup.ALL // ALL
|| e.getOperator() == CMGroup.SEQUENCE) // SEQUENCE
{
// visit all of the content
super.visitCMGroup(e);
}
}
cmGroupStack.pop();
}
static int count = 0;
public void visitCMAttributeDeclaration(CMAttributeDeclaration ad) {
if (alwaysVisit || buildOptionalAttributes(buildPolicy) || ad.getUsage() == CMAttributeDeclaration.REQUIRED) {
alwaysVisit = false;
String name = computeName(ad, currentParent);
String value = valueHelper.getValue(ad, namespaceTable);
Attr attr = createAttribute(ad, name, currentParent);
attr.setValue(value != null ? value : ""); //$NON-NLS-1$
linkNode(attr);
}
}
protected boolean isAbstract(CMNode ed) {
boolean result = false;
if (ed != null) {
Object value = ed.getProperty("Abstract"); //$NON-NLS-1$
result = (value == Boolean.TRUE);
}
return result;
}
protected CMElementDeclaration getSubstitution(CMElementDeclaration ed) {
CMElementDeclaration result = ed;
CMNodeList l = (CMNodeList) ed.getProperty("SubstitutionGroup"); //$NON-NLS-1$
if (l != null) {
for (int i = 0; i < l.getLength(); i++) {
CMNode candidate = l.item(i);
if (!isAbstract(candidate) && (candidate instanceof CMElementDeclaration)) {
result = (CMElementDeclaration) candidate;
break;
}
}
}
return result;
}
protected CMElementDeclaration getParentCMElementDeclaration() {
CMElementDeclaration ed = null;
int listSize = visitedCMElementDeclarationList.size();
if (listSize > 0) {
ed = (CMElementDeclaration) visitedCMElementDeclarationList.get(listSize - 1);
}
return ed;
}
public void visitCMAnyElement(CMAnyElement anyElement) {
// ingnore buildPolicy for ANY elements... only create elements if
// absolutely needed
//
int forcedMin = alwaysVisit ? 1 : 0;
int min = Math.max(anyElement.getMinOccur(), forcedMin);
alwaysVisit = false;
String uri = anyElement.getNamespaceURI();
String targetNSProperty = "http://org.eclipse.wst/cm/properties/targetNamespaceURI"; //$NON-NLS-1$
CMDocument parentCMDocument = (CMDocument) anyElement.getProperty("CMDocument"); //$NON-NLS-1$
CMElementDeclaration ed = null;
// System.out.println("parentCMDocument = " + parentCMDocument);
// //$NON-NLS-1$
if (parentCMDocument != null) {
if (uri == null || uri.startsWith("##") || uri.equals(parentCMDocument.getProperty(targetNSProperty))) //$NON-NLS-1$
{
ed = getSuitableElement(getParentCMElementDeclaration(), parentCMDocument);
}
}
if (ed == null && externalCMDocumentSupport != null && uri != null && !uri.startsWith("##") && currentParent instanceof Element) //$NON-NLS-1$
{
CMDocument externalCMDocument = externalCMDocumentSupport.getCMDocument((Element) currentParent, uri);
if (externalCMDocument != null) {
ed = getSuitableElement(null, externalCMDocument);
}
}
for (int i = 1; i <= min; i++) {
if (ed != null) {
visitCMElementDeclaration(ed);
}
else {
Element element = document.createElement("ANY-ELEMENT"); //$NON-NLS-1$
linkNode(element);
}
}
}
protected CMElementDeclaration getSuitableElement(CMNamedNodeMap nameNodeMap) {
CMElementDeclaration result = null;
int size = nameNodeMap.getLength();
for (int i = 0; i < size; i++) {
CMElementDeclaration candidate = (CMElementDeclaration) nameNodeMap.item(i);
if (!visitedCMElementDeclarationList.contains(candidate)) {
result = candidate;
break;
}
}
return result;
}
protected CMElementDeclaration getSuitableElement(CMElementDeclaration ed, CMDocument cmDocument) {
CMElementDeclaration result = null;
if (ed != null) {
result = getSuitableElement(ed.getLocalElements());
}
if (result == null && cmDocument != null) {
result = getSuitableElement(cmDocument.getElements());
}
return result;
}
public void linkNode(Node node) {
if (attachNodesToParent && currentParent != null) {
if (node.getNodeType() == Node.ATTRIBUTE_NODE) {
((Element) currentParent).setAttributeNode((Attr) node);
}
else {
currentParent.appendChild(node);
}
}
else if (resultList != null) {
resultList.add(node);
}
}
public static void testPopulateDocumentFromGrammarFile(Document document, String grammarFileName, String rootElementName, boolean hack) {
try {
CMDocument cmDocument = ContentModelManager.getInstance().createCMDocument(grammarFileName, null);
CMNamedNodeMap elementMap = cmDocument.getElements();
CMElementDeclaration element = (CMElementDeclaration) elementMap.getNamedItem(rootElementName);
DOMContentBuilderImpl contentBuilder = new DOMContentBuilderImpl(document);
contentBuilder.supressCreationOfDoctypeAndXMLDeclaration = hack;
contentBuilder.createDefaultRootContent(cmDocument, element);
System.out.println();
System.out.println("-----------------------------"); //$NON-NLS-1$
DOMWriter writer = new DOMWriter();
if (hack) {
writer.print(document, grammarFileName);
}
else {
writer.print(document);
}
System.out.println("-----------------------------"); //$NON-NLS-1$
}
catch (Exception e) {
System.out.println("Error: " + e); //$NON-NLS-1$
e.printStackTrace();
}
}
// test
//
/*
* public static void main(String arg[]) { if (arg.length >= 2) { try {
* CMDocumentFactoryRegistry.getInstance().registerCMDocumentBuilderWithClassName("org.eclipse.wst.xml.core.internal.contentmodel.mofimpl.CMDocumentBuilderImpl");
*
* String grammarFileName = arg[0]; String rootElementName = arg[1];
*
* Document document =
* (Document)Class.forName("org.apache.xerces.dom.DocumentImpl").newInstance();
* testPopulateDocumentFromGrammarFile(document, grammarFileName,
* rootElementName, true); } catch (Exception e) {
* System.out.println("DOMContentBuilderImpl error"); e.printStackTrace(); } }
* else { System.out.println("Usage : java
* org.eclipse.wst.xml.util.DOMContentBuildingCMVisitor grammarFileName
* rootElementName"); } }
*/
}