Mercurial > pidgin
view src/win32/IdleTracker/idletrack.c @ 13241:0d73a9019013
[gaim-migrate @ 15607]
Kevin Hunter managed to track down what was causing the problems in the
status selector with Gtk < 2.4. This is presumably a bug in the older
versions of Gtk but we can work around it here.
Thanks Kevin!
committer: Tailor Script <tailor@pidgin.im>
author | Stu Tomlinson <stu@nosnilmot.com> |
---|---|
date | Sun, 12 Feb 2006 16:19:21 +0000 |
parents | 5c50be815a4e |
children | 099bb2149158 |
line wrap: on
line source
/* * idletrack.c * * Authors: mrgentry @ http://www.experts-exchange.com * Herman Bloggs <hermanator12002@yahoo.com> * Date: February, 2003 * Description: Track user inactivity. * * Andrew Whewell <awhewell@users.sourceforge.net> - 25th June 2004. Added * support for GetLastInputInfo under Windows 2000 and above. This avoids having * IDLETRACK.DLL hook itself into every process on the machine, which makes * upgrades easier. The hook mechanism is also used by key loggers, so not * using hooks doesn't put the willys up programs that keep an eye out for * loggers. * * Windows 9x doesn't have GetLastInputInfo - when Gaim runs on these machines * the code silently falls back onto the old hooking scheme. */ #define _WIN32_WINNT 0x0500 #include <windows.h> #define EXPORT __declspec(dllexport) static HANDLE hMapObject = NULL; static DWORD *lastTime = NULL; static HHOOK keyHook = NULL; static HHOOK mouseHook = NULL; static HINSTANCE g_hInstance = NULL; static POINT g_point; /* GetLastInputInfo address and module - if g_GetLastInputInfo == NULL then * we fall back on the old "hook the world" method. GetLastInputInfo was brought * in with Windows 2000 so Windows 9x will still hook everything. */ typedef BOOL (WINAPI *GETLASTINPUTINFO)(LASTINPUTINFO *); static HMODULE g_user32 = NULL; static GETLASTINPUTINFO g_GetLastInputInfo = NULL; static DWORD* setup_shared_mem() { BOOL fInit; /* Set up the shared memory. */ hMapObject = CreateFileMapping((HANDLE) 0xFFFFFFFF, /* use paging file */ NULL, /* no security attributes */ PAGE_READWRITE, /* read/write access */ 0, /* size: high 32-bits */ sizeof(DWORD), /* size: low 32-bits */ "timermem"); /* name of map object */ if (hMapObject == NULL) return NULL; /* The first process to attach initializes memory. */ fInit = (GetLastError() != ERROR_ALREADY_EXISTS); /* Get a pointer to the file-mapped shared memory. */ lastTime = (DWORD*) MapViewOfFile(hMapObject, /* object to map view of */ FILE_MAP_WRITE, /* read/write access */ 0, /* high offset: map from */ 0, /* low offset: beginning */ 0); /* default: map entire file */ if (lastTime == NULL) return NULL; *lastTime = GetTickCount(); return lastTime; } LRESULT CALLBACK KeyboardProc(int code, WPARAM wParam, LPARAM lParam) { if (!(code < 0)) { if (lastTime == NULL) lastTime = setup_shared_mem(); if (lastTime) *lastTime = GetTickCount(); } return CallNextHookEx(keyHook, code, wParam, lParam); } LRESULT CALLBACK MouseProc(int code, WPARAM wParam, LPARAM lParam) { /* We need to verify that the Mouse pointer has actually moved. */ if(!(code < 0) && !((g_point.x == ((MOUSEHOOKSTRUCT*)lParam)->pt.x) && (g_point.y == ((MOUSEHOOKSTRUCT*)lParam)->pt.y))) { g_point.x = ((MOUSEHOOKSTRUCT*)lParam)->pt.x; g_point.y = ((MOUSEHOOKSTRUCT*)lParam)->pt.y; if (lastTime == NULL) lastTime = setup_shared_mem(); if (lastTime) *lastTime = GetTickCount(); } return CallNextHookEx(mouseHook, code, wParam, lParam); } EXPORT DWORD wgaim_get_lastactive() { DWORD result = 0; /* If we have GetLastInputInfo then use it, otherwise use the hooks*/ if (g_GetLastInputInfo != NULL) { LASTINPUTINFO lii; memset(&lii, 0, sizeof(lii)); lii.cbSize = sizeof(lii); if (g_GetLastInputInfo(&lii)) { result = lii.dwTime; } } else { if (lastTime == NULL) lastTime = setup_shared_mem(); if (lastTime) result = *lastTime; } return result; } EXPORT BOOL wgaim_set_idlehooks() { /* Is GetLastInputInfo available?*/ g_user32 = LoadLibrary("user32.dll"); if (g_user32) { g_GetLastInputInfo = (GETLASTINPUTINFO)GetProcAddress(g_user32, "GetLastInputInfo"); } /* If we couldn't find GetLastInputInfo then fall back onto the hooking scheme*/ if (g_GetLastInputInfo == NULL) { /* Set up the shared memory.*/ lastTime = setup_shared_mem(); if (lastTime == NULL) return FALSE; *lastTime = GetTickCount(); /* Set up the keyboard hook.*/ keyHook = SetWindowsHookEx(WH_KEYBOARD, KeyboardProc, g_hInstance, 0); if (keyHook == NULL) { UnmapViewOfFile(lastTime); CloseHandle(hMapObject); return FALSE; } /* Set up the mouse hook.*/ mouseHook = SetWindowsHookEx(WH_MOUSE, MouseProc, g_hInstance, 0); if (mouseHook == NULL) { UnhookWindowsHookEx(keyHook); UnmapViewOfFile(lastTime); CloseHandle(hMapObject); return FALSE; } } return TRUE; } EXPORT void wgaim_remove_idlehooks() { if (g_user32 != NULL) FreeLibrary(g_user32); if (keyHook) UnhookWindowsHookEx(keyHook); if (mouseHook) UnhookWindowsHookEx(mouseHook); if (lastTime) UnmapViewOfFile(lastTime); if (hMapObject) CloseHandle(hMapObject); } int WINAPI DllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved) { switch(dwReason) { case DLL_PROCESS_ATTACH: g_hInstance = hInstance; g_point.x = 0; g_point.y = 0; break; case DLL_PROCESS_DETACH: break; } return TRUE; }