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

com.alee.managers.style.data.SkinInfoConverter Maven / Gradle / Ivy

/*
 * This file is part of WebLookAndFeel library.
 *
 * WebLookAndFeel library is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * WebLookAndFeel library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with WebLookAndFeel library.  If not, see .
 */

package com.alee.managers.style.data;

import com.alee.api.annotations.NotNull;
import com.alee.api.annotations.Nullable;
import com.alee.api.merge.Merge;
import com.alee.api.resource.ClassResource;
import com.alee.managers.icon.set.IconSet;
import com.alee.managers.style.Skin;
import com.alee.managers.style.StyleException;
import com.alee.utils.ReflectUtils;
import com.alee.utils.XmlUtils;
import com.alee.utils.xml.XStreamContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.converters.reflection.ReflectionConverter;
import com.thoughtworks.xstream.converters.reflection.ReflectionProvider;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.mapper.Mapper;

import javax.swing.*;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

/**
 * Custom XStream converter for {@link com.alee.managers.style.data.SkinInfo} class.
 *
 * @author Mikle Garin
 * @see How to use StyleManager
 * @see com.alee.managers.style.StyleManager
 * @see com.alee.managers.style.data.SkinInfo
 */
public final class SkinInfoConverter extends ReflectionConverter
{
    /**
     * Context variables.
     */
    public static final String SKIN_CLASS = "skin.class";
    public static final String META_DATA_ONLY_KEY = "meta.data.only";

    /**
     * Converter constants.
     */
    public static final String ID_NODE = "id";
    public static final String CLASS_NODE = "class";
    public static final String SUPPORTED_SYSTEMS_NODE = "supportedSystems";
    public static final String ICON_NODE = "icon";
    public static final String TITLE_NODE = "title";
    public static final String DESCRIPTION_NODE = "description";
    public static final String AUTHOR_NODE = "author";
    public static final String EXTENDS_NODE = "extends";
    public static final String INCLUDE_NODE = "include";
    public static final String ICON_SET_NODE = "iconSet";
    public static final String STYLE_NODE = "style";
    public static final String NEAR_CLASS_ATTRIBUTE = "nearClass";

    /**
     * Custom resource map used by StyleEditor to link resources and modified XML files.
     * In other circumstances this map shouldn't be required and will be empty.
     * todo This is a temporary hack for StyleEditor support, it will be removed once #540 is implemented
     */
    protected static final Map> resourceMap = new LinkedHashMap> ();

    /**
     * Constructs SkinInfoConverter with the specified mapper and reflection provider.
     *
     * @param mapper             mapper
     * @param reflectionProvider reflection provider
     */
    public SkinInfoConverter ( @NotNull final Mapper mapper, @NotNull final ReflectionProvider reflectionProvider )
    {
        super ( mapper, reflectionProvider );
    }

    /**
     * Adds custom resource that will be used to change the default resources load strategy.
     * If specified skin XML files will be taken from this map instead of being read from the actual file.
     * This was generally added to support quick styles replacement by the {@link com.alee.extended.style.StyleEditor} application.
     *
     * @param nearClass class near which real XML is located
     * @param src       real XML location
     * @param xml       XML to use instead the real one
     */
    public static void addCustomResource ( @NotNull final String nearClass, @NotNull final String src, @NotNull final String xml )
    {
        Map nearClassMap = resourceMap.get ( nearClass );
        if ( nearClassMap == null )
        {
            nearClassMap = new LinkedHashMap ();
            resourceMap.put ( nearClass, nearClassMap );
        }
        nearClassMap.put ( src, xml );
    }

    @Override
    public boolean canConvert ( @NotNull final Class type )
    {
        return type.equals ( SkinInfo.class );
    }

    @NotNull
    @Override
    public Object unmarshal ( @NotNull final HierarchicalStreamReader reader, @NotNull final UnmarshallingContext context )
    {
        // Skin class provided for include skin or extension skin
        final String superSkinClass = ( String ) context.get ( SKIN_CLASS );

        // Have to perform read in try-catch to properly cleanup skin class
        // Cleanup will only be performed if the skin was read by this specific method call
        try
        {
            final SkinInfo skinInfo = new SkinInfo ();

            // Updating skin class with initial value
            // It might get replaced before we get to actual usage of this value
            // That depends on whether or not skin class node is provided
            skinInfo.setSkinClass ( superSkinClass );

            // Checking unmarshalling mode
            final Object mdo = context.get ( META_DATA_ONLY_KEY );
            final boolean metaDataOnly = mdo != null && ( Boolean ) mdo;

            // Creating component style
            List iconSets = new ArrayList ( 1 );
            final List styles = new ArrayList ( 10 );
            while ( reader.hasMoreChildren () )
            {
                // Read next node
                reader.moveDown ();
                final String nodeName = reader.getNodeName ();
                if ( nodeName.equals ( ID_NODE ) )
                {
                    // Reading skin unique ID
                    skinInfo.setId ( reader.getValue () );
                }
                else if ( nodeName.equals ( CLASS_NODE ) )
                {
                    // Reading skin class canonical name
                    final String skinClass = reader.getValue ();

                    // Validating skin class existence
                    getSkinClass ( skinClass );

                    // Saving skin class into skin information
                    skinInfo.setSkinClass ( skinClass );

                    // Adding skin into context, even if this is a subsequent skin
                    // Since it should be defined in the beginning of XML this value will be passed to underlying converters
                    // This way we can provide it into {@link com.alee.managers.style.data.ComponentStyleConverter}
                    context.put ( SKIN_CLASS, skinClass );
                }
                else if ( nodeName.equals ( SUPPORTED_SYSTEMS_NODE ) )
                {
                    // Reading OS systems supported by this skin
                    skinInfo.setSupportedSystems ( reader.getValue () );
                }
                else if ( nodeName.equals ( ICON_NODE ) )
                {
                    // Reading skin icon which should always be located near skin class
                    final Class skinClass = getSkinClass ( skinInfo.getSkinClass () );
                    skinInfo.setIcon ( new ImageIcon ( skinClass.getResource ( reader.getValue () ) ) );
                }
                else if ( nodeName.equals ( TITLE_NODE ) )
                {
                    // Reading skin title
                    skinInfo.setTitle ( reader.getValue () );
                }
                else if ( nodeName.equals ( DESCRIPTION_NODE ) )
                {
                    // Reading skin description
                    skinInfo.setDescription ( reader.getValue () );
                }
                else if ( nodeName.equals ( AUTHOR_NODE ) )
                {
                    // Reading skin author
                    skinInfo.setAuthor ( reader.getValue () );
                }
                else if ( nodeName.equals ( EXTENDS_NODE ) )
                {
                    // Reading skins supported by the extension
                    List extendedSkins = skinInfo.getExtendedSkins ();
                    if ( extendedSkins == null )
                    {
                        extendedSkins = new ArrayList ( 1 );
                        skinInfo.setExtendedSkins ( extendedSkins );
                    }
                    extendedSkins.add ( reader.getValue () );
                }
                else if ( nodeName.equals ( ICON_SET_NODE ) && !metaDataOnly )
                {
                    // Reading included icon set
                    final String className = reader.getValue ();
                    final Class realClass = mapper.realClass ( className );
                    final IconSet iconSet = readIconSet ( realClass );
                    iconSets.add ( iconSet );
                }
                else if ( nodeName.equals ( STYLE_NODE ) && !metaDataOnly )
                {
                    // Reading separate style
                    final ComponentStyle style = ( ComponentStyle ) context.convertAnother ( styles, ComponentStyle.class );

                    // Simply adding additional style to the end
                    styles.add ( style );
                }
                else if ( nodeName.equals ( INCLUDE_NODE ) && !metaDataOnly )
                {
                    // Reading include information
                    final String nearClass = reader.getAttribute ( NEAR_CLASS_ATTRIBUTE );
                    final String path = reader.getValue ();

                    // Reading included skin
                    final SkinInfo include = readInclude ( skinInfo, nearClass, path );

                    // Merging icon sets to avoid duplicates
                    iconSets = Merge.basicRaw ().nonNullMerge ( iconSets, include.getIconSets () );

                    // Simply adding additional styles to the end
                    styles.addAll ( include.getStyles () );
                }
                reader.moveUp ();
            }

            // Saving all read icon sets
            skinInfo.setIconSets ( iconSets );

            // Saving all read styles into the skin
            // At this point there might be more than one style with the same ID
            skinInfo.setStyles ( styles );

            // Returning complete skin information
            return skinInfo;
        }
        finally
        {
            // Restoring previous skin class value
            // This will also reset the skin class back to {@code null} if this is main skin
            context.put ( SKIN_CLASS, superSkinClass );
        }
    }

