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, 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.ade.sitemap;
029
030import org.opencms.ade.configuration.CmsADEConfigData;
031import org.opencms.ade.configuration.CmsADEManager;
032import org.opencms.ade.configuration.CmsFunctionAvailability;
033import org.opencms.ade.configuration.CmsFunctionReference;
034import org.opencms.ade.configuration.CmsModelPageConfig;
035import org.opencms.ade.configuration.CmsResourceTypeConfig;
036import org.opencms.ade.configuration.formatters.CmsFormatterConfigurationCacheState;
037import org.opencms.ade.detailpage.CmsDetailPageConfigurationWriter;
038import org.opencms.ade.detailpage.CmsDetailPageInfo;
039import org.opencms.ade.sitemap.shared.CmsClientSitemapEntry;
040import org.opencms.ade.sitemap.shared.CmsClientSitemapEntry.EntryType;
041import org.opencms.ade.sitemap.shared.CmsDetailPageTable;
042import org.opencms.ade.sitemap.shared.CmsGalleryFolderEntry;
043import org.opencms.ade.sitemap.shared.CmsGalleryType;
044import org.opencms.ade.sitemap.shared.CmsLocaleComparePropertyData;
045import org.opencms.ade.sitemap.shared.CmsModelInfo;
046import org.opencms.ade.sitemap.shared.CmsModelPageEntry;
047import org.opencms.ade.sitemap.shared.CmsNewResourceInfo;
048import org.opencms.ade.sitemap.shared.CmsSitemapAttributeData;
049import org.opencms.ade.sitemap.shared.CmsSitemapCategoryData;
050import org.opencms.ade.sitemap.shared.CmsSitemapChange;
051import org.opencms.ade.sitemap.shared.CmsSitemapChange.ChangeType;
052import org.opencms.ade.sitemap.shared.CmsSitemapClipboardData;
053import org.opencms.ade.sitemap.shared.CmsSitemapData;
054import org.opencms.ade.sitemap.shared.CmsSitemapData.EditorMode;
055import org.opencms.ade.sitemap.shared.CmsSitemapInfo;
056import org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService;
057import org.opencms.configuration.CmsDefaultUserSettings;
058import org.opencms.db.CmsAlias;
059import org.opencms.db.CmsAliasManager;
060import org.opencms.db.CmsRewriteAlias;
061import org.opencms.file.CmsFile;
062import org.opencms.file.CmsObject;
063import org.opencms.file.CmsProject;
064import org.opencms.file.CmsProperty;
065import org.opencms.file.CmsPropertyDefinition;
066import org.opencms.file.CmsRequestContext;
067import org.opencms.file.CmsResource;
068import org.opencms.file.CmsResourceFilter;
069import org.opencms.file.CmsUser;
070import org.opencms.file.CmsVfsResourceAlreadyExistsException;
071import org.opencms.file.CmsVfsResourceNotFoundException;
072import org.opencms.file.history.CmsHistoryResourceHandler;
073import org.opencms.file.types.CmsResourceTypeFolder;
074import org.opencms.file.types.CmsResourceTypeFolderExtended;
075import org.opencms.file.types.CmsResourceTypeFolderSubSitemap;
076import org.opencms.file.types.CmsResourceTypeFunctionConfig;
077import org.opencms.file.types.CmsResourceTypeUnknownFolder;
078import org.opencms.file.types.CmsResourceTypeXmlContainerPage;
079import org.opencms.file.types.I_CmsResourceType;
080import org.opencms.flex.CmsFlexController;
081import org.opencms.gwt.CmsCoreService;
082import org.opencms.gwt.CmsGwtService;
083import org.opencms.gwt.CmsIconUtil;
084import org.opencms.gwt.CmsPropertyEditorHelper;
085import org.opencms.gwt.CmsRpcException;
086import org.opencms.gwt.CmsTemplateFinder;
087import org.opencms.gwt.CmsVfsService;
088import org.opencms.gwt.shared.CmsCategoryTreeEntry;
089import org.opencms.gwt.shared.CmsClientLock;
090import org.opencms.gwt.shared.CmsCoreData;
091import org.opencms.gwt.shared.CmsCoreData.AdeContext;
092import org.opencms.gwt.shared.CmsGwtConstants;
093import org.opencms.gwt.shared.CmsListInfoBean;
094import org.opencms.gwt.shared.alias.CmsAliasEditValidationReply;
095import org.opencms.gwt.shared.alias.CmsAliasEditValidationRequest;
096import org.opencms.gwt.shared.alias.CmsAliasImportResult;
097import org.opencms.gwt.shared.alias.CmsAliasInitialFetchResult;
098import org.opencms.gwt.shared.alias.CmsAliasSaveValidationRequest;
099import org.opencms.gwt.shared.alias.CmsAliasTableRow;
100import org.opencms.gwt.shared.alias.CmsRewriteAliasTableRow;
101import org.opencms.gwt.shared.alias.CmsRewriteAliasValidationReply;
102import org.opencms.gwt.shared.alias.CmsRewriteAliasValidationRequest;
103import org.opencms.gwt.shared.property.CmsClientProperty;
104import org.opencms.gwt.shared.property.CmsPropertyModification;
105import org.opencms.i18n.CmsLocaleManager;
106import org.opencms.json.JSONArray;
107import org.opencms.jsp.CmsJspNavBuilder;
108import org.opencms.jsp.CmsJspNavBuilder.Visibility;
109import org.opencms.jsp.CmsJspNavElement;
110import org.opencms.jsp.CmsJspTagLink;
111import org.opencms.loader.CmsLoaderException;
112import org.opencms.lock.CmsLock;
113import org.opencms.lock.CmsLockUtil;
114import org.opencms.main.CmsException;
115import org.opencms.main.CmsLog;
116import org.opencms.main.OpenCms;
117import org.opencms.main.OpenCmsServlet;
118import org.opencms.relations.CmsCategory;
119import org.opencms.relations.CmsCategoryService;
120import org.opencms.search.galleries.CmsGallerySearch;
121import org.opencms.search.galleries.CmsGallerySearchResult;
122import org.opencms.security.CmsPermissionSet;
123import org.opencms.security.CmsRole;
124import org.opencms.security.CmsRoleViolationException;
125import org.opencms.security.CmsSecurityException;
126import org.opencms.site.CmsSite;
127import org.opencms.ui.apps.CmsQuickLaunchLocationCache;
128import org.opencms.util.CmsDateUtil;
129import org.opencms.util.CmsFileUtil;
130import org.opencms.util.CmsMacroResolver;
131import org.opencms.util.CmsPath;
132import org.opencms.util.CmsStringUtil;
133import org.opencms.util.CmsUUID;
134import org.opencms.workplace.CmsWorkplaceManager;
135import org.opencms.workplace.CmsWorkplaceMessages;
136import org.opencms.workplace.explorer.CmsExplorerTypeAccess;
137import org.opencms.workplace.explorer.CmsExplorerTypeSettings;
138import org.opencms.workplace.galleries.A_CmsAjaxGallery;
139import org.opencms.xml.CmsXmlException;
140import org.opencms.xml.I_CmsXmlDocument;
141import org.opencms.xml.containerpage.CmsADESessionCache;
142import org.opencms.xml.containerpage.CmsContainerBean;
143import org.opencms.xml.containerpage.CmsContainerElementBean;
144import org.opencms.xml.containerpage.CmsContainerPageBean;
145import org.opencms.xml.containerpage.CmsXmlContainerPage;
146import org.opencms.xml.containerpage.CmsXmlContainerPageFactory;
147import org.opencms.xml.containerpage.CmsXmlDynamicFunctionHandler;
148import org.opencms.xml.containerpage.mutable.CmsContainerPageWrapper;
149import org.opencms.xml.containerpage.mutable.CmsMutableContainer;
150import org.opencms.xml.containerpage.mutable.CmsMutableContainerPage;
151import org.opencms.xml.content.CmsXmlContent;
152import org.opencms.xml.content.CmsXmlContentFactory;
153import org.opencms.xml.content.CmsXmlContentProperty;
154import org.opencms.xml.content.CmsXmlContentPropertyHelper;
155import org.opencms.xml.types.I_CmsXmlContentValue;
156
157import java.text.DateFormat;
158import java.util.ArrayList;
159import java.util.Arrays;
160import java.util.Collection;
161import java.util.Collections;
162import java.util.Comparator;
163import java.util.Date;
164import java.util.HashMap;
165import java.util.LinkedHashMap;
166import java.util.List;
167import java.util.ListIterator;
168import java.util.Locale;
169import java.util.Map;
170import java.util.regex.Pattern;
171import java.util.regex.PatternSyntaxException;
172
173import javax.servlet.http.HttpServletRequest;
174
175import org.apache.commons.logging.Log;
176
177import com.google.common.collect.ComparisonChain;
178import com.google.common.collect.Lists;
179import com.google.common.collect.Maps;
180import com.google.common.collect.Ordering;
181
182/**
183 * Handles all RPC services related to the vfs sitemap.<p>
184 *
185 * @since 8.0.0
186 *
187 * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService
188 * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapServiceAsync
189 */
190public class CmsVfsSitemapService extends CmsGwtService implements I_CmsSitemapService {
191
192    /** Helper class for representing information about a  lock. */
193    protected class LockInfo {
194
195        /** The lock state. */
196        private CmsLock m_lock;
197
198        /** True if the lock was just locked. */
199        private boolean m_wasJustLocked;
200
201        /**
202         * Creates a new LockInfo object.<p>
203         *
204         * @param lock the lock state
205         * @param wasJustLocked true if the lock was just locked
206         */
207        public LockInfo(CmsLock lock, boolean wasJustLocked) {
208
209            m_lock = lock;
210            m_wasJustLocked = wasJustLocked;
211        }
212
213        /**
214         * Returns the lock state.<p>
215         *
216         * @return the lock state
217         */
218        public CmsLock getLock() {
219
220            return m_lock;
221        }
222
223        /**
224         * Returns true if the lock was just locked.<p>
225         *
226         * @return true if the lock was just locked
227         */
228        public boolean wasJustLocked() {
229
230            return m_wasJustLocked;
231        }
232    }
233
234    /** The path of the JSP used to download aliases. */
235    public static final String ALIAS_DOWNLOAD_PATH = "/system/workplace/commons/download-aliases.jsp";
236
237    /** The path to the JSP used to upload aliases. */
238    public static final String ALIAS_IMPORT_PATH = "/system/workplace/commons/import-aliases.jsp";
239
240    /** The show model edit confirm dialog attribute name. */
241    public static final String ATTR_SHOW_MODEL_EDIT_CONFIRM = "showModelEditConfirm";
242
243    /** Properties to remove from the copied template when creating a new sitemap entry. */
244    public static final List<String> FILTER_PROPERTIES = Arrays.asList(
245        new String[] {
246            CmsPropertyDefinition.PROPERTY_TITLE,
247            CmsPropertyDefinition.PROPERTY_DESCRIPTION,
248            CmsPropertyDefinition.PROPERTY_DESCRIPTION_HTML,
249            CmsPropertyDefinition.PROPERTY_NAVTEXT,
250            CmsPropertyDefinition.PROPERTY_NAVINFO});
251
252    /** The galleries folder name. */
253    public static final String GALLERIES_FOLDER_NAME = ".galleries";
254
255    /** The configuration key for the functionDetail attribute in the container.info property. */
256    public static final String KEY_FUNCTION_DETAIL = "functionDetail";
257
258    /** The additional user info key for deleted list. */
259    private static final String ADDINFO_ADE_DELETED_LIST = "ADE_DELETED_LIST";
260
261    /** The additional user info key for modified list. */
262    private static final String ADDINFO_ADE_MODIFIED_LIST = "ADE_MODIFIED_LIST";
263
264    /** The lock table to prevent multiple users from editing the alias table concurrently. */
265    private static CmsAliasEditorLockTable aliasEditorLockTable = new CmsAliasEditorLockTable();
266
267    /** The table containing the alias import results. */
268    private static CmsAliasImportResponseTable aliasImportResponseTable = new CmsAliasImportResponseTable();
269
270    /** The static log object for this class. */
271    private static final Log LOG = CmsLog.getLog(CmsVfsSitemapService.class);
272
273    /** The redirect recource type name. */
274    private static final String RECOURCE_TYPE_NAME_REDIRECT = "htmlredirect";
275
276    /** The redirect target XPath. */
277    private static final String REDIRECT_LINK_TARGET_XPATH = "Link";
278
279    /** Serialization uid. */
280    private static final long serialVersionUID = -7236544324371767330L;
281
282    /** The VFS path of the redirect copy page for navigation level entries. */
283    private static final String SUB_LEVEL_REDIRECT_COPY_PAGE = "/system/modules/org.opencms.base/copyresources/sub-level-redirect.html";
284
285    /** The navigation builder. */
286    private CmsJspNavBuilder m_navBuilder;
287
288    /**
289     * Adds an alias import result.<p>
290     *
291     * @param results the list of alias import results to add
292     *
293     * @return the key to retrieve the alias import results
294     */
295    public static String addAliasImportResult(List<CmsAliasImportResult> results) {
296
297        return aliasImportResponseTable.addImportResult(results);
298    }
299
300    /**
301     * Creates a client property bean from a server-side property.<p>
302     *
303     * @param prop the property from which to create the client property
304     * @param preserveOrigin if true, the origin will be copied into the new object
305     *
306     * @return the new client property
307     */
308    public static CmsClientProperty createClientProperty(CmsProperty prop, boolean preserveOrigin) {
309
310        CmsClientProperty result = new CmsClientProperty(
311            prop.getName(),
312            prop.getStructureValue(),
313            prop.getResourceValue());
314        if (preserveOrigin) {
315            result.setOrigin(prop.getOrigin());
316        }
317        return result;
318    }
319
320    /**
321     * Fetches the sitemap data.<p>
322     *
323     * @param request the current request
324     * @param sitemapUri the site relative path
325     *
326     * @return the sitemap data
327     *
328     * @throws CmsRpcException if something goes wrong
329     */
330    public static CmsSitemapData prefetch(HttpServletRequest request, String sitemapUri) throws CmsRpcException {
331
332        CmsVfsSitemapService service = new CmsVfsSitemapService();
333        service.setCms(CmsFlexController.getCmsObject(request));
334        service.setRequest(request);
335        CmsSitemapData result = null;
336        try {
337            result = service.prefetch(sitemapUri);
338        } finally {
339            service.clearThreadStorage();
340        }
341        return result;
342    }
343
344    /**
345     * Converts a sequence of properties to a map of client-side property beans.<p>
346     *
347     * @param props the sequence of properties
348     * @param preserveOrigin if true, the origin of the properties should be copied to the client properties
349     *
350     * @return the map of client properties
351     *
352     */
353    static Map<String, CmsClientProperty> createClientProperties(Iterable<CmsProperty> props, boolean preserveOrigin) {
354
355        Map<String, CmsClientProperty> result = new HashMap<String, CmsClientProperty>();
356        for (CmsProperty prop : props) {
357            CmsClientProperty clientProp = createClientProperty(prop, preserveOrigin);
358            result.put(prop.getName(), clientProp);
359        }
360        return result;
361    }
362
363    /**
364     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#changeCategory(java.lang.String, org.opencms.util.CmsUUID, java.lang.String, java.lang.String)
365     */
366    public void changeCategory(String entryPoint, CmsUUID id, String title, String name) throws CmsRpcException {
367
368        try {
369            name = OpenCms.getResourceManager().getFileTranslator().translateResource(name.trim().replace('/', '-'));
370            CmsObject cms = getCmsObject();
371            CmsResource categoryResource = cms.readResource(id);
372            ensureLock(categoryResource);
373            String sitePath = cms.getSitePath(categoryResource);
374            String newPath = CmsStringUtil.joinPaths(CmsResource.getParentFolder(sitePath), name);
375            cms.writePropertyObject(sitePath, new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, title, null));
376            if (!CmsStringUtil.joinPaths("/", newPath, "/").equals(CmsStringUtil.joinPaths("/", sitePath, "/"))) {
377                cms.moveResource(sitePath, newPath);
378            }
379        } catch (Exception e) {
380            error(e);
381        }
382    }
383
384    /**
385     * @see org.opencms.gwt.CmsGwtService#checkPermissions(org.opencms.file.CmsObject)
386     */
387    @Override
388    public void checkPermissions(CmsObject cms) throws CmsRoleViolationException {
389
390        OpenCms.getRoleManager().checkRole(cms, CmsRole.EDITOR);
391    }
392
393    /**
394     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#createCategory(java.lang.String, org.opencms.util.CmsUUID, java.lang.String, java.lang.String)
395     */
396    public void createCategory(String entryPoint, CmsUUID id, String title, String name) throws CmsRpcException {
397
398        try {
399            if ((name == null) || (name.length() == 0)) {
400                name = title;
401            }
402            name = OpenCms.getResourceManager().getFileTranslator().translateResource(name.trim().replace('/', '-'));
403            CmsObject cms = getCmsObject();
404            CmsCategoryService catService = CmsCategoryService.getInstance();
405
406            CmsCategory createdCategory = null;
407            if (id.isNullUUID()) {
408                String localRepositoryPath = CmsStringUtil.joinPaths(
409                    entryPoint,
410                    CmsCategoryService.getInstance().getRepositoryBaseFolderName(getCmsObject()));
411
412                // ensure category repository exists
413                if (!cms.existsResource(localRepositoryPath)) {
414                    tryUnlock(
415                        cms.createResource(
416                            localRepositoryPath,
417                            OpenCms.getResourceManager().getResourceType(CmsResourceTypeFolder.getStaticTypeName())));
418                }
419                createdCategory = catService.createCategory(cms, null, name, title, "", localRepositoryPath);
420            } else {
421                CmsResource parentResource = cms.readResource(id);
422                CmsCategory parent = catService.getCategory(cms, parentResource);
423                createdCategory = catService.createCategory(
424                    cms,
425                    parent,
426                    name,
427                    title,
428                    "",
429                    cms.getRequestContext().removeSiteRoot(parentResource.getRootPath()));
430            }
431            tryUnlock(cms.readResource(createdCategory.getId()));
432        } catch (Exception e) {
433            error(e);
434        }
435    }
436
437    /**
438     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#createNewGalleryFolder(java.lang.String, java.lang.String, int)
439     */
440    public CmsGalleryFolderEntry createNewGalleryFolder(String parentFolder, String title, int folderTypeId)
441    throws CmsRpcException {
442
443        CmsObject cms = getCmsObject();
444        CmsGalleryFolderEntry folderEntry = null;
445        try {
446            if (!cms.existsResource(parentFolder, CmsResourceFilter.IGNORE_EXPIRATION)) {
447                CmsResource parent = cms.createResource(
448                    parentFolder,
449                    OpenCms.getResourceManager().getResourceType(CmsResourceTypeFolder.getStaticTypeName()));
450                tryUnlock(parent);
451            }
452            String folderName = OpenCms.getResourceManager().getFileTranslator().translateResource(title);
453
454            folderName = OpenCms.getResourceManager().getNameGenerator().getUniqueFileName(
455                cms,
456                parentFolder,
457                folderName);
458            String folderPath = CmsStringUtil.joinPaths(parentFolder, folderName);
459
460            @SuppressWarnings("deprecation")
461            CmsResource galleryFolder = cms.createResource(
462                folderPath,
463                folderTypeId,
464                null,
465                Collections.singletonList(new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, title, null)));
466            folderEntry = readGalleryFolderEntry(
467                galleryFolder,
468                OpenCms.getResourceManager().getResourceType(galleryFolder).getTypeName());
469            tryUnlock(galleryFolder);
470        } catch (CmsException e) {
471            error(e);
472        }
473        return folderEntry;
474    }
475
476    /**
477     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#createNewModelPage(java.lang.String, java.lang.String, java.lang.String, org.opencms.util.CmsUUID, boolean)
478     */
479    public CmsModelPageEntry createNewModelPage(
480        String entryPointUri,
481        String title,
482        String description,
483        CmsUUID copyId,
484        boolean isModelGroup)
485    throws CmsRpcException {
486
487        try {
488            CmsObject cms = getCmsObject();
489            CmsResource rootResource = cms.readResource(entryPointUri);
490            CmsModelPageHelper helper = new CmsModelPageHelper(getCmsObject(), rootResource);
491            CmsResource page;
492            if (isModelGroup) {
493                page = helper.createModelGroupPage(title, description, copyId);
494            } else {
495                page = helper.createPageInModelFolder(title, description, copyId);
496                String configPath = CmsStringUtil.joinPaths(entryPointUri, ".content/.config");
497                CmsResource configResource = cms.readResource(configPath);
498                helper.addModelPageToSitemapConfiguration(configResource, page, false);
499            }
500            CmsModelPageEntry result = helper.createModelPageEntry(page, false, false, getWorkplaceLocale());
501            OpenCms.getADEManager().waitForCacheUpdate(false);
502            return result;
503        } catch (Throwable e) {
504            error(e);
505            return null;
506        }
507    }
508
509    /**
510     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#createSubSitemap(org.opencms.util.CmsUUID)
511     */
512    public CmsSitemapChange createSubSitemap(CmsUUID entryId) throws CmsRpcException {
513
514        CmsObject cms = getCmsObject();
515        try {
516            ensureSession();
517            CmsResource subSitemapFolder = cms.readResource(entryId);
518            ensureLock(subSitemapFolder);
519            String sitePath = cms.getSitePath(subSitemapFolder);
520            createSitemapContentFolder(cms, subSitemapFolder);
521            subSitemapFolder.setType(getSubsitemapType());
522            cms.writeResource(subSitemapFolder);
523            tryUnlock(subSitemapFolder);
524            CmsSitemapClipboardData clipboard = getClipboardData();
525            CmsClientSitemapEntry entry = toClientEntry(
526                getNavBuilder().getNavigationForResource(sitePath, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED),
527                false);
528            clipboard.addModified(entry);
529            setClipboardData(clipboard);
530            CmsSitemapChange result = new CmsSitemapChange(entry.getId(), entry.getSitePath(), ChangeType.modify);
531            result.setUpdatedEntry(entry);
532            return result;
533        } catch (Exception e) {
534            error(e);
535        }
536        return null;
537
538    }
539
540    /**
541     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#disableModelPage(java.lang.String, org.opencms.util.CmsUUID, boolean)
542     */
543    public void disableModelPage(String baseUri, CmsUUID modelPageId, boolean disabled) throws CmsRpcException {
544
545        try {
546            CmsObject cms = getCmsObject();
547            CmsResource rootResource = cms.readResource(baseUri);
548            CmsModelPageHelper helper = new CmsModelPageHelper(getCmsObject(), rootResource);
549            String configPath = CmsStringUtil.joinPaths(baseUri, ".content/.config");
550            CmsResource configResource = cms.readResource(configPath);
551            helper.disableModelPage(configResource, modelPageId, disabled);
552            OpenCms.getADEManager().waitForCacheUpdate(false);
553        } catch (Throwable e) {
554            error(e);
555        }
556    }
557
558    /**
559     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#editAttributeData(org.opencms.util.CmsUUID)
560     */
561    public CmsSitemapAttributeData editAttributeData(CmsUUID rootId) throws CmsRpcException {
562
563        try {
564            CmsObject cms = getCmsObject();
565            CmsResource root = cms.readResource(rootId);
566            CmsADEConfigData config = OpenCms.getADEManager().lookupConfiguration(cms, root.getRootPath());
567            String sitePath = cms.getSitePath(root);
568            String configPath = CmsStringUtil.joinPaths(sitePath, "/.content/.config");
569            CmsResource configResource = cms.readResource(configPath);
570            cms.lockResourceTemporary(configResource);
571            CmsListInfoBean configInfo = CmsVfsService.getPageInfo(cms, configResource);
572            Map<String, CmsXmlContentProperty> definitions = config.getAttributeEditorConfiguration().getAttributeDefinitions();
573            cms.getRequestContext().setAttribute(
574                CmsRequestContext.ATTRIBUTE_ADE_CONTEXT_PATH,
575                configResource.getRootPath());
576            definitions = CmsXmlContentPropertyHelper.resolveMacrosInProperties(
577                definitions,
578                CmsMacroResolver.newWorkplaceLocaleResolver(cms));
579
580            HashMap<String, String> values = new HashMap<>();
581            for (String key : definitions.keySet()) {
582                values.put(key, config.getAttribute(key, ""));
583            }
584            String unlockUrl = CmsStringUtil.joinPaths(
585                OpenCms.getStaticExportManager().getVfsPrefix(),
586                OpenCmsServlet.HANDLE_BUILTIN_SERVICE,
587                CmsGwtConstants.UNLOCK_FILE_PREFIX,
588                configResource.getStructureId().toString());
589            CmsSitemapAttributeData result = new CmsSitemapAttributeData(configInfo, definitions, values, unlockUrl);
590            return result;
591        } catch (Exception e) {
592            error(e);
593            return null;
594        }
595    }
596
597    /**
598     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getAliasImportResult(java.lang.String)
599     */
600    public List<CmsAliasImportResult> getAliasImportResult(String resultKey) {
601
602        return aliasImportResponseTable.getAndRemove(resultKey);
603    }
604
605    /**
606     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getAliasTable()
607     */
608    public CmsAliasInitialFetchResult getAliasTable() throws CmsRpcException {
609
610        try {
611            CmsObject cms = getCmsObject();
612            CmsAliasManager aliasManager = OpenCms.getAliasManager();
613            List<CmsAlias> aliases = aliasManager.getAliasesForSite(cms, cms.getRequestContext().getSiteRoot());
614            List<CmsAliasTableRow> rows = new ArrayList<CmsAliasTableRow>();
615            CmsAliasInitialFetchResult result = new CmsAliasInitialFetchResult();
616            for (CmsAlias alias : aliases) {
617                CmsAliasTableRow row = createAliasTableEntry(cms, alias);
618                rows.add(row);
619            }
620            result.setRows(rows);
621
622            List<CmsRewriteAlias> rewriteAliases = aliasManager.getRewriteAliases(
623                cms,
624                cms.getRequestContext().getSiteRoot());
625
626            List<CmsRewriteAliasTableRow> rewriteRows = new ArrayList<CmsRewriteAliasTableRow>();
627            for (CmsRewriteAlias rewriteAlias : rewriteAliases) {
628                CmsRewriteAliasTableRow rewriteRow = new CmsRewriteAliasTableRow(
629                    rewriteAlias.getId(),
630                    rewriteAlias.getPatternString(),
631                    rewriteAlias.getReplacementString(),
632                    rewriteAlias.getMode());
633                rewriteRows.add(rewriteRow);
634            }
635            result.setRewriteRows(rewriteRows);
636            CmsUser otherLockOwner = aliasEditorLockTable.update(cms, cms.getRequestContext().getSiteRoot());
637            if (otherLockOwner != null) {
638                result.setAliasLockOwner(otherLockOwner.getName());
639            }
640            result.setDownloadUrl(OpenCms.getLinkManager().substituteLinkForRootPath(cms, ALIAS_DOWNLOAD_PATH));
641            return result;
642        } catch (Throwable e) {
643            error(e);
644            return null;
645        }
646    }
647
648    /**
649     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getCategoryData(java.lang.String)
650     */
651    public CmsSitemapCategoryData getCategoryData(String entryPoint) throws CmsRpcException {
652
653        CmsObject cms = getCmsObject();
654        try {
655            CmsResource entryPointResource = cms.readResource(entryPoint);
656            String basePath = CmsStringUtil.joinPaths(
657                entryPointResource.getRootPath(),
658                CmsCategoryService.getInstance().getRepositoryBaseFolderName(getCmsObject()));
659
660            List<CmsCategoryTreeEntry> entries = CmsCoreService.getCategoriesForSitePathStatic(
661                cms,
662                entryPoint,
663                cms.getRequestContext().removeSiteRoot(basePath));
664            CmsSitemapCategoryData categoryData = new CmsSitemapCategoryData();
665            for (CmsCategoryTreeEntry entry : entries) {
666                categoryData.add(entry);
667            }
668
669            categoryData.setBasePath(basePath);
670            return categoryData;
671        } catch (Exception e) {
672            error(e);
673            return null;
674        }
675    }
676
677    /**
678     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getChildren(java.lang.String, org.opencms.util.CmsUUID, int)
679     */
680    public CmsClientSitemapEntry getChildren(String entryPointUri, CmsUUID entryId, int levels) throws CmsRpcException {
681
682        CmsClientSitemapEntry entry = null;
683
684        try {
685            CmsObject cms = getCmsObject();
686
687            //ensure that root ends with a '/' if it's a folder
688            CmsResource rootRes = cms.readResource(entryId, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
689            String root = cms.getSitePath(rootRes);
690            CmsJspNavElement navElement = getNavBuilder().getNavigationForResource(
691                root,
692                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
693            boolean isRoot = root.equals(entryPointUri);
694            entry = toClientEntry(navElement, isRoot);
695            if ((levels > 0) && (isRoot || (rootRes.isFolder()))) {
696                entry.setSubEntries(getChildren(root, levels, null), null);
697            }
698        } catch (Throwable e) {
699            error(e);
700        }
701        return entry;
702    }
703
704    /**
705     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getGalleryData(java.lang.String)
706     */
707    public Map<CmsGalleryType, List<CmsGalleryFolderEntry>> getGalleryData(String entryPointUri) {
708
709        List<String> subSitePaths = OpenCms.getADEManager().getSubSitePaths(
710            getCmsObject(),
711            getCmsObject().getRequestContext().addSiteRoot(entryPointUri));
712        List<CmsGalleryType> galleryTypes = collectGalleryTypes();
713        Map<CmsGalleryType, List<CmsGalleryFolderEntry>> result = new HashMap<CmsGalleryType, List<CmsGalleryFolderEntry>>();
714        for (CmsGalleryType type : galleryTypes) {
715            List<CmsGalleryFolderEntry> galleries = null;
716            try {
717                galleries = getGalleriesForType(entryPointUri, type, subSitePaths);
718            } catch (CmsException e) {
719                log(e.getLocalizedMessage(), e);
720            }
721            result.put(type, galleries);
722        }
723        return result;
724    }
725
726    /**
727     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getModelInfos(org.opencms.util.CmsUUID)
728     */
729    public CmsModelInfo getModelInfos(CmsUUID rootId) throws CmsRpcException {
730
731        try {
732            CmsObject cms = getCmsObject();
733            CmsResource rootResource = cms.readResource(rootId);
734            CmsModelPageHelper modelPageHelper = new CmsModelPageHelper(getCmsObject(), rootResource);
735            return modelPageHelper.getModelInfo();
736        } catch (Throwable e) {
737            error(e);
738            return null; // will  never be reached
739        }
740
741    }
742
743    /**
744     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getNewElementInfo(java.lang.String)
745     */
746    public List<CmsNewResourceInfo> getNewElementInfo(String entryPointUri) throws CmsRpcException {
747
748        try {
749            CmsObject cms = getCmsObject();
750            String rootPath = cms.getRequestContext().addSiteRoot(entryPointUri);
751            CmsADEConfigData config = OpenCms.getADEManager().lookupConfiguration(cms, rootPath);
752            List<CmsNewResourceInfo> result = getNewResourceInfos(cms, config, getWorkplaceLocale());
753            return result;
754        } catch (Throwable e) {
755            error(e);
756            return null;
757        }
758    }
759
760    /**
761     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#getResourceLink(org.opencms.util.CmsUUID, java.lang.String)
762     */
763    public String getResourceLink(CmsUUID baseId, String sitePath) throws CmsRpcException {
764
765        try {
766            CmsObject cms = OpenCms.initCmsObject(getCmsObject());
767            CmsResource baseResource = cms.readResource(baseId, CmsResourceFilter.IGNORE_EXPIRATION);
768            String contextPath = cms.getSitePath(baseResource);
769            cms.getRequestContext().setUri(contextPath);
770            Locale locale = CmsJspTagLink.getBaseUriLocale(cms, contextPath);
771            if (locale != null) {
772                cms.getRequestContext().setLocale(locale);
773            }
774            String result = OpenCms.getLinkManager().substituteLinkForUnknownTarget(cms, sitePath);
775            return result;
776        } catch (Exception e) {
777            error(e);
778        }
779        return null;
780    }
781
782    /**
783     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#loadPropertyDataForLocaleCompareView(org.opencms.util.CmsUUID, org.opencms.util.CmsUUID)
784     */
785    public CmsLocaleComparePropertyData loadPropertyDataForLocaleCompareView(CmsUUID id, CmsUUID rootId)
786    throws CmsRpcException {
787
788        try {
789            CmsLocaleComparePropertyData result = new CmsLocaleComparePropertyData();
790            CmsObject cms = getCmsObject();
791            CmsResourceFilter filter = CmsResourceFilter.IGNORE_EXPIRATION;
792            CmsResource resource = cms.readResource(id, filter);
793            CmsResource defaultFile = cms.readDefaultFile(id.toString());
794            result.setDefaultFileId(defaultFile != null ? defaultFile.getStructureId() : null);
795            result.setId(id);
796            result.setFolder(resource.isFolder());
797            List<CmsProperty> props = cms.readPropertyObjects(resource, false);
798            List<CmsProperty> defaultFileProps = cms.readPropertyObjects(defaultFile, false);
799            Map<String, CmsClientProperty> clientProps = createClientProperties(props, true);
800            Map<String, CmsClientProperty> clientDefaultFileProps = createClientProperties(defaultFileProps, true);
801
802            result.setOwnProperties(clientProps);
803            result.setDefaultFileProperties(clientDefaultFileProps);
804            List<CmsResource> blockingLocked = cms.getBlockingLockedResources(resource);
805
806            CmsResource parent = cms.readParentFolder(resource.getStructureId());
807            List<CmsResource> resourcesInSameFolder = cms.readResources(parent, CmsResourceFilter.ALL, false);
808            List<String> forbiddenUrlNames = Lists.newArrayList();
809            for (CmsResource resourceInSameFolder : resourcesInSameFolder) {
810                String otherName = CmsResource.getName(resourceInSameFolder.getRootPath());
811                forbiddenUrlNames.add(otherName);
812            }
813            result.setForbiddenUrlNames(forbiddenUrlNames);
814            result.setInheritedProperties(createClientProperties(cms.readPropertyObjects(parent, true), true));
815            result.setPath(resource.getRootPath());
816            String name = CmsFileUtil.removeTrailingSeparator(CmsResource.getName(resource.getRootPath()));
817            result.setName(name);
818            result.setHasEditableName(!CmsStringUtil.isEmptyOrWhitespaceOnly(name) && blockingLocked.isEmpty());
819            return result;
820        } catch (Exception e) {
821            error(e);
822            return null;
823        }
824    }
825
826    /**
827     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#mergeSubSitemap(java.lang.String, org.opencms.util.CmsUUID)
828     */
829    @SuppressWarnings("deprecation")
830    public CmsSitemapChange mergeSubSitemap(String entryPoint, CmsUUID subSitemapId) throws CmsRpcException {
831
832        CmsObject cms = getCmsObject();
833        try {
834            ensureSession();
835            CmsResource subSitemapFolder = cms.readResource(subSitemapId, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
836            ensureLock(subSitemapFolder);
837            subSitemapFolder.setType(
838                OpenCms.getResourceManager().getResourceType(CmsResourceTypeFolder.RESOURCE_TYPE_NAME).getTypeId());
839            cms.writeResource(subSitemapFolder);
840            String sitePath = cms.getSitePath(subSitemapFolder);
841            String sitemapConfigName = CmsStringUtil.joinPaths(
842                sitePath,
843                CmsADEManager.CONTENT_FOLDER_NAME,
844                CmsADEManager.CONFIG_FILE_NAME);
845            if (cms.existsResource(sitemapConfigName)) {
846                cms.deleteResource(sitemapConfigName, CmsResource.DELETE_PRESERVE_SIBLINGS);
847            }
848            tryUnlock(subSitemapFolder);
849            CmsSitemapClipboardData clipboard = getClipboardData();
850            CmsClientSitemapEntry entry = toClientEntry(
851                getNavBuilder().getNavigationForResource(
852                    cms.getSitePath(subSitemapFolder),
853                    CmsResourceFilter.ONLY_VISIBLE_NO_DELETED),
854                false);
855            clipboard.addModified(entry);
856            setClipboardData(clipboard);
857            entry = getChildren(entryPoint, subSitemapId, 1);
858            CmsSitemapChange result = new CmsSitemapChange(entry.getId(), entry.getSitePath(), ChangeType.modify);
859            result.setUpdatedEntry(entry);
860            return result;
861        } catch (Exception e) {
862            error(e);
863        }
864        return null;
865    }
866
867    /**
868     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#prefetch(java.lang.String)
869     */
870    public CmsSitemapData prefetch(String sitemapUri) throws CmsRpcException {
871
872        CmsSitemapData result = null;
873        CmsObject cms = getCmsObject();
874
875        try {
876            OpenCms.getRoleManager().checkRole(cms, CmsRole.EDITOR);
877            String openPath = getRequest().getParameter(CmsCoreData.PARAM_PATH);
878            if (!isValidOpenPath(cms, openPath)) {
879                // if no path is supplied, start from root
880                openPath = "/";
881            }
882            CmsQuickLaunchLocationCache.getLocationCache(getRequest().getSession()).setSitemapEditorLocation(
883                cms.getRequestContext().getSiteRoot(),
884                openPath);
885            CmsADEConfigData configData = OpenCms.getADEManager().lookupConfiguration(
886                cms,
887                cms.getRequestContext().addSiteRoot(openPath));
888            Map<String, CmsXmlContentProperty> propertyConfig = new LinkedHashMap<String, CmsXmlContentProperty>(
889                configData.getPropertyConfigurationAsMap());
890            propertyConfig = CmsXmlContentPropertyHelper.resolveMacrosInProperties(
891                propertyConfig,
892                CmsMacroResolver.newWorkplaceLocaleResolver(getCmsObject()));
893
894            Map<String, CmsClientProperty> parentProperties = generateParentProperties(configData.getBasePath());
895            String siteRoot = cms.getRequestContext().getSiteRoot();
896            String exportRfsPrefix = OpenCms.getStaticExportManager().getDefaultRfsPrefix();
897            CmsSite site = OpenCms.getSiteManager().getSiteForSiteRoot(siteRoot);
898            boolean isSecure = (site != null) && site.hasSecureServer();
899            String parentSitemap = null;
900
901            if (configData.getBasePath() != null) {
902                CmsADEConfigData parentConfigData = OpenCms.getADEManager().lookupConfiguration(
903                    cms,
904                    CmsResource.getParentFolder(configData.getBasePath()));
905                parentSitemap = parentConfigData.getBasePath();
906                if (parentSitemap != null) {
907                    parentSitemap = cms.getRequestContext().removeSiteRoot(parentSitemap);
908                }
909            }
910            String noEdit = "";
911            CmsNewResourceInfo defaultNewInfo = null;
912            List<CmsNewResourceInfo> newResourceInfos = null;
913            CmsDetailPageTable detailPages = null;
914            List<CmsNewResourceInfo> resourceTypeInfos = null;
915            boolean canEditDetailPages = false;
916            boolean isOnlineProject = CmsProject.isOnlineProject(cms.getRequestContext().getCurrentProject().getUuid());
917            String defaultGalleryFolder = GALLERIES_FOLDER_NAME;
918            try {
919                CmsObject rootCms = OpenCms.initCmsObject(cms);
920                rootCms.getRequestContext().setSiteRoot("");
921                CmsResource baseDir = rootCms.readResource(
922                    configData.getBasePath(),
923                    CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
924                CmsProperty galleryFolderProp = cms.readPropertyObject(
925                    baseDir,
926                    CmsPropertyDefinition.PROPERTY_GALLERIES_FOLDER,
927                    true);
928                if (!galleryFolderProp.isNullProperty()
929                    && CmsStringUtil.isNotEmptyOrWhitespaceOnly(galleryFolderProp.getValue())) {
930                    defaultGalleryFolder = galleryFolderProp.getValue();
931                }
932                CmsPropertyEditorHelper.updateWysiwygConfig(propertyConfig, cms, baseDir);
933            } catch (CmsException e) {
934                LOG.warn(e.getLocalizedMessage(), e);
935            }
936
937            detailPages = new CmsDetailPageTable(configData.getAllDetailPages());
938            if (!isOnlineProject) {
939                newResourceInfos = getNewResourceInfos(cms, configData, getWorkplaceLocale());
940                CmsResource modelResource = null;
941                if (configData.getDefaultModelPage() != null) {
942                    if (cms.existsResource(configData.getDefaultModelPage().getResource().getStructureId())) {
943                        modelResource = configData.getDefaultModelPage().getResource();
944                    } else {
945                        try {
946                            modelResource = cms.readResource(
947                                cms.getSitePath(configData.getDefaultModelPage().getResource()),
948                                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
949                        } catch (CmsException e) {
950                            LOG.warn(e.getLocalizedMessage(), e);
951                        }
952                    }
953                }
954                if ((modelResource == null) && !newResourceInfos.isEmpty()) {
955                    try {
956                        modelResource = cms.readResource(newResourceInfos.get(0).getCopyResourceId());
957                    } catch (CmsException e) {
958                        LOG.warn(e.getLocalizedMessage(), e);
959                    }
960                }
961                CmsFormatterConfigurationCacheState formatterState = OpenCms.getADEManager().getCachedFormatters(
962                    cms.getRequestContext().getCurrentProject().isOnlineProject());
963                if (modelResource != null) {
964                    resourceTypeInfos = getResourceTypeInfos(
965                        getCmsObject(),
966                        configData.getResourceTypes(),
967                        configData.getFunctionReferences(),
968                        configData.getDynamicFunctionAvailability(formatterState),
969                        modelResource,
970                        getWorkplaceLocale());
971                    try {
972                        defaultNewInfo = createNewResourceInfo(cms, modelResource, getWorkplaceLocale());
973                    } catch (CmsException e) {
974                        LOG.warn(e.getLocalizedMessage(), e);
975                    }
976                }
977                canEditDetailPages = !(configData.isModuleConfiguration());
978            }
979            if (isOnlineProject) {
980                noEdit = Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_SITEMAP_NO_EDIT_ONLINE_0);
981            }
982            List<String> allPropNames = getPropertyNames(cms);
983            String returnCode = getRequest().getParameter(CmsCoreData.PARAM_RETURNCODE);
984            if (!isValidReturnCode(returnCode)) {
985                returnCode = null;
986            }
987            cms.getRequestContext().getSiteRoot();
988            String aliasImportUrl = OpenCms.getLinkManager().substituteLinkForRootPath(cms, ALIAS_IMPORT_PATH);
989            boolean canEditAliases = OpenCms.getAliasManager().hasPermissionsForMassEdit(cms, siteRoot);
990            List<CmsListInfoBean> subsitemapFolderTypeInfos = collectSitemapTypeInfos(cms, configData);
991
992            // evaluate the editor mode
993            EditorMode editorMode = CmsADESessionCache.getCache(getRequest(), getCmsObject()).getSitemapEditorMode();
994            if ((editorMode == null) || (editorMode == EditorMode.compareLocales)) {
995                editorMode = EditorMode.navigation;
996            }
997
998            String basePath = configData.getBasePath();
999
1000            if (!cms.existsResource(
1001                cms.getRequestContext().removeSiteRoot(basePath),
1002                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED)) {
1003                basePath = cms.getRequestContext().getSiteRoot();
1004            }
1005            Boolean showModelEditConfirm = (Boolean)getRequest().getSession().getAttribute(
1006                ATTR_SHOW_MODEL_EDIT_CONFIRM);
1007            if (showModelEditConfirm == null) {
1008                showModelEditConfirm = Boolean.TRUE;
1009            }
1010            boolean showLocaleComparison = !(getCmsObject().getRequestContext().getCurrentProject().isOnlineProject())
1011                && (site != null)
1012                && (site.getMainTranslationLocale(null) != null);
1013
1014            result = new CmsSitemapData(
1015                (new CmsTemplateFinder(cms)).getTemplates(),
1016                propertyConfig,
1017                getClipboardData(),
1018                CmsCoreService.getContextMenuEntries(
1019                    cms,
1020                    configData.getResource().getStructureId(),
1021                    AdeContext.sitemapeditor,
1022                    new HashMap<>()),
1023                parentProperties,
1024                allPropNames,
1025                exportRfsPrefix,
1026                isSecure,
1027                noEdit,
1028                isDisplayToolbar(getRequest()),
1029                defaultNewInfo,
1030                newResourceInfos,
1031                createResourceTypeInfo(OpenCms.getResourceManager().getResourceType(RECOURCE_TYPE_NAME_REDIRECT), null),
1032                createNavigationLevelTypeInfo(),
1033                getSitemapInfo(basePath),
1034                parentSitemap,
1035                getRootEntry(basePath, CmsResource.getFolderPath(openPath)),
1036                openPath,
1037                30,
1038                detailPages,
1039                resourceTypeInfos,
1040                returnCode,
1041                canEditDetailPages,
1042                aliasImportUrl,
1043                canEditAliases,
1044                OpenCms.getWorkplaceManager().getDefaultUserSettings().getSubsitemapCreationMode() == CmsDefaultUserSettings.SubsitemapCreationMode.createfolder,
1045                OpenCms.getRoleManager().hasRole(cms, CmsRole.GALLERY_EDITOR),
1046                OpenCms.getRoleManager().hasRole(cms, CmsRole.CATEGORY_EDITOR),
1047                subsitemapFolderTypeInfos,
1048                editorMode,
1049                defaultGalleryFolder,
1050                showModelEditConfirm.booleanValue());
1051
1052            CmsUUID rootId = cms.readResource("/", CmsResourceFilter.ALL).getStructureId();
1053            result.setSiteRootId(rootId);
1054            result.setLocaleComparisonEnabled(showLocaleComparison);
1055            boolean allowCreateNestedGalleries = Boolean.parseBoolean(
1056                "" + OpenCms.getRuntimeProperty("ade.sitemap.allowCreateNestedGalleries"));
1057            result.setAllowCreateNestedGalleries(allowCreateNestedGalleries);
1058        } catch (Throwable e) {
1059            error(e);
1060        }
1061        return result;
1062    }
1063
1064    /**
1065     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#prepareReloadSitemap(org.opencms.util.CmsUUID, org.opencms.ade.sitemap.shared.CmsSitemapData.EditorMode)
1066     */
1067    public String prepareReloadSitemap(CmsUUID rootId, EditorMode mode) throws CmsRpcException {
1068
1069        try {
1070            CmsObject cms = getCmsObject();
1071            CmsResource res = cms.readResource(CmsADEManager.PATH_SITEMAP_EDITOR_JSP);
1072
1073            CmsResource target = cms.readResource(rootId, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
1074            String targetRootPath = OpenCms.getADEManager().getSubSiteRoot(cms, target.getRootPath());
1075            CmsSite targetSite = OpenCms.getSiteManager().getSiteForRootPath(targetRootPath);
1076            CmsADESessionCache.getCache(getRequest(), getCmsObject()).setSitemapEditorMode(mode);
1077            if (targetSite != null) {
1078                cms.getRequestContext().setSiteRoot(targetSite.getSiteRoot());
1079                String path = cms.getRequestContext().removeSiteRoot(targetRootPath);
1080                String link = OpenCms.getLinkManager().substituteLink(cms, res);
1081                link = link + "?path=" + path;
1082                return link;
1083            }
1084            return null;
1085        } catch (Exception e) {
1086            error(e);
1087            return null;
1088        }
1089    }
1090
1091    /**
1092     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#removeModelPage(java.lang.String, org.opencms.util.CmsUUID)
1093     */
1094    public void removeModelPage(String entryPointUri, CmsUUID id) throws CmsRpcException {
1095
1096        try {
1097            CmsObject cms = getCmsObject();
1098            CmsResource rootResource = cms.readResource(entryPointUri);
1099            CmsModelPageHelper helper = new CmsModelPageHelper(getCmsObject(), rootResource);
1100            String configPath = CmsStringUtil.joinPaths(entryPointUri, ".content/.config");
1101            CmsResource configResource = cms.readResource(configPath);
1102            helper.removeModelPage(configResource, id);
1103            OpenCms.getADEManager().waitForCacheUpdate(false);
1104        } catch (Throwable e) {
1105            error(e);
1106
1107        }
1108
1109    }
1110
1111    /**
1112     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#save(java.lang.String, org.opencms.ade.sitemap.shared.CmsSitemapChange)
1113     */
1114    public CmsSitemapChange save(String entryPoint, CmsSitemapChange change) throws CmsRpcException {
1115
1116        CmsSitemapChange result = null;
1117        try {
1118            result = saveInternal(entryPoint, change);
1119        } catch (Exception e) {
1120            error(e);
1121        }
1122        return result;
1123    }
1124
1125    /**
1126     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#saveAliases(org.opencms.gwt.shared.alias.CmsAliasSaveValidationRequest)
1127     */
1128    public CmsAliasEditValidationReply saveAliases(CmsAliasSaveValidationRequest saveRequest) throws CmsRpcException {
1129
1130        CmsObject cms = getCmsObject();
1131        CmsAliasBulkEditHelper helper = new CmsAliasBulkEditHelper(cms);
1132        try {
1133            return helper.saveAliases(saveRequest);
1134        } catch (Exception e) {
1135            error(e);
1136            return null;
1137        }
1138    }
1139
1140    /**
1141     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#savePropertiesForLocaleCompareMode(org.opencms.util.CmsUUID, java.lang.String, java.util.List, boolean)
1142     */
1143    public void savePropertiesForLocaleCompareMode(
1144        CmsUUID id,
1145        String newUrlName,
1146        List<CmsPropertyModification> propertyChanges,
1147        boolean editedName)
1148    throws CmsRpcException {
1149
1150        try {
1151            CmsObject cms = getCmsObject();
1152            CmsResource ownRes = cms.readResource(id, CmsResourceFilter.IGNORE_EXPIRATION);
1153            CmsResource defaultFileRes = cms.readDefaultFile("" + id);
1154            boolean shallow = !editedName;
1155            try (AutoCloseable c = CmsLockUtil.withLockedResources(cms, shallow, ownRes, defaultFileRes)) {
1156                updateProperties(cms, ownRes, defaultFileRes, propertyChanges);
1157                if (editedName) {
1158                    String parent = CmsResource.getParentFolder(ownRes.getRootPath());
1159                    newUrlName = CmsFileUtil.removeTrailingSeparator(newUrlName);
1160                    String newPath = CmsStringUtil.joinPaths(parent, newUrlName);
1161                    CmsObject rootCms = OpenCms.initCmsObject(cms);
1162                    rootCms.getRequestContext().setSiteRoot("");
1163                    if (!CmsPath.equal(ownRes.getRootPath(), newPath)) {
1164                        rootCms.moveResource(ownRes.getRootPath(), newPath);
1165                    }
1166                }
1167            }
1168        } catch (Exception e) {
1169            error(e);
1170        }
1171    }
1172
1173    /**
1174     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#saveSitemapAttributes(org.opencms.util.CmsUUID, java.util.Map)
1175     */
1176    public void saveSitemapAttributes(CmsUUID rootId, Map<String, String> attributes) throws CmsRpcException {
1177
1178        CmsResource configResource = null;
1179        try {
1180            CmsObject cms = getCmsObject();
1181            CmsResource root = cms.readResource(rootId);
1182            String sitePath = cms.getSitePath(root);
1183            String configPath = CmsStringUtil.joinPaths(sitePath, "/.content/.config");
1184            configResource = cms.readResource(configPath);
1185            CmsXmlContent configContent = CmsXmlContentFactory.unmarshal(cms, cms.readFile(configResource));
1186            CmsSitemapAttributeUpdater updater = new CmsSitemapAttributeUpdater(cms, configContent);
1187            if (updater.saveAttributesFromEditorDialog(attributes)) {
1188                byte[] newContentBytes = configContent.marshal();
1189                CmsFile file = configContent.getFile();
1190                file.setContents(newContentBytes);
1191                cms.writeFile(file);
1192                OpenCms.getADEManager().waitForCacheUpdate(false);
1193            }
1194        } catch (Exception e) {
1195            error(e);
1196        } finally {
1197            try {
1198                if (configResource != null) {
1199                    getCmsObject().unlockResource(configResource);
1200                }
1201            } catch (Exception e) {
1202                LOG.debug(e.getLocalizedMessage(), e);
1203            }
1204        }
1205
1206    }
1207
1208    /**
1209     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#setDefaultModel(org.opencms.util.CmsUUID, org.opencms.util.CmsUUID)
1210     */
1211    public CmsModelInfo setDefaultModel(CmsUUID rootId, CmsUUID modelId) throws CmsRpcException {
1212
1213        CmsModelInfo result = null;
1214        try {
1215            CmsObject cms = getCmsObject();
1216            CmsResource rootResource = cms.readResource(rootId);
1217            CmsModelPageHelper modelPageHelper = new CmsModelPageHelper(getCmsObject(), rootResource);
1218            CmsResource configFile = OpenCms.getADEManager().lookupConfiguration(
1219                cms,
1220                rootResource.getRootPath()).getResource();
1221            ensureLock(configFile);
1222            result = modelPageHelper.setDefaultModelPage(configFile, modelId);
1223            tryUnlock(configFile);
1224        } catch (Throwable e) {
1225            error(e);
1226        }
1227        return result;
1228    }
1229
1230    /**
1231     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#setEditorMode(org.opencms.ade.sitemap.shared.CmsSitemapData.EditorMode)
1232     */
1233    public void setEditorMode(EditorMode editorMode) {
1234
1235        CmsADESessionCache.getCache(getRequest(), getCmsObject()).setSitemapEditorMode(editorMode);
1236    }
1237
1238    /**
1239     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#updateAliasEditorStatus(boolean)
1240     */
1241    public void updateAliasEditorStatus(boolean editing) {
1242
1243        CmsObject cms = getCmsObject();
1244        if (editing) {
1245            aliasEditorLockTable.update(cms, cms.getRequestContext().getSiteRoot());
1246        } else {
1247            aliasEditorLockTable.clear(cms, cms.getRequestContext().getSiteRoot());
1248        }
1249    }
1250
1251    /**
1252     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#validateAliases(org.opencms.gwt.shared.alias.CmsAliasEditValidationRequest)
1253     */
1254    public CmsAliasEditValidationReply validateAliases(CmsAliasEditValidationRequest validationRequest) {
1255
1256        CmsObject cms = getCmsObject();
1257        CmsAliasBulkEditHelper helper = new CmsAliasBulkEditHelper(cms);
1258        return helper.validateAliases(validationRequest);
1259    }
1260
1261    /**
1262     * @see org.opencms.ade.sitemap.shared.rpc.I_CmsSitemapService#validateRewriteAliases(org.opencms.gwt.shared.alias.CmsRewriteAliasValidationRequest)
1263     */
1264    public CmsRewriteAliasValidationReply validateRewriteAliases(CmsRewriteAliasValidationRequest validationRequest) {
1265
1266        CmsRewriteAliasValidationReply result = new CmsRewriteAliasValidationReply();
1267        for (CmsRewriteAliasTableRow editedRow : validationRequest.getEditedRewriteAliases()) {
1268            try {
1269                String patternString = editedRow.getPatternString();
1270                Pattern.compile(patternString);
1271            } catch (PatternSyntaxException e) {
1272                result.addError(editedRow.getId(), "Syntax error in regular expression: " + e.getMessage());
1273            }
1274        }
1275        return result;
1276    }
1277
1278    /**
1279     * Locks the given resource with a temporary, if not already locked by the current user.
1280     * Will throw an exception if the resource could not be locked for the current user.<p>
1281     *
1282     * @param resource the resource to lock
1283     *
1284     * @return the assigned lock
1285     *
1286     * @throws CmsException if the resource could not be locked
1287     */
1288    protected LockInfo ensureLockAndGetInfo(CmsResource resource) throws CmsException {
1289
1290        CmsObject cms = getCmsObject();
1291        boolean justLocked = false;
1292        List<CmsResource> blockingResources = cms.getBlockingLockedResources(resource);
1293        if ((blockingResources != null) && !blockingResources.isEmpty()) {
1294            throw new CmsException(
1295                org.opencms.gwt.Messages.get().container(
1296                    org.opencms.gwt.Messages.ERR_RESOURCE_HAS_BLOCKING_LOCKED_CHILDREN_1,
1297                    cms.getSitePath(resource)));
1298        }
1299        CmsUser user = cms.getRequestContext().getCurrentUser();
1300        CmsLock lock = cms.getLock(resource);
1301        if (!lock.isOwnedBy(user)) {
1302            cms.lockResourceTemporary(resource);
1303            lock = cms.getLock(resource);
1304            justLocked = true;
1305        } else if (!lock.isOwnedInProjectBy(user, cms.getRequestContext().getCurrentProject())) {
1306            cms.changeLock(resource);
1307            lock = cms.getLock(resource);
1308            justLocked = true;
1309        }
1310        return new LockInfo(lock, justLocked);
1311    }
1312
1313    /**
1314     * Internal method for saving a sitemap.<p>
1315     *
1316     * @param entryPoint the URI of the sitemap to save
1317     * @param change the change to save
1318     *
1319     * @return list of changed sitemap entries
1320     *
1321     * @throws CmsException if something goes wrong
1322     */
1323    protected CmsSitemapChange saveInternal(String entryPoint, CmsSitemapChange change) throws CmsException {
1324
1325        ensureSession();
1326        switch (change.getChangeType()) {
1327            case clipboardOnly:
1328                // do nothing
1329                break;
1330            case remove:
1331                change = removeEntryFromNavigation(change);
1332                break;
1333            case undelete:
1334                change = undelete(change);
1335                break;
1336            default:
1337                change = applyChange(entryPoint, change);
1338        }
1339        setClipboardData(change.getClipBoardData());
1340        return change;
1341    }
1342
1343    /**
1344     * Gets the properties of a resource as a map of client properties.<p>
1345     *
1346     * @param cms the CMS context to use
1347     * @param res the resource whose properties to read
1348     * @param search true if the inherited properties should be read
1349     *
1350     * @return the client properties as a map
1351     *
1352     * @throws CmsException if something goes wrong
1353     */
1354    Map<String, CmsClientProperty> getClientProperties(CmsObject cms, CmsResource res, boolean search)
1355    throws CmsException {
1356
1357        List<CmsProperty> props = cms.readPropertyObjects(res, search);
1358        Map<String, CmsClientProperty> result = createClientProperties(props, false);
1359        return result;
1360    }
1361
1362    /**
1363     * Adds a function detail element to a container page.<p>
1364     *
1365     * @param cms the current CMS context
1366     * @param page the container page which should be changed
1367     * @param containerName the name of the container which should be used for function detail elements
1368     * @param elementId the structure id of the element to add
1369     * @param formatterId the structure id of the formatter for the element
1370     *
1371     * @throws CmsException if something goes wrong
1372     */
1373    private void addFunctionDetailElement(
1374        CmsObject cms,
1375        CmsXmlContainerPage page,
1376        String containerName,
1377        CmsUUID elementId,
1378        CmsUUID formatterId)
1379    throws CmsException {
1380
1381        CmsContainerPageBean bean = page.getContainerPage(cms);
1382        List<CmsContainerBean> containerBeans = new ArrayList<CmsContainerBean>();
1383        Collection<CmsContainerBean> originalContainers = bean.getContainers().values();
1384        if ((containerName == null) && !originalContainers.isEmpty()) {
1385            CmsContainerBean firstContainer = originalContainers.iterator().next();
1386            containerName = firstContainer.getName();
1387        }
1388        boolean foundContainer = false;
1389        for (CmsContainerBean cntBean : originalContainers) {
1390            boolean isDetailTarget = cntBean.getName().equals(containerName);
1391            if (isDetailTarget && !foundContainer) {
1392                foundContainer = true;
1393                List<CmsContainerElementBean> newElems = new ArrayList<CmsContainerElementBean>();
1394                newElems.addAll(cntBean.getElements());
1395                CmsContainerElementBean newElement = new CmsContainerElementBean(
1396                    elementId,
1397                    formatterId,
1398                    new HashMap<String, String>(),
1399                    false);
1400                newElems.add(0, newElement);
1401                CmsContainerBean newCntBean = cntBean.copyWithNewElements(newElems);
1402                containerBeans.add(newCntBean);
1403            } else {
1404                containerBeans.add(cntBean);
1405            }
1406        }
1407        if (!foundContainer) {
1408            throw new CmsException(
1409                Messages.get().container(Messages.ERR_NO_FUNCTION_DETAIL_CONTAINER_1, page.getFile().getRootPath()));
1410        }
1411        CmsContainerPageBean bean2 = new CmsContainerPageBean(new ArrayList<CmsContainerBean>(containerBeans));
1412        page.writeContainerPage(cms, bean2);
1413
1414    }
1415
1416    /**
1417     * Applys the given change to the VFS.<p>
1418     *
1419     * @param entryPoint the sitemap entry-point
1420     * @param change the change
1421     *
1422     * @return the updated entry
1423     *
1424     * @throws CmsException if something goes wrong
1425     */
1426    private CmsSitemapChange applyChange(String entryPoint, CmsSitemapChange change) throws CmsException {
1427
1428        CmsObject cms = getCmsObject();
1429        CmsResource configFile = null;
1430        // lock the config file first, to avoid half done changes
1431        if (change.hasDetailPageInfos()) {
1432            CmsADEConfigData configData = OpenCms.getADEManager().lookupConfiguration(
1433                cms,
1434                cms.getRequestContext().addSiteRoot(entryPoint));
1435            if (!configData.isModuleConfiguration() && (configData.getResource() != null)) {
1436                configFile = configData.getResource();
1437            }
1438            if (configFile != null) {
1439                ensureLock(configFile);
1440            }
1441        }
1442        if (change.isNew()) {
1443            CmsClientSitemapEntry newEntry = createNewEntry(entryPoint, change);
1444            change.setUpdatedEntry(newEntry);
1445            change.setEntryId(newEntry.getId());
1446        } else if (change.getChangeType() == ChangeType.delete) {
1447            delete(change);
1448        } else if (change.getEntryId() != null) {
1449            modifyEntry(change);
1450        }
1451        if (change.hasDetailPageInfos() && (configFile != null)) {
1452            saveDetailPages(change.getDetailPageInfos(), configFile, change.getEntryId(), change.getUpdatedEntry());
1453            tryUnlock(configFile);
1454        }
1455
1456        return change;
1457    }
1458
1459    /**
1460     * Changes the navigation for a moved entry and its neighbors.<p>
1461     *
1462     * @param change the sitemap change
1463     * @param entryFolder the moved entry
1464     *
1465     * @throws CmsException if something goes wrong
1466     */
1467    private void applyNavigationChanges(CmsSitemapChange change, CmsResource entryFolder) throws CmsException {
1468
1469        CmsObject cms = getCmsObject();
1470        String parentPath = null;
1471        if (change.hasNewParent()) {
1472            CmsResource parent = cms.readResource(change.getParentId());
1473            parentPath = cms.getSitePath(parent);
1474        } else {
1475            parentPath = CmsResource.getParentFolder(cms.getSitePath(entryFolder));
1476        }
1477        List<CmsJspNavElement> navElements = getNavBuilder().getNavigationForFolder(
1478            parentPath,
1479            Visibility.all,
1480            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
1481        CmsSitemapNavPosCalculator npc = new CmsSitemapNavPosCalculator(navElements, entryFolder, change.getPosition());
1482        List<CmsJspNavElement> navs = npc.getNavigationChanges();
1483        List<CmsResource> needToUnlock = new ArrayList<CmsResource>();
1484
1485        try {
1486            for (CmsJspNavElement nav : navs) {
1487                LockInfo lockInfo = ensureLockAndGetInfo(nav.getResource());
1488                if (!nav.getResource().equals(entryFolder) && lockInfo.wasJustLocked()) {
1489                    needToUnlock.add(nav.getResource());
1490                }
1491            }
1492            for (CmsJspNavElement nav : navs) {
1493                CmsProperty property = new CmsProperty(
1494                    CmsPropertyDefinition.PROPERTY_NAVPOS,
1495                    "" + nav.getNavPosition(),
1496                    null);
1497                cms.writePropertyObject(cms.getSitePath(nav.getResource()), property);
1498            }
1499        } finally {
1500            for (CmsResource lockedRes : needToUnlock) {
1501                try {
1502                    cms.unlockResource(lockedRes);
1503                } catch (CmsException e) {
1504                    // we catch this because we still want to unlock the other resources
1505                    LOG.error(e.getLocalizedMessage(), e);
1506                }
1507            }
1508        }
1509    }
1510
1511    /**
1512     * Builds the list info bean for a resource type which can be used as a sub-sitemap folder.<p>
1513     *
1514     * @param sitemapType the sitemap folder type
1515     *
1516     * @return the list info bean for the given type
1517     */
1518    private CmsListInfoBean buildSitemapTypeInfo(I_CmsResourceType sitemapType) {
1519
1520        CmsListInfoBean result = new CmsListInfoBean();
1521        CmsWorkplaceManager wm = OpenCms.getWorkplaceManager();
1522        String typeName = sitemapType.getTypeName();
1523        Locale wpLocale = wm.getWorkplaceLocale(getCmsObject());
1524        String title = typeName;
1525        String description = "";
1526        try {
1527            title = CmsWorkplaceMessages.getResourceTypeName(wpLocale, typeName);
1528        } catch (Throwable e) {
1529            LOG.warn(e.getLocalizedMessage(), e);
1530        }
1531        try {
1532            description = CmsWorkplaceMessages.getResourceTypeDescription(wpLocale, typeName);
1533        } catch (Throwable e) {
1534            LOG.warn(e.getLocalizedMessage(), e);
1535        }
1536        result.setResourceType(typeName);
1537        result.setTitle(title);
1538        result.setSubTitle(description);
1539        return result;
1540    }
1541
1542    /**
1543     * Returns all available gallery folder resource types.<p>
1544     *
1545     * @return the gallery folder resource types
1546     */
1547    @SuppressWarnings("deprecation")
1548    private List<CmsGalleryType> collectGalleryTypes() {
1549
1550        Locale wpLocale = OpenCms.getWorkplaceManager().getWorkplaceLocale(getCmsObject());
1551        Map<String, List<String>> galleryContentTypes = new HashMap<String, List<String>>();
1552        List<CmsGalleryType> result = new ArrayList<CmsGalleryType>();
1553        for (I_CmsResourceType resourceType : OpenCms.getResourceManager().getResourceTypes()) {
1554            if (resourceType instanceof CmsResourceTypeFolderExtended) {
1555                // found a configured extended folder resource type
1556                CmsResourceTypeFolderExtended galleryType = (CmsResourceTypeFolderExtended)resourceType;
1557                String folderClassName = galleryType.getFolderClassName();
1558                if (CmsStringUtil.isNotEmpty(folderClassName)) {
1559                    // only process this as a gallery if the folder name is not empty
1560                    try {
1561                        // check, if the folder class is a subclass of A_CmsGallery
1562                        if (A_CmsAjaxGallery.class.isAssignableFrom(Class.forName(folderClassName))) {
1563                            CmsGalleryType gallery = new CmsGalleryType();
1564                            gallery.setTypeId(resourceType.getTypeId());
1565                            gallery.setResourceType(resourceType.getTypeName());
1566                            gallery.setTitle(
1567                                CmsWorkplaceMessages.getResourceTypeName(wpLocale, resourceType.getTypeName()));
1568                            gallery.setSubTitle(
1569                                CmsWorkplaceMessages.getResourceTypeDescription(wpLocale, resourceType.getTypeName()));
1570                            gallery.setBigIconClasses(
1571                                CmsIconUtil.getIconClasses(resourceType.getTypeName(), null, false));
1572                            result.add(gallery);
1573                        }
1574                    } catch (Exception e) {
1575                        this.log(e.getLocalizedMessage(), e);
1576                    }
1577                }
1578            } else {
1579                List<I_CmsResourceType> galleryTypes = resourceType.getGalleryTypes();
1580                if ((galleryTypes != null) && !galleryTypes.isEmpty()) {
1581                    for (I_CmsResourceType galleryType : galleryTypes) {
1582                        List<String> typeList = galleryContentTypes.get(galleryType.getTypeName());
1583                        if (typeList == null) {
1584                            typeList = new ArrayList<String>();
1585                            galleryContentTypes.put(galleryType.getTypeName(), typeList);
1586                        }
1587                        typeList.add(resourceType.getTypeName());
1588                    }
1589                }
1590            }
1591        }
1592        for (CmsGalleryType galleryType : result) {
1593            galleryType.setContentTypeNames(galleryContentTypes.get(galleryType.getResourceType()));
1594        }
1595        return result;
1596    }
1597
1598    /**
1599     * Gets the information for the available sitemap folder types.<p>
1600     *
1601     * @param cms the current CMS context
1602     * @param configData the configuration data for the current subsitemap
1603     * @return the list info beans corresponding to available sitemap folder types
1604     */
1605    private List<CmsListInfoBean> collectSitemapTypeInfos(CmsObject cms, CmsADEConfigData configData) {
1606
1607        List<CmsListInfoBean> subsitemapFolderTypeInfos = new ArrayList<CmsListInfoBean>();
1608        List<Integer> sitemapTypeIds = CmsResourceTypeFolderSubSitemap.getSubSitemapResourceTypeIds();
1609        String checkPath = configData.getBasePath();
1610        if (checkPath != null) {
1611            checkPath = cms.getRequestContext().removeSiteRoot(checkPath);
1612        } else {
1613            checkPath = "/";
1614        }
1615        CmsResource checkResource = null;
1616        try {
1617            checkResource = cms.readResource(checkPath);
1618        } catch (CmsException e) {
1619            LOG.warn(e.getLocalizedMessage(), e);
1620        }
1621
1622        for (Integer typeId : sitemapTypeIds) {
1623            try {
1624                I_CmsResourceType sitemapType = OpenCms.getResourceManager().getResourceType(typeId.intValue());
1625                String typeName = sitemapType.getTypeName();
1626                CmsExplorerTypeSettings explorerType = OpenCms.getWorkplaceManager().getExplorerTypeSetting(typeName);
1627
1628                // if explorer type and check resource available, perform a permission check
1629                if ((explorerType != null) && (checkResource != null)) {
1630                    try {
1631                        CmsExplorerTypeAccess access = explorerType.getAccess();
1632                        if (!access.getPermissions(cms, checkResource).requiresControlPermission()) {
1633                            continue;
1634                        }
1635                    } catch (Exception e) {
1636                        LOG.error(e.getLocalizedMessage(), e);
1637                    }
1638                }
1639                CmsListInfoBean infoBean = buildSitemapTypeInfo(sitemapType);
1640                subsitemapFolderTypeInfos.add(infoBean);
1641
1642            } catch (CmsLoaderException e) {
1643                LOG.warn("Could not read sitemap folder type " + typeId + ": " + e.getLocalizedMessage(), e);
1644            }
1645        }
1646        return subsitemapFolderTypeInfos;
1647    }
1648
1649    /**
1650     * Creates a client alias bean from a server-side alias.<p>
1651     *
1652     * @param cms the current CMS context
1653     * @param alias the alias to convert
1654     *
1655     * @return the alias table row
1656     *
1657     * @throws CmsException if something goes wrong
1658     */
1659    private CmsAliasTableRow createAliasTableEntry(CmsObject cms, CmsAlias alias) throws CmsException {
1660
1661        CmsResource resource = cms.readResource(alias.getStructureId(), CmsResourceFilter.ALL);
1662        CmsAliasTableRow result = new CmsAliasTableRow();
1663        result.setStructureId(alias.getStructureId());
1664        result.setOriginalStructureId(alias.getStructureId());
1665        result.setAliasPath(alias.getAliasPath());
1666        result.setResourcePath(cms.getSitePath(resource));
1667        result.setKey((new CmsUUID()).toString());
1668        result.setMode(alias.getMode());
1669        result.setChanged(false);
1670        return result;
1671    }
1672
1673    /**
1674     * Creates a navigation level type info.<p>
1675     *
1676     * @return the navigation level type info bean
1677     *
1678     * @throws CmsException if reading the sub level redirect copy page fails
1679     */
1680    private CmsNewResourceInfo createNavigationLevelTypeInfo() throws CmsException {
1681
1682        String name = CmsResourceTypeFolder.getStaticTypeName();
1683        Locale locale = getWorkplaceLocale();
1684        String subtitle = Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_NAVIGATION_LEVEL_SUBTITLE_0);
1685        if (CmsStringUtil.isEmptyOrWhitespaceOnly(subtitle)) {
1686            subtitle = CmsWorkplaceMessages.getResourceTypeName(locale, name);
1687        }
1688        CmsNewResourceInfo result = new CmsNewResourceInfo(
1689            CmsResourceTypeFolder.getStaticTypeId(),
1690            name,
1691            Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_NAVIGATION_LEVEL_TITLE_0),
1692            subtitle,
1693            getCmsObject().readResource(SUB_LEVEL_REDIRECT_COPY_PAGE).getStructureId(),
1694            false,
1695            subtitle);
1696        result.setBigIconClasses(CmsIconUtil.ICON_NAV_LEVEL_BIG);
1697        result.setCreateParameter(CmsJspNavBuilder.NAVIGATION_LEVEL_FOLDER);
1698        return result;
1699    }
1700
1701    /**
1702     * Creates new content elements if required by the model page.<p>
1703     *
1704     * @param cms the cms context
1705     * @param page the page
1706     * @param sitePath the resource site path
1707     *
1708     * @throws CmsException when unable to create the content elements
1709     */
1710    private boolean createNewContainerElements(CmsObject cms, CmsMutableContainerPage page, String sitePath)
1711    throws CmsException {
1712
1713        boolean needsChanges = false;
1714        CmsADEConfigData configData = OpenCms.getADEManager().lookupConfiguration(cms, cms.addSiteRoot(sitePath));
1715        Locale contentLocale = OpenCms.getLocaleManager().getDefaultLocale(cms, CmsResource.getFolderPath(sitePath));
1716        CmsObject cloneCms = OpenCms.initCmsObject(cms);
1717        cloneCms.getRequestContext().setLocale(contentLocale);
1718        for (CmsMutableContainer container : page.containers()) {
1719            ListIterator<CmsContainerElementBean> iter = container.elements().listIterator();
1720            while (iter.hasNext()) {
1721                CmsContainerElementBean elem = iter.next();
1722                if (elem.isCreateNew() && !elem.isGroupContainer(cms) && !elem.isInheritedContainer(cms)) {
1723                    String typeName = OpenCms.getResourceManager().getResourceType(elem.getResource()).getTypeName();
1724                    CmsResourceTypeConfig typeConfig = configData.getResourceType(typeName);
1725                    if (typeConfig == null) {
1726                        throw new IllegalArgumentException(
1727                            "Can not copy template model element '"
1728                                + elem.getResource().getRootPath()
1729                                + "' because the resource type '"
1730                                + typeName
1731                                + "' is not available in this sitemap.");
1732                    }
1733
1734                    CmsResource newResource = typeConfig.createNewElement(
1735                        cloneCms,
1736                        elem.getResource(),
1737                        CmsResource.getParentFolder(cms.getRequestContext().addSiteRoot(sitePath)));
1738                    CmsContainerElementBean newBean = new CmsContainerElementBean(
1739                        newResource.getStructureId(),
1740                        elem.getFormatterId(),
1741                        elem.getIndividualSettings(),
1742                        false);
1743                    iter.set(newBean);
1744                    needsChanges = true;
1745                }
1746            }
1747        }
1748        return needsChanges;
1749    }
1750
1751    /**
1752     * Creates new content elements if required by the model page.<p>
1753     *
1754     * @param cms the cms context
1755     * @param page the page
1756     * @param sitePath the resource site path
1757     *
1758     * @throws CmsException when unable to create the content elements
1759     */
1760    private void createNewContainerElements(CmsObject cms, CmsXmlContainerPage page, String sitePath)
1761    throws CmsException {
1762
1763        CmsADEConfigData configData = OpenCms.getADEManager().lookupConfiguration(cms, cms.addSiteRoot(sitePath));
1764        Locale contentLocale = OpenCms.getLocaleManager().getDefaultLocale(cms, CmsResource.getFolderPath(sitePath));
1765        CmsObject cloneCms = OpenCms.initCmsObject(cms);
1766        cloneCms.getRequestContext().setLocale(contentLocale);
1767        CmsContainerPageBean pageBean = page.getContainerPage(cms);
1768        boolean needsChanges = false;
1769        List<CmsContainerBean> updatedContainers = new ArrayList<CmsContainerBean>();
1770        for (CmsContainerBean container : pageBean.getContainers().values()) {
1771            List<CmsContainerElementBean> updatedElements = new ArrayList<CmsContainerElementBean>();
1772            for (CmsContainerElementBean element : container.getElements()) {
1773                if (element.isCreateNew() && !element.isGroupContainer(cms) && !element.isInheritedContainer(cms)) {
1774                    needsChanges = true;
1775                    String typeName = OpenCms.getResourceManager().getResourceType(element.getResource()).getTypeName();
1776                    CmsResourceTypeConfig typeConfig = configData.getResourceType(typeName);
1777                    if (typeConfig == null) {
1778                        throw new IllegalArgumentException(
1779                            "Can not copy template model element '"
1780                                + element.getResource().getRootPath()
1781                                + "' because the resource type '"
1782                                + typeName
1783                                + "' is not available in this sitemap.");
1784                    }
1785
1786                    CmsResource newResource = typeConfig.createNewElement(
1787                        cloneCms,
1788                        element.getResource(),
1789                        CmsResource.getParentFolder(cms.getRequestContext().addSiteRoot(sitePath)));
1790                    CmsContainerElementBean newBean = new CmsContainerElementBean(
1791                        newResource.getStructureId(),
1792                        element.getFormatterId(),
1793                        element.getIndividualSettings(),
1794                        false);
1795                    updatedElements.add(newBean);
1796                } else {
1797                    updatedElements.add(element);
1798                }
1799            }
1800
1801            CmsContainerBean updatedContainer = container.copyWithNewElements(updatedElements);
1802            updatedContainers.add(updatedContainer);
1803        }
1804        if (needsChanges) {
1805            CmsContainerPageBean updatedPage = new CmsContainerPageBean(updatedContainers);
1806            page.writeContainerPage(cms, updatedPage);
1807        }
1808    }
1809
1810    /**
1811     * Creates a new page in navigation.<p>
1812     *
1813     * @param entryPoint the site-map entry-point
1814     * @param change the new change
1815     *
1816     * @return the updated entry
1817     *
1818     * @throws CmsException if something goes wrong
1819     */
1820    @SuppressWarnings("deprecation")
1821    private CmsClientSitemapEntry createNewEntry(String entryPoint, CmsSitemapChange change) throws CmsException {
1822
1823        CmsObject cms = getCmsObject();
1824        CmsClientSitemapEntry newEntry = null;
1825        if (change.getParentId() != null) {
1826            CmsResource parentFolder = cms.readResource(
1827                change.getParentId(),
1828                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
1829            String entryPath = "";
1830            CmsResource entryFolder = null;
1831            CmsResource newRes = null;
1832            byte[] content = null;
1833            List<CmsProperty> properties = Collections.emptyList();
1834            CmsResource copyPage = null;
1835            if (change.getNewCopyResourceId() != null) {
1836                copyPage = cms.readResource(change.getNewCopyResourceId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
1837                content = cms.readFile(copyPage).getContents();
1838                properties = cms.readPropertyObjects(copyPage, false);
1839            }
1840            List<CmsProperty> filteredProperties = new ArrayList<CmsProperty>();
1841            for (CmsProperty property : properties) {
1842                boolean filter = false;
1843                if (FILTER_PROPERTIES.contains(property.getName())) {
1844                    filter = true;
1845
1846                } else {
1847                    // filter localized versions also
1848                    for (String filterProp : FILTER_PROPERTIES) {
1849                        if (property.getName().startsWith(filterProp + "_")) {
1850                            filter = true;
1851                            break;
1852                        }
1853                    }
1854                }
1855                if (!filter) {
1856                    filteredProperties.add(property);
1857                }
1858            }
1859            properties = filteredProperties;
1860            if (isRedirectType(change.getNewResourceTypeId())) {
1861                entryPath = CmsStringUtil.joinPaths(cms.getSitePath(parentFolder), change.getName());
1862                newRes = cms.createResource(
1863                    entryPath,
1864                    change.getNewResourceTypeId(),
1865                    null,
1866                    Collections.singletonList(
1867                        new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, change.getName(), null)));
1868                cms.writePropertyObjects(newRes, generateInheritProperties(change, newRes));
1869                applyNavigationChanges(change, newRes);
1870            } else {
1871                String entryFolderPath = CmsStringUtil.joinPaths(cms.getSitePath(parentFolder), change.getName() + "/");
1872                boolean idWasNull = change.getEntryId() == null;
1873                // we don't really need to create a folder object here anymore.
1874                if (idWasNull) {
1875                    // need this for calculateNavPosition, even though the id will get overwritten
1876                    change.setEntryId(new CmsUUID());
1877                }
1878
1879                boolean isFunctionDetail = false;
1880                boolean isNavigationLevel = false;
1881                if (change.getCreateParameter() != null) {
1882                    if (CmsUUID.isValidUUID(change.getCreateParameter())) {
1883                        isFunctionDetail = true;
1884                    } else if (CmsJspNavBuilder.NAVIGATION_LEVEL_FOLDER.equals(change.getCreateParameter())) {
1885                        isNavigationLevel = true;
1886                    }
1887                }
1888                String folderType = CmsResourceTypeFolder.getStaticTypeName();
1889                String createSitemapFolderType = change.getCreateSitemapFolderType();
1890                if (createSitemapFolderType != null) {
1891                    folderType = createSitemapFolderType;
1892                }
1893
1894                int folderTypeId = OpenCms.getResourceManager().getResourceType(folderType).getTypeId();
1895                entryFolder = new CmsResource(
1896                    change.getEntryId(),
1897                    new CmsUUID(),
1898                    entryFolderPath,
1899                    folderTypeId,
1900                    true,
1901                    0,
1902                    cms.getRequestContext().getCurrentProject().getUuid(),
1903                    CmsResource.STATE_NEW,
1904                    System.currentTimeMillis(),
1905                    cms.getRequestContext().getCurrentUser().getId(),
1906                    System.currentTimeMillis(),
1907                    cms.getRequestContext().getCurrentUser().getId(),
1908                    CmsResource.DATE_RELEASED_DEFAULT,
1909                    CmsResource.DATE_EXPIRED_DEFAULT,
1910                    1,
1911                    0,
1912                    System.currentTimeMillis(),
1913                    0);
1914                List<CmsProperty> folderProperties = generateInheritProperties(change, entryFolder);
1915                if (isNavigationLevel) {
1916                    folderProperties.add(
1917                        new CmsProperty(
1918                            CmsPropertyDefinition.PROPERTY_DEFAULT_FILE,
1919                            CmsJspNavBuilder.NAVIGATION_LEVEL_FOLDER,
1920                            null));
1921                }
1922                entryFolder = cms.createResource(entryFolderPath, folderTypeId, null, folderProperties);
1923                if (createSitemapFolderType != null) {
1924                    createSitemapContentFolder(cms, entryFolder);
1925                }
1926                if (idWasNull) {
1927                    change.setEntryId(entryFolder.getStructureId());
1928                }
1929                applyNavigationChanges(change, entryFolder);
1930                entryPath = CmsStringUtil.joinPaths(entryFolderPath, "index.html");
1931                boolean isContainerPage = change.getNewResourceTypeId() == CmsResourceTypeXmlContainerPage.getContainerPageTypeIdSafely();
1932                if (isContainerPage && (copyPage != null)) {
1933
1934                    // do *NOT* get this from the cache, because we perform some destructive operation on the XML content
1935                    CmsXmlContainerPage page = CmsXmlContainerPageFactory.unmarshal(
1936                        cms,
1937                        cms.readFile(copyPage),
1938                        true,
1939                        true);
1940                    CmsContainerPageWrapper wrapper = new CmsContainerPageWrapper(cms, page);
1941                    if (isFunctionDetail) {
1942                        String functionDetailContainer = getFunctionDetailContainerName(parentFolder);
1943                        if (functionDetailContainer != null) {
1944                            CmsUUID functionStructureId = new CmsUUID(change.getCreateParameter());
1945                            CmsResource functionRes = cms.readResource(
1946                                functionStructureId,
1947                                CmsResourceFilter.IGNORE_EXPIRATION);
1948                            CmsResource functionFormatter;
1949                            if (OpenCms.getResourceManager().matchResourceType(
1950                                CmsResourceTypeFunctionConfig.TYPE_NAME,
1951                                functionRes.getTypeId())) {
1952                                functionFormatter = cms.readResource(CmsResourceTypeFunctionConfig.FORMATTER_PATH);
1953                            } else {
1954                                functionFormatter = cms.readResource(
1955                                    CmsResourceTypeFunctionConfig.FORMATTER_PATH,
1956                                    CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
1957                            }
1958                            if (!wrapper.addElementToContainer(
1959                                functionDetailContainer,
1960                                new CmsContainerElementBean(
1961                                    functionStructureId,
1962                                    functionFormatter.getStructureId(),
1963                                    new HashMap<>(),
1964                                    false))) {
1965
1966                                throw new CmsException(
1967                                    Messages.get().container(
1968                                        Messages.ERR_NO_FUNCTION_DETAIL_CONTAINER_1,
1969                                        page.getFile().getRootPath()));
1970                            }
1971                        } else {
1972                            LOG.debug("function detail container is null for " + parentFolder.getRootPath());
1973                        }
1974                    }
1975                    createNewContainerElements(cms, wrapper.page(), entryPath);
1976                    content = wrapper.marshal();
1977                }
1978                newRes = cms.createResource(
1979                    entryPath,
1980                    copyPage != null ? copyPage.getTypeId() : change.getNewResourceTypeId(),
1981                    content,
1982                    properties);
1983                cms.writePropertyObjects(newRes, generateOwnProperties(change));
1984            }
1985
1986            if (entryFolder != null) {
1987                tryUnlock(entryFolder);
1988                String sitePath = cms.getSitePath(entryFolder);
1989                newEntry = toClientEntry(
1990                    getNavBuilder().getNavigationForResource(sitePath, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED),
1991                    false);
1992                newEntry.setSubEntries(getChildren(sitePath, 1, null), null);
1993                newEntry.setChildrenLoadedInitially(true);
1994            }
1995            if (newRes != null) {
1996                tryUnlock(newRes);
1997            }
1998            if (newEntry == null) {
1999                newEntry = toClientEntry(
2000                    getNavBuilder().getNavigationForResource(
2001                        cms.getSitePath(newRes),
2002                        CmsResourceFilter.ONLY_VISIBLE_NO_DELETED),
2003                    false);
2004            }
2005            // mark position as not set
2006            newEntry.setPosition(-1);
2007            newEntry.setNew(true);
2008            change.getClipBoardData().getModifications().remove(null);
2009            change.getClipBoardData().getModifications().put(newEntry.getId(), newEntry);
2010        }
2011        return newEntry;
2012    }
2013
2014    /**
2015     * Creates a new resource info to a given model page resource.<p>
2016     *
2017     * @param cms the current CMS context
2018     * @param modelResource the model page resource
2019     * @param locale the locale used for retrieving descriptions/titles
2020     *
2021     * @return the new resource info
2022     *
2023     * @throws CmsException if something goes wrong
2024     */
2025    private CmsNewResourceInfo createNewResourceInfo(CmsObject cms, CmsResource modelResource, Locale locale)
2026    throws CmsException {
2027
2028        // if model page got overwritten by another resource, reread from site path
2029        if (!cms.existsResource(modelResource.getStructureId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED)) {
2030            modelResource = cms.readResource(cms.getSitePath(modelResource), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2031        }
2032        int typeId = modelResource.getTypeId();
2033        String name = OpenCms.getResourceManager().getResourceType(typeId).getTypeName();
2034        String title = cms.readPropertyObject(
2035            modelResource,
2036            CmsPropertyDefinition.PROPERTY_TITLE,
2037            false,
2038            locale).getValue();
2039        String description = cms.readPropertyObject(
2040            modelResource,
2041            CmsPropertyDefinition.PROPERTY_DESCRIPTION,
2042            false,
2043            locale).getValue();
2044
2045        boolean editable = false;
2046        try {
2047            CmsResource freshModelResource = cms.readResource(
2048                modelResource.getStructureId(),
2049                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2050            editable = cms.hasPermissions(
2051                freshModelResource,
2052                CmsPermissionSet.ACCESS_WRITE,
2053                false,
2054                CmsResourceFilter.DEFAULT);
2055        } catch (CmsException e) {
2056            LOG.warn(e.getLocalizedMessage(), e);
2057        }
2058        CmsNewResourceInfo info = new CmsNewResourceInfo(
2059            typeId,
2060            name,
2061            title,
2062            description,
2063            modelResource.getStructureId(),
2064            editable,
2065            description);
2066        info.setBigIconClasses(CmsIconUtil.getIconClasses(name, null, false));
2067        Float navpos = null;
2068        try {
2069            CmsProperty navposProp = cms.readPropertyObject(modelResource, CmsPropertyDefinition.PROPERTY_NAVPOS, true);
2070            String navposStr = navposProp.getValue();
2071            if (navposStr != null) {
2072                try {
2073                    navpos = Float.valueOf(navposStr);
2074                } catch (NumberFormatException e) {
2075                    // noop
2076                }
2077            }
2078        } catch (CmsException e) {
2079            LOG.warn(e.getLocalizedMessage(), e);
2080        }
2081
2082        info.setNavPos(navpos);
2083        info.setDate(
2084            CmsDateUtil.getDate(new Date(modelResource.getDateLastModified()), DateFormat.LONG, getWorkplaceLocale()));
2085        info.setVfsPath(modelResource.getRootPath());
2086        return info;
2087    }
2088
2089    /**
2090     * Creates a resource type info bean for a given resource type.<p>
2091     *
2092     * @param resType the resource type
2093     * @param copyResource the structure id of the copy resource
2094     *
2095     * @return the resource type info bean
2096     */
2097    @SuppressWarnings("deprecation")
2098    private CmsNewResourceInfo createResourceTypeInfo(I_CmsResourceType resType, CmsResource copyResource) {
2099
2100        String name = resType.getTypeName();
2101        Locale locale = getWorkplaceLocale();
2102        String subtitle = CmsWorkplaceMessages.getResourceTypeDescription(locale, name);
2103        if (CmsStringUtil.isEmptyOrWhitespaceOnly(subtitle)) {
2104            subtitle = CmsWorkplaceMessages.getResourceTypeName(locale, name);
2105        }
2106        if (copyResource != null) {
2107            CmsNewResourceInfo info = new CmsNewResourceInfo(
2108                copyResource.getTypeId(),
2109                name,
2110                CmsWorkplaceMessages.getResourceTypeName(locale, name),
2111                CmsWorkplaceMessages.getResourceTypeDescription(locale, name),
2112                copyResource.getStructureId(),
2113                false,
2114                subtitle);
2115            info.setBigIconClasses(CmsIconUtil.getIconClasses(name, null, false));
2116            return info;
2117        } else {
2118            CmsNewResourceInfo info = new CmsNewResourceInfo(
2119                resType.getTypeId(),
2120                name,
2121                CmsWorkplaceMessages.getResourceTypeName(locale, name),
2122                CmsWorkplaceMessages.getResourceTypeDescription(locale, name),
2123                null,
2124                false,
2125                subtitle);
2126            info.setBigIconClasses(CmsIconUtil.getIconClasses(name, null, false));
2127            return info;
2128        }
2129    }
2130
2131    /**
2132     * Helper method for creating the .content folder of a sub-sitemap.<p>
2133     *
2134     * @param cms the current CMS context
2135     * @param subSitemapFolder the sub-sitemap folder in which the .content folder should be created
2136     *
2137     * @throws CmsException if something goes wrong
2138     */
2139    @SuppressWarnings("deprecation")
2140    private void createSitemapContentFolder(CmsObject cms, CmsResource subSitemapFolder) throws CmsException {
2141
2142        String sitePath = cms.getSitePath(subSitemapFolder);
2143        String folderName = CmsStringUtil.joinPaths(sitePath, CmsADEManager.CONTENT_FOLDER_NAME + "/");
2144        String sitemapConfigName = CmsStringUtil.joinPaths(folderName, CmsADEManager.CONFIG_FILE_NAME);
2145        if (!cms.existsResource(folderName)) {
2146            cms.createResource(
2147                folderName,
2148                OpenCms.getResourceManager().getResourceType(CmsADEManager.CONFIG_FOLDER_TYPE));
2149        }
2150        I_CmsResourceType configType = OpenCms.getResourceManager().getResourceType(CmsADEManager.CONFIG_TYPE);
2151        if (cms.existsResource(sitemapConfigName)) {
2152            CmsResource configFile = cms.readResource(sitemapConfigName);
2153            if (configFile.getTypeId() != configType.getTypeId()) {
2154                throw new CmsException(
2155                    Messages.get().container(
2156                        Messages.ERR_CREATING_SUB_SITEMAP_WRONG_CONFIG_FILE_TYPE_2,
2157                        sitemapConfigName,
2158                        CmsADEManager.CONFIG_TYPE));
2159            }
2160        } else {
2161            cms.createResource(
2162                sitemapConfigName,
2163                OpenCms.getResourceManager().getResourceType(CmsADEManager.CONFIG_TYPE));
2164        }
2165
2166    }
2167
2168    /**
2169     * Deletes a resource according to the change data.<p>
2170     *
2171     * @param change the change data
2172     *
2173     * @return CmsClientSitemapEntry always null
2174     *
2175     *
2176     * @throws CmsException if something goes wrong
2177     */
2178    private CmsClientSitemapEntry delete(CmsSitemapChange change) throws CmsException {
2179
2180        CmsObject cms = getCmsObject();
2181        CmsResource resource = cms.readResource(change.getEntryId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2182        ensureLock(resource);
2183        cms.deleteResource(cms.getSitePath(resource), CmsResource.DELETE_PRESERVE_SIBLINGS);
2184        tryUnlock(resource);
2185        return null;
2186    }
2187
2188    /**
2189     * Generates a client side lock info object representing the current lock state of the given resource.<p>
2190     *
2191     * @param resource the resource
2192     *
2193     * @return the client lock
2194     *
2195     * @throws CmsException if something goes wrong
2196     */
2197    private CmsClientLock generateClientLock(CmsResource resource) throws CmsException {
2198
2199        CmsObject cms = getCmsObject();
2200        CmsLock lock = cms.getLock(resource);
2201        CmsClientLock clientLock = new CmsClientLock();
2202        clientLock.setLockType(CmsClientLock.LockType.valueOf(lock.getType().getMode()));
2203        CmsUUID ownerId = lock.getUserId();
2204        if (!lock.isUnlocked() && (ownerId != null)) {
2205            clientLock.setLockOwner(cms.readUser(ownerId).getDisplayName(cms, cms.getRequestContext().getLocale()));
2206            clientLock.setOwnedByUser(cms.getRequestContext().getCurrentUser().getId().equals(ownerId));
2207        }
2208        return clientLock;
2209    }
2210
2211    /**
2212     * Generates a list of property object to save to the sitemap entry folder to apply the given change.<p>
2213     *
2214     * @param change the change
2215     * @param entryFolder the entry folder
2216     *
2217     * @return the property objects
2218     */
2219    private List<CmsProperty> generateInheritProperties(CmsSitemapChange change, CmsResource entryFolder) {
2220
2221        List<CmsProperty> result = new ArrayList<CmsProperty>();
2222        Map<String, CmsClientProperty> clientProps = change.getOwnInternalProperties();
2223        boolean hasTitle = false;
2224        if (clientProps != null) {
2225            for (CmsClientProperty clientProp : clientProps.values()) {
2226                if (CmsPropertyDefinition.PROPERTY_TITLE.equals(clientProp.getName())) {
2227                    hasTitle = true;
2228                }
2229                CmsProperty prop = new CmsProperty(
2230                    clientProp.getName(),
2231                    clientProp.getStructureValue(),
2232                    clientProp.getResourceValue());
2233                result.add(prop);
2234            }
2235        }
2236        if (!hasTitle) {
2237            result.add(new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, change.getName(), null));
2238        }
2239        return result;
2240    }
2241
2242    /**
2243     * Generates a list of property object to save to the sitemap entry resource to apply the given change.<p>
2244     *
2245     * @param change the change
2246     *
2247     * @return the property objects
2248     */
2249    private List<CmsProperty> generateOwnProperties(CmsSitemapChange change) {
2250
2251        List<CmsProperty> result = new ArrayList<CmsProperty>();
2252
2253        Map<String, CmsClientProperty> clientProps = change.getDefaultFileProperties();
2254        boolean hasTitle = false;
2255        if (clientProps != null) {
2256            for (CmsClientProperty clientProp : clientProps.values()) {
2257                if (CmsPropertyDefinition.PROPERTY_TITLE.equals(clientProp.getName())) {
2258                    hasTitle = true;
2259                }
2260                CmsProperty prop = new CmsProperty(
2261                    clientProp.getName(),
2262                    clientProp.getStructureValue(),
2263                    clientProp.getResourceValue());
2264                result.add(prop);
2265            }
2266        }
2267        if (!hasTitle) {
2268            result.add(new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, change.getName(), null));
2269        }
2270        return result;
2271    }
2272
2273    /**
2274     * Generates a list of property values inherited to the site-map root entry.<p>
2275     *
2276     * @param rootPath the root entry name
2277     *
2278     * @return the list of property values inherited to the site-map root entry
2279     *
2280     * @throws CmsException if something goes wrong reading the properties
2281     */
2282    private Map<String, CmsClientProperty> generateParentProperties(String rootPath) throws CmsException {
2283
2284        CmsObject cms = getCmsObject();
2285        if (rootPath == null) {
2286            rootPath = cms.getRequestContext().addSiteRoot("/");
2287        }
2288        CmsObject rootCms = OpenCms.initCmsObject(cms);
2289        rootCms.getRequestContext().setSiteRoot("");
2290        String parentRootPath = CmsResource.getParentFolder(rootPath);
2291
2292        Map<String, CmsClientProperty> result = new HashMap<String, CmsClientProperty>();
2293        if (parentRootPath != null) {
2294            List<CmsProperty> props = rootCms.readPropertyObjects(parentRootPath, true);
2295            for (CmsProperty prop : props) {
2296                CmsClientProperty clientProp = createClientProperty(prop, true);
2297                result.put(clientProp.getName(), clientProp);
2298            }
2299        }
2300        return result;
2301    }
2302
2303    /**
2304     * Returns the sitemap children for the given path with all descendants up to the given level or to the given target path, ie.
2305     * <dl><dt>levels=1 <dd>only children<dt>levels=2<dd>children and great children</dl>
2306     * and so on.<p>
2307     *
2308     * @param root the site relative root
2309     * @param levels the levels to recurse
2310     * @param targetPath the target path
2311     *
2312     * @return the sitemap children
2313     */
2314    private List<CmsClientSitemapEntry> getChildren(String root, int levels, String targetPath) {
2315
2316        List<CmsClientSitemapEntry> children = new ArrayList<CmsClientSitemapEntry>();
2317        int i = 0;
2318        for (CmsJspNavElement navElement : getNavBuilder().getNavigationForFolder(
2319            root,
2320            Visibility.all,
2321            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED)) {
2322            try {
2323                CmsClientSitemapEntry child = toClientEntry(navElement, false);
2324                if (child != null) {
2325                    child.setPosition(i);
2326                    children.add(child);
2327                    int nextLevels = levels;
2328                    if ((nextLevels == 2) && (targetPath != null) && targetPath.startsWith(child.getSitePath())) {
2329                        nextLevels = 3;
2330                    }
2331                    if (child.isFolderType() && ((nextLevels > 1) || (nextLevels == -1)) && !isSubSitemap(navElement)) {
2332
2333                        child.setSubEntries(getChildren(child.getSitePath(), nextLevels - 1, targetPath), null);
2334                        child.setChildrenLoadedInitially(true);
2335                    }
2336                    i++;
2337                }
2338            } catch (CmsException e) {
2339                LOG.error("Could not read sitemap entry.", e);
2340            }
2341        }
2342        return children;
2343    }
2344
2345    /**
2346     * Returns the clipboard data from the current user.<p>
2347     *
2348     * @return the clipboard data
2349     */
2350    private CmsSitemapClipboardData getClipboardData() {
2351
2352        CmsSitemapClipboardData result = new CmsSitemapClipboardData();
2353        result.setModifications(getModifiedList());
2354        result.setDeletions(getDeletedList());
2355        return result;
2356    }
2357
2358    /**
2359     * Returns the deleted list from the current user.<p>
2360     *
2361     * @return the deleted list
2362     */
2363    private LinkedHashMap<CmsUUID, CmsClientSitemapEntry> getDeletedList() {
2364
2365        CmsObject cms = getCmsObject();
2366        CmsUser user = cms.getRequestContext().getCurrentUser();
2367        Object obj = user.getAdditionalInfo(ADDINFO_ADE_DELETED_LIST);
2368        LinkedHashMap<CmsUUID, CmsClientSitemapEntry> result = new LinkedHashMap<CmsUUID, CmsClientSitemapEntry>();
2369        if (obj instanceof String) {
2370            try {
2371                JSONArray array = new JSONArray((String)obj);
2372                for (int i = 0; i < array.length(); i++) {
2373                    try {
2374                        CmsUUID delId = new CmsUUID(array.getString(i));
2375                        CmsResource res = cms.readResource(delId, CmsResourceFilter.ALL);
2376                        if (res.getState().isDeleted()) {
2377                            // make sure resource is still deleted
2378                            CmsClientSitemapEntry delEntry = new CmsClientSitemapEntry();
2379                            delEntry.setSitePath(cms.getSitePath(res));
2380                            delEntry.setOwnProperties(getClientProperties(cms, res, false));
2381                            delEntry.setName(res.getName());
2382                            delEntry.setVfsPath(cms.getSitePath(res));
2383                            delEntry.setResourceTypeName(
2384                                OpenCms.getResourceManager().getResourceType(res).getTypeName());
2385                            delEntry.setEntryType(
2386                                res.isFolder()
2387                                ? EntryType.folder
2388                                : isRedirectType(res.getTypeId()) ? EntryType.redirect : EntryType.leaf);
2389                            delEntry.setNavModeIcon(
2390                                CmsIconUtil.getIconClasses(
2391                                    delEntry.getResourceTypeName(),
2392                                    delEntry.getVfsPath(),
2393                                    false));
2394                            delEntry.setId(delId);
2395                            result.put(delId, delEntry);
2396                        }
2397                    } catch (Throwable e) {
2398                        // should never happen, catches wrong or no longer existing values
2399                        LOG.warn(e.getLocalizedMessage());
2400                    }
2401                }
2402            } catch (Throwable e) {
2403                // should never happen, catches json parsing
2404                LOG.warn(e.getLocalizedMessage());
2405            }
2406        }
2407        return result;
2408    }
2409
2410    /**
2411     * Gets the container name for function detail elements depending on the parent folder.<p>
2412     *
2413     * @param parent the parent folder
2414     * @return the name of the function detail container
2415     */
2416    private String getFunctionDetailContainerName(CmsResource parent) {
2417
2418        CmsObject cms = getCmsObject();
2419        String notFound = null;
2420        String containerInfo = OpenCms.getTemplateContextManager().readPropertyFromTemplate(
2421            cms,
2422            parent,
2423            CmsPropertyDefinition.PROPERTY_CONTAINER_INFO,
2424            notFound);
2425        if (containerInfo == notFound) {
2426            return null;
2427        }
2428        Map<String, String> attrs = CmsStringUtil.splitAsMap(containerInfo, "|", "="); //$NON-NLS-2$
2429        String functionDetailContainerName = attrs.get(KEY_FUNCTION_DETAIL);
2430        return functionDetailContainerName;
2431    }
2432
2433    /**
2434     * Returns the galleries of the given sub site for the requested gallery type.<p>
2435     *
2436     * @param entryPointUri the sub site entry point
2437     * @param galleryType the gallery type
2438     * @param subSitePaths the sub site paths
2439     *
2440     * @return the gallery folder entries
2441     *
2442     * @throws CmsException if reading the resources fails
2443     */
2444    private List<CmsGalleryFolderEntry> getGalleriesForType(
2445        String entryPointUri,
2446        CmsGalleryType galleryType,
2447        List<String> subSitePaths)
2448    throws CmsException {
2449
2450        List<CmsGalleryFolderEntry> galleries = new ArrayList<CmsGalleryFolderEntry>();
2451        @SuppressWarnings("deprecation")
2452        List<CmsResource> galleryFolders = getCmsObject().readResources(
2453            entryPointUri,
2454            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED.addRequireType(galleryType.getTypeId()));
2455        for (CmsResource folder : galleryFolders) {
2456            try {
2457                if (!isInSubsite(subSitePaths, folder.getRootPath())) {
2458                    galleries.add(readGalleryFolderEntry(folder, galleryType.getResourceType()));
2459                }
2460            } catch (CmsException ex) {
2461                log(ex.getLocalizedMessage(), ex);
2462            }
2463        }
2464        // create a tree structure
2465        Collections.sort(galleries, new Comparator<CmsGalleryFolderEntry>() {
2466
2467            public int compare(CmsGalleryFolderEntry o1, CmsGalleryFolderEntry o2) {
2468
2469                return o1.getSitePath().compareTo(o2.getSitePath());
2470            }
2471        });
2472        List<CmsGalleryFolderEntry> galleryTree = new ArrayList<CmsGalleryFolderEntry>();
2473        for (int i = 0; i < galleries.size(); i++) {
2474            boolean isSubGallery = false;
2475            if (i > 0) {
2476                for (int j = i - 1; j >= 0; j--) {
2477                    if (galleries.get(i).getSitePath().startsWith(galleries.get(j).getSitePath())) {
2478                        galleries.get(j).addSubGallery(galleries.get(i));
2479                        isSubGallery = true;
2480                        break;
2481                    }
2482                }
2483            }
2484            if (!isSubGallery) {
2485                galleryTree.add(galleries.get(i));
2486            }
2487        }
2488        return galleryTree;
2489    }
2490
2491    /**
2492     * Returns the modified list from the current user.<p>
2493     *
2494     * @return the modified list
2495     */
2496    private LinkedHashMap<CmsUUID, CmsClientSitemapEntry> getModifiedList() {
2497
2498        CmsObject cms = getCmsObject();
2499        CmsUser user = cms.getRequestContext().getCurrentUser();
2500        Object obj = user.getAdditionalInfo(ADDINFO_ADE_MODIFIED_LIST);
2501        LinkedHashMap<CmsUUID, CmsClientSitemapEntry> result = new LinkedHashMap<CmsUUID, CmsClientSitemapEntry>();
2502        if (obj instanceof String) {
2503            try {
2504                JSONArray array = new JSONArray((String)obj);
2505                for (int i = 0; i < array.length(); i++) {
2506                    try {
2507                        CmsUUID modId = new CmsUUID(array.getString(i));
2508                        CmsResource res = cms.readResource(modId, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2509                        String sitePath = cms.getSitePath(res);
2510                        CmsJspNavElement navEntry = getNavBuilder().getNavigationForResource(
2511                            sitePath,
2512                            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2513                        if (navEntry.isInNavigation()) {
2514                            CmsClientSitemapEntry modEntry = toClientEntry(navEntry, false);
2515                            result.put(modId, modEntry);
2516                        }
2517                    } catch (Throwable e) {
2518                        // should never happen, catches wrong or no longer existing values
2519                        LOG.warn(e.getLocalizedMessage());
2520                    }
2521                }
2522            } catch (Throwable e) {
2523                // should never happen, catches json parsing
2524                LOG.warn(e.getLocalizedMessage());
2525            }
2526        }
2527        return result;
2528    }
2529
2530    /**
2531     * Returns a navigation builder reference.<p>
2532     *
2533     * @return the navigation builder
2534     */
2535    private CmsJspNavBuilder getNavBuilder() {
2536
2537        if (m_navBuilder == null) {
2538            m_navBuilder = new CmsJspNavBuilder(getCmsObject());
2539        }
2540        return m_navBuilder;
2541    }
2542
2543    /**
2544     * Returns the new resource infos.<p>
2545     *
2546     * @param cms the current CMS context
2547     * @param configData the configuration data from which the new resource infos should be read
2548     * @param locale locale used for retrieving descriptions/titles
2549     *
2550     * @return the new resource infos
2551     */
2552    private List<CmsNewResourceInfo> getNewResourceInfos(CmsObject cms, CmsADEConfigData configData, Locale locale) {
2553
2554        List<CmsNewResourceInfo> result = new ArrayList<CmsNewResourceInfo>();
2555        for (CmsModelPageConfig modelConfig : configData.getModelPages()) {
2556            try {
2557                CmsNewResourceInfo info = createNewResourceInfo(cms, modelConfig.getResource(), locale);
2558                info.setDefault(modelConfig.isDefault());
2559                result.add(info);
2560            } catch (CmsException e) {
2561                LOG.debug(e.getLocalizedMessage(), e);
2562            }
2563        }
2564        Collections.sort(result, new Comparator<CmsNewResourceInfo>() {
2565
2566            public int compare(CmsNewResourceInfo a, CmsNewResourceInfo b) {
2567
2568                return ComparisonChain.start().compareTrueFirst(a.isDefault(), b.isDefault()).compare(
2569                    a.getNavPos(),
2570                    b.getNavPos(),
2571                    Ordering.natural().nullsLast()).result();
2572            }
2573        });
2574        return result;
2575    }
2576
2577    /**
2578     * Gets the names of all available properties.<p>
2579     *
2580     * @param cms the CMS context
2581     *
2582     * @return the list of all property names
2583     *
2584     * @throws CmsException if something goes wrong
2585     */
2586    private List<String> getPropertyNames(CmsObject cms) throws CmsException {
2587
2588        List<CmsPropertyDefinition> propDefs = cms.readAllPropertyDefinitions();
2589        List<String> result = new ArrayList<String>();
2590        for (CmsPropertyDefinition propDef : propDefs) {
2591            result.add(propDef.getName());
2592        }
2593        return result;
2594    }
2595
2596    /**
2597     * Gets the resource type info beans for types for which new detail pages can be created.<p>
2598     *
2599     * @param cms the current CMS context
2600     * @param resourceTypeConfigs the resource type configurations
2601     * @param functionReferences the function references
2602     * @param dynamicFunctionRestriction
2603     * @param modelResource the model resource
2604     * @param locale the locale used for retrieving descriptions/titles
2605     *
2606     * @return the resource type info beans for types for which new detail pages can be created
2607     */
2608    private List<CmsNewResourceInfo> getResourceTypeInfos(
2609        CmsObject cms,
2610        List<CmsResourceTypeConfig> resourceTypeConfigs,
2611        List<CmsFunctionReference> functionReferences,
2612        CmsFunctionAvailability dynamicFunctionAvailability,
2613        CmsResource modelResource,
2614        Locale locale) {
2615
2616        List<CmsNewResourceInfo> result = new ArrayList<CmsNewResourceInfo>();
2617        CmsNewResourceInfo defaultPageInfo;
2618        if (modelResource != null) {
2619            defaultPageInfo = new CmsNewResourceInfo(
2620                modelResource.getTypeId(),
2621                CmsADEManager.DEFAULT_DETAILPAGE_TYPE,
2622                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_TITLE_0),
2623                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_DESCRIPTION_0),
2624                modelResource.getStructureId(),
2625                false,
2626                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_DESCRIPTION_0));
2627
2628        } else {
2629            defaultPageInfo = new CmsNewResourceInfo(
2630                CmsResourceTypeXmlContainerPage.getContainerPageTypeIdSafely(),
2631                CmsADEManager.DEFAULT_DETAILPAGE_TYPE,
2632                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_TITLE_0),
2633                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_DESCRIPTION_0),
2634                null,
2635                false,
2636                Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_DEFAULT_DETAIL_PAGE_DESCRIPTION_0));
2637        }
2638
2639        defaultPageInfo.setBigIconClasses(
2640            CmsIconUtil.getIconClasses(CmsResourceTypeXmlContainerPage.getStaticTypeName(), null, false));
2641        result.add(defaultPageInfo);
2642        for (CmsResourceTypeConfig typeConfig : resourceTypeConfigs) {
2643            if (typeConfig.isDetailPagesDisabled()) {
2644                continue;
2645            }
2646            String typeName = typeConfig.getTypeName();
2647            try {
2648                I_CmsResourceType resourceType = OpenCms.getResourceManager().getResourceType(typeName);
2649                result.add(createResourceTypeInfo(resourceType, modelResource));
2650            } catch (CmsLoaderException e) {
2651                LOG.error(e.getLocalizedMessage(), e);
2652            }
2653        }
2654        for (CmsFunctionReference functionRef : functionReferences) {
2655            if (dynamicFunctionAvailability.isDefined()) {
2656                if (!dynamicFunctionAvailability.checkAvailable(functionRef.getStructureId())) {
2657                    continue;
2658                }
2659            }
2660            try {
2661                CmsResource functionRes = cms.readResource(functionRef.getStructureId());
2662                String description = cms.readPropertyObject(
2663                    functionRes,
2664                    CmsPropertyDefinition.PROPERTY_DESCRIPTION,
2665                    false).getValue();
2666                String subtitle = description;
2667                try {
2668                    CmsGallerySearchResult searchResult = CmsGallerySearch.searchById(
2669                        cms,
2670                        functionRef.getStructureId(),
2671                        getWorkplaceLocale());
2672                    subtitle = searchResult.getDescription();
2673                } catch (CmsException e) {
2674                    LOG.warn(e.getLocalizedMessage(), e);
2675                }
2676
2677                CmsNewResourceInfo info = new CmsNewResourceInfo(
2678                    modelResource.getTypeId(),
2679                    CmsDetailPageInfo.FUNCTION_PREFIX + functionRef.getName(),
2680                    functionRef.getName(),
2681                    description,
2682                    modelResource.getStructureId(),
2683                    false,
2684                    subtitle);
2685                info.setCreateParameter(functionRef.getStructureId().toString());
2686                info.setIsFunction(true);
2687                info.setBigIconClasses(
2688                    CmsIconUtil.getIconClasses(CmsXmlDynamicFunctionHandler.TYPE_FUNCTION, null, false));
2689                result.add(info);
2690            } catch (CmsVfsResourceNotFoundException e) {
2691                LOG.warn(e.getLocalizedMessage(), e);
2692            } catch (CmsException e) {
2693                LOG.error(e.getLocalizedMessage(), e);
2694            }
2695        }
2696        return result;
2697    }
2698
2699    /**
2700     * Reeds the site root entry.<p>
2701     *
2702     * @param rootPath the root path of the sitemap root
2703     * @param targetPath the target path to open
2704     *
2705     * @return the site root entry
2706     *
2707     * @throws CmsSecurityException in case of insufficient permissions
2708     * @throws CmsException if something goes wrong
2709     */
2710    private CmsClientSitemapEntry getRootEntry(String rootPath, String targetPath)
2711    throws CmsSecurityException, CmsException {
2712
2713        String sitePath = "/";
2714        if ((rootPath != null)) {
2715            sitePath = getCmsObject().getRequestContext().removeSiteRoot(rootPath);
2716        }
2717        CmsJspNavElement navElement = getNavBuilder().getNavigationForResource(
2718            sitePath,
2719            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2720        CmsClientSitemapEntry result = toClientEntry(navElement, true);
2721        if (result != null) {
2722            result.setPosition(0);
2723            result.setChildrenLoadedInitially(true);
2724            result.setSubEntries(getChildren(sitePath, 2, targetPath), null);
2725        }
2726        return result;
2727    }
2728
2729    /**
2730     * Returns the sitemap info for the given base path.<p>
2731     *
2732     * @param basePath the base path
2733     *
2734     * @return the sitemap info
2735     *
2736     * @throws CmsException if something goes wrong reading the resources
2737     */
2738    private CmsSitemapInfo getSitemapInfo(String basePath) throws CmsException {
2739
2740        CmsObject cms = getCmsObject();
2741        CmsResource baseFolder = null;
2742        if (CmsStringUtil.isNotEmptyOrWhitespaceOnly(basePath)) {
2743            baseFolder = cms.readResource(
2744                cms.getRequestContext().removeSiteRoot(basePath),
2745                CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2746        } else {
2747            // in case of an empty base path, use base folder of the current site
2748            basePath = "/";
2749            baseFolder = cms.readResource("/");
2750        }
2751        CmsResource defaultFile = cms.readDefaultFile(baseFolder, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2752        String title = cms.readPropertyObject(baseFolder, CmsPropertyDefinition.PROPERTY_TITLE, false).getValue();
2753        if (CmsStringUtil.isEmptyOrWhitespaceOnly(title) && (defaultFile != null)) {
2754            title = cms.readPropertyObject(defaultFile, CmsPropertyDefinition.PROPERTY_TITLE, false).getValue();
2755        }
2756        String description = cms.readPropertyObject(
2757            baseFolder,
2758            CmsPropertyDefinition.PROPERTY_DESCRIPTION,
2759            false).getValue();
2760        if (CmsStringUtil.isEmptyOrWhitespaceOnly(description) && (defaultFile != null)) {
2761            description = cms.readPropertyObject(
2762                defaultFile,
2763                CmsPropertyDefinition.PROPERTY_DESCRIPTION,
2764                false).getValue();
2765        }
2766        return new CmsSitemapInfo(
2767            cms.getRequestContext().getCurrentProject().getName(),
2768            description,
2769            OpenCms.getLocaleManager().getDefaultLocale(cms, baseFolder).toString(),
2770            OpenCms.getSiteManager().getCurrentSite(cms).getUrl()
2771                + OpenCms.getLinkManager().substituteLinkForUnknownTarget(cms, basePath),
2772            title);
2773    }
2774
2775    /**
2776     * Gets the default type id for subsitemap folders.<p>
2777     *
2778     * @return the default type id for subsitemap folders
2779     *
2780     * @throws CmsRpcException in case of an error
2781     */
2782    @SuppressWarnings("deprecation")
2783    private int getSubsitemapType() throws CmsRpcException {
2784
2785        try {
2786            return OpenCms.getResourceManager().getResourceType(
2787                CmsResourceTypeFolderSubSitemap.TYPE_SUBSITEMAP).getTypeId();
2788        } catch (CmsLoaderException e) {
2789            error(e);
2790        }
2791        return CmsResourceTypeUnknownFolder.getStaticTypeId();
2792    }
2793
2794    /**
2795     * Returns the workplace locale for the current user.<p>
2796     *
2797     * @return the workplace locale
2798     */
2799    private Locale getWorkplaceLocale() {
2800
2801        Locale result = OpenCms.getWorkplaceManager().getWorkplaceLocale(getCmsObject());
2802        if (result == null) {
2803            result = CmsLocaleManager.getDefaultLocale();
2804        }
2805        if (result == null) {
2806            result = Locale.getDefault();
2807        }
2808        return result;
2809    }
2810
2811    /**
2812     * Checks whether the sitemap change has default file changes.<p>
2813     *
2814     * @param change a sitemap change
2815     * @return true if the change would change the default file
2816     */
2817    private boolean hasDefaultFileChanges(CmsSitemapChange change) {
2818
2819        return (change.getDefaultFileId() != null) && !change.isNew();
2820    }
2821
2822    /**
2823     * Checks whether the sitemap change has changes for the sitemap entry resource.<p>
2824     *
2825     * @param change the sitemap change
2826     * @return true if the change would change the original sitemap entry resource
2827     */
2828    private boolean hasOwnChanges(CmsSitemapChange change) {
2829
2830        return !change.isNew();
2831    }
2832
2833    /**
2834     * Checks whether a resource is a default file of a folder.<p>
2835     *
2836     * @param resource the resource to check
2837     *
2838     * @return true if the resource is the default file of a folder
2839     *
2840     * @throws CmsException if something goes wrong
2841     */
2842    private boolean isDefaultFile(CmsResource resource) throws CmsException {
2843
2844        CmsObject cms = getCmsObject();
2845        if (resource.isFolder()) {
2846            return false;
2847        }
2848
2849        CmsResource parent = cms.readResource(
2850            CmsResource.getParentFolder(cms.getSitePath(resource)),
2851            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2852        CmsResource defaultFile = cms.readDefaultFile(parent, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2853        return resource.equals(defaultFile);
2854    }
2855
2856    /**
2857     * Checks if the toolbar should be displayed.<p>
2858     *
2859     * @param request the current request to get the default locale from
2860     *
2861     * @return <code>true</code> if the toolbar should be displayed
2862     */
2863    private boolean isDisplayToolbar(HttpServletRequest request) {
2864
2865        // display the toolbar by default
2866        boolean displayToolbar = true;
2867        if (CmsHistoryResourceHandler.isHistoryRequest(request)) {
2868            // we do not want to display the toolbar in case of an historical request
2869            displayToolbar = false;
2870        }
2871        return displayToolbar;
2872    }
2873
2874    /**
2875     * Returns if the given path is located below one of the given sub site paths.<p>
2876     *
2877     * @param subSitePaths the sub site root paths
2878     * @param path the root path to check
2879     *
2880     * @return <code>true</code> if the given path is located below one of the given sub site paths
2881     */
2882    private boolean isInSubsite(List<String> subSitePaths, String path) {
2883
2884        boolean result = false;
2885        for (String subSite : subSitePaths) {
2886            if (path.startsWith(subSite)) {
2887                result = true;
2888                break;
2889            }
2890        }
2891        return result;
2892    }
2893
2894    /**
2895     * Returns if the given type id matches the xml-redirect resource type.<p>
2896     *
2897     * @param typeId the resource type id
2898     *
2899     * @return <code>true</code> if the given type id matches the xml-redirect resource type
2900     */
2901    @SuppressWarnings("deprecation")
2902    private boolean isRedirectType(int typeId) {
2903
2904        try {
2905            return typeId == OpenCms.getResourceManager().getResourceType(RECOURCE_TYPE_NAME_REDIRECT).getTypeId();
2906        } catch (Exception e) {
2907            return false;
2908        }
2909    }
2910
2911    /**
2912     * Returns if the given nav-element resembles a sub-sitemap entry-point.<p>
2913     *
2914     * @param navElement the nav-element
2915     *
2916     * @return <code>true</code> if the given nav-element resembles a sub-sitemap entry-point.<p>
2917     */
2918    private boolean isSubSitemap(CmsJspNavElement navElement) {
2919
2920        return CmsResourceTypeFolderSubSitemap.isSubSitemap(navElement.getResource());
2921    }
2922
2923    /**
2924     * Checks if the given open path is valid.<p>
2925     *
2926     * @param cms the cms context
2927     * @param openPath the open path
2928     *
2929     * @return <code>true</code> if the given open path is valid
2930     */
2931    private boolean isValidOpenPath(CmsObject cms, String openPath) {
2932
2933        if (CmsStringUtil.isEmptyOrWhitespaceOnly(openPath)) {
2934            return false;
2935        }
2936        if (!cms.existsResource(openPath)) {
2937            // in case of a detail-page check the parent folder
2938            String parent = CmsResource.getParentFolder(openPath);
2939            if (CmsStringUtil.isEmptyOrWhitespaceOnly(parent) || !cms.existsResource(parent)) {
2940                return false;
2941            }
2942        }
2943        return true;
2944    }
2945
2946    /**
2947     * Returns if the given return code is valid.<p>
2948     *
2949     * @param returnCode the return code to check
2950     *
2951     * @return <code>true</code> if the return code is valid
2952     */
2953    private boolean isValidReturnCode(String returnCode) {
2954
2955        if (CmsStringUtil.isEmptyOrWhitespaceOnly(returnCode)) {
2956            return false;
2957        }
2958        int pos = returnCode.indexOf(":");
2959        if (pos > 0) {
2960            return CmsUUID.isValidUUID(returnCode.substring(0, pos))
2961                && CmsUUID.isValidUUID(returnCode.substring(pos + 1));
2962        } else {
2963            return CmsUUID.isValidUUID(returnCode);
2964        }
2965    }
2966
2967    /**
2968     * Applys the given changes to the entry.<p>
2969     *
2970     * @param change the change to apply
2971     *
2972     * @throws CmsException if something goes wrong
2973     */
2974    private void modifyEntry(CmsSitemapChange change) throws CmsException {
2975
2976        CmsObject cms = getCmsObject();
2977        CmsResource entryPage = null;
2978        CmsResource entryFolder = null;
2979
2980        CmsResource ownRes = null;
2981        CmsResource defaultFileRes = null;
2982        try {
2983            // lock all resources necessary first to avoid doing changes only half way through
2984
2985            if (hasOwnChanges(change)) {
2986                ownRes = cms.readResource(change.getEntryId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2987                boolean shallow = !change.hasChangedName() && !change.hasChangedPosition() && !change.hasNewParent();
2988
2989                ensureLock(ownRes, shallow);
2990            }
2991
2992            if (hasDefaultFileChanges(change)) {
2993                defaultFileRes = cms.readResource(change.getDefaultFileId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
2994                ensureLock(defaultFileRes, false);
2995            }
2996
2997            if ((ownRes != null) && ownRes.isFolder()) {
2998                entryFolder = ownRes;
2999            }
3000
3001            if ((ownRes != null) && ownRes.isFile()) {
3002                entryPage = ownRes;
3003            }
3004
3005            if (defaultFileRes != null) {
3006                entryPage = defaultFileRes;
3007            }
3008
3009            if (change.isLeafType()) {
3010                entryFolder = entryPage;
3011            }
3012
3013            String moveSrc = null;
3014            String moveDest = null;
3015
3016            if (entryFolder != null) {
3017                if (change.hasNewParent() || change.hasChangedName()) {
3018                    String destinationPath;
3019                    if (change.hasNewParent()) {
3020                        CmsResource futureParent = cms.readResource(
3021                            change.getParentId(),
3022                            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
3023                        destinationPath = CmsStringUtil.joinPaths(cms.getSitePath(futureParent), change.getName());
3024                    } else {
3025                        destinationPath = CmsStringUtil.joinPaths(
3026                            CmsResource.getParentFolder(cms.getSitePath(entryFolder)),
3027                            change.getName());
3028                    }
3029                    if (change.isLeafType() && destinationPath.endsWith("/")) {
3030                        destinationPath = destinationPath.substring(0, destinationPath.length() - 1);
3031                    }
3032                    // only if the site-path has really changed
3033                    if (!CmsFileUtil.removeTrailingSeparator(cms.getSitePath(entryFolder)).equals(
3034                        CmsFileUtil.removeTrailingSeparator(destinationPath))) {
3035                        moveSrc = cms.getSitePath(entryFolder);
3036                        moveDest = CmsFileUtil.removeTrailingSeparator(destinationPath);
3037                    }
3038                }
3039            }
3040            if ((moveDest != null) && cms.existsResource(moveDest, CmsResourceFilter.IGNORE_EXPIRATION)) {
3041                throw new CmsVfsResourceAlreadyExistsException(
3042                    org.opencms.db.generic.Messages.get().container(
3043                        org.opencms.db.generic.Messages.ERR_RESOURCE_WITH_NAME_ALREADY_EXISTS_1,
3044                        moveDest));
3045            }
3046
3047            updateProperties(cms, ownRes, defaultFileRes, change.getPropertyChanges());
3048            if (change.hasChangedPosition()) {
3049                updateNavPos(ownRes, change);
3050            }
3051
3052            if (moveDest != null) {
3053                cms.moveResource(moveSrc, moveDest);
3054            }
3055            entryFolder = cms.readResource(entryFolder.getStructureId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
3056
3057        } finally {
3058            if (entryPage != null) {
3059                tryUnlock(entryPage);
3060            }
3061            if (entryFolder != null) {
3062                tryUnlock(entryFolder);
3063            }
3064        }
3065    }
3066
3067    /**
3068     * Reads the gallery folder properties.<p>
3069     *
3070     * @param folder the folder resource
3071     * @param typeName the  resource type name
3072     *
3073     * @return the folder entry data
3074     *
3075     * @throws CmsException if the folder properties can not be read
3076     */
3077    private CmsGalleryFolderEntry readGalleryFolderEntry(CmsResource folder, String typeName) throws CmsException {
3078
3079        CmsObject cms = getCmsObject();
3080        CmsGalleryFolderEntry folderEntry = new CmsGalleryFolderEntry();
3081        folderEntry.setResourceType(typeName);
3082        folderEntry.setSitePath(cms.getSitePath(folder));
3083        folderEntry.setStructureId(folder.getStructureId());
3084        folderEntry.setOwnProperties(getClientProperties(cms, folder, false));
3085        folderEntry.setIconClasses(CmsIconUtil.getIconClasses(typeName, null, false));
3086        return folderEntry;
3087    }
3088
3089    /**
3090     * Helper method for removing all locales except one from a container page.<p>
3091     *
3092     * @param page the container page to proces
3093     * @param localeToKeep the locale which should be kept
3094     *
3095     * @throws CmsXmlException if something goes wrong
3096     */
3097    private void removeAllLocalesExcept(CmsXmlContainerPage page, Locale localeToKeep) throws CmsXmlException {
3098
3099        List<Locale> locales = page.getLocales();
3100        for (Locale locale : locales) {
3101            if (!locale.equals(localeToKeep)) {
3102                page.removeLocale(locale);
3103            }
3104        }
3105    }
3106
3107    /**
3108     * Applys the given remove change.<p>
3109     *
3110     * @param change the change to apply
3111     *
3112     * @return the changed client sitemap entry or <code>null</code>
3113     *
3114     * @throws CmsException if something goes wrong
3115     */
3116    private CmsSitemapChange removeEntryFromNavigation(CmsSitemapChange change) throws CmsException {
3117
3118        CmsObject cms = getCmsObject();
3119        CmsResource entryFolder = cms.readResource(change.getEntryId(), CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
3120        ensureLock(entryFolder);
3121        List<CmsProperty> properties = new ArrayList<CmsProperty>();
3122        properties.add(
3123            new CmsProperty(
3124                CmsPropertyDefinition.PROPERTY_NAVTEXT,
3125                CmsProperty.DELETE_VALUE,
3126                CmsProperty.DELETE_VALUE));
3127        properties.add(
3128            new CmsProperty(CmsPropertyDefinition.PROPERTY_NAVPOS, CmsProperty.DELETE_VALUE, CmsProperty.DELETE_VALUE));
3129        cms.writePropertyObjects(cms.getSitePath(entryFolder), properties);
3130        tryUnlock(entryFolder);
3131        return change;
3132    }
3133
3134    /**
3135     * Saves the detail page information of a sitemap to the sitemap's configuration file.<p>
3136     *
3137     * @param detailPages saves the detailpage configuration
3138     * @param resource the configuration file resource
3139     * @param newId the structure id to use for new detail page entries
3140     * @param updateEntry the new detail page entry
3141     *
3142     * @throws CmsException if something goes wrong
3143     */
3144    private void saveDetailPages(
3145        List<CmsDetailPageInfo> detailPages,
3146        CmsResource resource,
3147        CmsUUID newId,
3148        CmsClientSitemapEntry updateEntry)
3149    throws CmsException {
3150
3151        CmsObject cms = getCmsObject();
3152        if (updateEntry != null) {
3153            for (CmsDetailPageInfo info : detailPages) {
3154                if (info.getId() == null) {
3155                    updateEntry.setDetailpageTypeName(info.getType());
3156                    break;
3157                }
3158            }
3159        }
3160        CmsDetailPageConfigurationWriter writer = new CmsDetailPageConfigurationWriter(cms, resource);
3161        writer.updateAndSave(detailPages, newId);
3162    }
3163
3164    /**
3165     * Saves the given clipboard data to the session.<p>
3166     *
3167     * @param clipboardData the clipboard data to save
3168     *
3169     * @throws CmsException if something goes wrong writing the user
3170     */
3171    private void setClipboardData(CmsSitemapClipboardData clipboardData) throws CmsException {
3172
3173        CmsObject cms = getCmsObject();
3174        CmsUser user = cms.getRequestContext().getCurrentUser();
3175        if (clipboardData != null) {
3176            JSONArray modified = new JSONArray();
3177            if (clipboardData.getModifications() != null) {
3178                for (CmsUUID id : clipboardData.getModifications().keySet()) {
3179                    if (id != null) {
3180                        modified.put(id.toString());
3181                    }
3182                }
3183            }
3184            user.setAdditionalInfo(ADDINFO_ADE_MODIFIED_LIST, modified.toString());
3185            JSONArray deleted = new JSONArray();
3186            if (clipboardData.getDeletions() != null) {
3187                for (CmsUUID id : clipboardData.getDeletions().keySet()) {
3188                    if (id != null) {
3189                        deleted.put(id.toString());
3190                    }
3191                }
3192            }
3193            user.setAdditionalInfo(ADDINFO_ADE_DELETED_LIST, deleted.toString());
3194            cms.writeUser(user);
3195        }
3196    }
3197
3198    /**
3199     * Determines if the title property of the default file should be changed.<p>
3200     *
3201     * @param properties the current default file properties
3202     * @param folderNavtext the 'NavText' property of the folder
3203     *
3204     * @return <code>true</code> if the title property should be changed
3205     */
3206    private boolean shouldChangeDefaultFileTitle(Map<String, CmsProperty> properties, CmsProperty folderNavtext) {
3207
3208        return (properties == null)
3209            || (properties.get(CmsPropertyDefinition.PROPERTY_TITLE) == null)
3210            || (properties.get(CmsPropertyDefinition.PROPERTY_TITLE).getValue() == null)
3211            || ((folderNavtext != null)
3212                && properties.get(CmsPropertyDefinition.PROPERTY_TITLE).getValue().equals(folderNavtext.getValue()));
3213    }
3214
3215    /**
3216     * Determines if the title property should be changed in case of a 'NavText' change.<p>
3217     *
3218     * @param properties the current resource properties
3219     *
3220     * @return <code>true</code> if the title property should be changed in case of a 'NavText' change
3221     */
3222    private boolean shouldChangeTitle(Map<String, CmsProperty> properties) {
3223
3224        return (properties == null)
3225            || (properties.get(CmsPropertyDefinition.PROPERTY_TITLE) == null)
3226            || (properties.get(CmsPropertyDefinition.PROPERTY_TITLE).getValue() == null)
3227            || ((properties.get(CmsPropertyDefinition.PROPERTY_NAVTEXT) != null)
3228                && properties.get(CmsPropertyDefinition.PROPERTY_TITLE).getValue().equals(
3229                    properties.get(CmsPropertyDefinition.PROPERTY_NAVTEXT).getValue()));
3230    }
3231
3232    /**
3233     * Synchronizes the title from the NavText for a specific resource after properties have been edited, if NavText isn't empty and the title isn't already set to a different non-empty value.
3234     *
3235     * @param resource the resource
3236     * @param oldNavText the NavText before editing the properties
3237     * @param newNavText the NavText after editing the properties (may be the same as oldNavText)
3238     * @throws CmsException if something goes wrong
3239     */
3240    private void synchTitleFromNavText(CmsResource resource, String oldNavText, String newNavText) throws CmsException {
3241
3242        CmsObject cms = getCmsObject();
3243
3244        if (CmsStringUtil.isEmpty(newNavText)) {
3245            return;
3246        }
3247        CmsProperty titleProp = cms.readPropertyObject(resource, CmsPropertyDefinition.PROPERTY_TITLE, false);
3248        String title = titleProp.getValue();
3249        if (title == null) {
3250            title = "";
3251        }
3252        // We don't check if oldNavText is different from newNavText, because we also want the synchronization to happen
3253        // when we don't actually change the NavText, but e.g. when we change the title to an empty string.
3254        if (CmsStringUtil.isEmpty(title) || title.equals(oldNavText)) {
3255            if (!newNavText.equals(title)) {
3256                cms.writePropertyObjects(
3257                    resource,
3258                    Arrays.asList(new CmsProperty(CmsPropertyDefinition.PROPERTY_TITLE, newNavText, null)));
3259            }
3260        }
3261    }
3262
3263    /**
3264     * Converts a jsp navigation element into a client sitemap entry.<p>
3265     *
3266     * @param navElement the jsp navigation element
3267     * @param isRoot true if the entry is a root entry
3268     *
3269     * @return the client sitemap entry
3270     *
3271     * @throws CmsException if something goes wrong
3272     */
3273    private CmsClientSitemapEntry toClientEntry(CmsJspNavElement navElement, boolean isRoot) throws CmsException {
3274
3275        CmsResource entryPage = null;
3276        CmsObject cms = getCmsObject();
3277        CmsClientSitemapEntry clientEntry = new CmsClientSitemapEntry();
3278        CmsResource entryFolder = null;
3279
3280        CmsResource ownResource = navElement.getResource();
3281        clientEntry.setResourceState(ownResource.getState());
3282        CmsResource defaultFileResource = null;
3283        if (ownResource.isFolder() && !navElement.isNavigationLevel()) {
3284            defaultFileResource = cms.readDefaultFile(ownResource, CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
3285        }
3286
3287        Map<String, CmsClientProperty> ownProps = getClientProperties(cms, ownResource, false);
3288
3289        Map<String, CmsClientProperty> defaultFileProps = null;
3290        if (defaultFileResource != null) {
3291            defaultFileProps = getClientProperties(cms, defaultFileResource, false);
3292            clientEntry.setDefaultFileId(defaultFileResource.getStructureId());
3293            clientEntry.setDefaultFileType(
3294                OpenCms.getResourceManager().getResourceType(defaultFileResource.getTypeId()).getTypeName());
3295            clientEntry.setDefaultFileReleased(defaultFileResource.isReleasedAndNotExpired(System.currentTimeMillis()));
3296        } else {
3297            defaultFileProps = new HashMap<String, CmsClientProperty>();
3298        }
3299        boolean isDefault = isDefaultFile(ownResource);
3300        clientEntry.setId(ownResource.getStructureId());
3301        clientEntry.setResourceTypeId(ownResource.getTypeId());
3302        clientEntry.setFolderDefaultPage(isDefault);
3303        if (navElement.getResource().isFolder()) {
3304            entryFolder = navElement.getResource();
3305            entryPage = defaultFileResource;
3306            clientEntry.setName(entryFolder.getName());
3307            if (entryPage == null) {
3308                entryPage = entryFolder;
3309            }
3310            if (!isRoot && isSubSitemap(navElement)) {
3311                clientEntry.setEntryType(EntryType.subSitemap);
3312                clientEntry.setDefaultFileType(null);
3313            } else if (navElement.isNavigationLevel()) {
3314                clientEntry.setEntryType(EntryType.navigationLevel);
3315            }
3316            CmsLock folderLock = cms.getLock(entryFolder);
3317            clientEntry.setHasForeignFolderLock(
3318                !folderLock.isUnlocked() && !folderLock.isOwnedBy(cms.getRequestContext().getCurrentUser()));
3319            if (!cms.getRequestContext().getCurrentProject().isOnlineProject()) {
3320                List<CmsResource> blockingChildren = cms.getBlockingLockedResources(entryFolder);
3321                clientEntry.setBlockingLockedChildren((blockingChildren != null) && !blockingChildren.isEmpty());
3322            }
3323        } else {
3324            entryPage = navElement.getResource();
3325            clientEntry.setName(entryPage.getName());
3326            if (isRedirectType(entryPage.getTypeId())) {
3327                clientEntry.setEntryType(EntryType.redirect);
3328                CmsFile file = getCmsObject().readFile(entryPage);
3329                I_CmsXmlDocument content = CmsXmlContentFactory.unmarshal(getCmsObject(), file);
3330                I_CmsXmlContentValue linkValue = content.getValue(
3331                    REDIRECT_LINK_TARGET_XPATH,
3332                    getCmsObject().getRequestContext().getLocale());
3333                String link = linkValue != null
3334                ? linkValue.getStringValue(getCmsObject())
3335                : Messages.get().getBundle(getWorkplaceLocale()).key(Messages.GUI_REDIRECT_SUB_LEVEL_0);
3336                clientEntry.setRedirectTarget(link);
3337            } else {
3338                clientEntry.setEntryType(EntryType.leaf);
3339            }
3340        }
3341        if (entryPage.isFile()) {
3342            List<CmsAlias> aliases = OpenCms.getAliasManager().getAliasesForStructureId(
3343                getCmsObject(),
3344                entryPage.getStructureId());
3345            if (!aliases.isEmpty()) {
3346                List<String> aliasList = new ArrayList<String>();
3347                for (CmsAlias alias : aliases) {
3348                    String aliasPath = alias.getAliasPath();
3349                    aliasList.add(aliasPath);
3350                }
3351                clientEntry.setAliases(aliasList);
3352            }
3353        }
3354        long dateExpired = navElement.getResource().getDateExpired();
3355        if (dateExpired != CmsResource.DATE_EXPIRED_DEFAULT) {
3356            clientEntry.setDateExpired(
3357                CmsDateUtil.getDate(new Date(dateExpired), DateFormat.SHORT, getWorkplaceLocale()));
3358        }
3359        long dateReleased = navElement.getResource().getDateReleased();
3360        if (dateReleased != CmsResource.DATE_RELEASED_DEFAULT) {
3361            clientEntry.setDateReleased(
3362                CmsDateUtil.getDate(new Date(dateReleased), DateFormat.SHORT, getWorkplaceLocale()));
3363        }
3364        clientEntry.setResleasedAndNotExpired(
3365            navElement.getResource().isReleasedAndNotExpired(System.currentTimeMillis()));
3366        String path = cms.getSitePath(entryPage);
3367        clientEntry.setVfsPath(path);
3368        clientEntry.setOwnProperties(ownProps);
3369        clientEntry.setDefaultFileProperties(defaultFileProps);
3370        clientEntry.setSitePath(entryFolder != null ? cms.getSitePath(entryFolder) : path);
3371        clientEntry.setLock(generateClientLock(entryPage));
3372        clientEntry.setInNavigation(isRoot || navElement.isInNavigation());
3373        String type = OpenCms.getResourceManager().getResourceType(ownResource).getTypeName();
3374        clientEntry.setResourceTypeName(type);
3375        clientEntry.setVfsModeIcon(CmsIconUtil.getIconClasses(type, ownResource.getName(), false));
3376
3377        if (!clientEntry.isSubSitemapType()) {
3378            if (clientEntry.isNavigationLevelType()) {
3379                clientEntry.setNavModeIcon(CmsIconUtil.ICON_NAV_LEVEL_BIG);
3380            } else if (defaultFileResource != null) {
3381                clientEntry.setNavModeIcon(
3382                    CmsIconUtil.getIconClasses(clientEntry.getDefaultFileType(), defaultFileResource.getName(), false));
3383            }
3384        }
3385        clientEntry.setPermissionInfo(OpenCms.getADEManager().getPermissionInfo(cms, ownResource, null));
3386        return clientEntry;
3387    }
3388
3389    /**
3390     * Un-deletes a resource according to the change data.<p>
3391     *
3392     * @param change the change data
3393     *
3394     * @return the changed entry or <code>null</code>
3395     *
3396     * @throws CmsException if something goes wrong
3397     */
3398    private CmsSitemapChange undelete(CmsSitemapChange change) throws CmsException {
3399
3400        CmsObject cms = getCmsObject();
3401        CmsResource deleted = cms.readResource(change.getEntryId(), CmsResourceFilter.ALL);
3402        if (deleted.getState().isDeleted()) {
3403            ensureLock(deleted);
3404            cms.undeleteResource(getCmsObject().getSitePath(deleted), true);
3405            tryUnlock(deleted);
3406        }
3407        String parentPath = CmsResource.getParentFolder(cms.getSitePath(deleted));
3408        CmsJspNavElement navElement = getNavBuilder().getNavigationForResource(
3409            parentPath,
3410            CmsResourceFilter.ONLY_VISIBLE_NO_DELETED);
3411        CmsClientSitemapEntry entry = toClientEntry(navElement, navElement.isInNavigation());
3412        entry.setSubEntries(getChildren(parentPath, 2, null), null);
3413        change.setUpdatedEntry(entry);
3414        change.setParentId(cms.readParentFolder(deleted.getStructureId()).getStructureId());
3415        return change;
3416    }
3417
3418    /**
3419     * Updates the navigation position for a resource.<p>
3420     *
3421     * @param res the resource for which to update the navigation position
3422     * @param change the sitemap change
3423     *
3424     * @throws CmsException if something goes wrong
3425     */
3426    private void updateNavPos(CmsResource res, CmsSitemapChange change) throws CmsException {
3427
3428        if (change.hasChangedPosition()) {
3429            applyNavigationChanges(change, res);
3430        }
3431    }
3432
3433    /**
3434     * Updates properties for a resource and possibly its detail page.<p>
3435     *
3436     * @param cms the CMS context
3437     * @param ownRes the resource
3438     * @param defaultFileRes the default file resource (possibly null)
3439     * @param propertyModifications the property modifications
3440     *
3441     * @throws CmsException if something goes wrong
3442     */
3443    private void updateProperties(
3444        CmsObject cms,
3445        CmsResource ownRes,
3446        CmsResource defaultFileRes,
3447        List<CmsPropertyModification> propertyModifications)
3448    throws CmsException {
3449
3450        Map<String, CmsProperty> ownProps = getPropertiesByName(cms.readPropertyObjects(ownRes, false));
3451        String oldNavText = null;
3452        if (ownProps.containsKey(CmsPropertyDefinition.PROPERTY_NAVTEXT)) {
3453            oldNavText = ownProps.get(CmsPropertyDefinition.PROPERTY_NAVTEXT).getValue();
3454        }
3455        if (oldNavText == null) {
3456            oldNavText = "";
3457        }
3458        Map<String, CmsProperty> defaultFileProps = Maps.newHashMap();
3459        if (defaultFileRes != null) {
3460            defaultFileProps = getPropertiesByName(cms.readPropertyObjects(defaultFileRes, false));
3461        }
3462        String newNavText = oldNavText;
3463        List<CmsProperty> ownPropertyChanges = new ArrayList<CmsProperty>();
3464        List<CmsProperty> defaultFilePropertyChanges = new ArrayList<CmsProperty>();
3465        for (CmsPropertyModification propMod : propertyModifications) {
3466            CmsProperty propToModify = null;
3467            Map<String, CmsProperty> propMap = null;
3468            List<CmsProperty> changeList = null;
3469
3470            if (ownRes.getStructureId().equals(propMod.getId())) {
3471                if (CmsPropertyDefinition.PROPERTY_NAVTEXT.equals(propMod.getName())) {
3472                    newNavText = propMod.getValue();
3473                }
3474                propMap = ownProps;
3475                changeList = ownPropertyChanges;
3476            } else {
3477                propMap = defaultFileProps;
3478                changeList = defaultFilePropertyChanges;
3479            }
3480            propToModify = propMap.get(propMod.getName());
3481            if (propToModify == null) {
3482                propToModify = new CmsProperty(propMod.getName(), null, null);
3483            }
3484            String newValue = propMod.getValue();
3485            if (newValue == null) {
3486                newValue = "";
3487            }
3488            if (propMod.isStructureValue()) {
3489                propToModify.setStructureValue(newValue);
3490            } else {
3491                propToModify.setResourceValue(newValue);
3492            }
3493            changeList.add(propToModify);
3494        }
3495
3496        if (!ownPropertyChanges.isEmpty()) {
3497            cms.writePropertyObjects(ownRes, ownPropertyChanges);
3498        }
3499        if (!defaultFilePropertyChanges.isEmpty() && (defaultFileRes != null)) {
3500            cms.writePropertyObjects(defaultFileRes, defaultFilePropertyChanges);
3501        }
3502        synchTitleFromNavText(ownRes, oldNavText, newNavText);
3503        if (defaultFileRes != null) {
3504            synchTitleFromNavText(defaultFileRes, oldNavText, newNavText);
3505        }
3506    }
3507
3508}