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

org.ditchnet.xml.dom.DomUtils Maven / Gradle / Ivy

/*
 * The contents of this file are subject to the GNU Lesser General Public
 * License Version 2.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.gnu.org/copyleft/lesser.html
 *
 * 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.
 *
 * Developer:
 * Todd Ditchendorf, [email protected]
 *
 */

/**
 *	@author Todd Ditchendorf
 *	@since 2005-03-13
 *
 */
package org.ditchnet.xml.dom;

import java.util.List;
import java.util.Iterator;
import java.util.ArrayList;
import org.w3c.dom.*;
import org.ditchnet.xml.Xhtml;

/**
 *	A Utility class that may not be instanciated (private constructor) that
 *	contains static utility methods that ease DOM traversal pain.
 *	@author Todd Ditchendorf
 *
 *
 */
public class DomUtils {
	
	/**
	 *	Private constructor -- class may not be instanciated.
	 */
	private DomUtils() { }
	
	/**
	 *	Test Element param to see if it's HTML id
	 *	attribute matches id.
	 *	@param	target	The Element to test.
	 *	@param	id		Check target for this value in it's HTML
	 *					id attribute.
	 */
	public static boolean hasId(final Element target,final String id) {
		return target.getAttribute(Xhtml.Attr.ID.toString()).equals(id);
	}
	
	/**
	 *	Test the target param to see if it's HTML class
	 *	attribute contains className. Keep in mind that the HTML
	 *	4.01 spec allows an element's HTML class attribute value
	 *	to consist of multiple space-separated values. This method takes that
	 *	fact into account, and tests target's HTML class attribute
	 *	for the existence of className anywhere in the string.

* *

So for example, say you had an org.w3c.dom.Element * reference to the following HTML element in a variable named * target:

*

<div class="ditch-tab-pane ditch-focused"></div> *

*

note that this attribute value is taking advantage of the * ability to list multiple space separated values.

*

DomUtils.hasClassName(target,"ditch-tab-pane");

*

Would return true.

*

DomUtils.hasClassName(target,"ditch-tab");

*

Would return false.

* * @param target The Element to test. * @param className Check target for this value in it's HTML * class attribute. */ //* @throws IllegalArgumentException public static boolean hasClassName(final Element target, String className) { className = className.trim(); /*if (className.indexOf(" ") > -1) { throw new IllegalArgumentException("Whitespace is not allowed " + "in the className arg. hasClassName() accepts only one " + "class name at a time"); }*/ String cn = target.getAttribute(Xhtml.Attr.CLASS.toString()); if (null == cn || 0 == cn.length()) { return false; } cn = cn.trim(); if (cn.equals(className)) { return true; } if (cn.indexOf(className + " ") > -1) { return true; } if (_isLastOfMultpleClassNames(cn,className)) { return true; } return false; } private static boolean _isLastOfMultpleClassNames(final String all, final String className) { int spaceBefore = all.lastIndexOf(className)-1; return all.endsWith(className) && all.substring(spaceBefore,spaceBefore+1).equals(" "); } /** * Tests to see if target's getNodeType() * method returns Node.ELEMENT_NODE. * */ public static boolean isElementNode(final Node target) { return Node.ELEMENT_NODE == target.getNodeType(); } /** * Search up the DOM tree for the first ancestor element of * target with an HTML class attribute value of * className. */ public static Element getFirstAncestorByClassName(final Node target, final String className) { Element parent = (Element)target; while ((parent = (Element)parent.getParentNode()) != null) { if (hasClassName(parent,className)) { return parent; } } return null; } /** * Search up the DOM tree for the first ancestor element of * target with an HTML class attribute value of * className. */ public static Element getFirstAncestorOrSelfByClassName( final Node target, final String className) { Element parent = (Element)target; do { if (hasClassName(parent,className)) { return parent; } } while ((parent = (Element)parent.getParentNode()) != null); return null; } /** * Search target's children for the first element node * with an HTML class attribute value of * className. */ public static Element getFirstChildByClassName( final Node target, final String className) { Node childNode; for (int i = 0; i < target.getChildNodes().getLength(); i++) { childNode = target.getChildNodes().item(i); if ( Node.ELEMENT_NODE != childNode.getNodeType() ) { continue; } if (hasClassName(((Element)childNode),className)) { return (Element)childNode; } } return null; } /** * Retreives target's first descendant element with an * HTML class attribute value that includes * className using a breadth-first algorithm. */ public static Element getFirstDescendantByClassNameBreadthFirst( final Node target, final String className) { Element result; if ((result = getFirstChildByClassName(target,className)) != null) { return result; } for (int i = 0; i < target.getChildNodes().getLength(); i++) { result = getFirstDescendantByClassNameBreadthFirst( target.getChildNodes().item(i), className ); if (null != result) { return result; } } return null; } /** * Retreives target's first descendant element with an * HTML class attribute value that includes * className using a depth-first algorithm. */ public static Element getFirstDescendantByClassNameDepthFirst( final Node target, final String className) { Node child; Element result; for (int i = 0; i < target.getChildNodes().getLength(); i++) { child = target.getChildNodes().item(i); if (isElementNode(child) && hasClassName(((Element)child),className)) { return (Element)child; } result = getFirstDescendantByClassNameDepthFirst( target.getChildNodes().item(i), className ); if (null != result) { return result; } } return null; } /** * Returns all child elements of target with HTML * class attribute values that contain className * as an array of type {@link java.util.List}. NOTE that unlike the * algorithms specified in the DOM spec, a NodeList is NOT * returned. This method searches only one level deep... * target's child nodes. */ public static List getChildrenByClassName(final Node target, final String className) { List result = new ArrayList(); Node child; for (int i = 0; i < target.getChildNodes().getLength(); i++) { child = target.getChildNodes().item(i); if (isElementNode(child) && hasClassName(((Element)child),className)) { result.add(child); } } return result; } /** * Returns all descendant elements of target with HTML * class attribute values that contain className * as an array of type {@link java.util.List}. NOTE that unlike the * algorithms specified in the DOM spec, a NodeList is NOT * returned. This method searched for all descendants of target * meeting the criteria using a breadth-first algorithm. */ public static List getDescendantsByClassName(final Node target, final String className) { List result = new ArrayList(); result.addAll(getChildrenByClassName(target,className)); for (int i = 0; i < target.getChildNodes().getLength(); i++) { result.addAll(getDescendantsByClassName( target.getChildNodes().item(i), className)); } return result; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy