All Downloads are FREE. Search and download functionalities are using the official Maven repository.

de.fau.cs.osr.ptk.common.jxpath.AstNodeFieldPointer Maven / Gradle / Ivy

/**
 * Copyright 2011 The Open Source Research Group,
 *                University of Erlangen-Nürnberg
 *
 * Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package de.fau.cs.osr.ptk.common.jxpath;

import org.apache.commons.jxpath.ri.Compiler;
import org.apache.commons.jxpath.ri.QName;
import org.apache.commons.jxpath.ri.compiler.NodeNameTest;
import org.apache.commons.jxpath.ri.compiler.NodeTest;
import org.apache.commons.jxpath.ri.compiler.NodeTypeTest;
import org.apache.commons.jxpath.ri.model.NodeIterator;
import org.apache.commons.jxpath.ri.model.NodePointer;

import de.fau.cs.osr.ptk.common.ast.AstNode;
import de.fau.cs.osr.utils.FmtInternalLogicError;

public class AstNodeFieldPointer
		extends
			NodePointer
{
	private static final long serialVersionUID = 1L;

	/** This pointer points to a field of this node. */
	private AstNode node;

	/** The index of the field in `node' this pointer points to. */
	private int fieldIndex;

	// =========================================================================

	public AstNodeFieldPointer(
			NodePointer parent,
			AstNode node,
			int index)
	{
		// Only called by an AstNodeFieldIterator.
		// The `parent' pointer points to `node'. Index refers to the field
		// in `node'.

		super(parent);
		this.node = node;
		this.fieldIndex = index;
		//debug("AstNodeFieldPointer");
	}

	// =========================================================================

	@Override
	public NodeIterator attributeIterator(QName name)
	{
		//debug("attributeIterator");

		/* This looks like a bug!
		// Shortcut ...
		if (((AstNodeInterface) getImmediateNode()).hasAttributes())
			return null;
		*/

		return new AstPropertyIterator(this, name);
	}

	@Override
	public NodeIterator childIterator(
			NodeTest test,
			boolean reverse,
			NodePointer startWith)
	{
		//debug("childIterator");

		AstNode pointee = (AstNode) getImmediateNode();

		// Shortcut ...
		if (isLeaf())
		{
			//throw new FormattedInternalLogicError("This should not happen!?");
			// But it does :(
			return null;
		}

		if (pointee.isNodeType(AstNode.NT_NODE_LIST))
		{
			return new AstNodeFieldIterator(
					this,
					test,
					reverse,
					startWith);
		}
		else
		{
			return new AstNodeFieldIterator(
					new AstNodePointer(this, pointee),
					test,
					reverse,
					startWith);
		}
	}

	@Override
	public boolean isLeaf()
	{
		AstNode pointee = (AstNode) getImmediateNode();
		return pointee == null || pointee.isEmpty();
	}

	@Override
	public boolean isActual()
	{
		return true;
	}

	@Override
	public boolean isCollection()
	{
		return false;
	}

	@Override
	public int getLength()
	{
		return 1;
	}

	@Override
	public QName getName()
	{
		// Since we don't handle namespaces, we probably should return 
		// `null' as namespace prefix in names.
		return new QName(null, getSimpleName());
	}

	@Override
	public Object getBaseValue()
	{
		return getImmediateNode();
	}

	@Override
	public Object getImmediateNode()
	{
		return node.get(fieldIndex);
	}

	@Override
	public void setValue(Object value)
	{
		throw new UnsupportedOperationException("Not supported yet.");
	}

	@Override
	public int compareChildNodePointers(
			NodePointer pointer1,
			NodePointer pointer2)
	{
		//debug("compareChildNodePointers");

		if (pointer1.getBaseValue() == pointer2.getBaseValue())
			return 0;

		boolean p1IsProp = pointer1 instanceof AstPropertyPointer;
		boolean p2IsProp = pointer2 instanceof AstPropertyPointer;
		if (p1IsProp && !p2IsProp)
		{
			return -1;
		}
		else if (!p1IsProp && p2IsProp)
		{
			return 1;
		}
		else if (p1IsProp && p2IsProp)
		{
			AstPropertyPointer p1 = (AstPropertyPointer) pointer1;
			AstPropertyPointer p2 = (AstPropertyPointer) pointer2;

			int cmp = p1.getName2().compareTo(p2.getName2());
			if (cmp == 0)
				throw new FmtInternalLogicError();

			return cmp;
		}
		else
		{
			AstNode node1 = (AstNode) pointer1.getBaseValue();
			AstNode node2 = (AstNode) pointer2.getBaseValue();

			for (AstNode child : (AstNode) getImmediateNode())
			{
				if (child == node1)
				{
					return -1;
				}
				else if (child == node2)
				{
					return 1;
				}
			}

			throw new FmtInternalLogicError();
		}
	}

	// =========================================================================

	@Override
	public int hashCode()
	{
		return getImmediateNode().hashCode();
	}

	@Override
	public boolean equals(Object obj)
	{
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		AstNodeFieldPointer other = (AstNodeFieldPointer) obj;
		if (getImmediateNode() != other.getImmediateNode())
			return false;
		return true;
	}

	// =========================================================================

	@Override
	public String asPath()
	{
		StringBuffer buffer = new StringBuffer();
		if (parent != null)
			buffer.append(parent.asPath());

		// If the parent pointer is not one of our pointer types, it is the 
		// parent's responsibility to produce the node test part of the path
		if (parent instanceof AstNodePointer || parent instanceof AstNodeFieldPointer)
		{
			if (buffer.length() == 0 || buffer.charAt(buffer.length() - 1) != '/')
				buffer.append('/');

			buffer.append(getSimpleName());
			if (parent != null)
			{
				buffer.append('[');
				buffer.append(getRelativePosition());
				buffer.append(']');
			}
		}

		return buffer.toString();
	}

	private int getRelativePosition()
	{
		if (parent != null)
		{
			if (parent instanceof AstNodePointer || parent instanceof AstNodeFieldPointer)
			{
				NodePointer p = (NodePointer) parent;

				AstNode thisNode = (AstNode) getImmediateNode();

				int i = 1;
				for (AstNode n : (AstNode) p.getImmediateNode())
				{
					if (n == thisNode)
						return i;
					++i;
				}

				throw new FmtInternalLogicError();
			}
			/*
			else if (parent instanceof AstNodeFieldPointer)
			{
				AstNodeFieldPointer p = (AstNodeFieldPointer) parent;
				
				int i = 1;
				for (AstNodeInterface n : (AstNodeInterface) p.getImmediateNode())
				{
					if (n == node)
						return i;
					++i;
				}
				
				throw new FormattedInternalLogicError();
			}
			*/
			else
				return 1;
		}
		else
			return 1;
	}

	// =========================================================================

	@Override
	public boolean testNode(NodeTest test)
	{
		//debug("testNode", test);

		if (test == null)
		{
			return true;
		}
		else if (test instanceof NodeNameTest)
		{
			NodeNameTest nodeNameTest = (NodeNameTest) test;

			QName testName = nodeNameTest.getNodeName();

			// We don't support prefixes
			if (testName.getPrefix() != null)
				return false;

			// Wildcards don't match when prefixes differ, so check this after 
			// we made sure that we are not looking for a prefixed name.
			if (nodeNameTest.isWildcard())
				return true;

			// Perform actual name check
			String s1 = getSimpleName();
			String s2 = testName.getName();

			// s1 cannot be `null'
			return s1 == s2 || s1.equals(s2);
		}
		else if (test instanceof NodeTypeTest)
		{
			NodeTypeTest nodeTypeTest = (NodeTypeTest) test;
			switch (nodeTypeTest.getNodeType())
			{
				case Compiler.NODE_TYPE_NODE:
					return true;

				case Compiler.NODE_TYPE_TEXT:
				case Compiler.NODE_TYPE_COMMENT:
				case Compiler.NODE_TYPE_PI:
				default:
					return false;
			}
		}
		else
		{
			// Unhandled: ProcessingInstructionTest
			//   An AstNodeInterface can never be a PI.

			return false;
		}
	}

	// =========================================================================

	private String getSimpleName()
	{
		return node.getChildNames()[fieldIndex];
	}

	// =========================================================================

	/*
	private void debug(String where, Object... params)
	{
		StringBuilder b = new StringBuilder();
		
		String path = asPath();
		b.append(String.format(
		        "@%8x : %s : %s%s.%s ; ",
		        System.identityHashCode(this),
		        path,
		        StringUtils.strrep(' ', 28 - path.length()),
		        getClass().getSimpleName(),
		        where));
		
		int i = 0;
		for (Object o : params)
		{
			if (i++ != 0)
				b.append(", ");
			b.append(o == null ? "null" : o.toString());
		}
		
		System.out.println(b.toString());
	}
	*/
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy