/* -*- 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 . */ #define _DIR_CXX #include #include #include #include #include "comdep.hxx" #include DBG_NAME( Dir ) /// determines whether insertion is required sal_Bool Dir::ImpInsertPointReached( const DirEntry& rNewEntry, const FileStat& rNewStat, size_t nCurPos, size_t nSortIndex ) const { #define VALUE( nKindFlags ) \ ( ( FSYS_KIND_FILE | FSYS_KIND_DIR | FSYS_KIND_DEV | \ FSYS_KIND_CHAR | FSYS_KIND_BLOCK ) & nKindFlags ) if ( pLst->empty() ) return sal_True; FSysSort nSort = (*pSortLst)[ nSortIndex ]; FileStat *pOldStat = NULL; DirEntry *pCurLstObj = (*pLst)[ nCurPos ]; if ( pStatLst ) pOldStat = (*pStatLst)[ nCurPos ]; switch( nSort ) { case FSYS_SORT_NAME: case (FSYS_SORT_NAME | FSYS_SORT_ASCENDING): if ( pCurLstObj->aName > rNewEntry.aName ) return sal_True; if ( !(pCurLstObj->aName == rNewEntry.aName) ) return sal_False; break; case (FSYS_SORT_NAME | FSYS_SORT_DESCENDING): if ( pCurLstObj->aName < rNewEntry.aName ) return sal_True; if ( !(pCurLstObj->aName == rNewEntry.aName) ) return sal_False; break; case FSYS_SORT_EXT: case (FSYS_SORT_EXT | FSYS_SORT_ASCENDING): { if ( pCurLstObj->GetExtension() > rNewEntry.GetExtension() ) return sal_True; if ( !(pCurLstObj->GetExtension() == rNewEntry.GetExtension()) ) return sal_False; break; } case (FSYS_SORT_EXT | FSYS_SORT_DESCENDING): { if ( pCurLstObj->GetExtension() < rNewEntry.GetExtension() ) return sal_True; if ( !(pCurLstObj->GetExtension() == rNewEntry.GetExtension()) ) return sal_False; break; } case FSYS_SORT_KIND: case (FSYS_SORT_KIND | FSYS_SORT_ASCENDING ): if ( VALUE(pOldStat->nKindFlags) > VALUE(rNewStat.nKindFlags) ) return sal_True; if ( !(VALUE(pOldStat->nKindFlags) == VALUE(rNewStat.nKindFlags)) ) return sal_False; break; case (FSYS_SORT_KIND | FSYS_SORT_DESCENDING): if ( VALUE(pOldStat->nKindFlags) < VALUE(rNewStat.nKindFlags) ) return sal_True; if ( !(VALUE(pOldStat->nKindFlags) == VALUE(rNewStat.nKindFlags)) ) return sal_False; break; case FSYS_SORT_SIZE: case (FSYS_SORT_SIZE | FSYS_SORT_ASCENDING): if ( pOldStat->nSize > rNewStat.nSize ) return sal_True; if ( !(pOldStat->nSize == rNewStat.nSize) ) return sal_False; break; case (FSYS_SORT_SIZE | FSYS_SORT_DESCENDING): if ( pOldStat->nSize < rNewStat.nSize ) return sal_True; if ( !(pOldStat->nSize == rNewStat.nSize) ) return sal_False; break; case FSYS_SORT_MODIFYED: case (FSYS_SORT_MODIFYED | FSYS_SORT_ASCENDING): if ( (pOldStat->aDateModified >= rNewStat.aDateModified) && (pOldStat->aTimeModified > rNewStat.aTimeModified) ) return sal_True; if ( !((pOldStat->aDateModified == rNewStat.aDateModified) && (pOldStat->aTimeModified == rNewStat.aTimeModified)) ) return sal_False; break; case (FSYS_SORT_MODIFYED | FSYS_SORT_DESCENDING): if ( (pOldStat->aDateModified <= rNewStat.aDateModified) && (pOldStat->aTimeModified < rNewStat.aTimeModified) ) return sal_True; if ( !((pOldStat->aDateModified == rNewStat.aDateModified) && (pOldStat->aTimeModified == rNewStat.aTimeModified)) ) return sal_False; break; case FSYS_SORT_CREATED: case (FSYS_SORT_CREATED | FSYS_SORT_ASCENDING): if ( (pOldStat->aDateCreated >= rNewStat.aDateCreated) && (pOldStat->aTimeCreated > rNewStat.aTimeCreated) ) return sal_True; if ( !((pOldStat->aDateCreated == rNewStat.aDateCreated) && (pOldStat->aTimeCreated == rNewStat.aTimeCreated)) ) return sal_False; break; case (FSYS_SORT_CREATED | FSYS_SORT_DESCENDING): if ( (pOldStat->aDateCreated <= rNewStat.aDateCreated) && (pOldStat->aTimeCreated < rNewStat.aTimeCreated) ) return sal_True; if ( !((pOldStat->aDateCreated == rNewStat.aDateCreated) && (pOldStat->aTimeCreated == rNewStat.aTimeCreated)) ) return sal_False; break; case FSYS_SORT_ACCESSED: case (FSYS_SORT_ACCESSED | FSYS_SORT_ASCENDING): if ( (pOldStat->aDateAccessed >= rNewStat.aDateAccessed) && (pOldStat->aTimeAccessed > rNewStat.aTimeAccessed) ) return sal_True; if ( !((pOldStat->aDateAccessed == rNewStat.aDateAccessed) && (pOldStat->aTimeAccessed == rNewStat.aTimeAccessed)) ) return sal_False; break; case (FSYS_SORT_ACCESSED | FSYS_SORT_DESCENDING): if ( (pOldStat->aDateAccessed <= rNewStat.aDateAccessed) && (pOldStat->aTimeAccessed < rNewStat.aTimeAccessed) ) return sal_True; if ( !((pOldStat->aDateAccessed == rNewStat.aDateAccessed) && (pOldStat->aTimeAccessed == rNewStat.aTimeAccessed)) ) return sal_False; break; default: /* Kann nicht sein */; } if ( nSortIndex == ( pSortLst->size() - 1 ) ) return sal_True; else return ImpInsertPointReached( rNewEntry, rNewStat, nCurPos, nSortIndex + 1 ); #undef VALUE } /// Insert as sorted void Dir::ImpSortedInsert( const DirEntry *pNewEntry, const FileStat *pNewStat ) { // special case: no sorting required if ( !pSortLst ) { pLst->push_back( (DirEntry*)pNewEntry ); return; } for ( size_t i = 0, n = pLst->size(); i < n; ++i ) { if ( ImpInsertPointReached( *pNewEntry, *pNewStat, i, 0 ) ) { if ( pStatLst ) { FileStatList::iterator it = pStatLst->begin(); ::std::advance( it, i ); pStatLst->insert( it, (FileStat*)pNewStat ); } DirEntryList::iterator it = pLst->begin(); ::std::advance( it, i ); pLst->insert( it, (DirEntry*)pNewEntry ); return; } } if ( pStatLst ) pStatLst->push_back( (FileStat*)pNewStat ); pLst->push_back( (DirEntry*)pNewEntry ); } /// shared implementation of CTORs void Dir::Construct( DirEntryKind nKindFlags ) { pLst = NULL; pSortLst = NULL; pStatLst = NULL; eAttrMask = nKindFlags; rtl::OString aTempName(rtl::OUStringToOString(GetName(), osl_getThreadTextEncoding())); if (aTempName.indexOf('*') != -1 || aTempName.indexOf('?') != -1) { #if defined( WNT ) rtl::OString aTStr(rtl::OUStringToOString(CutName(), osl_getThreadTextEncoding())); char* pBuffer = new char[aTStr.getLength()+1]; strcpy( pBuffer, aTStr.getStr() ); CharLowerBuff( pBuffer, aTStr.getLength() ); aNameMask = WildCard( String(pBuffer, osl_getThreadTextEncoding()), ';' ); delete [] pBuffer; #else aNameMask = WildCard( CutName(), ';' ); #endif } else aNameMask.setGlob(rtl::OUString(static_cast('*'))); } void Dir::Reset() { // remove old Reader if necessary if ( pReader && pReader->bInUse ) DELETEZ(pReader); // Remove all DirEntries from List and free memory if ( pLst ) { for ( size_t i = 0, n = pLst->size(); i < n; ++i ) { delete (*pLst)[ i ]; } pLst->clear(); } else pLst = new DirEntryList(); // Remove old File-Stats if ( pStatLst ) { for ( size_t i = 0, n = pStatLst->size(); i < n; ++i ) { delete (*pStatLst)[ i ]; } pStatLst->clear(); delete pStatLst; pStatLst = NULL; } // Does sorting require FileStats? if ( pSortLst ) { for ( size_t i = 0, n = pSortLst->size(); i < n; ++i ) { if ( (*pSortLst)[ i ] & ( FSYS_SORT_KIND | FSYS_SORT_SIZE | FSYS_SORT_CREATED | FSYS_SORT_MODIFYED | FSYS_SORT_ACCESSED ) ) { pStatLst = new FileStatList(); break; } } } // Create new reader if necessary if ( !pReader ) pReader = new DirReader_Impl( *this ); // Does the directory exist at all? #if !defined(UNX) //explanation: see DirReader_Impl::Read() in unx.cxx if( !pReader->pDosDir ) { nError = FSYS_ERR_NOTADIRECTORY; DELETEZ( pReader ); return; } #endif } sal_uInt16 Dir::Scan( sal_uInt16 nCount ) { sal_uInt16 nRead = 0; // Number of read entries in this round FSysFailOnErrorImpl(); // did not complete if ( pReader ) { // is this a new reader? if ( pLst->empty() ) { // Scan directories pReader->bInUse = sal_True; nRead = pReader->Init(); } // continue reading while ( nRead <= nCount && !pReader->bReady ) nRead = nRead + pReader->Read(); // done? if ( pReader && pReader->bReady ) DELETEZ( pReader ); } // Return read entry count return nRead; } Dir::Dir( const DirEntry& rDirEntry, DirEntryKind nKindFlags ): DirEntry( rDirEntry ), pReader( 0 ) { DBG_CTOR( Dir, NULL ); Construct( nKindFlags ); Reset(); } Dir::~Dir() { DBG_DTOR( Dir, NULL ); // Remove all DirEntries and free memory if ( pLst ) { for ( size_t i = 0, n = pLst->size(); i < n; ++i ) { delete (*pLst)[ i ]; } pLst->clear(); delete pLst; } // Remove all Sorts from list and free memory if ( pSortLst ) { pSortLst->clear(); delete pSortLst; } // Remove all FileStats from list and free memory if ( pStatLst ) { for ( size_t i = 0, n = pStatLst->size(); i < n; ++i ) { delete (*pStatLst)[ i ]; } pStatLst->clear(); delete pStatLst; } delete pReader; } DirEntry& Dir::operator[] ( size_t nIndex ) const { DBG_ASSERT( nIndex < Count(), "Dir::operator[] : nIndex > Count()" ); DirEntry *pEntry = (*pLst)[ nIndex ]; return *pEntry; } Dir& Dir::operator+=( const Dir& rDir ) { // Read the rest of the directory if ( pReader ) Scan( USHRT_MAX ); DBG_ASSERT( !rDir.pReader, "Dir::+= with incomplete Dir" ); if ( !pLst ) pLst = new DirEntryList(); // FileStats required by sorting criteria? sal_Bool bStat = sal_False; if ( pSortLst ) { for ( size_t i = 0, n = pSortLst->size(); i < n && !bStat; ++i ) { if ( (*pSortLst)[ i ] & ( FSYS_SORT_CREATED | FSYS_SORT_MODIFYED | FSYS_SORT_SIZE | FSYS_SORT_ACCESSED | FSYS_SORT_KIND ) ) { bStat = sal_True; } } } FileStat* stat = NULL; for ( size_t nNr = 0; nNr < rDir.Count(); nNr++ ) { if ( bStat ) { if ( rDir.pStatLst ) stat = new FileStat( *(*rDir.pStatLst)[ nNr ] ); else stat = new FileStat( rDir[nNr] ); } ImpSortedInsert( new DirEntry( rDir[nNr] ), stat ); } return *this; } size_t Dir::Count( sal_Bool bUpdated ) const { // Read the rest of the directory if ( bUpdated && pReader ) ((Dir*)this)->Scan( USHRT_MAX ); return pLst == NULL ? 0 : pLst->size(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */