/* * 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.core.util.resource.locator; import java.util.Arrays; import java.util.Locale; import org.apache.wicket.WicketRuntimeException; import org.apache.wicket.util.string.Strings; /** * Contains the logic to locate a resource based on a path, style (see * {@link org.apache.wicket.Session}), variation, locale and extension strings. The full filename * will be built like: * <path>_<variation>_<_<style>_<locale>.<extension>. * <p> * Resource matches will be attempted in the following order: * <ol> * <li>1. <path>_<style>_<locale>.<extension></li> * <li>2. <path>_<locale>.<extension></li> * <li>3. <path>_<style>.<extension></li> * <li>4. <path>.<extension></li> * </ol> * <p> * Locales may contain a language, a country and a region or variant. Combinations of these * components will be attempted in the following order: * <ol> * <li>locale.toString() see javadoc for Locale for more details</li> * <li><language>_<country></li> * <li><language></li> * </ol> * * @author Juergen Donnerstag */ public class ResourceNameIterator implements IResourceNameIterator { // The base path without extension, style, locale etc. private final String path; // The extensions to search for the resource file private final Iterable<String> extensions; // The locale to search for the resource file private final Locale locale; // Do not test any combinations. Just return the full path based on the locale, style etc. // provided. Only iterate over the extensions provided. private final boolean strict; // The various iterators used to locate the resource file private final StyleAndVariationResourceNameIterator styleIterator; private LocaleResourceNameIterator localeIterator; private ExtensionResourceNameIterator extensionsIterator; /** * Construct. * * @param path * The path of the resource. In case the parameter 'extensions' is null, the path * will be checked and if a filename extension is present, it'll be used instead. * @param style * A theme or style (see {@link org.apache.wicket.Session}) * @param variation * The component's variation (of the style) * @param locale * The Locale to apply * @param extensions * the filename's extensions * @param strict * If false, weaker combinations of style, locale, etc. are tested as well */ public ResourceNameIterator(final String path, final String style, final String variation, final Locale locale, final Iterable<String> extensions, final boolean strict) { this.locale = locale; boolean noext = extensions == null || !extensions.iterator().hasNext(); if (noext && (path != null) && (path.indexOf('.') != -1)) { String[] extns = Strings.split(Strings.afterLast(path, '.'), ','); this.extensions = Arrays.asList(extns); this.path = Strings.beforeLast(path, '.'); } else { this.extensions = extensions; this.path = path; } styleIterator = newStyleAndVariationResourceNameIterator(style, variation); this.strict = strict; } /** * Get the exact Locale which has been used for the latest resource path. * * @return current Locale */ @Override public final Locale getLocale() { return localeIterator.getLocale(); } /** * Get the exact Style which has been used for the latest resource path. * * @return current Style */ @Override public final String getStyle() { return styleIterator.getStyle(); } /** * Get the exact Variation which has been used for the latest resource path. * * @return current Variation */ @Override public final String getVariation() { return styleIterator.getVariation(); } /** * Get the exact filename extension used for the latest resource path. * * @return current filename extension */ @Override public final String getExtension() { return extensionsIterator.getExtension(); } /** * @see java.util.Iterator#hasNext() */ @Override public boolean hasNext() { // Most inner loop. Loop through all extensions provided if (extensionsIterator != null) { if (extensionsIterator.hasNext() == true) { return true; } // If there are no more extensions, than return to the next outer // loop (locale). Get the next value from that loop and start // over again with the first extension in the list. extensionsIterator = null; } // 2nd inner loop: Loop through all Locale combinations if (localeIterator != null) { while (localeIterator.hasNext()) { localeIterator.next(); extensionsIterator = newExtensionResourceNameIterator(extensions); if (extensionsIterator.hasNext() == true) { return true; } } localeIterator = null; } // Most outer loop: Loop through all combinations of styles and variations while (styleIterator.hasNext()) { styleIterator.next(); localeIterator = newLocaleResourceNameIterator(locale, strict); while (localeIterator.hasNext()) { localeIterator.next(); extensionsIterator = newExtensionResourceNameIterator(extensions); if (extensionsIterator.hasNext() == true) { return true; } } if (strict) { break; } } // No more combinations found. End of iteration. return false; } /** * @see java.util.Iterator#next() */ @Override public String next() { if (extensionsIterator != null) { extensionsIterator.next(); return toString(); } throw new WicketRuntimeException( "Illegal call of next(). Iterator not properly initialized"); } /** * @see java.lang.Object#toString() */ @Override public String toString() { return path + prepend(getVariation(), '_') + prepend(getStyle(), '_') + prepend(getLocale(), '_') + prepend(getExtension(), '.'); } /** * * @param string * @param prepend * @return The string prepended with the char */ private String prepend(Object string, char prepend) { return (string != null) ? prepend + string.toString() : ""; } /** * @param locale * @param strict * @return New iterator */ protected LocaleResourceNameIterator newLocaleResourceNameIterator(final Locale locale, boolean strict) { return new LocaleResourceNameIterator(locale, strict); } /** * * @param style * @param variation * @return new iterator */ protected StyleAndVariationResourceNameIterator newStyleAndVariationResourceNameIterator( final String style, final String variation) { return new StyleAndVariationResourceNameIterator(style, variation); } /** * @param extensions * @return New iterator */ protected ExtensionResourceNameIterator newExtensionResourceNameIterator( final Iterable<String> extensions) { return new ExtensionResourceNameIterator(extensions); } /** * @see java.util.Iterator#remove() */ @Override public void remove() { // ignore } }