| /* |
| * shell change notification |
| * |
| * Copyright 2000 Juergen Schmied |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <stdarg.h> |
| #include <string.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "wine/list.h" |
| #include "wine/debug.h" |
| #include "shell32_main.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(shell); |
| |
| static CRITICAL_SECTION SHELL32_ChangenotifyCS; |
| static CRITICAL_SECTION_DEBUG critsect_debug = |
| { |
| 0, 0, &SHELL32_ChangenotifyCS, |
| { &critsect_debug.ProcessLocksList, &critsect_debug.ProcessLocksList }, |
| 0, 0, { (DWORD_PTR)(__FILE__ ": SHELL32_ChangenotifyCS") } |
| }; |
| static CRITICAL_SECTION SHELL32_ChangenotifyCS = { &critsect_debug, -1, 0, 0, 0, 0 }; |
| |
| typedef SHChangeNotifyEntry *LPNOTIFYREGISTER; |
| |
| /* internal list of notification clients (internal) */ |
| typedef struct _NOTIFICATIONLIST |
| { |
| struct list entry; |
| HWND hwnd; /* window to notify */ |
| DWORD uMsg; /* message to send */ |
| LPNOTIFYREGISTER apidl; /* array of entries to watch*/ |
| UINT cidl; /* number of pidls in array */ |
| LONG wEventMask; /* subscribed events */ |
| DWORD dwFlags; /* client flags */ |
| ULONG id; |
| } NOTIFICATIONLIST, *LPNOTIFICATIONLIST; |
| |
| static struct list notifications = LIST_INIT( notifications ); |
| static LONG next_id; |
| |
| #define SHCNE_NOITEMEVENTS ( \ |
| SHCNE_ASSOCCHANGED ) |
| |
| #define SHCNE_ONEITEMEVENTS ( \ |
| SHCNE_ATTRIBUTES | SHCNE_CREATE | SHCNE_DELETE | SHCNE_DRIVEADD | \ |
| SHCNE_DRIVEADDGUI | SHCNE_DRIVEREMOVED | SHCNE_FREESPACE | \ |
| SHCNE_MEDIAINSERTED | SHCNE_MEDIAREMOVED | SHCNE_MKDIR | \ |
| SHCNE_NETSHARE | SHCNE_NETUNSHARE | SHCNE_RMDIR | \ |
| SHCNE_SERVERDISCONNECT | SHCNE_UPDATEDIR | SHCNE_UPDATEIMAGE ) |
| |
| #define SHCNE_TWOITEMEVENTS ( \ |
| SHCNE_RENAMEFOLDER | SHCNE_RENAMEITEM | SHCNE_UPDATEITEM ) |
| |
| /* for dumping events */ |
| static const char * DumpEvent( LONG event ) |
| { |
| if( event == SHCNE_ALLEVENTS ) |
| return "SHCNE_ALLEVENTS"; |
| #define DUMPEV(x) ,( event & SHCNE_##x )? #x " " : "" |
| return wine_dbg_sprintf( "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s" |
| DUMPEV(RENAMEITEM) |
| DUMPEV(CREATE) |
| DUMPEV(DELETE) |
| DUMPEV(MKDIR) |
| DUMPEV(RMDIR) |
| DUMPEV(MEDIAINSERTED) |
| DUMPEV(MEDIAREMOVED) |
| DUMPEV(DRIVEREMOVED) |
| DUMPEV(DRIVEADD) |
| DUMPEV(NETSHARE) |
| DUMPEV(NETUNSHARE) |
| DUMPEV(ATTRIBUTES) |
| DUMPEV(UPDATEDIR) |
| DUMPEV(UPDATEITEM) |
| DUMPEV(SERVERDISCONNECT) |
| DUMPEV(UPDATEIMAGE) |
| DUMPEV(DRIVEADDGUI) |
| DUMPEV(RENAMEFOLDER) |
| DUMPEV(FREESPACE) |
| DUMPEV(EXTENDED_EVENT) |
| DUMPEV(ASSOCCHANGED) |
| DUMPEV(INTERRUPT) |
| ); |
| #undef DUMPEV |
| } |
| |
| static const char * NodeName(const NOTIFICATIONLIST *item) |
| { |
| const char *str; |
| WCHAR path[MAX_PATH]; |
| |
| if(SHGetPathFromIDListW(item->apidl[0].pidl, path )) |
| str = wine_dbg_sprintf("%s", debugstr_w(path)); |
| else |
| str = wine_dbg_sprintf("<not a disk file>" ); |
| return str; |
| } |
| |
| static void DeleteNode(LPNOTIFICATIONLIST item) |
| { |
| UINT i; |
| |
| TRACE("item=%p\n", item); |
| |
| /* remove item from list */ |
| list_remove( &item->entry ); |
| |
| /* free the item */ |
| for (i=0; i<item->cidl; i++) |
| SHFree((LPITEMIDLIST)item->apidl[i].pidl); |
| SHFree(item->apidl); |
| SHFree(item); |
| } |
| |
| void InitChangeNotifications(void) |
| { |
| } |
| |
| void FreeChangeNotifications(void) |
| { |
| LPNOTIFICATIONLIST ptr, next; |
| |
| TRACE("\n"); |
| |
| EnterCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| LIST_FOR_EACH_ENTRY_SAFE( ptr, next, ¬ifications, NOTIFICATIONLIST, entry ) |
| DeleteNode( ptr ); |
| |
| LeaveCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| DeleteCriticalSection(&SHELL32_ChangenotifyCS); |
| } |
| |
| /************************************************************************* |
| * SHChangeNotifyRegister [SHELL32.2] |
| * |
| */ |
| ULONG WINAPI |
| SHChangeNotifyRegister( |
| HWND hwnd, |
| int fSources, |
| LONG wEventMask, |
| UINT uMsg, |
| int cItems, |
| SHChangeNotifyEntry *lpItems) |
| { |
| LPNOTIFICATIONLIST item; |
| int i; |
| |
| item = SHAlloc(sizeof(NOTIFICATIONLIST)); |
| |
| TRACE("(%p,0x%08x,0x%08x,0x%08x,%d,%p) item=%p\n", |
| hwnd, fSources, wEventMask, uMsg, cItems, lpItems, item); |
| |
| item->cidl = cItems; |
| item->apidl = SHAlloc(sizeof(SHChangeNotifyEntry) * cItems); |
| for(i=0;i<cItems;i++) |
| { |
| item->apidl[i].pidl = ILClone(lpItems[i].pidl); |
| item->apidl[i].fRecursive = lpItems[i].fRecursive; |
| } |
| item->hwnd = hwnd; |
| item->uMsg = uMsg; |
| item->wEventMask = wEventMask; |
| item->dwFlags = fSources; |
| item->id = InterlockedIncrement( &next_id ); |
| |
| TRACE("new node: %s\n", NodeName( item )); |
| |
| EnterCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| list_add_tail( ¬ifications, &item->entry ); |
| |
| LeaveCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| return item->id; |
| } |
| |
| /************************************************************************* |
| * SHChangeNotifyDeregister [SHELL32.4] |
| */ |
| BOOL WINAPI SHChangeNotifyDeregister(ULONG hNotify) |
| { |
| LPNOTIFICATIONLIST node; |
| |
| TRACE("(0x%08x)\n", hNotify); |
| |
| EnterCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| LIST_FOR_EACH_ENTRY( node, ¬ifications, NOTIFICATIONLIST, entry ) |
| { |
| if (node->id == hNotify) |
| { |
| DeleteNode( node ); |
| LeaveCriticalSection(&SHELL32_ChangenotifyCS); |
| return TRUE; |
| } |
| } |
| LeaveCriticalSection(&SHELL32_ChangenotifyCS); |
| return FALSE; |
| } |
| |
| /************************************************************************* |
| * SHChangeNotifyUpdateEntryList [SHELL32.5] |
| */ |
| BOOL WINAPI SHChangeNotifyUpdateEntryList(DWORD unknown1, DWORD unknown2, |
| DWORD unknown3, DWORD unknown4) |
| { |
| FIXME("(0x%08x, 0x%08x, 0x%08x, 0x%08x)\n", |
| unknown1, unknown2, unknown3, unknown4); |
| |
| return TRUE; |
| } |
| |
| struct new_delivery_notification |
| { |
| LONG event; |
| DWORD pidl1_size; |
| DWORD pidl2_size; |
| LPITEMIDLIST pidls[2]; |
| BYTE data[1]; |
| }; |
| |
| static BOOL should_notify( LPCITEMIDLIST changed, LPCITEMIDLIST watched, BOOL sub ) |
| { |
| TRACE("%p %p %d\n", changed, watched, sub ); |
| if ( !watched ) |
| return FALSE; |
| if (ILIsEqual( watched, changed ) ) |
| return TRUE; |
| if( sub && ILIsParent( watched, changed, FALSE ) ) |
| return TRUE; |
| return FALSE; |
| } |
| |
| /************************************************************************* |
| * SHChangeNotify [SHELL32.@] |
| */ |
| void WINAPI SHChangeNotify(LONG wEventId, UINT uFlags, LPCVOID dwItem1, LPCVOID dwItem2) |
| { |
| struct notification_recipients { |
| struct list entry; |
| HWND hwnd; |
| DWORD msg; |
| DWORD flags; |
| } *cur, *next; |
| |
| HANDLE shared_data = NULL; |
| LPITEMIDLIST Pidls[2]; |
| LPNOTIFICATIONLIST ptr; |
| struct list recipients; |
| |
| Pidls[0] = NULL; |
| Pidls[1] = NULL; |
| |
| TRACE("(0x%08x,0x%08x,%p,%p)\n", wEventId, uFlags, dwItem1, dwItem2); |
| |
| if(uFlags & ~(SHCNF_TYPE|SHCNF_FLUSH)) |
| FIXME("ignoring unsupported flags: %x\n", uFlags); |
| |
| if( ( wEventId & SHCNE_NOITEMEVENTS ) && ( dwItem1 || dwItem2 ) ) |
| { |
| TRACE("dwItem1 and dwItem2 are not zero, but should be\n"); |
| dwItem1 = 0; |
| dwItem2 = 0; |
| return; |
| } |
| else if( ( wEventId & SHCNE_ONEITEMEVENTS ) && dwItem2 ) |
| { |
| TRACE("dwItem2 is not zero, but should be\n"); |
| dwItem2 = 0; |
| return; |
| } |
| |
| if( ( ( wEventId & SHCNE_NOITEMEVENTS ) && |
| ( wEventId & ~SHCNE_NOITEMEVENTS ) ) || |
| ( ( wEventId & SHCNE_ONEITEMEVENTS ) && |
| ( wEventId & ~SHCNE_ONEITEMEVENTS ) ) || |
| ( ( wEventId & SHCNE_TWOITEMEVENTS ) && |
| ( wEventId & ~SHCNE_TWOITEMEVENTS ) ) ) |
| { |
| WARN("mutually incompatible events listed\n"); |
| return; |
| } |
| |
| /* convert paths in IDLists*/ |
| switch (uFlags & SHCNF_TYPE) |
| { |
| case SHCNF_PATHA: |
| if (dwItem1) Pidls[0] = SHSimpleIDListFromPathA(dwItem1); |
| if (dwItem2) Pidls[1] = SHSimpleIDListFromPathA(dwItem2); |
| break; |
| case SHCNF_PATHW: |
| if (dwItem1) Pidls[0] = SHSimpleIDListFromPathW(dwItem1); |
| if (dwItem2) Pidls[1] = SHSimpleIDListFromPathW(dwItem2); |
| break; |
| case SHCNF_IDLIST: |
| Pidls[0] = ILClone(dwItem1); |
| Pidls[1] = ILClone(dwItem2); |
| break; |
| case SHCNF_PRINTERA: |
| case SHCNF_PRINTERW: |
| FIXME("SHChangeNotify with (uFlags & SHCNF_PRINTER)\n"); |
| return; |
| case SHCNF_DWORD: |
| default: |
| FIXME("unknown type %08x\n", uFlags & SHCNF_TYPE); |
| return; |
| } |
| |
| list_init(&recipients); |
| EnterCriticalSection(&SHELL32_ChangenotifyCS); |
| LIST_FOR_EACH_ENTRY( ptr, ¬ifications, NOTIFICATIONLIST, entry ) |
| { |
| struct notification_recipients *item; |
| BOOL notify = FALSE; |
| DWORD i; |
| |
| for( i=0; (i<ptr->cidl) && !notify ; i++ ) |
| { |
| LPCITEMIDLIST pidl = ptr->apidl[i].pidl; |
| BOOL subtree = ptr->apidl[i].fRecursive; |
| |
| if (wEventId & ptr->wEventMask) |
| { |
| if( !pidl ) /* all ? */ |
| notify = TRUE; |
| else if( wEventId & SHCNE_NOITEMEVENTS ) |
| notify = TRUE; |
| else if( wEventId & ( SHCNE_ONEITEMEVENTS | SHCNE_TWOITEMEVENTS ) ) |
| notify = should_notify( Pidls[0], pidl, subtree ); |
| else if( wEventId & SHCNE_TWOITEMEVENTS ) |
| notify = should_notify( Pidls[1], pidl, subtree ); |
| } |
| } |
| |
| if( !notify ) |
| continue; |
| |
| item = SHAlloc(sizeof(struct notification_recipients)); |
| if(!item) { |
| ERR("out of memory\n"); |
| continue; |
| } |
| |
| item->hwnd = ptr->hwnd; |
| item->msg = ptr->uMsg; |
| item->flags = ptr->dwFlags; |
| list_add_tail(&recipients, &item->entry); |
| } |
| LeaveCriticalSection(&SHELL32_ChangenotifyCS); |
| |
| LIST_FOR_EACH_ENTRY_SAFE(cur, next, &recipients, struct notification_recipients, entry) |
| { |
| TRACE("notifying %p, event %s(%x)\n", cur->hwnd, DumpEvent(wEventId), wEventId); |
| |
| if (cur->flags & SHCNRF_NewDelivery) { |
| if(!shared_data) { |
| struct new_delivery_notification *notification; |
| UINT size1 = ILGetSize(Pidls[0]), size2 = ILGetSize(Pidls[1]); |
| UINT offset = (size1+sizeof(int)-1)/sizeof(int)*sizeof(int); |
| |
| notification = SHAlloc(sizeof(struct new_delivery_notification)+offset+size2); |
| if(!notification) { |
| ERR("out of memory\n"); |
| } else { |
| notification->event = wEventId; |
| notification->pidl1_size = size1; |
| notification->pidl2_size = size2; |
| if(size1) |
| memcpy(notification->data, Pidls[0], size1); |
| if(size2) |
| memcpy(notification->data+offset, Pidls[1], size2); |
| |
| shared_data = SHAllocShared(notification, |
| sizeof(struct new_delivery_notification)+size1+size2, |
| GetCurrentProcessId()); |
| SHFree(notification); |
| } |
| } |
| |
| if(shared_data) |
| SendMessageA(cur->hwnd, cur->msg, (WPARAM)shared_data, GetCurrentProcessId()); |
| else |
| ERR("out of memory\n"); |
| } else { |
| SendMessageA(cur->hwnd, cur->msg, (WPARAM)Pidls, wEventId); |
| } |
| |
| list_remove(&cur->entry); |
| SHFree(cur); |
| } |
| SHFreeShared(shared_data, GetCurrentProcessId()); |
| SHFree(Pidls[0]); |
| SHFree(Pidls[1]); |
| |
| if (wEventId & SHCNE_ASSOCCHANGED) |
| { |
| static const WCHAR args[] = {' ','-','a',0 }; |
| TRACE("refreshing file type associations\n"); |
| run_winemenubuilder( args ); |
| } |
| } |
| |
| /************************************************************************* |
| * NTSHChangeNotifyRegister [SHELL32.640] |
| * NOTES |
| * Idlist is an array of structures and Count specifies how many items in the array |
| * (usually just one I think). |
| */ |
| DWORD WINAPI NTSHChangeNotifyRegister( |
| HWND hwnd, |
| LONG events1, |
| LONG events2, |
| DWORD msg, |
| int count, |
| SHChangeNotifyEntry *idlist) |
| { |
| FIXME("(%p,0x%08x,0x%08x,0x%08x,0x%08x,%p):semi stub.\n", |
| hwnd,events1,events2,msg,count,idlist); |
| |
| return SHChangeNotifyRegister(hwnd, events1, events2, msg, count, idlist); |
| } |
| |
| /************************************************************************* |
| * SHChangeNotification_Lock [SHELL32.644] |
| */ |
| HANDLE WINAPI SHChangeNotification_Lock( |
| HANDLE hChange, |
| DWORD dwProcessId, |
| LPITEMIDLIST **lppidls, |
| LPLONG lpwEventId) |
| { |
| struct new_delivery_notification *ndn; |
| UINT offset; |
| |
| TRACE("%p %08x %p %p\n", hChange, dwProcessId, lppidls, lpwEventId); |
| |
| ndn = SHLockShared(hChange, dwProcessId); |
| if(!ndn) { |
| WARN("SHLockShared failed\n"); |
| return NULL; |
| } |
| |
| if(lppidls) { |
| offset = (ndn->pidl1_size+sizeof(int)-1)/sizeof(int)*sizeof(int); |
| ndn->pidls[0] = ndn->pidl1_size ? (LPITEMIDLIST)ndn->data : NULL; |
| ndn->pidls[1] = ndn->pidl2_size ? (LPITEMIDLIST)(ndn->data+offset) : NULL; |
| *lppidls = ndn->pidls; |
| } |
| |
| if(lpwEventId) |
| *lpwEventId = ndn->event; |
| |
| return ndn; |
| } |
| |
| /************************************************************************* |
| * SHChangeNotification_Unlock [SHELL32.645] |
| */ |
| BOOL WINAPI SHChangeNotification_Unlock ( HANDLE hLock) |
| { |
| TRACE("%p\n", hLock); |
| return SHUnlockShared(hLock); |
| } |
| |
| /************************************************************************* |
| * NTSHChangeNotifyDeregister [SHELL32.641] |
| */ |
| DWORD WINAPI NTSHChangeNotifyDeregister(ULONG x1) |
| { |
| FIXME("(0x%08x):semi stub.\n",x1); |
| |
| return SHChangeNotifyDeregister( x1 ); |
| } |