/*
 * #%L
 * Alfresco Repository
 * %%
 * Copyright (C) 2005 - 2016 Alfresco Software Limited
 * %%
 * This file is part of the Alfresco software. 
 * If the software was purchased under a paid Alfresco license, the terms of 
 * the paid license agreement will prevail.  Otherwise, the software is 
 * provided under the following open source license terms:
 * 
 * Alfresco is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 * 
 * Alfresco 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 Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public License
 * along with Alfresco. If not, see .
 * #L%
 */
package org.alfresco.repo.copy;
import java.io.Serializable;
import java.util.Map;
import org.alfresco.repo.copy.traitextender.DefaultCopyBehaviourCallbackExtension;
import org.alfresco.repo.copy.traitextender.DefaultCopyBehaviourCallbackTrait;
import org.alfresco.service.namespace.QName;
import org.alfresco.traitextender.AJExtender;
import org.alfresco.traitextender.Extend;
import org.alfresco.traitextender.ExtendedTrait;
import org.alfresco.traitextender.Extensible;
import org.alfresco.traitextender.Trait;
import org.alfresco.util.Pair;
/**
 * The default behaviour that a type of aspect implements if there is no associated
 * <{@link CopyBehaviourCallback behaviour}.
 * 
 * This implementation is {@link #getInstance() stateless} and therefore thread-safe.
 * 
 * The default behaviour is:
 * 
 *    - Must Copy:         YES*
- Must Cascade:      YES, if cascade is on*
- Properties to Copy:ALL*
* 
 * @author Derek Hulley
 * @since 3.2
 */
public class DefaultCopyBehaviourCallback extends AbstractCopyBehaviourCallback implements Extensible
{
    private static CopyBehaviourCallback instance = new DefaultCopyBehaviourCallback();
    
    private final ExtendedTrait defaultCopyBehaviourCallbackTrait;
    
    public DefaultCopyBehaviourCallback()
    {
        defaultCopyBehaviourCallbackTrait=new ExtendedTrait(createTrait());
    }
    /**
     * @return          Returns a stateless singleton
     */
    public static CopyBehaviourCallback getInstance()
    {
        return instance;
    }
    
    /**
     * Default behaviour: Always copy
     * 
     * @return          Returns true always
     */
    @Extend(traitAPI=DefaultCopyBehaviourCallbackTrait.class,extensionAPI=DefaultCopyBehaviourCallbackExtension.class)
    public boolean getMustCopy(QName classQName, CopyDetails copyDetails)
    {
        return true;
    }
    /**
     * Default behaviour:
     * * AssocCopySourceAction.COPY_REMOVE_EXISTING
     * * AssocCopyTargetAction.USE_COPIED_OTHERWISE_ORIGINAL_TARGET
     */
    @Override
    public Pair getAssociationCopyAction(
            QName classQName,
            CopyDetails copyDetails,
            CopyAssociationDetails assocCopyDetails)
    {
        return new Pair(
                AssocCopySourceAction.COPY_REMOVE_EXISTING,
                AssocCopyTargetAction.USE_COPIED_OTHERWISE_ORIGINAL_TARGET);
    }
    /**
     * Default behaviour: Cascade if we are copying children AND the association is primary
     * 
     * @return          Returns true if the association is primary and copyChildren == true
     */
    public ChildAssocCopyAction getChildAssociationCopyAction(
            QName classQName,
            CopyDetails copyDetails,
            CopyChildAssociationDetails childAssocCopyDetails)
    {
        if (!childAssocCopyDetails.isCopyChildren())
        {
            return ChildAssocCopyAction.IGNORE;
        }
        if (childAssocCopyDetails.getChildAssocRef().isPrimary())
        {
            return ChildAssocCopyAction.COPY_CHILD;
        }
        else
        {
            return ChildAssocCopyAction.COPY_ASSOC;
        }
    }
    /**
     * Default behaviour: Copy all associated properties
     * 
     * @return          Returns all the properties passes in
     */
    public Map getCopyProperties(
            QName classQName,
            CopyDetails copyDetails,
            Map properties)
    {
        return properties;
    }
    private DefaultCopyBehaviourCallbackTrait createTrait()
    {
        return new DefaultCopyBehaviourCallbackTrait()
        {
            @Override
            public boolean getMustCopy(final QName classQName, final CopyDetails copyDetails)
            {
                return AJExtender.run(new AJExtender.ExtensionBypass()
                {
                    @Override
                    public Boolean run()
                    {
                        return getInstance().getMustCopy(classQName, copyDetails);
                    };
                });
                
            }
        };
    }
    @Override
    public  ExtendedTrait getTrait(Class extends T> traitAPI)
    {
        return (ExtendedTrait) defaultCopyBehaviourCallbackTrait;
    }
}