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

org.apache.myfaces.trinidadinternal.style.util.CSSGenerationUtils 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.style.util;

import java.beans.Beans;

import java.io.PrintWriter;
import java.io.StringWriter;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Comparator;
import java.util.Date;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Pattern;

import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidadinternal.agent.TrinidadAgent;
import org.apache.myfaces.trinidadinternal.renderkit.core.xhtml.SkinSelectors;
import org.apache.myfaces.trinidadinternal.style.StyleContext;
import org.apache.myfaces.trinidadinternal.style.xml.parse.PropertyNode;
import org.apache.myfaces.trinidadinternal.style.xml.parse.StyleNode;


/**
 * CSS-generation related utilities used when we write out our css-2 stylesheet
 * document based on the skin's css-3 stylesheet document.
 *
 * @version $Name:  $ ($Revision: adfrt/faces/adf-faces-impl/src/main/java/oracle/adfinternal/view/faces/style/util/CSSUtils.java#0 $) $Date: 10-nov-2005.18:58:49 $
 */

public class CSSGenerationUtils
{
  /**
   * Returns a MatchingStyles object of normalized propertyString to the StyleNodes that have them as properties.
   * MatchingStyles segregates the clientRules from the normal selectors for ease of css rendering.
   * MatchingStyles object uses LinkedHashMap within, to indicate that relative ordering is preserved between the first
   * appearance of selectors with unshared properties.
   * However, a later style rule that shares its properties with a much earlier style rule will be defined with that
   * style rule, thus having its definition moved up in the file.  Therefore, customers should rely on
   * specificity rather than ordering to override styles.
   * defined with 
   * @param styleNodes
   * @return
   */
  private static MatchingStyles _buildMatchingStylesMap(List styleNodes)
  {
    int styleCount = styleNodes.size();

    MatchingStyles matchingStyles = new MatchingStyles(styleCount);

    // at this point the styles List can contain both Styles with
    // non-null selector or non-null name(aka alias). We only generate
    // the styles where getSelector is non-null.
    for (StyleNode styleNode : styleNodes)
    {
      if (styleNode.getSelector() != null)
      {
        // Get the property string (properties are sorted so that
        // the order doesn't affect whether styles match).
        String propertyString = _getSortedPropertyString(styleNode);
        if (!("".equals(propertyString)))
        {
          if (styleNode.hasClientRule())
          {
            matchingStyles.addStyle(styleNode.getClientRule(), propertyString, styleNode);
          }
          else
          {
            matchingStyles.addStyle(propertyString, styleNode);
          }
        }
      }
    }
    
    return matchingStyles;
  }

  /**
   * Writes the properties of a merged set of style rules
   * @param out
   * @param properties
   * @param styleSheetName
   * @param compressStyles
   * @param baseURI
   * @return
   */
  private static void _writeMergedProperties(
    PrintWriter out, Iterable properties, String styleSheetName, boolean compressStyles, String baseURI)
  {
    out.print(" {");

    boolean first = true;

    for (PropertyNode property : properties)
    {
      String propName  = property.getName();
      String propValue = property.getValue();

      if ((propName != null) &&
          (propValue != null) &&
          (propValue.length() > 0 ))
      {
        // insert separator before all except the first property
        if (first)
          first = false;
        else
          out.print(';');

        out.print(propName);
        out.print(':');
        String resolvedPropValue = CSSUtils.resolvePropertyValue(styleSheetName, baseURI, propName, propValue);
        out.print(resolvedPropValue);
      }
    }

    out.print('}'); 

    // take out the newlines for performance
    if (!compressStyles)
    {
      out.println();
    }
  }

  /**
   * Writes out a List of valid selectors
   * @param out
   * @param validSelectors
   */
  private static void _writeValidSelectors(PrintWriter out, List validSelectors)
  {
    boolean first = true;

    // Write out all of the style selectors for this property string
    for (String validSelector : validSelectors)
    {
      // insert separator before all except the first property
      if (first)
        first = false;
      else
        out.print(',');

      out.print(validSelector);
    }
  }

  /**
   * Given a List of mappedSelectors, returns a List of valid selectors to write
   * @param compressStyles
   * @param shortStyleClassMap
   * @param namespacePrefixArray
   * @param mappedSelectors
   * @return
   */
  private static List  _calculateValidSelectors(
    boolean compressStyles, Map shortStyleClassMap,
    String[] namespacePrefixArray, List mappedSelectors)
  {
     List validSelectors = new ArrayList(mappedSelectors.size() * 2);

    // TODO: figure out why we write both the uncompressed & compressed styles for styles
    // without a '|' character, shouldn't the uncompressed be enough on its own? This results
    // in some ugly code here.
      
    // Write out all of the style selectors for this property string
    for (String mappedSelector : mappedSelectors)
    { 
      String validFullNameSelector = null;

      // write out the full selector if we aren't compressing styles or
      // it doesn't have a '|' in the name which means it may be a user's public styleclass
      // and we don't want to compress those; we will also write out the compressed
      // version the public styleclasses in the next step.
      if (!compressStyles || (mappedSelector.indexOf('|') == -1))
      {
        validFullNameSelector = getValidFullNameSelector(mappedSelector, namespacePrefixArray);

        if (validFullNameSelector != null)
        {
          validSelectors.add(validFullNameSelector);
        }
      }

      if (compressStyles)
      {
        String shortSelector = getShortSelector(shortStyleClassMap, namespacePrefixArray, mappedSelector);

        // if the transformed full name is different than the shortSelector
        // then write out the shortSelector, too.
        if (shortSelector != null)
        {
          String validShortSelector = getValidFullNameSelector(shortSelector, namespacePrefixArray);

          // if we wrote out a full style, check to see if we need to write out the short, too.
          // if it is something different, write out the short, too.
          if (validFullNameSelector != null)
          {
            //Since validFullNameSelector is not null, we know we wrote out a full style
            // we write out a short style too in this case if it is different
            // example: .PublicStyleClass is written out fully even in compressed mode, but
            // it is different in compressed mode, so we write that out, too.
            if (!validFullNameSelector.equals(validShortSelector))
            {
              validSelectors.add(validShortSelector);
            }
          }
          else
          {
            validSelectors.add(validShortSelector);
          }
        }
      }
    }
    
    return validSelectors;
  }

  /**
   * Writes the merged selectors represented by a mappedSelectors, returning how many selectors were written. If the number
   * of selectors that would be written is greater to or equal than maxSelectors, nothing will be written and
   * 0 will be returned.  It is up to the caller to retry with this entry later.
   * @param out
   * @param compressStyles
   * @param shortStyleClassMap
   * @param namespacePrefixArray
   * @param maxSelectors
   * @param mappedSelectors
   * @return
   */
  private static int _writeMergedSelectors(
    PrintWriter out, boolean compressStyles, Map shortStyleClassMap,
    String[] namespacePrefixArray, int maxSelectors, List mappedSelectors)
  {
    // to make this atomic, we first calculate all of the selectors that we will write out.  If we have space, we will
    // then write the selectors out
    List validSelectors = _calculateValidSelectors(compressStyles, shortStyleClassMap, namespacePrefixArray,
                                                           mappedSelectors);

    int selectorsToWrite = validSelectors.size();

    if (selectorsToWrite >= maxSelectors)
    {
      // not enough space, so abort
      return 0;
    }
    else
    {
      _writeValidSelectors(out, validSelectors);

      return selectorsToWrite;
    }
  }

  /**
   * For a List of matching StyleNode, return the mapped version of their Selectors
   */
  private static List _calculateMappedSelectors(
    String[] namespacePrefixArray, Map afSelectorMap, List matchingStyleNodes)
  {
    List mappedSelectors = new ArrayList(matchingStyleNodes.size());

    for (StyleNode matchingNode : matchingStyleNodes)
    {
      String matchingSelector = matchingNode.getSelector();

      // We should always have a selector at this point
      assert (matchingSelector != null);
      
      String mappedSelector = getMappedSelector(afSelectorMap, namespacePrefixArray, matchingSelector);

      mappedSelectors.add(mappedSelector);
    }
    
    return mappedSelectors;
  }

