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

au.id.jericho.lib.html.Segment Maven / Gradle / Ivy

Go to download

Jericho HTML Parser is a java library allowing analysis and manipulation of parts of an HTML document, including server-side tags, while reproducing verbatim any unrecognised or invalid HTML.

There is a newer version: 3.4
Show newest version
// Jericho HTML Parser - Java based library for analysing and manipulating HTML
// Version 2.6
// Copyright (C) 2007 Martin Jericho
// http://jerichohtml.sourceforge.net/
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of either one of the following licences:
//
// 1. The Eclipse Public License (EPL) version 1.0,
// included in this distribution in the file licence-epl-1.0.html
// or available at http://www.eclipse.org/legal/epl-v10.html
//
// 2. The GNU Lesser General Public License (LGPL) version 2.1 or later,
// included in this distribution in the file licence-lgpl-2.1.txt
// or available at http://www.gnu.org/licenses/lgpl.txt
//
// This library is distributed on an "AS IS" basis,
// WITHOUT WARRANTY OF ANY KIND, either express or implied.
// See the individual licence texts for more details.

package au.id.jericho.lib.html;

import java.util.*;

/**
 * Represents a segment of a {@link Source} document.
 * 

* Many of the tag search methods are defined in this class. *

* The span of a segment is defined by the combination of its begin and end character positions. */ public class Segment implements Comparable, CharSequence { final int begin; final int end; final Source source; List childElements=null; private static final char[] WHITESPACE={' ','\n','\r','\t','\f','\u200B'}; // see comments in isWhiteSpace(char) method /** * Constructs a new Segment within the specified {@linkplain Source source} document with the specified begin and end character positions. * @param source the {@link Source} document, must not be null. * @param begin the character position in the source where this segment {@linkplain #getBegin() begins}, inclusive. * @param end the character position in the source where this segment {@linkplain #getEnd() ends}, exclusive. */ public Segment(final Source source, final int begin, final int end) { if (begin==-1 || end==-1 || begin>end) throw new IllegalArgumentException(); this.begin=begin; this.end=end; if (source==null) throw new IllegalArgumentException("source argument must not be null"); this.source=source; } // Only called from Source constructor Segment(final int length) { begin=0; this.end=length; source=(Source)this; } // Only used for creating dummy flag instances of this type (see Tag.NOT_CACHED and Element.NOT_CACHED) Segment() { begin=0; end=0; source=null; } /** * Returns the {@link Source} document containing this segment. * @return the {@link Source} document containing this segment. */ public final Source getSource() { return source; } /** * Returns the character position in the {@link Source} document at which this segment begins, inclusive. * @return the character position in the {@link Source} document at which this segment begins, inclusive. */ public final int getBegin() { return begin; } /** * Returns the character position in the {@link Source} document immediately after the end of this segment. *

* The character at the position specified by this property is not included in the segment. * * @return the character position in the {@link Source} document immediately after the end of this segment. */ public final int getEnd() { return end; } /** * Compares the specified object with this Segment for equality. *

* Returns true if and only if the specified object is also a Segment, * and both segments have the same {@link Source}, and the same begin and end positions. * @param object the object to be compared for equality with this Segment. * @return true if the specified object is equal to this Segment, otherwise false. */ public final boolean equals(final Object object) { if (this==object) return true; if (object==null || !(object instanceof Segment)) return false; final Segment segment=(Segment)object; return segment.begin==begin && segment.end==end && segment.source==source; } /** * Returns a hash code value for the segment. *

* The current implementation returns the sum of the begin and end positions, although this is not * guaranteed in future versions. * * @return a hash code value for the segment. */ public int hashCode() { return begin+end; } /** * Returns the length of the segment. * This is defined as the number of characters between the begin and end positions. * @return the length of the segment. */ public final int length() { return end-begin; } /** * Indicates whether this Segment encloses the specified Segment. *

* This is the case if {@link #getBegin()}<=segment.{@link #getBegin()} && {@link #getEnd()}>=segment.{@link #getEnd()}. * * @param segment the segment to be tested for being enclosed by this segment. * @return true if this Segment encloses the specified Segment, otherwise false. */ public final boolean encloses(final Segment segment) { return begin<=segment.begin && end>=segment.end; } /** * Indicates whether this segment encloses the specified character position in the source document. *

* This is the case if {@link #getBegin()} <= pos < {@link #getEnd()}. * * @param pos the position in the {@link Source} document. * @return true if this segment encloses the specified character position in the source document, otherwise false. */ public final boolean encloses(final int pos) { return begin<=pos && posString. *

* The returned String is newly created with every call to this method, unless this * segment is itself an instance of {@link Source}. *

* Note that before version 2.0 this returned a representation of this object useful for debugging purposes, * which can now be obtained via the {@link #getDebugInfo()} method. * * @return the source text of this segment as a String. */ public String toString() { return source.string.substring(begin,end).toString(); } /** * Performs a simple rendering of the HTML markup in this segment into text. *

* The output can be configured by setting any number of properties on the returned {@link Renderer} instance before * {@linkplain Renderer#writeTo(Writer) obtaining its output}. * * @return an instance of {@link Renderer} based on this segment. * @see #getTextExtractor() */ public Renderer getRenderer() { return new Renderer(this); } /** * Extracts the textual content from the HTML markup of this segment. *

* The output can be configured by setting properties on the returned {@link TextExtractor} instance before * {@linkplain TextExtractor#writeTo(Writer) obtaining its output}. *

* @return an instance of {@link TextExtractor} based on this segment. * @see #getRenderer() */ public TextExtractor getTextExtractor() { return new TextExtractor(this); } /** * Returns an iterator over every tag and text segment contained within this segment. *

* Every tag found in the {@link #findAllTags()} list is included in this iterator, including all {@linkplain TagType#isServerTag() server tags}. *

* Segments of the document between the tags are also included, resulting in a sequential walk-through of every "node" in this segment, where a node is either * a tag or a segment of text. * The {@linkplain #getEnd() end} position of each segment should correspond with the {@linkplain #getBegin() begin} position of the subsequent segment, * unless any of the tags are enclosed by other tags, which is common when {@linkplain TagType#isServerTag() server tags} are present. *

* The {@link CharacterReference#decodeCollapseWhiteSpace(CharSequence)} method can be used to retrieve the text from each text segment. *

*

*
Example:
*
*

* The following code demonstrates the typical usage of this method. *

*

	 * for (Iterator nodeIterator=segment.getNoteIterator(); nodeIterator.hasNext();) {
	 *   Segment nodeSegment=(Segment)nodeIterator.next();
	 *   if (nodeSegment instanceof Tag) {
	 *     Tag tag=(Tag)nodeSegment;
	 *     if (tag.getTagType().isServerTag()) continue; // ignore server tags
	 *     // Process the tag (just output it in this example):
	 *     System.out.println(tag.tidy());
	 *   } else {
	 *     // Segment is a text segment.
	 *     // Process the text segment (just output its text in this example):
	 *     String text=CharacterReference.decodeCollapseWhiteSpace(nodeSegment);
	 *     System.out.println(text);
	 *   }
	 * }
*
*
* @return an iterator over every tag and text segment contained within this segment. */ public Iterator getNodeIterator() { return new NodeIterator(this); } /** * Returns a list of all {@link Tag} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The {@link Source#fullSequentialParse()} method should be called after construction of the {@link Source} object * if this method is to be used on a large proportion of the source. * It is called automatically if this method is called on the {@link Source} object itself. *

* See the {@link Tag} class documentation for more details about the behaviour of this method. * * @return a list of all {@link Tag} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllTags() { return findAllTags(null); } /** * Returns a list of all {@link Tag} objects of the specified {@linkplain TagType type} that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* See the {@link Tag} class documentation for more details about the behaviour of this method. *

* Specifying a null argument to the tagType parameter is equivalent to {@link #findAllTags()}. * * @param tagType the {@linkplain TagType type} of tags to find. * @return a list of all {@link Tag} objects of the specified {@linkplain TagType type} that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllTags(final TagType tagType) { Tag tag=checkEnclosure(Tag.findNextTag(source,begin,tagType)); if (tag==null) return Collections.EMPTY_LIST; final ArrayList list=new ArrayList(); do { list.add(tag); tag=checkEnclosure(tagType==null ? tag.findNextTag() : Tag.findNextTag(source,tag.begin+1,tagType)); // tag.findNext() may be more efficient than Tag.findNextTag(source,tag.begin+1,null) due to caching } while (tag!=null); return list; } /** * Returns a list of all {@link StartTag} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The {@link Source#fullSequentialParse()} method should be called after construction of the {@link Source} object * if this method is to be used on a large proportion of the source. * It is called automatically if this method is called on the {@link Source} object itself. *

* See the {@link Tag} class documentation for more details about the behaviour of this method. * * @return a list of all {@link StartTag} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllStartTags() { return findAllStartTags(null); } /** * Returns a list of all {@linkplain StartTagType#NORMAL normal} {@link StartTag} objects with the specified {@linkplain StartTag#getName() name} that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* See the {@link Tag} class documentation for more details about the behaviour of this method. *

* Specifying a null argument to the name parameter is equivalent to {@link #findAllStartTags()}. *

* This method also returns {@linkplain Tag#isUnregistered() unregistered} tags if the specified name is not a valid {@linkplain Tag#isXMLName(CharSequence) XML tag name}. * * @param name the {@linkplain StartTag#getName() name} of the start tags to find. * @return a list of all {@linkplain StartTagType#NORMAL normal} {@link StartTag} objects with the specified {@linkplain StartTag#getName() name} that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllStartTags(String name) { if (name!=null) name=name.toLowerCase(); final boolean isXMLTagName=Tag.isXMLName(name); StartTag startTag=(StartTag)checkEnclosure(StartTag.findNext(source,begin,name,StartTagType.NORMAL,isXMLTagName)); if (startTag==null) return Collections.EMPTY_LIST; final ArrayList list=new ArrayList(); do { list.add(startTag); startTag=(StartTag)checkEnclosure(StartTag.findNext(source,startTag.begin+1,name,StartTagType.NORMAL,isXMLTagName)); } while (startTag!=null); return list; } /** * Returns a list of all {@link StartTag} objects with the specified attribute name/value pair * that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* See the {@link Tag} class documentation for more details about the behaviour of this method. * * @param attributeName the attribute name (case insensitive) to search for, must not be null. * @param value the value of the specified attribute to search for, must not be null. * @param valueCaseSensitive specifies whether the attribute value matching is case sensitive. * @return a list of all {@link StartTag} objects with the specified attribute name/value pair that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllStartTags(final String attributeName, final String value, final boolean valueCaseSensitive) { StartTag startTag=(StartTag)checkEnclosure(source.findNextStartTag(begin,attributeName,value,valueCaseSensitive)); if (startTag==null) return Collections.EMPTY_LIST; final ArrayList list=new ArrayList(); do { list.add(startTag); startTag=(StartTag)checkEnclosure(source.findNextStartTag(startTag.begin+1,attributeName,value,valueCaseSensitive)); } while (startTag!=null); return list; } /** * Returns a list of the immediate children of this segment in the document element hierarchy. *

* The returned list may include an element that extends beyond the end of this segment, as long as it begins within this segment. *

* An element found at the start of this segment is included in the list. * Note however that if this segment is an {@link Element}, the overriding {@link Element#getChildElements()} method is called instead, * which only returns the children of the element. *

* Calling getChildElements() on an Element is usually more efficient than calling it on a Segment. *

* The objects in the list are all of type {@link Element}. *

* The {@link Source#fullSequentialParse()} method should be called after construction of the {@link Source} object * if this method is to be used on a large proportion of the source. * It is called automatically if this method is called on the {@link Source} object itself. *

* See the {@link Source#getChildElements()} method for more details. * * @return the a list of the immediate children of this segment in the document element hierarchy, guaranteed not null. * @see Element#getParentElement() */ public List getChildElements() { if (childElements==null) { if (length()==0) { childElements=Collections.EMPTY_LIST; } else { childElements=new ArrayList(); int pos=begin; while (true) { final StartTag childStartTag=source.findNextStartTag(pos); if (childStartTag==null || childStartTag.begin>=end) break; if (!Config.IncludeServerTagsInElementHierarchy && childStartTag.getTagType().isServerTag()) { pos=childStartTag.end; continue; } final Element childElement=childStartTag.getElement(); childElements.add(childElement); childElement.getChildElements(); pos=childElement.end; } } } return childElements; } /** * Returns a list of all {@link Element} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The {@link Source#fullSequentialParse()} method should be called after construction of the {@link Source} object * if this method is to be used on a large proportion of the source. * It is called automatically if this method is called on the {@link Source} object itself. *

* The elements returned correspond exactly with the start tags returned in the {@link #findAllStartTags()} method. * * @return a list of all {@link Element} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllElements() { return findAllElements((String)null); } /** * Returns a list of all {@link Element} objects with the specified name that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The elements returned correspond exactly with the start tags returned in the {@link #findAllStartTags(String name)} method. *

* Specifying a null argument to the name parameter is equivalent to {@link #findAllElements()}. *

* This method also returns elements consisting of {@linkplain Tag#isUnregistered() unregistered} tags if the specified name is not a valid {@linkplain Tag#isXMLName(CharSequence) XML tag name}. * * @param name the {@linkplain Element#getName() name} of the elements to find. * @return a list of all {@link Element} objects with the specified name that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllElements(String name) { if (name!=null) name=name.toLowerCase(); final List startTags=findAllStartTags(name); if (startTags.isEmpty()) return Collections.EMPTY_LIST; final ArrayList elements=new ArrayList(startTags.size()); for (final Iterator i=startTags.iterator(); i.hasNext();) { final StartTag startTag=(StartTag)i.next(); final Element element=startTag.getElement(); if (element.end>end) break; elements.add(element); } return elements; } /** * Returns a list of all {@link Element} objects with start tags of the specified {@linkplain StartTagType type} that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The elements returned correspond exactly with the start tags returned in the {@link #findAllTags(TagType)} method. * * @param startTagType the {@linkplain StartTagType type} of start tags to find, must not be null. * @return a list of all {@link Element} objects with start tags of the specified {@linkplain StartTagType type} that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllElements(final StartTagType startTagType) { final List startTags=findAllTags(startTagType); if (startTags.isEmpty()) return Collections.EMPTY_LIST; final ArrayList elements=new ArrayList(startTags.size()); for (final Iterator i=startTags.iterator(); i.hasNext();) { final StartTag startTag=(StartTag)i.next(); final Element element=startTag.getElement(); if (element.end>end) break; elements.add(element); } return elements; } /** * Returns a list of all {@link Element} objects with the specified attribute name/value pair * that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* The elements returned correspond exactly with the start tags returned in the {@link #findAllStartTags(String attributeName, String value, boolean valueCaseSensitive)} method. * * @param attributeName the attribute name (case insensitive) to search for, must not be null. * @param value the value of the specified attribute to search for, must not be null. * @param valueCaseSensitive specifies whether the attribute value matching is case sensitive. * @return a list of all {@link Element} objects with the specified attribute name/value pair that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllElements(final String attributeName, final String value, final boolean valueCaseSensitive) { final List startTags=findAllStartTags(attributeName,value,valueCaseSensitive); if (startTags.isEmpty()) return Collections.EMPTY_LIST; final ArrayList elements=new ArrayList(startTags.size()); for (final Iterator i=startTags.iterator(); i.hasNext();) { final StartTag startTag=(StartTag)i.next(); final Element element=startTag.getElement(); if (element.end>end) break; elements.add(element); } return elements; } /** * Returns a list of all {@link CharacterReference} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. * @return a list of all {@link CharacterReference} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findAllCharacterReferences() { CharacterReference characterReference=findNextCharacterReference(begin); if (characterReference==null) return Collections.EMPTY_LIST; final ArrayList list=new ArrayList(); do { list.add(characterReference); characterReference=findNextCharacterReference(characterReference.end); } while (characterReference!=null); return list; } /** * Returns a list of the {@link FormControl} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. * @return a list of the {@link FormControl} objects that are {@linkplain #encloses(Segment) enclosed} by this segment. */ public List findFormControls() { return FormControl.findAll(this); } /** * Returns the {@link FormFields} object representing all form fields that are {@linkplain #encloses(Segment) enclosed} by this segment. *

* This is equivalent to {@link FormFields#FormFields(Collection) new FormFields}({@link #findFormControls()}). * * @return the {@link FormFields} object representing all form fields that are {@linkplain #encloses(Segment) enclosed} by this segment. * @see #findFormControls() */ public FormFields findFormFields() { return new FormFields(findFormControls()); } /** * Parses any {@link Attributes} within this segment. * This method is only used in the unusual situation where attributes exist outside of a start tag. * The {@link StartTag#getAttributes()} method should be used in normal situations. *

* This is equivalent to source.{@link Source#parseAttributes(int,int) parseAttributes}({@link #getBegin()},{@link #getEnd()}). * * @return the {@link Attributes} within this segment, or null if too many errors occur while parsing. */ public Attributes parseAttributes() { return source.parseAttributes(begin,end); } /** * Causes the this segment to be ignored when parsing. *

* Ignored segments are treated as blank spaces by the parsing mechanism, but are included as normal text in all other functions. *

* This method was originally the only means of preventing {@linkplain TagType#isServerTag() server tags} located inside * {@linkplain StartTagType#NORMAL normal} tags from interfering with the parsing of the tags * (such as where an {@linkplain Attribute attribute} of a normal tag uses a server tag to dynamically set its value), * as well as preventing non-server tags from being recognised inside server tags. *

* As of version 2.4 it is no longer necessary to use this method to ignore {@linkplain TagType#isServerTag() server tags} located inside normal tags, * as the attributes parser now automatically ignores any server tags. *

* As of version 2.5 it is no longer necessary to use this method to ignore the contents of {@link HTMLElementName#SCRIPT SCRIPT} elements, * as the parser now does this automatically when performing a {@linkplain Source#fullSequentialParse() full sequential parse}. *

* As of version 2.6 is is no longer necessary to use this method to ignore non-server tags inside server tags, * as the parser now does this automatically when performing a {@linkplain Source#fullSequentialParse() full sequential parse}. *

* This leaves only very few scenarios where calling this method still provides a significant benefit. *

* One such case is where XML-style server tags are used inside {@linkplain StartTagType#NORMAL normal} tags. * Here is an example using an XML-style JSP tag: *

<a href="<i18n:resource path="/Portal"/>?BACK=TRUE">back</a>
* The first double-quote of "/Portal" will be interpreted as the end quote for the href attribute, * as there is no way for the parser to recognise the il8n:resource element as a server tag. * Such use of XML-style server tags inside {@linkplain StartTagType#NORMAL normal} tags is generally seen as bad practice, * but it is nevertheless valid JSP. The only way to ensure that this library is able to parse the normal tag surrounding it is to * find these server tags first and call the ignoreWhenParsing method to ignore them before parsing the rest of the document. *

* It is important to understand the difference between ignoring the segment when parsing and removing the segment completely. * Any text inside a segment that is ignored when parsing is treated by most functions as content, and as such is included in the output of * tools such as {@link TextExtractor} and {@link Renderer}. *

* To remove segments completely, create an {@link OutputDocument} and call its {@link OutputDocument#remove(Segment) remove(Segment)} or * {@link OutputDocument#replaceWithSpaces(int,int) replaceWithSpaces(int begin, int end)} method for each segment. * Then create a new source document using {@link Source#Source(CharSequence) new Source(outputDocument.toString())} * and perform the desired operations on this new source object. *

* Calling this method after the {@link Source#fullSequentialParse()} method has been called is not permitted and throws an IllegalStateException. *

* Any tags appearing in this segment that are found before this method is called will remain in the {@linkplain Source#getCacheDebugInfo() tag cache}, * and so will continue to be found by the tag search methods. * If this is undesirable, the {@link Source#clearCache()} method can be called to remove them from the cache. * Calling the {@link Source#fullSequentialParse()} method after this method clears the cache automatically. *

* For best performance, this method should be called on all segments that need to be ignored without calling * any of the tag search methods in between. * * @see Source#ignoreWhenParsing(Collection segments) */ public void ignoreWhenParsing() { source.ignoreWhenParsing(begin,end); } /** * Compares this Segment object to another object. *

* If the argument is not a Segment, a ClassCastException is thrown. *

* A segment is considered to be before another segment if its begin position is earlier, * or in the case that both segments begin at the same position, its end position is earlier. *

* Segments that begin and end at the same position are considered equal for * the purposes of this comparison, even if they relate to different source documents. *

* Note: this class has a natural ordering that is inconsistent with equals. * This means that this method may return zero in some cases where calling the * {@link #equals(Object)} method with the same argument returns false. * * @param o the segment to be compared * @return a negative integer, zero, or a positive integer as this segment is before, equal to, or after the specified segment. * @throws ClassCastException if the argument is not a Segment */ public int compareTo(final Object o) { if (this==o) return 0; final Segment segment=(Segment)o; if (beginsegment.begin) return 1; if (endsegment.end) return 1; return 0; } /** * Indicates whether this segment consists entirely of {@linkplain #isWhiteSpace(char) white space}. * @return true if this segment consists entirely of {@linkplain #isWhiteSpace(char) white space}, otherwise false. */ public final boolean isWhiteSpace() { for (int i=begin; iwhite space. *

* The HTML 4.01 specification section 9.1 * specifies the following white space characters: *

    *
  • space (U+0020) *
  • tab (U+0009) *
  • form feed (U+000C) *
  • line feed (U+000A) *
  • carriage return (U+000D) *
  • zero-width space (U+200B) *
*

* Despite the explicit inclusion of the zero-width space in the HTML specification, Microsoft IE6 does not * recognise them as white space and renders them as an unprintable character (empty square). * Even zero-width spaces included using the numeric character reference &#x200B; are rendered this way. * * @param ch the character to test. * @return true if the specified character is white space, otherwise false. */ public static final boolean isWhiteSpace(final char ch) { for (int i=0; i * This is logically equivalent to toString().charAt(index) * for valid argument values 0 <= index < length(). *

* However because this implementation works directly on the underlying document source string, * it should not be assumed that an IndexOutOfBoundsException is thrown * for an invalid argument value. * * @param index the index of the character. * @return the character at the specified index. */ public final char charAt(final int index) { return source.string.charAt(begin+index); } /** * Returns a new character sequence that is a subsequence of this sequence. *

* This is logically equivalent to toString().subSequence(beginIndex,endIndex) * for valid values of beginIndex and endIndex. *

* However because this implementation works directly on the underlying document source string, * it should not be assumed that an IndexOutOfBoundsException is thrown * for invalid argument values as described in the String.subSequence(int,int) method. * * @param beginIndex the begin index, inclusive. * @param endIndex the end index, exclusive. * @return a new character sequence that is a subsequence of this sequence. */ public final CharSequence subSequence(final int beginIndex, final int endIndex) { return source.string.subSequence(begin+beginIndex,begin+endIndex); } /** * Extracts the textual content from the HTML markup of this segment. *

* This method has been deprecated as of version 2.4 and replaced with the {@link #getTextExtractor()} method. * * @return the textual content from the HTML markup of this segment. * @deprecated Use {@link #getTextExtractor()}.{@link TextExtractor#toString() toString()} instead. */ public String extractText() { return new TextExtractor(this).toString(); } /** * Extracts the textual content from the HTML markup of this segment. *

* This method has been deprecated as of version 2.4 and replaced with the {@link #getTextExtractor()} method. * * @param includeAttributes specifies whether the values of title, alt, label, and summary attributes are included in the output. * @return the textual content from the HTML markup of this segment. * @deprecated Use {@link #getTextExtractor()}.{@link TextExtractor#setIncludeAttributes(boolean) setIncludeAttributes(includeAttributes)}.{@link TextExtractor#toString() toString()} instead. */ public String extractText(final boolean includeAttributes) { return new TextExtractor(this).setIncludeAttributes(includeAttributes).toString(); } /** * Collapses the {@linkplain #isWhiteSpace(char) white space} in the specified text. * All leading and trailing white space is omitted, and any sections of internal white space are replaced by a single space. */ static final StringBuffer appendCollapseWhiteSpace(final StringBuffer sb, final CharSequence text) { final int textLength=text.length(); int i=0; boolean lastWasWhiteSpace=false; while (true) { if (i>=textLength) return sb; if (!isWhiteSpace(text.charAt(i))) break; i++; } do { final char ch=text.charAt(i++); if (isWhiteSpace(ch)) { lastWasWhiteSpace=true; } else { if (lastWasWhiteSpace) { sb.append(' '); lastWasWhiteSpace=false; } sb.append(ch); } } while (iend) return null; return tag; } private CharacterReference findNextCharacterReference(final int pos) { final CharacterReference characterReference=source.findNextCharacterReference(pos); if (characterReference==null || characterReference.end>end) return null; return characterReference; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy