All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.myfaces.trinidadinternal.skin.provider.BaseSkinProvider 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.provider;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.faces.context.ExternalContext;

import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidad.skin.Skin;
import org.apache.myfaces.trinidad.skin.SkinMetadata;
import org.apache.myfaces.trinidad.skin.SkinProvider;
import org.apache.myfaces.trinidad.skin.SkinVersion;

/**
 * This is the common base class for Trinidad SkinProviders. This class abstracts out some common
 * code that is useful across Trinidad internal SkinProviders. One such example worth mentioning is
 * the code to finding a Skin match for a given SkinMetadata search criteria.
 *
 * @See TrinidadBaseSkinProvider
 * @See TrinidadSkinProvider
 * @See ExternalSkinProvider
 */
abstract class BaseSkinProvider extends SkinProvider
{
  public BaseSkinProvider()
  {
    initSkins();
  }

  /**
   * {@inheritDoc}
   */
  @Override
  public Skin getSkin(ExternalContext context, SkinMetadata skinMetadata)
  {
    synchronized (this)
    {
      return _getMatchingSkin(context, skinMetadata);
    }
  }

  /**
   * {@inheritDoc}
   */
  @Override
  public Collection getSkinMetadata(ExternalContext context)
  {
    synchronized (this)
    {
      return Collections.unmodifiableCollection(_skins.keySet());
    }
  }

  /**
   * add method used to register a skin with the provider.
   *
   * @param metadata
   * @param skin
   * @return Any previously registered skins existing with the same metadata
   */
  public Skin addSkin(SkinMetadata metadata, Skin skin)
  {
    if (metadata == null || skin == null)
    {
      _LOG.warning("CANNOT_ADD_SKIN");
      return null;
    }

    if (_LOG.isFine())
      _LOG.fine("Adding skin to cache [metadata: {0}, skin: {1}]", new Object[] {metadata, skin});

    synchronized (this)
    {
      return _skins.put(metadata, skin);
    }
  }

  /**
   * template method to be implemented by subclasses with logic to load a skin that base class knows
   * as available with the current skin provider being asked to load the skin
   *
   * @param context
   * @param skinMetadata
   */
  protected abstract Skin loadAvailableSkin(ExternalContext context, SkinMetadata skinMetadata);

  /**
   * getter for the skins, only used by ExternalSkinProvider.reload to cache and restore the skins
   * if reload fails
   *
   * @return
   */
  protected Map getSkins()
  {
    synchronized (this)
    {
      return _skins;
    }
  }

  /**
   * setter for the skins, only used by ExternalSkinProvider.reload to cache and restore the skins
   * if reload fails
   *
   * @param skins
   */
  protected void setSkins(Map skins)
  {
    synchronized (this)
    {
      // do not allow a null to be set.
      if (skins == null)
        skins = new HashMap();

      _skins = skins;
    }
  }

  /**
   * initialize a new HashMap for the skins, used by constructor and ExternalSkinProvider.reload
   */
  protected void initSkins()
  {
    synchronized (this)
    {
      _skins = new HashMap();
    }
  }

  /**
   * Hook to do any kind of initialization before loading a skin or skin metadata sub classes can
   * choose to implement
   *
   * @param context
   */
  protected void initialize(ExternalContext context)
  {
  }

  /**
   * one point method for searching skins
   *
   * @param context
   * @param searchMetadata to search for
   * @return matching skin
   */
  private final Skin _getMatchingSkin(ExternalContext context, SkinMetadata searchMetadata)
  {
    if (searchMetadata == null)
      throw new NullPointerException("SkinMetadata passed for search is null");

    // find a skin that is already loaded using the exact search metadata passed.
    Skin availableSkin = _skins.get(searchMetadata);
    if (availableSkin != null)
      return availableSkin;

    // there is no skin already available, so find a skin which matches the search criteria
    // verify that either id or family is set for the search
    if (null == searchMetadata.getId() && null == searchMetadata.getFamily())
    {
      if (_LOG.isWarning())
        _LOG.warning("SP_CANNOT_FIND_SKIN_WITHOUT_FAMILY_OR_ID");

      throw new NullPointerException(_LOG.getMessage("SP_CANNOT_FIND_SKIN_WITHOUT_FAMILY_OR_ID"));
    }

    initialize(context);

    if (_LOG.isFine())
      _LOG.fine("SP_FINDING_SKIN_FOR", new Object[]{searchMetadata.toString()});

    // first check if a skin matching the requirement is present in this provider

    SkinMetadata availableMetadata = _findSkinMetadata(context, searchMetadata);
    if (availableMetadata == null)
    {
      if (_LOG.isFine())
        _LOG.fine(this + " Cannot find skin in this provider: " + searchMetadata);

      return null;
    }

    // find a skin that is already loaded
    // this is different from doing it at the beginning of the method
    // because the metadata with which the skin is added into _skins
    // may not have matched completely, since searchMetadata can provide
    // only certain conditions.
    availableSkin = _skins.get(availableMetadata);
    if (availableSkin != null)
      return availableSkin;

    // now we know that there is a skin available but not loaded
    // so load the skin
    availableSkin = loadAvailableSkin(context, availableMetadata);
    assert (availableSkin != null);

    _skins.put(availableMetadata, availableSkin);

    return availableSkin;
  }

