mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
Fallout: 1. Policy handling for 'onCopy' has been changed to 'getCopyCallback' 2. All existing policy usage was refactored to control behaviour more closely 3. The default child association behaviour has changed: 3.1 Types and aspects control their own child association behaviour 3.2 cm:folder recurses into primary children, but merely copies the secondary association 3.3 cm:rule recurses into primary children 3.4 unless behaviour is defined for a child association, there is no recursion or copying 4. Node association behavior has changed 4.1 There is no copying of node associations. Each type and aspect must handle this by recording nodes and fixing up the required associations in the onCopyComplete. 4.2 If there is a requirement, this can be added to the callback later See 'org.alfresco.repo.copy.AbstractCopyBehaviourCallback' and derived classes for examples. Areas to test with particular attention: 1. Normal copy behaviour 2. Copy of documents with discussions 3. Check-in check-out 4. Check-in, check-out of documents where a discussion was added to working copy 5. Copies of documents with thumbnails 6. Copies of documents with rules 7. Copying of hierarchies that contain rules to copy to another location within the hierarchy 8. Copying into folders where named children already exist git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@13915 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
205 lines
8.0 KiB
Java
205 lines
8.0 KiB
Java
/*
|
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
|
*
|
|
* This program 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 2
|
|
* of the License, or (at your option) any later version.
|
|
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
* As a special exception to the terms and conditions of version 2.0 of
|
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
|
* FLOSS exception. You should have recieved a copy of the text describing
|
|
* the FLOSS exception, and it is also available here:
|
|
* http://www.alfresco.com/legal/licensing"
|
|
*/
|
|
package org.alfresco.repo.model.ml;
|
|
|
|
import java.io.Serializable;
|
|
import java.util.Locale;
|
|
import java.util.Map;
|
|
|
|
import org.alfresco.model.ContentModel;
|
|
import org.alfresco.repo.copy.CopyBehaviourCallback;
|
|
import org.alfresco.repo.copy.CopyDetails;
|
|
import org.alfresco.repo.copy.CopyServicePolicies;
|
|
import org.alfresco.repo.copy.DoNothingCopyBehaviourCallback;
|
|
import org.alfresco.repo.node.NodeServicePolicies;
|
|
import org.alfresco.repo.policy.JavaBehaviour;
|
|
import org.alfresco.repo.policy.PolicyComponent;
|
|
import org.alfresco.service.cmr.ml.MultilingualContentService;
|
|
import org.alfresco.service.cmr.repository.NodeRef;
|
|
import org.alfresco.service.cmr.repository.NodeService;
|
|
import org.alfresco.service.cmr.repository.datatype.DefaultTypeConverter;
|
|
import org.alfresco.service.namespace.NamespaceService;
|
|
import org.alfresco.service.namespace.QName;
|
|
import org.alfresco.service.namespace.RegexQNamePattern;
|
|
|
|
|
|
/**
|
|
* Class containing behaviour for the multilingual document aspect.
|
|
*
|
|
* {@link ContentModel#ASPECT_MULTILINGUAL_DOCUMENT ml document aspect}
|
|
*
|
|
* @author yanipig
|
|
*/
|
|
public class MultilingualDocumentAspect implements
|
|
CopyServicePolicies.OnCopyNodePolicy,
|
|
NodeServicePolicies.BeforeDeleteNodePolicy,
|
|
NodeServicePolicies.OnUpdatePropertiesPolicy
|
|
{
|
|
|
|
// Dependencies
|
|
private PolicyComponent policyComponent;
|
|
private MultilingualContentService multilingualContentService;
|
|
private NodeService nodeService;
|
|
|
|
|
|
/**
|
|
* Initialise the Multilingual Aspect
|
|
*
|
|
* Ensures that the {@link ContentModel#ASPECT_MULTILINGUAL_DOCUMENT ml document aspect}
|
|
*/
|
|
public void init()
|
|
{
|
|
this.policyComponent.bindClassBehaviour(
|
|
QName.createQName(NamespaceService.ALFRESCO_URI, "getCopyCallback"),
|
|
ContentModel.ASPECT_MULTILINGUAL_DOCUMENT,
|
|
new JavaBehaviour(this, "getCopyCallback"));
|
|
|
|
this.policyComponent.bindClassBehaviour(
|
|
QName.createQName(NamespaceService.ALFRESCO_URI, "beforeDeleteNode"),
|
|
ContentModel.ASPECT_MULTILINGUAL_DOCUMENT,
|
|
new JavaBehaviour(this, "beforeDeleteNode"));
|
|
|
|
this.policyComponent.bindClassBehaviour(
|
|
QName.createQName(NamespaceService.ALFRESCO_URI, "onUpdateProperties"),
|
|
ContentModel.ASPECT_MULTILINGUAL_DOCUMENT,
|
|
new JavaBehaviour(this, "onUpdateProperties"));
|
|
|
|
}
|
|
|
|
/**
|
|
* @param policyComponent the policy component to register behaviour with
|
|
*/
|
|
public void setPolicyComponent(PolicyComponent policyComponent)
|
|
{
|
|
this.policyComponent = policyComponent;
|
|
}
|
|
|
|
/**
|
|
* @param multilingualContentService the Multilingual Content Service to set
|
|
*/
|
|
public void setMultilingualContentService(
|
|
MultilingualContentService multilingualContentService)
|
|
{
|
|
this.multilingualContentService = multilingualContentService;
|
|
}
|
|
|
|
/**
|
|
* @param nodeService the Node Service to set
|
|
*/
|
|
public void setNodeService(NodeService nodeService)
|
|
{
|
|
this.nodeService = nodeService;
|
|
}
|
|
|
|
/**
|
|
* The copy of a <b>cm:mlDocument</b> can't keep the Multilingual aspect.
|
|
*
|
|
* @return Returns the {@link DoNothingCopyBehaviourCallback}
|
|
*/
|
|
public CopyBehaviourCallback getCopyCallback(QName classRef, CopyDetails copyDetails)
|
|
{
|
|
return DoNothingCopyBehaviourCallback.getInstance();
|
|
}
|
|
|
|
/**
|
|
* If this is not an empty translation, then ensure that the node is properly
|
|
* unhooked from the translation mechanism first.
|
|
*/
|
|
public void beforeDeleteNode(NodeRef nodeRef)
|
|
{
|
|
if (nodeService.hasAspect(nodeRef, ContentModel.ASPECT_MULTILINGUAL_EMPTY_TRANSLATION))
|
|
{
|
|
// We just let it get deleted
|
|
}
|
|
else
|
|
{
|
|
// First unhook it
|
|
multilingualContentService.unmakeTranslation(nodeRef);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Ensure that the locale is unique inside the <b>mlContainer</b>.
|
|
*
|
|
* If the locale of a pivot translation is modified, the pivot locale reference of the mlContainer
|
|
* must be modified too.
|
|
*/
|
|
public void onUpdateProperties(NodeRef nodeRef, Map<QName, Serializable> before, Map<QName, Serializable> after)
|
|
{
|
|
/*
|
|
* TODO: Move this into MultilingualContentService#setTranslationLocale
|
|
*/
|
|
Locale localeBefore = (Locale)before.get(ContentModel.PROP_LOCALE);
|
|
|
|
Locale localeAfter = null;
|
|
Serializable objLocaleAfter = after.get(ContentModel.PROP_LOCALE);
|
|
if (objLocaleAfter != null)
|
|
{
|
|
localeAfter = DefaultTypeConverter.INSTANCE.convert(Locale.class, objLocaleAfter);
|
|
}
|
|
|
|
// if the local has been modified
|
|
if (localeBefore == null || !localeBefore.equals(localeAfter))
|
|
{
|
|
NodeRef mlContainer = multilingualContentService.getTranslationContainer(nodeRef);
|
|
|
|
// Since the map returned by the getTranslations doesn't duplicate keys, the size of this map will be
|
|
// different of the size of the number of children of the mlContainer if a duplicate locale is found.
|
|
int transSize = multilingualContentService.getTranslations(mlContainer).size();
|
|
int childSize = nodeService.getChildAssocs(mlContainer, ContentModel.ASSOC_MULTILINGUAL_CHILD, RegexQNamePattern.MATCH_ALL).size();
|
|
|
|
// if duplicate locale found
|
|
if(transSize != childSize)
|
|
{
|
|
// throw an exception and the current transaction will be rolled back. The properties will not be
|
|
// longer in an illegal state.
|
|
throw new IllegalArgumentException("The locale " + localeAfter +
|
|
" can't be changed for the node " + nodeRef +
|
|
" because this locale is already in use in an other translation of the same " +
|
|
ContentModel.TYPE_MULTILINGUAL_CONTAINER + ".");
|
|
}
|
|
|
|
// get the locale of ML Container
|
|
Locale localMlContainer = (Locale) nodeService.getProperty(
|
|
mlContainer,
|
|
ContentModel.PROP_LOCALE);
|
|
|
|
// if locale of the container is equals to the locale of
|
|
// the node (before update). The nodeRef is the pivot language
|
|
// and the locale of the mlContainer must be modified
|
|
if(localeBefore != null && localeBefore.equals(localMlContainer))
|
|
{
|
|
nodeService.setProperty(
|
|
mlContainer,
|
|
ContentModel.PROP_LOCALE,
|
|
localeAfter);
|
|
}
|
|
|
|
}
|
|
|
|
// else no action to perform
|
|
}
|
|
}
|