reactos/subsystems/win32/csrss/win32csr/appswitch.c
Amine Khaldi c424146e2c Create a branch for cmake bringup.
svn path=/branches/cmake-bringup/; revision=48236
2010-07-24 18:52:44 +00:00

269 lines
6 KiB
C

/*
* COPYRIGHT: See COPYING in the top level directory
* PROJECT: ReactOS system libraries
* FILE: subsys/csrss/win32csr/appswitch.c
* PURPOSE: app switching functionality
* PROGRAMMERS: Johannes Anderwald (janderwald@reactos.org)
*/
#define NDEBUG
#include "w32csr.h"
#include <debug.h>
typedef struct APPSWITCH_ITEM
{
HWND hwndDlg;
DWORD zPos;
HICON hIcon;
BOOL bFocus;
struct APPSWITCH_ITEM * Next;
WCHAR szText[1];
} APPSWITCH_ITEM, *PAPPSWITCH_ITEM;
static PAPPSWITCH_ITEM pRoot = NULL;
static DWORD NumOfWindows = 0;
static HWND hAppWindowDlg = NULL;
static HHOOK hhk = NULL;
UINT WINAPI PrivateExtractIconExW(LPCWSTR,int,HICON*,HICON*,UINT);
BOOL
CALLBACK
EnumWindowEnumProc(
HWND hwnd,
LPARAM lParam
)
{
PAPPSWITCH_ITEM pItem;
UINT Length;
HICON hIcon;
PAPPSWITCH_ITEM pCurItem;
DWORD dwPid;
HANDLE hProcess;
WCHAR szFileName[MAX_PATH] = {0};
/* check if the enumerated window is visible */
if (!IsWindowVisible(hwnd))
return TRUE;
/* get window icon */
hIcon = (HICON)SendMessage(hwnd, WM_GETICON, ICON_BIG, 0);
if (!hIcon)
{
GetWindowThreadProcessId(hwnd, &dwPid);
hProcess = OpenProcess(PROCESS_QUERY_INFORMATION|PROCESS_VM_READ, FALSE, dwPid);
if (hProcess)
{
if (GetModuleFileNameExW(hProcess, NULL, szFileName, MAX_PATH))
{
szFileName[MAX_PATH-1] = L'\0';
PrivateExtractIconExW(szFileName, 0, &hIcon, NULL, 1);
}
}
}
else
{
/* icons from WM_GETICON need to be copied */
hIcon = CopyIcon(hIcon);
}
/* get the text length */
Length = SendMessageW(hwnd, WM_GETTEXTLENGTH, 0, 0);
/* allocate item structure for it */
pItem = (PAPPSWITCH_ITEM)HeapAlloc(Win32CsrApiHeap, HEAP_ZERO_MEMORY, sizeof(APPSWITCH_ITEM) + Length * sizeof(WCHAR));
if (!pItem)
return TRUE;
if (Length)
{
/* retrieve the window text when available */
SendMessageW(hwnd, WM_GETTEXT, Length+1, (LPARAM)pItem->szText);
}
/* copy the window icon */
pItem->hIcon = hIcon;
/* store window handle */
pItem->hwndDlg = hwnd;
/* is the window the active window */
if (GetActiveWindow() == hwnd)
pItem->bFocus = TRUE;
if (!pRoot)
{
/* first item */
pRoot = pItem;
return TRUE;
}
/* enumerate the last item */
pCurItem = pRoot;
while(pCurItem->Next)
pCurItem = pCurItem->Next;
/* insert it into the list */
pCurItem->Next = pItem;
NumOfWindows++;
return TRUE;
}
VOID
EnumerateAppWindows(HDESK hDesk, HWND hwndDlg)
{
/* initialize defaults */
pRoot = NULL;
NumOfWindows = 0;
hAppWindowDlg = hwndDlg;
/* enumerate all windows */
EnumDesktopWindows(hDesk, EnumWindowEnumProc, (LPARAM)NULL);
if (NumOfWindows > 7)
{
/* FIXME resize window */
}
}
VOID
MarkNextEntryAsActive()
{
PAPPSWITCH_ITEM pItem;
pItem = pRoot;
if (!pRoot)
return;
while(pItem)
{
if (pItem->bFocus)
{
pItem->bFocus = FALSE;
if (pItem->Next)
pItem->Next->bFocus = TRUE;
else
pRoot->bFocus = TRUE;
}
pItem = pItem->Next;
}
InvalidateRgn(hAppWindowDlg, NULL, TRUE);
}
LRESULT
CALLBACK
KeyboardHookProc(
int nCode,
WPARAM wParam,
LPARAM lParam
)
{
PKBDLLHOOKSTRUCT hk = (PKBDLLHOOKSTRUCT) lParam;
if (wParam == WM_SYSKEYUP)
{
/* is tab key pressed */
if (hk->vkCode == VK_TAB)
{
if (hAppWindowDlg == NULL)
{
/* FIXME
* launch window
*/
DPRINT1("launch alt-tab window\n");
}
else
{
MarkNextEntryAsActive();
}
}
}
return CallNextHookEx(hhk, nCode, wParam, lParam);
}
VOID
PaintAppWindows(HWND hwndDlg, HDC hDc)
{
DWORD dwIndex, X, Y;
PAPPSWITCH_ITEM pCurItem;
RECT Rect;
DWORD XSize, YSize, XMax;
HBRUSH hBrush;
X = 10;
Y = 10;
XSize = GetSystemMetrics(SM_CXICON);
YSize = GetSystemMetrics(SM_CYICON);
XMax = (XSize+(XSize/2)) * 7 + X;
pCurItem = pRoot;
for (dwIndex = 0; dwIndex < NumOfWindows; dwIndex++)
{
if (X >= XMax)
{
X = 10;
Y += YSize + (YSize/2);
}
if (pCurItem->bFocus)
{
hBrush = CreateSolidBrush(RGB(30, 30, 255));
SetRect(&Rect, X-5, Y-5, X + XSize + 5, Y + YSize + 5);
FillRect(hDc, &Rect, hBrush);
DeleteObject((HGDIOBJ)hBrush);
SendDlgItemMessageW(hwndDlg, IDC_STATIC_CUR_APP, WM_SETTEXT, 0, (LPARAM)pCurItem->szText);
}
DrawIcon(hDc, X, Y, pCurItem->hIcon);
pCurItem = pCurItem->Next;
X += XSize +(XSize/2);
}
}
VOID
DestroyAppWindows()
{
PAPPSWITCH_ITEM pCurItem, pNextItem;
pCurItem = pRoot;
while(pCurItem)
{
pNextItem = pCurItem->Next;
DestroyIcon(pCurItem->hIcon);
HeapFree(Win32CsrApiHeap, 0, pCurItem);
pCurItem = pNextItem;
}
pRoot = NULL;
hAppWindowDlg = NULL;
NumOfWindows = 0;
}
INT_PTR
CALLBACK
SwitchWindowDlgProc(HWND hwndDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
PAINTSTRUCT Paint;
HDESK hInput;
switch (message)
{
case WM_INITDIALOG:
hInput = OpenInputDesktop(0,0, GENERIC_ALL);
if (hInput)
{
EnumerateAppWindows(hInput, hwndDlg);
CloseDesktop(hInput);
}
return TRUE;
case WM_PAINT:
BeginPaint(hwndDlg, &Paint);
PaintAppWindows(hwndDlg, Paint.hdc);
EndPaint(hwndDlg, &Paint);
break;
case WM_DESTROY:
DestroyAppWindows();
break;
}
return FALSE;
}
VOID
WINAPI
InitializeAppSwitchHook()
{
hhk = SetWindowsHookEx(WH_KEYBOARD_LL, KeyboardHookProc, NULL, 0);
DPRINT("InitializeAppSwitchHook hhk %p\n", hhk);
}