//------------------------------------------------------------------------------ // Copyright (c) 2001-2002, OpenBeOS // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // // File Name: RecentEntries.cpp // Author: Tyler Dauwalder (tyler@dauwalder.net) // Description: Recently launched apps list //------------------------------------------------------------------------------ /*! \file RecentEntries.cpp \brief RecentEntries class implementation */ #include "RecentEntries.h" #include #include #include #include // From the Storage Kit #include #include #include #include #include #include #include #define DBG(x) (x) //#define DBG(x) #define OUT printf //------------------------------------------------------------------------------ // recent_entry //------------------------------------------------------------------------------ /*! \struct recent_entry \brief A recent entry, the corresponding signature of the application that launched/used/opened/viewed/whatevered it, and an index used for keeping track of orderings when loading/storing the recent entries list from/to disk. */ /*! \brief Creates a new recent_entry object. */ recent_entry::recent_entry(const entry_ref *ref, const char *appSig, uint32 index) : ref(ref ? *ref : entry_ref()) , sig(appSig) , index(index) { } //------------------------------------------------------------------------------ // RecentEntries //------------------------------------------------------------------------------ /*! \class RecentEntries \brief Implements the common functionality used by the roster's recent folders and recent documents lists. */ /*! \var std::list RecentEntries::fEntryList \brief The list of entries and their corresponding app sigs, most recent first The signatures are expected to be stored all lowercase, as MIME signatures are case-independent. */ // constructor /*! \brief Creates a new list. The list is initially empty. */ RecentEntries::RecentEntries() { } // destructor /*! \brief Frees all resources associated with the object. */ RecentEntries::~RecentEntries() { Clear(); } // Add /*! \brief Places the given entry Places the app with the given signature at the front of the recent apps list. If the app already exists elsewhere in the list, that item is removed so only one instance exists in the list at any time. \param appSig The application's signature \param appFlags The application's flags. If \a appFlags contains either \c B_ARGV_ONLY or \c B_BACKGROUND_APP, the application is \b not added to the list (but \c B_OK is still returned). \return - \c B_OK: success (even if the app was not added due to appFlags) - error code: failure */ status_t RecentEntries::Add(const entry_ref *ref, const char *appSig) { std::string sig; status_t error = ref && appSig ? B_OK : B_BAD_VALUE; if (!error) { // Store all sigs as lowercase sig = BPrivate::Storage::to_lower(appSig); // Look for a previous instance of this entry std::list::iterator item; for (item = fEntryList.begin(); item != fEntryList.end(); item++) { if ((*item)->ref == *ref && (*item)->sig == sig) { fEntryList.erase(item); break; } } // Add this entry to the front of the list recent_entry *entry = new(nothrow) recent_entry(ref, appSig, 0); error = entry ? B_OK : B_NO_MEMORY; if (!error) fEntryList.push_front(entry); } return error; } // Get /*! \brief Returns the first \a maxCount recent apps in the \c BMessage pointed to by \a list. The message is cleared first, and \c entry_refs for the the apps are stored in the \c "refs" field of the message (\c B_REF_TYPE). If there are fewer than \a maxCount items in the list, the entire list is returned. Duplicate entries are never returned, i.e. if two instances of the same entry were added under different app sigs, and both instances match the given filter criterion, only the most recent instance is returned; the latter instance is ignored and not counted towards the \a maxCount number of entries to return. Since BRoster::GetRecentEntries() returns \c void, the message pointed to by \a list is simply cleared if maxCount is invalid (i.e. <= 0). \param fileTypes An array of file type filters. These file types are expected to be all lowercase. */ status_t RecentEntries::Get(int32 maxCount, const char *fileTypes[], int32 fileTypesCount, const char *appSig, BMessage *result) { status_t error = result && (fileTypesCount == 0 || (fileTypesCount > 0 && fileTypes)) ? B_OK : B_BAD_VALUE; if (!error) { result->MakeEmpty(); std::list duplicateList; std::list::iterator item; int count = 0; for (item = fEntryList.begin(); count < maxCount && item != fEntryList.end(); item++) { bool match = true; // Filter if necessary if (fileTypesCount > 0 || appSig) { match = false; // Filter by app sig if (appSig) match = (*item)->sig == appSig; // Filter by file type if (!match && fileTypesCount > 0) { char type[B_MIME_TYPE_LENGTH]; if (GetTypeForRef(&(*item)->ref, type) == B_OK) { for (int i = 0; i < fileTypesCount; i++) { if (strcmp(type, fileTypes[i]) == 0) { match = true; break; } } } } } if (match) { // Check for duplicates for (std::list::iterator dupItem = duplicateList.begin(); dupItem != duplicateList.end(); dupItem++) { if ((*dupItem)->ref == (*item)->ref) { match = false; break; } } } if (match) { // Add the ref to the list used to check // for duplicates, and then to the result duplicateList.push_back(*item); result->AddRef("refs", &(*item)->ref); count++; } } } return error; } // Clear /*! \brief Clears the list of recently launched apps */ status_t RecentEntries::Clear() { std::list::iterator i; for (i = fEntryList.begin(); i != fEntryList.end(); i++) delete *i; fEntryList.clear(); return B_OK; } // Print /*! \brief Dumps the the current list of entries to stdout. */ status_t RecentEntries::Print() { std::list::iterator item; int counter = 1; for (item = fEntryList.begin(); item != fEntryList.end(); item++) { printf("%d: device == '%ld', dir == '%lld', name == '%s', app == '%s', index == %ld\n", counter++, (*item)->ref.device, (*item)->ref.directory, (*item)->ref.name, (*item)->sig.c_str(), (*item)->index); } return B_OK; } // Save status_t RecentEntries::Save(FILE* file, const char *description, const char *tag) { status_t error = file ? B_OK : B_BAD_VALUE; if (!error) { fprintf(file, "# %s\n", description); /* In order to write our entries out in the format used by the Roster settings file, we need to collect all the signatures for each entry in one place, while at the same time updating the index values for each entry/sig pair to reflect the current ordering of the list. I believe this is the data structure R5 actually maintains all the time, as their indices do not change over time (whereas ours will). If our implementation proves to be slower that R5, we may want to consider using the data structure pervasively. */ std::map > map; uint32 count = fEntryList.size(); for (std::list::iterator item = fEntryList.begin(); item != fEntryList.end(); count--, item++) { recent_entry *entry = *item; if (entry) { entry->index = count; map[entry->ref].push_back(entry); } else { DBG(OUT("WARNING: RecentEntries::Save(): The entry %ld entries " "from the front of fEntryList was found to be NULL\n", fEntryList.size() - count)); } } for (std::map >::iterator mapItem = map.begin(); mapItem != map.end(); mapItem++) { // We're going to need to properly escape the path name we // get, which will at absolute worst double the length of // the string. char path[B_PATH_NAME_LENGTH]; char escapedPath[B_PATH_NAME_LENGTH*2]; status_t outputError = BPrivate::Storage::entry_ref_to_path(&mapItem->first, path, B_PATH_NAME_LENGTH); if (!outputError) { BPrivate::Storage::escape_path(path, escapedPath); fprintf(file, "%s %s", tag, escapedPath); std::list &list = mapItem->second; int32 i = 0; for (std::list::iterator item = list.begin(); item != list.end(); i++, item++) { recent_entry *entry = *item; if (entry) fprintf(file, " \"%s\" %ld", entry->sig.c_str(), entry->index); else { DBG(OUT("WARNING: RecentEntries::Save(): The entry %ld entries " "from the front of the compiled recent_entry* list for the " "entry ref (%ld, %lld, '%s') was found to be NULL\n", i, mapItem->first.device, mapItem->first.directory, mapItem->first.name)); } } fprintf(file, "\n"); } else { DBG(OUT("WARNING: RecentEntries::Save(): entry_ref_to_path() failed on " "the entry_ref (%ld, %lld, '%s') with error 0x%lx\n", mapItem->first.device, mapItem->first.directory, mapItem->first.name, outputError)); } } fprintf(file, "\n"); } return error; } // GetTypeForRef /*! \brief Fetches the file type of the given file. If the file has no type, an empty string is returned. The file is *not* sniffed. */ status_t RecentEntries::GetTypeForRef(const entry_ref *ref, char *result) { BNode node; status_t error = ref && result ? B_OK : B_BAD_VALUE; // Read the type and force to lowercase if (!error) error = node.SetTo(ref); if (!error) { ssize_t bytes = node.ReadAttr("BEOS:TYPE", B_MIME_STRING_TYPE, 0, result, B_MIME_TYPE_LENGTH-1); if (bytes < 0) error = bytes; else result[bytes] = '\0'; } if (!error) BPrivate::Storage::to_lower(result); return error; }