    /**
     * Returns {@link Skin} class for the specified canonical name.
     *
     * @param skinClass {@link Skin} class canonical name
     * @return {@link Skin} class for the specified canonical name
     */
    @NotNull
    private Class getSkinClass ( @NotNull final String skinClass )
    {
        try
        {
            return ReflectUtils.getClass ( skinClass );
        }
        catch ( final ClassNotFoundException e )
        {
            throw new StyleException ( "Unable to find skin class: " + skinClass, e );
        }
    }

    /**
     * Returns included {@link SkinInfo}.
     *
     * @param parent    parent {@link SkinInfo}
     * @param nearClass canonical name of the class to look for included skin nearby
     * @param path      path to included skin
     * @return included {@link SkinInfo}
     */
    @NotNull
    private SkinInfo readInclude ( @NotNull final SkinInfo parent, @Nullable final String nearClass, @NotNull final String path )
    {
        // Constructing resource
        final ClassResource resource;
        if ( nearClass != null )
        {
            resource = new ClassResource ( mapper.realClass ( nearClass ), path );
        }
        else
        {
            resource = new ClassResource ( parent.getSkinClass (), path );
        }

        // Reading skin part from included file
        return loadSkinInfo ( parent, resource );
    }

    /**
     * Loads SkinInfo from the specified resource file.
     * It will use an XML from a predefined resources map if it exists there.
     *
     * @param parent   parent {@link SkinInfo}
     * @param resource XML resource file
     * @return loaded SkinInfo
     */
    @NotNull
    private SkinInfo loadSkinInfo ( @NotNull final SkinInfo parent, @NotNull final ClassResource resource )
    {
        try
        {
            final SkinInfo skinInfo;
            final XStreamContext context = new XStreamContext ( SKIN_CLASS, parent.getSkinClass () );
            final Map nearClassMap = resourceMap.get ( resource.getClassName () );
            if ( nearClassMap != null )
            {
                final String xml = nearClassMap.get ( resource.getPath () );
                if ( xml != null )
                {
                    skinInfo = XmlUtils.fromXML ( xml, context );
                }
                else
                {
                    skinInfo = XmlUtils.fromXML ( resource, context );
                }
            }
            else
            {
                skinInfo = XmlUtils.fromXML ( resource, context );
            }
            return skinInfo;
        }
        catch ( final Exception e )
        {
            final String msg = "Included skin file '%s' cannot be read";
            throw new StyleException ( String.format ( msg, resource.getPath () ), e );
        }
    }

    /**
     * Returns icon set created using specified icon set class.
     *
     * @param className icon set class
     * @return icon set created using specified icon set class
     */
    @NotNull
    private IconSet readIconSet ( @NotNull final Class className )
    {
        try
        {
            return ReflectUtils.createInstance ( className );
        }
        catch ( final Exception e )
        {
            final String msg = "Unable to load icon set '%s'";
            throw new StyleException ( String.format ( msg, className ), e );
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy