Avoid calling FindExecutable16.
[wine] / dlls / shell32 / changenotify.c
1 /*
2  *      shell change notification
3  *
4  * Copyright 2000 Juergen Schmied
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #include <string.h>
22
23 #include "wine/debug.h"
24 #include "pidl.h"
25 #include "shell32_main.h"
26 #include "undocshell.h"
27
28 WINE_DEFAULT_DEBUG_CHANNEL(shell);
29
30 static CRITICAL_SECTION SHELL32_ChangenotifyCS = CRITICAL_SECTION_INIT("SHELL32_ChangenotifyCS");
31
32 /* internal list of notification clients (internal) */
33 typedef struct _NOTIFICATIONLIST
34 {
35         struct _NOTIFICATIONLIST *next;
36         struct _NOTIFICATIONLIST *prev;
37         HWND hwnd;              /* window to notify */
38         DWORD uMsg;             /* message to send */
39         LPNOTIFYREGISTER apidl; /* array of entries to watch*/
40         UINT cidl;              /* number of pidls in array */
41         LONG wEventMask;        /* subscribed events */
42         DWORD dwFlags;          /* client flags */
43 } NOTIFICATIONLIST, *LPNOTIFICATIONLIST;
44
45 static NOTIFICATIONLIST head;
46 static NOTIFICATIONLIST tail;
47
48 void InitChangeNotifications()
49 {
50         TRACE("head=%p tail=%p\n", &head, &tail);
51         head.next = &tail;
52         tail.prev = &head;
53 }
54
55 void FreeChangeNotifications()
56 {
57         LPNOTIFICATIONLIST ptr, item;
58
59         TRACE("\n");
60
61         EnterCriticalSection(&SHELL32_ChangenotifyCS);
62         ptr = head.next;
63
64         while(ptr != &tail)
65         {
66           int i;
67           item = ptr;
68           ptr = ptr->next;
69
70           TRACE("item=%p\n", item);
71
72           /* free the item */
73           for (i=0; i<item->cidl;i++) SHFree(item->apidl[i].pidlPath);
74           SHFree(item->apidl);
75           SHFree(item);
76         }
77         head.next = NULL;
78         tail.prev = NULL;
79
80         LeaveCriticalSection(&SHELL32_ChangenotifyCS);
81
82         DeleteCriticalSection(&SHELL32_ChangenotifyCS);
83 }
84
85 static BOOL AddNode(LPNOTIFICATIONLIST item)
86 {
87         LPNOTIFICATIONLIST last;
88
89         EnterCriticalSection(&SHELL32_ChangenotifyCS);
90
91         /* get last entry */
92         last = tail.prev;
93
94         /* link items */
95         last->next = item;
96         item->prev = last;
97         item->next = &tail;
98         tail.prev = item;
99         TRACE("item=%p prev=%p next=%p\n", item, item->prev, item->next);
100
101         LeaveCriticalSection(&SHELL32_ChangenotifyCS);
102
103         return TRUE;
104 }
105
106 static BOOL DeleteNode(LPNOTIFICATIONLIST item)
107 {
108         LPNOTIFICATIONLIST ptr;
109         int ret = FALSE;
110
111         TRACE("item=%p\n", item);
112
113         EnterCriticalSection(&SHELL32_ChangenotifyCS);
114
115         ptr = head.next;
116         while(ptr != &tail)
117         {
118           TRACE("ptr=%p\n", ptr);
119
120           if (ptr == item)
121           {
122             int i;
123
124             TRACE("item=%p prev=%p next=%p\n", item, item->prev, item->next);
125
126             /* remove item from list */
127             item->prev->next = item->next;
128             item->next->prev = item->prev;
129
130             /* free the item */
131             for (i=0; i<item->cidl;i++) SHFree(item->apidl[i].pidlPath);
132             SHFree(item->apidl);
133             SHFree(item);
134
135             ret = TRUE;
136             break;
137           }
138           ptr = ptr->next;
139         }
140
141         LeaveCriticalSection(&SHELL32_ChangenotifyCS);
142         return ret;
143
144 }
145
146 /*************************************************************************
147  * SHChangeNotifyRegister                       [SHELL32.2]
148  *
149  */
150 HANDLE WINAPI
151 SHChangeNotifyRegister(
152     HWND hwnd,
153     LONG dwFlags,
154     LONG wEventMask,
155     DWORD uMsg,
156     int cItems,
157     LPCNOTIFYREGISTER lpItems)
158 {
159         LPNOTIFICATIONLIST item;
160         int i;
161
162         item = SHAlloc(sizeof(NOTIFICATIONLIST));
163
164         TRACE("(0x%04x,0x%08lx,0x%08lx,0x%08lx,0x%08x,%p) item=%p\n",
165                 hwnd,dwFlags,wEventMask,uMsg,cItems,lpItems,item);
166
167         item->next = NULL;
168         item->prev = NULL;
169         item->cidl = cItems;
170         item->apidl = SHAlloc(sizeof(NOTIFYREGISTER) * cItems);
171         for(i=0;i<cItems;i++)
172         {
173           item->apidl[i].pidlPath = ILClone(lpItems[i].pidlPath);
174           item->apidl[i].bWatchSubtree = lpItems[i].bWatchSubtree;
175         }
176         item->hwnd = hwnd;
177         item->uMsg = uMsg;
178         item->wEventMask = wEventMask;
179         item->dwFlags = dwFlags;
180         AddNode(item);
181         return (HANDLE)item;
182 }
183
184 /*************************************************************************
185  * SHChangeNotifyDeregister                     [SHELL32.4]
186  */
187 BOOL WINAPI
188 SHChangeNotifyDeregister(
189         HANDLE hNotify)
190 {
191         TRACE("(0x%08x)\n",hNotify);
192
193         return DeleteNode((LPNOTIFICATIONLIST)hNotify);
194 }
195
196 /*************************************************************************
197  * SHChangeNotifyUpdateEntryList                [SHELL32.5]
198  */
199 BOOL WINAPI
200 SHChangeNotifyUpdateEntryList(DWORD unknown1, DWORD unknown2,
201                               DWORD unknown3, DWORD unknown4)
202 {
203         FIXME("(0x%08lx, 0x%08lx, 0x%08lx, 0x%08lx)\n",
204               unknown1, unknown2, unknown3, unknown4);
205
206         return -1;
207 }
208
209 /*************************************************************************
210  * SHChangeNotify                               [SHELL32.@]
211  */
212 void WINAPI SHChangeNotifyW (LONG wEventId, UINT  uFlags, LPCVOID dwItem1, LPCVOID dwItem2)
213 {
214         LPITEMIDLIST pidl1=(LPITEMIDLIST)dwItem1, pidl2=(LPITEMIDLIST)dwItem2;
215         LPNOTIFICATIONLIST ptr;
216
217         TRACE("(0x%08lx,0x%08x,%p,%p):stub.\n", wEventId,uFlags,dwItem1,dwItem2);
218
219         /* convert paths in IDLists*/
220         if(uFlags & SHCNF_PATHA)
221         {
222           DWORD dummy;
223           if (dwItem1) SHILCreateFromPathA((LPCSTR)dwItem1, &pidl1, &dummy);
224           if (dwItem2) SHILCreateFromPathA((LPCSTR)dwItem2, &pidl2, &dummy);
225         }
226
227         EnterCriticalSection(&SHELL32_ChangenotifyCS);
228
229         /* loop through the list */
230         ptr = head.next;
231         while(ptr != &tail)
232         {
233           TRACE("trying %p\n", ptr);
234
235           if(wEventId & ptr->wEventMask)
236           {
237             TRACE("notifying\n");
238             SendMessageA(ptr->hwnd, ptr->uMsg, (WPARAM)pidl1, (LPARAM)pidl2);
239           }
240           ptr = ptr->next;
241         }
242
243         LeaveCriticalSection(&SHELL32_ChangenotifyCS);
244
245         if(uFlags & SHCNF_PATHA)
246         {
247             if (pidl1) SHFree(pidl1);
248             if (pidl2) SHFree(pidl2);
249         }
250 }
251
252 /*************************************************************************
253  * SHChangeNotify                               [SHELL32.@]
254  */
255 void WINAPI SHChangeNotifyA (LONG wEventId, UINT  uFlags, LPCVOID dwItem1, LPCVOID dwItem2)
256 {
257         LPITEMIDLIST Pidls[2];
258         LPNOTIFICATIONLIST ptr;
259
260         Pidls[0] = (LPITEMIDLIST)dwItem1;
261         Pidls[1] = (LPITEMIDLIST)dwItem2;
262
263         TRACE("(0x%08lx,0x%08x,%p,%p):stub.\n", wEventId,uFlags,dwItem1,dwItem2);
264
265         /* convert paths in IDLists*/
266         if(uFlags & SHCNF_PATHA)
267         {
268           DWORD dummy;
269           if (Pidls[0]) SHILCreateFromPathA((LPCSTR)dwItem1, &Pidls[0], &dummy);
270           if (Pidls[1]) SHILCreateFromPathA((LPCSTR)dwItem2, &Pidls[1], &dummy);
271         }
272
273         EnterCriticalSection(&SHELL32_ChangenotifyCS);
274
275         /* loop through the list */
276         ptr = head.next;
277         while(ptr != &tail)
278         {
279           TRACE("trying %p\n", ptr);
280
281           if(wEventId & ptr->wEventMask)
282           {
283             TRACE("notifying\n");
284             SendMessageA(ptr->hwnd, ptr->uMsg, (WPARAM)&Pidls, (LPARAM)wEventId);
285           }
286           ptr = ptr->next;
287         }
288
289         LeaveCriticalSection(&SHELL32_ChangenotifyCS);
290
291         /* if we allocated it, free it */
292         if(uFlags & SHCNF_PATHA)
293         {
294             if (Pidls[0]) SHFree(Pidls[0]);
295             if (Pidls[1]) SHFree(Pidls[1]);
296         }
297 }
298
299 /*************************************************************************
300  * SHChangeNotify                               [SHELL32.@]
301  */
302 void WINAPI SHChangeNotifyAW (LONG wEventId, UINT  uFlags, LPCVOID dwItem1, LPCVOID dwItem2)
303 {
304         if(SHELL_OsIsUnicode())
305           SHChangeNotifyW (wEventId, uFlags, dwItem1, dwItem2);
306         else
307           SHChangeNotifyA (wEventId, uFlags, dwItem1, dwItem2);
308 }
309
310 /*************************************************************************
311  * NTSHChangeNotifyRegister                     [SHELL32.640]
312  * NOTES
313  *   Idlist is an array of structures and Count specifies how many items in the array
314  *   (usually just one I think).
315  */
316 DWORD WINAPI NTSHChangeNotifyRegister(
317     HWND hwnd,
318     LONG events1,
319     LONG events2,
320     DWORD msg,
321     int count,
322     LPNOTIFYREGISTER idlist)
323 {
324         FIXME("(0x%04x,0x%08lx,0x%08lx,0x%08lx,0x%08x,%p):stub.\n",
325                 hwnd,events1,events2,msg,count,idlist);
326         return 0;
327 }
328
329 /*************************************************************************
330  * SHChangeNotification_Lock                    [SHELL32.644]
331  */
332 HANDLE WINAPI SHChangeNotification_Lock(
333         HANDLE hMemoryMap,
334         DWORD dwProcessId,
335         LPCITEMIDLIST **lppidls,
336         LPLONG lpwEventId)
337 {
338         FIXME("\n");
339         return 0;
340 }
341
342 /*************************************************************************
343  * SHChangeNotification_Unlock                  [SHELL32.645]
344  */
345 BOOL WINAPI SHChangeNotification_Unlock (
346         HANDLE hLock)
347 {
348         FIXME("\n");
349         return 0;
350 }
351
352 /*************************************************************************
353  * NTSHChangeNotifyDeregister                   [SHELL32.641]
354  */
355 DWORD WINAPI NTSHChangeNotifyDeregister(LONG x1)
356 {
357         FIXME("(0x%08lx):stub.\n",x1);
358         return 0;
359 }
360