001/*
002 *  Copyright 2017 Anyware Services
003 *
004 *  Licensed under the Apache License, Version 2.0 (the "License");
005 *  you may not use this file except in compliance with the License.
006 *  You may obtain a copy of the License at
007 *
008 *      http://www.apache.org/licenses/LICENSE-2.0
009 *
010 *  Unless required by applicable law or agreed to in writing, software
011 *  distributed under the License is distributed on an "AS IS" BASIS,
012 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 *  See the License for the specific language governing permissions and
014 *  limitations under the License.
015 */
016package org.ametys.plugins.frontedition;
017
018import java.util.ArrayList;
019import java.util.Arrays;
020import java.util.Collection;
021import java.util.Collections;
022import java.util.Comparator;
023import java.util.HashMap;
024import java.util.LinkedHashMap;
025import java.util.List;
026import java.util.Map;
027import java.util.Set;
028import java.util.stream.Collectors;
029
030import javax.jcr.RepositoryException;
031
032import org.apache.avalon.framework.context.Context;
033import org.apache.avalon.framework.context.ContextException;
034import org.apache.avalon.framework.context.Contextualizable;
035import org.apache.avalon.framework.logger.LogEnabled;
036import org.apache.avalon.framework.logger.Logger;
037import org.apache.avalon.framework.service.ServiceException;
038import org.apache.avalon.framework.service.ServiceManager;
039import org.apache.avalon.framework.service.Serviceable;
040import org.apache.cocoon.components.ContextHelper;
041import org.apache.cocoon.environment.Request;
042import org.apache.commons.lang3.ArrayUtils;
043import org.apache.commons.lang3.StringUtils;
044import org.apache.commons.lang3.tuple.Pair;
045import org.w3c.dom.NodeList;
046
047import org.ametys.cms.CmsConstants;
048import org.ametys.cms.clientsideelement.styles.HTMLEditorStyle;
049import org.ametys.cms.content.RootContentHelper;
050import org.ametys.cms.contenttype.AttributeDefinition;
051import org.ametys.cms.contenttype.ContentType;
052import org.ametys.cms.contenttype.ContentTypeExtensionPoint;
053import org.ametys.cms.contenttype.ContentTypesHelper;
054import org.ametys.cms.repository.Content;
055import org.ametys.cms.repository.WorkflowAwareContent;
056import org.ametys.cms.workflow.ContentWorkflowHelper;
057import org.ametys.core.right.RightManager;
058import org.ametys.core.right.RightManager.RightResult;
059import org.ametys.core.ui.ClientSideElement;
060import org.ametys.core.ui.ClientSideElement.Script;
061import org.ametys.core.ui.ClientSideElement.ScriptFile;
062import org.ametys.core.ui.widgets.richtext.RichTextConfiguration;
063import org.ametys.core.ui.widgets.richtext.RichTextConfigurationExtensionPoint;
064import org.ametys.core.util.I18nizableSerializer;
065import org.ametys.core.util.JSONUtils;
066import org.ametys.plugins.core.ui.minimize.HashCache;
067import org.ametys.plugins.explorer.resources.actions.ExplorerResourcesDAO;
068import org.ametys.plugins.repository.AmetysObject;
069import org.ametys.plugins.repository.AmetysObjectIterable;
070import org.ametys.plugins.repository.AmetysObjectResolver;
071import org.ametys.plugins.repository.AmetysRepositoryException;
072import org.ametys.plugins.repository.RepositoryConstants;
073import org.ametys.plugins.repository.UnknownAmetysObjectException;
074import org.ametys.plugins.repository.jcr.DefaultTraversableAmetysObject;
075import org.ametys.plugins.repository.provider.RequestAttributeWorkspaceSelector;
076import org.ametys.plugins.repository.version.VersionableAmetysObject;
077import org.ametys.plugins.workflow.support.WorkflowHelper;
078import org.ametys.plugins.workflow.support.WorkflowProvider;
079import org.ametys.plugins.workflow.support.WorkflowProvider.AmetysObjectWorkflow;
080import org.ametys.runtime.i18n.I18nizableText;
081import org.ametys.runtime.model.ModelItem;
082import org.ametys.runtime.model.ModelItemGroup;
083import org.ametys.runtime.plugin.PluginsManager;
084import org.ametys.web.cache.PageHelper;
085import org.ametys.web.renderingcontext.RenderingContext;
086import org.ametys.web.repository.page.ModifiablePage;
087import org.ametys.web.repository.page.Page;
088import org.ametys.web.repository.page.SitemapElement;
089import org.ametys.web.repository.page.Zone;
090import org.ametys.web.repository.page.ZoneItem;
091import org.ametys.web.repository.site.Site;
092import org.ametys.web.repository.site.SiteManager;
093import org.ametys.web.repository.sitemap.Sitemap;
094import org.ametys.web.transformation.xslt.AmetysXSLTHelper;
095
096import com.opensymphony.workflow.loader.ActionDescriptor;
097import com.opensymphony.workflow.loader.StepDescriptor;
098import com.opensymphony.workflow.loader.WorkflowDescriptor;
099import com.opensymphony.workflow.spi.Step;
100
101/**
102 * Helper for preparing the Ametys Edition
103 */
104public class AmetysFrontEditionHelper implements Serviceable, Contextualizable, LogEnabled
105{
106    /** The right's id from front-edition access */
107    public static final String FRONT_EDITION_RIGHT_ID = "Front_Edition_Access_Right";
108    
109    private static AmetysObjectResolver _ametysObjectResolver;
110    private static HashCache _hashCache;
111    private static RichTextConfigurationExtensionPoint _richTextConfigurationExtensionPoint;
112    private static HTMLEditorStyle _htmlEditorStyle;
113    private static RightManager _rightManager;
114    private static ContentWorkflowHelper _contentWorkflowHelper;
115    private static PageHelper _pageHelper;
116    private static WorkflowHelper _workflowHelper;
117    private static JSONUtils _jsonUtils;
118    private static SiteManager _siteManager;
119    private static WorkflowProvider _workflowProvider;
120    private static ContentTypesHelper _contentTypesHelper;
121    private static ContentTypeExtensionPoint _contentTypesEP;
122    private static RootContentHelper _rootContentHelper;
123    
124    private static Context _context;
125
126    private static Logger _logger;
127
128    public void service(ServiceManager manager) throws ServiceException
129    {
130        _ametysObjectResolver = (AmetysObjectResolver) manager.lookup(AmetysObjectResolver.ROLE);
131        _siteManager = (SiteManager) manager.lookup(SiteManager.ROLE);
132        _hashCache = (HashCache) manager.lookup(HashCache.ROLE);
133        _richTextConfigurationExtensionPoint = (RichTextConfigurationExtensionPoint) manager.lookup(RichTextConfigurationExtensionPoint.ROLE);
134        _htmlEditorStyle = (HTMLEditorStyle) manager.lookup(HTMLEditorStyle.ROLE);
135        _rightManager = (RightManager) manager.lookup(RightManager.ROLE);
136        _contentWorkflowHelper = (ContentWorkflowHelper) manager.lookup(ContentWorkflowHelper.ROLE);
137        _contentTypesHelper = (ContentTypesHelper) manager.lookup(ContentTypesHelper.ROLE);
138        _contentTypesEP = (ContentTypeExtensionPoint) manager.lookup(ContentTypeExtensionPoint.ROLE);
139        _pageHelper = (PageHelper) manager.lookup(PageHelper.ROLE);
140        _workflowHelper = (WorkflowHelper) manager.lookup(WorkflowHelper.ROLE);
141        _jsonUtils = (JSONUtils) manager.lookup(JSONUtils.ROLE);
142        _workflowProvider = (WorkflowProvider) manager.lookup(WorkflowProvider.ROLE);
143        _rootContentHelper = (RootContentHelper) manager.lookup(RootContentHelper.ROLE);
144    }
145    
146    @Override
147    public void contextualize(Context context) throws ContextException
148    {
149        _context = context;
150    }
151    
152    @Override
153    public void enableLogging(Logger logger)
154    {
155        _logger = logger;
156    }
157    
158    /**
159     * Prepare a hashcode for js files
160     * @param locale The language code to use
161     * @param theme The url for the theme
162     * @return The new hashcode to read the minimified concatened file
163     */
164    public static String prepareJSFiles(String locale, String theme)
165    {
166        String themeName = StringUtils.substringAfterLast(theme, "/");
167        
168        Map<String, String> filesInfos = new HashMap<>();
169        filesInfos.put("media", "");
170        filesInfos.put("tag", "script");
171        
172        Map<String, Map<String, String>> files = new LinkedHashMap<>();
173        files.put("/plugins/extjs7/resources/ext-all.js", filesInfos);
174        files.put("/plugins/extjs7/resources/classic/locale/locale-" + locale + ".js", filesInfos);
175        files.put(theme + "/" + themeName + ".js", filesInfos);
176
177        files.put("/plugins/core-ui/resources/js/Ext.fixes." + locale + ".js", filesInfos);
178        files.put("/plugins/core-ui/resources/js/Ext.enhancements." + locale + ".js", filesInfos);
179        
180        files.put("/plugins/core-ui/resources/js/Ametys." + locale + ".js", filesInfos);
181        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys." + locale + ".js", filesInfos);
182        files.put("/plugins/core-ui/resources/js/Ametys/mask/GlobalLoadMask." + locale + ".js", filesInfos);
183        
184        files.put("/plugins/core-ui/resources/js/Ametys/log/Logger." + locale + ".js", filesInfos);
185        files.put("/plugins/core-ui/resources/js/Ametys/log/Logger/Entry." + locale + ".js", filesInfos);
186        files.put("/plugins/core-ui/resources/js/Ametys/log/LoggerFactory." + locale + ".js", filesInfos);
187        files.put("/plugins/core-ui/resources/js/Ametys/log/ErrorDialog." + locale + ".js", filesInfos);
188        
189        files.put("/plugins/core-ui/resources/js/Ametys/window/DialogBox." + locale + ".js", filesInfos);
190        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/window/DialogBox." + locale + ".js", filesInfos);
191        files.put("/plugins/core-ui/resources/js/Ametys/window/MessageBox." + locale + ".js", filesInfos);
192        
193        files.put("/plugins/core-ui/resources/js/Ametys/form/AbstractField." + locale + ".js", filesInfos);
194        files.put("/plugins/core-ui/resources/js/Ametys/form/AbstractFieldsWrapper." + locale + ".js", filesInfos);
195        files.put("/plugins/core-ui/resources/js/Ametys/form/AbstractQueryableComboBox." + locale + ".js", filesInfos);
196        files.put("/plugins/core-ui/resources/js/Ametys/form/AbstractQueryableComboBox/SplitterTracker." + locale + ".js", filesInfos);
197        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/form/widget/AbstractQueryableComboBox." + locale + ".js", filesInfos);
198        files.put("/plugins/core-ui/resources/js/Ametys/form/field/DateTime." + locale + ".js", filesInfos);
199        files.put("/plugins/core-ui/resources/js/Ametys/form/field/StringTime." + locale + ".js", filesInfos);
200        files.put("/plugins/core-ui/resources/js/Ametys/form/field/Password." + locale + ".js", filesInfos);
201        files.put("/plugins/core-ui/resources/js/Ametys/form/field/ChangePassword." + locale + ".js", filesInfos);
202        files.put("/plugins/core-ui/resources/js/Ametys/form/field/ReferencedNumberField." + locale + ".js", filesInfos);
203        files.put("/plugins/core-ui/resources/js/Ametys/form/field/RichText." + locale + ".js", filesInfos);
204        files.put("/plugins/cms/resources/js/Ametys/cms/form/field/RichText." + locale + ".js", filesInfos);
205        files.put("/plugins/core-ui/resources/js/Ametys/form/field/RichText/SplitterTracker." + locale + ".js", filesInfos);
206        files.put("/plugins/core-ui/resources/js/Ametys/form/field/TextArea." + locale + ".js", filesInfos);
207        files.put("/plugins/core-ui/resources/js/Ametys/form/field/ColorSelector." + locale + ".js", filesInfos);
208        files.put("/plugins/core-ui/resources/js/Ametys/form/field/SelectUserDirectory." + locale + ".js", filesInfos);
209        files.put("/plugins/core-ui/resources/js/Ametys/form/field/SelectGroupDirectories." + locale + ".js", filesInfos);
210        files.put("/plugins/core-ui/resources/js/Ametys/form/field/Code." + locale + ".js", filesInfos);
211        
212        files.put("/plugins/core-ui/resources/js/Ametys/helper/ChooseLocationLeaflet." + locale + ".js", filesInfos);
213        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/GeoCode." + locale + ".js", filesInfos);
214        files.put("/plugins/core-ui/resources/js/Ametys/grid/GridColumnHelper." + locale + ".js", filesInfos);
215        files.put("/plugins/tiny_mce/resources/js/tinymce.js", filesInfos);
216        files.put("/plugins/codemirror/resources/js/codemirror.js", filesInfos);
217        files.put("/plugins/codemirror/resources/js/addon/edit/matchbrackets.js", filesInfos);
218        files.put("/plugins/codemirror/resources/js/addon/selection/active-line.js", filesInfos);
219        files.put("/plugins/codemirror/resources/js/mode/xml/xml.js", filesInfos);
220        files.put("/plugins/codemirror/resources/js/mode/javascript/javascript.js", filesInfos);
221        files.put("/plugins/codemirror/resources/js/mode/css/css.js", filesInfos);
222        files.put("/plugins/codemirror/resources/js/mode/htmlmixed/htmlmixed.js", filesInfos);
223        
224        files.put("/plugins/core-ui/resources/js/Ametys/data/ServerCaller." + locale + ".js", filesInfos);
225        files.put("/plugins/core-ui/resources/js/Ametys/data/ServerComm." + locale + ".js", filesInfos);
226        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/data/ServerComm." + locale + ".js", filesInfos);
227        files.put("/plugins/core-ui/resources/js/Ametys/data/ServerCommProxy." + locale + ".js", filesInfos);
228        files.put("/plugins/core-ui/resources/js/Ametys/data/ServerComm/TimeoutDialog." + locale + ".js", filesInfos);
229
230        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/RichText/RichTextConfigurationBase." + locale + ".js", filesInfos);
231        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/RichText/RichTextConfiguration." + locale + ".js", filesInfos);
232        
233        files.put("/plugins/core-ui/resources/js/Ametys/plugins/coreui/system/devmode/StacktraceHelper." + locale + ".js", filesInfos);
234
235        files.put("/plugins/front-edition/resources/js/front-comm." + locale + ".js", filesInfos);
236        files.put("/plugins/front-edition/resources/js/front-widget." + locale + ".js", filesInfos);
237//        files.put("/plugins/front-edition/resources/js/front-page." + locale + ".js", filesInfos);
238        files.put("/plugins/core-ui/resources/js/Ametys/helper/EnterURL." + locale + ".js", filesInfos);
239        files.put("/plugins/core-ui/resources/js/Ametys/helper/FileUpload." + locale + ".js", filesInfos);
240        files.put("/plugins/web/resources/js/Ametys/web/helper/ChoosePage." + locale + ".js", filesInfos);
241        files.put("/plugins/web/resources/js/Ametys/web/helper/ContextToolbar." + locale + ".js", filesInfos);
242        files.put("/plugins/web/resources/js/Ametys/web/sitemap/SitemapTree." + locale + ".js", filesInfos);
243        files.put("/plugins/web/resources/js/Ametys/web/sitemap/SitemapTree/Page." + locale + ".js", filesInfos);
244        files.put("/plugins/web/resources/js/Ametys/web/sitemap/SitemapTree/Sitemap." + locale + ".js", filesInfos);
245        files.put("/plugins/web/resources/js/Ametys/web/site/SitesTree/Site." + locale + ".js", filesInfos);
246        files.put("/plugins/cms/resources/js/Ametys/cms/editor/LinkHandler." + locale + ".js", filesInfos);
247        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/editor/Links." + locale + ".js", filesInfos);
248        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/editor/Links/LinkHandler." + locale + ".js", filesInfos);
249        files.put("/plugins/web/resources/js/Ametys/plugins/web/editor/Links." + locale + ".js", filesInfos);
250        files.put("/plugins/web/resources/js/Ametys/plugins/web/editor/PageLinkHandler." + locale + ".js", filesInfos);
251        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/editor/Images." + locale + ".js", filesInfos);
252        
253        files.put("/plugins/cms/resources/js/tinymce/basicstyles." + locale + ".js", filesInfos);
254        files.put("/plugins/cms/resources/js/tinymce/lists." + locale + ".js", filesInfos);
255        files.put("/plugins/cms/resources/js/tinymce/links." + locale + ".js", filesInfos);
256        files.put("/plugins/web/resources/js/tinymce/links." + locale + ".js", filesInfos);
257        files.put("/plugins/front-edition/resources/js/tinymce/links." + locale + ".js", filesInfos);
258        files.put("/plugins/cms/resources/js/tinymce/styles." + locale + ".js", filesInfos);
259        files.put("/plugins/cms/resources/js/tinymce/images." + locale + ".js", filesInfos);
260        files.put("/plugins/front-edition/resources/js/tinymce/images." + locale + ".js", filesInfos);
261        
262        if (PluginsManager.getInstance().getPluginNames().contains("inlinemedia"))
263        {
264            
265            files.put("/plugins/inlinemedia/resources/js/Ametys/plugins/inlinemedia/Media." + locale + ".js", filesInfos);
266            files.put("/plugins/inlinemedia/resources/js/Ametys/plugins/inlinemedia/InsertMediaHelper." + locale + ".js", filesInfos);
267            files.put("/plugins/inlinemedia/resources/js/tinymce/images." + locale + ".js", filesInfos);
268        }
269        
270        files.put("/plugins/front-edition/resources/js/tinymce/save." + locale + ".js", filesInfos);
271        
272        //TinyMCE Tables
273        files.put("/plugins/cms/resources/js/tinymce/tables." + locale + ".js", filesInfos);
274        files.put("/plugins/front-edition/resources/js/tinymce/tables." + locale + ".js", filesInfos);
275        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/editor/Tables." + locale + ".js", filesInfos);
276        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/editor/BasicActions." + locale + ".js", filesInfos);
277        //TinyMCE Images
278        files.put("/plugins/cms/resources/js/Ametys/cms/uihelper/ChooseResource." + locale + ".js", filesInfos);
279        files.put("/plugins/explorer/resources/js/Ametys/explorer/tree/ExplorerTree." + locale + ".js", filesInfos);
280        files.put("/plugins/explorer/resources/js/Ametys/explorer/tree/ExplorerTree/NodeEntry." + locale + ".js", filesInfos);
281        files.put("/plugins/explorer/resources/js/Ametys/explorer/Resource." + locale + ".js", filesInfos);
282        files.put("/plugins/explorer/resources/js/Ametys/explorer/ExplorerNodeDAO." + locale + ".js", filesInfos);
283        files.put("/plugins/explorer/resources/js/Ametys/explorer/ExplorerNode." + locale + ".js", filesInfos);
284        files.put("/plugins/web/resources/js/Ametys/plugins/web/explorer/ExplorerNode." + locale + ".js", filesInfos);
285        files.put("/plugins/cms/resources/js/Ametys/cms/attach/AttachmentsExplorerTree." + locale + ".js", filesInfos);
286        files.put("/plugins/cms/resources/js/Ametys/cms/uihelper/ChooseAttachmentFile." + locale + ".js", filesInfos);
287        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/cms/uihelper/ChooseAttachmentFile." + locale + ".js", filesInfos);
288        files.put("/plugins/explorer/resources/js/Ametys/explorer/resources/actions/FileActions." + locale + ".js", filesInfos);
289        files.put("/plugins/explorer/resources/js/Ametys/explorer/resources/helper/ResourceUpload." + locale + ".js", filesInfos);
290        files.put("/plugins/core-ui/resources/js/Ametys/file/AbstractFileExplorerTree/FileNode." + locale + ".js", filesInfos);
291        files.put("/plugins/core-ui/resources/js/Ametys/file/AbstractFileExplorerTree." + locale + ".js", filesInfos);
292
293        //Empty Message class
294        files.put("/plugins/core-ui/resources/js/Ametys/message/Message." + locale + ".js", filesInfos);
295        files.put("/plugins/core-ui/resources/js/Ametys/message/MessageTarget." + locale + ".js", filesInfos);
296        files.put("/plugins/core-ui/resources/js/Ametys/message/MessageTargetFactory." + locale + ".js", filesInfos);
297        files.put("/plugins/core-ui/resources/js/Ametys/message/factory/DefaultMessageTargetFactory." + locale + ".js", filesInfos);
298        files.put("/plugins/front-edition/resources/js/Ametys/message/MessageTargetHelper." + locale + ".js", filesInfos);
299        //add page
300        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard." + locale + ".js", filesInfos);
301        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/plugins/web/page/AddPageWizard." + locale + ".js", filesInfos);
302        files.put("/plugins/web/resources/js/Ametys/web/sitemap/SitemapDAO." + locale + ".js", filesInfos);
303        files.put("/plugins/web/resources/js/Ametys/web/sitemap/Sitemap." + locale + ".js", filesInfos);
304        files.put("/plugins/web/resources/js/Ametys/web/page/PageDAO." + locale + ".js", filesInfos);
305        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/web/page/PageDAO." + locale + ".js", filesInfos);
306        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/Card." + locale + ".js", filesInfos);
307        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/CreatePageCard." + locale + ".js", filesInfos);
308        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/PageContentCard." + locale + ".js", filesInfos);
309        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/ContentPropertiesCard." + locale + ".js", filesInfos);
310        files.put("/plugins/core-ui/resources/js/Ametys/form/Widget." + locale + ".js", filesInfos);
311        files.put("/plugins/core-ui/resources/js/Ametys/form/ConfigurableFormPanel." + locale + ".js", filesInfos);
312        files.put("/plugins/front-edition/resources/js/Ametys/userprefs/UserPrefsDAO." + locale + ".js", filesInfos);
313        files.put("/plugins/core-ui/resources/js/Ametys/form/ConfigurableFormPanel/Repeater." + locale + ".js", filesInfos);
314        files.put("/plugins/core-ui/resources/js/Ametys/form/ConfigurableFormPanel/FieldCheckersManager." + locale + ".js", filesInfos);
315        files.put("/plugins/front-edition/resources/js/Ametys/message/MessageBus." + locale + ".js", filesInfos);
316        files.put("/plugins/core-ui/resources/js/Ametys/form/WidgetManager." + locale + ".js", filesInfos);
317        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/DefaultWidgets." + locale + ".js", filesInfos);
318        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/Date." + locale + ".js", filesInfos);
319        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/DateTime." + locale + ".js", filesInfos);
320        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/form/WidgetManager." + locale + ".js", filesInfos);
321        
322        files.put("/plugins/cms/resources/js/Ametys/cms/content/ContentDAO." + locale + ".js", filesInfos);
323        files.put("/plugins/web/resources/js/Ametys/web/content/ContentDAO." + locale + ".js", filesInfos);
324        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/cms/content/ContentDAO." + locale + ".js", filesInfos);
325        files.put("/plugins/cms/resources/js/Ametys/cms/content/Content." + locale + ".js", filesInfos);
326        files.put("/plugins/core-ui/resources/js/Ametys/navhistory/HistoryDAO." + locale + ".js", filesInfos);
327        files.put("/plugins/core-ui/resources/js/Ametys/window/MessageBox." + locale + ".js", filesInfos);
328        files.put("/plugins/front-edition/resources/js/Ametys/tool/ToolsManager." + locale + ".js", filesInfos);
329
330        files.put("/plugins/cms/resources/js/Ametys/cms/form/widget/RichText." + locale + ".js", filesInfos);
331        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/cms/form/widget/SelectContent/ContentEntry." + locale + ".js", filesInfos);
332        files.put("/plugins/cms/resources/js/Ametys/cms/form/widget/SelectContent." + locale + ".js", filesInfos);
333        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/cms/form/widget/SelectContent." + locale + ".js", filesInfos);
334        files.put("/plugins/cms/resources/js/Ametys/cms/form/widget/SelectReferenceTableContent." + locale + ".js", filesInfos);
335        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/cms/form/widget/SelectReferenceTableContent." + locale + ".js", filesInfos);
336        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/PageTypeCard." + locale + ".js", filesInfos);
337        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/plugins/web/page/AddPageWizard/PageTypeCard." + locale + ".js", filesInfos);
338        files.put("/plugins/web/resources/js/Ametys/web/form/widget/SelectPage." + locale + ".js", filesInfos);
339        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/web/form/widget/SelectPage." + locale + ".js", filesInfos);
340        files.put("/plugins/web/resources/js/Ametys/web/form/widget/SelectSite." + locale + ".js", filesInfos);
341        files.put("/plugins/web/resources/js/Ametys/web/helper/ChooseSite." + locale + ".js", filesInfos);
342        files.put("/plugins/web/resources/js/Ametys/web/site/SitesTree." + locale + ".js", filesInfos);
343        files.put("/plugins/web/resources/js/Ametys/web/site/SitesTree/Site." + locale + ".js", filesInfos);
344        files.put("/plugins/web/resources/js/Ametys/web/helper/ContextToolbar." + locale + ".js", filesInfos);
345        files.put("/plugins/web/resources/js/Ametys/web/site/SitesTree/Site." + locale + ".js", filesInfos);
346        files.put("/plugins/web/resources/js/Ametys/web/form/widget/SelectPage/PageEntry." + locale + ".js", filesInfos);
347        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/TagsCard." + locale + ".js", filesInfos);
348        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/AddPageWizard/TagsCardLite." + locale + ".js", filesInfos);
349        files.put("/plugins/web/resources/js/Ametys/web/page/Page." + locale + ".js", filesInfos);
350        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/tag/TagsTreePanel." + locale + ".js", filesInfos);
351        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/tag/TagsTreePanel/TagNode." + locale + ".js", filesInfos);
352
353        //delete page
354        files.put("/plugins/web/resources/js/Ametys/plugins/web/sitemap/SitemapActions." + locale + ".js", filesInfos);
355
356        //page tags
357        files.put("/plugins/web/resources/js/Ametys/plugins/web/tag/AffectTagAction." + locale + ".js", filesInfos);
358        files.put("/plugins/cms/resources/js/Ametys/cms/uihelper/ChooseTagHelper." + locale + ".js", filesInfos);
359        files.put("/plugins/cms/resources/js/Ametys/cms/uihelper/ChooseTag." + locale + ".js", filesInfos);
360        files.put("/plugins/web/resources/js/Ametys/web/helper/ChooseTag." + locale + ".js", filesInfos);
361        files.put("/plugins/cms/resources/js/Ametys/plugins/cms/tag/TagActions." + locale + ".js", filesInfos);
362        files.put("/plugins/web/resources/js/Ametys/plugins/web/tag/TagsTreePanel." + locale + ".js", filesInfos);
363        
364        // scheduled publication
365        files.put("/plugins/web/resources/js/Ametys/plugins/web/page/SchedulePublication." + locale + ".js", filesInfos);
366        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/plugins/web/page/SchedulePublication." + locale + ".js", filesInfos);
367
368        // ping
369        files.put("/plugins/core-ui/resources/js/Ametys/plugins/coreui/system/StartTimeChecker." + locale + ".js", filesInfos);
370
371        // actions on content
372        files.put("/plugins/cms/resources/js/Ametys/cms/uihelper/EditContent." + locale + ".js", filesInfos);
373        files.put("/plugins/web/resources/js/Ametys/plugins/web/zone/ZoneActions." + locale + ".js", filesInfos);
374        files.put("/plugins/web/resources/js/Ametys/plugins/web/zone/ZoneDAO." + locale + ".js", filesInfos);
375
376        // file widget (+cropping for image filter)
377        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/File." + locale + ".js", filesInfos);
378        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/form/widget/File." + locale + ".js", filesInfos);
379        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/File/FileSource." + locale + ".js", filesInfos);
380        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/File/External." + locale + ".js", filesInfos);
381        files.put("/plugins/cms/resources/js/Ametys/cms/form/widget/File/Resource." + locale + ".js", filesInfos);
382        files.put("/plugins/core-ui/resources/js/Ametys/helper/crop/CropDialog." + locale + ".js", filesInfos);
383        files.put("/plugins/core-ui/resources/js/Jcrop/jquery.Jcrop." + locale + ".js", filesInfos);
384        files.put("/plugins/core-ui/resources/js/Ametys/form/widget/Image." + locale + ".js", filesInfos);
385        files.put("/plugins/front-edition/resources/js/Ametys/override/Ametys/form/widget/Image." + locale + ".js", filesInfos);
386        
387        // save errors and warnings
388        files.put("/plugins/core-ui/resources/js/Ametys/form/SaveHelper." + locale + ".js", filesInfos);
389        files.put("/plugins/core-ui/resources/js/Ametys/form/SaveHelper/SaveErrorDialog." + locale + ".js", filesInfos);
390        files.put("/plugins/cms/resources/js/Ametys/cms/content/ContentErrorHelper." + locale + ".js", filesInfos);
391        
392        // Let's add all richtextconfiguration files
393        _addRichTextConfigurationFiles(files, filesInfos, locale, true);
394
395        return _hashCache.createHash(files, locale);
396    }
397
398    /**
399     * Prepare a hashcode for css files
400     * @param locale The language code to use
401     * @param theme The url for the theme
402     * @return The new hashcode to read the minimified concatened file
403     */
404    public static String prepareCSSFiles(String locale, String theme)
405    {
406        Map<String, String> filesInfos = new HashMap<>();
407        filesInfos.put("media", "");
408        filesInfos.put("tag", "link");
409        
410        Map<String, Map<String, String>> files = new LinkedHashMap<>();
411        String themeName = StringUtils.substringAfterLast(theme, "/");
412        files.put(theme + "/" + themeName + "-all.css", filesInfos);
413        files.put("/plugins/front-edition/resources/css/theme-ametys-base-light.css", filesInfos);
414        files.put("/plugins/core-ui/resources/font/ametys/AmetysIcon.css", filesInfos);
415        files.put("/plugins/core-ui/resources/css/Ametys/form/image.css", filesInfos);
416        files.put("/plugins/core-ui/resources/css/Jcrop/jquery.Jcrop.css", filesInfos);
417        files.put("/plugins/web/resources/css/pages/addpage.css", filesInfos);
418        files.put("/plugins/cms/resources/css/selectcontent.css", filesInfos);
419        files.put("/plugins/codemirror/resources/css/codemirror.css", filesInfos);
420        files.put("/plugins/codemirror/resources/js/addon/hint/show-hint.css", filesInfos);
421        
422        // Let's add all richtextconfiguration files
423        _addRichTextConfigurationFiles(files, filesInfos, locale, false);
424        
425        return _hashCache.createHash(files, locale);
426    }
427
428    private static List<ScriptFile> _getRichTextConfigurationFiles(boolean scripts, Map<String, Object> contextParameters)
429    {
430        List<ScriptFile> files = new ArrayList<>();
431        
432        for (String richTextConfigurationId : _richTextConfigurationExtensionPoint.getExtensionsIds())
433        {
434            RichTextConfiguration richTextConfiguration = _richTextConfigurationExtensionPoint.getExtension(richTextConfigurationId);
435            for (String category : richTextConfiguration.getCategories())
436            {
437                Set<ClientSideElement> convertors = richTextConfiguration.getConvertors(category, contextParameters);
438                _addFilesForClientSideElements(scripts, files, convertors);
439
440                Set<ClientSideElement> validators = richTextConfiguration.getValidators(category, contextParameters);
441                _addFilesForClientSideElements(scripts, files, validators);
442                
443                if (!scripts)
444                {
445                    List<ScriptFile> backOfficeCSSFiles = _htmlEditorStyle.getBackOfficeCSSFiles(category, contextParameters);
446                    if (backOfficeCSSFiles != null)
447                    {
448                        files.addAll(backOfficeCSSFiles);
449                    }
450                }
451            }
452        }
453
454        return files;
455    }
456
457    private static void _addFilesForClientSideElements(boolean scripts, List<ScriptFile> files, Set<ClientSideElement> clientSideElmts)
458    {
459        if (clientSideElmts != null)
460        {
461            for (ClientSideElement convertor : clientSideElmts)
462            {
463                List<Script> scriptsToAdd = convertor.getScripts(Collections.EMPTY_MAP);
464                if (scriptsToAdd != null)
465                {
466                    for (Script script : scriptsToAdd)
467                    {
468                        List<ScriptFile> scriptFiles = scripts ? script.getScriptFiles() : script.getCSSFiles();
469                        if (scriptFiles != null)
470                        {
471                            files.addAll(scriptFiles);
472                        }
473                    }
474                }
475            }
476        }
477    }
478
479    private static void _addRichTextConfigurationFiles(Map<String, Map<String, String>> files, Map<String, String> filesInfos, String locale, boolean scripts)
480    {
481        Map<String, Object> contextParameters = new HashMap<>();
482        contextParameters.put("siteName", AmetysXSLTHelper.site());
483        
484        List<ScriptFile> richTextConfigurationFiles = _getRichTextConfigurationFiles(scripts, contextParameters);
485        for (ScriptFile file : richTextConfigurationFiles)
486        {
487            String uri = null;
488            if (file.isLangSpecific())
489            {
490                if (file.getLangPaths().containsKey(locale))
491                {
492                    uri = file.getLangPaths().get(locale);
493                }
494                else if (file.getLangPaths().containsKey(file.getDefaultLang()))
495                {
496                    uri = file.getLangPaths().get(file.getDefaultLang());
497                }
498            }
499            else
500            {
501                uri = file.getPath();
502            }
503            
504            if (uri != null)
505            {
506                files.put(uri, filesInfos);
507            }
508        }
509    }
510    /**
511     * Check if we can display the front edition button for a specific right
512     * Checks :
513     * * Rendering context == front
514     * * Edition mode
515     * * Front_Edition_Access_Right on current page
516     * * RightId available  on object
517     * @param rightId right to check (can be null)
518     * @param objectId id of page/content to check. Can be null or empty to get the current page
519     * @return true if all is OK, false if at least one is false
520     * @deprecated Use the version with three arguments
521     */
522    @Deprecated
523    public static boolean hasFrontEditionRight(String rightId, String objectId)
524    {
525        return hasFrontEditionRight(rightId, objectId, true);
526    }
527    /**
528     * Check if we can display the front edition button for a specific right
529     * Checks :
530     * * Rendering context == front
531     * * Edition mode
532     * * Front_Edition_Access_Right on current page
533     * * RightId available on object
534     * @param rightId right to check (can be null)
535     * @param objectId id of page/content to check. Can be null or empty to get the current page
536     * @param editionModeOnly Check if the user is in edition mode
537     * @return true if all is OK, false if at least one is false
538     */
539    public static boolean hasFrontEditionRight(String rightId, String objectId, boolean editionModeOnly)
540    {
541        String aoId = objectId;
542        if (StringUtils.isEmpty(aoId))
543        {
544            aoId = AmetysXSLTHelper.pageId();
545        }
546        
547        if (StringUtils.isEmpty(aoId)                                 // No page or content to check
548            || !RenderingContext.FRONT.toString().equals(AmetysXSLTHelper.renderingContext())     // We are no on front
549            || editionModeOnly && !AmetysXSLTHelper.isEditionMode()   // Not in edition mode
550            || !hasFrontEditionRight())                                 // The user has the front edition right on current page
551        {
552            
553            return false;
554        }
555        
556        Request request = ContextHelper.getRequest(_context);
557        
558        // Retrieve the current workspace.
559        String currentWsp = RequestAttributeWorkspaceSelector.getForcedWorkspace(request);
560        
561        try
562        {
563            // Force default workspace (rights have to be checked on default workspace)
564            // The target objectId can be a node page, available of front only when a first subpage will be created
565            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, RepositoryConstants.DEFAULT_WORKSPACE);
566            
567            if (StringUtils.contains(aoId, ";"))
568            {
569                String[] pageIds = StringUtils.split(aoId, ";");
570                for (String pid : pageIds)
571                {
572                    if (_rightManager.currentUserHasRight(rightId, _ametysObjectResolver.resolveById(pid)) == RightResult.RIGHT_ALLOW)   // Check right on page (If this page does not exists an exception will be thrown... this is fine)
573                    {
574                        return true;
575                    }
576                    // else check the next one
577                }
578                return false; // No right inside the list
579            }
580            else
581            {
582                return StringUtils.isEmpty(rightId) // There is no right to check OR the user has the right
583                    || _rightManager.currentUserHasRight(rightId, _ametysObjectResolver.resolveById(aoId)) == RightResult.RIGHT_ALLOW;   // Check right on page (If this page does not exists an exception will be thrown... this is fine)
584            }
585        }
586        finally
587        {
588            // Restore workspace
589            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, currentWsp);
590        }
591    }
592    
593    /**
594     * Check if current user has right on Ametys object for front edition
595     * Checks :
596     * * Rendering context == front
597     * * Edition mode
598     * * Front_Edition_Access_Right on current page
599     * * RightId available on object
600     * @param rightId right to check (can be null)
601     * @param ao The ametys object check. Cannot be null
602     * @param editionModeOnly Check if the user is in edition mode
603     * @return true if all is OK, false if at least one is false
604     */
605    public static boolean hasFrontEditionRight(String rightId, AmetysObject ao, boolean editionModeOnly)
606    {
607        if (!RenderingContext.FRONT.toString().equals(AmetysXSLTHelper.renderingContext())     // We are no on front
608            || editionModeOnly && !AmetysXSLTHelper.isEditionMode()   // Not in edition mode
609            || !hasFrontEditionRight())                                 // The user has the front edition right on current page
610        {
611            return false;
612        }
613        
614        Request request = ContextHelper.getRequest(_context);
615        
616        // Retrieve the current workspace.
617        String currentWsp = RequestAttributeWorkspaceSelector.getForcedWorkspace(request);
618        
619        try
620        {
621            // Force default workspace (rights have to be checked on default workspace)
622            // The target objectId can be a node page, available of front only when a first subpage will be created
623            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, RepositoryConstants.DEFAULT_WORKSPACE);
624            
625            return StringUtils.isEmpty(rightId) // There is no right to check OR the user has the right
626                    || _rightManager.currentUserHasRight(rightId, ao) == RightResult.RIGHT_ALLOW;   // Check right on page (If this page does not exists an exception will be thrown... this is fine)
627        }
628        finally
629        {
630            // Restore workspace
631            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, currentWsp);
632        }
633    }
634    
635    /**
636     * Check if the current user has the Front_Edition_Access_Right right
637     * @return true if right granted
638     */
639    public static boolean hasFrontEditionRight()
640    {
641        Request request = ContextHelper.getRequest(_context);
642        
643        // Retrieve the current workspace.
644        String currentWsp = RequestAttributeWorkspaceSelector.getForcedWorkspace(request);
645        
646        try
647        {
648            // Force default workspace (rights have to be checked on default workspace)
649            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, RepositoryConstants.DEFAULT_WORKSPACE);
650            
651            String pageId = AmetysXSLTHelper.pageId();
652            if (StringUtils.isNotEmpty(pageId))
653            {
654                Page page = _ametysObjectResolver.resolveById(pageId);
655                return _rightManager.currentUserHasRight(FRONT_EDITION_RIGHT_ID, page) == RightResult.RIGHT_ALLOW;
656            }
657            
658            String lang = AmetysXSLTHelper.lang();
659            if (StringUtils.isEmpty(lang))
660            {
661                lang = (String) request.getAttribute("lang");
662            }
663            String siteName = AmetysXSLTHelper.site();
664            
665            if (StringUtils.isNotEmpty(siteName) && StringUtils.isNoneEmpty(lang))
666            {
667                Site site = _siteManager.getSite(siteName);
668                Sitemap sitemap = site.getSitemap(lang);
669                return _rightManager.currentUserHasRight(FRONT_EDITION_RIGHT_ID, sitemap) == RightResult.RIGHT_ALLOW;
670            }
671            
672            return false;
673        }
674        finally
675        {
676            // Restore workspace
677            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, currentWsp);
678        }
679        
680        
681    }
682    /**
683     * Check if the current workflow action is available on this content
684     * @param actionId action to check
685     * @param contentId content to check
686     * @param checkEditionMode check also if we are in edition mode
687     * @return true if right is granted
688     */
689    public static boolean hasWorkflowRight(int actionId, String contentId, boolean checkEditionMode)
690    {
691        List<Integer> actionIds = new ArrayList<>();
692        actionIds.add(actionId);
693        return hasWorkflowRight(actionIds, contentId, checkEditionMode);
694    }
695    /**
696     * Check if a list of workflow actions are available on this content
697     * @param actionIds list of action ids
698     * @param contentId id of the content to check
699     * @param checkEditionMode check also if we are in edition mode
700     * @return true if all actions are available
701     */
702    public static boolean hasWorkflowRight(List<Integer> actionIds, String contentId, boolean checkEditionMode)
703    {
704        Content content = _ametysObjectResolver.resolveById(contentId);
705        return hasWorkflowRight(actionIds, content, checkEditionMode);
706    }
707    /**
708     * Check if a list of workflow actions are available on this content
709     * @param actionIds list of action ids
710     * @param content content to check
711     * @param checkEditionMode check also if we are in edition mode
712     * @return true if all actions are available
713     */
714    public static boolean hasWorkflowRight(List<Integer> actionIds, Content content, boolean checkEditionMode)
715    {
716        if (checkEditionMode)
717        {
718            boolean editionModeOk = AmetysXSLTHelper.isEditionMode();
719            if (!editionModeOk)
720            {
721                return false;
722            }
723        }
724        boolean result = false;
725        if (content instanceof WorkflowAwareContent)
726        {
727            WorkflowAwareContent wcontent = (WorkflowAwareContent) content;
728            int[] availableActions = _contentWorkflowHelper.getAvailableActions(wcontent);
729            for (int actionId : actionIds)
730            {
731                result = result || ArrayUtils.contains(availableActions, actionId);
732                if (result)
733                {
734                    break;
735                }
736            }
737        }
738        return result;
739    }
740    
741    /**
742     * Get the toolbar configuration for page: retrieve the user rights, the parent and page position
743     * @param pageId The page id
744     * @param addContentsConfigAsJson The items configuration for adding contents as JSON string
745     * @param editionModeOnly true to check edition mode
746     * @return the user's rights, parent and page position
747     * @throws UnknownAmetysObjectException if content is unknown
748     * @throws AmetysRepositoryException if an error occurred
749     * @throws RepositoryException if an error occurred
750     */
751    public static String getPageToolbarConfig(String pageId, String addContentsConfigAsJson, boolean editionModeOnly) throws UnknownAmetysObjectException, AmetysRepositoryException, RepositoryException
752    {
753        Map<String, Object> result = new HashMap<>();
754        
755        Page page = _ametysObjectResolver.resolveById(pageId);
756        SitemapElement parent = page.getParent();
757        long position = _getPagePosition(page);
758        
759        List<Object> addContentsConfig = _jsonUtils.convertJsonToList(addContentsConfigAsJson);
760        
761        @SuppressWarnings("unchecked")
762        Set<Object> allowedContentTypes = addContentsConfig.stream()
763            .filter(Map.class::isInstance)
764            .filter(cfg -> _hasContentCreationRight((Map) cfg))
765            .map(cfg -> (String) ((Map) cfg).get("contentType"))
766            .collect(Collectors.toSet());
767        result.put("allowedContentTypes", allowedContentTypes);
768        
769        result.put("title", page.getTitle());
770        result.put("parentId", parent.getId());
771        result.put("position", position);
772
773        // Get the active status for each action depending on user's rights, page position, FO edition mode, ...
774        result.put("actionStatus", _getActionStatusOnPage(page, position, parent.getChildrenPages().getSize(), editionModeOnly));
775        
776        return _jsonUtils.convertObjectToJson(result);
777    }
778    
779    private static boolean _hasContentCreationRight(Map<String, Object> config)
780    {
781        if (config.containsKey("contentType"))
782        {
783            String cTypeId = (String) config.get("contentType");
784            
785            ContentType contentType = _contentTypesEP.getExtension(cTypeId);
786            if (contentType == null)
787            {
788                _logger.warn("Unknown content type '" + cTypeId + "'");
789                return false;
790            }
791            
792            String right = contentType.getRight();
793            if (StringUtils.isEmpty(right))
794            {
795                return true;
796            }
797            
798            return _rightManager.currentUserHasRight(right, "/cms") == RightResult.RIGHT_ALLOW || _rightManager.currentUserHasRight(right, _rootContentHelper.getRootContent()) == RightResult.RIGHT_ALLOW;
799        }
800        else
801        {
802            _logger.warn("Configuration to add content is invalid: missing 'contentType'. It will be ignored");
803            return false;
804        }
805    }
806    
807    private static long _getPagePosition(Page page) throws AmetysRepositoryException
808    {
809        SitemapElement parent = page.getParent();
810        AmetysObjectIterable< ? extends Page> children = parent.getChildrenPages();
811        
812        long count = 1;
813        for (Page child : children)
814        {
815            if (child.getId().equals(page.getId()))
816            {
817                return count;
818            }
819            count++;
820        }
821        
822        // Not found
823        return -1;
824    }
825    
826    private static Map<String, Boolean> _getActionStatusOnPage(Page page, long position, long size, boolean editionModeOnly) throws UnknownAmetysObjectException, AmetysRepositoryException
827    {
828        Map<String, Boolean> rights = new HashMap<>();
829        
830        if (page instanceof ModifiablePage)
831        {
832            rights.put("add-content", hasFrontEditionRight("Web_Rights_Page_AddContent", page, editionModeOnly));
833            rights.put("add-page", hasFrontEditionRight("Web_Rights_Page_Create", page, editionModeOnly));
834            rights.put("tag", hasFrontEditionRight("Web_Rights_Page_Tag", page, editionModeOnly));
835            rights.put("rename", hasFrontEditionRight("Web_Rights_Page_Rename", page, editionModeOnly));
836            rights.put("schedule-publication", hasFrontEditionRight("Web_Rights_Page_Schedule", page, editionModeOnly));
837            rights.put("delete", hasFrontEditionRight("Web_Rights_Page_Delete", page, editionModeOnly));
838            
839            boolean canMove = hasFrontEditionRight("Web_Rights_Page_Create", page.getParent().getId(), editionModeOnly);
840            boolean moveUp = canMove && position > 1;
841            boolean moveDown = canMove && position < size;
842            
843            rights.put("move-up", moveUp);
844            rights.put("move-down", moveDown);
845            rights.put("move", moveDown || moveUp);
846        }
847        
848        return rights;
849    }
850    
851    /**
852     * Get the toolbar configuration for content: retrieve user rights, available workflow actions, content current step and status history
853     * @param contentId The content id
854     * @param zoneItemId The zone item id
855     * @param restrictedActions The actions ids to retrieve. Set to null or empty to get all available workflow actions with no restriction
856     * @param jsonStepConfig The step configuration
857     * @param locale the locale for translation
858     * @param editionModeOnly true to check edition mode
859     * @return the user's rights, available workflow actions and content current step
860     * @throws UnknownAmetysObjectException if content is unknown
861     * @throws AmetysRepositoryException if an error occurred
862     * @throws RepositoryException if an error occurred
863     */
864    public static String getContentToolbarConfig(String contentId, String zoneItemId, String restrictedActions, String jsonStepConfig, String locale, boolean editionModeOnly) throws UnknownAmetysObjectException, AmetysRepositoryException, RepositoryException
865    {
866        Map<String, Object> result = new HashMap<>();
867        
868        Content content = _ametysObjectResolver.resolveById(contentId);
869
870        // Get the active status of each action depending on user rights, FO mode, content's position, ...
871        result.put("actionStatus", _getActionStatusOnContent(content, zoneItemId, editionModeOnly));
872        
873        if (content instanceof WorkflowAwareContent)
874        {
875            WorkflowDescriptor workflowDescriptor = _getWorkflowDescriptor((WorkflowAwareContent) content);
876            
877            WorkflowAwareContent wcontent = (WorkflowAwareContent) content;
878            
879            Map<String, Object> step = new HashMap<>();
880            long currentStepId = wcontent.getCurrentStepId();
881            
882            // Current step
883            StepDescriptor stepDescriptor = _getStepDescriptor(workflowDescriptor, currentStepId);
884            step.put("id", currentStepId);
885            String name = stepDescriptor.getName();
886            step.put("label", new I18nizableText(StringUtils.substringBefore(name, ":"), StringUtils.substringAfter(name, ":")));
887            result.put("step", step);
888            
889            Map<String, Object> stepConfig = StringUtils.isEmpty(jsonStepConfig) ? null : _jsonUtils.convertJsonToMap(jsonStepConfig);
890            result.put("statusHistory", _getStatusHistory(wcontent, currentStepId, stepConfig, workflowDescriptor));
891            
892            List<Integer> availableActions = Arrays.stream(_contentWorkflowHelper.getAvailableActions(wcontent)).boxed().collect(Collectors.toList());
893            result.put("availableWorkflowActions", availableActions);
894            
895            if (!editionModeOnly || AmetysXSLTHelper.isEditionMode())
896            {
897                List<Object> restrictedActionsAsList = StringUtils.isEmpty(restrictedActions) ? null : _jsonUtils.convertJsonToList(restrictedActions);
898                
899                List<Object> restrictedActionIds = restrictedActionsAsList == null ? null : restrictedActionsAsList.stream()
900                    .map(o -> o instanceof Map ? ((Map) o).get("id") : o)
901                    .collect(Collectors.toList());
902                
903                @SuppressWarnings("unchecked")
904                Map<Object, Map<String, Object>> workflowActions = availableActions.stream()
905                    .filter(actionId -> restrictedActionIds == null || restrictedActionIds.contains(actionId))
906                    .map(actionId -> {
907                        Map<String, Object> wa = new HashMap<>();
908                        
909                        _addDefaultWorkflowActionProperties(wa, workflowDescriptor, actionId);
910                        
911                        if (restrictedActionIds != null && restrictedActionsAsList != null && restrictedActionIds.contains(actionId))
912                        {
913                            for (Object object : restrictedActionsAsList)
914                            {
915                                if (object instanceof Map && ((Map) object).get("id") == actionId)
916                                {
917                                    // get or override properties from map sent by client side
918                                    wa.putAll((Map<String, Object>) object);
919                                    break;
920                                }
921                            }
922                            
923                        }
924                        return Pair.of(actionId, wa);
925                    })
926                    .collect(Collectors.toMap(Pair::getLeft, Pair::getRight));
927                
928                result.put("workflowActions", workflowActions);
929            }
930        }
931        
932        Request request = ContextHelper.getRequest(_context);
933        try
934        {
935            request.setAttribute(I18nizableSerializer.REQUEST_ATTR_LOCALE, locale);
936            return _jsonUtils.convertObjectToJson(result);
937        }
938        finally
939        {
940            request.removeAttribute(I18nizableSerializer.REQUEST_ATTR_LOCALE);
941        }
942    }
943    
944    private static Map<String, Boolean> _getActionStatusOnContent(Content content, String zoneItemId, boolean editionModeOnly) throws UnknownAmetysObjectException, AmetysRepositoryException
945    {
946        Map<String, Boolean> rights = new HashMap<>();
947        
948        ZoneItem zoneItem = _ametysObjectResolver.resolveById(zoneItemId);
949        Zone zone = zoneItem.getZone();
950        SitemapElement sitemapElement = zone.getSitemapElement();
951        
952        rights.put("tag", hasFrontEditionRight("CMS_Rights_Content_Tag", content, editionModeOnly));
953        rights.put("remove", sitemapElement instanceof ModifiablePage && hasFrontEditionRight("Web_Rights_Page_DeleteZoneItem", sitemapElement, editionModeOnly));
954        rights.put("delete", hasFrontEditionRight("CMS_Rights_DeleteContent", content, editionModeOnly));
955        
956        boolean moveUp = false;
957        boolean moveDown = false;
958        
959        long position = getZoneItemPosition(zoneItem);
960        long size = getZoneSize(zone);
961        
962        if (position != -1 && sitemapElement instanceof ModifiablePage && hasFrontEditionRight("Web_Rights_Page_OrganizeZoneItem", sitemapElement, editionModeOnly))
963        {
964            if (position > 0)
965            {
966                moveUp = true;
967            }
968            
969            if (position + 1 < size)
970            {
971                moveDown = true;
972            }
973        }
974        rights.put("move-up", moveUp);
975        rights.put("move-down", moveDown);
976        rights.put("move", moveDown || moveUp);
977        
978        return rights;
979    }
980    
981    private static Map<Integer, Map<String, Object>> _getStatusHistory(WorkflowAwareContent content, long currentStepId, Map<String, Object> stepsConfig, WorkflowDescriptor workflowDescriptor)
982    {
983        List<Step> historyStatus = new ArrayList<>();
984            
985        if (stepsConfig.containsKey(String.valueOf(currentStepId)))
986        {
987            AmetysObjectWorkflow workflow = _workflowProvider.getAmetysObjectWorkflow(content);
988            
989            long workflowId = content.getWorkflowId();
990            List<Step> historySteps = workflow.getHistorySteps(workflowId);
991
992            // Sort by start date descendant
993            Collections.sort(historySteps, new Comparator<Step>()
994            {
995                public int compare(Step s1, Step s2)
996                {
997                    return -s1.getStartDate().compareTo(s2.getStartDate());
998                }
999            });
1000            
1001            @SuppressWarnings("unchecked")
1002            Map<String, Object> currentStepCfg = (Map<String, Object>) stepsConfig.get(String.valueOf(currentStepId));
1003            String currentStepType = currentStepCfg != null ? (String) currentStepCfg.get("type") : null;
1004            
1005            boolean stop = false;
1006            for (Step step : historySteps)
1007            {
1008                if (stop)
1009                {
1010                    break;
1011                }
1012
1013                long stepId = step.getStepId();
1014                @SuppressWarnings("unchecked")
1015                Map<String, Object> stepCfg = (Map<String, Object>) stepsConfig.get(String.valueOf(stepId));
1016                String stepType = stepCfg != null ? (String) stepCfg.get("type") : null;
1017                
1018                stop = "positive".equals(currentStepType) && !"positive".equals(stepType)
1019                        || ("negative".equals(currentStepType) || "neutral".equals(currentStepType)) && !"negative".equals(stepType)
1020                        || currentStepId == stepId;
1021                
1022                if (!stop)
1023                {
1024                    historyStatus.add(step);
1025                }
1026            }
1027        }
1028        
1029        Map<Integer, Map<String, Object>> jsonHistoryStatus = new LinkedHashMap<>();
1030        for (Step step : historyStatus)
1031        {
1032            if (!jsonHistoryStatus.containsKey(step.getStepId()))
1033            {
1034                StepDescriptor stepDescriptor = _getStepDescriptor(workflowDescriptor, step.getStepId());
1035                String name = stepDescriptor.getName();
1036                I18nizableText label = new I18nizableText(StringUtils.substringBefore(name, ":"), StringUtils.substringAfter(name, ":"));
1037                jsonHistoryStatus.put(step.getStepId(), Map.of("label", label));
1038            }
1039        }
1040        
1041        return jsonHistoryStatus;
1042    }
1043    
1044    private static void _addDefaultWorkflowActionProperties(Map<String, Object> wa, WorkflowDescriptor workflowDescriptor, int actionId)
1045    {
1046        // Add workflow action label
1047        ActionDescriptor wAction = workflowDescriptor.getAction(actionId);
1048        String name = wAction.getName();
1049        wa.put("label", new I18nizableText(StringUtils.substringBefore(name, ":"), StringUtils.substringAfter(name, ":")));
1050        
1051        // Default workflow icon
1052        switch (actionId)
1053        {
1054            case 3:
1055                wa.put("icon", "fas fa-hand-point-up"); // Propose
1056                break;
1057            case 4:
1058                wa.put("icon", "fas fa-check"); // Validate
1059                break;
1060            case 7:
1061                wa.put("icon", "fas fa-times"); // Refuse
1062                break;
1063            case 10:
1064                wa.put("icon", "fas fa-ban"); // Unpublish
1065                break;
1066            default:
1067                break;
1068        }
1069    }
1070    
1071    private static WorkflowDescriptor _getWorkflowDescriptor(WorkflowAwareContent content)
1072    {
1073        long workflowId = content.getWorkflowId();
1074        
1075        AmetysObjectWorkflow workflow = _workflowProvider.getAmetysObjectWorkflow(content);
1076        
1077        String workflowName = workflow.getWorkflowName(workflowId);
1078        return workflow.getWorkflowDescriptor(workflowName);
1079    }
1080    
1081    private static StepDescriptor _getStepDescriptor(WorkflowDescriptor workflowDescriptor, long stepId)
1082    {
1083        if (workflowDescriptor != null)
1084        {
1085            StepDescriptor stepDescriptor = workflowDescriptor.getStep((int) stepId);
1086            if (stepDescriptor != null)
1087            {
1088                return stepDescriptor;
1089            }
1090        }
1091        
1092        return null;
1093    }
1094    
1095    /**
1096     * Get a string representing a json map of the contentIds with true/false if they can/can't be modified
1097     * @param actionId action to check
1098     * @param inputPageId page (can be null)
1099     * @param checkEditionMode check also if we are in edition mode
1100     * @return {contentId : true/false, ....}
1101     */
1102    public static String getModifiableContents(int actionId, String inputPageId, boolean checkEditionMode)
1103    {
1104        Map<String, Object> jsonObject = new HashMap<>();
1105        String pageId = inputPageId;
1106        if (StringUtils.isEmpty(inputPageId))
1107        {
1108            pageId = AmetysXSLTHelper.pageId();
1109        }
1110        
1111        if (!StringUtils.isEmpty(pageId))
1112        {
1113            AmetysObject resolveById = _ametysObjectResolver.resolveById(pageId);
1114            if (resolveById instanceof Page)
1115            {
1116                Page page = (Page) resolveById;
1117                List<Content> contents = _pageHelper.getAllContents(page);
1118                for (Content content : contents)
1119                {
1120                    Map<String, Object> contentInfo = new HashMap<>();
1121                    contentInfo.put("unmodifiableAttributes", AmetysFrontEditionHelper.getUnmodifiableAttributes(content, List.of(actionId), checkEditionMode));
1122                    contentInfo.put("rights", AmetysFrontEditionHelper.getRightsForContent(content));
1123                    
1124                    jsonObject.put(content.getId(), contentInfo);
1125                }
1126            }
1127        }
1128
1129        return _jsonUtils.convertObjectToJson(jsonObject);
1130    }
1131    
1132    /**
1133     * Get path of unmodifiable attributes. Check if content can be modified and the restrictions on attributes
1134     * @param content the content
1135     * @param actionIds the workflow action ids to check
1136     * @param checkEditionMode to check edition mode
1137     * @return the list with the path of unmodifiable attributes (regarding of restrictions) or null if the content cannot be modified
1138     */
1139    public static List<String> getUnmodifiableAttributes(Content content, List<Integer> actionIds, boolean checkEditionMode)
1140    {
1141        try
1142        {
1143            boolean hasWorkflowRight = hasWorkflowRight(actionIds, content, checkEditionMode);
1144            if (!hasWorkflowRight)
1145            {
1146                // content is not editable
1147                return null;
1148            }
1149            
1150            Collection<? extends ModelItem> modelItems = _contentTypesHelper.getModelItems(content.getTypes());
1151            List<AttributeDefinition> attributes = _getAttributeDefinitionsRecursively(modelItems);
1152            return attributes.stream()
1153                .filter(mi -> !mi.canWrite(content))
1154                .map(mi -> mi.getPath())
1155                .collect(Collectors.toList());
1156        }
1157        catch (IllegalArgumentException e)
1158        {
1159            _logger.error("Fail to get restricted model items for content " + content.getId(), e);
1160            return List.of();
1161        }
1162    }
1163    
1164    /**
1165     * Get a map of chosen rights for a content
1166     * @param content the content
1167     * @return the chosen rights for a content
1168     */
1169    public static Map<String, Object> getRightsForContent(Content content)
1170    {
1171        Map<String, Object> rights = new HashMap<>();
1172        rights.put("hasAddFileRight", _rightManager.currentUserHasRight(ExplorerResourcesDAO.RIGHTS_RESOURCE_ADD, content) == RightResult.RIGHT_ALLOW);
1173        return rights;
1174    }
1175    
1176    private static List<AttributeDefinition> _getAttributeDefinitionsRecursively(Collection<? extends ModelItem> modelItems)
1177    {
1178        List<AttributeDefinition> attributeDefs = new ArrayList<>();
1179        
1180        for (ModelItem modelItem : modelItems)
1181        {
1182            if (modelItem instanceof AttributeDefinition)
1183            {
1184                attributeDefs.add((AttributeDefinition) modelItem);
1185            }
1186            else if (modelItem instanceof ModelItemGroup)
1187            {
1188                attributeDefs.addAll(_getAttributeDefinitionsRecursively(((ModelItemGroup) modelItem).getModelItems()));
1189            }
1190        }
1191        
1192        return attributeDefs;
1193    }
1194    
1195    /**
1196     * Get the name of a workflow action
1197     * @param workflowName workflow name
1198     * @param actionId action id in the workflow
1199     * @return name of the workflow action
1200     */
1201    public static String getWorkflowName(String workflowName, int actionId)
1202    {
1203        String workflowNameKey = _workflowHelper.getActionName(workflowName, actionId);
1204        String translatedName = org.ametys.core.util.AmetysXSLTHelper.translate(workflowNameKey);
1205        return translatedName;
1206    }
1207    /**
1208     * Get the status (validated/draft) of all contents in a page
1209     * @param pageId id of the page to check
1210     * @return "" if not a page, "draft" if all contents are drafts, "validated" if all contents are validated, and "mixed" if there are both
1211     */
1212    public static String getPageStatus(String pageId)
1213    {
1214        boolean hasDraft = false;
1215        boolean hasValidated = false;
1216        AmetysObject resolveById = _ametysObjectResolver.resolveById(pageId);
1217        if (resolveById instanceof Page)
1218        {
1219            Page page = (Page) resolveById;
1220            List<Content> contents = _pageHelper.getAllContents(page);
1221            for (Content content : contents)
1222            {
1223                if (isContentLive(content))
1224                {
1225                    hasValidated = true;
1226                }
1227                else
1228                {
1229                    hasDraft = true;
1230                }
1231            }
1232            String result = "none";
1233            if (hasDraft && !hasValidated)
1234            {
1235                result = "draft";
1236            }
1237            else if (!hasDraft && hasValidated)
1238            {
1239                result = "validated";
1240            }
1241            else if (hasDraft && hasValidated)
1242            {
1243                result = "mixed";
1244            }
1245            return result;
1246        }
1247        else
1248        {
1249            return "";
1250        }
1251    }
1252
1253    /**
1254     * Get the workflow step id of a content
1255     * @param contentId id of the content
1256     * @return the step Id
1257     */
1258    public static long getContentWorkflowId(String contentId)
1259    {
1260        AmetysObject resolvedById = _ametysObjectResolver.resolveById(contentId);
1261        if (resolvedById instanceof Content)
1262        {
1263            Content content = (Content) resolvedById;
1264            return getContentWorkflowId(content);
1265        }
1266        else
1267        {
1268            return 0;
1269        }
1270    }
1271    /**
1272     * Get the workflow step id of a content
1273     * @param content content
1274     * @return the step Id
1275     */
1276    public static long getContentWorkflowId(Content content)
1277    {
1278        WorkflowAwareContent wContent = (WorkflowAwareContent) content;
1279        return wContent.getCurrentStepId();
1280    }
1281
1282    /**
1283     * check if a content is published ot not
1284     * @param contentId id of the content
1285     * @return "validated" or "draft", "" if not found
1286     */
1287    public static String getContentStatus(String contentId)
1288    {
1289        AmetysObject resolvedById = _ametysObjectResolver.resolveById(contentId);
1290        if (resolvedById instanceof Content)
1291        {
1292            Content content = (Content) resolvedById;
1293            return getContentStatus(content);
1294        }
1295        else
1296        {
1297            return "";
1298        }
1299    }
1300    /**
1301     * check if a content is published ot not
1302     * @param content content
1303     * @return "validated" or "draft"
1304     */
1305    public static String getContentStatus(Content content)
1306    {
1307        return isContentLive(content) ? "validated" : "draft";
1308    }
1309    /**
1310     * check if a content is published ot not
1311     * @param contentId id of the content
1312     * @return true if validated
1313     */
1314    public static boolean isContentLive(String contentId)
1315    {
1316        AmetysObject resolvedById = _ametysObjectResolver.resolveById(contentId);
1317        if (resolvedById instanceof Content)
1318        {
1319            Content content = (Content) resolvedById;
1320            return isContentLive(content);
1321        }
1322        else
1323        {
1324            return false;
1325        }
1326    }
1327    /**
1328     * check if a content is published ot not
1329     * @param content content
1330     * @return true if validated
1331     */
1332    public static boolean isContentLive(Content content)
1333    {
1334        return Arrays.asList(((VersionableAmetysObject) content).getLabels()).contains(CmsConstants.LIVE_LABEL);
1335    }
1336    /**
1337     * Get the position of this zoneItem in the zone, or -1 if not found
1338     * @param zoneItemId zoneitem to search
1339     * @return zero based position, -1 if not found
1340     * @throws UnknownAmetysObjectException If zone item is not found
1341     * @throws AmetysRepositoryException If an error occurred
1342     */
1343    public static long getZoneItemPosition(String zoneItemId) throws UnknownAmetysObjectException, AmetysRepositoryException
1344    {
1345        ZoneItem zoneItem = _ametysObjectResolver.resolveById(zoneItemId);
1346        return getZoneItemPosition(zoneItem);
1347    }
1348    
1349    /**
1350     * Get the position of this zoneItem in its parent zone, or -1 if not found
1351     * @param zoneItem zoneitem
1352     * @return zero based position, -1 if not found
1353     */
1354    public static long getZoneItemPosition(ZoneItem zoneItem)
1355    {
1356        AmetysObject parent = zoneItem.getParent();
1357        if (parent instanceof DefaultTraversableAmetysObject)
1358        {
1359            return ((DefaultTraversableAmetysObject) parent).getChildPosition(zoneItem);
1360        }
1361        return -1; // virtual page
1362    }
1363    
1364    /**
1365     * Return the size of a zone
1366     * @param zoneName zone name
1367     * @param pageId page Id
1368     * @return size of the zone (-1 if not found)
1369     */
1370    public static long getZoneSize(String zoneName, String pageId)
1371    {
1372        Page page = _ametysObjectResolver.resolveById(pageId);
1373        Zone zone = page.getZone(zoneName);
1374        return getZoneSize(zone);
1375    }
1376    
1377    /**
1378     * Return the size of a zone
1379     * @param zone the zone
1380     * @return size of the zone or -1 is zone is null
1381     */
1382    public static long getZoneSize(Zone zone)
1383    {
1384        if (zone != null)
1385        {
1386            return zone.getZoneItems().getSize();
1387        }
1388        return -1;
1389    }
1390    /**
1391     * Test if a page is modifiable
1392     * @param pageId id of the page to check
1393     * @return true if the page is modifiable, false otherwise
1394     */
1395    public static boolean isPageModifiable(String pageId)
1396    {
1397        try
1398        {
1399            AmetysObject resolveById = _ametysObjectResolver.resolveById(pageId);
1400            return resolveById instanceof ModifiablePage;
1401        }
1402        catch (AmetysRepositoryException e)
1403        {
1404            // Nothing, this is not a page
1405        }
1406        
1407        return false;
1408    }
1409    
1410    /**
1411     * Returns the ids of the pages tagged with the specified tag in default workspace
1412     * @param tag The tag id
1413     * @return Array of pages ids
1414     */
1415    public static NodeList findPagesIdsByTagInDefaultWorkspace(String tag)
1416    {
1417        return findPagesIdsByTagInDefaultWorkspace(tag, false);
1418    }
1419    
1420    /**
1421     * Returns the ids of the pages tagged with the specified tag  in default workspace
1422     * @param tag The tag id
1423     * @param checkReadAccess true to return only pages with read access for current user
1424     * @return Array of pages ids
1425     */
1426    public static NodeList findPagesIdsByTagInDefaultWorkspace(String tag, boolean checkReadAccess)
1427    {
1428        Request request = ContextHelper.getRequest(_context);
1429        
1430        // Retrieve the current workspace.
1431        String currentWsp = RequestAttributeWorkspaceSelector.getForcedWorkspace(request);
1432        
1433        try
1434        {
1435            // Force default workspace
1436            // The tagged page can be a node page, available of front only when a first subpage will be created
1437            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, RepositoryConstants.DEFAULT_WORKSPACE);
1438            return AmetysXSLTHelper.findPagesIdsByTag(tag, checkReadAccess);
1439        }
1440        finally
1441        {
1442         // Restore workspace
1443            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, currentWsp);
1444        }
1445    }
1446    /**
1447     * Returns the ids of the pages tagged with the specified tag in default workspace
1448     * @param sitename The site id
1449     * @param lang The language code
1450     * @param tag The tag id
1451     * @return Array of pages ids
1452     */
1453    public static NodeList findPagesIdsByTagInDefaultWorkspace(String sitename, String lang, String tag)
1454    {
1455        return findPagesIdsByTagInDefaultWorkspace(sitename, lang, tag, false);
1456    }
1457    
1458    /**
1459     * Returns the ids of the pages tagged with the specified tag in default workspace
1460     * @param sitename The site id
1461     * @param lang The language code
1462     * @param tag The tag id
1463     * @param checkReadAccess true to return only pages with read access for current user
1464     * @return Array of pages ids
1465     */
1466    public static NodeList findPagesIdsByTagInDefaultWorkspace(String sitename, String lang, String tag, boolean checkReadAccess)
1467    {
1468        Request request = ContextHelper.getRequest(_context);
1469        
1470        // Retrieve the current workspace.
1471        String currentWsp = RequestAttributeWorkspaceSelector.getForcedWorkspace(request);
1472        
1473        try
1474        {
1475            // Force default workspace
1476            // The tagged page can be a node page, available of front only when a first subpage will be created
1477            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, RepositoryConstants.DEFAULT_WORKSPACE);
1478            return AmetysXSLTHelper.findPagesIdsByTag(sitename, lang, tag, checkReadAccess);
1479        }
1480        finally
1481        {
1482         // Restore workspace
1483            RequestAttributeWorkspaceSelector.setForcedWorkspace(request, currentWsp);
1484        }
1485    }
1486}