|
|
// File: mrulist.cpp
#include "precomp.h"
#include "mrulist2.h"
typedef struct { LPTSTR psz1; LPTSTR psz2; } SZSZ; typedef SZSZ * PSZSZ;
typedef struct { LPTSTR psz1; LPTSTR psz2; DWORD dw; } SZSZDW; typedef SZSZDW * PSZSZDW;
static LPTSTR PszAlloc(LPCTSTR pszSrc) { if (NULL == pszSrc) return NULL;
LPTSTR pszDest = new TCHAR[lstrlen(pszSrc) + 1]; if (NULL != pszDest) { lstrcpy(pszDest, pszSrc); } return pszDest; }
/* C M R U L I S T */ /*-------------------------------------------------------------------------
%%Function: CMRUList2
-------------------------------------------------------------------------*/ CMRUList2::CMRUList2(const DWSTR * prgDwStr, int cEntryMax, BOOL fReversed) : m_prgDwStr (prgDwStr), m_cEntryMax (cEntryMax), m_fReversed (fReversed), m_cEntry (0), m_rgpEntry (NULL), m_fDirty (FALSE) { DBGENTRY(CMRUList2::CMRUList2);
ASSERT(NULL != prgDwStr); m_cCol = m_prgDwStr[0].dw;
int cb = m_cEntryMax * sizeof(PMRUE); m_rgpEntry = new PMRUE[cb]; if (NULL == m_rgpEntry) { ERROR_OUT(("CMRUList2 - out of memory")); return; } ZeroMemory(m_rgpEntry, cb);
RegEntry re(PszRegKey(), HKEY_CURRENT_USER); if (ERROR_SUCCESS != re.GetError()) return;
m_cEntry = min(re.GetNumber(REGVAL_MRU_COUNT, 0), m_cEntryMax); for (int i = 0; i < m_cEntry; i++) { m_rgpEntry[i] = LoadEntry(&re, i); } }
CMRUList2::~CMRUList2() { DBGENTRY(CMRUList2::~CMRUList2);
if (m_fDirty) { Save(); }
for (int i = 0; i < m_cEntry; i++) { DeleteEntry(m_rgpEntry[i]); } delete m_rgpEntry; }
///////////////////////////////////////////////////////////////////////////
PMRUE CMRUList2::LoadEntry(RegEntry * pre, int iItem) { if (m_fReversed) { iItem = (m_cEntry - (iItem+1)); }
PMRUE pEntry = (PMRUE) new PVOID[m_cCol*sizeof(PVOID)]; if (NULL != pEntry) { PVOID ** ppv = (PVOID **) pEntry; for (int iCol = 0; iCol < m_cCol; iCol++, ppv++) { TCHAR szKey[MAX_PATH]; wsprintf(szKey, TEXT("%s%d"), PszPrefixForCol(iCol), iItem); switch (MruTypeForCol(iCol)) { default: case MRUTYPE_SZ: * (LPTSTR *)ppv = PszAlloc(pre->GetString(szKey)); break; case MRUTYPE_DW: * (DWORD *) ppv = pre->GetNumber(szKey); break; } } }
return pEntry; }
VOID CMRUList2::StoreEntry(RegEntry * pre, int iItem) { PVOID ** ppv = (PVOID **) GetEntry(iItem);
if (m_fReversed) { iItem = (m_cEntry - (iItem+1)); }
for (int iCol = 0; iCol < m_cCol; iCol++, ppv++) { TCHAR szKey[MAX_PATH]; wsprintf(szKey, TEXT("%s%d"), PszPrefixForCol(iCol), iItem); switch (MruTypeForCol(iCol)) { default: case MRUTYPE_SZ: pre->SetValue(szKey, * (LPCTSTR *)ppv); break; case MRUTYPE_DW: pre->SetValue(szKey, * (ULONG *) ppv); break; } } }
VOID CMRUList2::DeleteEntry(PMRUE pEntry) { PVOID ** ppv = (PVOID **) pEntry; for (int iCol = 0; iCol < m_cCol; iCol++, ppv++) { switch (MruTypeForCol(iCol)) { default: case MRUTYPE_SZ: delete *ppv; break; case MRUTYPE_DW: break; } } delete pEntry; }
VOID CMRUList2::DeleteEntry(int iItem) { if ((iItem < 0) || (iItem >= m_cEntry)) return; // nothing to do
// delete the data
DeleteEntry(m_rgpEntry[iItem]);
// decrement the count
m_cEntry--;
// shift items up
for ( ; iItem < m_cEntry; iItem++) { m_rgpEntry[iItem] = m_rgpEntry[iItem+1]; }
// the list has been modified
m_fDirty = TRUE; }
//--------------------------------------------------------------------------//
// CMRUList2::DeleteEntry. //
// This DeleteEntry() deletes the first entry it finds thats primary //
// string matches the one passed in. //
//--------------------------------------------------------------------------//
void CMRUList2::DeleteEntry ( const TCHAR * const primaryString ){ int items = GetNumEntries();
for( int nn = 0; nn < items; nn++ ) { if( StrCmpI( primaryString, * ((const TCHAR * const * const) m_rgpEntry[ nn ]) ) == 0 ) { DeleteEntry( nn ); break; } }
} // End of CMRUList2::DeleteEntry.
/* C O M P A R E E N T R Y */ /*-------------------------------------------------------------------------
%%Function: CompareEntry
-------------------------------------------------------------------------*/ int CMRUList2::CompareEntry(int iItem, PMRUE pEntry) { ASSERT(NULL != pEntry);
int iRet = 0;
PVOID * ppv1 = (PVOID *) GetEntry(iItem); PVOID * ppv2 = (PVOID *) pEntry; for (int iCol = 0; iCol < m_cCol; iCol++, ppv1++, ppv2++) { switch (MruTypeForCol(iCol)) { default: case MRUTYPE_SZ: iRet = lstrcmpi(* (LPCTSTR *) ppv1, * (LPCTSTR *) ppv2); break; case MRUTYPE_DW: iRet = (* (int *) ppv1) - (* (int *) ppv2); break; }
if (0 != iRet) break; }
return iRet; }
/* F I N D E N T R Y */ /*-------------------------------------------------------------------------
%%Function: FindEntry
Return -1 if the item is not found. -------------------------------------------------------------------------*/ int CMRUList2::FindEntry(PMRUE pEntry) { int cItems = GetNumEntries(); for (int i = 0; i < cItems; i++) { if (0 == CompareEntry(i, pEntry)) { return i; } }
return -1; // not found
}
/* S A V E */ /*-------------------------------------------------------------------------
%%Function: Save
-------------------------------------------------------------------------*/ HRESULT CMRUList2::Save(void) { DBGENTRY(CMRUList2::Save);
// Retrieve the data from the registry
RegEntry re(PszRegKey(), HKEY_CURRENT_USER); if (ERROR_SUCCESS != re.GetError()) return E_FAIL;
re.SetValue(REGVAL_MRU_COUNT, m_cEntry); for (int i = 0; i < m_cEntry; i++) { StoreEntry(&re, i); }
return S_OK; }
/* S H I F T E N T R I E S D O W N */ /*-------------------------------------------------------------------------
%%Function: ShiftEntriesDown
Shift the entires down by one slot leaving the first position open. -------------------------------------------------------------------------*/ VOID CMRUList2::ShiftEntriesDown(int cItem) { if (cItem < 1) return; // nothing to do
int iItem; for (iItem = cItem; iItem > 0; iItem--) { m_rgpEntry[iItem] = m_rgpEntry[iItem-1]; }
// the list has been modified
m_fDirty = TRUE; }
/* M O V E E N T R Y T O T O P */ /*-------------------------------------------------------------------------
%%Function: MoveEntryToTop
-------------------------------------------------------------------------*/ VOID CMRUList2::MoveEntryToTop(int iItem) { DBGENTRY(CMRUList2::MoveEntryToTop);
if ((iItem < 1) || (iItem >= m_cEntry)) return; // nothing to do
PMRUE pEntry = GetEntry(iItem); ShiftEntriesDown(iItem); m_rgpEntry[0] = pEntry; }
/* A D D E N T R Y */ /*-------------------------------------------------------------------------
%%Function: AddEntry
Put the entry into the top of the list. The data is owned by the list after this. Returns: S_OK - added to the head of the list S_FALSE - already in list (item is moved to top) -------------------------------------------------------------------------*/ HRESULT CMRUList2::AddEntry(PMRUE pEntry) { DBGENTRY(CMRUList2::AddEntry);
// the list has been modified
m_fDirty = TRUE;
int iItem = FindEntry(pEntry); if (-1 != iItem) { // This entry already exists, move it to the top:
MoveEntryToTop(iItem); DeleteEntry(pEntry); // don't need this data
return S_FALSE; // Success, but already in the list
}
int cShift; if (m_cEntryMax == m_cEntry) { // drop the last item
DeleteEntry(m_rgpEntry[m_cEntry-1]); cShift = m_cEntry-1; } else { cShift = m_cEntry; m_cEntry++; } ShiftEntriesDown(cShift);
// add it to the head of the list
m_rgpEntry[0] = pEntry;
return S_OK; }
HRESULT CMRUList2::AddEntry(LPCTSTR pcsz) { LPTSTR * ppsz = new LPTSTR; LPTSTR psz = PszAlloc(pcsz); if ((NULL == ppsz) || (NULL == psz)) { delete ppsz; delete psz; return E_OUTOFMEMORY; }
*ppsz = psz;
return AddEntry((PMRUE) ppsz); }
HRESULT CMRUList2::AddEntry(LPCTSTR pcsz1, LPCTSTR pcsz2) { PSZSZ pSzSz = new SZSZ; if (NULL == pSzSz) return E_OUTOFMEMORY;
pSzSz->psz1 = PszAlloc(pcsz1); pSzSz->psz2 = PszAlloc(pcsz2); if ((NULL == pSzSz->psz1) || (NULL == pSzSz->psz1)) { // something failed - don't add anything
DeleteEntry(pSzSz); return E_OUTOFMEMORY; } return AddEntry((PMRUE) pSzSz); }
HRESULT CMRUList2::AddEntry(LPCTSTR pcsz1, LPCTSTR pcsz2, DWORD dw3) { PSZSZDW pData = new SZSZDW; if (NULL == pData) return E_OUTOFMEMORY;
pData->psz1 = PszAlloc(pcsz1); pData->psz2 = PszAlloc(pcsz2); if ((NULL == pData->psz1) || (NULL == pData->psz1)) { // something failed - don't add anything
DeleteEntry(pData); return E_OUTOFMEMORY; } pData->dw = dw3; return AddEntry((PMRUE) pData); }
/* P S Z E N T R Y */ /*-------------------------------------------------------------------------
%%Function: PszEntry
Return the main string associated with the entry -------------------------------------------------------------------------*/ LPCTSTR CMRUList2::PszEntry(int iItem) { PMRUE pEntry = GetEntry(iItem); return (LPCTSTR) * ((LPTSTR *)pEntry); }
LPCTSTR CMRUList2::PszData2(int iItem) { PMRUE pEntry = GetEntry(iItem); LPTSTR * ppsz = ((LPTSTR *)pEntry); return * (ppsz+1); }
DWORD_PTR CMRUList2::PszData3(int iItem) { PMRUE pEntry = GetEntry(iItem); LPTSTR * ppsz = ((LPTSTR *)pEntry); return (DWORD_PTR) * (ppsz+2); }
|