Как отобразить текст в значке в системном трее с Win32 API?

Попытка создать небольшое приложение для мониторинга, которое отображает текущее использование Интернета в процентах в системном трее в C, используя Win32 API.

Также желая использовать цветной фон или цветной текст, основываясь на том, сколько используется по отношению к дням, оставшимся в месяце.

РЕДАКТИРОВАТЬ: Чтобы уточнить, я хочу, чтобы значок в системном трее был динамическим. По мере изменения процента я обновляю значок в системном трее. Ищите решение, которое использует просто старый win32 (т.е. без MFC или WTL).

4 ответа

Решение

Хорошо, вот мое решение для Win32:

HICON CreateSmallIcon( HWND hWnd )
{
    static TCHAR *szText = TEXT ( "100" );
    HDC hdc, hdcMem;
    HBITMAP hBitmap = NULL;
    HBITMAP hOldBitMap = NULL;
    HBITMAP hBitmapMask = NULL;
    ICONINFO iconInfo;
    HFONT hFont;
    HICON hIcon;

    hdc = GetDC ( hWnd );
    hdcMem = CreateCompatibleDC ( hdc );
    hBitmap = CreateCompatibleBitmap ( hdc, 16, 16 );
    hBitmapMask = CreateCompatibleBitmap ( hdc, 16, 16 );
    ReleaseDC ( hWnd, hdc );
    hOldBitMap = (HBITMAP) SelectObject ( hdcMem, hBitmap );
    PatBlt ( hdcMem, 0, 0, 16, 16, WHITENESS );

    // Draw percentage
    hFont = CreateFont (12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
                    TEXT ("Arial"));
    hFont = (HFONT) SelectObject ( hdcMem, hFont );
    TextOut ( hdcMem, 0, 0, szText, lstrlen (szText) );

    SelectObject ( hdc, hOldBitMap );
    hOldBitMap = NULL;

    iconInfo.fIcon = TRUE;
    iconInfo.xHotspot = 0;
    iconInfo.yHotspot = 0;
    iconInfo.hbmMask = hBitmapMask;
    iconInfo.hbmColor = hBitmap;

    hIcon = CreateIconIndirect ( &iconInfo );

    DeleteObject ( SelectObject ( hdcMem, hFont ) );
    DeleteDC ( hdcMem );
    DeleteDC ( hdc );
    DeleteObject ( hBitmap );
    DeleteObject ( hBitmapMask );

    return hIcon;
}

Под текстом вы подразумеваете "Советы"?

Предполагая, что у вас есть значок на панели задач

NOTIFYICONDATA _stNotifyIconData;

// For a simple Tip
_stNotifyIconData.uFlags = NIF_TIP;
strcpy_s(_stNotifyIconData.szTip, "Little Tip"); // Copy Tip    
Shell_NotifyIcon(NIM_MODIFY, &_stNotifyIconData);

// For a Ballon Tip
_stNotifyIconData.uFlags = NIF_INFO;
strcpy_s(_stNotifyIconData.szInfoTitle, "Title of the Ballon"); // Title
strcpy_s(_stNotifyIconData.szInfo, "Text..." ); // Copy Tip
_stNotifyIconData.uTimeout = 3000;  // 3 Seconds
_stNotifyIconData.dwInfoFlags = NIIF_INFO;

Shell_NotifyIcon(NIM_MODIFY, &_stNotifyIconData);

На панели задач отображаются только значки, а не текст.

Чтобы получить текст, показанный там, вы должны сначала создать растровое изображение памяти, нарисовать на нем свой текст, затем преобразовать это растровое изображение памяти в значок памяти и заставить системный трей показать этот значок.

Пример кода ниже:

CDC dcMem;
dcMem.CreateCompatibleDC(NULL);

CBitmap* pOld = dcMem.SelectObject( &m_bmpIcon );

CBrush back( RGB(0,0,0) );
dcMem.FillRect( CRect(0,0,16,16), &back );

CBrush brush( COLORDOWN );
dcMem.FillRect( rcRecv, &brush );

dcMem.SelectObject( pOld );

HICON hIcon = CreateIconIndirect( &m_TaskBarIconInfo );  

Для тех, кто ищет решение Python, используяpywin32, вот что я в итоге сделал:

      this_files_dir = os.path.abspath(os.path.dirname(__file__))
icon_path = os.path.join(this_files_dir, 'custom.ico')

def txt_to_bmp(txt):
    img = Image.new('L', (256,256), color=255)
    img_w, img_h = img.size
    font = ImageFont.truetype('arial.ttf', 180) # font & font-size
    mask = font.getmask(txt, mode='L')
    mask_w, mask_h = mask.size
    #print(mask_w,mask_h)
    d = Image.core.draw(img.im, 0)
    d.draw_bitmap(((img_w - mask_w)/2, (img_h - mask_h)/2), mask, 0)
    img.save(icon_path)

# pywin32 code to display a SysTray icon, etc
# forked from https://github.com/jfoote/watchme/blob/master/systrayicon.py
class SysTrayIcon:
    def refresh_icon(self):
        # Try and find a custom icon
        hinst = win32gui.GetModuleHandle(None)
        if os.path.isfile(self.icon):
            icon_flags = win32con.LR_LOADFROMFILE | win32con.LR_DEFAULTSIZE
            hicon = win32gui.LoadImage(hinst,
                                       icon_path,
                                       win32con.IMAGE_ICON,
                                       0,
                                       0,
                                       icon_flags)
        else:
            print("Can't find icon file - using default.")
            hicon = win32gui.LoadIcon(0, win32con.IDI_APPLICATION)

        if self.notify_id: message = win32gui.NIM_MODIFY
        else: message = win32gui.NIM_ADD

        self.notify_id = (self.hwnd,
                          0,
                          win32gui.NIF_ICON | win32gui.NIF_MESSAGE | win32gui.NIF_TIP,
                          win32con.WM_USER+20,
                          hicon,
                          'Hovertext', 'msg',200, 'title', 4) #NIIF_USER==4, keeps Icon displayed through something about balloontips
        win32gui.Shell_NotifyIcon(message, self.notify_id)
Другие вопросы по тегам