  /**
   * Writes a merged style rule represented by a mergedEntry, returning how many selectors were written.  If the number
   * of selectors that would be written is greater to or equal than maxSelectors, nothing will be written and
   * 0 will be returned.  It is up to the caller to retry with this entry later.
   * @param styleSheetName
   * @param compressStyles
   * @param shortStyleClassMap
   * @param namespacePrefixArray
   * @param afSelectorMap
   * @param out
   * @param maxSelectors
   * @param mergedEntry
   * @return
   */
  private static int _writeMergedEntry(
    String                             styleSheetName,
    String                             baseURI,
    boolean                            compressStyles,
    Map                shortStyleClassMap,
    String[]                           namespacePrefixArray,
    Map                afSelectorMap,
    PrintWriter                        out,
    int                                maxSelectors,
    Map.Entry> mergedEntry
    )
  {
    String propertyString = mergedEntry.getKey();

    // we shouldn't have any null or empty Strings
    assert (propertyString != null);
    assert (!"".equals(propertyString));

    // Get all of the styles which share this property string.
    List matchingStyleNodes = mergedEntry.getValue();

    // Actually, we should always have at least one StyleNode here
    assert (!matchingStyleNodes.isEmpty());
    
    List mappedSelectors = _calculateMappedSelectors(namespacePrefixArray, afSelectorMap, matchingStyleNodes);

 
    // Write out all of the style selectors for this property string.  If 0 is returned, we've aborted
    int numberSelectorsWritten = _writeMergedSelectors(out, compressStyles, shortStyleClassMap, namespacePrefixArray,
                                                       maxSelectors, mappedSelectors);

    // Now that we have written out the selectors, write out
    // the properties
    if (numberSelectorsWritten > 0)
    {
      // At this point, we could just write out the property string
      // that we already created, but this string contains the properties
      // in sorted order.  We prefer to attempt to preserve the order
      // of the properties as specified in the XSS document.  So,
      // we get the properties from the StyleNode object instead of
      // using the propertyString
      // because of this lameness, we use the order from the original StyleNode
      Iterable properties = matchingStyleNodes.get(0).getProperties();
  
      _writeMergedProperties(out, properties, styleSheetName, compressStyles, baseURI);
    }
    
    return numberSelectorsWritten;
  }
  
    
  /**
   * Returns the maximum number of selectors per file that this agent supports
   * @param context
   * @return
   */
  private static int _getMaxSelectorsPerFile(StyleContext context)
  {
    if (TrinidadAgent.Application.IEXPLORER == context.getAgent().getAgentApplication())
    {
      return _MSIE_SELECTOR_LIMIT;
    }
    else
    {
      return Integer.MAX_VALUE;
    }
  }
  
  /**
   * Converts the specified set of StyleNodes to CSS. We output either full styleclass names or
   * compressed styleclass names.
   *
   * @param context The current StyleContext
   * @param styleSheetName The stylesheet name that is registered with the skin.
   *     e.g. skins/purple/purpleSkin.css
   * @param styleNodes The style nodes to convert
   * @param writerFactory The factory to obtain {@link PrintWriter} instances
   * @param compressStyles This tells us whether or not we want to output the short names.
   * @param shortStyleClassMap A Map which maps style
   *   class names to short names.
   * @param namespacePrefixArray an array with the namespace prefixes of our
   *  special selectors. e.g., "af|" or "tr|" .
   * @param afSelectorMap A Map which maps the namespaced component selectors
   *   to their base names (e.g., 'af|menuPath::step' maps to 'af|menuPath A')
   */
  public static void writeCSS(
    StyleContext        context,
    String              styleSheetName,
    List     styleNodes,
    StyleWriterFactory  writerFactory,
    boolean             compressStyles,
    Map shortStyleClassMap,
    String[]            namespacePrefixArray,
    Map afSelectorMap)
  {
    // writeCSS() attempts to produce a minimal set of style rules
    // by combining selectors with identical properties into a
    // single rule.  For example, we convert the following two
    // styles:
    //
    //   .OraBGAccentVeryDark,.x8 {background-color:#999966}
    //   .OraTable,.x1x {background-color:#999966}
    //
    // into a single style:
    //
    //   .OraBGAccentVeryDark,.x8,.OraTable,.x1x {background-color:#999966}
    //
    // The intention of this optimization is to improve performance by
    // reducing the overall size of the generated CSS file - and hopefully
    // speed up download time.
    //
    // To implement this optimization, we perform two passes over the
    // set of styles.  During the first pass, we build up a hash map
    // which maps from the property string to an array of StyleNodes
    // which share the properties defined by the property string.
    // During the second pass over the styles, we write out all matching
    // selectors for each style followed by the shared set of properties.

    // This is the first pass where MatchingStyles object is created from
    // the styleNodes. MatchingStyles object segregates the normal selectors
    // rendered as selector {prop: value} and clientRule selectors which are
    // rendered as clientRule { selector { prop: value }}
    // The segregation of clientRules allow us to segregate the rendering logic.
    MatchingStyles matchingStyles = _buildMatchingStylesMap(styleNodes);

    // This is the second pass in which we write out the style rules
    // Get the baseURI up front so we don't have to recalculate it every time we find
    // a property value that contains url() and need to resolve the uri.
    String baseURI = CSSUtils.getBaseSkinStyleSheetURI(styleSheetName);

    // Because of IE's selector limit, we need a slightly more complicated scheme
    // for writing out the styles, so that we can create new CSS files as necessary
    final int maxSelectorsPerFile = _getMaxSelectorsPerFile(context);
    int fileSelectorsWritten = 0;

    // read out the normal selectors, we render these first
    LinkedHashMap> noClientRuleMap = matchingStyles.getNonClientRuleMap();
    Iterator>> noClientRuleEntries = noClientRuleMap.entrySet().iterator();

    PrintWriter out = writerFactory.createWriter();

    if (out == null)
    {
      return;
    }

    _beginCssFile(out, compressStyles);

    // loop over all of the entries
    while (noClientRuleEntries.hasNext())
    {

      Map.Entry> abortedEntry = null;

      // loop over all of the entries we can fit in a CSS file
      while (abortedEntry == null && noClientRuleEntries.hasNext())
      {
        // get the entry to write out.  This handles retrying any aborted entry
        Map.Entry> currEntry;

        // aborted entry is set when the max number of selectors have already
        // written out for the current css file. if aborted entry is set then
        // a new file needs to be created to write remaining selectors
        if (abortedEntry != null)
        {
          currEntry = abortedEntry;
          abortedEntry = null;
        }
        else
        {
          currEntry = noClientRuleEntries.next();
        }

        // write the entry
        int selectorsLeft = maxSelectorsPerFile - fileSelectorsWritten;
        int entrySelectorsWritten = _writeMergedEntry(styleSheetName, baseURI, compressStyles, shortStyleClassMap,
                                                      namespacePrefixArray, afSelectorMap, out, selectorsLeft,
                                                      currEntry);
        // detect abort
        if (entrySelectorsWritten == 0)
        {
          // no selectors written, so we have reached the threshold for this css file
          // set the aborted entry so that we will create a new css file
          abortedEntry = currEntry;
        }
        else
        {
          // add up the selectors written to track the threshold for the css file
          fileSelectorsWritten += entrySelectorsWritten;
        }
      }

      if (abortedEntry != null)
      {
        // abort detected, close the current out and create a new one
        _endCssFile(out, styleSheetName, compressStyles, fileSelectorsWritten);
        fileSelectorsWritten = 0;

        // create new out
        out = writerFactory.createWriter();

        if (out == null)
        {
          return;
        }

        // put in the headers
        _beginCssFile(out, compressStyles);
      }
    }

    Set clientRules = matchingStyles.getClientRules();

    // while writing a client rule, all selectors within the clien rule has to fit into the current css file.
    // If that cannot happen, then we need to create a new css file and continue writing.
    for (String clientRule : clientRules)
    {
      // write out all client rule entries into a temp out, so that we will know if it can be fitted in
      // the current css file
      StringWriter tempStringWriter = new StringWriter();
      PrintWriter tempWriter = new PrintWriter(tempStringWriter);
      
      // we cannot count the selectors within the client rule to decide the number of selectors
      // there are cases where both compressed and non-compressed selectors get written
      // so we create a temp StringWriter and get the selectors and properties written into that
      // we check if all selectors enclosed in a particular client rule can be incorporated in the current out
      // otherwise we create a new out and write out the client rule followed by the selectors within
      int clientRuleSelectorsWritten = _writeClientRuleStyles(tempWriter, clientRule, matchingStyles, maxSelectorsPerFile,
                                                              fileSelectorsWritten, styleSheetName, baseURI, compressStyles,
                                                              shortStyleClassMap,namespacePrefixArray, afSelectorMap);

      // if no selectors were written, then we need a new file to contain the contents of the current clientRule
      if (clientRuleSelectorsWritten == 0)
      {
        // all selectors inside client rule cannot be accommodated in current file
        _endCssFile(out, styleSheetName, compressStyles, fileSelectorsWritten);
        fileSelectorsWritten = 0;

        // create new out
        out = writerFactory.createWriter();

        if (out == null)
        {
          return;
        }

        // write headers
        _beginCssFile(out, compressStyles);

        // close the tempWriter and create a new one
        // some selectors inside the client rule may have got written to tempWriter
        tempWriter.close();
        tempStringWriter = new StringWriter();
        tempWriter = new PrintWriter(tempStringWriter);

        // now that we recreated the writer, so it should have space to write all the selectors in the client rule
        clientRuleSelectorsWritten = _writeClientRuleStyles(tempWriter, clientRule, matchingStyles, 
                                                            maxSelectorsPerFile, fileSelectorsWritten, styleSheetName, 
                                                            baseURI, compressStyles, shortStyleClassMap,
                                                            namespacePrefixArray, afSelectorMap);
      }

      // all selectors inside the current client rule should now be written to tempWriter
      assert (clientRuleSelectorsWritten != 0);

      if (tempWriter.checkError())
      {
        // check for errors in tempWriter and log it and return
        _LOG.severe("Error writing stylesheet:" + styleSheetName);
        return;
      }

      // the client rule can now be written to the actual out
      out.print(clientRule);
      _writeString(out, " {", compressStyles);

      // write the selectors and properties to the actual out
      out.print(tempStringWriter.toString());

      // closing braces for client rule
      _writeString(out, "}", compressStyles);

      fileSelectorsWritten += clientRuleSelectorsWritten;

      tempWriter.close();
    }

    _endCssFile(out, styleSheetName, compressStyles, fileSelectorsWritten);
  }

