/hudson-core/src/main/java/hudson/MarkupText.java
http://github.com/hudson/hudson · Java · 334 lines · 170 code · 44 blank · 120 comment · 23 complexity · cc9960cf9b36b9560bbab8358fc65017 MD5 · raw file
- /*
- * The MIT License
- *
- * Copyright (c) 2004-2010, Sun Microsystems, Inc., Kohsuke Kawaguchi
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- */
- package hudson;
- import java.util.ArrayList;
- import java.util.Collections;
- import java.util.List;
- import java.util.regex.Matcher;
- import java.util.regex.Pattern;
- /**
- * Mutable representation of string with HTML mark up.
- *
- * <p>
- * This class is used to put mark up on plain text.
- * See <a href="https://github.com/hudson/hudson/blob/master/hudson-core/src/test/java/hudson/MarkupTextTest.java">
- * the test code</a> for a typical usage and its result.
- *
- * @author Kohsuke Kawaguchi
- * @since 1.70
- */
- public class MarkupText extends AbstractMarkupText {
- private final String text;
- /**
- * Added mark up tags.
- */
- private final List<Tag> tags = new ArrayList<Tag>();
- /**
- * Represents one mark up inserted into text.
- */
- private static final class Tag implements Comparable<Tag> {
- /**
- * Char position of this tag in {@link MarkupText#text}.
- * This tag is placed in front of the character of this index.
- */
- private final int pos;
- private final String markup;
- public Tag(int pos, String markup) {
- this.pos = pos;
- this.markup = markup;
- }
- public int compareTo(Tag that) {
- return this.pos-that.pos;
- }
- }
- /**
- * Represents a substring of a {@link MarkupText}.
- */
- public final class SubText extends AbstractMarkupText {
- private final int start,end;
- private final int[] groups;
- public SubText(Matcher m, int textOffset) {
- start = m.start() + textOffset;
- end = m.end() + textOffset;
- int cnt = m.groupCount();
- groups = new int[cnt*2];
- for( int i=0; i<cnt; i++ ) {
- groups[i*2 ] = m.start(i+1) + textOffset;
- groups[i*2+1] = m.end(i+1) + textOffset;
- }
- }
- public SubText(int start, int end) {
- this.start = start;
- this.end = end;
- groups = new int[0];
- }
- @Override
- public SubText subText(int start, int end) {
- return MarkupText.this.subText(this.start+start,
- end<0 ? this.end+1+end : this.start+end);
- }
- @Override
- public String getText() {
- return text.substring(start,end);
- }
- @Override
- public void addMarkup(int startPos, int endPos, String startTag, String endTag) {
- MarkupText.this.addMarkup(startPos+start, endPos+start, startTag, endTag);
- }
- /**
- * Surrounds this subtext with the specified start tag and the end tag.
- *
- * <p>
- * Start/end tag text can contain special tokens "$0", "$1", ...
- * and they will be replaced by their {@link #group(int) group match}.
- * "\$" can be used to escape characters.
- */
- public void surroundWith(String startTag, String endTag) {
- addMarkup(0,length(),replace(startTag),replace(endTag));
- }
- /**
- * Works like {@link #surroundWith(String, String)} except
- * that the token replacement is not performed on parameters.
- */
- public void surroundWithLiteral(String startTag, String endTag) {
- addMarkup(0,length(),startTag,endTag);
- }
- /**
- * Surrounds this subtext with <a>...</a>.
- */
- public void href(String url) {
- addHyperlink(0,length(),url);
- }
- /**
- * Gets the start index of the captured group within {@link MarkupText#getText()}.
- *
- * @param groupIndex
- * 0 means the start of the whole subtext. 1, 2, ... are
- * groups captured by '(...)' in the regexp.
- */
- public int start(int groupIndex) {
- if(groupIndex==0) return start;
- return groups[groupIndex*2-2];
- }
- /**
- * Gets the start index of this subtext within {@link MarkupText#getText()}.
- */
- public int start() {
- return start;
- }
- /**
- * Gets the end index of the captured group within {@link MarkupText#getText()}.
- */
- public int end(int groupIndex) {
- if(groupIndex==0) return end;
- return groups[groupIndex*2-1];
- }
- /**
- * Gets the end index of this subtext within {@link MarkupText#getText()}.
- */
- public int end() {
- return end;
- }
- /**
- * Gets the text that represents the captured group.
- */
- public String group(int groupIndex) {
- if(start(groupIndex)==-1)
- return null;
- return text.substring(start(groupIndex),end(groupIndex));
- }
- /**
- * How many captured groups are in this subtext.
- * @since 1.357
- */
- public int groupCount() {
- return groups.length / 2;
- }
- /**
- * Replaces the group tokens like "$0", "$1", and etc with their actual matches.
- */
- public String replace(String s) {
- StringBuffer buf = new StringBuffer();
- for( int i=0; i<s.length(); i++) {
- char ch = s.charAt(i);
- if (ch == '\\') {// escape char
- i++;
- buf.append(s.charAt(i));
- } else if (ch == '$') {// replace by group
- i++;
- ch = s.charAt(i);
- // get the group number
- int groupId = ch - '0';
- if (groupId < 0 || groupId > 9) {
- buf.append('$').append(ch);
- } else {
- // add the group text
- String group = group(groupId);
- if (group != null)
- buf.append(group);
- }
- } else {
- // other chars
- buf.append(ch);
- }
- }
- return buf.toString();
- }
- @Override
- protected SubText createSubText(Matcher m) {
- return new SubText(m,start);
- }
- }
- /**
- *
- * @param text
- * Plain text. This shouldn't include any markup nor escape. Those are done later in {@link #toString(boolean)}.
- */
- public MarkupText(String text) {
- this.text = text;
- }
- @Override
- public String getText() {
- return text;
- }
- /**
- * Returns a subtext.
- *
- * @param end
- * If negative, -N means "trim the last N-1 chars". That is, (s,-1) is the same as (s,length)
- */
- public SubText subText(int start, int end) {
- return new SubText(start, end<0 ? text.length()+1+end : end);
- }
- @Override
- public void addMarkup( int startPos, int endPos, String startTag, String endTag ) {
- rangeCheck(startPos);
- rangeCheck(endPos);
- if(startPos>endPos) throw new IndexOutOfBoundsException();
- // when multiple tags are added to the same range, we want them to show up like
- // <b><i>abc</i></b>, not <b><i>abc</b></i>. Also, we'd like <b>abc</b><i>def</i>,
- // not <b>abc<i></b>def</i>. Do this by inserting them to different places.
- tags.add(new Tag(startPos, startTag));
- tags.add(0,new Tag(endPos,endTag));
- }
- public void addMarkup(int pos, String tag) {
- rangeCheck(pos);
- tags.add(new Tag(pos,tag));
- }
- private void rangeCheck(int pos) {
- if(pos<0 || pos>text.length())
- throw new IndexOutOfBoundsException();
- }
- /**
- * Returns the fully marked-up text.
- *
- * @deprecated as of 1.350.
- * Use {@link #toString(boolean)} to be explicit about the escape mode.
- */
- @Override
- public String toString() {
- return toString(false);
- }
- /**
- * Returns the fully marked-up text.
- *
- * @param preEscape
- * If true, the escaping is for the <PRE> context. This leave SP and CR/LF intact.
- * If false, the escape is for the normal HTML, thus SP becomes &nbsp; and CR/LF becomes <BR>
- */
- public String toString(boolean preEscape) {
- if(tags.isEmpty())
- return preEscape? Util.xmlEscape(text) : Util.escape(text); // the most common case
- Collections.sort(tags);
- StringBuilder buf = new StringBuilder();
- int copied = 0; // # of chars already copied from text to buf
- for (Tag tag : tags) {
- if (copied<tag.pos) {
- String portion = text.substring(copied, tag.pos);
- buf.append(preEscape ? Util.xmlEscape(portion) : Util.escape(portion));
- copied = tag.pos;
- }
- buf.append(tag.markup);
- }
- if (copied<text.length()) {
- String portion = text.substring(copied, text.length());
- buf.append(preEscape ? Util.xmlEscape(portion) : Util.escape(portion));
- }
- return buf.toString();
- }
- // perhaps this method doesn't need to be here to remain binary compatible with past versions,
- // but having this seems to be safer.
- @Override
- public List<SubText> findTokens(Pattern pattern) {
- return super.findTokens(pattern);
- }
- @Override
- protected SubText createSubText(Matcher m) {
- return new SubText(m,0);
- }
- }