1
0
mirror of https://github.com/narzoul/DDrawCompat synced 2024-12-30 08:55:36 +01:00
DDrawCompat/DDrawCompat/CompatGdiTitleBar.cpp

158 lines
3.6 KiB
C++

#include "CompatGdi.h"
#include "CompatGdiTitleBar.h"
#include "CompatPrimarySurface.h"
#include "Hook.h"
namespace
{
enum TitleBarInfoIndex
{
TBII_TITLEBAR = 0,
TBII_MINIMIZE_BUTTON = 2,
TBII_MAXIMIZE_BUTTON = 3,
TBII_HELP_BUTTON = 4,
TBII_CLOSE_BUTTON = 5
};
}
namespace CompatGdi
{
TitleBar::TitleBar(HWND hwnd, HDC compatDc) :
m_hwnd(hwnd), m_compatDc(compatDc), m_buttonWidth(0), m_buttonHeight(0), m_tbi(),
m_windowRect(), m_hasIcon(false), m_hasTitleBar(false)
{
m_hasTitleBar = 0 != (GetWindowLongPtr(hwnd, GWL_STYLE) & WS_CAPTION);
if (!m_hasTitleBar)
{
return;
}
m_tbi.cbSize = sizeof(m_tbi);
SendMessage(hwnd, WM_GETTITLEBARINFOEX, 0, reinterpret_cast<LPARAM>(&m_tbi));
m_hasTitleBar = !IsRectEmpty(&m_tbi.rcTitleBar);
if (!m_hasTitleBar)
{
return;
}
POINT origin = {};
ClientToScreen(hwnd, &origin);
m_hasIcon = m_tbi.rcTitleBar.left > origin.x;
m_tbi.rcTitleBar.left = origin.x;
m_tbi.rcTitleBar.bottom -= 1;
GetWindowRect(hwnd, &m_windowRect);
OffsetRect(&m_tbi.rcTitleBar, -m_windowRect.left, -m_windowRect.top);
m_buttonWidth = GetSystemMetrics(SM_CXSIZE) - 2;
m_buttonHeight = GetSystemMetrics(SM_CYSIZE) - 4;
for (std::size_t i = TBII_MINIMIZE_BUTTON; i <= TBII_CLOSE_BUTTON; ++i)
{
if (isVisible(i))
{
OffsetRect(&m_tbi.rgrect[i], -m_windowRect.left, -m_windowRect.top);
adjustButtonSize(m_tbi.rgrect[i]);
}
}
}
void TitleBar::adjustButtonSize(RECT& rect) const
{
rect.left += (rect.right - rect.left - m_buttonWidth) / 2;
rect.top += (rect.bottom - rect.top - m_buttonHeight) / 2;
rect.right = rect.left + m_buttonWidth;
rect.bottom = rect.top + m_buttonHeight;
}
void TitleBar::drawAll() const
{
drawCaption();
drawButtons();
}
void TitleBar::drawButtons() const
{
if (!m_hasTitleBar)
{
return;
}
drawButton(TBII_MINIMIZE_BUTTON, DFCS_CAPTIONMIN);
drawButton(TBII_MAXIMIZE_BUTTON, IsZoomed(m_hwnd) ? DFCS_CAPTIONRESTORE : DFCS_CAPTIONMAX);
drawButton(TBII_HELP_BUTTON, DFCS_CAPTIONHELP);
drawButton(TBII_CLOSE_BUTTON, DFCS_CAPTIONCLOSE);
}
void TitleBar::drawCaption() const
{
if (!m_hasTitleBar)
{
return;
}
UINT flags = 0;
if (GetActiveWindow() == m_hwnd)
{
flags |= DC_ACTIVE;
}
if (CompatPrimarySurface::pixelFormat.dwRGBBitCount > 8)
{
flags |= DC_GRADIENT;
}
RECT clipRect = m_tbi.rcTitleBar;
OffsetRect(&clipRect, m_windowRect.left, m_windowRect.top);
HRGN clipRgn = CreateRectRgnIndirect(&clipRect);
SelectClipRgn(m_compatDc, clipRgn);
DeleteObject(clipRgn);
RECT textRect = m_tbi.rcTitleBar;
if (m_hasIcon)
{
CALL_ORIG_FUNC(DrawCaption)(m_hwnd, m_compatDc, &m_tbi.rcTitleBar, DC_ICON | flags);
textRect.left -= 1;
}
textRect.top -= 1;
CALL_ORIG_FUNC(DrawCaption)(m_hwnd, m_compatDc, &textRect, DC_TEXT | flags);
SelectClipRgn(m_compatDc, nullptr);
}
void TitleBar::drawButton(std::size_t tbiIndex, UINT dfcState) const
{
if (!isVisible(tbiIndex))
{
return;
}
DWORD stateFlags = 0;
if (m_tbi.rgstate[tbiIndex] & STATE_SYSTEM_UNAVAILABLE)
{
stateFlags |= DFCS_INACTIVE;
}
else if (m_tbi.rgstate[tbiIndex] & STATE_SYSTEM_PRESSED)
{
stateFlags |= DFCS_PUSHED;
}
RECT rect = m_tbi.rgrect[tbiIndex];
CALL_ORIG_FUNC(DrawFrameControl)(m_compatDc, &rect, DFC_CAPTION, dfcState | stateFlags);
}
void TitleBar::excludeFromClipRegion() const
{
if (m_hasTitleBar)
{
const RECT& r = m_tbi.rcTitleBar;
ExcludeClipRect(m_compatDc, r.left, r.top, r.right, r.bottom);
}
}
bool TitleBar::isVisible(std::size_t tbiIndex) const
{
return !(m_tbi.rgstate[tbiIndex] & (STATE_SYSTEM_INVISIBLE | STATE_SYSTEM_OFFSCREEN));
}
}