  /**
   * tests if all the selectors under the client rule can fit into the current css file.
   * writes the selectors and properties into the PrintWriter passed.
   * 
   * @param out
   * @param clientRule
   * @param matchingStyles
   * @param maxSelectorsPerFile
   * @param fileSelectorsWritten
   * @param styleSheetName
   * @param baseURI
   * @param compressStyles
   * @param shortStyleClassMap
   * @param namespacePrefixArray
   * @param afSelectorMap
   * @return number of selectors written into the css file. Returns 0, if all selectors does not fit
   *
   */
  private static int _writeClientRuleStyles(PrintWriter out, 
                                            String clientRule, 
                                            MatchingStyles matchingStyles, 
                                            int maxSelectorsPerFile,
                                            int fileSelectorsWritten, 
                                            String styleSheetName, 
                                            String baseURI,
                                            boolean compressStyles, 
                                            Map shortStyleClassMap,
                                            String[] namespacePrefixArray,
                                            Map afSelectorMap)
  {
    int selectorsWritten = 0;
    int entrySelectorsWritten = 0;
    
    // we know this cannot be null because the caller is using the client rule keySet from matchingStyles
    LinkedHashMap> clientRuleMap = matchingStyles.getClientRuleMap(clientRule);
    Iterator>> clientRuleEntries = clientRuleMap.entrySet().iterator();
    
    while (clientRuleEntries.hasNext())
    {
      Map.Entry> currEntry = clientRuleEntries.next();
      int selectorsLeft = maxSelectorsPerFile - (fileSelectorsWritten + selectorsWritten);
      entrySelectorsWritten = _writeMergedEntry(styleSheetName, baseURI, compressStyles, shortStyleClassMap,
                                                namespacePrefixArray, afSelectorMap, out, selectorsLeft,
                                                currEntry);
      
      // we want to write all selectors in clientRuleEntries into the same css file
      // so, if we cannot write any one of these selector we need to abort
      if (entrySelectorsWritten == 0)
      {
        return 0;
      }
      else
      {
        selectorsWritten += entrySelectorsWritten;
      }
    }

    return selectorsWritten;
  }

  /**
   * outputs header for a css file
   * @param out
   * @param compressStyles
   */
  private static void _beginCssFile(PrintWriter out, boolean compressStyles)
  {
    if (!compressStyles)
    {
      Date date = new Date();

      // write out the header with a time stamp
      out.println("/* This CSS file generated on " + date + " */");
    }
  }

  /**
   * outputs the footer for a css file
   * @param out
   * @param styleSheetName
   * @param compressStyles
   * @param fileSelectorsWritten
   */
  private static void _endCssFile(PrintWriter out, String styleSheetName, boolean compressStyles, int fileSelectorsWritten)
  {
    if (!compressStyles)
    {
      out.print("/* The number of CSS selectors in this file is ");
      out.print(fileSelectorsWritten);
      out.println(" */");
    }

    if (out.checkError())
    {
      _LOG.severe("Error writing stylesheet:" + styleSheetName);
    }

    out.close();
  }

  /**
   * Util method to output a outStr based on compressStyles
   * @param out
   * @param outStr
   * @param compressStyles
   */
  private static void _writeString(PrintWriter out, String outStr, boolean compressStyles)
  {
    if (compressStyles)
      out.print(outStr); // take out the newlines for performance
    else
      out.println(outStr);
  }

  /**
   * Shorten (compress) the selector.
   * @param shortStyleClassMap
   * @param namespacePrefixArray
   * @param selector
   * @return the shortened selector, or selector if nothing could be shortened.
   */
  public static String getShortSelector(
    Map shortStyleClassMap,
    String[]            namespacePrefixArray,
    String              selector)
  {
    // shorten all the css-2 style class selectors (those that start with
    // '.' and don't have a namespace prefix in it)
    // and return the shortened string.
    // e.g., selector of '.OraBulletedList A' is shortened to '.xj A'
    // e.g., selector of af|inputText::content is not shortened since
    // it has no css-2 style class selector piece that starts with '.'.
    // e.g., selector of af|foo.Bar will shorten the '.Bar' piece
    // af|foo.xz
    // e.g., .Foo:hover -> .x0:hover
    String shortSelector = _getShortNonNamespacedSelector(selector,
                                             shortStyleClassMap);

    if (shortSelector == null)
      shortSelector = selector;

    // run it through a shortener one more time to shorten any
    // of the af component selectors.
    // e.g., 'af|menuPath' is shortened to '.x11'

    if (_hasNamespacePrefix(shortSelector, namespacePrefixArray))
    {
      String[] shortSelectorArray  = StyleUtils.splitStringByWhitespace(shortSelector);

      shortSelector = _getMappedNSSelector(shortStyleClassMap,
                                           namespacePrefixArray,
                                           shortSelectorArray,
                                           true);
    }
    return shortSelector;
  }

