[4121] | 1 | /*
|
---|
[10604] | 2 | * Systray
|
---|
[4121] | 3 | *
|
---|
[10604] | 4 | * Copyright 1999 Kai Morich <kai.morich@bigfoot.de>
|
---|
[4121] | 5 | *
|
---|
| 6 | * Manage the systray window. That it actually appears in the docking
|
---|
| 7 | * area of KDE or GNOME is delegated to windows/x11drv/wnd.c,
|
---|
| 8 | * X11DRV_WND_DockWindow.
|
---|
| 9 | *
|
---|
[21592] | 10 | * Modified by ErOs2 and Dmitriy Kuminov for usage with systray_os2.c
|
---|
[10604] | 11 | *
|
---|
[4121] | 12 | */
|
---|
[5618] | 13 |
|
---|
| 14 | #ifndef __WIN32OS2__
|
---|
[4121] | 15 | #include <unistd.h>
|
---|
| 16 | #endif
|
---|
| 17 | #include <stdlib.h>
|
---|
| 18 | #include <string.h>
|
---|
| 19 |
|
---|
[5618] | 20 | #include "winnls.h"
|
---|
| 21 | #include "shlobj.h"
|
---|
[4121] | 22 | #include "shellapi.h"
|
---|
| 23 | #include "shell32_main.h"
|
---|
| 24 | #include "commctrl.h"
|
---|
| 25 | #include "debugtools.h"
|
---|
[5618] | 26 | #include "heap.h"
|
---|
[4121] | 27 | #include "config.h"
|
---|
[10604] | 28 | #include "winuser32.h"
|
---|
[21600] | 29 | #include "auxthread.h"
|
---|
[4121] | 30 |
|
---|
[21591] | 31 | #include "systray.h"
|
---|
| 32 |
|
---|
[5618] | 33 | DEFAULT_DEBUG_CHANNEL(shell);
|
---|
[4121] | 34 |
|
---|
[10604] | 35 |
|
---|
[21591] | 36 | struct _SystrayItem {
|
---|
[21592] | 37 | HWND hWnd;
|
---|
| 38 | HWND hWndToolTip;
|
---|
| 39 | NOTIFYICONDATAA notifyIcon;
|
---|
| 40 | UINT uIdx;
|
---|
| 41 | struct _SystrayItem *nextTrayItem;
|
---|
[21591] | 42 | };
|
---|
[4121] | 43 |
|
---|
[21592] | 44 | static SystrayItem *systray = NULL;
|
---|
[4121] | 45 |
|
---|
[21591] | 46 | BOOL (*SYSTRAY_ItemInit)(SystrayItem *ptrayItem) = 0;
|
---|
| 47 | void (*SYSTRAY_ItemTerm)(SystrayItem *ptrayItem) = 0;
|
---|
[21592] | 48 | void (*SYSTRAY_ItemUpdate)(SystrayItem *ptrayItem, ULONG uFlags) = 0;
|
---|
[4121] | 49 |
|
---|
| 50 | static BOOL SYSTRAY_ItemIsEqual(PNOTIFYICONDATAA pnid1, PNOTIFYICONDATAA pnid2)
|
---|
| 51 | {
|
---|
[21592] | 52 | if (pnid1->hWnd != pnid2->hWnd) return FALSE;
|
---|
| 53 | if (pnid1->uID != pnid2->uID) return FALSE;
|
---|
| 54 | return TRUE;
|
---|
[4121] | 55 | }
|
---|
| 56 |
|
---|
| 57 | static BOOL SYSTRAY_Add(PNOTIFYICONDATAA pnid)
|
---|
| 58 | {
|
---|
[21592] | 59 | SystrayItem *pItem = systray, *pPrevItem = NULL, *ptrayItem;
|
---|
| 60 | ULONG uIdx = 0;
|
---|
[4121] | 61 |
|
---|
[21592] | 62 | /* Search for the first free index and also check if already in list */
|
---|
| 63 | while (pItem)
|
---|
| 64 | {
|
---|
| 65 | if (SYSTRAY_ItemIsEqual(pnid, &pItem->notifyIcon))
|
---|
| 66 | return FALSE;
|
---|
[4121] | 67 |
|
---|
[21592] | 68 | uIdx = pPrevItem->uIdx + 1;
|
---|
[4121] | 69 |
|
---|
[21592] | 70 | if (pPrevItem && pItem->uIdx - pPrevItem->uIdx > 1)
|
---|
| 71 | {
|
---|
| 72 | /* found a hole in the index row, will insert here */
|
---|
| 73 | break;
|
---|
| 74 | }
|
---|
| 75 |
|
---|
| 76 | pPrevItem = pItem;
|
---|
| 77 | pItem = pItem->nextTrayItem;
|
---|
| 78 | }
|
---|
| 79 |
|
---|
| 80 | /* check the rest (if any) for duplicates */
|
---|
| 81 | ptrayItem = pItem;
|
---|
| 82 | while (ptrayItem)
|
---|
| 83 | {
|
---|
| 84 | if (SYSTRAY_ItemIsEqual(pnid, &ptrayItem->notifyIcon))
|
---|
| 85 | return FALSE;
|
---|
| 86 | ptrayItem = ptrayItem->nextTrayItem;
|
---|
| 87 | }
|
---|
| 88 |
|
---|
| 89 | /* Allocate SystrayItem for the new element and insert it */
|
---|
| 90 | ptrayItem = ( SystrayItem *)malloc(sizeof(SystrayItem));
|
---|
| 91 | memset(ptrayItem, 0, sizeof(SystrayItem));
|
---|
| 92 | ptrayItem->nextTrayItem = pItem;
|
---|
| 93 |
|
---|
| 94 | if (pPrevItem)
|
---|
| 95 | {
|
---|
| 96 | pPrevItem->nextTrayItem = ptrayItem;
|
---|
| 97 | ptrayItem->uIdx = pPrevItem->uIdx + 1;
|
---|
| 98 | }
|
---|
| 99 | else
|
---|
| 100 | {
|
---|
| 101 | systray = ptrayItem;
|
---|
| 102 | ptrayItem->uIdx = 0;
|
---|
| 103 | }
|
---|
| 104 |
|
---|
| 105 | /* Initialize and set data for the tray element. */
|
---|
| 106 | ptrayItem->uIdx = uIdx;
|
---|
| 107 |
|
---|
| 108 | ptrayItem->notifyIcon.uID = pnid->uID;
|
---|
| 109 | ptrayItem->notifyIcon.hWnd = pnid->hWnd;
|
---|
| 110 |
|
---|
| 111 | ptrayItem->notifyIcon.hIcon =
|
---|
| 112 | (pnid->uFlags & NIF_ICON) ? GetOS2Icon(pnid->hIcon) : 0;
|
---|
| 113 | ptrayItem->notifyIcon.uCallbackMessage =
|
---|
| 114 | (pnid->uFlags & NIF_MESSAGE) ? pnid->uCallbackMessage : 0;
|
---|
| 115 |
|
---|
| 116 | if (pnid->uFlags & NIF_TIP)
|
---|
| 117 | CharToOemA(pnid->szTip, ptrayItem->notifyIcon.szTip);
|
---|
| 118 | else
|
---|
| 119 | ptrayItem->notifyIcon.szTip[0] = '\0';
|
---|
| 120 |
|
---|
| 121 | /* Implementation specific initialization */
|
---|
| 122 | if (!SYSTRAY_ItemInit(ptrayItem))
|
---|
| 123 | {
|
---|
| 124 | free(ptrayItem);
|
---|
| 125 | if (pPrevItem)
|
---|
| 126 | pPrevItem->nextTrayItem = pItem;
|
---|
| 127 | else
|
---|
| 128 | systray = NULL;
|
---|
| 129 | return FALSE;
|
---|
| 130 | }
|
---|
| 131 |
|
---|
| 132 | /* Trigger the data update (flags = 0 means it's the first time) */
|
---|
| 133 | SYSTRAY_ItemUpdate(ptrayItem, 0);
|
---|
| 134 |
|
---|
| 135 | TRACE("SYSTRAY_Add %p: uIdx %u, hWnd 0x%08x, uID %d, hIcon 0x%08x, "
|
---|
| 136 | "uCallbackMessage 0x%08x, szTip [%s]\n",
|
---|
| 137 | ptrayItem, ptrayItem->uIdx, ptrayItem->hWnd, ptrayItem->notifyIcon.uID,
|
---|
| 138 | ptrayItem->notifyIcon.hIcon,ptrayItem->notifyIcon.uCallbackMessage,
|
---|
| 139 | ptrayItem->notifyIcon.szTip);
|
---|
| 140 |
|
---|
| 141 | return TRUE;
|
---|
[4121] | 142 | }
|
---|
| 143 |
|
---|
| 144 | static BOOL SYSTRAY_Modify(PNOTIFYICONDATAA pnid)
|
---|
| 145 | {
|
---|
[21592] | 146 | SystrayItem *ptrayItem = systray;
|
---|
[4121] | 147 |
|
---|
[21592] | 148 | while (ptrayItem)
|
---|
| 149 | {
|
---|
| 150 | if (SYSTRAY_ItemIsEqual(pnid, &ptrayItem->notifyIcon))
|
---|
| 151 | {
|
---|
| 152 | if (pnid->uFlags & NIF_ICON)
|
---|
| 153 | ptrayItem->notifyIcon.hIcon = GetOS2Icon(pnid->hIcon);
|
---|
| 154 | if (pnid->uFlags & NIF_MESSAGE)
|
---|
| 155 | ptrayItem->notifyIcon.uCallbackMessage = pnid->uCallbackMessage;
|
---|
| 156 | if (pnid->uFlags & NIF_TIP)
|
---|
| 157 | CharToOemA(pnid->szTip, ptrayItem->notifyIcon.szTip);
|
---|
[4121] | 158 |
|
---|
[21592] | 159 | SYSTRAY_ItemUpdate(ptrayItem, pnid->uFlags);
|
---|
| 160 |
|
---|
| 161 | TRACE("SYSTRAY_Modify %p: uIdx %u, hWnd 0x%08x, uID %d, hIcon 0x%08x, "
|
---|
| 162 | "uCallbackMessage 0x%08x, szTip [%s]\n",
|
---|
| 163 | ptrayItem, ptrayItem->uIdx, ptrayItem->hWnd, ptrayItem->notifyIcon.uID,
|
---|
| 164 | ptrayItem->notifyIcon.hIcon,ptrayItem->notifyIcon.uCallbackMessage,
|
---|
| 165 | ptrayItem->notifyIcon.szTip);
|
---|
| 166 |
|
---|
| 167 | return TRUE;
|
---|
| 168 | }
|
---|
| 169 | ptrayItem = ptrayItem->nextTrayItem;
|
---|
[4121] | 170 | }
|
---|
[21592] | 171 |
|
---|
| 172 | return FALSE; /* not found */
|
---|
[4121] | 173 | }
|
---|
| 174 |
|
---|
| 175 | static BOOL SYSTRAY_Delete(PNOTIFYICONDATAA pnid)
|
---|
| 176 | {
|
---|
[21592] | 177 | SystrayItem **ptrayItem = &systray;
|
---|
[4121] | 178 |
|
---|
[21592] | 179 | while (*ptrayItem)
|
---|
| 180 | {
|
---|
| 181 | if (SYSTRAY_ItemIsEqual(pnid, &(*ptrayItem)->notifyIcon))
|
---|
| 182 | {
|
---|
| 183 | SystrayItem *next = (*ptrayItem)->nextTrayItem;
|
---|
[4121] | 184 |
|
---|
[21592] | 185 | TRACE("SYSTRAY_Delete %p: uIdx %u, hWnd 0x%08x, uID %d\n",
|
---|
| 186 | *ptrayItem, (*ptrayItem)->uIdx, (*ptrayItem)->notifyIcon.hWnd,
|
---|
| 187 | (*ptrayItem)->notifyIcon.uID);
|
---|
[4121] | 188 |
|
---|
[21592] | 189 | SYSTRAY_ItemTerm(*ptrayItem);
|
---|
| 190 |
|
---|
| 191 | free(*ptrayItem);
|
---|
[21602] | 192 |
|
---|
[21592] | 193 | *ptrayItem = next;
|
---|
| 194 |
|
---|
| 195 | return TRUE;
|
---|
| 196 | }
|
---|
[21602] | 197 |
|
---|
[21592] | 198 | ptrayItem = &((*ptrayItem)->nextTrayItem);
|
---|
[4121] | 199 | }
|
---|
| 200 |
|
---|
[21592] | 201 | return FALSE; /* not found */
|
---|
[4121] | 202 | }
|
---|
| 203 |
|
---|
[21591] | 204 | #ifndef __WIN32OS2__
|
---|
[4121] | 205 | /*************************************************************************
|
---|
| 206 | *
|
---|
| 207 | */
|
---|
| 208 | BOOL SYSTRAY_Init(void)
|
---|
| 209 | {
|
---|
[21592] | 210 | return TRUE;
|
---|
[4121] | 211 | }
|
---|
[21591] | 212 | #endif
|
---|
[4121] | 213 |
|
---|
| 214 | /*************************************************************************
|
---|
[21592] | 215 | *
|
---|
| 216 | */
|
---|
| 217 | SystrayItem *SYSTRAY_FindItem(ULONG uIdx)
|
---|
| 218 | {
|
---|
| 219 | SystrayItem *ptrayItem = systray;
|
---|
| 220 |
|
---|
| 221 | while (ptrayItem)
|
---|
| 222 | {
|
---|
| 223 | if (ptrayItem->uIdx == uIdx)
|
---|
| 224 | return ptrayItem;
|
---|
| 225 |
|
---|
| 226 | ptrayItem = ptrayItem->nextTrayItem;
|
---|
| 227 | }
|
---|
| 228 |
|
---|
| 229 | return NULL; /* not found */
|
---|
| 230 | }
|
---|
| 231 |
|
---|
[21602] | 232 | /*************************************************************************
|
---|
| 233 | *
|
---|
| 234 | */
|
---|
| 235 | void SYSTRAY_PruneAllItems(void)
|
---|
| 236 | {
|
---|
| 237 | SystrayItem *ptrayItem = systray;
|
---|
| 238 |
|
---|
| 239 | while (ptrayItem)
|
---|
| 240 | {
|
---|
| 241 | SystrayItem *next = ptrayItem->nextTrayItem;
|
---|
| 242 |
|
---|
| 243 | TRACE("SYSTRAY_PruneAllItems %p: uIdx %u, hWnd 0x%08x, uID %d\n",
|
---|
| 244 | ptrayItem, ptrayItem->uIdx, ptrayItem->notifyIcon.hWnd,
|
---|
| 245 | ptrayItem->notifyIcon.uID);
|
---|
| 246 |
|
---|
| 247 | SYSTRAY_ItemTerm(ptrayItem);
|
---|
| 248 |
|
---|
| 249 | free(ptrayItem);
|
---|
| 250 |
|
---|
| 251 | ptrayItem = next;
|
---|
| 252 | }
|
---|
| 253 |
|
---|
| 254 | systray = NULL;
|
---|
| 255 | }
|
---|
| 256 |
|
---|
[21600] | 257 | static PVOID Do_Shell_NotifyIconA(PVOID arg1, PVOID arg2,
|
---|
| 258 | PVOID arg3, PVOID arg4)
|
---|
[4121] | 259 | {
|
---|
[21600] | 260 | DWORD dwMessage = (DWORD)arg1;
|
---|
| 261 | PNOTIFYICONDATAA pnid = (PNOTIFYICONDATAA)arg2;
|
---|
| 262 |
|
---|
[21592] | 263 | BOOL flag = FALSE;
|
---|
| 264 | switch(dwMessage)
|
---|
| 265 | {
|
---|
| 266 | case NIM_ADD:
|
---|
| 267 | flag = SYSTRAY_Add(pnid);
|
---|
| 268 | break;
|
---|
| 269 | case NIM_MODIFY:
|
---|
| 270 | flag = SYSTRAY_Modify(pnid);
|
---|
| 271 | break;
|
---|
| 272 | case NIM_DELETE:
|
---|
| 273 | flag = SYSTRAY_Delete(pnid);
|
---|
| 274 | break;
|
---|
| 275 | }
|
---|
[21600] | 276 | return (PVOID)flag;
|
---|
[4121] | 277 | }
|
---|
| 278 |
|
---|
| 279 | /*************************************************************************
|
---|
[21600] | 280 | * Shell_NotifyIconA [SHELL32.297][SHELL32.296]
|
---|
| 281 | */
|
---|
| 282 | BOOL WINAPI Shell_NotifyIconA(DWORD dwMessage, PNOTIFYICONDATAA pnid )
|
---|
| 283 | {
|
---|
| 284 | PVOID ret;
|
---|
| 285 | BOOL brc = RunOnAuxThreadAndWait(Do_Shell_NotifyIconA,
|
---|
| 286 | (PVOID)dwMessage, (PVOID)pnid,
|
---|
| 287 | NULL, NULL, &ret);
|
---|
| 288 | if (brc)
|
---|
| 289 | brc = (BOOL)ret;
|
---|
| 290 | return brc;
|
---|
| 291 | }
|
---|
| 292 |
|
---|
| 293 | /*************************************************************************
|
---|
[10604] | 294 | * Shell_NotifyIconW [SHELL32.298]
|
---|
[4121] | 295 | */
|
---|
| 296 | BOOL WINAPI Shell_NotifyIconW (DWORD dwMessage, PNOTIFYICONDATAW pnid )
|
---|
| 297 | {
|
---|
[10604] | 298 | BOOL ret;
|
---|
[4121] | 299 |
|
---|
[10604] | 300 | PNOTIFYICONDATAA p = HeapAlloc(GetProcessHeap(),0,sizeof(NOTIFYICONDATAA));
|
---|
| 301 | memcpy(p, pnid, sizeof(NOTIFYICONDATAA));
|
---|
[5618] | 302 | WideCharToMultiByte( CP_ACP, 0, pnid->szTip, -1, p->szTip, sizeof(p->szTip), NULL, NULL );
|
---|
| 303 | p->szTip[sizeof(p->szTip)-1] = 0;
|
---|
[4121] | 304 |
|
---|
[10604] | 305 | ret = Shell_NotifyIconA(dwMessage, p );
|
---|
[4121] | 306 |
|
---|
[10604] | 307 | HeapFree(GetProcessHeap(),0,p);
|
---|
| 308 | return ret;
|
---|
[4121] | 309 | }
|
---|
[10604] | 310 |
|
---|
| 311 |
|
---|
| 312 | #ifdef __WIN32OS2__
|
---|
| 313 |
|
---|
| 314 | BOOL DoWin32PostMessage(HWND w, ULONG m, WPARAM mp1, LPARAM mp2 )
|
---|
| 315 | {
|
---|
| 316 | TRACE("DoWin32WinPostMsg: HWND 0x%08x MSG 0x%08x ID 0x%08x MMSG 0x%08x\n", w, m, mp1, mp2);
|
---|
| 317 | if ( w && m )
|
---|
| 318 | {
|
---|
| 319 | return PostMessageA( w, m, mp1, mp2 );
|
---|
| 320 | }
|
---|
| 321 | return FALSE;
|
---|
| 322 | }
|
---|
| 323 |
|
---|
| 324 | #endif
|
---|