org.apache.wicket.util.template.PackageTextTemplate.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.wicket.util.template.PackageTextTemplate.java

Source

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.wicket.util.template;

import java.io.IOException;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;

import org.apache.wicket.Application;
import org.apache.wicket.util.io.Streams;
import org.apache.wicket.util.lang.Packages;
import org.apache.wicket.util.resource.IResourceStream;
import org.apache.wicket.util.resource.ResourceStreamNotFoundException;
import org.apache.wicket.core.util.resource.locator.ResourceStreamLocator;
import org.apache.wicket.util.string.interpolator.MapVariableInterpolator;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A <code>String</code> resource that can be appended to.
 * 
 * @author Eelco Hillenius
 * @since 1.2.6
 */
public class PackageTextTemplate extends TextTemplate {
    /** log. */
    private static final Logger log = LoggerFactory.getLogger(PackageTextTemplate.class);

    private static final long serialVersionUID = 1L;

    /** The content type used if not provided in the constructor */
    public static final String DEFAULT_CONTENT_TYPE = "text";

    /** The encoding used if not provided in the constructor */
    public static final String DEFAULT_ENCODING = null;

    /** contents */
    private final StringBuilder buffer = new StringBuilder();

    private final Class<?> scope;

    private final String fileName;

    private String encoding;

    /**
     * Constructor.
     * 
     * @param clazz
     *            the <code>Class</code> to be used for retrieving the classloader for loading the
     *            <code>PackagedTextTemplate</code>
     * @param fileName
     *            the name of the file, relative to the <code>clazz</code> position
     */
    public PackageTextTemplate(final Class<?> clazz, final String fileName) {
        this(clazz, fileName, DEFAULT_CONTENT_TYPE);
    }

    /**
     * Constructor.
     * 
     * @param clazz
     *            the <code>Class</code> to be used for retrieving the classloader for loading the
     *            <code>PackagedTextTemplate</code>
     * @param fileName
     *            the name of the file, relative to the <code>clazz</code> position
     * @param contentType
     *            the mime type of this resource, such as "<code>image/jpeg</code>" or "
     *            <code>text/html</code>"
     */
    public PackageTextTemplate(final Class<?> clazz, final String fileName, final String contentType) {
        this(clazz, fileName, contentType, DEFAULT_ENCODING);
    }

    /**
     * Constructor.
     * 
     * @param clazz
     *            the <code>Class</code> to be used for retrieving the classloader for loading the
     *            <code>PackagedTextTemplate</code>
     * @param fileName
     *            the name of the file, relative to the <code>clazz</code> position
     * @param contentType
     *            the mime type of this resource, such as "<code>image/jpeg</code>" or "
     *            <code>text/html</code>"
     * @param encoding
     *            the file's encoding, for example, "<code>UTF-8</code>"
     */
    public PackageTextTemplate(final Class<?> clazz, final String fileName, final String contentType,
            final String encoding) {
        this(clazz, fileName, null, null, null, contentType, encoding);
    }

    /**
     * Constructor.
     *
     * @param clazz
     *            the <code>Class</code> to be used for retrieving the classloader for loading the
     *            <code>PackagedTextTemplate</code>
     * @param fileName
     *            the name of the file, relative to the <code>clazz</code> position
     * @param style
     *            Any resource style, such as a skin style (see {@link org.apache.wicket.Session})
     * @param variation
     *            The template's variation (of the style)
     * @param locale
     *            The locale of the resource to load
     * @param contentType
     *            the mime type of this resource, such as "<code>image/jpeg</code>" or "
     *            <code>text/html</code>"
     * @param encoding
     *            the file's encoding, for example, "<code>UTF-8</code>"
     */
    public PackageTextTemplate(final Class<?> clazz, final String fileName, final String style,
            final String variation, final Locale locale, final String contentType, final String encoding) {
        super(contentType);

        this.scope = clazz;
        this.fileName = fileName;
        this.encoding = encoding;

        setStyle(style);
        setVariation(variation);
        setLocale(locale);
    }

    @Override
    public void setStyle(String style) {
        if (Objects.equals(style, getStyle()) == false) {
            buffer.setLength(0);
        }
        super.setStyle(style);
    }

    @Override
    public void setLocale(Locale locale) {
        if (Objects.equals(locale, getLocale()) == false) {
            buffer.setLength(0);
        }
        super.setLocale(locale);
    }

    @Override
    public void setVariation(String variation) {
        if (Objects.equals(variation, getVariation()) == false) {
            buffer.setLength(0);
        }
        super.setVariation(variation);
    }

    public void setEncoding(String encoding) {
        if (Objects.equals(encoding, this.encoding) == false) {
            buffer.setLength(0);
        }
        this.encoding = encoding == null ? DEFAULT_ENCODING : encoding;
    }

    /**
     * Loads the template if it is not loaded yet
     */
    private void load() {
        if (buffer.length() == 0) {
            String path = Packages.absolutePath(scope, fileName);

            Application app = Application.get();

            // first try default class loading locator to find the resource
            IResourceStream stream = app.getResourceSettings().getResourceStreamLocator().locate(scope, path,
                    getStyle(), getVariation(), getLocale(), null, false);

            if (stream == null) {
                // if the default locator didn't find the resource then fallback
                stream = new ResourceStreamLocator().locate(scope, path, getStyle(), getVariation(), getLocale(),
                        null, false);
            }

            if (stream == null) {
                throw new IllegalArgumentException(
                        "resource " + fileName + " not found for scope " + scope + " (path = " + path + ")");
            }

            setLastModified(stream.lastModifiedTime());

            try {
                if (encoding != null) {
                    buffer.append(Streams.readString(stream.getInputStream(), encoding));
                } else {
                    buffer.append(Streams.readString(stream.getInputStream()));
                }
            } catch (IOException e) {
                throw new RuntimeException(e);
            } catch (ResourceStreamNotFoundException e) {
                throw new RuntimeException(e);
            } finally {
                try {
                    stream.close();
                } catch (IOException e) {
                    log.error(e.getMessage(), e);
                }
            }
        }
    }

    /**
     * @see org.apache.wicket.util.resource.AbstractStringResourceStream#getString()
     */
    @Override
    public String getString() {
        load();
        return buffer.toString();
    }

    /**
     * Interpolates a <code>Map</code> of variables with the content and replaces the content with
     * the result. Variables are denoted in the <code>String</code> by the
     * <code>syntax ${variableName}</code>. The contents will be altered by replacing each variable
     * of the form <code>${variableName}</code> with the value returned by
     * <code>variables.getValue("variableName")</code>.
     * <p>
     * WARNING: there is no going back to the original contents after the interpolation is done. If
     * you need to do different interpolations on the same original contents, use the method
     * {@link #asString(Map)} instead.
     * </p>
     * 
     * @param variables
     *            a <code>Map</code> of variables to interpolate
     * @return this for chaining
     */
    @Override
    public final TextTemplate interpolate(Map<String, ?> variables) {
        if (variables != null) {
            load();
            String result = new MapVariableInterpolator(buffer.toString(), variables).toString();
            buffer.setLength(0);
            buffer.append(result);
        }
        return this;
    }

}