windows-nt/Source/XPSP1/NT/net/config/shell/folder/ienumidl.cpp

495 lines
13 KiB
C++
Raw Permalink Normal View History

2020-09-26 03:20:57 -05:00
//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1997.
//
// File: I E N U M I D L . C P P
//
// Contents: IEnumIDList implementation for CConnectionFolderEnum
//
// Notes:
//
// Author: jeffspr 22 Sep 1997
//
//----------------------------------------------------------------------------
#include "pch.h"
#pragma hdrstop
#include "foldinc.h" // Standard shell\folder includes
#include "ncnetcon.h"
#include "webview.h"
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::CConnectionFolderEnum
//
// Purpose: Constructor for the enumerator
//
// Arguments:
// (none)
//
// Returns:
//
// Author: jeffspr 18 Mar 1998
//
// Notes:
//
CConnectionFolderEnum::CConnectionFolderEnum()
{
TraceFileFunc(ttidShellFolderIface);
m_pidlFolder.Clear();
m_dwFlags = 0;
m_fTray = FALSE;
m_dwEnumerationType = CFCOPT_ENUMALL; // all connection types
}
//+---------------------------------------------------------------------------
//
// Function: CConnectionFolderEnum
//
// Purpose: Destructor for the enumerator. Standard cleanup.
//
// Arguments:
// (none)
//
// Returns:
//
// Author: jeffspr 18 Mar 1998
//
// Notes:
//
CConnectionFolderEnum::~CConnectionFolderEnum()
{
TraceFileFunc(ttidShellFolderIface);
m_pidlFolder.Clear();
}
//+---------------------------------------------------------------------------
//
// Function: CConnectionFolderEnum::PidlInitialize
//
// Purpose: Initialization for the enumerator object
//
// Arguments:
// fTray [in] Are we owned by the tray
// pidlFolder [in] Pidl for the folder itself
// dwEnumerationType [in] Enumeration type (inbound/outbound/all)
//
// Returns:
//
// Author: jeffspr 18 Mar 1998
//
// Notes:
//
VOID CConnectionFolderEnum::PidlInitialize(
BOOL fTray,
const PCONFOLDPIDLFOLDER& pidlFolder,
DWORD dwEnumerationType)
{
TraceFileFunc(ttidShellFolderIface);
NETCFG_TRY
m_fTray = fTray;
m_pidlFolder = pidlFolder;
m_dwEnumerationType = dwEnumerationType;
NETCFG_CATCH_AND_RETHROW
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::CreateInstance
//
// Purpose: Create an instance of the CConnectionFolderEnum object, and
// returns the requested interface
//
// Arguments:
// riid [in] Interface requested
// ppv [out] Pointer to receive the requested interface
//
// Returns: Standard OLE HRESULT
//
// Author: jeffspr 5 Nov 1997
//
// Notes:
//
HRESULT CConnectionFolderEnum::CreateInstance(
REFIID riid,
void** ppv)
{
TraceFileFunc(ttidShellFolderIface);
HRESULT hr = E_OUTOFMEMORY;
CConnectionFolderEnum * pObj = NULL;
pObj = new CComObject <CConnectionFolderEnum>;
if (pObj)
{
// Do the standard CComCreator::CreateInstance stuff.
//
pObj->SetVoid (NULL);
pObj->InternalFinalConstructAddRef ();
hr = pObj->FinalConstruct ();
pObj->InternalFinalConstructRelease ();
if (SUCCEEDED(hr))
{
hr = pObj->QueryInterface (riid, ppv);
}
if (FAILED(hr))
{
delete pObj;
}
}
TraceHr(ttidError, FAL, hr, FALSE, "CConnectionFolderEnum::CreateInstance");
return hr;
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::Next
//
// Purpose: Retrieves the specified number of item identifiers in the
// enumeration sequence and advances the current position
// by the number of items retrieved.
//
// Arguments:
// celt [] Max number requested
// rgelt [] Array to fill
// pceltFetched [] Return count for # filled.
//
// Returns: S_OK if successful, S_FALSE if there are no more items
// in the enumeration sequence, or an OLE-defined error value
// otherwise.
//
// Author: jeffspr 5 Nov 1997
//
// Notes:
//
STDMETHODIMP CConnectionFolderEnum::Next(
ULONG celt,
LPITEMIDLIST * rgelt,
ULONG * pceltFetched)
{
TraceFileFunc(ttidShellFolderIface);
HRESULT hr = S_OK;
Assert(celt >= 1);
Assert(rgelt);
Assert(pceltFetched || (celt == 1));
// If the caller asks for the fetch count, zero it out for now.
//
if (pceltFetched)
{
*pceltFetched = 0;
}
// Init the output list pointer
//
*rgelt = NULL;
// If there's not currently a list, build one.
//
if (m_apidl.empty())
{
hr = Reset();
// This will have returned either S_FALSE (no wizard? weird!), an
// error (meaning creating the wizard failed), or S_OK, meaning
// that (at least) the wizard creation succeeded. Enum of the connections
// failing will get filtered by Reset().
}
if (SUCCEEDED(hr))
{
// If there are NOW items in the list
//
if (!m_apidl.empty() )
{
BOOL fMatchFound = FALSE;
// Check that we've set the current pointer to at least the root
//
// Normalize the return code
hr = S_OK;
while ((S_OK == hr) && !fMatchFound)
{
// If there are no remaining entries, return S_FALSE.
//
if ( m_iterPidlCurrent == m_apidl.end() )
{
hr = S_FALSE;
}
else
{
const PCONFOLDPIDL& pcfp = *m_iterPidlCurrent;
// Else, Return the first entry, then increment the current
// pointer
//
Assert(!pcfp.empty());
// Check to see if we want to return this type, based on
// the enumeration type & connection type. The wizard
// should always be included.
//
if ( WIZARD_NOT_WIZARD != pcfp->wizWizard )
{
if (HrIsWebViewEnabled() == S_OK)
{
m_iterPidlCurrent++; // skip over this item
continue;
}
else
{
fMatchFound = TRUE;
}
}
else
{
switch(m_dwEnumerationType)
{
case CFCOPT_ENUMALL:
fMatchFound = TRUE;
break;
case CFCOPT_ENUMINCOMING:
fMatchFound = (pcfp->dwCharacteristics & NCCF_INCOMING_ONLY);
break;
case CFCOPT_ENUMOUTGOING:
fMatchFound = !(pcfp->dwCharacteristics & NCCF_INCOMING_ONLY);
break;
}
}
// If we've found one that needn't be filtered out,
// then fill in the return param, etc.
//
if (fMatchFound)
{
// Copy the pidl for return
//
rgelt[0] = m_iterPidlCurrent->TearOffItemIdList();
if (!rgelt[0])
{
hr = E_OUTOFMEMORY;
}
else
{
// If they requested a return count, fill it in.
//
if (pceltFetched)
{
*pceltFetched = 1;
}
// ISSUE:
// IsValidPIDL is debug code. However, we're doing this in release mode until we
// find the bug from NTRAID#NTBUG9-125787-2000/07/26-deonb.
#ifdef DBG_VALIDATE_PIDLS
if (!IsValidPIDL(rgelt[0]))
{
return E_ABORT;
}
#endif
}
}
// Move the pointer to the next pidl in the list.
//
m_iterPidlCurrent++;
}
}
}
else
{
// There are no items in the list, return S_FALSE
//
hr = S_FALSE;
}
}
#ifdef DBG
if (pceltFetched)
{
TraceTag(ttidShellFolderIface, "IEnumIDList::Next generated PIDL: 0x%08x", rgelt[0]);
}
#endif
TraceHr(ttidError, FAL, hr, (S_FALSE == hr), "CConnectionFolderEnum::Next");
return hr;
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::Skip
//
// Purpose: Skips over the specified number of elements in the
// enumeration sequence.
//
// Arguments:
// celt [in] Number of item identifiers to skip.
//
// Returns: Returns S_OK if successful, or an OLE-defined error
// value otherwise.
//
// Author: jeffspr 5 Nov 1997
//
// Notes:
//
STDMETHODIMP CConnectionFolderEnum::Skip(
ULONG celt)
{
TraceFileFunc(ttidShellFolderIface);
HRESULT hr = S_OK;
NYI("CConnectionFolderEnum::Skip");
// Currently, do nothing
//
TraceHr(ttidError, FAL, hr, FALSE, "CConnectionFolderEnum::Skip");
return S_OK;
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::Reset
//
// Purpose: Returns to the beginning of the enumeration sequence. For us,
// this means do all of the actual enumeration
//
// Arguments:
// (none)
//
// Returns: Returns S_OK if successful, or an OLE-defined error
// value otherwise.
//
// Author: jeffspr 5 Nov 1997
//
// Notes:
//
STDMETHODIMP CConnectionFolderEnum::Reset()
{
TraceFileFunc(ttidShellFolderIface);
HRESULT hr = S_OK;
// If there's already a list, free it and rebuild.
//
if (!m_apidl.empty())
{
m_apidl.clear();
m_iterPidlCurrent = m_apidl.end();
}
// Yes, I know that the code below looks strange, as both cases do the same thing,
// but it makes it a bit easier to debug, and it makes the comments more obvious.
//
hr = HrRetrieveConManEntries();
if (SUCCEEDED(hr))
{
// Normalize the return code. HrRetrieveConManEntries... may have returned
// S_FALSE, meaning that there we no connections (fine).
//
hr = S_OK;
m_iterPidlCurrent = m_apidl.begin();
}
else
{
// Actually, we're still going to return noerror here after tracing the problem,
// as we don't want to keep the enumerator from returning an error
// if the wizard is present (no connections, but hey, better than nothing).
//
TraceHr(ttidError, FAL, hr, FALSE,
"CConnectionsFolderEnum failed in call to HrRetrieveConManEntries");
hr = S_FALSE;
}
TraceHr(ttidError, FAL, hr, FALSE, "CConnectionFolderEnum::Reset");
return hr;
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::Clone
//
// Purpose: Creates a new item enumeration object with the same contents
// and state as the current one.
//
// Arguments:
// ppenum [out] Return a clone of the current internal PIDL
//
// Returns: Returns S_OK if successful, or an OLE-defined error
// value otherwise.
//
// Author: jeffspr 5 Nov 1997
//
// Notes:
//
STDMETHODIMP CConnectionFolderEnum::Clone(
IEnumIDList ** ppenum)
{
TraceFileFunc(ttidShellFolderIface);
NYI("CConnectionFolderEnum::Clone");
*ppenum = NULL;
return E_NOTIMPL;
}
//+---------------------------------------------------------------------------
//
// Member: CConnectionFolderEnum::HrRetrieveConManEntries
//
// Purpose: Enumerate all connections from the ConnectionManagers, and
// add them to our IDL.
//
// Arguments:
// (none)
//
// Returns:
//
// Author: jeffspr 8 Oct 1997
//
// Notes:
//
HRESULT CConnectionFolderEnum::HrRetrieveConManEntries()
{
TraceFileFunc(ttidShellFolderIface);
HRESULT hr = S_OK;
NETCFG_TRY
PCONFOLDPIDLVEC apidlNew;
hr = g_ccl.HrRetrieveConManEntries(apidlNew);
if (SUCCEEDED(hr))
{
m_apidl.clear();
m_apidl = apidlNew;
}
TraceHr(ttidError, FAL, hr, FALSE, "CConnectionFolder::HrRetrieveConManEntries");
NETCFG_CATCH(hr)
return hr;
}