Windows2000/private/windows/shell/accessib/accwiz/lookprev.cpp
2020-09-30 17:12:32 +02:00

901 lines
26 KiB
C++

/* LOOKPREV.C
**
** Copyright (C) Microsoft, 1993, All Rights Reserved.
**
**
** History:
**
*/
#include "pch.hxx" // PCH
#pragma hdrstop
#include "AccWiz.h"
#include "desk.h"
//#include "deskid.h"
#include "resource.h"
#include "look.h"
#include "LookPrev.h"
#define RCZ(element) g_elements[element].rc
///
// Support function
void MyDrawBorderBelow(HDC hdc, LPRECT prc);
void MyDrawFrame(HDC hdc, LPRECT prc, HBRUSH hbrColor, int cl);
HDC g_hdcMem;
TCHAR g_szABC[] = TEXT("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz");
int cxSize;
////
// Declarations of static variables declared in classes
BOOL CLookPreviewGlobals::sm_bOneInstanceCreated = FALSE; // This variable insures that only one instance of CLookPreviewGlobals is created
CLookPreviewGlobals CLookPrev::sm_Globals;
////
// CLookPreviewGlobals member functions
BOOL CLookPreviewGlobals::Initialize()
{
if(m_bInitialized)
return TRUE;
m_bInitialized = TRUE;
// Make sure there is only one instance of this class created
_ASSERT(!sm_bOneInstanceCreated);
sm_bOneInstanceCreated = TRUE;
// Load our display strings.
VERIFY(LoadString(g_hInstDll, IDS_ACTIVE, m_szActive, ARRAYSIZE(m_szActive)));
VERIFY(LoadString(g_hInstDll, IDS_INACTIVE, m_szInactive, ARRAYSIZE(m_szInactive)));
VERIFY(LoadString(g_hInstDll, IDS_MINIMIZED, m_szMinimized, ARRAYSIZE(m_szMinimized)));
VERIFY(LoadString(g_hInstDll, IDS_ICONTITLE, m_szIconTitle, ARRAYSIZE(m_szIconTitle)));
VERIFY(LoadString(g_hInstDll, IDS_NORMAL, m_szNormal, ARRAYSIZE(m_szNormal)));
VERIFY(LoadString(g_hInstDll, IDS_DISABLED, m_szDisabled, ARRAYSIZE(m_szDisabled)));
VERIFY(LoadString(g_hInstDll, IDS_SELECTED, m_szSelected, ARRAYSIZE(m_szSelected)));
VERIFY(LoadString(g_hInstDll, IDS_MSGBOX, m_szMsgBox, ARRAYSIZE(m_szMsgBox)));
VERIFY(LoadString(g_hInstDll, IDS_BUTTONTEXT, m_szButton, ARRAYSIZE(m_szButton)));
// VERIFY(LoadString(g_hInstDll, IDS_SMCAPTION, m_szSmallCaption, ARRAYSIZE(m_szSmallCaption)));
VERIFY(LoadString(g_hInstDll, IDS_WINDOWTEXT, m_szWindowText, ARRAYSIZE(m_szWindowText)));
VERIFY(LoadString(g_hInstDll, IDS_MSGBOXTEXT, m_szMsgBoxText, ARRAYSIZE(m_szMsgBoxText)));
// Register Look Preview window class
WNDCLASS wc;
memset(&wc, 0, sizeof(wc));
wc.style = 0;
wc.lpfnWndProc = CLookPrev::LookPreviewWndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = g_hInstDll;
wc.hIcon = NULL;
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE+1);
wc.lpszMenuName = NULL;
wc.lpszClassName = LOOKPREV_CLASS;
VERIFY(RegisterClass(&wc));
// Global initialization of g_hdcMem
HBITMAP hbmDefault;
HDC hdc = GetDC(NULL);
VERIFY(g_hdcMem = CreateCompatibleDC(hdc));
ReleaseDC(NULL, hdc);
HBITMAP hbm = CreateBitmap(1, 1, 1, 1, NULL);
hbmDefault = (HBITMAP)SelectObject(g_hdcMem, hbm);
SelectObject(g_hdcMem, hbmDefault);
DeleteObject(hbm);
// Old initialization form Look_InitSysStuff()
int i;
NONCLIENTMETRICS ncm;
HKEY hkey;
hdc = GetDC(NULL);
g_LogDPI = GetDeviceCaps(hdc, LOGPIXELSY);
g_bPalette = GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE;
ReleaseDC(NULL, hdc);
// always make a palette even on non-pal device
if (g_bPalette || TRUE)
{
DWORD pal[21];
HPALETTE hpal = (HPALETTE)GetStockObject(DEFAULT_PALETTE);
pal[1] = RGB(255, 255, 255);
pal[2] = RGB(0, 0, 0 );
pal[3] = RGB(192, 192, 192);
pal[4] = RGB(128, 128, 128);
pal[5] = RGB(255, 0, 0 );
pal[6] = RGB(128, 0, 0 );
pal[7] = RGB(255, 255, 0 );
pal[8] = RGB(128, 128, 0 );
pal[9] = RGB(0 , 255, 0 );
pal[10] = RGB(0 , 128, 0 );
pal[11] = RGB(0 , 255, 255);
pal[12] = RGB(0 , 128, 128); // Needs to be changed to get Blue color
pal[13] = RGB(0 , 0, 255);
pal[14] = RGB(0 , 0, 128);
pal[15] = RGB(255, 0, 255);
pal[16] = RGB(128, 0, 128);
GetPaletteEntries(hpal, 11, 1, (LPPALETTEENTRY)&pal[17]);
pal[0] = MAKELONG(0x300, 17);
g_hpalVGA = CreatePalette((LPLOGPALETTE)pal);
// get magic colors
GetPaletteEntries(hpal, 8, 4, (LPPALETTEENTRY)&pal[17]);
pal[0] = MAKELONG(0x300, 20);
g_hpal3D = CreatePalette((LPLOGPALETTE)pal);
}
// system colors
for (i = 0; i < NT40_COLOR_MAX; i++)
{
g_Options.m_schemePreview.m_rgb[i] = GetSysColor(i);
g_brushes[i] = NULL;
}
// sizes and fonts
ncm.cbSize = sizeof(ncm);
SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm),
(void far *)(LPNONCLIENTMETRICS)&ncm, FALSE);
SetMyNonClientMetrics(&ncm);
SystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(LOGFONT),
(void far *)(LPLOGFONT)&(g_fonts[FONT_ICONTITLE].lf), FALSE);
SystemParametersInfo(SPI_SETGRADIENTCAPTIONS, 0, (PVOID)TRUE, 0);
// default shell icon sizes
g_sizes[ SIZE_ICON ].CurSize = GetSystemMetrics( SM_CXICON );
g_sizes[ SIZE_SMICON ].CurSize = g_sizes[ SIZE_ICON ].CurSize / 2;
if( RegOpenKey( HKEY_CURRENT_USER, c_szRegPathUserMetrics, &hkey )
== ERROR_SUCCESS )
{
TCHAR val[ 8 ];
LONG len = sizeof( val );
if( RegQueryValueEx( hkey, c_szRegValIconSize, 0, NULL, (LPBYTE)&val,
(LPDWORD)&len ) == ERROR_SUCCESS )
{
g_sizes[ SIZE_ICON ].CurSize = (int)MyStrToLong( val );
}
len = SIZEOF( val );
if( RegQueryValueEx( hkey, c_szRegValSmallIconSize, 0, NULL, (LPBYTE)&val,
(LPDWORD)&len ) == ERROR_SUCCESS )
{
g_sizes[ SIZE_SMICON ].CurSize = (int)MyStrToLong( val );
}
RegCloseKey( hkey );
}
g_sizes[ SIZE_DXICON ].CurSize =
GetSystemMetrics( SM_CXICONSPACING ) - g_sizes[ SIZE_ICON ].CurSize;
if( g_sizes[ SIZE_DXICON ].CurSize < 0 )
g_sizes[ SIZE_DXICON ].CurSize = 0;
g_sizes[ SIZE_DYICON ].CurSize =
GetSystemMetrics( SM_CYICONSPACING ) - g_sizes[ SIZE_ICON ].CurSize;
if( g_sizes[ SIZE_DYICON ].CurSize < 0 )
g_sizes[ SIZE_DYICON ].CurSize = 0;
// clean out the memory
for (i = 0; i < NUM_FONTS; i++)
{
g_fonts[i].hfont = NULL;
}
// build all the brushes/fonts we need
Look_RebuildSysStuff(TRUE);
// From Look_InitDialog
// initialize some globals
cyBorder = GetSystemMetrics(SM_CYBORDER);
cxBorder = GetSystemMetrics(SM_CXBORDER);
cxEdge = GetSystemMetrics(SM_CXEDGE);
cyEdge = GetSystemMetrics(SM_CYEDGE);
return TRUE;
}
////
// CLookPreviewGlobals member functions
// This is the static window proc function of CLookPrev
LRESULT CALLBACK CLookPrev::LookPreviewWndProc (HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
PAINTSTRUCT ps;
CLookPrev *pThis = (CLookPrev *)GetWindowLongPtr(hWnd, GWLP_USERDATA);
if(!pThis)
{
// Create the class to handle this object
// Store the 'this' pointer in
pThis = new CLookPrev;
pThis->m_hwnd = hWnd;
SetWindowLongPtr (hWnd, GWLP_USERDATA, (INT_PTR)pThis);
}
switch(message)
{
case WM_NCCREATE:
{
DWORD dw;
dw = GetWindowLong (hWnd,GWL_STYLE);
SetWindowLong (hWnd, GWL_STYLE, dw | WS_BORDER);
dw = GetWindowLong (hWnd,GWL_EXSTYLE);
SetWindowLong (hWnd, GWL_EXSTYLE, dw | WS_EX_CLIENTEDGE);
}
return TRUE;
case WM_CREATE:
pThis->OnCreate();
break;
case WM_PALETTECHANGED:
if ((HWND)wParam == hWnd)
break;
//fallthru
case WM_QUERYNEWPALETTE:
if (g_hpal3D)
InvalidateRect(hWnd, NULL, FALSE);
break;
case WM_PAINT:
BeginPaint(hWnd, &ps);
pThis->OnPaint(ps.hdc);
EndPaint(hWnd, &ps);
return 0;
case LPM_REPAINT:
pThis->OnRepaint();
return 0;
case LPM_RECALC:
pThis->OnRecalc();
return 0;
}
return DefWindowProc(hWnd,message,wParam,lParam);
}
void CLookPrev::OnCreate()
{
// Load menu for window
m_hmenuSample = LoadMenu(g_hInstDll, MAKEINTRESOURCE(IDR_MENU));
EnableMenuItem(m_hmenuSample, IDM_DISABLED, MF_GRAYED | MF_BYCOMMAND);
HiliteMenuItem(m_hwnd, m_hmenuSample, IDM_SELECTED, MF_HILITE | MF_BYCOMMAND);
// Create Bitmap for window
RECT rc;
HDC hdc;
GetClientRect(m_hwnd, &rc);
hdc = GetDC(NULL);
m_hbmLook = CreateCompatibleBitmap(hdc, rc.right - rc.left, rc.bottom - rc.top);
ReleaseDC(NULL, hdc);
// Mirror the memory DC if the window is mirrored to keep the text readable.
if (GetWindowLong(m_hwnd, GWL_EXSTYLE) & WS_EX_LAYOUTRTL) {
SetLayout(g_hdcMem, LAYOUT_RTL);
}
}
void CLookPrev::OnDestroy()
{
if (m_hbmLook)
DeleteObject(m_hbmLook);
if (m_hmenuSample)
DestroyMenu(m_hmenuSample);
// Un-allocate memory for this instance of the class
delete this;
}
void CLookPrev::OnPaint(HDC hdc)
{
if (m_hbmLook)
ShowBitmap(hdc);
else
Draw(hdc);
}
void CLookPrev::ShowBitmap(HDC hdc)
{
RECT rc;
HBITMAP hbmOld;
HPALETTE hpalOld = NULL;
if (g_hpal3D)
{
hpalOld = SelectPalette(hdc, g_hpal3D, FALSE);
RealizePalette(hdc);
}
GetClientRect(m_hwnd, &rc);
hbmOld = (HBITMAP)SelectObject(g_hdcMem, m_hbmLook);
BitBlt(hdc, 0, 0, rc.right - rc.left, rc.bottom - rc.top, g_hdcMem, 0, 0, SRCCOPY);
SelectObject(g_hdcMem, hbmOld);
if (hpalOld)
{
SelectPalette(hdc, hpalOld, FALSE);
RealizePalette(hdc);
}
}
void CLookPrev::Draw(HDC hdc)
{
RECT rcT;
int nMode;
DWORD rgbBk;
int cxSize, cySize;
HANDLE hOldColors;
HPALETTE hpalOld = NULL;
HICON hiconLogo;
// HFONT hfontOld;
SaveDC(hdc);
if (g_hpal3D)
{
hpalOld = SelectPalette(hdc, g_hpal3D, TRUE);
RealizePalette(hdc);
}
hOldColors = SetSysColorsTemp(g_Options.m_schemePreview.m_rgb, g_brushes, COLOR_MAX_97_NT5/*COLOR_MAX_95_NT4*/);
hiconLogo = (HICON)LoadImage(NULL, IDI_APPLICATION, IMAGE_ICON,
g_sizes[SIZE_CAPTION].CurSize - 2*cxBorder,
g_sizes[SIZE_CAPTION].CurSize - 2*cyBorder, 0);
// Setup drawing stuff
nMode = SetBkMode(hdc, TRANSPARENT);
rgbBk = GetTextColor(hdc);
cxSize = GetSystemMetrics(SM_CXSIZE);
cySize = GetSystemMetrics(SM_CYSIZE);
// Desktop
FillRect(hdc, &RCZ(ELEMENT_DESKTOP), g_brushes[COLOR_BACKGROUND]);
// Inactive window
// Border
rcT = RCZ(ELEMENT_INACTIVEBORDER);
DrawEdge(hdc, &rcT, EDGE_RAISED, BF_RECT | BF_ADJUST);
MyDrawFrame(hdc, &rcT, g_brushes[COLOR_INACTIVEBORDER], g_sizes[SIZE_FRAME].CurSize);
MyDrawFrame(hdc, &rcT, g_brushes[COLOR_3DFACE], 1);
// Caption
rcT = RCZ(ELEMENT_INACTIVECAPTION);
MyDrawBorderBelow(hdc, &rcT);
// NOTE: because USER draws icon stuff using its own DC and subsequently
// its own palette, we need to make sure to use the inactivecaption
// brush before USER does so that it will be realized against our palette.
// this might get fixed in USER by better be safe.
// "clip" the caption title under the buttons
rcT.left = RCZ(ELEMENT_INACTIVESYSBUT2).left - cyEdge;
FillRect(hdc, &rcT, g_brushes[COLOR_GRADIENTINACTIVECAPTION]);
rcT.right = rcT.left;
rcT.left = RCZ(ELEMENT_INACTIVECAPTION).left;
DrawCaptionTemp(NULL, hdc, &rcT, g_fonts[FONT_CAPTION].hfont, hiconLogo, sm_Globals.m_szInactive, DC_ICON | DC_TEXT | DC_GRADIENT);
DrawFrameControl(hdc, &RCZ(ELEMENT_INACTIVESYSBUT1), DFC_CAPTION, DFCS_CAPTIONCLOSE);
rcT = RCZ(ELEMENT_INACTIVESYSBUT2);
rcT.right -= (rcT.right - rcT.left)/2;
DrawFrameControl(hdc, &rcT, DFC_CAPTION, DFCS_CAPTIONMIN);
rcT.left = rcT.right;
rcT.right = RCZ(ELEMENT_INACTIVESYSBUT2).right;
DrawFrameControl(hdc, &rcT, DFC_CAPTION, DFCS_CAPTIONMAX);
#if 0
// small caption window
//
{
HICON hicon;
int temp;
rcT = RCZ(ELEMENT_SMCAPTION);
hicon = (HICON)LoadImage(NULL, IDI_APPLICATION,
IMAGE_ICON,
g_sizes[SIZE_SMCAPTION].CurSize - 2*cxBorder,
g_sizes[SIZE_SMCAPTION].CurSize - 2*cyBorder,
0);
DrawEdge(hdc, &rcT, EDGE_RAISED, BF_TOP | BF_LEFT | BF_RIGHT | BF_ADJUST);
MyDrawFrame(hdc, &rcT, g_brushes[COLOR_3DFACE], 1);
// "clip" the caption title under the buttons
temp = rcT.left; // remember start of actual caption
rcT.left = RCZ(ELEMENT_SMCAPSYSBUT).left - cxEdge;
FillRect(hdc, &rcT, g_brushes[COLOR_ACTIVECAPTION]);
rcT.right = rcT.left;
rcT.left = temp; // start of actual caption
DrawCaptionTemp(NULL, hdc, &rcT, g_fonts[FONT_SMCAPTION].hfont, hicon, sm_Globals.m_szSmallCaption, DC_SMALLCAP | DC_ICON | DC_TEXT);
DestroyIcon(hicon);
DrawFrameControl(hdc, &RCZ(ELEMENT_SMCAPSYSBUT), DFC_CAPTION, DFCS_CAPTIONCLOSE);
}
#endif
// Active window
// Border
rcT = RCZ(ELEMENT_ACTIVEBORDER);
DrawEdge(hdc, &rcT, EDGE_RAISED, BF_RECT | BF_ADJUST);
MyDrawFrame(hdc, &rcT, g_brushes[COLOR_ACTIVEBORDER], g_sizes[SIZE_FRAME].CurSize);
MyDrawFrame(hdc, &rcT, g_brushes[COLOR_3DFACE], 1);
// Caption
rcT = RCZ(ELEMENT_ACTIVECAPTION);
MyDrawBorderBelow(hdc, &rcT);
// "clip" the caption title under the buttons
rcT.left = RCZ(ELEMENT_ACTIVESYSBUT2).left - cxEdge;
FillRect(hdc, &rcT, g_brushes[COLOR_GRADIENTACTIVECAPTION]);
rcT.right = rcT.left;
rcT.left = RCZ(ELEMENT_ACTIVECAPTION).left;
DrawCaptionTemp(NULL, hdc, &rcT, g_fonts[FONT_CAPTION].hfont, hiconLogo, sm_Globals.m_szActive, DC_ACTIVE | DC_ICON | DC_TEXT | DC_GRADIENT);
DrawFrameControl(hdc, &RCZ(ELEMENT_ACTIVESYSBUT1), DFC_CAPTION, DFCS_CAPTIONCLOSE);
rcT = RCZ(ELEMENT_ACTIVESYSBUT2);
rcT.right -= (rcT.right - rcT.left)/2;
DrawFrameControl(hdc, &rcT, DFC_CAPTION, DFCS_CAPTIONMIN);
rcT.left = rcT.right;
rcT.right = RCZ(ELEMENT_ACTIVESYSBUT2).right;
DrawFrameControl(hdc, &rcT, DFC_CAPTION, DFCS_CAPTIONMAX);
// Menu
rcT = RCZ(ELEMENT_MENUNORMAL);
#if 0 // HACK TO SLIP USING DrawMenuBarTemp() which is not available on Memphis
DrawMenuBarTemp(m_hwnd, hdc, &rcT, g_Options.m_hmenuSample, g_fonts[FONT_MENU].hfont);
#else
{
// JMC: HACK - HARD CODED TEXT
HFONT hOldFont = (HFONT)SelectObject(hdc, g_fonts[FONT_MENU].hfont);
COLORREF clrrefOldText = SetTextColor(hdc, g_Options.m_schemePreview.m_rgb[COLOR_MENUTEXT]);
COLORREF clrrefOldBk = SetBkColor(hdc, g_Options.m_schemePreview.m_rgb[COLOR_MENU]);
int nOldMode = SetBkMode(hdc, OPAQUE);
// LPCTSTR lpszText = __TEXT(" File Edit Help");
TCHAR szText[200];
LoadString(g_hInstDll, IDS_PREVIEWMENUTEXT, szText, ARRAYSIZE(szText));
ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rcT, NULL, 0, NULL);
DrawText(hdc, szText, lstrlen(szText), &rcT, DT_VCENTER | DT_SINGLELINE | DT_EXPANDTABS);
SetTextColor(hdc, clrrefOldText);
SetBkColor(hdc, clrrefOldBk);
SetBkMode(hdc, nOldMode);
SelectObject(hdc, hOldFont);
}
#endif
MyDrawBorderBelow(hdc, &rcT);
// Client area
rcT = RCZ(ELEMENT_WINDOW);
DrawEdge(hdc, &rcT, EDGE_SUNKEN, BF_RECT | BF_ADJUST);
FillRect(hdc, &rcT, g_brushes[COLOR_WINDOW]);
// window text
SetBkMode(hdc, TRANSPARENT);
SetTextColor(hdc, g_Options.m_schemePreview.m_rgb[COLOR_WINDOWTEXT]);
TextOut(hdc, RCZ(ELEMENT_WINDOW).left + 2*cxEdge, RCZ(ELEMENT_WINDOW).top + 2*cyEdge, sm_Globals.m_szWindowText, lstrlen(sm_Globals.m_szWindowText));
// scroll bar
rcT = RCZ(ELEMENT_SCROLLBAR);
//MyDrawFrame(hdc, &rcT, g_brushes[COLOR_3DSHADOW], 1);
//g_brushes[COLOR_SCROLLBAR]);
//FillRect(hdc, &rcT, (HBRUSH)DefWindowProc(m_hwnd, WM_CTLCOLORSCROLLBAR, (WPARAM)hdc, (LPARAM)m_hwnd));
FillRect(hdc, &rcT, g_brushes[COLOR_SCROLLBAR]);
DrawFrameControl(hdc, &RCZ(ELEMENT_SCROLLUP), DFC_SCROLL, DFCS_SCROLLUP);
DrawFrameControl(hdc, &RCZ(ELEMENT_SCROLLDOWN), DFC_SCROLL, DFCS_SCROLLDOWN);
#if 0 // Don't draw message box
// MessageBox
rcT = RCZ(ELEMENT_MSGBOX);
DrawEdge(hdc, &rcT, EDGE_RAISED, BF_RECT | BF_ADJUST);
FillRect(hdc, &rcT, g_brushes[COLOR_3DFACE]);
rcT = RCZ(ELEMENT_MSGBOXCAPTION);
MyDrawBorderBelow(hdc, &rcT);
// "clip" the caption title under the buttons
rcT.left = RCZ(ELEMENT_MSGBOXSYSBUT).left - cxEdge;
FillRect(hdc, &rcT, g_brushes[COLOR_GRADIENTACTIVECAPTION]);
rcT.right = rcT.left;
rcT.left = RCZ(ELEMENT_MSGBOXCAPTION).left;
DrawCaptionTemp(NULL, hdc, &rcT, g_fonts[FONT_CAPTION].hfont, hiconLogo, sm_Globals.m_szMsgBox, DC_ACTIVE | DC_ICON | DC_TEXT | DC_GRADIENT);
DrawFrameControl(hdc, &RCZ(ELEMENT_MSGBOXSYSBUT), DFC_CAPTION, DFCS_CAPTIONCLOSE);
// message box text
SetBkMode(hdc, TRANSPARENT);
SetTextColor(hdc, g_Options.m_schemePreview.m_rgb[COLOR_WINDOWTEXT]);
hfontOld = (HFONT)SelectObject(hdc, g_fonts[FONT_MSGBOX].hfont);
TextOut(hdc, RCZ(ELEMENT_MSGBOX).left + 3*cxEdge, RCZ(ELEMENT_MSGBOXCAPTION).bottom + cyEdge,
sm_Globals.m_szMsgBoxText, lstrlen(sm_Globals.m_szMsgBoxText));
if (hfontOld)
SelectObject(hdc, hfontOld);
// Button
rcT = RCZ(ELEMENT_BUTTON);
DrawFrameControl(hdc, &rcT, DFC_BUTTON, DFCS_BUTTONPUSH);
// ?????? what font should this use ??????
SetBkMode(hdc, TRANSPARENT);
SetTextColor(hdc, g_Options.m_schemePreview.m_rgb[COLOR_BTNTEXT]);
DrawText(hdc, sm_Globals.m_szButton, -1, &rcT, DT_CENTER | DT_NOPREFIX |
DT_SINGLELINE | DT_VCENTER);
#endif
SetBkColor(hdc, rgbBk);
SetBkMode(hdc, nMode);
if (hiconLogo)
DestroyIcon(hiconLogo);
SetSysColorsTemp(NULL, NULL, (UINT_PTR)hOldColors);
if (hpalOld)
{
hpalOld = SelectPalette(hdc, hpalOld, FALSE);
RealizePalette(hdc);
}
RestoreDC(hdc, -1);
}
void CLookPrev::OnRepaint()
{
HBITMAP hbmOld;
if (m_hbmLook)
{
hbmOld = (HBITMAP)SelectObject(g_hdcMem, m_hbmLook);
Draw(g_hdcMem);
SelectObject(g_hdcMem, hbmOld);
}
InvalidateRect(m_hwnd, NULL, FALSE);
}
void CLookPrev::OnRecalc()
{
DWORD cxNormal;
int cxDisabled, cxSelected;
int cxAvgCharx2;
RECT rc;
HFONT hfontT;
int cxFrame, cyFrame;
int cyCaption;
int i;
SIZE sizButton;
GetClientRect(m_hwnd, &rc);
// Get our drawing data
cxSize = GetSystemMetrics(SM_CXSIZE);
cxFrame = (g_sizes[SIZE_FRAME].CurSize + 1) * cxBorder + cxEdge;
cyFrame = (g_sizes[SIZE_FRAME].CurSize + 1) * cyBorder + cyEdge;
cyCaption = g_sizes[SIZE_CAPTION].CurSize;
// Get text dimensions, with proper font.
hfontT = (HFONT)SelectObject(g_hdcMem, g_fonts[FONT_MENU].hfont);
GetTextExtentPoint32(g_hdcMem, sm_Globals.m_szNormal, lstrlen(sm_Globals.m_szNormal), &sizButton);
cxNormal = sizButton.cx;
GetTextExtentPoint32(g_hdcMem, sm_Globals.m_szDisabled, lstrlen(sm_Globals.m_szDisabled), &sizButton);
cxDisabled = sizButton.cx;
GetTextExtentPoint32(g_hdcMem, sm_Globals.m_szSelected, lstrlen(sm_Globals.m_szSelected), &sizButton);
cxSelected = sizButton.cx;
// get the average width (USER style) of menu font
GetTextExtentPoint32(g_hdcMem, g_szABC, 52, &sizButton);
cxAvgCharx2 = 2 * (sizButton.cx / 52);
// actual menu-handling widths of strings is bigger
cxDisabled += cxAvgCharx2;
cxSelected += cxAvgCharx2;
cxNormal += cxAvgCharx2;
SelectObject(g_hdcMem, hfontT);
GetTextExtentPoint32(g_hdcMem, sm_Globals.m_szButton, lstrlen(sm_Globals.m_szButton), &sizButton);
// Desktop
RCZ(ELEMENT_DESKTOP) = rc;
InflateRect(&rc, -8*cxBorder, -8*cyBorder);
// Windows
rc.bottom -= cyFrame + cyCaption;
RCZ(ELEMENT_ACTIVEBORDER) = rc;
OffsetRect(&RCZ(ELEMENT_ACTIVEBORDER), cxFrame,
cyFrame + cyCaption + cyBorder);
RCZ(ELEMENT_ACTIVEBORDER).bottom -= cyCaption;
// Inactive window
rc.right -= cyCaption;
RCZ(ELEMENT_INACTIVEBORDER) = rc;
// Caption
InflateRect(&rc, -cxFrame, -cyFrame);
rc.bottom = rc.top + cyCaption + cyBorder;
RCZ(ELEMENT_INACTIVECAPTION) = rc;
// close button
InflateRect(&rc, -cxEdge, -cyEdge);
rc.bottom -= cyBorder; // compensate for magic line under caption
RCZ(ELEMENT_INACTIVESYSBUT1) = rc;
RCZ(ELEMENT_INACTIVESYSBUT1).left = rc.right - (cyCaption - cxEdge);
// min/max buttons
RCZ(ELEMENT_INACTIVESYSBUT2) = rc;
RCZ(ELEMENT_INACTIVESYSBUT2).right = RCZ(ELEMENT_INACTIVESYSBUT1).left - cxEdge;
RCZ(ELEMENT_INACTIVESYSBUT2).left = RCZ(ELEMENT_INACTIVESYSBUT2).right -
2 * (cyCaption - cxEdge);
#if 0
// small caption window
RCZ(ELEMENT_SMCAPTION) = RCZ(ELEMENT_ACTIVEBORDER);
RCZ(ELEMENT_SMCAPTION).bottom = RCZ(ELEMENT_SMCAPTION).top;
RCZ(ELEMENT_SMCAPTION).top -= g_sizes[SIZE_SMCAPTION].CurSize + cyEdge + 2 * cyBorder;
RCZ(ELEMENT_SMCAPTION).right -= cxFrame;
RCZ(ELEMENT_SMCAPTION).left = RCZ(ELEMENT_INACTIVECAPTION).right + 2 * cxFrame;
RCZ(ELEMENT_SMCAPSYSBUT) = RCZ(ELEMENT_SMCAPTION);
// deflate inside frame/border to caption and then another edge's worth
RCZ(ELEMENT_SMCAPSYSBUT).right -= 2 * cxEdge + cxBorder;
RCZ(ELEMENT_SMCAPSYSBUT).top += 2 * cxEdge + cxBorder;
RCZ(ELEMENT_SMCAPSYSBUT).bottom -= cxEdge + cxBorder;
RCZ(ELEMENT_SMCAPSYSBUT).left = RCZ(ELEMENT_SMCAPSYSBUT).right -
(g_sizes[SIZE_SMCAPTION].CurSize - cxEdge);
#endif
// Active window
// Caption
rc = RCZ(ELEMENT_ACTIVEBORDER);
InflateRect(&rc, -cxFrame, -cyFrame);
RCZ(ELEMENT_ACTIVECAPTION) = rc;
RCZ(ELEMENT_ACTIVECAPTION).bottom =
RCZ(ELEMENT_ACTIVECAPTION).top + cyCaption + cyBorder;
// close button
RCZ(ELEMENT_ACTIVESYSBUT1) = RCZ(ELEMENT_ACTIVECAPTION);
InflateRect(&RCZ(ELEMENT_ACTIVESYSBUT1), -cxEdge, -cyEdge);
RCZ(ELEMENT_ACTIVESYSBUT1).bottom -= cyBorder; // compensate for magic line under caption
RCZ(ELEMENT_ACTIVESYSBUT1).left = RCZ(ELEMENT_ACTIVESYSBUT1).right -
(cyCaption - cxEdge);
// min/max buttons
RCZ(ELEMENT_ACTIVESYSBUT2) = RCZ(ELEMENT_ACTIVESYSBUT1);
RCZ(ELEMENT_ACTIVESYSBUT2).right = RCZ(ELEMENT_ACTIVESYSBUT1).left - cxEdge;
RCZ(ELEMENT_ACTIVESYSBUT2).left = RCZ(ELEMENT_ACTIVESYSBUT2).right -
2 * (cyCaption - cxEdge);
// Menu
rc.top = RCZ(ELEMENT_ACTIVECAPTION).bottom;
RCZ(ELEMENT_MENUNORMAL) = rc;
rc.top = RCZ(ELEMENT_MENUNORMAL).bottom = RCZ(ELEMENT_MENUNORMAL).top + g_sizes[SIZE_MENU].CurSize;
RCZ(ELEMENT_MENUDISABLED) = RCZ(ELEMENT_MENUSELECTED) = RCZ(ELEMENT_MENUNORMAL);
RCZ(ELEMENT_MENUDISABLED).left = RCZ(ELEMENT_MENUNORMAL).left + cxNormal;
RCZ(ELEMENT_MENUDISABLED).right = RCZ(ELEMENT_MENUSELECTED).left =
RCZ(ELEMENT_MENUDISABLED).left + cxDisabled;
RCZ(ELEMENT_MENUSELECTED).right = RCZ(ELEMENT_MENUSELECTED).left + cxSelected;
// Client
RCZ(ELEMENT_WINDOW) = rc;
// Scrollbar
InflateRect(&rc, -cxEdge, -cyEdge); // take off client edge
RCZ(ELEMENT_SCROLLBAR) = rc;
rc.right = RCZ(ELEMENT_SCROLLBAR).left = rc.right - g_sizes[SIZE_SCROLL].CurSize;
RCZ(ELEMENT_SCROLLUP) = RCZ(ELEMENT_SCROLLBAR);
RCZ(ELEMENT_SCROLLUP).bottom = RCZ(ELEMENT_SCROLLBAR).top + g_sizes[SIZE_SCROLL].CurSize;
RCZ(ELEMENT_SCROLLDOWN) = RCZ(ELEMENT_SCROLLBAR);
RCZ(ELEMENT_SCROLLDOWN).top = RCZ(ELEMENT_SCROLLBAR).bottom - g_sizes[SIZE_SCROLL].CurSize;
// Message Box
rc.top = RCZ(ELEMENT_WINDOW).top + (RCZ(ELEMENT_WINDOW).bottom - RCZ(ELEMENT_WINDOW).top) / 2;
rc.bottom = RCZ(ELEMENT_DESKTOP).bottom - 2*cyEdge;
rc.left = RCZ(ELEMENT_WINDOW).left + 2*cyEdge;
rc.right = RCZ(ELEMENT_WINDOW).left + (RCZ(ELEMENT_WINDOW).right - RCZ(ELEMENT_WINDOW).left) / 2 + 3*cyCaption;
RCZ(ELEMENT_MSGBOX) = rc;
// Caption
RCZ(ELEMENT_MSGBOXCAPTION) = rc;
RCZ(ELEMENT_MSGBOXCAPTION).top += cyEdge + cyBorder;
RCZ(ELEMENT_MSGBOXCAPTION).bottom = RCZ(ELEMENT_MSGBOXCAPTION).top + cyCaption + cyBorder;
RCZ(ELEMENT_MSGBOXCAPTION).left += cxEdge + cxBorder;
RCZ(ELEMENT_MSGBOXCAPTION).right -= cxEdge + cxBorder;
RCZ(ELEMENT_MSGBOXSYSBUT) = RCZ(ELEMENT_MSGBOXCAPTION);
InflateRect(&RCZ(ELEMENT_MSGBOXSYSBUT), -cxEdge, -cyEdge);
RCZ(ELEMENT_MSGBOXSYSBUT).left = RCZ(ELEMENT_MSGBOXSYSBUT).right -
(cyCaption - cxEdge);
RCZ(ELEMENT_MSGBOXSYSBUT).bottom -= cyBorder; // line under caption
// Button
RCZ(ELEMENT_BUTTON).bottom = RCZ(ELEMENT_MSGBOX).bottom - (4*cyBorder + cyEdge);
RCZ(ELEMENT_BUTTON).top = RCZ(ELEMENT_BUTTON).bottom - (sizButton.cy + 8 * cyBorder);
i = (RCZ(ELEMENT_BUTTON).bottom - RCZ(ELEMENT_BUTTON).top) * 3;
RCZ(ELEMENT_BUTTON).left = (rc.left + (rc.right - rc.left)/2) - i/2;
RCZ(ELEMENT_BUTTON).right = RCZ(ELEMENT_BUTTON).left + i;
}
////
// Support functions
// ---------
// MyDrawFrame() -
// Draws bordered frame, border size cl, and adjusts passed in rect.
// ---------
void MyDrawFrame(HDC hdc, LPRECT prc, HBRUSH hbrColor, int cl)
{
HBRUSH hbr;
int cx, cy;
RECT rcT;
rcT = *prc;
cx = cl * cxBorder;
cy = cl * cyBorder;
hbr = (HBRUSH)SelectObject(hdc, hbrColor);
PatBlt(hdc, rcT.left, rcT.top, cx, rcT.bottom - rcT.top, PATCOPY);
rcT.left += cx;
PatBlt(hdc, rcT.left, rcT.top, rcT.right - rcT.left, cy, PATCOPY);
rcT.top += cy;
rcT.right -= cx;
PatBlt(hdc, rcT.right, rcT.top, cx, rcT.bottom - rcT.top, PATCOPY);
rcT.bottom -= cy;
PatBlt(hdc, rcT.left, rcT.bottom, rcT.right - rcT.left, cy, PATCOPY);
hbr = (HBRUSH)SelectObject(hdc, hbr);
*prc = rcT;
}
/*
** draw a cyBorder band of 3DFACE at the bottom of the given rectangle.
** also, adjust the rectangle accordingly.
*/
void MyDrawBorderBelow(HDC hdc, LPRECT prc)
{
int i;
i = prc->top;
prc->top = prc->bottom - cyBorder;
FillRect(hdc, prc, g_brushes[COLOR_3DFACE]);
prc->top = i;
prc->bottom -= cyBorder;
}
/*
** draw a full window caption with system menu, minimize button,
** maximize button, and text.
***/
void DrawFullCaption(HDC hdc, LPRECT prc, LPTSTR lpszTitle, UINT flags)
{
int iRight;
int iFont;
SaveDC(hdc);
// special case gross for small caption that already drew on bottom
if (!(flags & DC_SMALLCAP))
MyDrawBorderBelow(hdc, prc);
iRight = prc->right;
prc->right = prc->left + cxSize;
DrawFrameControl(hdc, prc, DFC_CAPTION, DFCS_CAPTIONCLOSE);
prc->left = prc->right;
prc->right = iRight - 2*cxSize;
iFont = flags & DC_SMALLCAP ? FONT_SMCAPTION : FONT_CAPTION;
DrawCaptionTemp(NULL, hdc, prc, g_fonts[iFont].hfont, NULL, lpszTitle, flags | DC_ICON | DC_TEXT);
prc->left = prc->right;
prc->right = prc->left + cxSize;
DrawFrameControl(hdc, prc, DFC_CAPTION, DFCS_CAPTIONMIN);
prc->left = prc->right;
prc->right = prc->left + cxSize;
DrawFrameControl(hdc, prc, DFC_CAPTION, DFCS_CAPTIONMAX);
RestoreDC(hdc, -1);
}