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

com.helger.xml.microdom.IMicroNode Maven / Gradle / Ivy

/**
 * Copyright (C) 2014-2016 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * 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 com.helger.xml.microdom;

import java.io.Serializable;
import java.util.function.Predicate;

import javax.annotation.Nonnegative;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.Nonempty;
import com.helger.commons.annotation.ReturnsMutableCopy;
import com.helger.commons.collection.ext.CommonsArrayList;
import com.helger.commons.collection.ext.ICommonsList;
import com.helger.commons.collection.ext.ICommonsMap;
import com.helger.commons.collection.ext.ICommonsSet;
import com.helger.commons.hierarchy.IHasChildrenRecursive;
import com.helger.commons.hierarchy.IHasChildrenSorted;
import com.helger.commons.hierarchy.IHasParent;
import com.helger.commons.lang.ICloneable;
import com.helger.commons.state.EChange;

/**
 * This is the base interface for all kind of nodes in the micro document object
 * model.
 *
 * @author Philip Helger
 */
public interface IMicroNode extends
                            ICloneable ,
                            IHasChildrenSorted ,
                            IHasChildrenRecursive ,
                            IHasParent ,
                            Serializable
{
  /**
   * @return Just an abstract name that depends on the implementing class. For
   *         {@link IMicroElement} nodes this is the same as the tag name.
   */
  @Nonnull
  @Nonempty
  String getNodeName ();

  /**
   * @return The value of this node. This depends on the concrete implementation
   *         class. It is currently implemented for {@link IMicroText},
   *         {@link IMicroComment} and {@link IMicroEntityReference}.
   */
  default String getNodeValue ()
  {
    return "";
  }

  /**
   * Get a list of all direct child nodes.
   *
   * @return May be null if the node has no children.
   */
  @Nullable
  ICommonsList  getAllChildren ();

  /**
   * Check if any direct child matching the provided filter is contained.
   *
   * @param aFilter
   *        The filter that is applied to all child nodes. May not be
   *        null.
   * @return true if any child matching the provided filter is
   *         contained, false otherwise.
   */
  default boolean containsAnyChild (@Nonnull final Predicate  aFilter)
  {
    ValueEnforcer.notNull (aFilter, "Filter");
    if (hasNoChildren ())
      return false;
    return getAllChildren ().containsAny (aFilter);
  }

  /**
   * @return The first child node of this node, or null if this
   *         node has no children.
   */
  @Nullable
  IMicroNode getFirstChild ();

  /**
   * @return The last child node of this node, or null if this node
   *         has no children.
   */
  @Nullable
  IMicroNode getLastChild ();

  /**
   * Recursively get all children. Micro container are contained in this list
   * (incl. their children of course)
   *
   * @return A list containing all recursively contained child elements. May be
   *         null if this node has no children.
   */
  @Nullable
  default ICommonsList  getAllChildrenRecursive ()
  {
    if (hasNoChildren ())
      return null;

    final ICommonsList  ret = new CommonsArrayList<> ();
    forAllChildrenRecursive (aNode -> ret.add (aNode));
    return ret;
  }

  /**
   * @return The previous node on the same level as this node, or
   *         null if this node has no preceding siblings.
   */
  @Nullable
  IMicroNode getPreviousSibling ();

  /**
   * @return The next node on the same level as this node, or null
   *         if this node has no succeeding siblings.
   */
  @Nullable
  IMicroNode getNextSibling ();

  /**
   * @return true if this node has a parent node assigned,
   *         false otherwise.
   */
  boolean hasParent ();

  /**
   * @return May be null.
   */
  @Nullable
  IMicroNode getParent ();

  /**
   * Detach this node from the parent node so it can be inserted into another
   * node without problems. Otherwise you would get an
   * {@link IllegalStateException} if adding this node again to another parent
   * since each node can only have one parent.
   *
   * @return this
   */
  @Nonnull
  IMicroNode detachFromParent ();

  @Nullable
  IMicroElement getParentElementWithName (@Nullable String sTagName);

  @Nullable
  IMicroElement getParentElementWithName (@Nullable String sNamespaceURI, @Nullable String sTagName);

  /**
   * Append any child to the node.
   *
   * @param 
   *        Parameter type == return type
   * @param aChildNode
   *        The child node to append. May be null.
   * @return The appended node, or null if the parameter was
   *         null.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nullable
   NODETYPE appendChild (@Nullable NODETYPE aChildNode) throws MicroException;

  /**
   * Insert an existing node before a certain child node of this.
   *
   * @param 
   *        Parameter type == return type
   * @param aChildNode
   *        The new child node to be inserted.
   * @param aSuccessor
   *        The node before which the new node will be inserted.
   * @return The newly inserted node
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nullable
   NODETYPE insertBefore (@Nullable NODETYPE aChildNode,
                                                       @Nonnull IMicroNode aSuccessor) throws MicroException;

  /**
   * Insert an existing node after a certain child node of this.
   *
   * @param 
   *        Parameter type == return type
   * @param aChildNode
   *        The new child node to be inserted.
   * @param aPredecessor
   *        The node after which the new node will be inserted.
   * @return The newly inserted node
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nullable
   NODETYPE insertAfter (@Nullable NODETYPE aChildNode,
                                                      @Nonnull IMicroNode aPredecessor) throws MicroException;

  /**
   * Insert an existing node as a child at the specified index.
   *
   * @param 
   *        Parameter type == return type
   * @param nIndex
   *        The index to insert. Must be ≥ 0.
   * @param aChildNode
   *        The new child node to be inserted.
   * @return The newly inserted node
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nullable
   NODETYPE insertAtIndex (@Nonnegative int nIndex,
                                                        @Nullable NODETYPE aChildNode) throws MicroException;

  /**
   * Append a text node to this node.
   *
   * @param sText
   *        text to be added
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroText appendText (@Nullable CharSequence sText) throws MicroException;

  /**
   * Append a text node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  default IMicroText appendText (@Nonnull final char [] aChars) throws MicroException
  {
    return appendText (aChars, 0, aChars.length);
  }

  /**
   * Append a text node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @param nOfs
   *        Offset into the array where to start copying data. May not be <
   *        0.
   * @param nLen
   *        Number of bytes to take from the array. May not be < 0.
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroText appendText (@Nonnull char [] aChars, @Nonnegative int nOfs, @Nonnegative int nLen) throws MicroException;

  /**
   * Append a text node to this node. If the type of the value is not
   * {@link String}, the {@link com.helger.commons.typeconvert.TypeConverter} is
   * invoked to convert it to a {@link String} object.
   *
   * @param aValue
   *        text to be added
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroText appendTextWithConversion (@Nullable Object aValue) throws MicroException;

  /**
   * Append a text node which is ignorable whitespace content to this node.
   *
   * @param sText
   *        The whitespace content to be added.
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroText appendIgnorableWhitespaceText (@Nullable CharSequence sText) throws MicroException;

  /**
   * Append a text node which is ignorable whitespace content to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  default IMicroText appendIgnorableWhitespaceText (@Nonnull final char [] aChars) throws MicroException
  {
    return appendIgnorableWhitespaceText (aChars, 0, aChars.length);
  }

  /**
   * Append a text node which is ignorable whitespace content to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @param nOfs
   *        Offset into the array where to start copying data. May not be <
   *        0.
   * @param nLen
   *        Number of bytes to take from the array. May not be < 0.
   * @return The created text node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroText appendIgnorableWhitespaceText (@Nonnull char [] aChars,
                                            @Nonnegative int nOfs,
                                            @Nonnegative int nLen) throws MicroException;

  /**
   * Append a CDATA node to this node.
   *
   * @param sText
   *        CDATA text
   * @return The created CDATA node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroCDATA appendCDATA (@Nullable CharSequence sText) throws MicroException;

  /**
   * Append a CDATA node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @return The created CDATA node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  default IMicroCDATA appendCDATA (@Nonnull final char [] aChars) throws MicroException
  {
    return appendCDATA (aChars, 0, aChars.length);
  }

  /**
   * Append a CDATA node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @param nOfs
   *        Offset into the array where to start copying data. May not be <
   *        0.
   * @param nLen
   *        Number of bytes to take from the array. May not be < 0.
   * @return The created CDATA node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroCDATA appendCDATA (@Nonnull char [] aChars, @Nonnegative int nOfs, @Nonnegative int nLen) throws MicroException;

  /**
   * Append a CDATA node to this node. If the type of the value is not
   * {@link String}, the {@link com.helger.commons.typeconvert.TypeConverter} is
   * invoked to convert it to a {@link String} object.
   *
   * @param aValue
   *        CDATA to be added
   * @return The created CDATA node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroCDATA appendCDATAWithConversion (@Nullable Object aValue) throws MicroException;

  /**
   * Append a comment node to this node.
   *
   * @param sText
   *        comment text
   * @return The created comment.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroComment appendComment (@Nullable CharSequence sText) throws MicroException;

  /**
   * Append a comment node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @return The created comment.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  default IMicroComment appendComment (@Nonnull final char [] aChars) throws MicroException
  {
    return appendComment (aChars, 0, aChars.length);
  }

  /**
   * Append a comment node to this node.
   *
   * @param aChars
   *        Characters to append. May not be null
   * @param nOfs
   *        Offset into the array where to start copying data. May not be <
   *        0.
   * @param nLen
   *        Number of bytes to take from the array. May not be < 0.
   * @return The created comment.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroComment appendComment (@Nonnull char [] aChars,
                               @Nonnegative int nOfs,
                               @Nonnegative int nLen) throws MicroException;

  /**
   * Append a comment node to this node. If the type of the value is not
   * {@link String}, the {@link com.helger.commons.typeconvert.TypeConverter} is
   * invoked to convert it to a {@link String} object.
   *
   * @param aValue
   *        Comment to be added
   * @return The created comment node.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroComment appendCommentWithConversion (@Nullable Object aValue) throws MicroException;

  /**
   * Append an entity reference to this node.
   *
   * @param sName
   *        The name of the entity reference.
   * @return The created entity reference.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroEntityReference appendEntityReference (@Nonnull String sName) throws MicroException;

  /**
   * Append an element without namespace to this node.
   *
   * @param sTagName
   *        Element name to be created. May neither be null nor
   *        empty.
   * @return The created element
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroElement appendElement (@Nonnull @Nonempty String sTagName) throws MicroException;

  /**
   * Append an element with namespace to this node.
   *
   * @param sNamespaceURI
   *        Namespace URI to use. May be null.
   * @param sTagName
   *        Element name to be created. May neither be null nor
   *        empty.
   * @return The created element
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroElement appendElement (@Nullable String sNamespaceURI,
                               @Nonnull @Nonempty String sTagName) throws MicroException;

  /**
   * Append a processing instruction to this node.
   *
   * @param sTarget
   *        The PI target
   * @param sData
   *        The PI data
   * @return The created element
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroProcessingInstruction appendProcessingInstruction (@Nonnull @Nonempty String sTarget,
                                                           @Nullable String sData) throws MicroException;

  /**
   * Append a new container to this node
   *
   * @return The created container.
   * @throws MicroException
   *         if this node cannot have children
   */
  @Nonnull
  IMicroContainer appendContainer () throws MicroException;

  /**
   * Remove the passed child.
   *
   * @param aChild
   *        The child to be removed. May not be null.
   * @return {@link EChange#CHANGED} if the child was successfully removed,
   *         {@link EChange#UNCHANGED} otherwise.
   */
  @Nonnull
  EChange removeChild (@Nonnull IMicroNode aChild);

  /**
   * Remove the child not at the specified index.
   *
   * @param nIndex
   *        The 0-based index of the item to be removed.
   * @return {@link EChange#CHANGED} if the node was successfully removed,
   *         {@link EChange#UNCHANGED} otherwise.
   */
  @Nonnull
  EChange removeChildAtIndex (@Nonnegative int nIndex);

  /**
   * Remove all children from this node.
   *
   * @return {@link EChange#CHANGED} if at least one child was present, and was
   *         successfully removed, {@link EChange#UNCHANGED} otherwise.
   */
  @Nonnull
  EChange removeAllChildren ();

  /**
   * Replace the passed old child with the new child.
   *
   * @param aOldChild
   *        The child to be removed. May not be null.
   * @param aNewChild
   *        The child to be inserted instead. May not be null.
   * @return {@link EChange#CHANGED} if the child was successfully replaced,
   *         {@link EChange#UNCHANGED} if old child and new child are identical.
   */
  @Nonnull
  EChange replaceChild (@Nonnull IMicroNode aOldChild, @Nonnull IMicroNode aNewChild);

  /**
   * @return The node type. Never null.
   */
  @Nonnull
  EMicroNodeType getType ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroDocument}.
   */
  boolean isDocument ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroDocumentType}.
   */
  boolean isDocumentType ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroText}.
   */
  boolean isText ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroCDATA}.
   */
  boolean isCDATA ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroComment}.
   */
  boolean isComment ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroEntityReference}.
   */
  boolean isEntityReference ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroElement}.
   */
  boolean isElement ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroProcessingInstruction}.
   */
  boolean isProcessingInstruction ();

  /**
   * @return true if this node can safely be casted to
   *         {@link IMicroContainer}.
   */
  boolean isContainer ();

  /**
   * Register a specific MicroDOM event listener. One event listener can only be
   * attached once to an event!
   *
   * @param eEventType
   *        The event type. May not be null.
   * @param aTarget
   *        The event target to be added. May not be null.
   * @return {@link EChange#CHANGED} if the event listener was registered,
   *         {@link EChange#UNCHANGED} otherwise.
   */
  @Nonnull
  EChange registerEventTarget (@Nonnull EMicroEvent eEventType, @Nonnull IMicroEventTarget aTarget);

  /**
   * Unregister a specific MicroDOM event listener.
   *
   * @param eEventType
   *        The event type. May not be null.
   * @param aTarget
   *        The event target to be added. May not be null.
   * @return {@link EChange#CHANGED} if the event listener was unregistered,
   *         {@link EChange#UNCHANGED} otherwise.
   */
  @Nonnull
  EChange unregisterEventTarget (@Nonnull EMicroEvent eEventType, @Nonnull IMicroEventTarget aTarget);

  /**
   * @return A map of all registered event targets. Never null.
   */
  @Nonnull
  @ReturnsMutableCopy
  ICommonsMap > getAllEventTargets ();

  /**
   * Get all event targets for a certain event.
   *
   * @param eEvent
   *        The event to be queried. May be null.
   * @return A map of all registered event targets. Never null.
   */
  @Nonnull
  @ReturnsMutableCopy
  ICommonsSet  getAllEventTargets (@Nullable EMicroEvent eEvent);

  /**
   * As instances of this class may not implement equals/hashCode we need a way
   * to determine, if 2 nodes are equal by content.
   *
   * @param aNode
   *        The node to compare to this.
   * @return true if the nodes are of the same type and the same
   *         content, false otherwise.
   */
  boolean isEqualContent (@Nullable IMicroNode aNode);
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy