Все знают, как удобно, когда приложение размещает свою иконку в трее (полностью - TrayBar или, как теперь называет это Microsoft, Область Уведомлений) и поэтому не мешается на экране или в панели задач. Но работа с треем осуществляется совсем не так, как с панелью задач. Самое интересное, что, чтобы найти информацию по этому вопросу в MSDN, мне потребовалось довольно много времени. Да и удобство изложения материала оставляло желать лучшего.
Итак, начнём по порядку. Все действия с областью уведомлений производятся с помощью одной функции - Shell_NotifyIcon. Так что по имени этой функции вы сможете найти в MSDN всё, что связано с треем.
Функция объявлена так:
BOOL Shell_NotifyIcon(
DWORD dwMessage,
PNOTIFYICONDATA lpdata
);
Первый параметр определяет выполняемое действие (NIM_ADD, NIM_DELETE и т. д.), а второй - это указатель на структуру типа NOTIFYICONDATA, содержащую дополнительную информацию.
Приступим к помещению иконки в область уведомлений. Нужно заметить, что нам необходимо будет обрабатывать сообщения, приходящие от этой иконки. Но эти сообщения мы должны определить сами:
#define WM_TRAYNOTIFY WM_USER + 2
Почему WM_USER + 2? Потому что WM_USER и WM_USER + 1, вероятно, уже используется в вашей программе (это сообщения DM_GETDEFID и DM_SETDEFID соответственно). Так что на всякий случай надо принять меры предосторожности.
Идём дальше. Объявляем (лучше глобально) структуру типа NOTIFYICONDATA:
NOTIFYICONDATA tr_icon_data;
Теперь где-нибудь после создания окна и до начала цикла обработки сообщений помещаем следующий код:
//создаём значок в TrayBar
tr_icon_data.cbSize = sizeof(tr_icon_data);
tr_icon_data.hWnd = hWndMain; //HWND главного окна
tr_icon_data.uID = 0x10; //идентификатор может быть любой
//следующая строка означает, что поля hIcon, szTip и uCallbackMessage
//заполнены правильно. Остальные игнорируются.
tr_icon_data.uFlags = NIF_ICON | NIF_TIP | NIF_MESSAGE;
tr_icon_data.hIcon = LoadIcon(hInst, MAKEINTRESOURCE(IDI_ICON1));
strcpy(tr_icon_data.szTip, "Всплывающая подсказка к нашей программе");
tr_icon_data.uCallbackMessage = WM_TRAYNOTIFY;
//теперь создаём иконку
Shell_NotifyIcon(NIM_ADD, &tr_icon_data);
Вот и всё. Если бы не одно но... Иногда это сильно подтормаживает программу. Я не знаю, что там так медленно работает, но я нашёл способ, как этого избежать. Весь последний фрагмент кода я поместил в отдельную функцию, которую вызываю в отдельном потоке. Результат мгновенный.
Итак, объявляем функцию
void WINAPI CreateNotifyIcon();
Затем помещаем в её тело вышеупомянутый фрагмент кода (т. е. заполнение структуры и вызов функции Shell_NotifyIcon). И запускаем нашу функцию как отдельный поток в том месте программы, где раньше находился всё тот же вышеупомянутый фрагмент кода:
HANDLE hThread = CreateThread(NULL, 0,
(LPTHREAD_START_ROUTINE)CreateNotifyIcon, NULL, 0, NULL);
CloseHandle(hThread);
Теперь всё. Спросите, как теперь удалить из трея эту прекрасную иконку? Всё просто - после завершения цикла обработки сообщений пишем
Shell_NotifyIcon(NIM_DELETE, &tr_icon_data);
Ну, теперь уже совсем всё. Удачи вам.
P. S. Чуть не забыл... Не пробуйте создавать элементы управления (вообще любые окна) в отдельном потоке - при завершении потока они удаляются!!!