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}