/*
 * Copyright (C) 2005-2010 Alfresco Software Limited.
 *
 * This file is part of Alfresco
 *
 * 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 .
 */
package org.alfresco.repo.node;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.alfresco.model.ContentModel;
import org.alfresco.service.cmr.repository.AssociationRef;
import org.alfresco.service.cmr.repository.ChildAssociationRef;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.util.collections.CollectionUtils;
import org.alfresco.util.collections.Filter;
import org.alfresco.util.collections.Function;
/**
 * @author Nick Smith
 * @since 4.0
 *
 */
public abstract class NodeUtils
{
    public static Function toNodeRef()
    {
        return new Function()
        {
            public NodeRef apply(String value)
            {
                return new NodeRef(value);
            }
        };
    }
    public static List toNodeRefs(Collection nodeIds)
    {
        return CollectionUtils.transform(nodeIds, toNodeRef());
    }
    public static Function toChildRef()
    {
        return new Function()
        {
            public NodeRef apply(ChildAssociationRef value)
            {
                return value.getChildRef();
            }
        };
    }
    
    public static List toChildRefs(Collection assocRefs)
    {
        return CollectionUtils.transform(assocRefs, toChildRef());
    }
    public static Function toParentRef()
    {
        return new Function()
        {
            public NodeRef apply(ChildAssociationRef value)
            {
                return value.getParentRef();
            }
        };
    }
    
    public static List toParentRefs(Collection assocRefs)
    {
        return CollectionUtils.transform(assocRefs, toParentRef());
    }
    public static Function toNodeRefQueitly()
    {
        return new Function()
        {
            public NodeRef apply(String value)
            {
                if(value!=null && NodeRef.isNodeRef(value))
                {
                    return new NodeRef(value);
                }
                return null;
            }
        };
    }
    
    public static Filter exists(final NodeService nodeService)
    {
        return new Filter()
        {
            public Boolean apply(NodeRef value)
            {
                return nodeService.exists(value);
            }
        };
    }
    
    public static boolean exists(NodeRef node, NodeService nodeService)
    {
        return node != null && nodeService.exists(node);
    }
    
    public static NodeRef getSingleChildAssocNode(Collection assocs, boolean getChild)
    {
        if(assocs != null && assocs.size()==1 )
        {
            ChildAssociationRef association = assocs.iterator().next();
            return getChild ? association.getChildRef() : association.getParentRef();
        }
        return null;
    }
    public static NodeRef getSingleAssocNode(Collection assocs, boolean getTarget)
    {
        if(assocs != null && assocs.size()==1 )
        {
            AssociationRef association = assocs.iterator().next();
            return getTarget ? association.getTargetRef() : association.getSourceRef();
        }
        return null;
    }
    
    public List sortByCreationDate(NodeService nodeService,Collection nodes)
    {
        ArrayList sorted = new ArrayList(nodes);
        Collections.sort(sorted, getNodeCreationDateComparator(nodeService));
        return sorted;
    }
    public Comparator getNodeCreationDateComparator(final NodeService nodeService)
    {
        return new Comparator()
        {
            private Map dates = new HashMap();
            
            public int compare(NodeRef o1, NodeRef o2)
            {
                long date1 = getDate(o1);
                long date2 = getDate(o1);
                return (int)(date1 - date2);
            }
            private long getDate(NodeRef node)
            {
                Long date = dates.get(node);
                if(date == null)
                {
                    Date dateObj = (Date) nodeService.getProperty(node, ContentModel.PROP_CREATED);
                    date = dateObj == null ? -1 : dateObj.getTime();
                    dates.put(node, date);
                }
                return date;
            }
        }; 
    }
}