org.spdx.licenseTemplate.HtmlTemplateOutputHandler Maven / Gradle / Ivy
/**
* Copyright (c) 2013 Source Auditor Inc.
*
* 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 org.spdx.licenseTemplate;
/**
* License template output handler for generating an HTML version of a license from a license template.
* Used when parsing a license template.
* @author Gary O'Neall
*
*/
public class HtmlTemplateOutputHandler implements ILicenseTemplateOutputHandler {
public static final String REPLACEABLE_LICENSE_TEXT_CLASS = "replacable-license-text";
public static final String OPTIONAL_LICENSE_TEXT_CLASS = "optional-license-text";
static final String END_PARAGRAPH_TAG = "";
private static final String STARTS_WITH_LETTER_REGEX = "[A-Za-z].*";
StringBuilder htmlString = new StringBuilder();
int optionalNestLevel = 0;
boolean movingParagraph = false; // true if we need to move the end of a
/* (non-Javadoc)
* @see org.spdx.licenseTemplate.ILicenseTemplateOutputHandler#text(java.lang.String)
*/
@Override
public void text(String text) {
if (optionalNestLevel > 0) {
htmlString.append((SpdxLicenseTemplateHelper.formatEscapeHTML(text, false)));
} else {
htmlString.append(SpdxLicenseTemplateHelper.formatEscapeHTML(text, this.movingParagraph));
this.movingParagraph = false;
}
}
/* (non-Javadoc)
* @see org.spdx.licenseTemplate.ILicenseTemplateOutputHandler#variableRule(org.spdx.licenseTemplate.LicenseTemplateRule)
*/
@Override
public void variableRule(LicenseTemplateRule rule) {
removeEndParagraphTag();
htmlString.append(formatReplaceabledHTML(rule.getOriginal(), rule.getName()));
}
/**
* Adds back an end paragraph tag if it was removed
*/
@SuppressWarnings("unused")
private void addRemovedEndParagraphTag() {
// this is a bit of a hack to deal with the formatting of HTML
// in normal text inserting end paragraphs at the end of the string
// which then creates line breaks. We need to move the end paragraph
// tags to after the rule generated text.
if (this.movingParagraph) {
if (endsInEndParagraph()) {
this.htmlString.append(END_PARAGRAPH_TAG);
}
this.movingParagraph = false;
}
}
private boolean endsInEndParagraph() {
int lastEndParagraph = this.htmlString.lastIndexOf(END_PARAGRAPH_TAG);
return (lastEndParagraph == this.htmlString.length()-END_PARAGRAPH_TAG.length()) ;
}
/**
* If the current htmlString ends with an HTML end paragraph tag, remove it
* and set the flag so that it can be put back when addRemovedEndParagraphTag
* is called.
*/
private void removeEndParagraphTag() {
if (endsInEndParagraph()) {
this.movingParagraph = true;
this.htmlString.delete(this.htmlString.length()-END_PARAGRAPH_TAG.length(), this.htmlString.length());
}
}
/**
* Format HTML for a replaceable string
* @param text text for the optional license string
* @param id ID used for the div
* @return
*/
public static String formatReplaceabledHTML(String text, String id) {
StringBuilder sb = new StringBuilder();
sb.append("\n");
sb.append(SpdxLicenseTemplateHelper.formatEscapeHTML(text));
sb.append("\n");
return sb.toString();
}
/**
* Escape the ID string to conform to the legal characters for an HTML ID string
* @param id
* @return
*/
public static String escapeIdString(String id) {
String retval = id;
if (!retval.matches(STARTS_WITH_LETTER_REGEX)) {
retval = "X" + retval;
}
for (int i = 0; i < retval.length(); i++) {
char c = retval.charAt(i);
if (!validIdChar(c)) {
// replace with "_"
retval = retval.replace(c, '_');
}
}
return retval;
}
/**
* @param c
* @return true if c is a valid character for an ID string
*/
private static boolean validIdChar(char c) {
return ((c >= 'a' && c <= 'z') ||
(c >= 'A' && c <= 'Z') ||
(c >= '0' && c <= '9') ||
(c == '-')||
(c == '_') ||
(c == '.'));
}
/**
* @return
*/
public String getHtml() {
return this.htmlString.toString();
}
/* (non-Javadoc)
* @see org.spdx.licenseTemplate.ILicenseTemplateOutputHandler#beginOptional(org.spdx.licenseTemplate.LicenseTemplateRule)
*/
@Override
public void beginOptional(LicenseTemplateRule rule) {
if (this.optionalNestLevel == 0) { // We only want to format the top level optional texts
removeEndParagraphTag();
this.htmlString.append(formatStartOptionalHTML(rule.getName()));
}
this.optionalNestLevel++;
}
/**
* Format HTML for an optional string
* @param text text for the optional license string
* @param id ID used for the div
* @return
*/
public static String formatStartOptionalHTML(String id) {
StringBuilder sb = new StringBuilder();
sb.append("\n\n");
return sb.toString();
}
public static String formatEndOptionalHTML(boolean inParagraph) {
if (inParagraph) {
return "\n\n";
} else {
return "