  /**
   * Tests whether the specified selector is a single style class
   * selector. A single style class selector is something like
   * ".AFInstructionText". Examples that are not single style class
   * selectors are "af|inputText" or ".AFFoo .AFBar" or ".foo:hover"
   */
  public static boolean isSingleStyleClassSelector(String selector)
  {
    if ((selector == null)      ||
        (selector.length() < 2) ||
        (selector.charAt(0) != '.'))
    {
      return false;
    }

    for (int i = 1; i < selector.length(); i++)
    {
      if (_isStyleClassTerminator(selector.charAt(i)))
        return false;
    }

    return true;
  }

  /**
   * Returns an
   * Iterator of all of the style class selectors included in the
   * specified selector.  For example, ".OraNav1Enabled" returns
   * a single element Iterator with the string "OraNav1Enabled".
   * "P.OraNav1Enabled SPAN.text" returns a two element Iterator
   * with "OraNav1Enabled" and "text".
   * .OraLink:visited returns "OraLink"
   * .star.moon returns "star" and "moon"
   * namespaced style classes like af|table are not returned
   */
  public static Iterator getNonNamespacedStyleClasses(String selector)
  {
    ArrayList styleClasses = null;
    boolean inAttributeSet = false;
    int styleClassStartIndex = -1;
    int length = selector.length();

    for (int i = 0; i < length; i++)
    {
      char c = selector.charAt(i);

      // attribute set can be id["something"] or img["foo/bar/baz.gif"]
      // inAttributeSet is set to true when we find '['
      // we want to skip till ']' so that we skip the entire attribute set
      if (inAttributeSet)
      {
        if (c == ']')
        {
          inAttributeSet = false;
        }

        // skip until and including ']'
        continue;
      }

      // If we aren't inside of a style class yet,
      // check to see if we are about to enter one.
      // Otherwise, check to see if we are at the
      // end of one.
      if (styleClassStartIndex == -1)
      {
        if (c == '.')
          styleClassStartIndex = i;
        else if (c == '[')
          inAttributeSet = true;

      }
      else
      {
        boolean end = _isStyleClassTerminator(c);

        if (!end)
        {
          // Check to see if we are at the end of the string
          if (i == (length - 1))
          {
            // We need to increment our counter to make sure
            // we include this last character when we pull out
            // the substring
            i++;
            end = true;
          }
        }

        // If we're at the end of the style class, add it to the list
        if (end)
        {
          // if we are at the end due to '[' we want to skip processing the entire
          // attributeSet before we start looking for next selector
          if (c == '[')
            inAttributeSet = true;

          String styleClass = selector.substring(styleClassStartIndex + 1, i);
          if (styleClasses == null)
            styleClasses = new ArrayList(3);

          styleClasses.add(styleClass);
          // if c is ., then this means we've found another styleclass
          if (c == '.')
            styleClassStartIndex = i;
          else
            styleClassStartIndex = -1;
        }
      }
    }

    if (styleClasses == null)
      return null;

    return styleClasses.iterator();
  }

  /**
   * Called when creating the shortened styleclass map.
   * Returns an Iterator of all of the component selectors that begin with the
   * given namespace. All styleclasses and pseudo-classes are ignored and NOT
   * returned.
   * For example, ".OraNav1Enabled af|menuPath" returns
   * a single element Iterator with the string "af|menuPath".
   * "af|menuPath.OraNav1Enabled af|treeTable.text" returns a two
   * element Iterator with "af|menuPath" and "af|treeTable".
   * af|inputText:disabled af|inputText::content returns two
   * "af|inputText" and "af|inputText::content".
   * It also looks at the afSelectorMap to map any special selectors if
   * necessary.
   * e.g., "af|menuPath::step" maps to "af|menuPath A", so
   * ".OraNav1Enabled af|menuPath::step" returns "af|menuPath"
   */
  public static Iterator getNamespacedSelectors(
    String              selector,
    String              namespace,
    Map afSelectorMap)
  {
    if (selector == null)
      return null;
    int afIndex = selector.indexOf(namespace);

    // no namespace in the selector, so just return null
    if (afIndex == -1)
      return null;

    ArrayList afUnmappedSelectorList = new ArrayList();

    // now find each af| component selector and map
    // e.g., af|menuPath::step maps to af|menuPath A

    // order the pseudo elements and classes
    String base = selector.substring(afIndex);
    String[] afSelectors =
        _orderPseudoElementsAndClasses(base);

    // loop through each of the af| parts.
    for (int i=0; i < afSelectors.length; i++)
    {
      if (afSelectors[i].startsWith(namespace))
      {
        // get the component selector, which is just the main part, nothing
        // that includes a '.' ':', or a ' '.
        String afComponentSelector =
          _getNSComponentSelector(afSelectors[i], false);
        afUnmappedSelectorList.add(afComponentSelector);
      }
    }

    // ok, now everything in afSelectorList at this point is the unmapped
    // af|xxx component selectors. Now map them, and return the base key
    // in the selector.
    // loop through again and map them
    ArrayList afSelectorList = new ArrayList();

    int size = afUnmappedSelectorList.size();
    for (int i=0; i < size; i++)
    {
      String afComponentSelector = afUnmappedSelectorList.get(i);
      String mappedSelector = null;

      if (afSelectorMap != null)
      {
        mappedSelector = afSelectorMap.get(afComponentSelector);

      }
      if (mappedSelector != null)
      {
        // In case the mapped selector does not start with the namespace,
        // start the string where the namespace starts.
        int namespaceIndex = mappedSelector.indexOf(namespace);
        if (namespaceIndex > -1)
        {
          // get the selector up until the space.
          String[] baseSelector = StyleUtils.splitStringByWhitespace(mappedSelector.substring(namespaceIndex));
          afComponentSelector = baseSelector[0];
          afSelectorList.add(afComponentSelector);
        }
      }
      else
      {
        afSelectorList.add(afComponentSelector);
      }

    }
    return afSelectorList.iterator();

  }

  /**
   * Add to the namespacePrefixes Set any namespace prefixes found in this selector.
   * @param namespacePrefixes
   * @param selector
   */
  public static void getNamespacePrefixes(
    Set namespacePrefixes,
    String selector)
  {

    int length = selector.length();
    int startSubstringIndex = 0;
    // Loop through each character of the selector looking for namespace prefixes.
    for (int i = 0; i < length; i++)
    {
      char c = selector.charAt(i);
      if (c == '|')
      {
        String prefix = selector.substring(startSubstringIndex, i+1);
        startSubstringIndex = i+1;
        // protect against just | in the prefix by checking length.
        if (prefix.length() > 1)
          namespacePrefixes.add(prefix);
      }
      else if(!_isStyleClassTerminator(c))
      {
        // keep going if it isn't a terminating character
      }
      else
      {
        // update the startSubstring index.
        startSubstringIndex = i+1;
      }
    }
    return;
  }

  /**
   * Called from getNamespacedSelectors.
   * Given a single selector that begins with a namespace prefix (most likely
   * af|), return the main portion -- the prefix+component+pseudo-element.
   * Styleclasses and pseudo-classes and anything after a space will be ignored,
   * and won't be returned.
   * @param singleAfSelector
   * @param allowPseudoClass
   * @return
   */
  private static String _getNSComponentSelector(
    String singleAfSelector,
    boolean allowPseudoClass)
  {
    int colonIndex = singleAfSelector.indexOf("::");

    // get the part after the ::.
    // colonIndex will be 0 if there is no ::.
    if (colonIndex != -1)
    {
      colonIndex += 2;
    }
    else
      colonIndex = 0;
    String afterDoubleColon = singleAfSelector.substring(colonIndex);

    // now find the part with a single ':', a ' ', or a '.'. That is where
    // I want to chop it to get my component selector.

    boolean end = false;
    int afterLength = afterDoubleColon.length();
    int endIndex=0;

    for (; ((endIndex < afterLength) && !end); endIndex++)
    {
      char c = afterDoubleColon.charAt(endIndex);
      if (allowPseudoClass)
        end = Character.isWhitespace(c);
      else
        end = _isStyleClassTerminator(c);
    }

    String afComponentSelector = null;
    if (end)
    {
      afComponentSelector =
        singleAfSelector.substring(0, colonIndex + endIndex-1);
    }
    else
    {
      // there was no end, so just take the entire string.
      afComponentSelector = singleAfSelector;
    }

    return afComponentSelector;
  }

