com.helger.css.decl.visit.CSSVisitor Maven / Gradle / Ivy
/**
* Copyright (C) 2014-2016 Philip Helger (www.helger.com)
* philip[at]helger[dot]com
*
* Licensed 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 com.helger.css.decl.visit;
import javax.annotation.Nonnull;
import javax.annotation.concurrent.Immutable;
import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.PresentForCodeCoverage;
import com.helger.css.decl.CSSDeclaration;
import com.helger.css.decl.CSSFontFaceRule;
import com.helger.css.decl.CSSImportRule;
import com.helger.css.decl.CSSKeyframesBlock;
import com.helger.css.decl.CSSKeyframesRule;
import com.helger.css.decl.CSSMediaRule;
import com.helger.css.decl.CSSNamespaceRule;
import com.helger.css.decl.CSSPageMarginBlock;
import com.helger.css.decl.CSSPageRule;
import com.helger.css.decl.CSSSelector;
import com.helger.css.decl.CSSStyleRule;
import com.helger.css.decl.CSSSupportsRule;
import com.helger.css.decl.CSSUnknownRule;
import com.helger.css.decl.CSSViewportRule;
import com.helger.css.decl.CascadingStyleSheet;
import com.helger.css.decl.ICSSPageRuleMember;
import com.helger.css.decl.ICSSTopLevelRule;
import com.helger.css.decl.IHasCSSDeclarations;
/**
* This class is used to walk a CSS domain object and call the respective
* {@link ICSSVisitor} and {@link ICSSUrlVisitor} interface methods. Method
* {@link #visitCSS(CascadingStyleSheet, ICSSVisitor)} is usually the entry
* point to iterate over the whole content of a parsed CSS.
*
* @author Philip Helger
*/
@Immutable
public final class CSSVisitor
{
@PresentForCodeCoverage
@SuppressWarnings ("unused")
private static final CSSVisitor s_aInstance = new CSSVisitor ();
private CSSVisitor ()
{}
/**
* Visit all elements of a single import rule.
*
* @param aImportRule
* The import rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitImportRule (@Nonnull final CSSImportRule aImportRule, @Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onImport (aImportRule);
}
/**
* Visit all elements of a single namespace rule.
*
* @param aNamespaceRule
* The namespace rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitNamespaceRule (@Nonnull final CSSNamespaceRule aNamespaceRule,
@Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onNamespace (aNamespaceRule);
}
/**
* Visit all declarations contained in the passed declaration container.
*
* @param aHasDeclarations
* The declarations to be visited. May not be null
.
* @param aVisitor
* The visitor to be invoked on each declaration. May not be
* null
.
*/
public static void visitAllDeclarations (@Nonnull final IHasCSSDeclarations > aHasDeclarations,
@Nonnull final ICSSVisitor aVisitor)
{
// for all declarations
for (final CSSDeclaration aDeclaration : aHasDeclarations.getAllDeclarations ())
aVisitor.onDeclaration (aDeclaration);
}
/**
* Visit all elements of a single style rule.
*
* @param aStyleRule
* The style rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitStyleRule (@Nonnull final CSSStyleRule aStyleRule, @Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginStyleRule (aStyleRule);
try
{
// for all selectors
for (final CSSSelector aSelector : aStyleRule.getAllSelectors ())
aVisitor.onStyleRuleSelector (aSelector);
// for all declarations
visitAllDeclarations (aStyleRule, aVisitor);
}
finally
{
aVisitor.onEndStyleRule (aStyleRule);
}
}
/**
* Visit all elements of a single page rule.
*
* @param aPageRule
* The page rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitPageRule (@Nonnull final CSSPageRule aPageRule, @Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginPageRule (aPageRule);
try
{
// for all declarations
for (final ICSSPageRuleMember aMember : aPageRule.getAllMembers ())
if (aMember instanceof CSSDeclaration)
aVisitor.onDeclaration ((CSSDeclaration) aMember);
else
{
final CSSPageMarginBlock aPageMarginBlock = (CSSPageMarginBlock) aMember;
aVisitor.onBeginPageMarginBlock (aPageMarginBlock);
try
{
// for all declarations
visitAllDeclarations (aPageMarginBlock, aVisitor);
}
finally
{
aVisitor.onEndPageMarginBlock (aPageMarginBlock);
}
}
}
finally
{
aVisitor.onEndPageRule (aPageRule);
}
}
/**
* Visit all elements of a single font-face rule.
*
* @param aFontFaceRule
* The font-face rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitFontFaceRule (@Nonnull final CSSFontFaceRule aFontFaceRule,
@Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginFontFaceRule (aFontFaceRule);
try
{
// for all declarations
visitAllDeclarations (aFontFaceRule, aVisitor);
}
finally
{
aVisitor.onEndFontFaceRule (aFontFaceRule);
}
}
/**
* Visit all elements of a single media rule.
*
* @param aMediaRule
* The media rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitMediaRule (@Nonnull final CSSMediaRule aMediaRule, @Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginMediaRule (aMediaRule);
try
{
// for all nested rules
for (final ICSSTopLevelRule aRule : aMediaRule.getAllRules ())
visitTopLevelRule (aRule, aVisitor);
}
finally
{
aVisitor.onEndMediaRule (aMediaRule);
}
}
/**
* Visit all elements of a single keyframes rule.
*
* @param aKeyframesRule
* The keyframes rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitKeyframesRule (@Nonnull final CSSKeyframesRule aKeyframesRule,
@Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginKeyframesRule (aKeyframesRule);
try
{
// for all keyframes blocks
for (final CSSKeyframesBlock aBlock : aKeyframesRule.getAllBlocks ())
{
aVisitor.onBeginKeyframesBlock (aBlock);
try
{
// for all declarations
visitAllDeclarations (aBlock, aVisitor);
}
finally
{
aVisitor.onEndKeyframesBlock (aBlock);
}
}
}
finally
{
aVisitor.onEndKeyframesRule (aKeyframesRule);
}
}
/**
* Visit all elements of a single viewport rule.
*
* @param aViewportRule
* The viewport rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitViewportRule (@Nonnull final CSSViewportRule aViewportRule,
@Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginViewportRule (aViewportRule);
try
{
// for all declarations
visitAllDeclarations (aViewportRule, aVisitor);
}
finally
{
aVisitor.onEndViewportRule (aViewportRule);
}
}
/**
* Visit all elements of a single supports rule.
*
* @param aSupportsRule
* The supports rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitSupportsRule (@Nonnull final CSSSupportsRule aSupportsRule,
@Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onBeginSupportsRule (aSupportsRule);
try
{
// for all nested rules
for (final ICSSTopLevelRule aRule : aSupportsRule.getAllRules ())
visitTopLevelRule (aRule, aVisitor);
}
finally
{
aVisitor.onEndSupportsRule (aSupportsRule);
}
}
/**
* Visit all elements of a single unknown @ rule.
*
* @param aUnknownRule
* The unknown rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitUnknownRule (@Nonnull final CSSUnknownRule aUnknownRule, @Nonnull final ICSSVisitor aVisitor)
{
aVisitor.onUnknownRule (aUnknownRule);
}
/**
* Visit all elements of a single top-level rule. This includes all rules
* except @import
and @namespace
rules.
*
* @param aTopLevelRule
* The top-level rule to visit. May not be null
.
* @param aVisitor
* The visitor to use. May not be null
.
*/
public static void visitTopLevelRule (@Nonnull final ICSSTopLevelRule aTopLevelRule,
@Nonnull final ICSSVisitor aVisitor)
{
if (aTopLevelRule instanceof CSSStyleRule)
{
visitStyleRule ((CSSStyleRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSPageRule)
{
visitPageRule ((CSSPageRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSFontFaceRule)
{
visitFontFaceRule ((CSSFontFaceRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSMediaRule)
{
visitMediaRule ((CSSMediaRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSKeyframesRule)
{
visitKeyframesRule ((CSSKeyframesRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSViewportRule)
{
visitViewportRule ((CSSViewportRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSSupportsRule)
{
visitSupportsRule ((CSSSupportsRule) aTopLevelRule, aVisitor);
}
else
if (aTopLevelRule instanceof CSSUnknownRule)
{
visitUnknownRule ((CSSUnknownRule) aTopLevelRule, aVisitor);
}
else
throw new IllegalStateException ("Top level rule " + aTopLevelRule + " is unsupported!");
}
/**
* Visit all CSS elements in the order of their declaration. import rules come
* first, namespace rules come next and all other top-level rules in the order
* of their declaration.
*
* @param aCSS
* The CSS to visit. May not be null
.
* @param aVisitor
* The callback to be invoked for each element found. May not be
* null
.
*/
public static void visitCSS (@Nonnull final CascadingStyleSheet aCSS, @Nonnull final ICSSVisitor aVisitor)
{
ValueEnforcer.notNull (aCSS, "CSS");
ValueEnforcer.notNull (aVisitor, "Visitor");
aVisitor.begin ();
try
{
// for all imports
for (final CSSImportRule aImportRule : aCSS.getAllImportRules ())
visitImportRule (aImportRule, aVisitor);
// for all namespaces
for (final CSSNamespaceRule aNamespaceRule : aCSS.getAllNamespaceRules ())
visitNamespaceRule (aNamespaceRule, aVisitor);
// for all other top level rules
for (final ICSSTopLevelRule aTopLevelRule : aCSS.getAllRules ())
visitTopLevelRule (aTopLevelRule, aVisitor);
}
finally
{
aVisitor.end ();
}
}
/**
* Visit all items that can contain URLs in CSS files. Therefore the special
* visitor class {@link CSSVisitorForUrl} is used.
*
* @param aCSS
* The CSS to visit. May not be null
.
* @param aVisitor
* The callback to invoke for each found occurrence. May not be
* null
.
* @see CSSVisitorForUrl
*/
public static void visitCSSUrl (@Nonnull final CascadingStyleSheet aCSS, @Nonnull final ICSSUrlVisitor aVisitor)
{
// Visit only the URLs of a CSS with a specific CSS visitor
visitCSS (aCSS, new CSSVisitorForUrl (aVisitor));
}
/**
* Visit all items that can contain URLs in CSS files. Therefore the special
* visitor class {@link CSSVisitorForUrl} is used.
*
* @param aCSS
* The CSS to visit. May not be null
.
* @param aVisitor
* The callback to invoke for each found occurrence. May not be
* null
.
*/
public static void visitAllDeclarationUrls (@Nonnull final IHasCSSDeclarations > aCSS,
@Nonnull final ICSSUrlVisitor aVisitor)
{
// Visit only the URLs of a CSS with a specific CSS visitor
visitAllDeclarations (aCSS, new CSSVisitorForUrl (aVisitor));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy