org.apache.myfaces.trinidadinternal.skin.SkinStyleProvider Maven / Gradle / Ivy
/*
* 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.myfaces.trinidadinternal.skin;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import javax.faces.context.FacesContext;
import org.apache.myfaces.trinidad.skin.Skin;
import org.apache.myfaces.trinidadinternal.style.StyleContext;
import org.apache.myfaces.trinidadinternal.style.StyleProvider;
import org.apache.myfaces.trinidadinternal.style.cache.FileSystemStyleCache;
import org.apache.myfaces.trinidadinternal.style.xml.StyleSheetDocumentUtils;
import org.apache.myfaces.trinidadinternal.style.xml.parse.StyleSheetDocument;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidadinternal.util.LRUCache;
/**
* An extension of the FileSystemStyleCache which defers to
* a Skin for loading the StyleSheetDocument.
*
* @version $Name: $ ($Revision: adfrt/faces/adf-faces-impl/src/main/java/oracle/adfinternal/view/faces/skin/SkinStyleProvider.java#0 $) $Date: 10-nov-2005.18:58:59 $
*/
public class SkinStyleProvider extends FileSystemStyleCache
{
/**
* Returns a shared instance of the SkinStyleProvider.
* The StyleProvider combines styles from two sources. First,
* styles are pulled from the current Skin, which is
* retrieved from the RenderingContext. Then, styles are pulled
* from the custom style sheet, as identified by the (possibly null)
* customStyleSheetPath argument. Styles specified by the custom
* style sheet take precedence over styles provided by the
* Skin.
*
* @param context The current RenderingContext, which provides
* access to the current Skin.
* @param targetDirectoryPath The full file system path of the
* directory where generated CSS files are stored.
* If the directory does not exist and cannot be
* created, an IllegalArgumentException is thrown.
* @throws IllegalArgumentException This exception is thrown
* if no Skin is found, or if either of the
* paths are invalid.
*/
public static StyleProvider getSkinStyleProvider(
Skin skin,
String targetDirectoryPath
) throws IllegalArgumentException
{
if (skin == null)
throw new IllegalArgumentException(_LOG.getMessage(
"NO_SKIN_SPECIFIED"));
// If the skin is actually one of our request-specific wrapper
// skins, rip off the wrapper and look up the StyleProvider
// for the wrapped skin. If we don't do this, we end up creating
// a new StyleProvider for every request.
if (skin instanceof RequestSkinWrapper)
skin = ((RequestSkinWrapper)skin).getWrappedSkin();
// Create the key object that we use to look up our
// shared SkinStyleProvider instance
ProviderKey key = new ProviderKey(skin,
targetDirectoryPath);
// Get our cache of existing StyleProviders
Map providers = _getProviders();
StyleProvider provider = null;
synchronized (providers)
{
provider = providers.get(key);
if (provider == null)
{
// If we haven't created an instance for this skin/custom style sheet
// yet, try creating it now.
provider = new SkinStyleProvider(skin,
targetDirectoryPath);
if (_LOG.isFine())
{
_LOG.fine("Create a new SkinStyleProvider for skin {0} and targetDirectoryPath {1}",
new Object[]{skin.getId(), targetDirectoryPath});
}
// Store the provider in our cache
providers.put(key, provider);
}
}
return provider;
}
/**
* Creates SkinStyleProvider instance.
* Only subclasses should call this method. All other
* clients should use getSkinStyleProvider().
* @param skin The Skin which defines the
* look and feel-specific style information for this
* StyleProvider.
* @param targetDirectoryPath The full file system path
* to the directory where generated CSS files will be
* stored.
* @see #SkinStyleProvider
* @throws IllegalArgumentException This exception is thrown
* if no Skin is null, or if either of the
* paths are invalid.
*/
protected SkinStyleProvider(
Skin skin,
String targetDirectoryPath
) throws IllegalArgumentException
{
super(targetDirectoryPath);
if (skin == null)
throw new IllegalArgumentException(_LOG.getMessage(
"NO_SKIN_SPECIFIED"));
_skin = skin;
}
/**
* Override of FileSystemStyleCache.createStyleSheetDocument().
* Merges the Skin's styles with custom styles to
* produce a single StyleSheetDocument
*/
@Override
protected StyleSheetDocument createStyleSheetDocument(
StyleContext context
)
{
// First, get the StyleSheetDocument for the custom style
// sheet from the FileSystemStyleCache.
StyleSheetDocument customDocument = super.createStyleSheetDocument(
context);
// Now, get the Skin's StyleSheetDocument
StyleSheetDocument skinDocument = null;
// Synchronize access to _skinDocument
synchronized (this)
{
// gets the skin's StyleSheetDocument (it creates it if needed)
skinDocument = _skinDocument =
((DocumentProviderSkin) _skin).getStyleSheetDocument(context);
}
// Merge the two StyleSheetDocuments
return StyleSheetDocumentUtils.mergeStyleSheetDocuments(skinDocument,
customDocument);
}
/**
* Override of FileSystemStyleCache.hasSourceDocumentChanged()
* which checks for changes to the Skin's style sheet.
*/
@Override
protected boolean hasSourceDocumentChanged(StyleContext context)
{
if (super.hasSourceDocumentChanged(context))
return true;
// Just check to see whether the Skin has a new
// StyleSheetDocument.
// Synchronize access to _skinDocument
synchronized (this)
{
return (_skinDocument !=
((DocumentProviderSkin) _skin).getStyleSheetDocument(context));
}
}
/**
* Override of FileSystemStyleCache.getTargetStyleSheetName().
*/
@Override
protected String getTargetStyleSheetName(
StyleContext context,
StyleSheetDocument document
)
{
// Get the base name from the FileSystemStyleCache.
String name = super.getTargetStyleSheetName(context, document);
// Use the LAF's id as a prefix
String id = _skin.getId();
if (id != null)
{
StringBuffer buffer = new StringBuffer(id.length() + name.length() + 1);
// We know that some LAF ids contain the '.' character. Replace
// this with '-' to make the file name look nicer.
buffer.append(id.replace('.', '-'));
buffer.append('-');
buffer.append(name);
return buffer.toString();
}
return name;
}
// Returns a Map which hashes ProviderKeys to shared instances
// of SkinStyleProviders.
private static Map _getProviders()
{
// =-=ags For now, we just use a global variable. But
// really, our cache should probably be hanging off
// of the ServletContext.
if (_sSharedProviders == null)
{
FacesContext context = FacesContext.getCurrentInstance();
String lruCacheSize =
context.getExternalContext().
getInitParameter(_MAX_SKINS_CACHED);
int lruCacheSizeInt = _MAX_SKINS_CACHED_DEFAULT;
boolean invalidInt = false;
if (lruCacheSize != null && !lruCacheSize.equals(""))
{
try
{
lruCacheSizeInt = Integer.parseInt(lruCacheSize);
}
catch (NumberFormatException nfe)
{
invalidInt = true;
}
if (lruCacheSizeInt <= 0)
{
invalidInt = true;
}
}
// The user typed in an invalid integer ( <=0 or a value that couldn't be formatted to a number)
// so log a warning
if (invalidInt)
{
lruCacheSizeInt = _MAX_SKINS_CACHED_DEFAULT;
if (_LOG.isWarning())
_LOG.warning("INVALID_INTEGER_MAX_SKINS_CACHED", new Object[]{lruCacheSize, _MAX_SKINS_CACHED_DEFAULT});
}
_sSharedProviders = new LRUCache(lruCacheSizeInt);
}
return _sSharedProviders;
}
// Key that we use to retrieve a shared SkinStyleProvider
// instance
private static class ProviderKey
{
public ProviderKey(
Skin skin,
String targetDirectoryPath
)
{
_skin = skin;
_targetDirectoryPath = targetDirectoryPath;
}
// Test for equality
@Override
public boolean equals(Object o)
{
if (o == this)
return true;
if (!(o instanceof ProviderKey))
return false;
ProviderKey key = (ProviderKey)o;
return (_equals(_skin, key._skin) &&
_equals(_targetDirectoryPath, key._targetDirectoryPath));
}
// Produce the hash code
@Override
public int hashCode()
{
int hashCode = _skin.hashCode();
if (_targetDirectoryPath != null)
hashCode ^= _targetDirectoryPath.hashCode();
return hashCode;
}
// Tests two objects for equality, taking possible nulls
// into account
private boolean _equals(Object o1, Object o2)
{
if (o1 == null)
return (o1 == o2);
return o1.equals(o2);
}
private Skin _skin;
private String _targetDirectoryPath;
}
// The Skin which provides styles
private Skin _skin;
// The Skin-specific StyleSheetDocument
private StyleSheetDocument _skinDocument;
// Cache of shared SkinStyleProvider instances
private static Map _sSharedProviders;
private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger(
SkinStyleProvider.class);
private static final String _MAX_SKINS_CACHED =
"org.apache.myfaces.trinidad.skin.MAX_SKINS_CACHED";
private static final int _MAX_SKINS_CACHED_DEFAULT = 20;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy