267 lines
7.7 KiB
C++
267 lines
7.7 KiB
C++
#include <windows.h>
|
|
#include <gdiplus.h>
|
|
#include <vector>
|
|
#include <thread>
|
|
#include <mutex>
|
|
#include <queue>
|
|
using namespace Gdiplus;
|
|
using namespace std;
|
|
#pragma comment (lib,"Gdiplus.lib")
|
|
|
|
LRESULT CALLBACK MouseProc(int nCode, WPARAM wParam, LPARAM lParam);
|
|
|
|
Gdiplus::Graphics* graphics;
|
|
ULONG_PTR gdiplusToken;
|
|
HWND hwnd;
|
|
mutex mtx;
|
|
|
|
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam);
|
|
void CreateFullScreenTransparentWindow(const wchar_t* window_name, const wchar_t* class_name);
|
|
int g_hotkeyId = 1;
|
|
typedef struct {
|
|
int x1, y1, x2, y2;
|
|
int width;
|
|
int time;
|
|
}VineDraw;
|
|
queue<VineDraw> vines;
|
|
|
|
DWORD WINAPI eraser(LPVOID lpParam);
|
|
void vine2(int x1, int y1, float anchor, float turn, int speed, int cost, Color color, int count, int clone);
|
|
|
|
int WINAPI WinMain(HINSTANCE hThisInstance, HINSTANCE hPrevInstance, LPSTR lpszArgument, int nCmdShow)
|
|
{
|
|
HANDLE hMutex = CreateMutex(NULL, FALSE, L"cursorAnim");
|
|
if (hMutex == NULL) {
|
|
return 1;
|
|
}
|
|
if (GetLastError() == ERROR_ALREADY_EXISTS) {
|
|
MessageBox(nullptr, L"已运行", L"错误", MB_OK);
|
|
CloseHandle(hMutex);
|
|
return 1;
|
|
}
|
|
|
|
CreateFullScreenTransparentWindow(L"鼠标特效", L"cursor_anim");
|
|
HHOOK hook = SetWindowsHookEx(WH_MOUSE_LL, MouseProc, NULL, 0);
|
|
|
|
|
|
|
|
// 创建线程
|
|
HANDLE hThread = CreateThread(NULL, 0, eraser, NULL, 0, NULL);
|
|
if (hThread == NULL) {
|
|
MessageBox(nullptr, L"eraser线程启动失败", L"错误", MB_OK);
|
|
CloseHandle(hMutex);
|
|
return 1;
|
|
}
|
|
|
|
// 设置线程优先级为低
|
|
SetThreadPriority(hThread, THREAD_PRIORITY_LOWEST);
|
|
SetPriorityClass(GetCurrentProcess(), IDLE_PRIORITY_CLASS);
|
|
|
|
|
|
|
|
MSG msg;
|
|
while (GetMessage(&msg, nullptr, 0, 0))
|
|
{
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
Sleep(3);
|
|
}
|
|
CloseHandle(hMutex);
|
|
// 等待线程结束
|
|
WaitForSingleObject(hThread, INFINITE);
|
|
// 关闭线程句柄
|
|
CloseHandle(hThread);
|
|
return 0;
|
|
}
|
|
|
|
void CreateFullScreenTransparentWindow(const wchar_t* window_name, const wchar_t* class_name)
|
|
{
|
|
WNDCLASS wc = {};
|
|
wc.lpfnWndProc = WndProc;
|
|
wc.hInstance = GetModuleHandle(nullptr);
|
|
wc.lpszClassName = class_name;
|
|
wc.style = CS_HREDRAW | CS_VREDRAW;
|
|
wc.hbrBackground = (HBRUSH)CreateSolidBrush(RGB(255, 0, 0));
|
|
if (!RegisterClass(&wc))
|
|
{
|
|
MessageBox(nullptr, L"窗口类注册失败", L"错误", MB_OK);
|
|
return;
|
|
}
|
|
|
|
hwnd = CreateWindowExW(WS_EX_TRANSPARENT | WS_EX_LAYERED | WS_EX_TOPMOST | WS_EX_TOOLWINDOW, class_name, window_name, WS_POPUP,
|
|
0, 0, GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN)-40,
|
|
nullptr, nullptr, GetModuleHandle(nullptr), nullptr);
|
|
if (hwnd == nullptr)
|
|
{
|
|
MessageBox(nullptr, L"窗口创建失败", L"错误", MB_OK);
|
|
return;
|
|
}
|
|
|
|
SetLayeredWindowAttributes(hwnd, RGB(255, 0, 0), 100, LWA_COLORKEY);
|
|
ShowWindow(hwnd, SW_SHOW);
|
|
UpdateWindow(hwnd);
|
|
}
|
|
|
|
|
|
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
switch (message)
|
|
{
|
|
case WM_CREATE:
|
|
{
|
|
Gdiplus::GdiplusStartupInput gdiplusStartupInput;
|
|
Gdiplus::GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL);
|
|
graphics = new Gdiplus::Graphics(hwnd);
|
|
break;
|
|
}
|
|
case WM_TIMER:
|
|
break;
|
|
case WM_DESTROY:
|
|
{
|
|
MessageBox(nullptr, L"关闭", L"关闭", MB_OK);
|
|
delete graphics;
|
|
UnregisterHotKey(hwnd, g_hotkeyId);
|
|
Gdiplus::GdiplusShutdown(gdiplusToken);
|
|
PostQuitMessage(0);
|
|
break;
|
|
}
|
|
default:
|
|
return DefWindowProc(hwnd, message, wparam, lparam);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void vine2(int x1, int y1, float anchor, float turn, int speed, int cost, Color color, int count, int clone)
|
|
{
|
|
if (speed <= 0)
|
|
{
|
|
Sleep(300);
|
|
return;
|
|
}
|
|
count++;
|
|
int x2 = x1 + (speed)*cos(anchor);
|
|
int y2 = y1 + (speed)*sin(anchor);
|
|
Pen penline(color, speed);
|
|
penline.SetEndCap(LineCapRound);
|
|
penline.SetStartCap(LineCapRound);
|
|
mtx.lock();
|
|
graphics->DrawLine(&penline, x1, y1, x2, y2);
|
|
vines.push(VineDraw{x1,y1,x2,y2,speed,10});
|
|
|
|
if (clone > 0 && rand() % 10 == 0)
|
|
{
|
|
thread t(vine2, x1, y1, anchor + rand() % 10 / 23.0, turn * (-1 + rand() % 2 * 2), speed + rand() % 2, cost, Color(rand() % 70, 155 + rand() % 100, 20 + rand() % 70), 0, --clone);
|
|
t.detach();
|
|
}
|
|
if (count % 9 == 0)
|
|
speed -= cost;
|
|
anchor += turn;
|
|
if (count % 3 == 0)
|
|
turn += turn / 2;
|
|
if (turn > 0.6)
|
|
speed = 0;
|
|
mtx.unlock();
|
|
Sleep(1);
|
|
vine2(x2, y2, anchor, turn, speed, cost, color, count, clone);
|
|
}
|
|
|
|
DWORD WINAPI eraser(LPVOID lpParam)
|
|
{
|
|
Pen erase(Color(255, 0, 0));
|
|
erase.SetEndCap(LineCapRound);
|
|
erase.SetStartCap(LineCapRound);
|
|
while(true)
|
|
{
|
|
if (!vines.empty())
|
|
{
|
|
vines.front().time--;
|
|
if (vines.front().time <= 0)
|
|
{
|
|
erase.SetWidth(vines.front().width * 2 + 2);
|
|
mtx.lock();
|
|
graphics->DrawLine(&erase, vines.front().x1, vines.front().y1, vines.front().x2, vines.front().y2);
|
|
vines.pop();
|
|
mtx.unlock();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Sleep(70);
|
|
}
|
|
}
|
|
}
|
|
|
|
//void vine(int x1, int y1, float anchor, float turn, int speed, int cost, Color color, int count, int clone)
|
|
//{
|
|
// if (speed <= 0)
|
|
// {
|
|
// Sleep(300);
|
|
// return;
|
|
// }
|
|
// count++;
|
|
// int x2 = x1 + (speed)*cos(anchor);
|
|
// int y2 = y1 + (speed)*sin(anchor);
|
|
// Pen penline(color, speed);
|
|
// penline.SetEndCap(LineCapRound);
|
|
// penline.SetStartCap(LineCapRound);
|
|
// mtx.lock();
|
|
// graphics->DrawLine(&penline, x1, y1, x2, y2);
|
|
// mtx.unlock();
|
|
// Sleep(1);
|
|
// if (clone > 0 && rand() % 10 == 0)
|
|
// {
|
|
// int x_copy = x1;
|
|
// int y_copy = y1;
|
|
// float anchor_copy = anchor + rand() % 10 / 26.0;
|
|
// float turn_copy = turn * (-1 + rand() % 2 * 2);
|
|
// int speed_copy = speed + rand() % 3;
|
|
// int cost_copy = cost + rand() % 3;
|
|
// thread t(vine, x_copy, y_copy, anchor_copy, turn_copy, speed_copy, cost_copy, Color(rand() % 50, 100 + rand() % 100, 20 + rand() % 50), 0, --clone);
|
|
// t.detach();
|
|
// }
|
|
// if (count % 10 == 0)
|
|
// speed -= cost;
|
|
// anchor += turn;
|
|
// if (count % 3 == 0)
|
|
// turn += turn / 2;
|
|
// if (turn > 0.6)
|
|
// speed = 0;
|
|
// vine(x2, y2, anchor, turn, speed, cost, color, count, clone);
|
|
// Sleep(count / 3);
|
|
// Pen erase(Color(255, 0, 0), speed * 2 + 3);
|
|
// erase.SetEndCap(LineCapRound);
|
|
// erase.SetStartCap(LineCapRound);
|
|
// mtx.lock();
|
|
// graphics->DrawLine(&erase, x1, y1, x2, y2);
|
|
// mtx.unlock();
|
|
//}
|
|
|
|
LRESULT CALLBACK MouseProc(int nCode, WPARAM wParam, LPARAM lParam) {
|
|
if (nCode >= 0) {
|
|
if (wParam == WM_LBUTTONDOWN) {
|
|
// 获取鼠标点击位置
|
|
POINT p;
|
|
GetCursorPos(&p);
|
|
float base = rand() % 10 / 3;
|
|
thread t0(vine2, p.x, p.y, 0 + base, 0.02, 5, 2, Color(0, 255, 0), 0, 3);
|
|
t0.detach();
|
|
if (rand() % 3 != 0)
|
|
{
|
|
thread t1(vine2, p.x, p.y, 2 + base, 0.010, 5, 2, Color(0, 255, 0), 0, 3);
|
|
t1.detach();
|
|
}
|
|
if (rand() % 2 == 0) {
|
|
thread t2(vine2, p.x, p.y, -2 + base, 0.025, 5, 2, Color(0, 255, 0), 0, 3);
|
|
t2.detach();
|
|
}
|
|
else {
|
|
thread t3(vine2, p.x, p.y, -0.5 + base, 0.025, 5, 2, Color(0, 255, 0), 0, 3);
|
|
t3.detach();
|
|
thread t4(vine2, p.x, p.y, -3 + base, 0.025, 5, 2, Color(0, 255, 0), 0, 3);
|
|
t4.detach();
|
|
}
|
|
}
|
|
}
|
|
return CallNextHookEx(NULL, nCode, wParam, lParam);
|
|
}
|