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}