  /**
   * find if there is a skin with the search condition supported by the current SkinProvider
   *
   * @param search
   * @return
   */
  private SkinMetadata _findSkinMetadata(ExternalContext context, SkinMetadata search)
  {
    SkinMetadata matchingSkinMetadata = null;
    String skinId = search.getId();

    Collection skinMetadata = getSkinMetadata(context);

    if (skinMetadata == null || skinMetadata.isEmpty())
      return null;

    // search is with either id or (family, renderkit, version)
    // we have ensure that either id or family is passed in the search
    if (skinId != null)
    {
      // Id is available then search using ID
      for (SkinMetadata m : skinMetadata)
        if (skinId.equals(m.getId()))
        {
          matchingSkinMetadata = m;
        }

      if (matchingSkinMetadata == null)
      {
        if (_LOG.isFine())
        {
          _LOG.fine("SP_CANNOT_FIND_MATCHING_SKIN_ID", new Object[]{skinId});
        }
      }
    }
    else
    {
      // search using family, renderkit, version
      String family = search.getFamily();

      // we need at least the family to go on with the search
      if (family == null)
        return null;

      // renderkit id cannot be null, we initialize it to APACHE_TRINIDAD_DESKTOP
      String renderKit = search.getRenderKitId();

      // version cannot be null as we initialize it to SkinVersion.EMPTY_SKIN_VERSION
      SkinVersion version = search.getVersion();

      List familyRenderKitMatches = new ArrayList(2);

      // search using family and renderkit id
      for (SkinMetadata m : skinMetadata)
        if (family.equalsIgnoreCase(m.getFamily()) &&
              renderKit.equalsIgnoreCase(m.getRenderKitId()))
          familyRenderKitMatches.add(m);

      if (familyRenderKitMatches.isEmpty())
      {
        // if we get here, that means we couldn't find an exact
        // family/renderKitId match, so return the simple skin
        // that matches the renderKitId.
        if (_LOG.isFine())
        {
          _LOG.fine("SP_CANNOT_FIND_MATCHING_SKIN", new Object[]{family, renderKit});
        }

        return null;
      }

      // at this point we know we have something in the familyRenderKitMatches
      // which is a list of matching family and renderKitId skins. Now match the version
      // to find the best matched skin.
      String versionName = version.getName();
      boolean versionIsDefault = version.isDefault();
      boolean foundMatchingSkin = false;

      // if the user didn't ask for the 'default' version, then look for the exact match
      if (!versionIsDefault)
      {
        matchingSkinMetadata = _findSkinMetadataForVersionName(familyRenderKitMatches, version);
      }

      // matchingSkinMetadata will be null if an exact version match (family+renderKitId+version)
      // was not found;
      // or if user was asking for a default version
      // we can have an exact version match if the user asks for null version,
      // and we find a skin with no
      // version set.
      if (matchingSkinMetadata == null || versionIsDefault)
      {
        // at this point either user is looking for default skin
        // or did not find the exact version match specified
        // so we find the default skin
        matchingSkinMetadata = _findSkinMetadataWithDefaultVersion(familyRenderKitMatches);

        if (matchingSkinMetadata == null)
        {
          // if we fail to find a default skin then
          // get the latest skin in the matchingSkinList if there is no skin marked default.
          if (_LOG.isFine())
            _LOG.fine(this + " did not find default / version skinMetadata. so getting leaf skin");

          matchingSkinMetadata = _findLeafSkinMetadata(familyRenderKitMatches);
        }
        else if ((matchingSkinMetadata != null) && versionIsDefault)
        {
          // found the default skin the user wanted
          if (_LOG.isFine())
            _LOG.fine(this + " found default skinMetadata");

          foundMatchingSkin = true;
        }
      }

      // log messages
      if (foundMatchingSkin)
      {
        if (_LOG.isFine())
          _LOG.fine("GET_SKIN_FOUND_SKIN_VERSION",
                    new Object[]{family, version, matchingSkinMetadata.getId()});
      }
      else if (matchingSkinMetadata != null)
      {
        if (_LOG.isFine())
        {
          if ("".equals(versionName))
          {
            _LOG.fine("GET_SKIN_CANNOT_FIND_NO_VERSION",
                      new Object[]{family, matchingSkinMetadata.getId()});
          }
          else
          {
            _LOG.fine("GET_SKIN_CANNOT_FIND_SKIN_VERSION",
                      new Object[]{family, version, matchingSkinMetadata.getId()});
          }
        }
      }

      if (matchingSkinMetadata == null)
      {
        matchingSkinMetadata = familyRenderKitMatches.get(familyRenderKitMatches.size() - 1);
      }
    }


    if (matchingSkinMetadata != null)
      if (_LOG.isFine())
        _LOG.fine(this + " found matching metadata: " + matchingSkinMetadata);

    return matchingSkinMetadata;
  }

