001/* 002 * This library is part of OpenCms - 003 * the Open Source Content Management System 004 * 005 * Copyright (c) Alkacon Software GmbH & Co. KG (http://www.alkacon.com) 006 * 007 * This library is free software; you can redistribute it and/or 008 * modify it under the terms of the GNU Lesser General Public 009 * License as published by the Free Software Foundation; either 010 * version 2.1 of the License, or (at your option) any later version. 011 * 012 * This library is distributed in the hope that it will be useful, 013 * but WITHOUT ANY WARRANTY; without even the implied warranty of 014 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 015 * Lesser General Public License for more details. 016 * 017 * For further information about Alkacon Software GmbH & Co. KG, please see the 018 * company website: http://www.alkacon.com 019 * 020 * For further information about OpenCms, please see the 021 * project website: http://www.opencms.org 022 * 023 * You should have received a copy of the GNU Lesser General Public 024 * License along with this library; if not, write to the Free Software 025 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 026 */ 027 028package org.opencms.workplace; 029 030import org.opencms.ade.configuration.CmsElementView; 031import org.opencms.ade.containerpage.shared.CmsCntPageData.ElementDeleteMode; 032import org.opencms.ade.galleries.shared.CmsGallerySearchScope; 033import org.opencms.ade.upload.CmsDefaultUploadRestriction; 034import org.opencms.ade.upload.I_CmsUploadRestriction; 035import org.opencms.ade.upload.I_CmsVirusScanner; 036import org.opencms.configuration.CmsAdditionalLogFolderConfig; 037import org.opencms.configuration.CmsDefaultUserSettings; 038import org.opencms.db.CmsExportPoint; 039import org.opencms.db.CmsUserSettings; 040import org.opencms.db.I_CmsProjectDriver; 041import org.opencms.file.CmsFolder; 042import org.opencms.file.CmsObject; 043import org.opencms.file.CmsProject; 044import org.opencms.file.CmsPropertyDefinition; 045import org.opencms.file.CmsRequestContext; 046import org.opencms.file.CmsResource; 047import org.opencms.file.CmsResourceFilter; 048import org.opencms.file.CmsUser; 049import org.opencms.file.CmsVfsResourceNotFoundException; 050import org.opencms.file.types.CmsResourceTypeFolder; 051import org.opencms.file.types.CmsResourceTypeFolderExtended; 052import org.opencms.file.types.I_CmsResourceType; 053import org.opencms.i18n.CmsAcceptLanguageHeaderParser; 054import org.opencms.i18n.CmsEncoder; 055import org.opencms.i18n.CmsI18nInfo; 056import org.opencms.i18n.CmsLocaleComparator; 057import org.opencms.i18n.CmsLocaleManager; 058import org.opencms.i18n.I_CmsLocaleHandler; 059import org.opencms.loader.CmsLoaderException; 060import org.opencms.main.CmsBroadcast.ContentMode; 061import org.opencms.main.CmsEvent; 062import org.opencms.main.CmsException; 063import org.opencms.main.CmsLog; 064import org.opencms.main.I_CmsEventListener; 065import org.opencms.main.OpenCms; 066import org.opencms.module.CmsModule; 067import org.opencms.module.CmsModuleManager; 068import org.opencms.relations.CmsCategoryService; 069import org.opencms.security.CmsOrganizationalUnit; 070import org.opencms.security.CmsPermissionSet; 071import org.opencms.security.CmsPermissionViolationException; 072import org.opencms.security.CmsRole; 073import org.opencms.security.CmsRoleViolationException; 074import org.opencms.security.CmsSecurityException; 075import org.opencms.security.I_CmsPrincipal; 076import org.opencms.util.CmsRfsFileViewer; 077import org.opencms.util.CmsStringUtil; 078import org.opencms.util.CmsUUID; 079import org.opencms.workplace.CmsAccountInfo.Field; 080import org.opencms.workplace.editors.CmsEditorDisplayOptions; 081import org.opencms.workplace.editors.CmsEditorHandler; 082import org.opencms.workplace.editors.CmsWorkplaceEditorManager; 083import org.opencms.workplace.editors.I_CmsEditorActionHandler; 084import org.opencms.workplace.editors.I_CmsEditorCssHandler; 085import org.opencms.workplace.editors.I_CmsEditorHandler; 086import org.opencms.workplace.editors.I_CmsPreEditorActionDefinition; 087import org.opencms.workplace.editors.directedit.CmsDirectEditDefaultProvider; 088import org.opencms.workplace.editors.directedit.I_CmsDirectEditProvider; 089import org.opencms.workplace.explorer.CmsExplorerTypeAccess; 090import org.opencms.workplace.explorer.CmsExplorerTypeSettings; 091import org.opencms.workplace.galleries.A_CmsAjaxGallery; 092import org.opencms.workplace.tools.CmsToolManager; 093 094import java.io.IOException; 095import java.io.UnsupportedEncodingException; 096import java.net.URL; 097import java.util.ArrayList; 098import java.util.Arrays; 099import java.util.Collections; 100import java.util.Enumeration; 101import java.util.HashMap; 102import java.util.HashSet; 103import java.util.Iterator; 104import java.util.List; 105import java.util.Locale; 106import java.util.Map; 107import java.util.Set; 108import java.util.concurrent.TimeUnit; 109import java.util.jar.Manifest; 110import java.util.regex.Pattern; 111import java.util.regex.PatternSyntaxException; 112 113import javax.servlet.http.HttpServletRequest; 114import javax.servlet.http.HttpSession; 115 116import org.apache.commons.logging.Log; 117 118import com.google.common.cache.Cache; 119import com.google.common.cache.CacheBuilder; 120import com.google.common.collect.Lists; 121import com.google.common.collect.Maps; 122import com.google.common.collect.Sets; 123 124/** 125 * Manages the global OpenCms workplace settings for all users.<p> 126 * 127 * This class reads the settings from the "opencms.properties" and stores them in member variables. 128 * For each setting one or more get methods are provided.<p> 129 * 130 * @since 6.0.0 131 */ 132public final class CmsWorkplaceManager implements I_CmsLocaleHandler, I_CmsEventListener { 133 134 /** 135 * Helper class used to easily define default view mappings for standard resource types.<p> 136 */ 137 static class ViewRules { 138 139 /** 140 * Internal view map.<p> 141 */ 142 private Map<String, String> m_viewMap = Maps.newHashMap(); 143 144 /** 145 * Creates a new instance.<p> 146 * 147 * @param rules an array of strings of the form 'foo,bar,baz:view1', where foo, ... are type names and view1 is a view name (explorertype) 148 */ 149 public ViewRules(String... rules) { 150 151 for (String rule : rules) { 152 int colIndex = rule.indexOf(':'); 153 if (colIndex != -1) { 154 String before = rule.substring(0, colIndex); 155 String after = rule.substring(colIndex + 1); 156 for (String token : CmsStringUtil.splitAsList(before, ",")) { 157 m_viewMap.put(token.trim(), after); 158 } 159 } 160 161 } 162 } 163 164 /** 165 * Gets the view configured for the given type.<p> 166 * 167 * @param type a resource type name 168 * @return the view explorer type for the given resource type 169 */ 170 public String getViewForType(String type) { 171 172 return m_viewMap.get(type); 173 174 } 175 } 176 177 /** The default encoding for the workplace (UTF-8). */ 178 public static final String DEFAULT_WORKPLACE_ENCODING = CmsEncoder.ENCODING_UTF_8; 179 180 /** The workplace localization manifest attribute name. */ 181 public static final String LOCALIZATION_ATTRIBUTE_NAME = "OpenCms-Localization"; 182 183 /** The manifest file resource name. */ 184 public static final String MANIFEST_RESOURCE_NAME = "META-INF/MANIFEST.MF"; 185 186 /** The id of the "requestedResource" parameter for the OpenCms login form. */ 187 public static final String PARAM_LOGIN_REQUESTED_RESOURCE = "requestedResource"; 188 189 /** Key name for the session workplace settings. */ 190 public static final String SESSION_WORKPLACE_SETTINGS = "__CmsWorkplace.WORKPLACE_SETTINGS"; 191 192 /** Default view configuration. */ 193 static ViewRules m_defaultViewRules = new ViewRules( 194 "folder,plain,jsp,htmlredirect,containerpage:view_basic", 195 "imagegallery,downloadgallery,linkgallery,subsitemap,content_folder:view_folders", 196 "formatter_config,xmlvfsbundle,propertyvfsbundle,bundledescriptor,sitemap_config,sitemap_master_config,site_plugin,attr_editor_config,module_config,elementview,seo_file,containerpage_template,inheritance_config,macro_formatter,flex_formatter,settings_config:view_configs", 197 "xmlcontent,pointer:view_other"); 198 199 /** The default account infos. */ 200 private static final CmsAccountInfo[] DEFAULT_ACCOUNT_INFOS = new CmsAccountInfo[] { 201 new CmsAccountInfo(Field.firstname, null, false), 202 new CmsAccountInfo(Field.lastname, null, false), 203 new CmsAccountInfo(Field.email, null, false), 204 new CmsAccountInfo(Field.institution, null, false)}; 205 206 /** The logger for this class. */ 207 private static final Log LOG = CmsLog.getLog(CmsWorkplaceManager.class); 208 209 /** Value of the acacia-unlock configuration option (may be null if not set). */ 210 private String m_acaciaUnlock; 211 212 /** The configured account infos. */ 213 private List<CmsAccountInfo> m_accountInfos; 214 215 /** The admin cms context. */ 216 private CmsObject m_adminCms; 217 218 /** If enabled, gives element authors more gallery-related permissions (mostly upload/replace). */ 219 private boolean m_allowElementAuthorToWorkInGalleries; 220 221 /** Indicates if auto-locking of resources is enabled or disabled. */ 222 private boolean m_autoLockResources; 223 224 /** The name of the local category folder(s). */ 225 private String m_categoryFolder; 226 227 /** The default access for explorer types. */ 228 private CmsExplorerTypeAccess m_defaultAccess; 229 230 /** The configured default locale of the workplace. */ 231 private Locale m_defaultLocale; 232 233 /** The default property setting for setting new property values. */ 234 private boolean m_defaultPropertiesOnStructure; 235 236 /** The default user settings. */ 237 private CmsDefaultUserSettings m_defaultUserSettings; 238 239 /** The configured dialog handlers. */ 240 private Map<String, I_CmsDialogHandler> m_dialogHandler; 241 242 /** The configured direct edit provider. */ 243 private I_CmsDirectEditProvider m_directEditProvider; 244 245 /** A flag, indicating if the categories should be displayed separated by repository in the category selection dialog. */ 246 private boolean m_displayCategoriesByRepository; 247 248 /** A flag, indicating if the categories should be displayed separated by repository in the category selection dialog. */ 249 private boolean m_displayCategorySelectionCollapsed; 250 251 /** The edit action handler. */ 252 private I_CmsEditorActionHandler m_editorAction; 253 254 /** The editor CSS handlers. */ 255 private List<I_CmsEditorCssHandler> m_editorCssHandlers; 256 257 /** The workplace editor display options. */ 258 private CmsEditorDisplayOptions m_editorDisplayOptions; 259 260 /** The editor handler. */ 261 private I_CmsEditorHandler m_editorHandler; 262 263 /** The editor manager. */ 264 private CmsWorkplaceEditorManager m_editorManager; 265 266 /** The element delete mode. */ 267 private ElementDeleteMode m_elementDeleteMode; 268 269 /** The flag if switching tabs in the advanced property dialog is enabled. */ 270 private boolean m_enableAdvancedPropertyTabs; 271 272 /** The configured encoding of the workplace. */ 273 private String m_encoding; 274 275 /** Categories column enabled? */ 276 private boolean m_explorerCategoriesEnabled; 277 278 /** Show only leaf categories in explorer? */ 279 private boolean m_explorerCategoriesLeavesOnly; 280 281 /** Show categories with their path in the explorer? */ 282 private boolean m_explorerCategoriesWithPath; 283 284 /** The explorer type settings. */ 285 private List<CmsExplorerTypeSettings> m_explorerTypeSettings; 286 287 /** The explorer type settings from the configured modules. */ 288 private List<CmsExplorerTypeSettings> m_explorerTypeSettingsFromModules; 289 290 /** The explorer type settings from the XML configuration. */ 291 private List<CmsExplorerTypeSettings> m_explorerTypeSettingsFromXml; 292 293 /** The explorer type settings as Map with resource type name as key. */ 294 private Map<String, CmsExplorerTypeSettings> m_explorerTypeSettingsMap; 295 296 /** The element views generated from explorer types. */ 297 private Map<CmsUUID, CmsElementView> m_explorerTypeViews = Maps.newHashMap(); 298 299 /** The workplace export points. */ 300 private Set<CmsExportPoint> m_exportPoints; 301 302 /** Maximum size of an upload file. */ 303 private int m_fileMaxUploadSize; 304 305 /** The instance used for reading portions of lines of a file to choose. */ 306 private CmsRfsFileViewer m_fileViewSettings; 307 308 /** The configured workplace galleries. */ 309 private Map<String, A_CmsAjaxGallery> m_galleries; 310 311 /** The configured gallery default scope. */ 312 private String m_galleryDefaultScope; 313 314 /** The group translation. */ 315 private I_CmsGroupNameTranslation m_groupNameTranslation; 316 317 /** The configured group translation class name. */ 318 private String m_groupTranslationClass; 319 320 /** Keep-alive flag. */ 321 private Boolean m_keepAlive; 322 323 /** Contains all folders that should be labeled if siblings exist. */ 324 private List<String> m_labelSiteFolders; 325 326 /** List of installed workplace locales, sorted ascending. */ 327 private List<Locale> m_locales; 328 329 /** The configured list of localized workplace folders. */ 330 private List<String> m_localizedFolders; 331 332 /** The additional log folder configuration. */ 333 private CmsAdditionalLogFolderConfig m_logFolderConfig = new CmsAdditionalLogFolderConfig(); 334 335 /** Max number of locale buttons to display in the editor. */ 336 private int m_maxLocaleButtons = 5; 337 338 /** The workplace localized messages (mapped to the locales). */ 339 private Map<Locale, CmsWorkplaceMessages> m_messages; 340 341 /** The post upload handler. */ 342 private I_CmsPostUploadDialogHandler m_postUploadHandler; 343 344 /** The condition definitions for the resource types which are triggered before opening the editor. */ 345 private List<I_CmsPreEditorActionDefinition> m_preEditorConditionDefinitions; 346 347 /** The repository folder handler. */ 348 private I_CmsRepositoryFolderHandler m_repositoryFolderHandler; 349 350 /** Indicates if the user management icon should be displayed in the workplace. */ 351 private boolean m_showUserGroupIcon; 352 353 /** The role required for editing the sitemap configuration. */ 354 private String m_sitemapConfigEditRole; 355 356 /** Exclude patterns for synchronization. */ 357 private ArrayList<Pattern> m_synchronizeExcludePatterns; 358 359 /** The temporary file project used by the editors. */ 360 private CmsProject m_tempFileProject; 361 362 /** The tool manager. */ 363 private CmsToolManager m_toolManager; 364 365 /** The upload restriction. */ 366 private I_CmsUploadRestriction m_uploadRestriction = CmsDefaultUploadRestriction.unrestricted(); 367 368 /** Keeps track of whether the upload restriction has been set. */ 369 private boolean m_uploadRestrictionSet; 370 371 /** The user additional information configuration. */ 372 private CmsWorkplaceUserInfoManager m_userInfoManager; 373 374 /** The user list mode. */ 375 private String m_userListMode; 376 377 /** The configured workplace views. */ 378 private List<CmsWorkplaceView> m_views; 379 380 /** The configured virus scanner. */ 381 private I_CmsVirusScanner m_virusScanner; 382 383 /** True if the virus scanner is enabled. */ 384 private boolean m_virusScannerEnabled; 385 386 /** Expiring cache used to limit the number of notifications sent because of invalid workplace server names. */ 387 private Cache<String, String> m_workplaceServerUserChecks; 388 389 /** The XML content auto correction flag. */ 390 private boolean m_xmlContentAutoCorrect; 391 392 /** 393 * Creates a new instance for the workplace manager, will be called by the workplace configuration manager.<p> 394 */ 395 public CmsWorkplaceManager() { 396 397 if (CmsLog.INIT.isInfoEnabled()) { 398 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_WORKPLACE_INITIALIZE_START_0)); 399 } 400 401 m_locales = new ArrayList<Locale>(); 402 m_labelSiteFolders = new ArrayList<String>(); 403 m_localizedFolders = new ArrayList<String>(); 404 m_autoLockResources = true; 405 m_categoryFolder = CmsCategoryService.REPOSITORY_BASE_FOLDER; 406 m_xmlContentAutoCorrect = true; 407 m_showUserGroupIcon = true; 408 m_dialogHandler = new HashMap<String, I_CmsDialogHandler>(); 409 m_views = new ArrayList<CmsWorkplaceView>(); 410 m_exportPoints = new HashSet<CmsExportPoint>(); 411 m_editorHandler = new CmsEditorHandler(); 412 m_fileMaxUploadSize = -1; 413 m_fileViewSettings = new CmsRfsFileViewer(); 414 m_explorerTypeSettingsFromXml = new ArrayList<CmsExplorerTypeSettings>(); 415 m_explorerTypeSettingsFromModules = new ArrayList<CmsExplorerTypeSettings>(); 416 m_defaultPropertiesOnStructure = true; 417 m_enableAdvancedPropertyTabs = true; 418 m_defaultUserSettings = new CmsDefaultUserSettings(); 419 m_defaultAccess = new CmsExplorerTypeAccess(); 420 m_galleries = new HashMap<String, A_CmsAjaxGallery>(); 421 flushMessageCache(); 422 m_preEditorConditionDefinitions = new ArrayList<I_CmsPreEditorActionDefinition>(); 423 m_editorCssHandlers = new ArrayList<I_CmsEditorCssHandler>(); 424 m_synchronizeExcludePatterns = new ArrayList<Pattern>(); 425 426 // important to set this to null to avoid unnecessary overhead during configuration phase 427 m_explorerTypeSettings = null; 428 CacheBuilder<Object, Object> cb = CacheBuilder.newBuilder().expireAfterWrite( 429 2, 430 TimeUnit.MINUTES).concurrencyLevel(3); 431 m_workplaceServerUserChecks = cb.build(); 432 } 433 434 /** 435 * Returns true if the provided request was done by a Workplace user.<p> 436 * 437 * @param req the request to check 438 * 439 * @return true if the provided request was done by a Workplace user 440 */ 441 public static boolean isWorkplaceUser(HttpServletRequest req) { 442 443 HttpSession session = req.getSession(false); 444 if (session != null) { 445 // if a session is available, check for a workplace configuration 446 CmsWorkplaceSettings workplaceSettings = (CmsWorkplaceSettings)session.getAttribute( 447 CmsWorkplaceManager.SESSION_WORKPLACE_SETTINGS); 448 return ((null != workplaceSettings) && !workplaceSettings.getUser().isGuestUser()); 449 } 450 // no session means no workplace use 451 return false; 452 } 453 454 /** 455 * Adds an account info.<p> 456 * 457 * @param info the account info to add 458 */ 459 public void addAccountInfo(CmsAccountInfo info) { 460 461 if (m_accountInfos == null) { 462 m_accountInfos = new ArrayList<CmsAccountInfo>(); 463 } 464 m_accountInfos.add(info); 465 } 466 467 /** 468 * Adds an account info.<p> 469 * 470 * @param field the field 471 * @param addInfoKey the additional info key 472 * @param editable the editable flag 473 */ 474 public void addAccountInfo(String field, String addInfoKey, String editable) { 475 476 addAccountInfo(new CmsAccountInfo(field, addInfoKey, editable)); 477 } 478 479 /** 480 * Adds a dialog handler instance to the list of configured dialog handlers.<p> 481 * 482 * @param clazz the instantiated dialog handler to add 483 */ 484 public void addDialogHandler(I_CmsDialogHandler clazz) { 485 486 m_dialogHandler.put(clazz.getDialogHandler(), clazz); 487 if (CmsLog.INIT.isInfoEnabled()) { 488 CmsLog.INIT.info( 489 Messages.get().getBundle().key( 490 Messages.INIT_ADD_DIALOG_HANDLER_2, 491 clazz.getDialogHandler(), 492 clazz.getClass().getName())); 493 } 494 } 495 496 /** 497 * Adds an editor CSS handler class to the list of handlers.<p> 498 * 499 * @param editorCssHandlerClassName full class name of the css handler class 500 */ 501 public void addEditorCssHandler(String editorCssHandlerClassName) { 502 503 try { 504 I_CmsEditorCssHandler editorCssHandler = (I_CmsEditorCssHandler)Class.forName( 505 editorCssHandlerClassName).newInstance(); 506 m_editorCssHandlers.add(editorCssHandler); 507 if (CmsLog.INIT.isInfoEnabled()) { 508 CmsLog.INIT.info( 509 Messages.get().getBundle().key(Messages.INIT_EDITOR_CSSHANDLER_CLASS_1, editorCssHandlerClassName)); 510 } 511 } catch (Exception e) { 512 LOG.error( 513 Messages.get().getBundle().key(Messages.LOG_INVALID_EDITOR_CSSHANDLER_1, editorCssHandlerClassName), 514 e); 515 } 516 } 517 518 /** 519 * Adds an editor CSS handler class at the first position of the list of handlers.<p> 520 * 521 * @param editorCssHandlerClassName full class name of the css handler class 522 */ 523 public void addEditorCssHandlerToHead(String editorCssHandlerClassName) { 524 525 try { 526 I_CmsEditorCssHandler editorCssHandler = (I_CmsEditorCssHandler)Class.forName( 527 editorCssHandlerClassName).newInstance(); 528 529 List<I_CmsEditorCssHandler> editorCssHandlers = new ArrayList<I_CmsEditorCssHandler>(); 530 editorCssHandlers.add(editorCssHandler); 531 editorCssHandlers.addAll(m_editorCssHandlers); 532 533 m_editorCssHandlers = editorCssHandlers; 534 535 if (CmsLog.INIT.isInfoEnabled()) { 536 CmsLog.INIT.info( 537 Messages.get().getBundle().key(Messages.INIT_EDITOR_CSSHANDLER_CLASS_1, editorCssHandlerClassName)); 538 } 539 } catch (Exception e) { 540 LOG.error( 541 Messages.get().getBundle().key(Messages.LOG_INVALID_EDITOR_CSSHANDLER_1, editorCssHandlerClassName), 542 e); 543 } 544 } 545 546 /** 547 * Adds an explorer type setting object to the list of type settings.<p> 548 * 549 * @param settings the explorer type settings 550 */ 551 public void addExplorerTypeSetting(CmsExplorerTypeSettings settings) { 552 553 m_explorerTypeSettingsFromXml.add(settings); 554 if (CmsLog.INIT.isInfoEnabled()) { 555 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_ADD_TYPE_SETTING_1, settings.getName())); 556 } 557 if (m_explorerTypeSettings != null) { 558 // reset the list of all explorer type settings, but not during startup 559 initExplorerTypeSettings(); 560 } 561 } 562 563 /** 564 * Adds the list of explorer type settings from the given module.<p> 565 * 566 * @param module the module witch contains the explorer type settings to add 567 */ 568 public void addExplorerTypeSettings(CmsModule module) { 569 570 List<CmsExplorerTypeSettings> explorerTypes = module.getExplorerTypes(); 571 if ((explorerTypes != null) && (explorerTypes.size() > 0)) { 572 Iterator<CmsExplorerTypeSettings> i = explorerTypes.iterator(); 573 while (i.hasNext()) { 574 CmsExplorerTypeSettings settings = i.next(); 575 if (m_explorerTypeSettingsFromModules.contains(settings)) { 576 m_explorerTypeSettingsFromModules.remove(settings); 577 } 578 m_explorerTypeSettingsFromModules.add(settings); 579 if (CmsLog.INIT.isInfoEnabled()) { 580 CmsLog.INIT.info( 581 Messages.get().getBundle().key(Messages.INIT_ADD_TYPE_SETTING_1, settings.getName())); 582 } 583 } 584 // reset the list of all explorer type settings 585 initExplorerTypeSettings(); 586 } 587 } 588 589 /** 590 * Adds newly created export point to the workplace configuration.<p> 591 * 592 * @param uri the export point uri 593 * @param destination the export point destination 594 */ 595 public void addExportPoint(String uri, String destination) { 596 597 CmsExportPoint point = new CmsExportPoint(uri, destination); 598 m_exportPoints.add(point); 599 if (CmsLog.INIT.isInfoEnabled() && (point.getDestinationPath() != null)) { 600 CmsLog.INIT.info( 601 Messages.get().getBundle().key( 602 Messages.INIT_ADD_EXPORT_POINT_2, 603 point.getUri(), 604 point.getDestinationPath())); 605 } 606 } 607 608 /** 609 * Adds a folder to the list of labeled folders.<p> 610 * 611 * @param uri the folder uri to add 612 */ 613 public void addLabeledFolder(String uri) { 614 615 m_labelSiteFolders.add(uri); 616 if (CmsLog.INIT.isInfoEnabled()) { 617 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_LABEL_LINKS_IN_FOLDER_1, uri)); 618 } 619 } 620 621 /** 622 * Adds a new folder to the list of localized workplace folders.<p> 623 * 624 * @param uri a new folder to add to the list of localized workplace folders 625 */ 626 public void addLocalizedFolder(String uri) { 627 628 m_localizedFolders.add(uri); 629 if (CmsLog.INIT.isInfoEnabled()) { 630 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_WORKPLACE_LOCALIZED_1, uri)); 631 } 632 } 633 634 /** 635 * Adds an initialized condition definition class that is triggered before opening the editor.<p> 636 * 637 * @param preEditorCondition the initialized condition definition class 638 */ 639 public void addPreEditorConditionDefinition(I_CmsPreEditorActionDefinition preEditorCondition) { 640 641 m_preEditorConditionDefinitions.add(preEditorCondition); 642 if (CmsLog.INIT.isInfoEnabled()) { 643 CmsLog.INIT.info( 644 Messages.get().getBundle().key( 645 Messages.INIT_EDITOR_PRE_ACTION_2, 646 preEditorCondition.getClass().getName(), 647 preEditorCondition.getResourceTypeName())); 648 } 649 } 650 651 /** 652 * Adds a condition definition class for a given resource type name that is triggered before opening the editor.<p> 653 * 654 * @param resourceTypeName the name of the resource type 655 * @param preEditorConditionDefinitionClassName full class name of the condition definition class 656 */ 657 public void addPreEditorConditionDefinition(String resourceTypeName, String preEditorConditionDefinitionClassName) { 658 659 try { 660 I_CmsPreEditorActionDefinition preEditorCondition = (I_CmsPreEditorActionDefinition)Class.forName( 661 preEditorConditionDefinitionClassName).newInstance(); 662 preEditorCondition.setResourceTypeName(resourceTypeName); 663 m_preEditorConditionDefinitions.add(preEditorCondition); 664 if (CmsLog.INIT.isInfoEnabled()) { 665 CmsLog.INIT.info( 666 Messages.get().getBundle().key( 667 Messages.INIT_EDITOR_PRE_ACTION_2, 668 preEditorConditionDefinitionClassName, 669 resourceTypeName)); 670 } 671 } catch (Exception e) { 672 LOG.error( 673 Messages.get().getBundle().key( 674 Messages.LOG_INVALID_EDITOR_PRE_ACTION_1, 675 preEditorConditionDefinitionClassName), 676 e); 677 } 678 } 679 680 /** 681 * Adds a pattern to be excluded in VFS synchronization.<p> 682 * 683 * @param pattern a java regex to applied on the file name 684 */ 685 public void addSynchronizeExcludePattern(String pattern) { 686 687 try { 688 m_synchronizeExcludePatterns.add(Pattern.compile(pattern)); 689 } catch (PatternSyntaxException e) { 690 LOG.error(Messages.get().getBundle().key(Messages.LOG_INVALID_SYNCHRONIZE_EXCLUDE_PATTERN_1, pattern), e); 691 } 692 } 693 694 /** 695 * Returns if the autolock resources feature is enabled.<p> 696 * 697 * @return true if the autolock resources feature is enabled, otherwise false 698 */ 699 public boolean autoLockResources() { 700 701 return m_autoLockResources; 702 } 703 704 /** 705 * Checks if the user in the given context has permissions for uploading. 706 * 707 * @param cms a CMS context 708 * @throws CmsSecurityException if the user doesn't have permission 709 */ 710 public void checkAdeGalleryUpload(CmsObject cms) throws CmsSecurityException { 711 712 OpenCms.getRoleManager().checkRole(cms, getUploadRole()); 713 } 714 715 /** 716 * Checks whether the workplace is accessed through the workplace server, and sends an error message otherwise.<p> 717 * 718 * @param request the request to check 719 * @param cms the CmsObject to use 720 */ 721 public void checkWorkplaceRequest(HttpServletRequest request, CmsObject cms) { 722 723 try { 724 if ((OpenCms.getSiteManager().getSites().size() > 1) 725 && !OpenCms.getSiteManager().isWorkplaceRequest(request)) { 726 // this is a multi site-configuration, but not a request to the configured Workplace site 727 728 CmsUser user = cms.getRequestContext().getCurrentUser(); 729 // to limit the number of times broadcast is called for a user, we use an expiring cache 730 // with the user name as key 731 if (null == m_workplaceServerUserChecks.getIfPresent(user.getName())) { 732 m_workplaceServerUserChecks.put(user.getName(), ""); 733 OpenCms.getSessionManager().sendBroadcast( 734 null, 735 Messages.get().getBundle(getWorkplaceLocale(cms)).key( 736 Messages.ERR_WORKPLACE_SERVER_CHECK_FAILED_0), 737 user, 738 ContentMode.plain); 739 740 } 741 742 } 743 } catch (Exception e) { 744 LOG.error(e.getLocalizedMessage(), e); 745 } 746 } 747 748 /** 749 * Implements the event listener of this class.<p> 750 * 751 * @see org.opencms.main.I_CmsEventListener#cmsEvent(org.opencms.main.CmsEvent) 752 */ 753 public void cmsEvent(CmsEvent event) { 754 755 switch (event.getType()) { 756 case I_CmsEventListener.EVENT_CLEAR_CACHES: 757 flushMessageCache(); 758 m_editorDisplayOptions.clearCache(); 759 if (LOG.isDebugEnabled()) { 760 LOG.debug(Messages.get().getBundle().key(Messages.LOG_EVENT_CLEAR_CACHES_0)); 761 } 762 break; 763 default: // no operation 764 } 765 } 766 767 /** 768 * Creates a temporary file which is needed while working in an editor with preview option.<p> 769 * 770 * <i>Note</i>: This method is synchronized to avoid rare issues that might be caused by 771 * double requests fired by some browser/OS combinations.<p> 772 * 773 * @param cms the cms context 774 * @param resourceName the name of the resource to copy 775 * @param currentProjectId the id of the project to work with 776 * 777 * @return the file name of the temporary file 778 * 779 * @throws CmsException if something goes wrong 780 */ 781 public synchronized String createTempFile(CmsObject cms, String resourceName, CmsUUID currentProjectId) 782 throws CmsException { 783 784 // check that the current user has write permissions 785 if (!cms.hasPermissions(cms.readResource(resourceName, CmsResourceFilter.ALL), CmsPermissionSet.ACCESS_WRITE)) { 786 throw new CmsPermissionViolationException( 787 org.opencms.db.Messages.get().container(org.opencms.db.Messages.ERR_PERM_DENIED_2, resourceName, "w")); 788 } 789 790 // initialize admin cms context 791 CmsObject adminCms = getAdminCms(cms); 792 793 // generate the filename of the temporary file 794 String temporaryFilename = CmsWorkplace.getTemporaryFileName(resourceName); 795 796 // check if the temporary file is already present 797 if (adminCms.existsResource(temporaryFilename, CmsResourceFilter.ALL)) { 798 // delete old temporary file 799 if (!cms.getLock(temporaryFilename).isUnlocked()) { 800 // steal lock 801 cms.changeLock(temporaryFilename); 802 } else { 803 // lock resource to current user 804 cms.lockResource(temporaryFilename); 805 } 806 cms.deleteResource(temporaryFilename, CmsResource.DELETE_PRESERVE_SIBLINGS); 807 } 808 809 try { 810 // switch to the temporary file project 811 adminCms.getRequestContext().setCurrentProject(cms.readProject(getTempFileProjectId())); 812 // copy the file to edit to a temporary file 813 adminCms.copyResource(resourceName, temporaryFilename, CmsResource.COPY_AS_NEW); 814 } finally { 815 // switch back to current project 816 adminCms.getRequestContext().setCurrentProject(cms.readProject(currentProjectId)); 817 } 818 819 try { 820 // switch to the temporary file project 821 cms.getRequestContext().setCurrentProject( 822 cms.readProject(OpenCms.getWorkplaceManager().getTempFileProjectId())); 823 // lock the temporary file 824 cms.changeLock(temporaryFilename); 825 // touch the temporary file 826 cms.setDateLastModified(temporaryFilename, System.currentTimeMillis(), false); 827 // set the temporary file flag 828 CmsResource tempFile = cms.readResource(temporaryFilename, CmsResourceFilter.ALL); 829 int flags = tempFile.getFlags(); 830 if ((flags & CmsResource.FLAG_TEMPFILE) == 0) { 831 flags += CmsResource.FLAG_TEMPFILE; 832 } 833 cms.chflags(temporaryFilename, flags); 834 // remove eventual release & expiration date from temporary file to make preview in editor work 835 cms.setDateReleased(temporaryFilename, CmsResource.DATE_RELEASED_DEFAULT, false); 836 cms.setDateExpired(temporaryFilename, CmsResource.DATE_EXPIRED_DEFAULT, false); 837 // remove visibility permissions for everybody on temporary file if possible 838 if (cms.hasPermissions(tempFile, CmsPermissionSet.ACCESS_CONTROL)) { 839 cms.chacc( 840 temporaryFilename, 841 I_CmsPrincipal.PRINCIPAL_GROUP, 842 OpenCms.getDefaultUsers().getGroupUsers(), 843 "-v"); 844 } 845 } finally { 846 // switch back to current project 847 cms.getRequestContext().setCurrentProject(cms.readProject(currentProjectId)); 848 } 849 850 return temporaryFilename; 851 } 852 853 /** 854 * Flushes the cached workplace messages.<p> 855 */ 856 public void flushMessageCache() { 857 858 // clear the cached message objects 859 m_messages = new HashMap<Locale, CmsWorkplaceMessages>(); 860 if (LOG.isDebugEnabled()) { 861 try { 862 throw new RuntimeException("Tracing exception"); 863 } catch (Exception e) { 864 LOG.info("Tracing call to CmsWorkplaceManager.flushMessageCache method.", e); 865 } 866 } 867 } 868 869 /** 870 * Gets the value of the acacia-unlock configuration option (null if not set explicitly).<p> 871 * 872 * @return the value of the acacia-unlock configuration option 873 */ 874 public String getAcaciaUnlock() { 875 876 return m_acaciaUnlock; 877 } 878 879 /** 880 * Returns the account infos.<p> 881 * 882 * @return the account infos 883 */ 884 public List<CmsAccountInfo> getAccountInfos() { 885 886 if (m_accountInfos == null) { 887 return Collections.unmodifiableList(Arrays.asList(DEFAULT_ACCOUNT_INFOS)); 888 } else { 889 return Collections.unmodifiableList(m_accountInfos); 890 } 891 } 892 893 /** 894 * Gets the additional log folder configuration.<p> 895 * 896 * @return the additional log folder configuration 897 */ 898 public CmsAdditionalLogFolderConfig getAdditionalLogFolderConfiguration() { 899 900 return m_logFolderConfig; 901 } 902 903 /** 904 * Returns the name of the local category folder(s).<p> 905 * 906 * @return the name of the local category folder(s) 907 */ 908 public String getCategoryFolder() { 909 910 return m_categoryFolder; 911 } 912 913 /** 914 * Returns the configured account infos.<p> 915 * 916 * @return the configured account infos 917 */ 918 public List<CmsAccountInfo> getConfiguredAccountInfos() { 919 920 return m_accountInfos; 921 } 922 923 /** 924 * Gets the access object of the type settings.<p> 925 * 926 * @return access object of the type settings 927 */ 928 public CmsExplorerTypeAccess getDefaultAccess() { 929 930 return m_defaultAccess; 931 } 932 933 /** 934 * Returns the Workplace default locale.<p> 935 * 936 * @return the Workplace default locale 937 */ 938 public Locale getDefaultLocale() { 939 940 return m_defaultLocale; 941 } 942 943 /** 944 * Gets the default name pattern for the given type.<p> 945 * 946 * @param type the type name 947 * @return the default name pattern for the type 948 */ 949 public String getDefaultNamePattern(String type) { 950 951 CmsExplorerTypeSettings settings = getExplorerTypeSetting(type); 952 if ((settings != null) && (settings.getNamePattern() != null)) { 953 return settings.getNamePattern(); 954 } 955 if (type.equals("sitemap_config") || type.equals("module_config")) { 956 return ".config%(number)"; 957 } 958 if (type.equals("content_folder")) { 959 return ".content%(number)"; 960 } 961 return "new_" + type + "%(number)"; 962 } 963 964 /** 965 * Returns the Workplace default user settings.<p> 966 * 967 * @return the Workplace default user settings 968 */ 969 public CmsDefaultUserSettings getDefaultUserSettings() { 970 971 return m_defaultUserSettings; 972 } 973 974 /** 975 * Returns all instantiated dialog handlers for the workplace.<p> 976 * 977 * @return all instantiated dialog handlers for the workplace 978 */ 979 public Map<String, I_CmsDialogHandler> getDialogHandler() { 980 981 return m_dialogHandler; 982 } 983 984 /** 985 * Returns the instantiated dialog handler class for the key or null, if there is no mapping for the key.<p> 986 * 987 * @param key the key whose associated value is to be returned 988 * 989 * @return the instantiated dialog handler class for the key 990 */ 991 public I_CmsDialogHandler getDialogHandler(String key) { 992 993 return m_dialogHandler.get(key); 994 } 995 996 /** 997 * Returns a new instance of the configured direct edit provider.<p> 998 * 999 * @return a new instance of the configured direct edit provider 1000 */ 1001 public I_CmsDirectEditProvider getDirectEditProvider() { 1002 1003 return m_directEditProvider.newInstance(); 1004 } 1005 1006 /** 1007 * Returns the instantiated editor action handler class.<p> 1008 * 1009 * @return the instantiated editor action handler class 1010 */ 1011 public I_CmsEditorActionHandler getEditorActionHandler() { 1012 1013 return m_editorAction; 1014 } 1015 1016 /** 1017 * Returns the instantiated editor CSS handler classes.<p> 1018 * 1019 * @return the instantiated editor CSS handler classes 1020 */ 1021 public List<I_CmsEditorCssHandler> getEditorCssHandlers() { 1022 1023 return m_editorCssHandlers; 1024 } 1025 1026 /** 1027 * Returns the instantiated editor display option class.<p> 1028 * 1029 * @return the instantiated editor display option class 1030 */ 1031 public CmsEditorDisplayOptions getEditorDisplayOptions() { 1032 1033 return m_editorDisplayOptions; 1034 } 1035 1036 /** 1037 * Returns the instantiated editor handler class.<p> 1038 * 1039 * @return the instantiated editor handler class 1040 */ 1041 public I_CmsEditorHandler getEditorHandler() { 1042 1043 return m_editorHandler; 1044 } 1045 1046 /** 1047 * Gets the maximum number of locale buttons to display in the content editor. 1048 */ 1049 public int getEditorMaxLocaleButtons() { 1050 1051 return m_maxLocaleButtons; 1052 } 1053 1054 /** 1055 * Returns the element delete mode.<p> 1056 * 1057 * @return the element delete mode 1058 */ 1059 public ElementDeleteMode getElementDeleteMode() { 1060 1061 return m_elementDeleteMode; 1062 } 1063 1064 /** 1065 * Returns the configured workplace encoding.<p> 1066 * 1067 * @return the configured workplace encoding 1068 */ 1069 public String getEncoding() { 1070 1071 return m_encoding; 1072 } 1073 1074 /** 1075 * Returns the explorer type settings for the specified resource type.<p> 1076 * 1077 * @param type the resource type for which the settings are required 1078 * 1079 * @return the explorer type settings for the specified resource type 1080 */ 1081 public CmsExplorerTypeSettings getExplorerTypeSetting(String type) { 1082 1083 return m_explorerTypeSettingsMap.get(type); 1084 } 1085 1086 /** 1087 * Returns the list of explorer type settings.<p> 1088 * 1089 * These settings provide information for the new resource dialog and the context menu appearance.<p> 1090 * 1091 * @return the list of explorer type settings 1092 */ 1093 public List<CmsExplorerTypeSettings> getExplorerTypeSettings() { 1094 1095 if (m_explorerTypeSettings == null) { 1096 // initialize all explorer type settings if not already done 1097 initExplorerTypeSettings(); 1098 } 1099 1100 return m_explorerTypeSettings; 1101 } 1102 1103 /** 1104 * Gets the explorer types for the given view name.<p> 1105 * 1106 * @param viewName the view name 1107 * 1108 * @return the explorer names for the given view names 1109 */ 1110 public List<CmsExplorerTypeSettings> getExplorerTypesForView(String viewName) { 1111 1112 List<CmsExplorerTypeSettings> result = Lists.newArrayList(); 1113 for (CmsExplorerTypeSettings explorerType : getExplorerTypeSettings()) { 1114 1115 String currentViewName = explorerType.getElementView(); 1116 if (currentViewName == null) { 1117 currentViewName = getDefaultView(explorerType.getName()); 1118 } 1119 if ((currentViewName != null) && currentViewName.equals(viewName)) { 1120 if (OpenCms.getResourceManager().hasResourceType(explorerType.getName())) { 1121 result.add(explorerType); 1122 } 1123 } else if (CmsResourceTypeFolder.getStaticTypeName().equals(explorerType.getName()) 1124 && "view_folders|view_basic".contains(viewName)) { 1125 result.add(explorerType); 1126 } 1127 1128 } 1129 return result; 1130 } 1131 1132 /** 1133 * Gets the element views generated from explorer types.<p> 1134 * 1135 * @return the map of element views from the explorer types 1136 */ 1137 public Map<CmsUUID, CmsElementView> getExplorerTypeViews() { 1138 1139 return Collections.unmodifiableMap(m_explorerTypeViews); 1140 1141 } 1142 1143 /** 1144 * Returns the set of configured export points for the workplace.<p> 1145 * 1146 * @return the set of configured export points for the workplace 1147 */ 1148 public Set<CmsExportPoint> getExportPoints() { 1149 1150 return m_exportPoints; 1151 } 1152 1153 /** 1154 * Returns the value (in bytes) for the maximum file upload size of the current user.<p> 1155 * 1156 * @param cms the initialized CmsObject 1157 * 1158 * @return the value (in bytes) for the maximum file upload size 1159 */ 1160 public long getFileBytesMaxUploadSize(CmsObject cms) { 1161 1162 int maxFileSize = getFileMaxUploadSize(); 1163 long maxFileSizeBytes = maxFileSize * 1024; 1164 // check if current user belongs to Admin group, if so no file upload limit 1165 if ((maxFileSize <= 0) || OpenCms.getRoleManager().hasRole(cms, CmsRole.VFS_MANAGER)) { 1166 maxFileSizeBytes = -1; 1167 } 1168 return maxFileSizeBytes; 1169 } 1170 1171 /** 1172 * Returns the value (in kb) for the maximum file upload size.<p> 1173 * 1174 * @return the value (in kb) for the maximum file upload size 1175 */ 1176 public int getFileMaxUploadSize() { 1177 1178 return m_fileMaxUploadSize; 1179 } 1180 1181 /** 1182 * Returns the system-wide file view settings for the workplace.<p> 1183 * 1184 * Note that this instance may not modified (invocation of setters) directly or a 1185 * <code>{@link org.opencms.main.CmsRuntimeException}</code> will be thrown.<p> 1186 * 1187 * It has to be cloned first and then may be written back to the workplace settings using 1188 * method {@link #setFileViewSettings(CmsObject, org.opencms.util.CmsRfsFileViewer)}.<p> 1189 * 1190 * @return the system-wide file view settings for the workplace 1191 */ 1192 public CmsRfsFileViewer getFileViewSettings() { 1193 1194 return m_fileViewSettings; 1195 } 1196 1197 /** 1198 * Returns a collection of all available galleries.<p> 1199 * 1200 * The Map has the gallery type name as key and an instance of the 1201 * gallery class (not completely initialized) as value.<p> 1202 * 1203 * @return a collection of all available galleries 1204 */ 1205 public Map<String, A_CmsAjaxGallery> getGalleries() { 1206 1207 return m_galleries; 1208 } 1209 1210 /** 1211 * Returns the gallery default scope.<p> 1212 * 1213 * @return the gallery default scope 1214 */ 1215 public CmsGallerySearchScope getGalleryDefaultScope() { 1216 1217 CmsGallerySearchScope result = CmsGallerySearchScope.siteShared; 1218 if (m_galleryDefaultScope != null) { 1219 try { 1220 result = CmsGallerySearchScope.valueOf(m_galleryDefaultScope); 1221 } catch (Throwable t) { 1222 // ignore 1223 } 1224 } 1225 return result; 1226 } 1227 1228 /** 1229 * Gets the configured gallery default scope as a string.<p> 1230 * 1231 * @return the gallery default scope as a string 1232 */ 1233 public String getGalleryDefaultScopeString() { 1234 1235 return m_galleryDefaultScope; 1236 } 1237 1238 /** 1239 * Returns the object used for translating group names.<p> 1240 * 1241 * @return the group name translator 1242 */ 1243 public I_CmsGroupNameTranslation getGroupNameTranslation() { 1244 1245 if (m_groupNameTranslation != null) { 1246 return m_groupNameTranslation; 1247 } 1248 if (m_groupTranslationClass != null) { 1249 try { 1250 m_groupNameTranslation = (I_CmsGroupNameTranslation)Class.forName( 1251 m_groupTranslationClass).newInstance(); 1252 return m_groupNameTranslation; 1253 } catch (ClassNotFoundException e) { 1254 LOG.error(e.getLocalizedMessage(), e); 1255 } catch (IllegalAccessException e) { 1256 LOG.error(e.getLocalizedMessage(), e); 1257 } catch (InstantiationException e) { 1258 LOG.error(e.getLocalizedMessage(), e); 1259 } catch (ClassCastException e) { 1260 LOG.error(e.getLocalizedMessage(), e); 1261 } 1262 m_groupNameTranslation = getDefaultGroupNameTranslation(); 1263 return m_groupNameTranslation; 1264 } else { 1265 m_groupNameTranslation = getDefaultGroupNameTranslation(); 1266 return m_groupNameTranslation; 1267 } 1268 } 1269 1270 /** 1271 * Returns the configured class name for translating group names.<p> 1272 * 1273 * @return the group translation class name 1274 */ 1275 public String getGroupTranslationClass() { 1276 1277 return m_groupTranslationClass; 1278 } 1279 1280 /** 1281 * @see org.opencms.i18n.I_CmsLocaleHandler#getI18nInfo(javax.servlet.http.HttpServletRequest, org.opencms.file.CmsUser, org.opencms.file.CmsProject, java.lang.String) 1282 */ 1283 public CmsI18nInfo getI18nInfo(HttpServletRequest req, CmsUser user, CmsProject project, String resource) { 1284 1285 Locale locale = null; 1286 // try to read locale from session 1287 if (req != null) { 1288 // set the request character encoding 1289 try { 1290 req.setCharacterEncoding(m_encoding); 1291 } catch (UnsupportedEncodingException e) { 1292 // should not ever really happen 1293 LOG.error(Messages.get().getBundle().key(Messages.LOG_UNSUPPORTED_ENCODING_SET_1, m_encoding), e); 1294 } 1295 // read workplace settings 1296 HttpSession session = req.getSession(false); 1297 if (session != null) { 1298 CmsWorkplaceSettings settings = (CmsWorkplaceSettings)session.getAttribute( 1299 CmsWorkplaceManager.SESSION_WORKPLACE_SETTINGS); 1300 if (settings != null) { 1301 locale = settings.getUserSettings().getLocale(); 1302 } 1303 } 1304 } 1305 1306 if (locale == null) { 1307 // no session available, try to read the locale form the user additional info 1308 if (!user.isGuestUser()) { 1309 // check user settings only for "real" users 1310 CmsUserSettings settings = new CmsUserSettings(user); 1311 locale = settings.getLocale(); 1312 1313 } 1314 if (req != null) { 1315 List<Locale> acceptedLocales = (new CmsAcceptLanguageHeaderParser( 1316 req, 1317 getDefaultLocale())).getAcceptedLocales(); 1318 if ((locale != null) && (!acceptedLocales.contains(locale))) { 1319 acceptedLocales.add(0, locale); 1320 } 1321 locale = OpenCms.getLocaleManager().getFirstMatchingLocale(acceptedLocales, m_locales); 1322 } 1323 1324 // if no locale was found, use the default 1325 if (locale == null) { 1326 locale = getDefaultLocale(); 1327 } 1328 } 1329 1330 return new CmsI18nInfo(locale, m_encoding); 1331 } 1332 1333 /** 1334 * Returns a list of site folders which generate labeled links.<p> 1335 * 1336 * @return a list of site folders which generate labeled links 1337 */ 1338 public List<String> getLabelSiteFolders() { 1339 1340 return m_labelSiteFolders; 1341 } 1342 1343 /** 1344 * Returns the list of available workplace locales, sorted ascending.<p> 1345 * 1346 * Please note: Be careful not to modify the returned Set as it is not a clone.<p> 1347 * 1348 * @return the set of available workplace locales 1349 */ 1350 public List<Locale> getLocales() { 1351 1352 return m_locales; 1353 } 1354 1355 /** 1356 * Returns the configured list of localized workplace folders.<p> 1357 * 1358 * @return the configured list of localized workplace folders 1359 */ 1360 public List<String> getLocalizedFolders() { 1361 1362 return m_localizedFolders; 1363 } 1364 1365 /** 1366 * Returns the {@link CmsWorkplaceMessages} for the given locale.<p> 1367 * 1368 * The workplace messages are a collection of resource bundles, containing the messages 1369 * for all OpenCms core bundles and of all initialized modules.<p> 1370 * 1371 * Please note that the message objects are cached internally. 1372 * The returned message object should therefore never be modified directly in any way.<p> 1373 * 1374 * @param locale the locale to get the messages for 1375 * 1376 * @return the {@link CmsWorkplaceMessages} for the given locale 1377 */ 1378 public CmsWorkplaceMessages getMessages(Locale locale) { 1379 1380 CmsWorkplaceMessages result = m_messages.get(locale); 1381 if (result != null) { 1382 // messages have already been read 1383 return result; 1384 } 1385 1386 // messages have not been read so far 1387 synchronized (this) { 1388 // check again 1389 result = m_messages.get(locale); 1390 if (result == null) { 1391 result = new CmsWorkplaceMessages(locale); 1392 m_messages.put(locale, result); 1393 } 1394 } 1395 return result; 1396 } 1397 1398 /** 1399 * Returns the post upload handler.<p> 1400 * 1401 * @return the post upload handler 1402 */ 1403 public I_CmsPostUploadDialogHandler getPostUploadHandler() { 1404 1405 return m_postUploadHandler; 1406 } 1407 1408 /** 1409 * Returns the condition definition for the given resource type that is triggered before opening the editor.<p> 1410 * 1411 * @param resourceType the resource type 1412 * 1413 * @return the condition definition for the given resource type class name or null if none is found 1414 */ 1415 public I_CmsPreEditorActionDefinition getPreEditorConditionDefinition(I_CmsResourceType resourceType) { 1416 1417 Iterator<I_CmsPreEditorActionDefinition> i = m_preEditorConditionDefinitions.iterator(); 1418 I_CmsPreEditorActionDefinition result = null; 1419 int matchResult = -1; 1420 while (i.hasNext()) { 1421 I_CmsPreEditorActionDefinition currentDefinition = i.next(); 1422 if (resourceType.getClass().isInstance(currentDefinition.getResourceType())) { 1423 // now determine the match count... 1424 int matchDistance = 0; 1425 Class<?> superClass = resourceType.getClass(); 1426 while (true) { 1427 // check if a super class is present 1428 if (superClass == null) { 1429 break; 1430 } 1431 if (superClass.getName().equals(currentDefinition.getResourceType().getClass().getName())) { 1432 break; 1433 } 1434 matchDistance += 1; 1435 superClass = superClass.getSuperclass(); 1436 } 1437 if (matchResult != -1) { 1438 if (matchDistance < matchResult) { 1439 matchResult = matchDistance; 1440 result = currentDefinition; 1441 } 1442 } else { 1443 matchResult = matchDistance; 1444 result = currentDefinition; 1445 } 1446 } 1447 } 1448 return result; 1449 } 1450 1451 /** 1452 * Returns the condition definitions for the different resource 1453 * types which are triggered before opening the editor.<p> 1454 * 1455 * @return the condition definitions 1456 */ 1457 public List<I_CmsPreEditorActionDefinition> getPreEditorConditionDefinitions() { 1458 1459 return m_preEditorConditionDefinitions; 1460 } 1461 1462 /** 1463 * Returns the repository folder handler.<p> 1464 * 1465 * @return the repository folder handler 1466 */ 1467 public I_CmsRepositoryFolderHandler getRepositoryFolderHandler() { 1468 1469 if (m_repositoryFolderHandler == null) { 1470 // handler has not been configured, use the default one 1471 m_repositoryFolderHandler = new CmsRepositoryFolderHandler(); 1472 } 1473 return m_repositoryFolderHandler; 1474 } 1475 1476 /** 1477 * Gets the name of the role necessary for editing the sitemap configuration. 1478 * 1479 * @return the name of the role necessary for editing the sitemap configuration 1480 */ 1481 public String getSitemapConfigEditRole() { 1482 1483 return m_sitemapConfigEditRole; 1484 } 1485 1486 /** 1487 * Returns Regex patterns that should be excluded from synchronization.<p> 1488 * 1489 * @return the exclude patterns 1490 */ 1491 public ArrayList<Pattern> getSynchronizeExcludePatterns() { 1492 1493 return m_synchronizeExcludePatterns; 1494 } 1495 1496 /** 1497 * Returns the id of the temporary file project required by the editors.<p> 1498 * 1499 * @return the id of the temporary file project required by the editors 1500 */ 1501 public CmsUUID getTempFileProjectId() { 1502 1503 if (m_tempFileProject != null) { 1504 return m_tempFileProject.getUuid(); 1505 } else { 1506 return null; 1507 } 1508 } 1509 1510 /** 1511 * Returns the tool manager.<p> 1512 * 1513 * @return the tool manager 1514 */ 1515 public CmsToolManager getToolManager() { 1516 1517 if (m_toolManager == null) { 1518 m_toolManager = new CmsToolManager(); 1519 } 1520 return m_toolManager; 1521 } 1522 1523 /** 1524 * Gets the upload hook URI which should be opened for an upload to a given folder.<p> 1525 * This method will return null if no upload hook should be used for the given upload folder.<p> 1526 * 1527 * The API for this upload hook is as follows: 1528 * 1529 * The upload hook will be called with the following parameters: 1530 * 1531 * resources (required): a comma separated list of the structure ids of the uploaded resources 1532 * if this is omitted 1533 * closelink (optional): a link which should be opened once the upload hook has finished whatever 1534 * it is doing 1535 * 1536 * @param cms the current CMS context 1537 * @param uploadFolder the folder for which the upload hook should be found 1538 * 1539 * @return the URI of the upload hook or null 1540 */ 1541 public String getUploadHook(CmsObject cms, String uploadFolder) { 1542 1543 if (m_postUploadHandler != null) { 1544 return m_postUploadHandler.getUploadHook(cms, uploadFolder); 1545 } 1546 I_CmsDialogHandler handler = getDialogHandler(CmsDialogSelector.DIALOG_PROPERTY); 1547 if ((handler != null) && (handler instanceof I_CmsPostUploadDialogHandler)) { 1548 return ((I_CmsPostUploadDialogHandler)handler).getUploadHook(cms, uploadFolder); 1549 } else { 1550 return null; 1551 } 1552 } 1553 1554 /** 1555 * Gets the upload restriction. 1556 * 1557 * @return the upload restriction 1558 */ 1559 public I_CmsUploadRestriction getUploadRestriction() { 1560 1561 return m_uploadRestriction; 1562 } 1563 1564 /** 1565 * Returns the user additional information configuration Manager.<p> 1566 * 1567 * @return the user additional information configuration manager 1568 */ 1569 public CmsWorkplaceUserInfoManager getUserInfoManager() { 1570 1571 return m_userInfoManager; 1572 } 1573 1574 /** 1575 * Returns the user list mode as a string.<p> 1576 * 1577 * @return the user list mode as a string 1578 */ 1579 public String getUserListModeString() { 1580 1581 return m_userListMode; 1582 } 1583 1584 /** 1585 * Returns the map of configured workplace views.<p> 1586 * 1587 * @return the map of configured workplace views 1588 */ 1589 public List<CmsWorkplaceView> getViews() { 1590 1591 return m_views; 1592 } 1593 1594 /** 1595 * Gets the configured virus scanner (may be null). 1596 * 1597 * @return the configured virus scanner 1598 */ 1599 public I_CmsVirusScanner getVirusScanner() { 1600 1601 return m_virusScanner; 1602 } 1603 1604 /** 1605 * Returns the instantiated workplace editor manager class.<p> 1606 * 1607 * @return the instantiated workplace editor manager class 1608 */ 1609 public CmsWorkplaceEditorManager getWorkplaceEditorManager() { 1610 1611 return m_editorManager; 1612 } 1613 1614 /** 1615 * Returns the list of explorer type settings configured in the opencms-workplace.xml file.<p> 1616 * 1617 * @return the list of explorer type settings 1618 */ 1619 public List<CmsExplorerTypeSettings> getWorkplaceExplorerTypeSettings() { 1620 1621 return Collections.unmodifiableList(m_explorerTypeSettingsFromXml); 1622 } 1623 1624 /** 1625 * Returns the workplace locale from the current user's settings.<p> 1626 * 1627 * @param cms the current cms object 1628 * 1629 * @return the workplace locale 1630 */ 1631 public Locale getWorkplaceLocale(CmsObject cms) { 1632 1633 return getWorkplaceLocale(cms.getRequestContext()); 1634 } 1635 1636 /** 1637 * Gets the workplace locale for the given request context.<p> 1638 * 1639 * @param requestContext the request context 1640 * 1641 * @return the workplace locale for the request context 1642 */ 1643 public Locale getWorkplaceLocale(CmsRequestContext requestContext) { 1644 1645 Locale wpLocale = new CmsUserSettings(requestContext.getCurrentUser()).getLocale(); 1646 if (wpLocale == null) { 1647 // fall back 1648 wpLocale = getDefaultLocale(); 1649 if (wpLocale == null) { 1650 // fall back 1651 wpLocale = requestContext.getLocale(); 1652 } 1653 } 1654 return wpLocale; 1655 1656 } 1657 1658 /** 1659 * Returns the workplace locale for the user. 1660 * @param user the user to get the workplace locale for. 1661 * @return the workplace locale for the user. 1662 */ 1663 public Locale getWorkplaceLocale(CmsUser user) { 1664 1665 Locale wpLocale = new CmsUserSettings(user).getLocale(); 1666 if (wpLocale == null) { 1667 // fall back 1668 wpLocale = OpenCms.getWorkplaceManager().getDefaultLocale(); 1669 if (wpLocale == null) { 1670 // fall back 1671 wpLocale = CmsLocaleManager.MASTER_LOCALE; 1672 } 1673 } 1674 return wpLocale; 1675 1676 } 1677 1678 /** 1679 * @see org.opencms.i18n.I_CmsLocaleHandler#initHandler(org.opencms.file.CmsObject) 1680 */ 1681 public void initHandler(CmsObject cms) { 1682 1683 // initialize the workplace locale set 1684 m_locales = initWorkplaceLocales(cms); 1685 } 1686 1687 /** 1688 * Initializes the workplace manager with the OpenCms system configuration.<p> 1689 * 1690 * @param cms an OpenCms context object that must have been initialized with "Admin" permissions 1691 * 1692 * @throws CmsRoleViolationException if the provided OpenCms user context does 1693 * not have <code>{@link CmsRole#WORKPLACE_MANAGER}</code> role permissions 1694 * @throws CmsException if something goes wrong 1695 */ 1696 public synchronized void initialize(CmsObject cms) throws CmsException, CmsRoleViolationException { 1697 1698 try { 1699 // ensure that the current user has permissions to initialize the workplace 1700 OpenCms.getRoleManager().checkRole(cms, CmsRole.WORKPLACE_MANAGER); 1701 1702 // set the workplace encoding 1703 try { 1704 // workplace encoding is set on the workplace parent folder /system/workplace/ 1705 CmsResource wpFolderRes = cms.readResource(CmsWorkplace.VFS_PATH_WORKPLACE); 1706 m_encoding = CmsLocaleManager.getResourceEncoding(cms, wpFolderRes); 1707 } catch (CmsVfsResourceNotFoundException e) { 1708 // workplace parent folder could not be read - use configured default encoding 1709 m_encoding = OpenCms.getSystemInfo().getDefaultEncoding(); 1710 } 1711 1712 // configure direct edit provider with default if not available 1713 if (m_directEditProvider == null) { 1714 m_directEditProvider = new CmsDirectEditDefaultProvider(); 1715 } 1716 1717 // throw away all currently configured module explorer types 1718 m_explorerTypeSettingsFromModules.clear(); 1719 // now add the additional explorer types found in the modules 1720 CmsModuleManager moduleManager = OpenCms.getModuleManager(); 1721 Iterator<String> moduleNameIterator = moduleManager.getModuleNames().iterator(); 1722 while (moduleNameIterator.hasNext()) { 1723 CmsModule module = moduleManager.getModule(moduleNameIterator.next()); 1724 if (module != null) { 1725 addExplorerTypeSettings(module); 1726 } 1727 } 1728 // initialize the explorer type settings 1729 initExplorerTypeSettings(); 1730 // initialize the workplace views 1731 initWorkplaceViews(cms); 1732 // initialize the workplace editor manager 1733 m_editorManager = new CmsWorkplaceEditorManager(cms); 1734 // initialize the locale handler 1735 initHandler(cms); 1736 1737 if (CmsLog.INIT.isInfoEnabled()) { 1738 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_VFS_ACCESS_INITIALIZED_0)); 1739 } 1740 try { 1741 // read the temporary file project 1742 m_tempFileProject = cms.readProject(I_CmsProjectDriver.TEMP_FILE_PROJECT_NAME); 1743 } catch (CmsException e) { 1744 // during initial setup of OpenCms the temp file project does not yet exist... 1745 LOG.error(Messages.get().getBundle().key(Messages.LOG_NO_TEMP_FILE_PROJECT_0)); 1746 } 1747 // create an instance of editor display options 1748 m_editorDisplayOptions = new CmsEditorDisplayOptions(); 1749 1750 // throw away all current gallery settings 1751 m_galleries.clear(); 1752 // read out the configured gallery classes 1753 Iterator<I_CmsResourceType> typesIterator = OpenCms.getResourceManager().getResourceTypes().iterator(); 1754 while (typesIterator.hasNext()) { 1755 I_CmsResourceType resourceType = typesIterator.next(); 1756 if (resourceType instanceof CmsResourceTypeFolderExtended) { 1757 // found a configured extended folder resource type 1758 CmsResourceTypeFolderExtended galleryType = (CmsResourceTypeFolderExtended)resourceType; 1759 String folderClassName = galleryType.getFolderClassName(); 1760 if (CmsStringUtil.isNotEmpty(folderClassName)) { 1761 // only process this as a gallery if the folder name is not empty 1762 try { 1763 // check, if the folder class is a subclass of A_CmsGallery 1764 if (A_CmsAjaxGallery.class.isAssignableFrom(Class.forName(folderClassName))) { 1765 // create gallery class instance 1766 A_CmsAjaxGallery galleryInstance = (A_CmsAjaxGallery)Class.forName( 1767 folderClassName).newInstance(); 1768 // set gallery folder resource type 1769 galleryInstance.setResourceType(galleryType); 1770 // store the gallery class instance with the type name as lookup key 1771 m_galleries.put(galleryType.getTypeName(), galleryInstance); 1772 } 1773 } catch (ClassNotFoundException e) { 1774 LOG.error(e.getLocalizedMessage()); 1775 } catch (InstantiationException e) { 1776 LOG.error(e.getLocalizedMessage()); 1777 } catch (IllegalAccessException e) { 1778 LOG.error(e.getLocalizedMessage()); 1779 } 1780 } 1781 } 1782 } 1783 1784 getDefaultUserSettings().initPreferences(this); 1785 1786 // configures the tool manager 1787 getToolManager().configure(cms); 1788 1789 flushMessageCache(); 1790 getUploadRestriction().setAdminCmsObject(cms); 1791 1792 // register this object as event listener 1793 OpenCms.addCmsEventListener(this, new int[] {I_CmsEventListener.EVENT_CLEAR_CACHES}); 1794 } catch (CmsException e) { 1795 if (LOG.isErrorEnabled()) { 1796 LOG.error(e.getLocalizedMessage(), e); 1797 } 1798 throw new CmsException(Messages.get().container(Messages.ERR_INITIALIZE_WORKPLACE_0)); 1799 } 1800 m_adminCms = cms; 1801 } 1802 1803 /** 1804 * Returns true if gallery upload is disabled for the user in the given context. 1805 * 1806 * @param cms a CMS context 1807 * @return true if the upload is disabled 1808 */ 1809 public boolean isAdeGalleryUploadDisabled(CmsObject cms) { 1810 1811 return !OpenCms.getRoleManager().hasRole(cms, getUploadRole()); 1812 1813 } 1814 1815 /** 1816 * Checks if element authors have special permission to work in galleries (upload/replace). 1817 * @return true in the case above 1818 */ 1819 1820 public boolean isAllowElementAuthorToWorkInGalleries() { 1821 1822 return m_allowElementAuthorToWorkInGalleries; 1823 } 1824 1825 /** 1826 * Returns the default property editing mode on resources.<p> 1827 * 1828 * @return the default property editing mode on resources 1829 */ 1830 public boolean isDefaultPropertiesOnStructure() { 1831 1832 return m_defaultPropertiesOnStructure; 1833 } 1834 1835 /** 1836 * Returns a flag, indicating if the categories should be displayed separated by repository in the category selection dialog. 1837 * 1838 * @return a flag, indicating if the categories should be displayed separated by repository in the category selection dialog. 1839 */ 1840 public boolean isDisplayCategoriesByRepository() { 1841 1842 return m_displayCategoriesByRepository; 1843 } 1844 1845 /** 1846 * Returns a flag, indicating if the category selection dialog should have all entries completely collapsed when opened. 1847 * 1848 * @return a flag, indicating if the category selection dialog should have all entries completely collapsed when opened. 1849 */ 1850 public boolean isDisplayCategorySelectionCollapsed() { 1851 1852 return m_displayCategorySelectionCollapsed; 1853 } 1854 1855 /** 1856 * Returns if tabs in the advanced property dialog are enabled.<p> 1857 * 1858 * @return <code>true</code> if tabs should be enabled, otherwise <code>false</code> 1859 */ 1860 public boolean isEnableAdvancedPropertyTabs() { 1861 1862 return m_enableAdvancedPropertyTabs; 1863 } 1864 1865 /** 1866 * Checks if the 'Categories' column in the explorer should be enabled 1867 * 1868 * @return true if the categories column in the explorer should be enabled 1869 */ 1870 public boolean isExplorerCategoriesEnabled() { 1871 1872 return m_explorerCategoriesEnabled; 1873 } 1874 1875 /** 1876 * Checks if only 'leaf' categories should be shown in the explorer 'Categories' column, i.e. only those which don't have child categories assigned to the same resource. 1877 * 1878 * @return true if only 'leaf' categories should be shown 1879 */ 1880 public boolean isExplorerCategoriesLeavesOnly() { 1881 1882 return m_explorerCategoriesLeavesOnly; 1883 } 1884 1885 /** 1886 * Checks if categories should be shown with their path in the explorer 'Categories' column. 1887 * 1888 * @return true if categories should be shown with their path 1889 */ 1890 public boolean isExplorerCategoriesWithPath() { 1891 1892 return m_explorerCategoriesWithPath; 1893 } 1894 1895 /** 1896 * Returns true if "keep alive" mode is active. 1897 * 1898 * @return true if the session should be kept alive 1899 */ 1900 public boolean isKeepAlive() { 1901 1902 return isKeepAlive(true).booleanValue(); 1903 } 1904 1905 /** 1906 * Returns true if the session should be kept alive.<p> 1907 * 1908 * @param useDefault if true, the default value will be returned if the "keep alive" setting is not explicitly configured 1909 * 1910 * @return True if the "keep alive" mode is active 1911 */ 1912 public Boolean isKeepAlive(boolean useDefault) { 1913 1914 if (m_keepAlive != null) { 1915 return m_keepAlive; 1916 } 1917 if (useDefault) { 1918 return Boolean.TRUE; 1919 } else { 1920 return null; 1921 } 1922 1923 } 1924 1925 /** 1926 * Checks if the virus scanner is enabled. 1927 * 1928 * @return true if the virus scanner is enabled 1929 */ 1930 public boolean isVirusScannerEnabled() { 1931 1932 return m_virusScannerEnabled; 1933 } 1934 1935 /** 1936 * Returns if XML content is automatically corrected when opened with the editor.<p> 1937 * 1938 * @return <code>true</code> if XML content is automatically corrected when opened with the editor, otherwise <code>false</code> 1939 */ 1940 public boolean isXmlContentAutoCorrect() { 1941 1942 return m_xmlContentAutoCorrect; 1943 } 1944 1945 /** 1946 * Returns if lazy user lists are enabled.<p> 1947 * 1948 * @return <code>true</code> if lazy user lists are enabled 1949 */ 1950 public boolean lazyUserListsEnabled() { 1951 1952 return true; 1953 } 1954 1955 /** 1956 * Removes the list of explorer type settings from the given module.<p> 1957 * 1958 * @param module the module witch contains the explorer type settings to remove 1959 */ 1960 public void removeExplorerTypeSettings(CmsModule module) { 1961 1962 List<CmsExplorerTypeSettings> explorerTypes = module.getExplorerTypes(); 1963 if ((explorerTypes != null) && (explorerTypes.size() > 0)) { 1964 Iterator<CmsExplorerTypeSettings> i = explorerTypes.iterator(); 1965 while (i.hasNext()) { 1966 CmsExplorerTypeSettings settings = i.next(); 1967 if (m_explorerTypeSettingsFromModules.contains(settings)) { 1968 m_explorerTypeSettingsFromModules.remove(settings); 1969 if (CmsLog.INIT.isInfoEnabled()) { 1970 CmsLog.INIT.info( 1971 Messages.get().getBundle().key( 1972 Messages.INIT_REMOVE_EXPLORER_TYPE_SETTING_1, 1973 settings.getName())); 1974 } 1975 } 1976 } 1977 // reset the list of all explorer type settings 1978 initExplorerTypeSettings(); 1979 } 1980 } 1981 1982 /** 1983 * Sets the value of the acacia-unlock configuration option.<p> 1984 * 1985 * @param value the value of the acacia-unlock configuration option 1986 */ 1987 public void setAcaciaUnlock(String value) { 1988 1989 m_acaciaUnlock = value; 1990 1991 } 1992 1993 /** 1994 * Sets the additional log folder configuration.<p> 1995 * 1996 * @param logConfig the additional log folder configuration 1997 */ 1998 public void setAdditionalLogFolderConfiguration(CmsAdditionalLogFolderConfig logConfig) { 1999 2000 m_logFolderConfig = logConfig; 2001 } 2002 2003 /** 2004 * Enables/disables special permissions for element authors to work with galleries (upload/replace). 2005 * @param allowElementAuthorToWorkInGalleries true if the special permissions should be enabled for element authors 2006 */ 2007 public void setAllowElementAuthorToWorkInGalleries(boolean allowElementAuthorToWorkInGalleries) { 2008 2009 m_allowElementAuthorToWorkInGalleries = allowElementAuthorToWorkInGalleries; 2010 } 2011 2012 /** 2013 * Sets if the autolock resources feature is enabled.<p> 2014 * 2015 * @param value <code>"true"</code> if the autolock resources feature is enabled, otherwise false 2016 */ 2017 public void setAutoLock(String value) { 2018 2019 m_autoLockResources = Boolean.valueOf(value).booleanValue(); 2020 if (CmsLog.INIT.isInfoEnabled()) { 2021 CmsLog.INIT.info( 2022 Messages.get().getBundle().key( 2023 m_autoLockResources ? Messages.INIT_AUTO_LOCK_ENABLED_0 : Messages.INIT_AUTO_LOCK_DISABLED_0)); 2024 } 2025 } 2026 2027 /** 2028 * Sets the category display options that affect how the category selection dialog is shown. 2029 * 2030 * @param displayCategoriesByRepository if true, the categories are shown separated by repository. 2031 * @param displayCategorySelectionCollapsed if true, the selection dialog opens showing only the top-level categories 2032 * (or the various repositories) in collapsed state. 2033 */ 2034 public void setCategoryDisplayOptions( 2035 String displayCategoriesByRepository, 2036 String displayCategorySelectionCollapsed) { 2037 2038 m_displayCategoriesByRepository = Boolean.parseBoolean(displayCategoriesByRepository); 2039 m_displayCategorySelectionCollapsed = Boolean.parseBoolean(displayCategorySelectionCollapsed); 2040 } 2041 2042 /** 2043 * Sets the name of the local category folder(s).<p> 2044 * 2045 * @param categoryFolder the name of the local category folder(s) 2046 */ 2047 public void setCategoryFolder(String categoryFolder) { 2048 2049 m_categoryFolder = categoryFolder; 2050 } 2051 2052 /** 2053 * Sets the access object of the type settings.<p> 2054 * 2055 * @param access access object 2056 */ 2057 public void setDefaultAccess(CmsExplorerTypeAccess access) { 2058 2059 m_defaultAccess = access; 2060 } 2061 2062 /** 2063 * Sets the Workplace default locale.<p> 2064 * 2065 * @param locale the locale to set 2066 */ 2067 public void setDefaultLocale(String locale) { 2068 2069 try { 2070 m_defaultLocale = CmsLocaleManager.getLocale(locale); 2071 } catch (Exception e) { 2072 if (CmsLog.INIT.isWarnEnabled()) { 2073 CmsLog.INIT.warn(Messages.get().getBundle().key(Messages.INIT_NONCRIT_ERROR_0), e); 2074 } 2075 } 2076 if (CmsLog.INIT.isInfoEnabled()) { 2077 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_DEFAULT_LOCALE_1, m_defaultLocale)); 2078 } 2079 } 2080 2081 /** 2082 * Sets the default property editing mode on resources.<p> 2083 * 2084 * @param defaultPropertiesOnStructure the default property editing mode on resources 2085 */ 2086 public void setDefaultPropertiesOnStructure(String defaultPropertiesOnStructure) { 2087 2088 m_defaultPropertiesOnStructure = Boolean.valueOf(defaultPropertiesOnStructure).booleanValue(); 2089 if (CmsLog.INIT.isInfoEnabled()) { 2090 CmsLog.INIT.info( 2091 Messages.get().getBundle().key( 2092 m_defaultPropertiesOnStructure 2093 ? Messages.INIT_PROP_ON_STRUCT_TRUE_0 2094 : Messages.INIT_PROP_ON_STRUCT_FALSE_0)); 2095 } 2096 } 2097 2098 /** 2099 * Sets the Workplace default user settings.<p> 2100 * 2101 * @param defaultUserSettings the user settings to set 2102 */ 2103 public void setDefaultUserSettings(CmsDefaultUserSettings defaultUserSettings) { 2104 2105 m_defaultUserSettings = defaultUserSettings; 2106 2107 if (CmsLog.INIT.isInfoEnabled()) { 2108 CmsLog.INIT.info( 2109 Messages.get().getBundle().key( 2110 Messages.INIT_DEFAULT_USER_SETTINGS_1, 2111 m_defaultUserSettings.getClass().getName())); 2112 } 2113 } 2114 2115 /** 2116 * Sets the direct edit provider.<p> 2117 * 2118 * @param clazz the direct edit provider to set 2119 */ 2120 public void setDirectEditProvider(I_CmsDirectEditProvider clazz) { 2121 2122 m_directEditProvider = clazz; 2123 if (CmsLog.INIT.isInfoEnabled()) { 2124 CmsLog.INIT.info( 2125 Messages.get().getBundle().key( 2126 Messages.INIT_DIRECT_EDIT_PROVIDER_1, 2127 m_directEditProvider.getClass().getName())); 2128 } 2129 } 2130 2131 /** 2132 * Sets the editor action class.<p> 2133 * 2134 * @param clazz the editor action class to set 2135 */ 2136 public void setEditorAction(I_CmsEditorActionHandler clazz) { 2137 2138 m_editorAction = clazz; 2139 if (CmsLog.INIT.isInfoEnabled()) { 2140 CmsLog.INIT.info( 2141 Messages.get().getBundle().key( 2142 Messages.INIT_EDITOR_ACTION_CLASS_1, 2143 m_editorAction.getClass().getName())); 2144 } 2145 } 2146 2147 /** 2148 * Sets the editor display option class.<p> 2149 * 2150 * @param clazz the editor display option class to set 2151 */ 2152 public void setEditorDisplayOptions(CmsEditorDisplayOptions clazz) { 2153 2154 m_editorDisplayOptions = clazz; 2155 if (CmsLog.INIT.isInfoEnabled()) { 2156 CmsLog.INIT.info( 2157 Messages.get().getBundle().key( 2158 Messages.INIT_EDITOR_DISPLAY_OPTS_1, 2159 m_editorAction.getClass().getName())); 2160 } 2161 } 2162 2163 /** 2164 * Sets the editor handler class.<p> 2165 * 2166 * @param clazz the editor handler class to set 2167 */ 2168 public void setEditorHandler(I_CmsEditorHandler clazz) { 2169 2170 m_editorHandler = clazz; 2171 if (CmsLog.INIT.isInfoEnabled()) { 2172 CmsLog.INIT.info( 2173 Messages.get().getBundle().key( 2174 Messages.INIT_EDITOR_HANDLER_CLASS_1, 2175 m_editorHandler.getClass().getName())); 2176 } 2177 } 2178 2179 /** 2180 * Sets the maximum number of locale buttons to display in the content editor. 2181 * @param maxLocaleButtons the number of buttons, as a string 2182 */ 2183 public void setEditorMaxLocaleButtons(String maxLocaleButtons) { 2184 2185 m_maxLocaleButtons = Math.max(0, Integer.valueOf(maxLocaleButtons)); 2186 } 2187 2188 /** 2189 * Sets the element delete mode.<p> 2190 * 2191 * @param deleteMode the element delete mode 2192 */ 2193 public void setElementDeleteMode(String deleteMode) { 2194 2195 try { 2196 m_elementDeleteMode = ElementDeleteMode.valueOf(deleteMode); 2197 } catch (Throwable t) { 2198 m_elementDeleteMode = ElementDeleteMode.askDelete; 2199 } 2200 } 2201 2202 /** 2203 * Sets if tabs in the advanced property dialog are enabled.<p> 2204 * 2205 * @param enableAdvancedPropertyTabs true if tabs should be enabled, otherwise false 2206 */ 2207 public void setEnableAdvancedPropertyTabs(String enableAdvancedPropertyTabs) { 2208 2209 m_enableAdvancedPropertyTabs = Boolean.valueOf(enableAdvancedPropertyTabs).booleanValue(); 2210 if (CmsLog.INIT.isInfoEnabled()) { 2211 CmsLog.INIT.info( 2212 Messages.get().getBundle().key( 2213 m_enableAdvancedPropertyTabs 2214 ? Messages.INIT_ADV_PROP_DIALOG_SHOW_TABS_0 2215 : Messages.INIT_ADV_PROP_DIALOG_HIDE_TABS_0)); 2216 } 2217 } 2218 2219 /** 2220 * Sets the options related to the 'Categories' column in the file explorer. 2221 * 2222 * @param enabled 'true' if the column should be enabled 2223 * @param leavesOnly 'true' if parent categories should be omitted 2224 * @param withPath 'true' if category labels should include the whole path (i.e. the titles of parent categories) 2225 */ 2226 public void setExplorerCategoryOptions(String enabled, String leavesOnly, String withPath) { 2227 2228 m_explorerCategoriesEnabled = Boolean.parseBoolean(enabled); 2229 m_explorerCategoriesLeavesOnly = Boolean.parseBoolean(leavesOnly); 2230 m_explorerCategoriesWithPath = Boolean.parseBoolean(withPath); 2231 2232 } 2233 2234 /** 2235 * Sets the value (in kb) for the maximum file upload size.<p> 2236 * 2237 * @param value the value (in kb) for the maximum file upload size 2238 */ 2239 public void setFileMaxUploadSize(String value) { 2240 2241 try { 2242 m_fileMaxUploadSize = Integer.valueOf(value).intValue(); 2243 } catch (NumberFormatException e) { 2244 // can usually be ignored 2245 if (LOG.isInfoEnabled()) { 2246 LOG.info(e.getLocalizedMessage()); 2247 } 2248 m_fileMaxUploadSize = -1; 2249 } 2250 if (CmsLog.INIT.isInfoEnabled()) { 2251 if (m_fileMaxUploadSize > 0) { 2252 CmsLog.INIT.info( 2253 Messages.get().getBundle().key( 2254 Messages.INIT_MAX_FILE_UPLOAD_SIZE_1, 2255 Integer.valueOf(m_fileMaxUploadSize))); 2256 } else { 2257 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_MAX_FILE_UPLOAD_SIZE_UNLIMITED_0)); 2258 } 2259 2260 } 2261 } 2262 2263 /** 2264 * Sets the system-wide file view settings for the workplace.<p> 2265 * 2266 * @param cms the CmsObject for ensuring security constraints. 2267 * 2268 * @param fileViewSettings the system-wide file view settings for the workplace to set 2269 * 2270 * @throws CmsRoleViolationException if the current user does not own the administrator role ({@link CmsRole#ROOT_ADMIN}) 2271 * */ 2272 public void setFileViewSettings(CmsObject cms, CmsRfsFileViewer fileViewSettings) throws CmsRoleViolationException { 2273 2274 if (OpenCms.getRunLevel() > OpenCms.RUNLEVEL_2_INITIALIZING) { 2275 OpenCms.getRoleManager().checkRole(cms, CmsRole.ROOT_ADMIN); 2276 } 2277 m_fileViewSettings = fileViewSettings; 2278 // disallow modifications of this "new original" 2279 m_fileViewSettings.setFrozen(true); 2280 } 2281 2282 /** 2283 * Sets the gallery default scope.<p> 2284 * 2285 * @param galleryDefaultScope the gallery default scope 2286 */ 2287 public void setGalleryDefaultScope(String galleryDefaultScope) { 2288 2289 m_galleryDefaultScope = galleryDefaultScope; 2290 try { 2291 CmsGallerySearchScope.valueOf(galleryDefaultScope); 2292 } catch (Throwable t) { 2293 LOG.warn(t.getLocalizedMessage(), t); 2294 } 2295 } 2296 2297 /** 2298 * Sets the group translation class name.<p> 2299 * 2300 * @param translationClassName the group translation class name 2301 */ 2302 public void setGroupTranslationClass(String translationClassName) { 2303 2304 m_groupTranslationClass = translationClassName; 2305 } 2306 2307 /** 2308 * Sets the "keep alive" mode.<p> 2309 * 2310 * @param keepAlive the keep-alive mode 2311 */ 2312 public void setKeepAlive(String keepAlive) { 2313 2314 m_keepAlive = Boolean.valueOf(keepAlive); 2315 } 2316 2317 /** 2318 * Sets the post upload dialog handler.<p> 2319 * 2320 * @param uploadHandler the post upload handler 2321 */ 2322 public void setPostUploadHandler(I_CmsPostUploadDialogHandler uploadHandler) { 2323 2324 m_postUploadHandler = uploadHandler; 2325 } 2326 2327 /** 2328 * Sets the repository folder handler.<p> 2329 * 2330 * @param clazz the repository folder handler 2331 */ 2332 public void setRepositoryFolderHandler(I_CmsRepositoryFolderHandler clazz) { 2333 2334 m_repositoryFolderHandler = clazz; 2335 if (CmsLog.INIT.isInfoEnabled()) { 2336 CmsLog.INIT.info( 2337 org.opencms.configuration.Messages.get().getBundle().key( 2338 org.opencms.configuration.Messages.INIT_REPOSITORY_FOLDER_1, 2339 m_repositoryFolderHandler.getClass().getName())); 2340 } 2341 } 2342 2343 /** 2344 * Sets the name of the role necessary for editing the sitemap configuration. 2345 * 2346 * @param roleName the name of the role necessary for editing the sitemap configuration 2347 */ 2348 public void setSitemapConfigEditRole(String roleName) { 2349 2350 m_sitemapConfigEditRole = roleName; 2351 } 2352 2353 /** 2354 * Sets the tool Manager object.<p> 2355 * 2356 * @param toolManager the tool Manager object to set 2357 */ 2358 public void setToolManager(CmsToolManager toolManager) { 2359 2360 m_toolManager = toolManager; 2361 } 2362 2363 /** 2364 * Sets the upload restriciton. 2365 * 2366 * @param uploadRestriction the upload restriction 2367 */ 2368 public void setUploadRestriction(I_CmsUploadRestriction uploadRestriction) { 2369 2370 if (m_uploadRestrictionSet) { 2371 throw new IllegalStateException("Upload restriction has already been set."); 2372 } 2373 m_uploadRestriction = uploadRestriction; 2374 m_uploadRestrictionSet = true; 2375 } 2376 2377 /** 2378 * Sets the user additional information configuration manager.<p> 2379 * 2380 * @param userInfoManager the manager to set 2381 */ 2382 public void setUserInfoManager(CmsWorkplaceUserInfoManager userInfoManager) { 2383 2384 m_userInfoManager = userInfoManager; 2385 } 2386 2387 /** 2388 * Sets the user list mode.<p> 2389 * 2390 * @param mode the user list mode 2391 */ 2392 public void setUserListMode(String mode) { 2393 2394 m_userListMode = mode; 2395 } 2396 2397 /** 2398 * Controls if the user/group icon in the administration view should be shown.<p> 2399 * 2400 * @param value <code>"true"</code> if the user/group icon in the administration view should be shown, otherwise false 2401 */ 2402 public void setUserManagementEnabled(String value) { 2403 2404 m_showUserGroupIcon = Boolean.valueOf(value).booleanValue(); 2405 if (CmsLog.INIT.isInfoEnabled()) { 2406 if (m_showUserGroupIcon) { 2407 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_USER_MANAGEMENT_ICON_ENABLED_0)); 2408 } else { 2409 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_USER_MANAGEMENT_ICON_DISABLED_0)); 2410 } 2411 } 2412 } 2413 2414 /** 2415 * Sets the virus scanner. 2416 * 2417 * @param virusScanner the virus scanner to set 2418 */ 2419 public void setVirusScanner(I_CmsVirusScanner virusScanner) { 2420 2421 m_virusScanner = virusScanner; 2422 } 2423 2424 /** 2425 * Sets the virus scanner to enabled/disabled. 2426 * 2427 * @param enabled true if the virus scanner should be enabled 2428 */ 2429 public void setVirusScannerEnabled(boolean enabled) { 2430 2431 m_virusScannerEnabled = enabled; 2432 } 2433 2434 /** 2435 * Sets the auto correction of XML contents when they are opened with the editor.<p> 2436 * 2437 * @param xmlContentAutoCorrect if "true", the content will be corrected without notification, otherwise a confirmation is needed 2438 */ 2439 public void setXmlContentAutoCorrect(String xmlContentAutoCorrect) { 2440 2441 m_xmlContentAutoCorrect = Boolean.valueOf(xmlContentAutoCorrect).booleanValue(); 2442 if (CmsLog.INIT.isInfoEnabled()) { 2443 CmsLog.INIT.info( 2444 Messages.get().getBundle().key( 2445 m_xmlContentAutoCorrect 2446 ? Messages.INIT_XMLCONTENT_AUTOCORRECT_ENABLED_0 2447 : Messages.INIT_XMLCONTENT_AUTOCORRECT_DISABLED_0)); 2448 } 2449 } 2450 2451 /** 2452 * Returns true if the Acacia editor in standalone mode should automatically unlock resources.<p> 2453 * 2454 * @return true if resources should be automatically unlocked in standalone mode 2455 */ 2456 public boolean shouldAcaciaUnlock() { 2457 2458 if (m_acaciaUnlock == null) { 2459 return true; 2460 } else { 2461 return Boolean.parseBoolean(m_acaciaUnlock); 2462 } 2463 } 2464 2465 /** 2466 * Returns if the user/group icon in the administration view should be shown.<p> 2467 * 2468 * @return true if the user/group icon in the administration view should be shown, otherwise false 2469 */ 2470 public boolean showUserGroupIcon() { 2471 2472 return m_showUserGroupIcon; 2473 } 2474 2475 /** 2476 * Returns true if lazy user lists should be used.<p> 2477 * 2478 * @return true if lazy user lists should be used 2479 */ 2480 public boolean supportsLazyUserLists() { 2481 2482 boolean result = "lazy".equalsIgnoreCase(m_userListMode); 2483 if (org.opencms.db.mssql.CmsUserDriver.isInstantiated()) { 2484 LOG.warn("Lazy user lists currently not supported on MSSQL, using classic user list mode as a fallback."); 2485 result = false; 2486 } 2487 return result; 2488 2489 } 2490 2491 /** 2492 * Translates a group name using the configured {@link I_CmsGroupNameTranslation}.<p> 2493 * 2494 * @param groupName the group name 2495 * @param keepOu if true, the OU will be appended to the translated name 2496 * 2497 * @return the translated group name 2498 */ 2499 public String translateGroupName(String groupName, boolean keepOu) { 2500 2501 I_CmsGroupNameTranslation translation = getGroupNameTranslation(); 2502 return translation.translateGroupName(groupName, keepOu); 2503 } 2504 2505 /** 2506 * Gets the default view name ( = explorer type) for the given type.<p> 2507 * 2508 * @param typeName a resource type name 2509 * @return the default view for the given type 2510 */ 2511 String getDefaultView(String typeName) { 2512 2513 String result = m_defaultViewRules.getViewForType(typeName); 2514 if (result == null) { 2515 result = "view_other"; 2516 try { 2517 if (OpenCms.getResourceManager().hasResourceType(typeName) 2518 && OpenCms.getResourceManager().getResourceType(typeName).isFolder()) { 2519 result = "view_folders"; 2520 } 2521 } catch (CmsLoaderException e) { 2522 LOG.error(e.getLocalizedMessage(), e); 2523 } 2524 } 2525 return result; 2526 } 2527 2528 /** 2529 * Creates a copy of the admin cms object which is initialize with the data of the current cms object.<p> 2530 * 2531 * @param cms the current cms object 2532 * @return the new admin cms object 2533 * 2534 * @throws CmsException if something goes wrong 2535 */ 2536 private CmsObject getAdminCms(CmsObject cms) throws CmsException { 2537 2538 CmsObject adminCms = OpenCms.initCmsObject(m_adminCms); 2539 adminCms.getRequestContext().setSiteRoot(cms.getRequestContext().getSiteRoot()); 2540 adminCms.getRequestContext().setRequestTime(cms.getRequestContext().getRequestTime()); 2541 adminCms.getRequestContext().setCurrentProject(cms.getRequestContext().getCurrentProject()); 2542 adminCms.getRequestContext().setEncoding(cms.getRequestContext().getEncoding()); 2543 adminCms.getRequestContext().setUri(cms.getRequestContext().getUri()); 2544 return adminCms; 2545 } 2546 2547 /** 2548 * Returns a dummy group name translation which leaves the group names unchanged.<p> 2549 * 2550 * @return a dummy group name translation 2551 */ 2552 private I_CmsGroupNameTranslation getDefaultGroupNameTranslation() { 2553 2554 return new I_CmsGroupNameTranslation() { 2555 2556 public String translateGroupName(String group, boolean keepOu) { 2557 2558 return keepOu ? group : CmsOrganizationalUnit.getSimpleName(group); 2559 } 2560 }; 2561 } 2562 2563 /** 2564 * Returns the role required for enabling the upload functionality. 2565 * 2566 * @return the upload role 2567 */ 2568 private CmsRole getUploadRole() { 2569 2570 return isAllowElementAuthorToWorkInGalleries() ? CmsRole.ELEMENT_AUTHOR : CmsRole.EDITOR; 2571 } 2572 2573 /** 2574 * Initializes the configured explorer type settings.<p> 2575 */ 2576 private synchronized void initExplorerTypeSettings() { 2577 2578 Map<String, CmsExplorerTypeSettings> explorerTypeSettingsMap = new HashMap<String, CmsExplorerTypeSettings>(); 2579 List<CmsExplorerTypeSettings> explorerTypeSettings = new ArrayList<CmsExplorerTypeSettings>(); 2580 2581 if (m_defaultAccess.getAccessControlList() == null) { 2582 try { 2583 // initialize the default access control configuration 2584 m_defaultAccess.createAccessControlList(CmsExplorerTypeAccess.PRINCIPAL_DEFAULT); 2585 } catch (CmsException e) { 2586 if (CmsLog.INIT.isInfoEnabled()) { 2587 CmsLog.INIT.info( 2588 Messages.get().getBundle().key( 2589 Messages.INIT_ADD_TYPE_SETTING_FAILED_1, 2590 CmsExplorerTypeAccess.PRINCIPAL_DEFAULT), 2591 e); 2592 } 2593 } 2594 } 2595 2596 explorerTypeSettings.addAll(m_explorerTypeSettingsFromXml); 2597 explorerTypeSettings.addAll(m_explorerTypeSettingsFromModules); 2598 2599 for (int i = 0; i < explorerTypeSettings.size(); i++) { 2600 CmsExplorerTypeSettings settings = explorerTypeSettings.get(i); 2601 // put the settings in the lookup map 2602 explorerTypeSettingsMap.put(settings.getName(), settings); 2603 if (getDefaultAccess() == settings.getAccess()) { 2604 continue; 2605 } 2606 try { 2607 // initialize the access control configuration of the explorer type 2608 settings.getAccess().createAccessControlList(settings.getName()); 2609 } catch (CmsException e) { 2610 if (CmsLog.INIT.isInfoEnabled()) { 2611 CmsLog.INIT.info( 2612 Messages.get().getBundle().key(Messages.INIT_ADD_TYPE_SETTING_FAILED_1, settings.getName()), 2613 e); 2614 } 2615 } 2616 } 2617 // sort the explorer type settings 2618 Collections.sort(explorerTypeSettings); 2619 // make the settings unmodifiable and store them in the global variables 2620 m_explorerTypeSettings = Collections.unmodifiableList(explorerTypeSettings); 2621 m_explorerTypeSettingsMap = Collections.unmodifiableMap(explorerTypeSettingsMap); 2622 2623 m_explorerTypeViews = Maps.newHashMap(); 2624 Set<String> explorerTypeViews = Sets.newHashSet(); 2625 for (CmsExplorerTypeSettings explorerType : getExplorerTypeSettings()) { 2626 if (explorerType.isView()) { 2627 explorerTypeViews.add(explorerType.getName()); 2628 } 2629 } 2630 2631 for (String typeName : explorerTypeViews) { 2632 CmsExplorerTypeSettings explorerType = OpenCms.getWorkplaceManager().getExplorerTypeSetting(typeName); 2633 CmsElementView elemView = new CmsElementView(explorerType); 2634 m_explorerTypeViews.put(elemView.getId(), elemView); 2635 } 2636 2637 } 2638 2639 /** 2640 * Initializes the workplace locale set.<p> 2641 * 2642 * Currently, this is defined by the existence of a special folder 2643 * <code>/system/workplace/locales/{locale-name}/</code>. 2644 * This is likely to change in future implementations.<p> 2645 * 2646 * @param cms an OpenCms context object that must have been initialized with "Admin" permissions 2647 * 2648 * @return the workplace locale set 2649 */ 2650 private List<Locale> initWorkplaceLocales(CmsObject cms) { 2651 2652 Set<Locale> locales = new HashSet<Locale>(); 2653 2654 // collect locales from the VFS 2655 if (cms.existsResource(CmsWorkplace.VFS_PATH_LOCALES)) { 2656 List<CmsResource> localeFolders; 2657 try { 2658 localeFolders = cms.getSubFolders(CmsWorkplace.VFS_PATH_LOCALES); 2659 } catch (CmsException e) { 2660 LOG.warn( 2661 Messages.get().getBundle().key( 2662 Messages.LOG_WORKPLACE_INIT_NO_LOCALES_1, 2663 CmsWorkplace.VFS_PATH_LOCALES), 2664 e); 2665 // can not throw exception here since then OpenCms would not even start in shell mode (runlevel 2) 2666 localeFolders = new ArrayList<CmsResource>(); 2667 } 2668 Iterator<CmsResource> i = localeFolders.iterator(); 2669 while (i.hasNext()) { 2670 CmsFolder folder = (CmsFolder)i.next(); 2671 Locale locale = CmsLocaleManager.getLocale(folder.getName()); 2672 // add locale 2673 locales.add(locale); 2674 // add less specialized locale 2675 locales.add(new Locale(locale.getLanguage(), locale.getCountry())); 2676 // add even less specialized locale 2677 locales.add(new Locale(locale.getLanguage())); 2678 } 2679 } 2680 // collect locales from JAR manifests 2681 try { 2682 Enumeration<URL> resources = getClass().getClassLoader().getResources(MANIFEST_RESOURCE_NAME); 2683 2684 while (resources.hasMoreElements()) { 2685 URL resUrl = resources.nextElement(); 2686 try { 2687 Manifest manifest = new Manifest(resUrl.openStream()); 2688 String localeString = manifest.getMainAttributes().getValue(LOCALIZATION_ATTRIBUTE_NAME); 2689 if (CmsStringUtil.isNotEmptyOrWhitespaceOnly(localeString)) { 2690 Locale locale = CmsLocaleManager.getLocale(localeString); 2691 // add locale 2692 locales.add(locale); 2693 // add less specialized locale 2694 locales.add(new Locale(locale.getLanguage(), locale.getCountry())); 2695 // add even less specialized locale 2696 locales.add(new Locale(locale.getLanguage())); 2697 } 2698 } catch (IOException e) { 2699 LOG.warn( 2700 "Error reading manifest from " + resUrl + " while evaluating available workplace localization.", 2701 e); 2702 } 2703 } 2704 } catch (IOException e) { 2705 LOG.error("Error evaluating available workplace localization from JAR manifests.", e); 2706 } 2707 2708 // sort the result 2709 ArrayList<Locale> result = new ArrayList<Locale>(); 2710 result.addAll(locales); 2711 Collections.sort(result, CmsLocaleComparator.getComparator()); 2712 return result; 2713 } 2714 2715 /** 2716 * Initializes the available workplace views.<p> 2717 * 2718 * Currently, this is defined by iterating the subfolder of the folder 2719 * <code>/system/workplace/views/</code>. 2720 * These subfolders must have the properties NavPos, NavText and default-file set.<p> 2721 * 2722 * @param cms an OpenCms context object that must have been initialized with "Admin" permissions 2723 * @return the available workplace views 2724 */ 2725 private List<CmsWorkplaceView> initWorkplaceViews(CmsObject cms) { 2726 2727 List<CmsResource> viewFolders; 2728 try { 2729 // get the subfolders of the "views" folder 2730 viewFolders = cms.getSubFolders(CmsWorkplace.VFS_PATH_VIEWS); 2731 } catch (CmsException e) { 2732 if ((OpenCms.getRunLevel() > OpenCms.RUNLEVEL_2_INITIALIZING) && LOG.isInfoEnabled()) { 2733 LOG.info( 2734 Messages.get().getBundle().key(Messages.LOG_WORKPLACE_INIT_NO_VIEWS_1, CmsWorkplace.VFS_PATH_VIEWS), 2735 e); 2736 } 2737 // can not throw exception here since then OpenCms would not even start in shell mode (runlevel 2) 2738 viewFolders = new ArrayList<CmsResource>(); 2739 } 2740 m_views = new ArrayList<CmsWorkplaceView>(viewFolders.size()); 2741 for (int i = 0; i < viewFolders.size(); i++) { 2742 // loop through all view folders 2743 CmsFolder folder = (CmsFolder)viewFolders.get(i); 2744 String folderPath = cms.getSitePath(folder); 2745 try { 2746 // get view information from folder properties 2747 String order = cms.readPropertyObject( 2748 folderPath, 2749 CmsPropertyDefinition.PROPERTY_NAVPOS, 2750 false).getValue(); 2751 String key = cms.readPropertyObject( 2752 folderPath, 2753 CmsPropertyDefinition.PROPERTY_NAVTEXT, 2754 false).getValue(); 2755 String viewUri = cms.readPropertyObject( 2756 folderPath, 2757 CmsPropertyDefinition.PROPERTY_DEFAULT_FILE, 2758 false).getValue(); 2759 if (viewUri == null) { 2760 // no view URI found 2761 viewUri = folderPath; 2762 } else if (!viewUri.startsWith("/")) { 2763 // default file is in current view folder, create absolute path to view URI 2764 viewUri = folderPath + viewUri; 2765 } 2766 if (order == null) { 2767 // no valid NavPos property value found, use loop count as order value 2768 order = "" + i; 2769 } 2770 Float orderValue; 2771 try { 2772 // create Float order object 2773 orderValue = Float.valueOf(order); 2774 } catch (NumberFormatException e) { 2775 // String was not formatted correctly, use loop counter 2776 orderValue = Float.valueOf(i); 2777 } 2778 if (key == null) { 2779 // no language key found, use default String to avoid NullPointerException 2780 key = "View " + i; 2781 // if no navtext is given do not display the view 2782 continue; 2783 } 2784 // create new view object 2785 CmsWorkplaceView view = new CmsWorkplaceView(key, viewUri, orderValue); 2786 m_views.add(view); 2787 // log the view 2788 if (CmsLog.INIT.isInfoEnabled()) { 2789 CmsLog.INIT.info(Messages.get().getBundle().key(Messages.INIT_WORKPLACE_VIEW_1, view.getUri())); 2790 } 2791 } catch (CmsException e) { 2792 // should usually never happen 2793 LOG.error(Messages.get().getBundle().key(Messages.LOG_READING_VIEW_FOLDER_FAILED_1, folderPath), e); 2794 } 2795 } 2796 // sort the views by their order number 2797 Collections.sort(m_views); 2798 return m_views; 2799 } 2800 2801}