2025-05-13 19:45:22 +03:00

1342 lines
35 KiB
C

/*
* Process Hacker -
* notification icon manager
*
* Copyright (C) 2011-2016 wj32
*
* This file is part of Process Hacker.
*
* Process Hacker is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Process Hacker 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Process Hacker. If not, see <http://www.gnu.org/licenses/>.
*/
#include <phapp.h>
#include <procprv.h>
#include <settings.h>
#include <extmgri.h>
#include <miniinfo.h>
#include <phplug.h>
#include <notifico.h>
#include <notificop.h>
#include <mainwndp.h>
#include <shellapi.h>
#include <windowsx.h>
BOOLEAN PhNfTerminating = FALSE;
ULONG PhNfIconMask;
ULONG PhNfIconNotifyMask;
ULONG PhNfMaximumIconId = PH_ICON_DEFAULT_MAXIMUM;
PPH_NF_ICON PhNfRegisteredIcons[32] = { 0 };
PPH_STRING PhNfIconTextCache[32] = { 0 };
BOOLEAN PhNfMiniInfoEnabled;
BOOLEAN PhNfMiniInfoPinned;
PH_NF_POINTERS PhNfpPointers;
PH_CALLBACK_REGISTRATION PhNfpProcessesUpdatedRegistration;
PH_NF_BITMAP PhNfpDefaultBitmapContext = { 0 };
PH_NF_BITMAP PhNfpBlackBitmapContext = { 0 };
HBITMAP PhNfpBlackBitmap = NULL;
HICON PhNfpBlackIcon = NULL;
static POINT IconClickLocation;
static PH_NF_MSG_SHOWMINIINFOSECTION_DATA IconClickShowMiniInfoSectionData;
static BOOLEAN IconClickUpDueToDown;
static BOOLEAN IconDisableHover;
VOID PhNfLoadStage1(
VOID
)
{
PPH_STRING iconList;
PH_STRINGREF part;
PH_STRINGREF remainingPart;
PhNfpPointers.UpdateRegisteredIcon = PhNfpUpdateRegisteredIcon;
PhNfpPointers.BeginBitmap = PhNfpBeginBitmap;
// Load settings for default icons.
PhNfIconMask = PhGetIntegerSetting(L"IconMask");
// Load settings for registered icons.
iconList = PhGetStringSetting(L"IconMaskList");
remainingPart = iconList->sr;
while (remainingPart.Length != 0)
{
PhSplitStringRefAtChar(&remainingPart, '|', &part, &remainingPart);
if (part.Length != 0)
{
PH_STRINGREF pluginName;
ULONG subId;
PPH_NF_ICON icon;
if (PhEmParseCompoundId(&part, &pluginName, &subId) &&
(icon = PhNfFindIcon(&pluginName, subId)))
{
PhNfIconMask |= icon->IconId;
}
}
}
PhDereferenceObject(iconList);
}
VOID PhNfLoadStage2(
VOID
)
{
ULONG i;
PhNfMiniInfoEnabled = WindowsVersion >= WINDOWS_VISTA && !!PhGetIntegerSetting(L"MiniInfoWindowEnabled");
for (i = PH_ICON_MINIMUM; i != PhNfMaximumIconId; i <<= 1)
{
if (PhNfIconMask & i)
PhNfpAddNotifyIcon(i);
}
PhRegisterCallback(
&PhProcessesUpdatedEvent,
PhNfpProcessesUpdatedHandler,
NULL,
&PhNfpProcessesUpdatedRegistration
);
}
VOID PhNfSaveSettings(
VOID
)
{
ULONG registeredIconMask;
PhSetIntegerSetting(L"IconMask", PhNfIconMask & PH_ICON_DEFAULT_ALL);
registeredIconMask = PhNfIconMask & ~PH_ICON_DEFAULT_ALL;
if (registeredIconMask != 0)
{
PH_STRING_BUILDER iconListBuilder;
ULONG i;
PhInitializeStringBuilder(&iconListBuilder, 60);
for (i = 0; i < sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON); i++)
{
if (PhNfRegisteredIcons[i])
{
if (registeredIconMask & PhNfRegisteredIcons[i]->IconId)
{
PhAppendFormatStringBuilder(
&iconListBuilder,
L"+%s+%u|",
PhNfRegisteredIcons[i]->Plugin->Name.Buffer,
PhNfRegisteredIcons[i]->SubId
);
}
}
}
if (iconListBuilder.String->Length != 0)
PhRemoveEndStringBuilder(&iconListBuilder, 1);
PhSetStringSetting2(L"IconMaskList", &iconListBuilder.String->sr);
PhDeleteStringBuilder(&iconListBuilder);
}
else
{
PhSetStringSetting(L"IconMaskList", L"");
}
}
VOID PhNfUninitialization(
VOID
)
{
ULONG i;
// Remove all icons to prevent them hanging around after we exit.
PhNfTerminating = TRUE; // prevent further icon updating
for (i = PH_ICON_MINIMUM; i != PhNfMaximumIconId; i <<= 1)
{
if (PhNfIconMask & i)
PhNfpRemoveNotifyIcon(i);
}
}
VOID PhNfForwardMessage(
_In_ ULONG_PTR WParam,
_In_ ULONG_PTR LParam
)
{
ULONG iconIndex = HIWORD(LParam);
PPH_NF_ICON registeredIcon = NULL;
if (iconIndex < sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON) && PhNfRegisteredIcons[iconIndex])
{
registeredIcon = PhNfRegisteredIcons[iconIndex];
if (registeredIcon->MessageCallback)
{
if (registeredIcon->MessageCallback(
registeredIcon,
WParam,
LParam,
registeredIcon->Context
))
{
return;
}
}
}
switch (LOWORD(LParam))
{
case WM_LBUTTONDOWN:
{
if (PhGetIntegerSetting(L"IconSingleClick"))
{
ProcessHacker_IconClick(PhMainWndHandle);
PhNfpDisableHover();
}
else
{
IconClickUpDueToDown = TRUE;
}
}
break;
case WM_LBUTTONUP:
{
if (!PhGetIntegerSetting(L"IconSingleClick") && PhNfMiniInfoEnabled && IconClickUpDueToDown)
{
PH_NF_MSG_SHOWMINIINFOSECTION_DATA showMiniInfoSectionData;
if (PhNfpGetShowMiniInfoSectionData(iconIndex, registeredIcon, &showMiniInfoSectionData))
{
GetCursorPos(&IconClickLocation);
if (IconClickShowMiniInfoSectionData.SectionName)
{
PhFree(IconClickShowMiniInfoSectionData.SectionName);
IconClickShowMiniInfoSectionData.SectionName = NULL;
}
if (showMiniInfoSectionData.SectionName)
{
IconClickShowMiniInfoSectionData.SectionName = PhDuplicateStringZ(showMiniInfoSectionData.SectionName);
}
SetTimer(PhMainWndHandle, TIMER_ICON_CLICK_ACTIVATE, GetDoubleClickTime() + NFP_ICON_CLICK_ACTIVATE_DELAY, PhNfpIconClickActivateTimerProc);
}
else
{
KillTimer(PhMainWndHandle, TIMER_ICON_CLICK_ACTIVATE);
}
}
}
break;
case WM_LBUTTONDBLCLK:
{
if (!PhGetIntegerSetting(L"IconSingleClick"))
{
if (PhNfMiniInfoEnabled)
{
// We will get another WM_LBUTTONUP message corresponding to the double-click,
// and we need to make sure that it doesn't start the activation timer again.
KillTimer(PhMainWndHandle, TIMER_ICON_CLICK_ACTIVATE);
IconClickUpDueToDown = FALSE;
PhNfpDisableHover();
}
ProcessHacker_IconClick(PhMainWndHandle);
}
}
break;
case WM_RBUTTONUP:
case WM_CONTEXTMENU:
{
POINT location;
if (!PhGetIntegerSetting(L"IconSingleClick") && PhNfMiniInfoEnabled)
KillTimer(PhMainWndHandle, TIMER_ICON_CLICK_ACTIVATE);
PhPinMiniInformation(MiniInfoIconPinType, -1, 0, 0, NULL, NULL);
GetCursorPos(&location);
PhShowIconContextMenu(location);
}
break;
case NIN_KEYSELECT:
// HACK: explorer seems to send two NIN_KEYSELECT messages when the user selects the icon and presses ENTER.
if (GetForegroundWindow() != PhMainWndHandle)
ProcessHacker_IconClick(PhMainWndHandle);
break;
case NIN_BALLOONUSERCLICK:
PhShowDetailsForIconNotification();
break;
case NIN_POPUPOPEN:
{
PH_NF_MSG_SHOWMINIINFOSECTION_DATA showMiniInfoSectionData;
POINT location;
if (PhNfMiniInfoEnabled && !IconDisableHover && PhNfpGetShowMiniInfoSectionData(iconIndex, registeredIcon, &showMiniInfoSectionData))
{
GetCursorPos(&location);
PhPinMiniInformation(MiniInfoIconPinType, 1, 0, PH_MINIINFO_DONT_CHANGE_SECTION_IF_PINNED,
showMiniInfoSectionData.SectionName, &location);
}
}
break;
case NIN_POPUPCLOSE:
PhPinMiniInformation(MiniInfoIconPinType, -1, 350, 0, NULL, NULL);
break;
}
}
ULONG PhNfGetMaximumIconId(
VOID
)
{
return PhNfMaximumIconId;
}
ULONG PhNfTestIconMask(
_In_ ULONG Id
)
{
return PhNfIconMask & Id;
}
VOID PhNfSetVisibleIcon(
_In_ ULONG Id,
_In_ BOOLEAN Visible
)
{
if (Visible)
{
PhNfIconMask |= Id;
PhNfpAddNotifyIcon(Id);
}
else
{
PhNfIconMask &= ~Id;
PhNfpRemoveNotifyIcon(Id);
}
}
BOOLEAN PhNfShowBalloonTip(
_In_opt_ ULONG Id,
_In_ PWSTR Title,
_In_ PWSTR Text,
_In_ ULONG Timeout,
_In_ ULONG Flags
)
{
NOTIFYICONDATA notifyIcon = { NOTIFYICONDATA_V3_SIZE };
if (Id == 0)
{
// Choose the first visible icon.
Id = PhNfIconMask;
}
if (!_BitScanForward(&Id, Id))
return FALSE;
notifyIcon.hWnd = PhMainWndHandle;
notifyIcon.uID = Id;
notifyIcon.uFlags = NIF_INFO;
wcsncpy_s(notifyIcon.szInfoTitle, sizeof(notifyIcon.szInfoTitle) / sizeof(WCHAR), Title, _TRUNCATE);
wcsncpy_s(notifyIcon.szInfo, sizeof(notifyIcon.szInfo) / sizeof(WCHAR), Text, _TRUNCATE);
notifyIcon.uTimeout = Timeout;
notifyIcon.dwInfoFlags = Flags;
Shell_NotifyIcon(NIM_MODIFY, &notifyIcon);
return TRUE;
}
HICON PhNfBitmapToIcon(
_In_ HBITMAP Bitmap
)
{
ICONINFO iconInfo;
PhNfpGetBlackIcon();
iconInfo.fIcon = TRUE;
iconInfo.xHotspot = 0;
iconInfo.yHotspot = 0;
iconInfo.hbmMask = PhNfpBlackBitmap;
iconInfo.hbmColor = Bitmap;
return CreateIconIndirect(&iconInfo);
}
PPH_NF_ICON PhNfRegisterIcon(
_In_ struct _PH_PLUGIN *Plugin,
_In_ ULONG SubId,
_In_opt_ PVOID Context,
_In_ PWSTR Text,
_In_ ULONG Flags,
_In_opt_ PPH_NF_ICON_UPDATE_CALLBACK UpdateCallback,
_In_opt_ PPH_NF_ICON_MESSAGE_CALLBACK MessageCallback
)
{
PPH_NF_ICON icon;
ULONG iconId;
ULONG iconIndex;
if (PhNfMaximumIconId == PH_ICON_LIMIT)
{
// No room for any more icons.
return NULL;
}
iconId = PhNfMaximumIconId;
if (!_BitScanReverse(&iconIndex, iconId) ||
iconIndex >= sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON))
{
// Should never happen.
return NULL;
}
PhNfMaximumIconId <<= 1;
icon = PhAllocate(sizeof(PH_NF_ICON));
icon->Plugin = Plugin;
icon->SubId = SubId;
icon->Context = Context;
icon->Pointers = &PhNfpPointers;
icon->Text = Text;
icon->Flags = Flags;
icon->IconId = iconId;
icon->UpdateCallback = UpdateCallback;
icon->MessageCallback = MessageCallback;
PhNfRegisteredIcons[iconIndex] = icon;
return icon;
}
PPH_NF_ICON PhNfGetIconById(
_In_ ULONG Id
)
{
ULONG iconIndex;
if (!_BitScanReverse(&iconIndex, Id))
return NULL;
return PhNfRegisteredIcons[iconIndex];
}
PPH_NF_ICON PhNfFindIcon(
_In_ PPH_STRINGREF PluginName,
_In_ ULONG SubId
)
{
ULONG i;
for (i = 0; i < sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON); i++)
{
if (PhNfRegisteredIcons[i])
{
if (PhNfRegisteredIcons[i]->SubId == SubId &&
PhEqualStringRef(PluginName, &PhNfRegisteredIcons[i]->Plugin->AppContext.AppName, FALSE))
{
return PhNfRegisteredIcons[i];
}
}
}
return NULL;
}
VOID PhNfNotifyMiniInfoPinned(
_In_ BOOLEAN Pinned
)
{
ULONG i;
ULONG id;
if (PhNfMiniInfoPinned != Pinned)
{
PhNfMiniInfoPinned = Pinned;
// Go through every icon and set/clear the NIF_SHOWTIP flag depending on whether the mini info window is
// pinned. If it's pinned then we want to show normal tooltips, because the section doesn't change
// automatically when the cursor hovers over an icon.
for (i = 0; i < sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON); i++)
{
id = 1 << i;
if (PhNfIconMask & id)
{
PhNfpModifyNotifyIcon(id, NIF_TIP, PhNfIconTextCache[i], NULL);
}
}
}
}
HICON PhNfpGetBlackIcon(
VOID
)
{
if (!PhNfpBlackIcon)
{
ULONG width;
ULONG height;
PVOID bits;
HDC hdc;
HBITMAP oldBitmap;
ICONINFO iconInfo;
PhNfpBeginBitmap2(&PhNfpBlackBitmapContext, &width, &height, &PhNfpBlackBitmap, &bits, &hdc, &oldBitmap);
memset(bits, 0, width * height * sizeof(ULONG));
iconInfo.fIcon = TRUE;
iconInfo.xHotspot = 0;
iconInfo.yHotspot = 0;
iconInfo.hbmMask = PhNfpBlackBitmap;
iconInfo.hbmColor = PhNfpBlackBitmap;
PhNfpBlackIcon = CreateIconIndirect(&iconInfo);
SelectObject(hdc, oldBitmap);
}
return PhNfpBlackIcon;
}
BOOLEAN PhNfpAddNotifyIcon(
_In_ ULONG Id
)
{
NOTIFYICONDATA notifyIcon = { sizeof(NOTIFYICONDATA) };
PPH_NF_ICON icon;
if (PhNfTerminating)
return FALSE;
if ((icon = PhNfGetIconById(Id)) && (icon->Flags & PH_NF_ICON_UNAVAILABLE))
return FALSE;
// The IDs we pass to explorer are bit indicies, not the normal mask values.
if (!_BitScanForward(&Id, Id))
return FALSE;
notifyIcon.hWnd = PhMainWndHandle;
notifyIcon.uID = Id;
notifyIcon.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP;
notifyIcon.uCallbackMessage = WM_PH_NOTIFY_ICON_MESSAGE;
wcsncpy_s(
notifyIcon.szTip, sizeof(notifyIcon.szTip) / sizeof(WCHAR),
PhGetStringOrDefault(PhNfIconTextCache[Id], PhApplicationName),
_TRUNCATE
);
notifyIcon.hIcon = PhNfpGetBlackIcon();
if (!PhNfMiniInfoEnabled || PhNfMiniInfoPinned || (icon && !(icon->Flags & PH_NF_ICON_SHOW_MINIINFO)))
notifyIcon.uFlags |= NIF_SHOWTIP;
Shell_NotifyIcon(NIM_ADD, &notifyIcon);
if (WindowsVersion >= WINDOWS_VISTA)
{
notifyIcon.uVersion = NOTIFYICON_VERSION_4;
Shell_NotifyIcon(NIM_SETVERSION, &notifyIcon);
}
return TRUE;
}
BOOLEAN PhNfpRemoveNotifyIcon(
_In_ ULONG Id
)
{
NOTIFYICONDATA notifyIcon = { sizeof(NOTIFYICONDATA) };
PPH_NF_ICON icon;
if ((icon = PhNfGetIconById(Id)) && (icon->Flags & PH_NF_ICON_UNAVAILABLE))
return FALSE;
if (!_BitScanForward(&Id, Id))
return FALSE;
notifyIcon.hWnd = PhMainWndHandle;
notifyIcon.uID = Id;
Shell_NotifyIcon(NIM_DELETE, &notifyIcon);
return TRUE;
}
BOOLEAN PhNfpModifyNotifyIcon(
_In_ ULONG Id,
_In_ ULONG Flags,
_In_opt_ PPH_STRING Text,
_In_opt_ HICON Icon
)
{
NOTIFYICONDATA notifyIcon = { sizeof(NOTIFYICONDATA) };
PPH_NF_ICON icon;
ULONG notifyId;
if (PhNfTerminating)
return FALSE;
if ((icon = PhNfGetIconById(Id)) && (icon->Flags & PH_NF_ICON_UNAVAILABLE))
return FALSE;
if (!_BitScanForward(&notifyId, Id))
return FALSE;
notifyIcon.hWnd = PhMainWndHandle;
notifyIcon.uID = notifyId;
notifyIcon.uFlags = Flags;
if (Flags & NIF_TIP)
{
PhSwapReference(&PhNfIconTextCache[notifyId], Text);
wcsncpy_s(
notifyIcon.szTip,
sizeof(notifyIcon.szTip) / sizeof(WCHAR),
PhGetStringOrDefault(Text, PhApplicationName),
_TRUNCATE
);
}
notifyIcon.hIcon = Icon;
if (!PhNfMiniInfoEnabled || PhNfMiniInfoPinned || (icon && !(icon->Flags & PH_NF_ICON_SHOW_MINIINFO)))
notifyIcon.uFlags |= NIF_SHOWTIP;
if (!Shell_NotifyIcon(NIM_MODIFY, &notifyIcon))
{
// Explorer probably died and we lost our icon. Try to add the icon, and try again.
PhNfpAddNotifyIcon(Id);
Shell_NotifyIcon(NIM_MODIFY, &notifyIcon);
}
return TRUE;
}
VOID PhNfpProcessesUpdatedHandler(
_In_opt_ PVOID Parameter,
_In_opt_ PVOID Context
)
{
ULONG registeredIconMask;
// We do icon updating on the provider thread so we don't block the main GUI when
// explorer is not responding.
if (PhNfIconMask & PH_ICON_CPU_HISTORY)
PhNfpUpdateIconCpuHistory();
if (PhNfIconMask & PH_ICON_IO_HISTORY)
PhNfpUpdateIconIoHistory();
if (PhNfIconMask & PH_ICON_COMMIT_HISTORY)
PhNfpUpdateIconCommitHistory();
if (PhNfIconMask & PH_ICON_PHYSICAL_HISTORY)
PhNfpUpdateIconPhysicalHistory();
if (PhNfIconMask & PH_ICON_CPU_USAGE)
PhNfpUpdateIconCpuUsage();
registeredIconMask = PhNfIconMask & ~PH_ICON_DEFAULT_ALL;
if (registeredIconMask != 0)
{
ULONG i;
for (i = 0; i < sizeof(PhNfRegisteredIcons) / sizeof(PPH_NF_ICON); i++)
{
if (PhNfRegisteredIcons[i] && (registeredIconMask & PhNfRegisteredIcons[i]->IconId))
{
PhNfpUpdateRegisteredIcon(PhNfRegisteredIcons[i]);
}
}
}
}
VOID PhNfpUpdateRegisteredIcon(
_In_ PPH_NF_ICON Icon
)
{
PVOID newIconOrBitmap;
ULONG updateFlags;
PPH_STRING newText;
HICON newIcon;
ULONG flags;
if (!Icon->UpdateCallback)
return;
newIconOrBitmap = NULL;
newText = NULL;
newIcon = NULL;
flags = 0;
Icon->UpdateCallback(
Icon,
&newIconOrBitmap,
&updateFlags,
&newText,
Icon->Context
);
if (newIconOrBitmap)
{
if (updateFlags & PH_NF_UPDATE_IS_BITMAP)
newIcon = PhNfBitmapToIcon(newIconOrBitmap);
else
newIcon = newIconOrBitmap;
flags |= NIF_ICON;
}
if (newText)
flags |= NIF_TIP;
if (flags != 0)
PhNfpModifyNotifyIcon(Icon->IconId, flags, newText, newIcon);
if (newIcon && (updateFlags & PH_NF_UPDATE_IS_BITMAP))
DestroyIcon(newIcon);
if (newIconOrBitmap && (updateFlags & PH_NF_UPDATE_DESTROY_RESOURCE))
{
if (updateFlags & PH_NF_UPDATE_IS_BITMAP)
DeleteObject(newIconOrBitmap);
else
DestroyIcon(newIconOrBitmap);
}
if (newText)
PhDereferenceObject(newText);
}
VOID PhNfpBeginBitmap(
_Out_ PULONG Width,
_Out_ PULONG Height,
_Out_ HBITMAP *Bitmap,
_Out_opt_ PVOID *Bits,
_Out_ HDC *Hdc,
_Out_ HBITMAP *OldBitmap
)
{
PhNfpBeginBitmap2(&PhNfpDefaultBitmapContext, Width, Height, Bitmap, Bits, Hdc, OldBitmap);
}
VOID PhNfpBeginBitmap2(
_Inout_ PPH_NF_BITMAP Context,
_Out_ PULONG Width,
_Out_ PULONG Height,
_Out_ HBITMAP *Bitmap,
_Out_opt_ PVOID *Bits,
_Out_ HDC *Hdc,
_Out_ HBITMAP *OldBitmap
)
{
if (!Context->Initialized)
{
HDC screenHdc;
screenHdc = GetDC(NULL);
Context->Hdc = CreateCompatibleDC(screenHdc);
memset(&Context->Header, 0, sizeof(BITMAPINFOHEADER));
Context->Header.biSize = sizeof(BITMAPINFOHEADER);
Context->Header.biWidth = PhSmallIconSize.X;
Context->Header.biHeight = PhSmallIconSize.Y;
Context->Header.biPlanes = 1;
Context->Header.biBitCount = 32;
Context->Bitmap = CreateDIBSection(screenHdc, (BITMAPINFO *)&Context->Header, DIB_RGB_COLORS, &Context->Bits, NULL, 0);
ReleaseDC(NULL, screenHdc);
Context->Initialized = TRUE;
}
*Width = PhSmallIconSize.X;
*Height = PhSmallIconSize.Y;
*Bitmap = Context->Bitmap;
if (Bits) *Bits = Context->Bits;
*Hdc = Context->Hdc;
*OldBitmap = SelectObject(Context->Hdc, Context->Bitmap);
}
VOID PhNfpUpdateIconCpuHistory(
VOID
)
{
static PH_GRAPH_DRAW_INFO drawInfo =
{
16,
16,
PH_GRAPH_USE_LINE_2,
2,
RGB(0x00, 0x00, 0x00),
16,
NULL,
NULL,
0,
0,
0,
0
};
ULONG maxDataCount;
ULONG lineDataCount;
PFLOAT lineData1;
PFLOAT lineData2;
HBITMAP bitmap;
PVOID bits;
HDC hdc;
HBITMAP oldBitmap;
HICON icon;
HANDLE maxCpuProcessId;
PPH_PROCESS_ITEM maxCpuProcessItem;
PH_FORMAT format[8];
PPH_STRING text;
// Icon
PhNfpBeginBitmap(&drawInfo.Width, &drawInfo.Height, &bitmap, &bits, &hdc, &oldBitmap);
maxDataCount = drawInfo.Width / 2 + 1;
lineData1 = _alloca(maxDataCount * sizeof(FLOAT));
lineData2 = _alloca(maxDataCount * sizeof(FLOAT));
lineDataCount = min(maxDataCount, PhCpuKernelHistory.Count);
PhCopyCircularBuffer_FLOAT(&PhCpuKernelHistory, lineData1, lineDataCount);
PhCopyCircularBuffer_FLOAT(&PhCpuUserHistory, lineData2, lineDataCount);
drawInfo.LineDataCount = lineDataCount;
drawInfo.LineData1 = lineData1;
drawInfo.LineData2 = lineData2;
drawInfo.LineColor1 = PhCsColorCpuKernel;
drawInfo.LineColor2 = PhCsColorCpuUser;
drawInfo.LineBackColor1 = PhHalveColorBrightness(PhCsColorCpuKernel);
drawInfo.LineBackColor2 = PhHalveColorBrightness(PhCsColorCpuUser);
if (bits)
PhDrawGraphDirect(hdc, bits, &drawInfo);
SelectObject(hdc, oldBitmap);
icon = PhNfBitmapToIcon(bitmap);
// Text
if (PhMaxCpuHistory.Count != 0)
maxCpuProcessId = UlongToHandle(PhGetItemCircularBuffer_ULONG(&PhMaxCpuHistory, 0));
else
maxCpuProcessId = NULL;
if (maxCpuProcessId)
maxCpuProcessItem = PhReferenceProcessItem(maxCpuProcessId);
else
maxCpuProcessItem = NULL;
PhInitFormatS(&format[0], L"CPU Usage: ");
PhInitFormatF(&format[1], (PhCpuKernelUsage + PhCpuUserUsage) * 100, 2);
PhInitFormatC(&format[2], '%');
if (maxCpuProcessItem)
{
PhInitFormatC(&format[3], '\n');
PhInitFormatSR(&format[4], maxCpuProcessItem->ProcessName->sr);
PhInitFormatS(&format[5], L": ");
PhInitFormatF(&format[6], maxCpuProcessItem->CpuUsage * 100, 2);
PhInitFormatC(&format[7], '%');
}
text = PhFormat(format, maxCpuProcessItem ? 8 : 3, 128);
if (maxCpuProcessItem) PhDereferenceObject(maxCpuProcessItem);
PhNfpModifyNotifyIcon(PH_ICON_CPU_HISTORY, NIF_TIP | NIF_ICON, text, icon);
DestroyIcon(icon);
PhDereferenceObject(text);
}
VOID PhNfpUpdateIconIoHistory(
VOID
)
{
static PH_GRAPH_DRAW_INFO drawInfo =
{
16,
16,
PH_GRAPH_USE_LINE_2,
2,
RGB(0x00, 0x00, 0x00),
16,
NULL,
NULL,
0,
0,
0,
0
};
ULONG maxDataCount;
ULONG lineDataCount;
PFLOAT lineData1;
PFLOAT lineData2;
FLOAT max;
ULONG i;
HBITMAP bitmap;
PVOID bits;
HDC hdc;
HBITMAP oldBitmap;
HICON icon;
HANDLE maxIoProcessId;
PPH_PROCESS_ITEM maxIoProcessItem;
PH_FORMAT format[8];
PPH_STRING text;
// Icon
PhNfpBeginBitmap(&drawInfo.Width, &drawInfo.Height, &bitmap, &bits, &hdc, &oldBitmap);
maxDataCount = drawInfo.Width / 2 + 1;
lineData1 = _alloca(maxDataCount * sizeof(FLOAT));
lineData2 = _alloca(maxDataCount * sizeof(FLOAT));
lineDataCount = min(maxDataCount, PhIoReadHistory.Count);
max = 1024 * 1024; // minimum scaling of 1 MB.
for (i = 0; i < lineDataCount; i++)
{
lineData1[i] =
(FLOAT)PhGetItemCircularBuffer_ULONG64(&PhIoReadHistory, i) +
(FLOAT)PhGetItemCircularBuffer_ULONG64(&PhIoOtherHistory, i);
lineData2[i] =
(FLOAT)PhGetItemCircularBuffer_ULONG64(&PhIoWriteHistory, i);
if (max < lineData1[i] + lineData2[i])
max = lineData1[i] + lineData2[i];
}
PhDivideSinglesBySingle(lineData1, max, lineDataCount);
PhDivideSinglesBySingle(lineData2, max, lineDataCount);
drawInfo.LineDataCount = lineDataCount;
drawInfo.LineData1 = lineData1;
drawInfo.LineData2 = lineData2;
drawInfo.LineColor1 = PhCsColorIoReadOther;
drawInfo.LineColor2 = PhCsColorIoWrite;
drawInfo.LineBackColor1 = PhHalveColorBrightness(PhCsColorIoReadOther);
drawInfo.LineBackColor2 = PhHalveColorBrightness(PhCsColorIoWrite);
if (bits)
PhDrawGraphDirect(hdc, bits, &drawInfo);
SelectObject(hdc, oldBitmap);
icon = PhNfBitmapToIcon(bitmap);
// Text
if (PhMaxIoHistory.Count != 0)
maxIoProcessId = UlongToHandle(PhGetItemCircularBuffer_ULONG(&PhMaxIoHistory, 0));
else
maxIoProcessId = NULL;
if (maxIoProcessId)
maxIoProcessItem = PhReferenceProcessItem(maxIoProcessId);
else
maxIoProcessItem = NULL;
PhInitFormatS(&format[0], L"I/O\nR: ");
PhInitFormatSize(&format[1], PhIoReadDelta.Delta);
PhInitFormatS(&format[2], L"\nW: ");
PhInitFormatSize(&format[3], PhIoWriteDelta.Delta);
PhInitFormatS(&format[4], L"\nO: ");
PhInitFormatSize(&format[5], PhIoOtherDelta.Delta);
if (maxIoProcessItem)
{
PhInitFormatC(&format[6], '\n');
PhInitFormatSR(&format[7], maxIoProcessItem->ProcessName->sr);
}
text = PhFormat(format, maxIoProcessItem ? 8 : 6, 128);
if (maxIoProcessItem) PhDereferenceObject(maxIoProcessItem);
PhNfpModifyNotifyIcon(PH_ICON_IO_HISTORY, NIF_TIP | NIF_ICON, text, icon);
DestroyIcon(icon);
PhDereferenceObject(text);
}
VOID PhNfpUpdateIconCommitHistory(
VOID
)
{
static PH_GRAPH_DRAW_INFO drawInfo =
{
16,
16,
0,
2,
RGB(0x00, 0x00, 0x00),
16,
NULL,
NULL,
0,
0,
0,
0
};
ULONG maxDataCount;
ULONG lineDataCount;
PFLOAT lineData1;
ULONG i;
HBITMAP bitmap;
PVOID bits;
HDC hdc;
HBITMAP oldBitmap;
HICON icon;
DOUBLE commitFraction;
PH_FORMAT format[5];
PPH_STRING text;
// Icon
PhNfpBeginBitmap(&drawInfo.Width, &drawInfo.Height, &bitmap, &bits, &hdc, &oldBitmap);
maxDataCount = drawInfo.Width / 2 + 1;
lineData1 = _alloca(maxDataCount * sizeof(FLOAT));
lineDataCount = min(maxDataCount, PhCommitHistory.Count);
for (i = 0; i < lineDataCount; i++)
lineData1[i] = (FLOAT)PhGetItemCircularBuffer_ULONG(&PhCommitHistory, i);
PhDivideSinglesBySingle(lineData1, (FLOAT)PhPerfInformation.CommitLimit, lineDataCount);
drawInfo.LineDataCount = lineDataCount;
drawInfo.LineData1 = lineData1;
drawInfo.LineColor1 = PhCsColorPrivate;
drawInfo.LineBackColor1 = PhHalveColorBrightness(PhCsColorPrivate);
if (bits)
PhDrawGraphDirect(hdc, bits, &drawInfo);
SelectObject(hdc, oldBitmap);
icon = PhNfBitmapToIcon(bitmap);
// Text
commitFraction = (DOUBLE)PhPerfInformation.CommittedPages / PhPerfInformation.CommitLimit;
PhInitFormatS(&format[0], L"Commit: ");
PhInitFormatSize(&format[1], UInt32x32To64(PhPerfInformation.CommittedPages, PAGE_SIZE));
PhInitFormatS(&format[2], L" (");
PhInitFormatF(&format[3], commitFraction * 100, 2);
PhInitFormatS(&format[4], L"%)");
text = PhFormat(format, 5, 96);
PhNfpModifyNotifyIcon(PH_ICON_COMMIT_HISTORY, NIF_TIP | NIF_ICON, text, icon);
DestroyIcon(icon);
PhDereferenceObject(text);
}
VOID PhNfpUpdateIconPhysicalHistory(
VOID
)
{
static PH_GRAPH_DRAW_INFO drawInfo =
{
16,
16,
0,
2,
RGB(0x00, 0x00, 0x00),
16,
NULL,
NULL,
0,
0,
0,
0
};
ULONG maxDataCount;
ULONG lineDataCount;
PFLOAT lineData1;
ULONG i;
HBITMAP bitmap;
PVOID bits;
HDC hdc;
HBITMAP oldBitmap;
HICON icon;
ULONG physicalUsage;
FLOAT physicalFraction;
PH_FORMAT format[5];
PPH_STRING text;
// Icon
PhNfpBeginBitmap(&drawInfo.Width, &drawInfo.Height, &bitmap, &bits, &hdc, &oldBitmap);
maxDataCount = drawInfo.Width / 2 + 1;
lineData1 = _alloca(maxDataCount * sizeof(FLOAT));
lineDataCount = min(maxDataCount, PhCommitHistory.Count);
for (i = 0; i < lineDataCount; i++)
lineData1[i] = (FLOAT)PhGetItemCircularBuffer_ULONG(&PhPhysicalHistory, i);
PhDivideSinglesBySingle(lineData1, (FLOAT)PhSystemBasicInformation.NumberOfPhysicalPages, lineDataCount);
drawInfo.LineDataCount = lineDataCount;
drawInfo.LineData1 = lineData1;
drawInfo.LineColor1 = PhCsColorPhysical;
drawInfo.LineBackColor1 = PhHalveColorBrightness(PhCsColorPhysical);
if (bits)
PhDrawGraphDirect(hdc, bits, &drawInfo);
SelectObject(hdc, oldBitmap);
icon = PhNfBitmapToIcon(bitmap);
// Text
physicalUsage = PhSystemBasicInformation.NumberOfPhysicalPages - PhPerfInformation.AvailablePages;
physicalFraction = (FLOAT)physicalUsage / PhSystemBasicInformation.NumberOfPhysicalPages;
PhInitFormatS(&format[0], L"Physical memory: ");
PhInitFormatSize(&format[1], UInt32x32To64(physicalUsage, PAGE_SIZE));
PhInitFormatS(&format[2], L" (");
PhInitFormatF(&format[3], physicalFraction * 100, 2);
PhInitFormatS(&format[4], L"%)");
text = PhFormat(format, 5, 96);
PhNfpModifyNotifyIcon(PH_ICON_PHYSICAL_HISTORY, NIF_TIP | NIF_ICON, text, icon);
DestroyIcon(icon);
PhDereferenceObject(text);
}
VOID PhNfpUpdateIconCpuUsage(
VOID
)
{
ULONG width;
ULONG height;
HBITMAP bitmap;
HDC hdc;
HBITMAP oldBitmap;
HICON icon;
HANDLE maxCpuProcessId;
PPH_PROCESS_ITEM maxCpuProcessItem;
PPH_STRING maxCpuText = NULL;
PPH_STRING text;
// Icon
PhNfpBeginBitmap(&width, &height, &bitmap, NULL, &hdc, &oldBitmap);
// This stuff is copied from CpuUsageIcon.cs (PH 1.x).
{
COLORREF kColor = PhCsColorCpuKernel;
COLORREF uColor = PhCsColorCpuUser;
COLORREF kbColor = PhHalveColorBrightness(PhCsColorCpuKernel);
COLORREF ubColor = PhHalveColorBrightness(PhCsColorCpuUser);
FLOAT k = PhCpuKernelUsage;
FLOAT u = PhCpuUserUsage;
LONG kl = (LONG)(k * height);
LONG ul = (LONG)(u * height);
RECT rect;
HBRUSH dcBrush;
HBRUSH dcPen;
POINT points[2];
dcBrush = GetStockObject(DC_BRUSH);
dcPen = GetStockObject(DC_PEN);
rect.left = 0;
rect.top = 0;
rect.right = width;
rect.bottom = height;
SetDCBrushColor(hdc, RGB(0x00, 0x00, 0x00));
FillRect(hdc, &rect, dcBrush);
// Draw the base line.
if (kl + ul == 0)
{
SelectObject(hdc, dcPen);
SetDCPenColor(hdc, uColor);
points[0].x = 0;
points[0].y = height - 1;
points[1].x = width;
points[1].y = height - 1;
Polyline(hdc, points, 2);
}
else
{
rect.left = 0;
rect.top = height - ul - kl;
rect.right = width;
rect.bottom = height - kl;
SetDCBrushColor(hdc, ubColor);
FillRect(hdc, &rect, dcBrush);
points[0].x = 0;
points[0].y = height - 1 - ul - kl;
if (points[0].y < 0) points[0].y = 0;
points[1].x = width;
points[1].y = points[0].y;
SelectObject(hdc, dcPen);
SetDCPenColor(hdc, uColor);
Polyline(hdc, points, 2);
if (kl != 0)
{
rect.left = 0;
rect.top = height - kl;
rect.right = width;
rect.bottom = height;
SetDCBrushColor(hdc, kbColor);
FillRect(hdc, &rect, dcBrush);
points[0].x = 0;
points[0].y = height - 1 - kl;
if (points[0].y < 0) points[0].y = 0;
points[1].x = width;
points[1].y = points[0].y;
SelectObject(hdc, dcPen);
SetDCPenColor(hdc, kColor);
Polyline(hdc, points, 2);
}
}
}
SelectObject(hdc, oldBitmap);
icon = PhNfBitmapToIcon(bitmap);
// Text
if (PhMaxCpuHistory.Count != 0)
maxCpuProcessId = UlongToHandle(PhGetItemCircularBuffer_ULONG(&PhMaxCpuHistory, 0));
else
maxCpuProcessId = NULL;
if (maxCpuProcessId)
{
if (maxCpuProcessItem = PhReferenceProcessItem(maxCpuProcessId))
{
maxCpuText = PhFormatString(
L"\n%s: %.2f%%",
maxCpuProcessItem->ProcessName->Buffer,
maxCpuProcessItem->CpuUsage * 100
);
PhDereferenceObject(maxCpuProcessItem);
}
}
text = PhFormatString(L"CPU usage: %.2f%%%s", (PhCpuKernelUsage + PhCpuUserUsage) * 100, PhGetStringOrEmpty(maxCpuText));
if (maxCpuText) PhDereferenceObject(maxCpuText);
PhNfpModifyNotifyIcon(PH_ICON_CPU_USAGE, NIF_TIP | NIF_ICON, text, icon);
DestroyIcon(icon);
PhDereferenceObject(text);
}
BOOLEAN PhNfpGetShowMiniInfoSectionData(
_In_ ULONG IconIndex,
_In_ PPH_NF_ICON RegisteredIcon,
_Out_ PPH_NF_MSG_SHOWMINIINFOSECTION_DATA Data
)
{
BOOLEAN showMiniInfo = FALSE;
if (RegisteredIcon)
{
Data->SectionName = NULL;
if (RegisteredIcon->Flags & PH_NF_ICON_SHOW_MINIINFO)
{
if (RegisteredIcon->MessageCallback)
{
RegisteredIcon->MessageCallback(
RegisteredIcon,
(ULONG_PTR)Data,
MAKELPARAM(PH_NF_MSG_SHOWMINIINFOSECTION, 0),
RegisteredIcon->Context
);
}
showMiniInfo = TRUE;
}
}
else
{
switch (1 << IconIndex)
{
case PH_ICON_CPU_HISTORY:
case PH_ICON_CPU_USAGE:
Data->SectionName = L"CPU";
break;
case PH_ICON_IO_HISTORY:
Data->SectionName = L"I/O";
break;
case PH_ICON_COMMIT_HISTORY:
Data->SectionName = L"Commit charge";
break;
case PH_ICON_PHYSICAL_HISTORY:
Data->SectionName = L"Physical memory";
break;
}
showMiniInfo = TRUE;
}
return showMiniInfo;
}
VOID PhNfpIconClickActivateTimerProc(
_In_ HWND hwnd,
_In_ UINT uMsg,
_In_ UINT_PTR idEvent,
_In_ DWORD dwTime
)
{
PhPinMiniInformation(MiniInfoActivePinType, 1, 0,
PH_MINIINFO_ACTIVATE_WINDOW | PH_MINIINFO_DONT_CHANGE_SECTION_IF_PINNED,
IconClickShowMiniInfoSectionData.SectionName, &IconClickLocation);
KillTimer(PhMainWndHandle, TIMER_ICON_CLICK_ACTIVATE);
}
VOID PhNfpDisableHover(
VOID
)
{
IconDisableHover = TRUE;
SetTimer(PhMainWndHandle, TIMER_ICON_RESTORE_HOVER, NFP_ICON_RESTORE_HOVER_DELAY, PhNfpIconRestoreHoverTimerProc);
}
VOID PhNfpIconRestoreHoverTimerProc(
_In_ HWND hwnd,
_In_ UINT uMsg,
_In_ UINT_PTR idEvent,
_In_ DWORD dwTime
)
{
IconDisableHover = FALSE;
KillTimer(PhMainWndHandle, TIMER_ICON_RESTORE_HOVER);
}