-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathbattery.cpp
More file actions
324 lines (318 loc) · 11.5 KB
/
battery.cpp
File metadata and controls
324 lines (318 loc) · 11.5 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
#pragma once
#include <Windows.h>
#include <cstring>
#include <cmath>
#include <ctime>
#include <fstream>
#include <ios>
#include <iosfwd>
#include <libloaderapi.h>
#include <sal.h>
#include <shellapi.h>
#include <hidusage.h>
#include "battery.h"
#include "resource.h"
constexpr int MAX_LOADSTRING = 100;
CHAR szTitle[MAX_LOADSTRING];
CHAR szWindowClass[MAX_LOADSTRING];
NOTIFYICONDATA nid;
HMENU hMenu;
static BOOL thermoicon(BOOL b = ACTION::KEEP){
static BOOL ti;
switch (b){
case ACTION::KEEP: break;
case ACTION::TOGGLE: ti = !ti; break;
default: ti = b; break;
}
return ti;
}
static void darken(PCOLOR color, double brightness){
PBYTE rgb = (PBYTE)color;
for (int j = 0; j < 3; j++) rgb[j] = min(255, (BYTE)round(rgb[j] * brightness));
}
static COLOR colormixer(COLOR color0, COLOR color1, double ratio){
ratio = max(0, min(1, ratio));
darken(&color0, 1 - ratio);
darken(&color1, ratio);
return (COLOR)((UINT)color0 + (UINT)color1);
}
static COLOR thermometer(double temp){
COLOR color = colormixer(COLOR::BLUE, COLOR::RED, (temp - TEMP.BLUE) / (TEMP.RED - TEMP.BLUE));
PBYTE rgb = (PBYTE)&color;
DEBUG(temp << "\t" << std::hex << (UINT)color << std::dec << "\n");
return color;
}
static BOOL cursoronicon(void){
NOTIFYICONIDENTIFIER ni = { sizeof(NOTIFYICONIDENTIFIER), nid.hWnd, nid.uID, nid.guidItem };
POINT p;
RECT r;
GetCursorPos(&p);
Shell_NotifyIconGetRect(&ni, &r);
return r.left < p.x && p.x < r.right && r.top < p.y && p.y < r.bottom;
}
static BOOL getfocus(void){
return GetFocus() || cursoronicon();
}
static BOOL isframe(PCOLOR color){
PBYTE rgb = (PBYTE)color;
for (int j = 1; j < 3; j++) if (rgb[0] != rgb[j]) return false;
return true;
}
static void setled(LED led){
HDC hdc = GetDC(NULL);
HICON hIcon = nid.hIcon;
ICONINFO ii;
GetIconInfo(hIcon, &ii);
BITMAP bm = { 0 };
GetObject(ii.hbmColor, sizeof(BITMAP), &bm);
SelectObject(hdc, &bm);
BITMAPINFO3 b3 = { 0 };
b3.bi.bmiHeader.biSize = sizeof(BITMAPINFO3);
GetDIBits(hdc, ii.hbmColor, 0, 1, NULL, &b3.bi, DIB_PAL_COLORS);
COLOR px[256];
if (countof(px) < bm.bmWidth) return;
for (int k = 0; k < bm.bmHeight; k++){
GetDIBits(hdc, ii.hbmColor, k, 1, px, &b3.bi, DIB_PAL_COLORS);
for (int j = 0; j < bm.bmWidth; j++) if ((UINT)px[j]) px[j] = isframe(&px[j]) ? led.frame : led.face;
SetDIBits(hdc, ii.hbmColor, k, 1, px, &b3.bi, DIB_PAL_COLORS);
}
nid.hIcon = CreateIconIndirect(&ii);
DestroyIcon(hIcon);
ReleaseDC(NULL, hdc);
Shell_NotifyIcon(NIM_MODIFY, &nid);
}
static void getmenutext(IDM item, LPSTR text, UINT cch){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_STRING;
mii.cch = cch;
mii.dwTypeData = text;
GetMenuItemInfo(hMenu, (UINT)item, false, &mii);
}
void menutext(IDM item, LPSTR text){
char otext[64];
getmenutext(item, otext, sizeof(otext));
if (strcmp(otext, text)){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_STRING;
mii.dwTypeData = text;
SetMenuItemInfo(hMenu, (UINT)item, false, &mii);
}
}
static BOOL getmenucheck(IDM item){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_STATE;
GetMenuItemInfo(hMenu, (UINT)item, false, &mii);
return false != (mii.fState & MFS_CHECKED);
}
static void menucheck(IDM item, BOOL b){
if (b != getmenucheck(item)){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_STATE;
mii.fState = b ? MFS_CHECKED : MFS_UNCHECKED;
SetMenuItemInfo(hMenu, (UINT)item, false, &mii);
}
}
static BOOL menuhilite(IDM item){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_STATE;
GetMenuItemInfo(hMenu, (UINT)item, false, &mii);
return false != (mii.fState & MFS_HILITE);
}
void tiptext(char* x){
if (strcmp(nid.szTip, x)){
strcpy_s(nid.szTip, x);
Shell_NotifyIcon(NIM_MODIFY, &nid);
}
}
static BOOL menuexists(IDM item){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_ID;
GetMenuItemInfo(hMenu, (UINT)item, false, &mii);
return mii.wID == (UINT)item;
}
BOOL getverbosity(void){
return menuexists(IDM::WATT);
}
void statusled(PBATTERY b){
static LED oled;
LED led = { COLOR::BLUE, COLOR::FRAME};
if (thermoicon()) led.face = thermometer(b->temperature[0]);
else if (b->temperature[0] > TEMP.THROTTLE) led.face = COLOR::RED;
else if (b->capacity.level < CAPACITY_LEVEL_WHITE) switch (b->state){
case STATE::CHARGE: led.face = b->flags.airplane ? COLOR::PURPLE : COLOR::ORANGE; break;
}
else switch (b->state){
case STATE::INACTIVE: led.face = COLOR::GRAY; break;
case STATE::CHARGE: led.face = COLOR::WHITE; break;
}
if (GetFocus()) { darken(&led.face, 0.8); darken(&led.frame, 0.8); }
if (memcmp(&oled, &led, sizeof(LED))) { setled(led); oled = led; }
menucheck(IDM::TEMPERATURE, b->temperature[0] > TEMP.THROTTLE);
menucheck(IDM::LEVEL, b->state == STATE::CHARGE);
menucheck(IDM::WATT, b->flags.airplane);
}
static void insertitem(IDM above, IDM wid, const char* text = "", HBITMAP bmp = NULL, UINT type = MFT_RADIOCHECK){
MENUITEMINFO mii = { sizeof(MENUITEMINFO) };
mii.fMask = MIIM_ID | MIIM_STRING | MIIM_BITMAP | MIIM_STATE | MIIM_FTYPE;
mii.wID = (UINT)wid;
mii.fType = type;
mii.hbmpItem = bmp;
mii.dwTypeData = (LPSTR)text;
InsertMenuItem(hMenu, (UINT)above, false, &mii);
}
static void insertitem(IDM wid, const char* text = "", HBITMAP bmp = NULL){ insertitem(IDM::ZERO, wid, text, bmp); }
void setverbosity(BOOL b){
BOOL v = getverbosity();
if (b == ACTION::TOGGLE) b = !v;
else { b = b > 0; if (!b && !menuhilite(IDM::TEMPERATURE) && getfocus()) return; }
if (b != v){
thermoicon(!b);
if (b) { insertitem(IDM::LEVEL); insertitem(IDM::WATT); }
else { RemoveMenu(hMenu, (UINT)IDM::LEVEL, false); RemoveMenu(hMenu, (UINT)IDM::WATT, false); }
}
}
static void setaccthreshold(int delta){
if (delta != ACTION::TOGGLE){
if (HIBYTE(GetAsyncKeyState(VK_LSHIFT))) delta *= 2;
if (!HIBYTE(GetAsyncKeyState(VK_LCONTROL))) delta *= 5;
}
setthreshold(delta);
}
static void rawedit(int delta, USHORT usButtonFlags = RI_MOUSE_WHEEL){
if (!getverbosity()) setverbosity(true);
else if (menuhilite(IDM::TEMPERATURE)) setverbosity(delta);
else if (menuhilite(IDM::LEVEL)) setaccthreshold(delta);
else if (menuhilite(IDM::WATT)) setairplane(delta);
else if (HIBYTE(GetAsyncKeyState(VK_RCONTROL))) setverbosity(delta);
else if (cursoronicon()) switch (usButtonFlags){
case RI_MOUSE_WHEEL: setaccthreshold(delta); break;
case RI_MOUSE_HWHEEL: setairplane(delta); break;
}
}
static void wmtimer(HWND hWnd, TIMER uElapse){
static TIMER delay, elapse;
delay = (TIMER)(uElapse < TIMER::SLOW ? 0 : (UINT)delay + (UINT)elapse);
if (elapse != uElapse && !((UINT)delay && delay < TIMER::DELAY)) { elapse = uElapse; SetTimer(hWnd, 1, (UINT)elapse, NULL); }
if (uElapse > TIMER::FAST) battery();
}
static void rawmouse(HWND hWnd, PRAWINPUT raw){
if (getfocus()){
int delta = COPYSIGN(1, *(short*)&raw->data.mouse.usButtonData);
USHORT flags = raw->data.mouse.usButtonFlags;
switch (flags) case RI_MOUSE_WHEEL: case RI_MOUSE_HWHEEL: rawedit(delta, flags);
}
}
static void rawkeyboard(HWND hWnd, PRAWINPUT raw){
if (raw->data.keyboard.Message == WM_KEYDOWN){
if (getfocus()){
if (HIBYTE(GetAsyncKeyState(VK_RSHIFT))) thermoicon(ACTION::TOGGLE);
if (HIBYTE(GetAsyncKeyState(VK_RCONTROL))) rawedit(ACTION::TOGGLE);
}
if (GetFocus()) switch (raw->data.keyboard.VKey){
case VK_SPACE: rawedit(ACTION::TOGGLE); break;
case VK_LEFT: rawedit(-1); break;
case VK_RIGHT: rawedit(1); break;
case 'A': setairplane(ACTION::TOGGLE); break;
case 'T': setthreshold(ACTION::TOGGLE); break;
case 'V': setverbosity(ACTION::TOGGLE); break;
case 'Q': if (HIBYTE(GetAsyncKeyState(VK_LCONTROL))) SendMessage(hWnd, WM_CLOSE, 0, 0); break;
}
}
}
static void wminput(HWND hWnd, LPARAM lParam){
RAWINPUT raw = { 0 };
UINT size = sizeof(RAWINPUT);
if (getfocus()) wmtimer(hWnd, TIMER::FAST);
GetRawInputData((HRAWINPUT)lParam, RID_INPUT, &raw, &size, sizeof(RAWINPUTHEADER));
switch (raw.header.dwType){
case RIM_TYPEMOUSE: rawmouse(hWnd, &raw); break;
case RIM_TYPEKEYBOARD: rawkeyboard(hWnd, &raw); break;
}
}
static void trackmenu(HWND hWnd, UINT uFlags){
POINT p;
if (SetForegroundWindow(hWnd) && GetCursorPos(&p)){
switch ((IDM)TrackPopupMenuEx(hMenu, TPM_BOTTOMALIGN | TPM_RETURNCMD | uFlags, p.x, p.y, hWnd, NULL)){
case IDM::QUIT: SendMessage(hWnd, WM_CLOSE, 0, 0); break;
case IDM::TEMPERATURE: setverbosity(ACTION::TOGGLE); break;
case IDM::LEVEL: setthreshold(ACTION::TOGGLE); break;
case IDM::WATT: setairplane(ACTION::TOGGLE); break;
}
}
}
static void trackquitmenu(HWND hWnd){
insertitem(IDM::TEMPERATURE, IDM::SEPARATOR, NULL);
insertitem(IDM::SEPARATOR, IDM::QUIT, "Quit", HBMMENU_POPUP_CLOSE);
trackmenu(hWnd, TPM_LEFTBUTTON);
RemoveMenu(hMenu, (UINT)IDM::SEPARATOR, false);
RemoveMenu(hMenu, (UINT)IDM::QUIT, false);
}
static void wmicon(HWND hWnd, LPARAM lParam){
switch (LOWORD(lParam)){
case WM_LBUTTONUP: trackmenu(hWnd, TPM_RIGHTBUTTON); break;
case WM_RBUTTONUP: trackquitmenu(hWnd); break;
}
}
static LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam){
switch (message){
case WM_CREATE: hMenu = CreatePopupMenu(); insertitem(IDM::TEMPERATURE); break;
case WM_DESTROY: DestroyMenu(hMenu); PostQuitMessage(0); break;
case WM_ICON: wmicon(hWnd, lParam); break;
case WM_INPUT: wminput(hWnd, lParam); break;
case WM_POWERBROADCAST: wmtimer(hWnd, TIMER::FAST); break;
case WM_TIMER: wmtimer(hWnd, TIMER::SLOW); break;
}
return DefWindowProc(hWnd, message, wParam, lParam);
}
static void icon(HINSTANCE hInstance, HWND hWnd){
nid.cbSize = sizeof(NOTIFYICONDATA);
nid.hWnd = hWnd;
nid.uID = IDI_ICON1;
nid.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP;
nid.uCallbackMessage = WM_ICON;
nid.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_ICON1));
Shell_NotifyIcon(NIM_ADD, &nid);
}
static void rawinput(HWND hWnd){
RAWINPUTDEVICE rid = { 0 };
rid.usUsagePage = HID_USAGE_PAGE_GENERIC;
rid.hwndTarget = hWnd;
rid.dwFlags = RIDEV_PAGEONLY | RIDEV_INPUTSINK;
RegisterRawInputDevices(&rid, 1, sizeof(RAWINPUTDEVICE));
}
static BOOL InitInstance(HINSTANCE hInstance){
HWND hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);
if (!hWnd) return FALSE;
icon(hInstance, hWnd);
rawinput(hWnd);
wmtimer(hWnd, TIMER::SLOW);
RegisterSuspendResumeNotification(hWnd, DEVICE_NOTIFY_WINDOW_HANDLE);
ShowWindow(hWnd, SW_HIDE);
UpdateWindow(hWnd);
return TRUE;
}
static ATOM MyRegisterClass(HINSTANCE hInstance){
WNDCLASSEX cex = { 0 };
cex.cbSize = sizeof(WNDCLASSEX);
cex.style = CS_HREDRAW | CS_VREDRAW;
cex.lpfnWndProc = WndProc;
cex.hInstance = hInstance;
cex.hIconSm = cex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_ICON1));
cex.hCursor = LoadCursor(nullptr, IDC_ARROW);
cex.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
cex.lpszMenuName = MAKEINTRESOURCE(IDC_BATTERY);
cex.lpszClassName = szWindowClass;
return RegisterClassEx(&cex);
}
int WinMain(_In_ HINSTANCE hi, _In_opt_ HINSTANCE hpi, _In_ LPSTR lpCmdLine, _In_ int nShowCmd){
LoadString(hi, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadString(hi, IDC_BATTERY, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hi);
if (!InitInstance(hi)) return 0;
MSG msg = { 0 };
while (GetMessage(&msg, nullptr, 0, 0)){
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return (int)msg.wParam;
}