/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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 org.apache.commons.jxpath.ri.model.beans; import java.util.Locale; import org.apache.commons.jxpath.JXPathInvalidAccessException; 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 org.apache.commons.jxpath.util.ValueUtils; /** * A pointer describing a node that has properties, each of which could be * a collection. * * @author Dmitri Plotnikov * @version $Revision: 652845 $ $Date: 2008-05-02 12:46:46 -0500 (Fri, 02 May 2008) $ */ public abstract class PropertyOwnerPointer extends NodePointer { private static final Object UNINITIALIZED = new Object(); private Object value = UNINITIALIZED; public NodeIterator childIterator(NodeTest test, boolean reverse, NodePointer startWith) { if (test == null) { return createNodeIterator(null, reverse, startWith); } if (test instanceof NodeNameTest) { NodeNameTest nodeNameTest = (NodeNameTest) test; QName testName = nodeNameTest.getNodeName(); if (isValidProperty(testName)) { return createNodeIterator(nodeNameTest.isWildcard() ? null : testName.toString(), reverse, startWith); } return null; } return test instanceof NodeTypeTest && ((NodeTypeTest) test).getNodeType() == Compiler.NODE_TYPE_NODE ? createNodeIterator(null, reverse, startWith) : null; } /** * Create a NodeIterator. * @param property property name * @param reverse whether to iterate in reverse * @param startWith first pointer to return * @return NodeIterator */ public NodeIterator createNodeIterator(String property, boolean reverse, NodePointer startWith) { return new PropertyIterator(this, property, reverse, startWith); } public NodeIterator attributeIterator(QName name) { return new BeanAttributeIterator(this, name); } /** * Create a new PropertyOwnerPointer. * @param parent parent pointer * @param locale Locale */ protected PropertyOwnerPointer(NodePointer parent, Locale locale) { super(parent, locale); } /** * Create a new PropertyOwnerPointer. * @param parent pointer */ protected PropertyOwnerPointer(NodePointer parent) { super(parent); } public void setIndex(int index) { if (this.index != index) { super.setIndex(index); value = UNINITIALIZED; } } public Object getImmediateNode() { if (value == UNINITIALIZED) { value = index == WHOLE_COLLECTION ? ValueUtils.getValue(getBaseValue()) : ValueUtils.getValue(getBaseValue(), index); } return value; } public abstract QName getName(); /** * Learn whether <code>name</code> is a valid child name for this PropertyOwnerPointer. * @param name the QName to test * @return <code>true</code> if <code>QName</code> is a valid property name. * @since JXPath 1.3 */ public boolean isValidProperty(QName name) { return isDefaultNamespace(name.getPrefix()); } /** * Throws an exception if you try to change the root element, otherwise * forwards the call to the parent pointer. * @param value to set */ public void setValue(Object value) { this.value = value; if (parent != null) { if (parent.isContainer()) { parent.setValue(value); } else { if (index == WHOLE_COLLECTION) { throw new UnsupportedOperationException( "Cannot setValue of an object that is not " + "some other object's property"); } throw new JXPathInvalidAccessException( "The specified collection element does not exist: " + this); } } else { throw new UnsupportedOperationException( "Cannot replace the root object"); } } /** * If this is a root node pointer, throws an exception; otherwise * forwards the call to the parent node. */ public void remove() { this.value = null; if (parent != null) { parent.remove(); } else { throw new UnsupportedOperationException( "Cannot remove an object that is not " + "some other object's property or a collection element"); } } /** * Get a PropertyPointer for this PropertyOwnerPointer. * @return PropertyPointer */ public abstract PropertyPointer getPropertyPointer(); /** * Learn whether dynamic property declaration is supported. * @return true if the property owner can set a property "does not exist". * A good example is a Map. You can always assign a value to any * key even if it has never been "declared". */ public boolean isDynamicPropertyDeclarationSupported() { return false; } public int compareChildNodePointers(NodePointer pointer1, NodePointer pointer2) { int r = pointer1.getName().toString().compareTo(pointer2.getName().toString()); return r == 0 ? pointer1.getIndex() - pointer2.getIndex() : r; } }