  // Converts all style class selectors that begin with '.'
  // to the short version if
  // there is a short version. does not shorten styles that start with the
  // namespace
  // returns null if it can't shorten the selector
  private static String _getShortNonNamespacedSelector(
    String              selector,
    Map shortStyleClassMap)
  {
    if (shortStyleClassMap == null)
      return null;

    // This will see if the selector is a single styleClass, and if so,
    // it will run it through the shortStyleClassMap and return the shortened
    // style class. A single style class selector is something like
    // ".AFInstructionText". Examples that are not single style class
    // selectors are "af|inputText" or ".AFFoo .AFBar" or ".foo:hover"
    if (isSingleStyleClassSelector(selector))
    {
      String shortStyleClass = shortStyleClassMap.get(selector.substring(1));

      return (shortStyleClass == null) ? null : "." + shortStyleClass;
    }

    // This will run if we do not have a one class selector definition
    // but a af|XXX class or a command class like .AFXYZ .AFzzz or
    // ".foo:hover" It shortens all the .xxx pieces.
    boolean isShorter = false;
    int length = selector.length();
    StringBuffer buffer = new StringBuffer(length);
    int styleClassStartIndex = -1;

    // loop through each character. If we don't come across a '.' character,
    // return the selector unchanged.
    for (int i = 0; i < length; i++)
    {
      char c = selector.charAt(i);

      if (styleClassStartIndex == -1)
      {
        if (c == '.')
          styleClassStartIndex = i;

        buffer.append(c);
      }
      else
      {
        // We are currently inside of a style class substring.
        // Check to see if the style class has been terminated.
        boolean end = _isStyleClassTerminator(c);
        if (!end)
        {
          // Check to see if we are at the end of the string
          if (i == (length - 1))
          {
            // We need to increment our counter to make sure
            // we include this last character when we pull out
            // the substring
            i++;
            end = true;
          }
        }

        if (end)
        {

          // If we are at the end of the style class, check to
          // see if we've got a shorter version
          String styleClass = selector.substring(styleClassStartIndex + 1, i);
          String shortStyleClass = null;
          // don't shorten the styles that contain a namespace, because we
          // do this in another step.
          if (styleClass.indexOf('|') == -1)
            shortStyleClass = shortStyleClassMap.get(styleClass);


          if (shortStyleClass == null)
          {
            buffer.append(styleClass);
          }
          else
          {
            buffer.append(shortStyleClass);
            isShorter = true;
          }

          // Don't forget the terminator character
          if (i < (length - 1))
            buffer.append(c);
          // if c is ., then this means we've found another styleclass
          if (c == '.')
            styleClassStartIndex = i;
          else
            styleClassStartIndex = -1;
        }
      }
    }

    // return the original selector if this isn't shorter.
    return isShorter ? buffer.toString() : selector;
  }
  
  /**
   * Runs a selector through a map. It returns the selector unchanged (except for converted
   * pseudo-classes) if there is no namespace in the selector.
   * This could be a map to convert the
   * public no-html selector to an internal selector that has html-specifics
   * (e.g., 'af|menuPath::step:hover' -> 'af|menuPath A:hover')
   * or it could be a map to shorten the selector
   * (e.g., 'af|menuPath A:hover' -> '.x11 A:hover')
   * We call this method first with the public->internal map, and then
   * to shorten it.
   * Only the pieces of the selector that start with the namespace are mapped.
   * @param afSelectorMap if shortenPass is true, then this map shortens the
   *                 af| selector. else, it maps the public af| selector
   *                 to the internal selector (a selector that is closer to what is written to the
*                    CSS file. 
*                    e.g., af|inputText:error::content becomes 
*                    af|inputText.p_AFError af|inputText::content
   * @param namespacePrefixArray   most likely, "af|". The selectors with this namespace
   *                               are the ones we map.
   * @param selector    selector to map.
   */
  public static String getMappedSelector (
    Map afSelectorMap,
    String[]            namespacePrefixArray,
    String              selector)
  {
    String mappedSelector;
    if (_hasNamespacePrefix(selector, namespacePrefixArray))
    {
      String[] selectorArray =
        _orderPseudoElementsAndClasses(selector);

      // map selectors, if needed
      // any of the selectors that start with a namespace prefix will
      // be mapped.
      // e.g., "af|menuPath::step" maps to "af|menuPath A"
      // if the selector does not need to be 'mapped', it will be returned
      // untouched. e.g., .AFInstructionText maps to .AFInstructionText
      mappedSelector = _getMappedNSSelector(afSelectorMap,
                                            namespacePrefixArray,
                                            selectorArray,
                                            false);
    }
    else
    {
      // there are no namespaces in this selector, but we still need to convert pseudo-classes.
      //Separate pseudoclasses and then call it
      int start = 0;
      StringBuilder b = new StringBuilder();
      for(int i = 0; i < selector.length(); i++)
      {
        char c = selector.charAt(i);
        if (c == ' ' )
        {
          if (start == i)
          {
            //group of spaces
            start = i+1; //Skip space
          }
          else
          {
            String subSelector = selector.substring(start,i);
            subSelector = _convertPseudoClassesInSelector(subSelector, selector);
            start = i+1; //Skip space
            b.append(subSelector);
            b.append(' ');
          }
        }
      } // end for loop

      // We reached the end of the selector, now convert the last bit
      if (start == 0)
      {
        //there is no space in selector, but we still need to map pseudo-classes.
        mappedSelector = _convertPseudoClassesInSelector(selector, selector);
      }
      else
      {
        String subSelector = selector.substring(start);
        subSelector = _convertPseudoClassesInSelector(subSelector, selector);
        b.append(subSelector);
        mappedSelector = b.toString();
      }

    }
    return mappedSelector;
  }

  /**
   * Runs a namespaced selector through a map.
   * This could be a map to convert the
   * public no-html selector to an internal selector that has html-specifics
   * (e.g., 'af|menuPath::step:hover' -> 'af|menuPath A:hover')
   * or it could be a map to shorten the selector
   * (e.g., 'af|menuPath A:hover' -> '.x11 A:hover')
   * We call this method first with the public->internal map, and then
   * to shorten it.
   * Only the pieces of the selector that start with the namespace are mapped.
   * @param map         if shortenPass is true, then this map shortens the
   *                    af| selector. else, it maps the public af| selector
   *                    to the internal selector.
   * @param selectorArray selectorArray is the selector split into pieces based on the ' '
   * @param shorten     if true, then we'll add the "." to the mapped selector.
   * @return            the selector, mapped.
   */
  private static String _getMappedNSSelector (
    Map map,
    String[]            nsPrefixArray,
    String[]            selectorArray,
    boolean             shorten)
  {
    // selectorArray is the selector broken into pieces.
    // Map each piece, then put back together.

    for (int i=0; i < selectorArray.length; i++)
    {
      // Assumption is the selector 'piece' only contains one namespace
      // prefix. af|foo.tr|bar is not a valid selector, so this assumption
      // is fine.
      int nsIndex = -1;
      int numNsPrefixes = nsPrefixArray.length;
      for (int j=0; (j <  numNsPrefixes) && (nsIndex == -1); j++)
      {
       nsIndex = selectorArray[i].indexOf(nsPrefixArray[j]);
      }

      if (nsIndex > -1)
      {
        selectorArray[i] = _getEachMappedSelector(map,
                                                  nsIndex,
                                                  selectorArray[i],
                                                  shorten);
      }
    }

    // build back the selectorArray into a string
    return StyleUtils.arrayToStringWithSpaces(selectorArray);
  }

  private static String _getEachMappedSelector(
    Map map,
    int                 indexOfNSPrefix,
    String              selector,
    boolean             shorten)
  {
    // break apart the selector into 2 parts:
    // main (af|foo::pseudo-element)
    // end (anything after, ':' or including and after '.')
    // map the main and the ending (pseudo-classes could be mapped to private
    // styleclasses, :disabled -> .p_AFDisabled
    // piece  back together.
    if (indexOfNSPrefix == -1)
     return selector;
    if (map == null)
      return selector;


    String wholeAfSelector = selector.substring(indexOfNSPrefix);

    // get main piece
    int firstDoubleColonIndex = wholeAfSelector.indexOf("::");

    // get the part after the ::.
    // colonIndex will be 0 if there is no ::.
    if (firstDoubleColonIndex != -1)
    {
      int lastDoubleColonIndex = wholeAfSelector.lastIndexOf("::");
      
      // We exceptionally support multiple occurence of double colon for the browser builtin pseudo elements
      if (lastDoubleColonIndex != firstDoubleColonIndex && 
          ! _BUILT_IN_PSEUDO_ELEMENTS.contains(wholeAfSelector.substring(lastDoubleColonIndex)))
      {
        _LOG.warning("UNSUPPORTED_CONSECUTIVE_SUB_ELEMENT_SYNTAX", selector);
      }

      firstDoubleColonIndex += 2;
    }
    else
      firstDoubleColonIndex = 0;
    String afterDoubleColon = wholeAfSelector.substring(firstDoubleColonIndex);

    // now find the part with a single ':', a ' ', a '[', or a '.'. That is where
    // I want to chop it to get my 'main' piece of the component selector.
    boolean end = false;
    int afterLength = afterDoubleColon.length();
    int endIndex=0;
    char c;
    for (; ((endIndex < afterLength) && !end); endIndex++)
    {
      c = afterDoubleColon.charAt(endIndex);
      end = (Character.isWhitespace(c)) || (c == '.') || (c == ':') || (c == '[');
    }

    // Set the main piece in the pieces object
    String mainSelector;
    if (endIndex == afterLength)
    {
      mainSelector = wholeAfSelector.substring(0, firstDoubleColonIndex + endIndex);
    }
    else
    {
      mainSelector = wholeAfSelector.substring(0, firstDoubleColonIndex + endIndex-1);
    }
    String afterMain = null;
    if (endIndex != afterLength)
    {
      // If I got here, that means there are characters after the 'main' part
      // of the selector.
      afterMain = wholeAfSelector.substring(firstDoubleColonIndex + endIndex-1);
      
      // map the afterMain part. It includes styleclasses and pseudo-classes.
      // we don't need to convert the pseudo-classes if we are shortening.
      // Also very likely we are dealing with inbuilt pseudo elements that 
      //  should not be converted, do not process them as pseudo classes.
      if (!shorten && ! _BUILT_IN_PSEUDO_ELEMENTS.contains(afterMain))
        afterMain = _convertPseudoClassesInSelector(afterMain, wholeAfSelector);
    }

    // piece back together the string
    StringBuffer buffer = new StringBuffer();

    // beginning. add '.' if needed.
    if (indexOfNSPrefix > 0)
    {
      buffer.append(selector.substring(0, indexOfNSPrefix));
      if (shorten && selector.charAt(indexOfNSPrefix-1) != '.')
        buffer.append('.');
    }
    else if (shorten)
      buffer.append('.');

    // map the mainSelector, and append the afterMain part.
    buffer.append(_runThroughMap(map, mainSelector));
    if (afterMain != null)
      buffer.append(afterMain);


    return buffer.toString();
  }

  private static boolean _hasNamespacePrefix(
    String   selector,
    String[] nsPrefixArray)
  {
    if (selector == null) return false;
    boolean hasNamespacePrefix = false;
    int numNamespaces = nsPrefixArray.length;
    for (int i=0; (i <  numNamespaces )&& !hasNamespacePrefix; i++)
      if (selector.indexOf(nsPrefixArray[i]) > -1)
        hasNamespacePrefix = true;
    return hasNamespacePrefix;
  }

  /*
   * if I see this: af|foo:class:class::element, return this
   * af|foo:class:class and af|foo::element in a String array.
   * if I see this: af|foo:p-class.StyleClass::element, return this:
   * 'af|foo:p-class.StyleClass' 'af|foo::element'
   * af|foo.StyleClass::element -> 'af|foo.StyleClass' 'af|foo::element'
   * If I see thiss: af|foo::p-element.StyleClass, return this
   * af|foo::p-element.StyleClass (leave alone).
   */
  private static String[]  _orderPseudoElementsAndClasses(
    String selector)
  {
    String[] input = StyleUtils.splitStringByWhitespace(selector);

    List output = new ArrayList();
    for (int i=0; i < input.length; i++)
    {

      int indexOfDoubleColon = input[i].indexOf("::");
      if (indexOfDoubleColon == -1)
      {
        // no double colon (aka pseudo-element)
        output.add(input[i]);
      }
      else
      {
        // you have a double colon index. Now look to see if we need to
        // reorder. We have to reorder if the pseudo-element comes after
        // the pseudo-class or composite style classes.
        int indexOfFirstColon = input[i].indexOf(':');
        int indexOfDot = input[i].indexOf('.');

        boolean pseudoClassBeforePseudoElement =
          (indexOfFirstColon < indexOfDoubleColon);
        boolean styleClassBeforePseudoElement =
          (indexOfDot != -1 && indexOfDot < indexOfDoubleColon);

        if (!(pseudoClassBeforePseudoElement ||
              styleClassBeforePseudoElement))
        {
          output.add(input[i]);
        }
        else
        {
          if (indexOfFirstColon == indexOfDoubleColon)
            indexOfFirstColon = -1;
          int indexOfClass = Math.min(indexOfFirstColon, indexOfDot);
          if (indexOfClass == -1)
            indexOfClass = Math.max(indexOfFirstColon, indexOfDot);

          // we have the condition where pseudo-class or styleClass is before
          // pseudo-element: af|foo:psdo-class::psdo-element
          // e.g.,
          // af|inputText:disabled::content
          // indexOfColon = 12, indexOfDoubleColon = 21
          // main = 'af|inputText'
          // mainPlusClasses = 'af|inputText:disabled'
          // end '::content'
          String main = input[i].substring(0, indexOfClass);
          String mainPlusClasses = input[i].substring(0, indexOfDoubleColon);
          String end = input[i].substring(indexOfDoubleColon);
          output.add(mainPlusClasses);
          output.add(main + end);
        }
      }

    }
    return output.toArray(new String[output.size()]);
  }



  /**
   * get rid of the | and :: that browsers don't like, and add the
   * '.' where needed to make the af| component selector
   * into a style class selector that is valid to be written to the css file.
   * @param selector
   * @return
   */
  public static String getValidFullNameSelector(
    String selector,
    String[] namespacePrefixArray)
  {
    if (selector.indexOf('|') == -1)
      return selector;

    // split on spaces.
    String[] spacerArray = StyleUtils.splitStringByWhitespace(selector);

    for (int i=0; i < spacerArray.length; i++)
    {
      // if this starts with any of our namespaces, then add a '.'
      if (spacerArray[i].indexOf('|') > -1)
      {
        for (int j=0; j < namespacePrefixArray.length; j++)
        {
          String nsPrefix = namespacePrefixArray[j];
          if (spacerArray[i].startsWith(nsPrefix))
          {
            spacerArray[i] = ".".concat(spacerArray[i]);
            break;
          }
        }
      }
    }
    return StyleUtils.convertToValidSelector(
      StyleUtils.arrayToStringWithSpaces(spacerArray));

  }


  // Tests whether the specified character or pseudo-class terminates a
  // style class selector
  private static boolean _isStyleClassTerminator(char c)
  {
    return (Character.isWhitespace(c) || (c == ':') || (c == '.') || (c == '['));
  }