  /**
   * find a skin with version passed
   *
   * @param skins
   * @param version
   * @return
   */
  private SkinMetadata _findSkinMetadataForVersionName(Collection skins,
                                                       SkinVersion version)
  {
    if (version == null)
      throw new IllegalArgumentException("skin version cannot be null");

    if (version.getName() == null || version.getName().isEmpty())
      return null;

    for (SkinMetadata metadata : skins)
    {
      // metadata cannot be null and also version inside it cannot be null
      if (version.getName().equals(metadata.getVersion().getName()))
      {
        if (_LOG.isFine())
          _LOG.fine("Found version match skinMetadata: " + metadata);

        return metadata;
      }
    }

    return null;
  }

  /**
   * Latest skin is the one which is last in the family hierarchy eg: fusion-v1 -> fusion-v2 ->
   * fusion-v3 Among this fusion-v3 is the latest. So we look for a skin that is not extended by any
   * other skin in the family.
   *
   * @param skins
   * @return
   */
  private SkinMetadata _findLeafSkinMetadata(Collection skins)
  {
    List leafSkinMetadata = new ArrayList();
    List parentIds = new ArrayList();

    // collect parents skins among the list
    for (SkinMetadata metadata : skins)
    {
      String parentId = metadata.getBaseSkinId();
      if (parentId != null)
        parentIds.add(metadata.getBaseSkinId());
    }

    // find leaf skins, which is not in parent list
    for (SkinMetadata metadata : skins)
    {
      String skinId = metadata.getId();
      if (skinId != null && !parentIds.contains(skinId))
      {
        leafSkinMetadata.add(metadata);
      }
    }

    // if there are no leaves, return null
    // this is a rare case, almost impossible since there will be
    // at least one skin which is not parent of another
    // but let us cover the corner case if any
    if (leafSkinMetadata.isEmpty())
      return null;

    // if there is one leaf skin return that
    // if there are many, return the last one among the leaves
    return leafSkinMetadata.get(leafSkinMetadata.size() - 1);
  }

  /**
   * find a skin that has its SkinVersion set to 'default', if it exists.
   *
   * @param matchingSkinList A list of Skins that we will look through to find the 'default'.
   * @return Skin with SkinVersion isDefault true, otherwise, null.
   */
  private SkinMetadata _findSkinMetadataWithDefaultVersion(
    Collection matchingSkinList)
  {
    for (SkinMetadata metadata : matchingSkinList)
    {
      SkinVersion skinVersion = metadata.getVersion();

      if (skinVersion != null && skinVersion.isDefault())
      {
        if (_LOG.isFine())
          _LOG.fine("Found default skinMetadata: " + metadata);

        return metadata;
      }
    }

    return null;
  }

  private Map _skins;
  private final static TrinidadLogger _LOG =
    TrinidadLogger.createTrinidadLogger(BaseSkinProvider.class);
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy