xref: /reactos/dll/shellext/netshell/enumlist.cpp (revision 3a49e26f)
1 /*
2  * PROJECT:     ReactOS Shell
3  * LICENSE:     LGPL-2.1-or-later (https://spdx.org/licenses/LGPL-2.1-or-later)
4  * PURPOSE:     CNetworkConnections Shell Folder items enumerator
5  * COPYRIGHT:   Copyright 2008 Johannes Anderwald (johannes.anderwald@reactos.org)
6  */
7 
8 #include "precomp.h"
9 
10 PNETCONIDSTRUCT ILGetConnData(PCITEMID_CHILD pidl)
11 {
12     if (!pidl || !pidl->mkid.cb || pidl->mkid.abID[0] != 0x99)
13         return NULL;
14     return (PNETCONIDSTRUCT)(&pidl->mkid.abID[0]);
15 }
16 
17 PWCHAR ILGetConnName(PCITEMID_CHILD pidl)
18 {
19     PNETCONIDSTRUCT pdata = ILGetConnData(pidl);
20     if (!pdata)
21         return NULL;
22     return (PWCHAR)&pidl->mkid.abID[pdata->uNameOffset];
23 }
24 
25 PWCHAR ILGetDeviceName(PCITEMID_CHILD pidl)
26 {
27     PNETCONIDSTRUCT pdata = ILGetConnData(pidl);
28     if (!pdata)
29         return NULL;
30     return (PWCHAR)&pidl->mkid.abID[pdata->uDeviceNameOffset];
31 }
32 
33 PITEMID_CHILD ILCreateNetConnectItem(INetConnection * pItem)
34 {
35     PITEMID_CHILD pidl;
36     ULONG_PTR size;
37     NETCON_PROPERTIES * pProperties;
38     PNETCONIDSTRUCT pnetid;
39     PWCHAR pwchName;
40 
41     if (pItem->GetProperties(&pProperties) != S_OK)
42         return NULL;
43 
44     size = sizeof(WORD); /* nr of bytes in this item */
45     size += sizeof(NETCONIDSTRUCT);
46     size += (wcslen(pProperties->pszwName) + 1) * sizeof(WCHAR);
47     size += (wcslen(pProperties->pszwDeviceName) + 1) * sizeof(WCHAR);
48 
49     /* Allocate enough memory for the trailing id which will indicate that this is a simple id */
50     pidl = static_cast<LPITEMIDLIST>(SHAlloc(size + sizeof(SHITEMID)));
51     pidl->mkid.cb = (WORD)size;
52     pidl->mkid.abID[0] = 0x99;
53 
54     /* Copy the connection properties */
55     pnetid = ILGetConnData(pidl);
56     pnetid->guidId = pProperties->guidId;
57     pnetid->Status = pProperties->Status;
58     pnetid->MediaType = pProperties->MediaType;
59     pnetid->dwCharacter = pProperties->dwCharacter;
60     pnetid->uNameOffset = sizeof(NETCONIDSTRUCT);
61     pnetid->uDeviceNameOffset = pnetid->uNameOffset + (wcslen(pProperties->pszwName) + 1) * sizeof(WCHAR);
62 
63     pwchName = ILGetConnName(pidl);
64     wcscpy(pwchName, pProperties->pszwName);
65 
66     pwchName = ILGetDeviceName(pidl);
67     wcscpy(pwchName, pProperties->pszwDeviceName);
68 
69     /* Set the trailing id to null */
70     memset((void*)((ULONG_PTR)pidl + size), 0, sizeof(SHITEMID));
71 
72     NcFreeNetconProperties(pProperties);
73 
74     return pidl;
75 }
76 
77 HRESULT ILGetConnection(PCITEMID_CHILD pidl, INetConnection ** pItem)
78 {
79     HRESULT hr;
80     CComPtr<INetConnectionManager> pNetConMan;
81     CComPtr<IEnumNetConnection> pEnumCon;
82     CComPtr<INetConnection> INetCon;
83     ULONG Count;
84     NETCON_PROPERTIES * pProperties;
85 
86     PNETCONIDSTRUCT pdata = ILGetConnData(pidl);
87     if (!pdata)
88         return E_FAIL;
89 
90     /* get an instance to of IConnectionManager */
91     hr = CNetConnectionManager_CreateInstance(IID_PPV_ARG(INetConnectionManager, &pNetConMan));
92     if (FAILED_UNEXPECTEDLY(hr))
93         return hr;
94 
95     hr = pNetConMan->EnumConnections(NCME_DEFAULT, &pEnumCon);
96     if (FAILED_UNEXPECTEDLY(hr))
97         return hr;
98 
99     while (TRUE)
100     {
101         hr = pEnumCon->Next(1, &INetCon, &Count);
102         if (hr != S_OK)
103             return E_FAIL;
104 
105         hr = INetCon->GetProperties(&pProperties);
106         if (FAILED_UNEXPECTEDLY(hr))
107             continue;
108 
109         BOOL bSame = !memcmp(&pProperties->guidId, &pdata->guidId, sizeof(GUID));
110 
111         NcFreeNetconProperties(pProperties);
112 
113         if (bSame)
114         {
115             *pItem = INetCon.Detach();
116             return S_OK;
117         }
118     }
119 
120     return E_FAIL;
121 }
122 
123 typedef struct tagENUMLIST
124 {
125     struct tagENUMLIST *pNext;
126     PITEMID_CHILD pidl;
127 } ENUMLIST, *LPENUMLIST;
128 
129 class CEnumIDList:
130     public CComObjectRootEx<CComMultiThreadModelNoCS>,
131     public IEnumIDList
132 {
133     public:
134         CEnumIDList();
135         ~CEnumIDList();
136 
137         HRESULT Initialize();
138 
139         // IEnumIDList
140         STDMETHOD(Next)(ULONG celt, PITEMID_CHILD *rgelt, ULONG *pceltFetched) override;
141         STDMETHOD(Skip)(ULONG celt) override;
142         STDMETHOD(Reset)() override;
143         STDMETHOD(Clone)(IEnumIDList **ppenum) override;
144 
145     private:
146         BOOL AddToEnumList(PITEMID_CHILD pidl);
147 
148         LPENUMLIST  m_pFirst;
149         LPENUMLIST  m_pLast;
150         LPENUMLIST  m_pCurrent;
151 
152     public:
153         DECLARE_NOT_AGGREGATABLE(CEnumIDList)
154         DECLARE_PROTECT_FINAL_CONSTRUCT()
155 
156         BEGIN_COM_MAP(CEnumIDList)
157             COM_INTERFACE_ENTRY_IID(IID_IEnumIDList, IEnumIDList)
158         END_COM_MAP()
159 };
160 
161 /**************************************************************************
162  *  AddToEnumList()
163  */
164 BOOL
165 CEnumIDList::AddToEnumList(PITEMID_CHILD pidl)
166 {
167     LPENUMLIST pNew;
168 
169     if (!pidl)
170         return FALSE;
171 
172     pNew = static_cast<LPENUMLIST>(SHAlloc(sizeof(ENUMLIST)));
173     if (pNew)
174     {
175         pNew->pNext = NULL;
176         pNew->pidl = pidl;
177 
178         if (!m_pFirst)
179         {
180             m_pFirst = pNew;
181             m_pCurrent = pNew;
182         }
183 
184         if (m_pLast)
185         {
186             /*add the new item to the end of the list */
187             m_pLast->pNext = pNew;
188         }
189 
190         /*update the last item pointer */
191         m_pLast = pNew;
192         return TRUE;
193     }
194     return FALSE;
195 }
196 
197 CEnumIDList::CEnumIDList() :
198     m_pFirst(NULL),
199     m_pLast(NULL),
200     m_pCurrent(NULL)
201 {
202 }
203 
204 CEnumIDList::~CEnumIDList()
205 {
206     LPENUMLIST pDelete;
207 
208     while (m_pFirst)
209     {
210         pDelete = m_pFirst;
211         m_pFirst = pDelete->pNext;
212         SHFree(pDelete->pidl);
213         SHFree(pDelete);
214     }
215 }
216 
217 HRESULT
218 CEnumIDList::Initialize()
219 {
220     HRESULT hr;
221     CComPtr<INetConnectionManager> pNetConMan;
222     CComPtr<IEnumNetConnection> pEnumCon;
223     ULONG Count;
224     PITEMID_CHILD pidl;
225 
226     /* get an instance to of IConnectionManager */
227     hr = CNetConnectionManager_CreateInstance(IID_PPV_ARG(INetConnectionManager, &pNetConMan));
228     if (FAILED_UNEXPECTEDLY(hr))
229         return S_OK;
230 
231     hr = pNetConMan->EnumConnections(NCME_DEFAULT, &pEnumCon);
232     if (FAILED_UNEXPECTEDLY(hr))
233         return S_OK;
234 
235     while (TRUE)
236     {
237         CComPtr<INetConnection> INetCon;
238 
239         hr = pEnumCon->Next(1, &INetCon, &Count);
240         if (hr != S_OK)
241             break;
242 
243         pidl = ILCreateNetConnectItem(INetCon);
244         if (pidl)
245         {
246             AddToEnumList(pidl);
247         }
248     }
249 
250     return S_OK;
251 }
252 
253 HRESULT
254 WINAPI
255 CEnumIDList::Next(
256     ULONG celt,
257     PITEMID_CHILD *rgelt,
258     ULONG *pceltFetched)
259 {
260     ULONG i;
261     HRESULT hr = S_OK;
262     PITEMID_CHILD temp;
263 
264     if (pceltFetched)
265         *pceltFetched = 0;
266 
267     if (celt > 1 && !pceltFetched)
268     {
269         return E_INVALIDARG;
270     }
271 
272     if (celt > 0 && !m_pCurrent)
273     {
274         return S_FALSE;
275     }
276 
277     for (i = 0; i < celt; i++)
278     {
279         if (!m_pCurrent)
280         {
281             hr = S_FALSE;
282             break;
283         }
284 
285         temp = ILClone(m_pCurrent->pidl);
286         if (!temp)
287         {
288             hr = i ? S_FALSE : E_OUTOFMEMORY;
289             break;
290         }
291         rgelt[i] = temp;
292         m_pCurrent = m_pCurrent->pNext;
293     }
294 
295     if (pceltFetched)
296         *pceltFetched = i;
297 
298     return hr;
299 }
300 
301 HRESULT
302 WINAPI
303 CEnumIDList::Skip(ULONG celt)
304 {
305     DWORD dwIndex;
306     HRESULT hr = S_OK;
307 
308     for (dwIndex = 0; dwIndex < celt; dwIndex++)
309     {
310         if (!m_pCurrent)
311         {
312             hr = S_FALSE;
313             break;
314         }
315         m_pCurrent = m_pCurrent->pNext;
316     }
317 
318     return hr;
319 }
320 
321 HRESULT
322 WINAPI
323 CEnumIDList::Reset()
324 {
325     m_pCurrent = m_pFirst;
326     return S_OK;
327 }
328 
329 HRESULT
330 WINAPI
331 CEnumIDList::Clone(
332     LPENUMIDLIST * ppenum)
333 {
334     return E_NOTIMPL;
335 }
336 
337 HRESULT CEnumIDList_CreateInstance(HWND hwndOwner, DWORD dwFlags, REFIID riid, LPVOID * ppv)
338 {
339     return ShellObjectCreatorInit<CEnumIDList>(riid, ppv);
340 }
341