  private static int _computeBuilderSize(PropertyNode[] properties)
  {
    // Compute the StringBuilder size
    int builderSize = 0;
    
    for (int i = 0; i < properties.length; i++)
    {
      PropertyNode property = properties[i];
      String name = property.getName();
      String value = property.getValue();

      if ((name != null) && (value != null))
      {
        builderSize += property.getName().length();
        builderSize += property.getValue().length();

        // Leave room for a separator
        builderSize++;
      }
    }
    
    return builderSize;
  }

  private static String _getSortedPropertyString(PropertyNode[] sortedProperties)
  {
    int builderSize = _computeBuilderSize(sortedProperties);
    
    StringBuilder builder = new StringBuilder(builderSize);
    boolean first = true;

    for (int i = 0; i < sortedProperties.length; i++)
    {
      PropertyNode property = sortedProperties[i];
      String       name     = property.getName();
      String       value    = property.getValue();

      if ((name != null) && (value != null))
      {
        if (!first)
          builder.append(';');
        else
          first = false;

        builder.append(name);
        builder.append(':');
        builder.append(value);
      }
    }

    return builder.toString();
  }
  
  /**
   * Returns the properties of the specified StyleNode in property name sorted order as a String.
   * If the Stylenode contains no properties, the empty String is returned
   */
  private static String _getSortedPropertyString(StyleNode styleNode)
  {
    Collection nodeProperties = styleNode.getProperties();
    int propertyCount = nodeProperties.size();
      
    if (propertyCount == 0)
      return "";
    
    // =-= bts the extra step of copying the PropertyNodes into the array is kind of lame.  I'm wondering
    // if it would be better for the StyleNodes to sort the properties for us
    List properties       = new ArrayList(nodeProperties);
    PropertyNode[]     sortedProperties = properties.toArray(new PropertyNode[propertyCount]);

    // Sort the properties so that the order of the properties won't
    // come into play when comparing property strings.
    Arrays.sort(sortedProperties, PropertyNodeNameComparator.sharedInstance());

    return _getSortedPropertyString(sortedProperties);
  }

  /**
   * Given a selector, convert the pseudo-classes. Non css-2 pseudo-classes
   * get converted to styleclasses so they don't get passed through to the generated css.
   *
   * @param selector String input selector. The assumption is there are no spaces.
   * If the original selector has spaces, then break the selector up into the pieces before
   * calling this method.
   * @param completeSelector to figure out if the selector is AF namespaced
   * @return String the selector with the pseudo-classes converted, if needed.
   * e.g., .StyleClass:error:active -> .StyleClass.p_AFError:active
   */
  private static String _convertPseudoClassesInSelector(String selector, String completeSelector)
  {
    if (selector == null || completeSelector ==null) return selector;

    boolean afNamespacedSelector = _isAFNamespacedSelector(completeSelector);

    StringBuffer completeBuffer = new StringBuffer();
    StringBuffer pseudoClassBuffer = new StringBuffer();
    boolean inPseudoClass = false;

    for (int i=0; i < selector.length(); i++)
    {
      char x = selector.charAt(i);

      if ((x == ':') || (x == '.') || (x == '['))
      {
        if (inPseudoClass)
        {
          // if we are in a pseudo-class already, and we get a ':' or '.' that means
          // this pseudo-class is complete. Get ready for another one.
          String convertedPseudoClass = _convertPseudoClass(completeSelector, pseudoClassBuffer.toString(), afNamespacedSelector);
          completeBuffer.append(convertedPseudoClass);
          pseudoClassBuffer = new StringBuffer();
          inPseudoClass = false;
        }
        if (x == ':')
        {
          inPseudoClass = true;
          pseudoClassBuffer.append(x);
        }
        else if (x == '.' || x == '[')
        {
          completeBuffer.append(x);
        }

      }
      else
      {
        if (!inPseudoClass)
          completeBuffer.append(x);
        else
          pseudoClassBuffer.append(x);
      }

    }
    if (inPseudoClass)
    {
      String mappedPseudoClass = _convertPseudoClass(completeSelector, pseudoClassBuffer.toString(), afNamespacedSelector);
      completeBuffer.append(mappedPseudoClass);

    }
    return completeBuffer.toString();
  }

  /**
   * Returns true if the passed selector is a 'af' selector.
   *
   * @param completeSelector
   * @return
   */
  private static boolean _isAFNamespacedSelector(String completeSelector)
  {

    Set namespaces = new HashSet();
    getNamespacePrefixes(namespaces, completeSelector);

    if (namespaces.isEmpty() && completeSelector.startsWith(_DEFAULT_AF_SELECTOR))
      return true;
    if (namespaces.contains(_DEFAULT_NAMESPACE))
      return true;

    return false;
  }

  // run through the map. If it's not in the map, return the selector unchanged.
  private static String _runThroughMap(Map map, String selector)
  {
    String mappedSelector = map.get(selector);
    return (mappedSelector != null) ? mappedSelector : selector;
  }


  // Comparator that sorts PropertyNodes by name
  private static class PropertyNodeNameComparator implements Comparator
  {
    public static Comparator sharedInstance()
    {
      return _INSTANCE;
    }

    @Override
    public int compare(PropertyNode o1, PropertyNode o2)
    {
      String name1 = (o1 == null) ? null : o1.getName();
      String name2 = (o2 == null) ? null : o2.getName();

      if ((name1 == null) || (name2 == null) )
      {
        if (name1 == name2)
          return 0;
        else if (name1 != null)
          return 1;

        return -1;
      }

      return name1.compareTo(name2);
    }
    
    @Override
    public boolean equals(Object o)
    {
      // we only have our singleton instance
      return (this == o);
    }

    private PropertyNodeNameComparator() {}

    private static final Comparator _INSTANCE = new PropertyNodeNameComparator();
  }

  /**
   * Converts the pseudo class into a "p_AF" prefixed styleclass.
   * This is because certain pseudo-classes are internal and are not recognized by any browsers.
   * The renderers are responsible for rendering out the same styleclass format.
   * Some examples are: 
   * :screen-reader becomes .p_AFScreenReader
   * :disabled becomes .p_AFDisabled
   *
   * Exception is CSS built in pseudo classes, which are recognized by browsers. These are rendered as is.
   *
   * @param pseudoClass
   * @param afNamespacedSelector true if the pseudoClass is part of AF namespaced selector
   * @return
   */
  static private String _convertPseudoClass(String completeSelector, String pseudoClass, boolean afNamespacedSelector)
  {
    // The design time needs the browser-supported pseudo-classes to be converted so they
    // can show a preview of the skinned component.
    String builtInPseudoClass = pseudoClass;
    int parenthesesIndex = pseudoClass.indexOf("(");

    // if there are no open parentheses check for closing parentheses
    if (parenthesesIndex == -1)
      parenthesesIndex = pseudoClass.indexOf(")");

    // if at least one parentheses exists strip the pseudoClass off it
    if (parenthesesIndex != -1)
      builtInPseudoClass = pseudoClass.substring(0, parenthesesIndex);

    if (_BUILT_IN_PSEUDO_CLASSES.contains(builtInPseudoClass) && !Beans.isDesignTime())
      return pseudoClass;

    // skip the pseudo selectors in _AT_PAGE_PSEUDO_CLASSES only for @page client rule
    if (completeSelector.contains(_AT_PAGE_SELECTOR) && _AT_PAGE_PSEUDO_CLASSES.contains(builtInPseudoClass))
      return pseudoClass;

    // _BACKWARD_COMPATIBLE_CSS3_PSEUDO_CLASSES is treated differently
    // for namespaced selectors we render it prefixed with "p_AF"
    // for non-namespaced selectors we render it directly
    if (!afNamespacedSelector && _BACKWARD_COMPATIBLE_CSS3_PSEUDO_CLASSES.contains(builtInPseudoClass))
      return pseudoClass;

    StringBuilder builder = new StringBuilder(pseudoClass.length() + 3);
    builder.append(".");
    builder.append(SkinSelectors.STATE_PREFIX);

    for (String content : _DASH_PATTERN.split(pseudoClass.substring(1)))
    {
      if (content.length() > 0)
      {
        builder.append(Character.toUpperCase(content.charAt(0)));
        builder.append(content.substring(1));
      }
    }

    return builder.toString();
  }

  /**
   * encapsulates the propertyString vs StyleNode maps separately for clientRule and normal selectors
   * ClientRule selectors the styles are rendered as:
   * clientRule { selector: {property1: value1, property2: value2 ... } }
   * Normal selectors are rendered as
   * selector: {property1: value1 ... }
   */
  private static final class MatchingStyles
  {
    private MatchingStyles(int styleCount)
    {
      this._nonClientRuleEntry = new MatchingEntry(styleCount);
      this._clientRuleEntries = new LinkedHashMap();
    }


    public LinkedHashMap> getNonClientRuleMap()
    {
      return _nonClientRuleEntry.getMatchingStyles();
    }

    public LinkedHashMap> getClientRuleMap(String clientRule)
    {
      return _clientRuleEntries.get(clientRule).getMatchingStyles();
    }

    public Set getClientRules()
    {
      return _clientRuleEntries.keySet();
    }

    public void addStyle(String propertyString, StyleNode styleNode)
    {
      _nonClientRuleEntry.addMatchingStyle(propertyString, styleNode);
    }

    public void addStyle(String clientRule, String propertyString, StyleNode styleNode)
    {
      MatchingEntry matchingStylesForClientRule = _clientRuleEntries.get(clientRule);

      if (matchingStylesForClientRule == null)
      {
        matchingStylesForClientRule = new MatchingEntry(1);
        _clientRuleEntries.put(clientRule, matchingStylesForClientRule);
      }

      matchingStylesForClientRule.addMatchingStyle(propertyString, styleNode);
    }

    // matching styles map with no client rules
    private final MatchingEntry _nonClientRuleEntry;

    // matching styles map with client rules
    private final LinkedHashMap _clientRuleEntries;
  }

  /**
   * encapsulates propertyString vs StyleNode map
   * manages addition of StyleNode for a matching propertyString
   */
  private final static class MatchingEntry
  {
    public MatchingEntry(int initialSize)
    {
      _matchingStyles = new LinkedHashMap>(initialSize);
    }

    public void addMatchingStyle(String propertyString, StyleNode styleNode)
    {
      if (propertyString == null || propertyString.equals(""))
        return;


      // See if we already have a StyleNode with the same properties
      List matchingStyles = _matchingStyles.get(propertyString);

      if (matchingStyles == null)
      {
        // If we don't already have matching StyleNodes, create a new match list and cache it
        matchingStyles = new ArrayList(1);
        _matchingStyles.put(propertyString, matchingStyles);
      }

      matchingStyles.add(styleNode);
    }

    public LinkedHashMap> getMatchingStyles()
    {
      return _matchingStyles;
    }

    @Override
    public boolean equals(Object o)
    {
      if (this == o) return true;
      if (o == null || getClass() != o.getClass()) return false;

      MatchingEntry that = (MatchingEntry) o;

      if (_matchingStyles != null ? !_matchingStyles.equals(that._matchingStyles) : that._matchingStyles != null)
        return false;

      return true;
    }

    @Override
    public int hashCode()
    {
      return _matchingStyles != null ? _matchingStyles.hashCode() : 0;
    }

    private LinkedHashMap> _matchingStyles;
  }

  // We want to output to the css the browser-supported pseudo-classes as is
  static private final Set _BUILT_IN_PSEUDO_CLASSES = new HashSet();
  static private final Set _BACKWARD_COMPATIBLE_CSS3_PSEUDO_CLASSES = new HashSet();

  // We want to output to the css the browser-supported pseudo-elements (HTML5/ CSS3) as is
  static private final Set _BUILT_IN_PSEUDO_ELEMENTS = new HashSet();
  static private final Set _AT_PAGE_PSEUDO_CLASSES = new HashSet();

  static
  {
    /** CSS 2 pseudo classes */
    _BUILT_IN_PSEUDO_CLASSES.add(":first-child");
    _BUILT_IN_PSEUDO_CLASSES.add(":link");
    _BUILT_IN_PSEUDO_CLASSES.add(":visited");
    _BUILT_IN_PSEUDO_CLASSES.add(":hover");
    _BUILT_IN_PSEUDO_CLASSES.add(":active");
    _BUILT_IN_PSEUDO_CLASSES.add(":focus");
    _BUILT_IN_PSEUDO_CLASSES.add(":-moz-placeholder"); // FF <= 18
    _BUILT_IN_PSEUDO_CLASSES.add(":-ms-input-placeholder"); // IE

    /** Special case CSS2 pseudo-elements used as pseudo-classes
     * for compatibility reasons.
     * Refer: http://www.w3.org/TR/selectors/#pseudo-elements */
    _BUILT_IN_PSEUDO_CLASSES.add(":after");
    _BUILT_IN_PSEUDO_CLASSES.add(":before");
    _BUILT_IN_PSEUDO_CLASSES.add(":first-line");
    _BUILT_IN_PSEUDO_CLASSES.add(":first-letter");

    /** CSS 3 pseudo classes */
    _BUILT_IN_PSEUDO_CLASSES.add(":nth-child");
    _BUILT_IN_PSEUDO_CLASSES.add(":nth-last-child");
    _BUILT_IN_PSEUDO_CLASSES.add(":nth-of-type");
    _BUILT_IN_PSEUDO_CLASSES.add(":nth-last-of-type");
    _BUILT_IN_PSEUDO_CLASSES.add(":last-child");
    _BUILT_IN_PSEUDO_CLASSES.add(":first-of-type");
    _BUILT_IN_PSEUDO_CLASSES.add(":last-of-type");
    _BUILT_IN_PSEUDO_CLASSES.add(":only-child");
    _BUILT_IN_PSEUDO_CLASSES.add(":only-of-type");
    _BUILT_IN_PSEUDO_CLASSES.add(":root");
    _BUILT_IN_PSEUDO_CLASSES.add(":target");
    _BUILT_IN_PSEUDO_CLASSES.add(":enabled");
    _BUILT_IN_PSEUDO_CLASSES.add(":checked");
    _BUILT_IN_PSEUDO_CLASSES.add(":not");
    _BUILT_IN_PSEUDO_CLASSES.add(":lang");
    _BUILT_IN_PSEUDO_CLASSES.add(":indeterminate");

    /** CSS3 pseudo classes for backward compatibility*/
    _BACKWARD_COMPATIBLE_CSS3_PSEUDO_CLASSES.add(":disabled");
    _BACKWARD_COMPATIBLE_CSS3_PSEUDO_CLASSES.add(":empty");

    /** CSS3 pseudo elements */
    _BUILT_IN_PSEUDO_ELEMENTS.add("::outside");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::before");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::after");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::alternate");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::first-line");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::first-letter");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::marker");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::line-marker");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::selection");
    _BUILT_IN_PSEUDO_ELEMENTS.add("::-webkit-input-placeholder"); // Chrome
    _BUILT_IN_PSEUDO_ELEMENTS.add("::-moz-placeholder"); // FF >= 19
    
    /** @page pseudo classes*/
    _AT_PAGE_PSEUDO_CLASSES.add(":first");
    _AT_PAGE_PSEUDO_CLASSES.add(":left");
    _AT_PAGE_PSEUDO_CLASSES.add(":right");
  }
  
  private static final Pattern _DASH_PATTERN =  Pattern.compile("-");
  private static final int _MSIE_SELECTOR_LIMIT = 4095;
  private static final String _DEFAULT_NAMESPACE = "af|";
  private static final String _DEFAULT_AF_SELECTOR = ".AF";
  private static final String _AT_PAGE_SELECTOR = "@page";

  private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger(CSSGenerationUtils.class);
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy