/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "TemplateScanner.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; namespace { const char TITLE[] = "Title"; const char TARGET_DIR_URL[] = "TargetDirURL"; const char DESCRIPTION[] = "TypeDescription"; const char TARGET_URL[] = "TargetURL"; const char DOCTEMPLATES[] = "com.sun.star.frame.DocumentTemplates"; // These strings are used to find impress templates in the tree of // template files. Should probably be determined dynamically. const char IMPRESS_BIN_TEMPLATE[] = "application/vnd.stardivision.impress"; const char IMPRESS_XML_TEMPLATE[] = MIMETYPE_VND_SUN_XML_IMPRESS_ASCII; // The following id comes from the bugdoc in #i2764#. const char IMPRESS_XML_TEMPLATE_B[] = "Impress 2.0"; const char IMPRESS_XML_TEMPLATE_OASIS[] = MIMETYPE_OASIS_OPENDOCUMENT_PRESENTATION_ASCII; class FolderDescriptor { public: FolderDescriptor ( int nPriority, const OUString& rsTitle, const OUString& rsTargetDir, const OUString& rsContentIdentifier, const Reference& rxFolderEnvironment) : mnPriority(nPriority), msTitle(rsTitle), msTargetDir(rsTargetDir), msContentIdentifier(rsContentIdentifier), mxFolderEnvironment(rxFolderEnvironment) { } int mnPriority; OUString msTitle; OUString msTargetDir; OUString msContentIdentifier; // Reference mxFolderResultSet; Reference mxFolderEnvironment; class Comparator { public: bool operator() (const FolderDescriptor& r1, const FolderDescriptor& r2) const { return r1.mnPriority < r2.mnPriority; } }; }; /** Use a heuristic based on the URL of a top-level template folder to assign a priority that is used to sort the folders. */ int Classify (const OUString&, const OUString& rsURL) { int nPriority (0); if (rsURL.isEmpty()) nPriority = 100; else if (rsURL.indexOf("presnt")>=0) { nPriority = 30; } else if (rsURL.indexOf("layout")>=0) { nPriority = 20; } else if (rsURL.indexOf("educate")>=0) { nPriority = 40; } else if (rsURL.indexOf("finance")>=0) { nPriority = 40; } else { // All other folders are taken for user supplied and have the // highest priority. nPriority = 10; } return nPriority; } } // end of anonymous namespace namespace sd { TemplateEntryCompare::TemplateEntryCompare(): mpStringSorter(new comphelper::string::NaturalStringSorter( ::comphelper::getProcessComponentContext(), Application::GetSettings().GetLanguageTag().getLocale())) {} bool TemplateEntryCompare::operator()(TemplateEntry* pA, TemplateEntry* pB) const { return 0 > mpStringSorter->compare(pA->msTitle, pB->msTitle); } void TemplateDir::EnableSorting(bool bSortingEnabled) { mbSortingEnabled = bSortingEnabled; if (mbSortingEnabled) { if (mpEntryCompare.get() == NULL) mpEntryCompare.reset(new TemplateEntryCompare); ::std::sort(maEntries.begin(), maEntries.end(), *mpEntryCompare); } } void TemplateDir::InsertEntry(TemplateEntry* pNewEntry) { if (mbSortingEnabled) { ::std::vector::iterator aPlaceToInsert = ::std::upper_bound(maEntries.begin(), maEntries.end(), pNewEntry, *mpEntryCompare); maEntries.insert(aPlaceToInsert, pNewEntry); } else maEntries.push_back(pNewEntry); } class TemplateScanner::FolderDescriptorList : public ::std::multiset { }; TemplateScanner::TemplateScanner (void) : meState(INITIALIZE_SCANNING), maFolderContent(), mpTemplateDirectory(NULL), maFolderList(), mbEntrySortingEnabled(false), mpLastAddedEntry(NULL), mpFolderDescriptors(new FolderDescriptorList()), mxTemplateRoot(), mxFolderEnvironment(), mxEntryEnvironment(), mxFolderResultSet(), mxEntryResultSet() { // empty; } TemplateScanner::~TemplateScanner (void) { mpFolderDescriptors.reset(); // Delete all entries of the template list that have not been // transferred to another object. std::vector::iterator I; for (I=maFolderList.begin(); I!=maFolderList.end(); ++I) if (*I != NULL) delete *I; } TemplateScanner::State TemplateScanner::GetTemplateRoot (void) { State eNextState (INITIALIZE_FOLDER_SCANNING); Reference< XComponentContext > xContext = ::comphelper::getProcessComponentContext(); Reference xTemplates = frame::DocumentTemplates::create(xContext); mxTemplateRoot = xTemplates->getContent(); return eNextState; } TemplateScanner::State TemplateScanner::InitializeEntryScanning (void) { State eNextState (SCAN_ENTRY); if (maFolderContent.isFolder()) { mxEntryEnvironment = Reference(); // We are interested only in three properties: the entry's name, // its URL, and its content type. Sequence aProps (3); aProps[0] = OUString(TITLE); aProps[1] = OUString(TARGET_URL); aProps[2] = OUString(DESCRIPTION); // Create a cursor to iterate over the templates in this folders. ::ucbhelper::ResultSetInclude eInclude = ::ucbhelper::INCLUDE_DOCUMENTS_ONLY; mxEntryResultSet = Reference( maFolderContent.createCursor(aProps, eInclude)); } else eNextState = ERROR; return eNextState; } TemplateScanner::State TemplateScanner::ScanEntry (void) { State eNextState (ERROR); Reference xContentAccess (mxEntryResultSet, UNO_QUERY); Reference xRow (mxEntryResultSet, UNO_QUERY); if (xContentAccess.is() && xRow.is() && mxEntryResultSet.is()) { if (mxEntryResultSet->next()) { OUString sTitle (xRow->getString (1)); OUString sTargetURL (xRow->getString (2)); OUString sContentType (xRow->getString (3)); OUString aId = xContentAccess->queryContentIdentifierString(); ::ucbhelper::Content aContent = ::ucbhelper::Content (aId, mxEntryEnvironment, comphelper::getProcessComponentContext()); if (aContent.isDocument ()) { // Check whether the entry is an impress template. If so // add a new entry to the resulting list (which is created // first if necessary). if ( (sContentType == MIMETYPE_OASIS_OPENDOCUMENT_PRESENTATION_TEMPLATE) || (sContentType == IMPRESS_XML_TEMPLATE_OASIS) || (sContentType == IMPRESS_BIN_TEMPLATE) || (sContentType == IMPRESS_XML_TEMPLATE) || (sContentType == IMPRESS_XML_TEMPLATE_B)) { OUString sLocalisedTitle = SfxDocumentTemplates::ConvertResourceString( STR_TEMPLATE_NAME1_DEF, STR_TEMPLATE_NAME1, NUM_TEMPLATE_NAMES, sTitle ); mpLastAddedEntry = new TemplateEntry(sLocalisedTitle, sTargetURL); mpTemplateDirectory->InsertEntry(mpLastAddedEntry); } } // Continue scanning entries. eNextState = SCAN_ENTRY; } else { if (mpTemplateDirectory->maEntries.empty()) { delete mpTemplateDirectory; mpTemplateDirectory = NULL; } else { SolarMutexGuard aGuard; maFolderList.push_back(mpTemplateDirectory); } // Continue with scanning the next folder. eNextState = SCAN_FOLDER; } } return eNextState; } TemplateScanner::State TemplateScanner::InitializeFolderScanning (void) { State eNextState (ERROR); mxFolderResultSet = Reference(); try { // Create content for template folders. mxFolderEnvironment = Reference(); ::ucbhelper::Content aTemplateDir (mxTemplateRoot, mxFolderEnvironment, comphelper::getProcessComponentContext()); // Define the list of properties we are interested in. Sequence aProps (2); aProps[0] = OUString(TITLE); aProps[1] = OUString(TARGET_DIR_URL); // Create an cursor to iterate over the template folders. ::ucbhelper::ResultSetInclude eInclude = ::ucbhelper::INCLUDE_FOLDERS_ONLY; mxFolderResultSet = Reference( aTemplateDir.createCursor(aProps, eInclude)); if (mxFolderResultSet.is()) eNextState = GATHER_FOLDER_LIST; } catch (::com::sun::star::uno::Exception&) { eNextState = ERROR; } return eNextState; } TemplateScanner::State TemplateScanner::GatherFolderList (void) { State eNextState (ERROR); Reference xContentAccess (mxFolderResultSet, UNO_QUERY); if (xContentAccess.is() && mxFolderResultSet.is()) { while (mxFolderResultSet->next()) { Reference xRow (mxFolderResultSet, UNO_QUERY); if (xRow.is()) { OUString sTitle (xRow->getString (1)); OUString sTargetDir (xRow->getString (2)); OUString aId = xContentAccess->queryContentIdentifierString(); mpFolderDescriptors->insert( FolderDescriptor( Classify(sTitle,sTargetDir), sTitle, sTargetDir, aId, mxFolderEnvironment)); } } eNextState = SCAN_FOLDER; } return eNextState; } TemplateScanner::State TemplateScanner::ScanFolder (void) { State eNextState (ERROR); if (mpFolderDescriptors->size() > 0) { FolderDescriptor aDescriptor (*mpFolderDescriptors->begin()); mpFolderDescriptors->erase(mpFolderDescriptors->begin()); OUString sTitle (aDescriptor.msTitle); OUString sTargetDir (aDescriptor.msTargetDir); OUString aId (aDescriptor.msContentIdentifier); maFolderContent = ::ucbhelper::Content (aId, aDescriptor.mxFolderEnvironment, comphelper::getProcessComponentContext()); if (maFolderContent.isFolder()) { // Scan the folder and insert it into the list of template // folders. mpTemplateDirectory = new TemplateDir (sTitle, sTargetDir); if (mpTemplateDirectory != NULL) { mpTemplateDirectory->EnableSorting(mbEntrySortingEnabled); // Continue with scanning all entries in the folder. eNextState = INITIALIZE_ENTRY_SCAN; } } } else { eNextState = DONE; } return eNextState; } void TemplateScanner::Scan (void) { while (HasNextStep()) RunNextStep(); } std::vector& TemplateScanner::GetFolderList (void) { return maFolderList; } void TemplateScanner::RunNextStep (void) { switch (meState) { case INITIALIZE_SCANNING: meState = GetTemplateRoot(); break; case INITIALIZE_FOLDER_SCANNING: meState = InitializeFolderScanning(); break; case SCAN_FOLDER: meState = ScanFolder(); break; case GATHER_FOLDER_LIST: meState = GatherFolderList(); break; case INITIALIZE_ENTRY_SCAN: meState = InitializeEntryScanning(); break; case SCAN_ENTRY: meState = ScanEntry(); break; default: break; } switch (meState) { case DONE: case ERROR: mxTemplateRoot.clear(); mxTemplateRoot.clear(); mxFolderEnvironment.clear(); mxEntryEnvironment.clear(); mxFolderResultSet.clear(); mxEntryResultSet.clear(); mpLastAddedEntry = NULL; break; default: break; } } bool TemplateScanner::HasNextStep (void) { switch (meState) { case DONE: case ERROR: return false; default: return true; } } const TemplateEntry* TemplateScanner::GetLastAddedEntry (void) const { return mpLastAddedEntry; } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */