3004 lines
74 KiB
C++
3004 lines
74 KiB
C++
|
|
|
|
// Microsoft Windows NT Security
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1999
|
|
|
|
// File: acuictl.cpp
|
|
|
|
// Contents: Authenticode Default UI controls
|
|
|
|
// History: 12-May-97 kirtd Created
|
|
|
|
|
|
#include <stdpch.h>
|
|
|
|
#include <richedit.h>
|
|
|
|
#include "secauth.h"
|
|
|
|
IACUIControl::IACUIControl(CInvokeInfoHelper& riih) : m_riih( riih ),
|
|
m_hrInvokeResult( TRUST_E_SUBJECT_NOT_TRUSTED )
|
|
{
|
|
m_hrInvokeResult = TRUST_E_SUBJECT_NOT_TRUSTED;
|
|
|
|
|
|
m_pszCopyActionText = NULL;
|
|
m_pszCopyActionTextNoTS = NULL;
|
|
m_pszCopyActionTextNotSigned = NULL;
|
|
|
|
if ((riih.ProviderData()) &&
|
|
(riih.ProviderData()->psPfns) &&
|
|
(riih.ProviderData()->psPfns->psUIpfns) &&
|
|
(riih.ProviderData()->psPfns->psUIpfns->psUIData))
|
|
{
|
|
if (_ISINSTRUCT(CRYPT_PROVUI_DATA,
|
|
riih.ProviderData()->psPfns->psUIpfns->psUIData->cbStruct,
|
|
pCopyActionTextNotSigned))
|
|
{
|
|
this->LoadActionText(&m_pszCopyActionText,
|
|
riih.ProviderData()->psPfns->psUIpfns->psUIData->pCopyActionText, IDS_ACTIONSIGNED);
|
|
this->LoadActionText(&m_pszCopyActionTextNoTS,
|
|
riih.ProviderData()->psPfns->psUIpfns->psUIData->pCopyActionTextNoTS, IDS_ACTIONSIGNED_NODATE);
|
|
this->LoadActionText(&m_pszCopyActionTextNotSigned,
|
|
riih.ProviderData()->psPfns->psUIpfns->psUIData->pCopyActionTextNotSigned, IDS_ACTIONNOTSIGNED);
|
|
}
|
|
}
|
|
|
|
if (!(m_pszCopyActionText))
|
|
{
|
|
this->LoadActionText(&m_pszCopyActionText, NULL, IDS_ACTIONSIGNED);
|
|
}
|
|
|
|
if (!(m_pszCopyActionTextNoTS))
|
|
{
|
|
this->LoadActionText(&m_pszCopyActionTextNoTS, NULL, IDS_ACTIONSIGNED_NODATE);
|
|
}
|
|
|
|
if (!(m_pszCopyActionTextNotSigned))
|
|
{
|
|
this->LoadActionText(&m_pszCopyActionTextNotSigned, NULL, IDS_ACTIONNOTSIGNED);
|
|
}
|
|
}
|
|
|
|
void IACUIControl::LoadActionText(WCHAR **ppszRet, WCHAR *pwszIn, DWORD dwDefId)
|
|
{
|
|
WCHAR sz[MAX_PATH];
|
|
|
|
*ppszRet = NULL;
|
|
sz[0] = NULL;
|
|
|
|
if ((pwszIn) && (*pwszIn))
|
|
{
|
|
sz[0] = NULL;
|
|
if (wcslen(pwszIn) < MAX_PATH)
|
|
{
|
|
wcscpy(&sz[0], pwszIn);
|
|
}
|
|
|
|
if (sz[0])
|
|
{
|
|
if (*ppszRet = new WCHAR[wcslen(&sz[0]) + 1])
|
|
{
|
|
wcscpy(*ppszRet, &sz[0]);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (!(sz[0]))
|
|
{
|
|
sz[0] = NULL;
|
|
LoadStringU(g_hModule, dwDefId, &sz[0], MAX_PATH);
|
|
|
|
if (sz[0])
|
|
{
|
|
if (*ppszRet = new WCHAR[wcslen(&sz[0]) + 1])
|
|
{
|
|
wcscpy(*ppszRet, &sz[0]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
IACUIControl::~IACUIControl ()
|
|
{
|
|
DELETE_OBJECT(m_pszCopyActionText);
|
|
DELETE_OBJECT(m_pszCopyActionTextNoTS);
|
|
DELETE_OBJECT(m_pszCopyActionTextNotSigned);
|
|
}
|
|
|
|
void IACUIControl::SetupButtons(HWND hWnd)
|
|
{
|
|
char sz[MAX_PATH];
|
|
|
|
if ((m_riih.ProviderData()) &&
|
|
(m_riih.ProviderData()->psPfns) &&
|
|
(m_riih.ProviderData()->psPfns->psUIpfns) &&
|
|
(m_riih.ProviderData()->psPfns->psUIpfns->psUIData))
|
|
{
|
|
if (m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pYesButtonText)
|
|
{
|
|
if (!(m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pYesButtonText[0]))
|
|
{
|
|
ShowWindow(GetDlgItem(hWnd, IDYES), SW_HIDE);
|
|
}
|
|
else
|
|
{
|
|
SetWindowTextU(GetDlgItem(hWnd, IDYES), m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pYesButtonText);
|
|
}
|
|
}
|
|
|
|
if (m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pNoButtonText)
|
|
{
|
|
if (!(m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pNoButtonText[0]))
|
|
{
|
|
ShowWindow(GetDlgItem(hWnd, IDNO), SW_HIDE);
|
|
}
|
|
else
|
|
{
|
|
SetWindowTextU(GetDlgItem(hWnd, IDNO), m_riih.ProviderData()->psPfns->psUIpfns->psUIData->pNoButtonText);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Member: IACUIControl::OnUIMessage, public
|
|
|
|
// Synopsis: responds to UI messages
|
|
|
|
// Arguments: [hwnd] -- window
|
|
// [uMsg] -- message id
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if message processing should continue, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
IACUIControl::OnUIMessage (
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
switch ( uMsg )
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
BOOL fReturn;
|
|
HICON hIcon;
|
|
|
|
fReturn = OnInitDialog(hwnd, wParam, lParam);
|
|
|
|
ACUICenterWindow(hwnd);
|
|
|
|
// hIcon = LoadIcon((HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE), MAKEINTRESOURCE(IDI_LOCK));
|
|
|
|
// dwOrigIcon = SetClassLongPtr(hwnd, GCLP_HICON,
|
|
// (LONG_PTR)LoadIcon((HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE),
|
|
// MAKEINTRESOURCE(IDI_LOCK)));
|
|
|
|
// PostMessage(hwnd, WM_SETICON, ICON_BIG, (LPARAM)hIcon);
|
|
// PostMessage(hwnd, WM_SETICON, ICON_SMALL, (LPARAM)hIcon);
|
|
|
|
return( fReturn );
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
{
|
|
WORD wNotifyCode = HIWORD(wParam);
|
|
WORD wId = LOWORD(wParam);
|
|
HWND hwndControl = (HWND)lParam;
|
|
|
|
if ( wNotifyCode == BN_CLICKED )
|
|
{
|
|
if ( wId == IDYES )
|
|
{
|
|
return( OnYes(hwnd) );
|
|
}
|
|
else if ( wId == IDNO )
|
|
{
|
|
return( OnNo(hwnd) );
|
|
}
|
|
else if ( wId == IDMORE )
|
|
{
|
|
return( OnMore(hwnd) );
|
|
}
|
|
}
|
|
|
|
return( FALSE );
|
|
}
|
|
break;
|
|
|
|
case WM_CLOSE:
|
|
return( OnNo(hwnd) );
|
|
break;
|
|
|
|
default:
|
|
return( FALSE );
|
|
}
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::CVerifiedTrustUI, public
|
|
|
|
// Synopsis: Constructor
|
|
|
|
// Arguments: [riih] -- invoke info helper reference
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CVerifiedTrustUI::CVerifiedTrustUI (CInvokeInfoHelper& riih, HRESULT& rhr)
|
|
: IACUIControl( riih ),
|
|
m_pszInstallAndRun( NULL ),
|
|
m_pszAuthenticity( NULL ),
|
|
m_pszCaution( NULL ),
|
|
m_pszPersonalTrust( NULL )
|
|
{
|
|
DWORD_PTR aMessageArgument[3];
|
|
|
|
|
|
// Initialize the hot-link subclass data
|
|
|
|
|
|
m_lsdPublisher.uId = IDC_PUBLISHER;
|
|
m_lsdPublisher.hwndParent = NULL;
|
|
m_lsdPublisher.wpPrev = (WNDPROC)NULL;
|
|
m_lsdPublisher.pvData = (LPVOID)&riih;
|
|
m_lsdPublisher.uToolTipText = IDS_CLICKHEREFORCERT;
|
|
|
|
m_lsdOpusInfo.uId = IDC_INSTALLANDRUN;
|
|
m_lsdOpusInfo.hwndParent = NULL;
|
|
m_lsdOpusInfo.wpPrev = (WNDPROC)NULL;
|
|
m_lsdOpusInfo.pvData = &riih;
|
|
m_lsdOpusInfo.uToolTipText = (DWORD_PTR)riih.ControlWebPage();
|
|
|
|
m_lsdCA.uId = IDC_AUTHENTICITY;
|
|
m_lsdCA.hwndParent = NULL;
|
|
m_lsdCA.wpPrev = (WNDPROC)NULL;
|
|
m_lsdCA.pvData = &riih;
|
|
m_lsdCA.uToolTipText = (DWORD_PTR)riih.CAWebPage(); // IDS_CLICKHEREFORCAINFO;
|
|
|
|
m_lsdAdvanced.uId = IDC_ADVANCED;
|
|
m_lsdAdvanced.hwndParent = NULL;
|
|
m_lsdAdvanced.wpPrev = (WNDPROC)NULL;
|
|
m_lsdAdvanced.pvData = &riih;
|
|
m_lsdAdvanced.uToolTipText = IDS_CLICKHEREFORADVANCED;
|
|
|
|
|
|
|
|
// Format the install and run string
|
|
|
|
|
|
aMessageArgument[2] = NULL;
|
|
|
|
if (m_riih.CertTimestamp())
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_pszCopyActionText;
|
|
aMessageArgument[1] = (DWORD_PTR)m_riih.Subject();
|
|
aMessageArgument[2] = (DWORD_PTR)m_riih.CertTimestamp();
|
|
}
|
|
else
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_pszCopyActionTextNoTS;
|
|
aMessageArgument[1] = (DWORD_PTR)m_riih.Subject();
|
|
aMessageArgument[2] = NULL;
|
|
}
|
|
|
|
rhr = FormatACUIResourceString(0, aMessageArgument, &m_pszInstallAndRun);
|
|
|
|
|
|
// Format the authenticity string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_riih.PublisherCertIssuer();
|
|
|
|
rhr = FormatACUIResourceString(
|
|
IDS_AUTHENTICITY,
|
|
aMessageArgument,
|
|
&m_pszAuthenticity
|
|
);
|
|
}
|
|
|
|
|
|
// Get the publisher as a message argument
|
|
|
|
|
|
aMessageArgument[0] = (DWORD_PTR)m_riih.Publisher();
|
|
|
|
|
|
// Format the caution string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
rhr = FormatACUIResourceString(
|
|
IDS_CAUTION,
|
|
aMessageArgument,
|
|
&m_pszCaution
|
|
);
|
|
}
|
|
|
|
|
|
// Format the personal trust string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
rhr = FormatACUIResourceString(
|
|
IDS_PERSONALTRUST,
|
|
aMessageArgument,
|
|
&m_pszPersonalTrust
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::~CVerifiedTrustUI, public
|
|
|
|
// Synopsis: Destructor
|
|
|
|
// Arguments: (none)
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CVerifiedTrustUI::~CVerifiedTrustUI ()
|
|
{
|
|
DELETE_OBJECT(m_pszInstallAndRun);
|
|
DELETE_OBJECT(m_pszAuthenticity);
|
|
DELETE_OBJECT(m_pszCaution);
|
|
DELETE_OBJECT(m_pszPersonalTrust);
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::InvokeUI, public
|
|
|
|
// Synopsis: invoke the UI
|
|
|
|
// Arguments: [hDisplay] -- parent window
|
|
|
|
// Returns: S_OK, user trusts the subject
|
|
// TRUST_E_SUBJECT_NOT_TRUSTED, user does NOT trust the subject
|
|
// Any other valid HRESULT
|
|
|
|
// Notes:
|
|
|
|
|
|
HRESULT
|
|
CVerifiedTrustUI::InvokeUI (HWND hDisplay)
|
|
{
|
|
|
|
// Bring up the dialog
|
|
|
|
|
|
if ( DialogBoxParamU(
|
|
g_hModule,
|
|
(LPWSTR) MAKEINTRESOURCE(IDD_DIALOG1_VERIFIED),
|
|
hDisplay,
|
|
ACUIMessageProc,
|
|
(LPARAM)this
|
|
) == -1 )
|
|
{
|
|
return( HRESULT_FROM_WIN32(GetLastError()) );
|
|
}
|
|
|
|
|
|
|
|
// The result has been stored as a member
|
|
|
|
|
|
return( m_hrInvokeResult );
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::OnInitDialog, public
|
|
|
|
// Synopsis: dialog initialization
|
|
|
|
// Arguments: [hwnd] -- dialog window
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if successful init, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CVerifiedTrustUI::OnInitDialog(HWND hwnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
WCHAR psz[MAX_LOADSTRING_BUFFER];
|
|
HWND hControl;
|
|
int deltavpos = 0;
|
|
int deltaheight;
|
|
int bmptosep;
|
|
int septodlg;
|
|
int savevpos;
|
|
int hkcharpos;
|
|
RECT rect;
|
|
|
|
|
|
// Setup the publisher link subclass data parent window
|
|
|
|
|
|
m_lsdPublisher.hwndParent = hwnd;
|
|
m_lsdOpusInfo.hwndParent = hwnd;
|
|
m_lsdCA.hwndParent = hwnd;
|
|
m_lsdAdvanced.hwndParent = hwnd;
|
|
|
|
|
|
// Render the install and run string
|
|
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_INSTALLANDRUN,
|
|
IDC_PUBLISHER,
|
|
m_pszInstallAndRun,
|
|
deltavpos,
|
|
(m_riih.ControlWebPage()) ? TRUE : FALSE,
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdOpusInfo,
|
|
0,
|
|
m_riih.Subject());
|
|
|
|
|
|
|
|
// Render the publisher, give it a "link" look and feel if it is a known
|
|
// publisher
|
|
|
|
|
|
|
|
// if there was a test cert in the chain, add it to the text...
|
|
|
|
if (m_riih.TestCertInChain())
|
|
{
|
|
WCHAR *pszCombine;
|
|
|
|
pszCombine = new WCHAR[wcslen(m_riih.Publisher()) + wcslen(m_riih.TestCertInChain()) + 3];
|
|
|
|
if (pszCombine != NULL)
|
|
{
|
|
wcscpy(pszCombine, m_riih.Publisher());
|
|
wcscat(pszCombine, L"\r\n");
|
|
wcscat(pszCombine, m_riih.TestCertInChain());
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_PUBLISHER,
|
|
IDC_AUTHENTICITY,
|
|
pszCombine,
|
|
deltavpos,
|
|
m_riih.IsKnownPublisher() &&
|
|
m_riih.IsCertViewPropertiesAvailable(),
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdPublisher,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
delete[] pszCombine;
|
|
}
|
|
|
|
if (LoadStringU(g_hModule, IDS_TESTCERTTITLE, psz, MAX_LOADSTRING_BUFFER) != 0)
|
|
{
|
|
int wtlen;
|
|
|
|
wtlen = wcslen(psz) + GetWindowTextLength(hwnd);
|
|
pszCombine = new WCHAR[wtlen + 1];
|
|
|
|
if (pszCombine != NULL)
|
|
{
|
|
GetWindowTextU(hwnd, pszCombine, wtlen + 1);
|
|
wcscat(pszCombine, psz);
|
|
SetWindowTextU(hwnd, pszCombine);
|
|
|
|
delete[] pszCombine;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_PUBLISHER,
|
|
IDC_AUTHENTICITY,
|
|
m_riih.Publisher(),
|
|
deltavpos,
|
|
m_riih.IsKnownPublisher() &&
|
|
m_riih.IsCertViewPropertiesAvailable(),
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdPublisher,
|
|
0,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
|
|
// Render the authenticity statement
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_AUTHENTICITY,
|
|
IDC_CAUTION,
|
|
m_pszAuthenticity,
|
|
deltavpos,
|
|
(m_riih.CAWebPage()) ? TRUE : FALSE,
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdCA,
|
|
0,
|
|
m_riih.PublisherCertIssuer());
|
|
|
|
|
|
|
|
// Render the caution statement
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_CAUTION,
|
|
IDC_ADVANCED,
|
|
m_pszCaution,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
|
|
// Render the advanced string
|
|
|
|
if ((m_riih.AdvancedLink()) &&
|
|
(m_riih.ProviderData()->psPfns->psUIpfns->pfnOnAdvancedClick))
|
|
{
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_ADVANCED,
|
|
IDC_PERSONALTRUST,
|
|
m_riih.AdvancedLink(),
|
|
deltavpos,
|
|
TRUE,
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdAdvanced,
|
|
0,
|
|
NULL
|
|
);
|
|
}
|
|
else
|
|
{
|
|
ShowWindow(GetDlgItem(hwnd, IDC_ADVANCED), SW_HIDE);
|
|
}
|
|
|
|
|
|
// Calculate the distances from the bottom of the bitmap to the top
|
|
// of the separator and from the bottom of the separator to the bottom
|
|
// of the dialog
|
|
|
|
|
|
bmptosep = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_VERBMP,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
septodlg = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
|
|
// Rebase the check box and render the personal trust statement or hide
|
|
// them the publisher is not known
|
|
|
|
|
|
if ( m_riih.IsKnownPublisher() == TRUE )
|
|
{
|
|
hControl = GetDlgItem(hwnd, IDC_PTCHECK);
|
|
|
|
RebaseControlVertical(
|
|
hwnd,
|
|
hControl,
|
|
NULL,
|
|
FALSE,
|
|
deltavpos,
|
|
0,
|
|
bmptosep,
|
|
&deltaheight
|
|
);
|
|
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Find the hotkey character position for the personal trust
|
|
// check box
|
|
|
|
|
|
hkcharpos = GetHotKeyCharPosition(GetDlgItem(hwnd, IDC_PTCHECK));
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_PERSONALTRUST,
|
|
IDC_SEPARATORLINE,
|
|
m_pszPersonalTrust,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
bmptosep,
|
|
NULL
|
|
);
|
|
|
|
if ( hkcharpos != 0 )
|
|
{
|
|
FormatHotKeyOnEditControl(
|
|
GetDlgItem(hwnd, IDC_PERSONALTRUST),
|
|
hkcharpos
|
|
);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ShowWindow(GetDlgItem(hwnd, IDC_PTCHECK), SW_HIDE);
|
|
ShowWindow(GetDlgItem(hwnd, IDC_PERSONALTRUST), SW_HIDE);
|
|
}
|
|
|
|
|
|
|
|
// Rebase the static line
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_SEPARATORLINE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Rebase the buttons
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDYES);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDMORE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Resize the bitmap and the dialog rectangle if necessary
|
|
|
|
|
|
if ( deltavpos > 0 )
|
|
{
|
|
int cyupd;
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_VERBMP);
|
|
GetWindowRect(hControl, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_VERBMP,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd -= bmptosep;
|
|
|
|
SetWindowPos(
|
|
hControl,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd = septodlg - cyupd;
|
|
|
|
SetWindowPos(
|
|
hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
}
|
|
|
|
|
|
// check for overridden button texts
|
|
|
|
this->SetupButtons(hwnd);
|
|
|
|
|
|
// Set focus to appropriate control
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
::PostMessage(hwnd, WM_NEXTDLGCTL, (WPARAM) hControl, (LPARAM) MAKEWORD(TRUE, 0));
|
|
|
|
return( FALSE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::OnYes, public
|
|
|
|
// Synopsis: process IDYES button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CVerifiedTrustUI::OnYes (HWND hwnd)
|
|
{
|
|
|
|
// Set the invoke result
|
|
|
|
|
|
m_hrInvokeResult = S_OK;
|
|
|
|
|
|
// Add the publisher to the trust database
|
|
|
|
// BUGBUG: What if this fails??
|
|
|
|
|
|
if ( SendDlgItemMessage(
|
|
hwnd,
|
|
IDC_PTCHECK,
|
|
BM_GETCHECK,
|
|
0,
|
|
0
|
|
) == BST_CHECKED )
|
|
{
|
|
m_riih.AddPublisherToPersonalTrust();
|
|
}
|
|
|
|
|
|
// End the dialog processing
|
|
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::OnNo, public
|
|
|
|
// Synopsis: process IDNO button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CVerifiedTrustUI::OnNo (HWND hwnd)
|
|
{
|
|
m_hrInvokeResult = TRUST_E_SUBJECT_NOT_TRUSTED;
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CVerifiedTrustUI::OnMore, public
|
|
|
|
// Synopsis: process the IDMORE button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CVerifiedTrustUI::OnMore (HWND hwnd)
|
|
{
|
|
WinHelp(hwnd, "SECAUTH.HLP", HELP_CONTEXT, IDH_SECAUTH_SIGNED);
|
|
|
|
// ACUIViewHTMLHelpTopic(hwnd, "sec_signed.htm");
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::CUnverifiedTrustUI, public
|
|
|
|
// Synopsis: Constructor
|
|
|
|
// Arguments: [riih] -- invoke info helper reference
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CUnverifiedTrustUI::CUnverifiedTrustUI (CInvokeInfoHelper& riih, HRESULT& rhr)
|
|
: IACUIControl( riih ),
|
|
m_pszNoAuthenticity( NULL ),
|
|
m_pszProblemsBelow( NULL ),
|
|
m_pszInstallAndRun3( NULL )
|
|
{
|
|
DWORD_PTR aMessageArgument[3];
|
|
|
|
|
|
// Initialize the publisher link subclass data
|
|
|
|
|
|
m_lsdPublisher.uId = IDC_PUBLISHER;
|
|
m_lsdPublisher.hwndParent = NULL;
|
|
m_lsdPublisher.wpPrev = (WNDPROC)NULL;
|
|
m_lsdPublisher.pvData = (LPVOID)&riih;
|
|
m_lsdPublisher.uToolTipText = IDS_CLICKHEREFORCERT;
|
|
|
|
m_lsdOpusInfo.uId = IDC_INSTALLANDRUN;
|
|
m_lsdOpusInfo.hwndParent = NULL;
|
|
m_lsdOpusInfo.wpPrev = (WNDPROC)NULL;
|
|
m_lsdOpusInfo.pvData = &riih;
|
|
m_lsdOpusInfo.uToolTipText = (DWORD_PTR)riih.ControlWebPage(); // IDS_CLICKHEREFOROPUSINFO;
|
|
|
|
m_lsdAdvanced.uId = IDC_ADVANCED;
|
|
m_lsdAdvanced.hwndParent = NULL;
|
|
m_lsdAdvanced.wpPrev = (WNDPROC)NULL;
|
|
m_lsdAdvanced.pvData = &riih;
|
|
m_lsdAdvanced.uToolTipText = IDS_CLICKHEREFORADVANCED;
|
|
|
|
|
|
|
|
// Format the no authenticity string
|
|
|
|
|
|
rhr = FormatACUIResourceString(
|
|
IDS_NOAUTHENTICITY,
|
|
NULL,
|
|
&m_pszNoAuthenticity
|
|
);
|
|
|
|
|
|
// Format the problems below string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_riih.ErrorStatement();
|
|
|
|
rhr = FormatACUIResourceString(
|
|
IDS_PROBLEMSBELOW,
|
|
aMessageArgument,
|
|
&m_pszProblemsBelow
|
|
);
|
|
}
|
|
|
|
|
|
// Format the install and run string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
if (m_riih.CertTimestamp())
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_pszCopyActionText;
|
|
aMessageArgument[1] = (DWORD_PTR)m_riih.Subject();
|
|
aMessageArgument[2] = (DWORD_PTR)m_riih.CertTimestamp();
|
|
}
|
|
else
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_pszCopyActionTextNoTS;
|
|
aMessageArgument[1] = (DWORD_PTR)m_riih.Subject();
|
|
aMessageArgument[2] = NULL;
|
|
}
|
|
|
|
rhr = FormatACUIResourceString(0, aMessageArgument, &m_pszInstallAndRun3);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::~CUnverifiedTrustUI, public
|
|
|
|
// Synopsis: Destructor
|
|
|
|
// Arguments: (none)
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CUnverifiedTrustUI::~CUnverifiedTrustUI ()
|
|
{
|
|
DELETE_OBJECT(m_pszNoAuthenticity);
|
|
DELETE_OBJECT(m_pszProblemsBelow);
|
|
DELETE_OBJECT(m_pszInstallAndRun3);
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::InvokeUI, public
|
|
|
|
// Synopsis: invoke the UI
|
|
|
|
// Arguments: [hDisplay] -- parent window
|
|
|
|
// Returns: S_OK, user trusts the subject
|
|
// TRUST_E_SUBJECT_NOT_TRUSTED, user does NOT trust the subject
|
|
// Any other valid HRESULT
|
|
|
|
// Notes:
|
|
|
|
|
|
HRESULT
|
|
CUnverifiedTrustUI::InvokeUI (HWND hDisplay)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
// Bring up the dialog
|
|
|
|
|
|
if ( DialogBoxParamU(
|
|
g_hModule,
|
|
(LPWSTR) MAKEINTRESOURCE(IDD_DIALOG2_UNVERIFIED),
|
|
hDisplay,
|
|
ACUIMessageProc,
|
|
(LPARAM)this
|
|
) == -1 )
|
|
{
|
|
return( HRESULT_FROM_WIN32(GetLastError()) );
|
|
}
|
|
|
|
|
|
// The result has been stored as a member
|
|
|
|
|
|
return( m_hrInvokeResult );
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::OnInitDialog, public
|
|
|
|
// Synopsis: dialog initialization
|
|
|
|
// Arguments: [hwnd] -- dialog window
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if successful init, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CUnverifiedTrustUI::OnInitDialog(HWND hwnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
HWND hControl;
|
|
int deltavpos = 0;
|
|
int deltaheight;
|
|
int bmptosep;
|
|
int septodlg;
|
|
RECT rect;
|
|
|
|
|
|
// Setup the publisher link subclass data parent window
|
|
|
|
|
|
m_lsdPublisher.hwndParent = hwnd;
|
|
m_lsdOpusInfo.hwndParent = hwnd;
|
|
m_lsdAdvanced.hwndParent = hwnd;
|
|
|
|
|
|
|
|
// Render the no authenticity statement
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_NOAUTHENTICITY,
|
|
IDC_PROBLEMSBELOW,
|
|
m_pszNoAuthenticity,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
|
|
// Render the problems below string
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_PROBLEMSBELOW,
|
|
IDC_INSTALLANDRUN3,
|
|
m_pszProblemsBelow,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
|
|
// Render the install and run string
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_INSTALLANDRUN3,
|
|
IDC_PUBLISHER2,
|
|
m_pszInstallAndRun3,
|
|
deltavpos,
|
|
(m_riih.ControlWebPage()) ? TRUE : FALSE,
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdOpusInfo,
|
|
0,
|
|
m_riih.Subject());
|
|
|
|
|
|
|
|
// Calculate the distances from the bottom of the bitmap to the top
|
|
// of the separator and from the bottom of the separator to the bottom
|
|
// of the dialog
|
|
|
|
|
|
bmptosep = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_NOVERBMP2,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
septodlg = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
|
|
// Render the publisher, give it a "link" look and feel
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_PUBLISHER2,
|
|
IDC_ADVANCED,
|
|
m_riih.Publisher(),
|
|
deltavpos,
|
|
m_riih.IsKnownPublisher() &&
|
|
m_riih.IsCertViewPropertiesAvailable(),
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdPublisher,
|
|
bmptosep,
|
|
NULL
|
|
);
|
|
|
|
if ((m_riih.AdvancedLink()) &&
|
|
(m_riih.ProviderData()->psPfns->psUIpfns->pfnOnAdvancedClick))
|
|
{
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_ADVANCED,
|
|
IDC_SEPARATORLINE,
|
|
m_riih.AdvancedLink(),
|
|
deltavpos,
|
|
TRUE,
|
|
(WNDPROC)ACUILinkSubclass,
|
|
&m_lsdAdvanced,
|
|
0,
|
|
NULL
|
|
);
|
|
}
|
|
else
|
|
{
|
|
ShowWindow(GetDlgItem(hwnd, IDC_ADVANCED), SW_HIDE);
|
|
}
|
|
|
|
|
|
// Rebase the static line
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_SEPARATORLINE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Rebase the buttons
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDYES);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDMORE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Resize the bitmap and the dialog rectangle if necessary
|
|
|
|
|
|
if ( deltavpos > 0 )
|
|
{
|
|
int cyupd;
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_NOVERBMP2);
|
|
GetWindowRect(hControl, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_NOVERBMP2,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd -= bmptosep;
|
|
|
|
SetWindowPos(
|
|
hControl,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd = septodlg - cyupd;
|
|
|
|
SetWindowPos(
|
|
hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
}
|
|
|
|
|
|
// check for overridden button texts
|
|
|
|
this->SetupButtons(hwnd);
|
|
|
|
|
|
// Set focus to appropriate control
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
::PostMessage(hwnd, WM_NEXTDLGCTL, (WPARAM) hControl, (LPARAM) MAKEWORD(TRUE, 0));
|
|
|
|
return( FALSE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::OnYes, public
|
|
|
|
// Synopsis: process IDYES button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CUnverifiedTrustUI::OnYes (HWND hwnd)
|
|
{
|
|
m_hrInvokeResult = S_OK;
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::OnNo, public
|
|
|
|
// Synopsis: process IDNO button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CUnverifiedTrustUI::OnNo (HWND hwnd)
|
|
{
|
|
m_hrInvokeResult = TRUST_E_SUBJECT_NOT_TRUSTED;
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CUnverifiedTrustUI::OnMore, public
|
|
|
|
// Synopsis: process the IDMORE button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CUnverifiedTrustUI::OnMore (HWND hwnd)
|
|
{
|
|
WinHelp(hwnd, "SECAUTH.HLP", HELP_CONTEXT, IDH_SECAUTH_SIGNED_N_INVALID);
|
|
|
|
// ACUIViewHTMLHelpTopic(hwnd, "sec_signed_n_invalid.htm");
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::CNoSignatureUI, public
|
|
|
|
// Synopsis: Constructor
|
|
|
|
// Arguments: [riih] -- invoke info helper
|
|
// [rhr] -- result code reference
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CNoSignatureUI::CNoSignatureUI (CInvokeInfoHelper& riih, HRESULT& rhr)
|
|
: IACUIControl( riih ),
|
|
m_pszInstallAndRun2( NULL ),
|
|
m_pszNoPublisherFound( NULL )
|
|
{
|
|
DWORD_PTR aMessageArgument[2];
|
|
|
|
|
|
// Format the install and run string
|
|
|
|
|
|
aMessageArgument[0] = (DWORD_PTR)m_pszCopyActionTextNotSigned;
|
|
aMessageArgument[1] = (DWORD_PTR)m_riih.Subject();
|
|
|
|
rhr = FormatACUIResourceString(0, aMessageArgument, &m_pszInstallAndRun2);
|
|
|
|
|
|
// Format the no publisher found string
|
|
|
|
|
|
if ( rhr == S_OK )
|
|
{
|
|
aMessageArgument[0] = (DWORD_PTR)m_riih.ErrorStatement();
|
|
|
|
rhr = FormatACUIResourceString(
|
|
IDS_NOPUBLISHERFOUND,
|
|
aMessageArgument,
|
|
&m_pszNoPublisherFound
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::~CNoSignatureUI, public
|
|
|
|
// Synopsis: Destructor
|
|
|
|
// Arguments: (none)
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
CNoSignatureUI::~CNoSignatureUI ()
|
|
{
|
|
DELETE_OBJECT(m_pszInstallAndRun2);
|
|
DELETE_OBJECT(m_pszNoPublisherFound);
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::InvokeUI, public
|
|
|
|
// Synopsis: invoke the UI
|
|
|
|
// Arguments: [hDisplay] -- parent window
|
|
|
|
// Returns: S_OK, user trusts the subject
|
|
// TRUST_E_SUBJECT_NOT_TRUSTED, user does NOT trust the subject
|
|
// Any other valid HRESULT
|
|
|
|
// Notes:
|
|
|
|
|
|
HRESULT
|
|
CNoSignatureUI::InvokeUI (HWND hDisplay)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
// Bring up the dialog
|
|
|
|
|
|
if ( DialogBoxParamU(
|
|
g_hModule,
|
|
(LPWSTR) MAKEINTRESOURCE(IDD_DIALOG3_NOSIGNATURE),
|
|
hDisplay,
|
|
ACUIMessageProc,
|
|
(LPARAM)this
|
|
) == -1 )
|
|
{
|
|
return( HRESULT_FROM_WIN32(GetLastError()) );
|
|
}
|
|
|
|
|
|
// The result has been stored as a member
|
|
|
|
|
|
return( m_hrInvokeResult );
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::OnInitDialog, public
|
|
|
|
// Synopsis: dialog initialization
|
|
|
|
// Arguments: [hwnd] -- dialog window
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if successful init, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CNoSignatureUI::OnInitDialog(HWND hwnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
HWND hControl;
|
|
int deltavpos = 0;
|
|
int deltaheight;
|
|
int bmptosep;
|
|
int septodlg;
|
|
RECT rect;
|
|
|
|
|
|
// Render the install and run string
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_INSTALLANDRUN2,
|
|
IDC_NOPUBLISHERFOUND,
|
|
m_pszInstallAndRun2,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
|
|
// Calculate the distances from the bottom of the bitmap to the top
|
|
// of the separator and from the bottom of the separator to the bottom
|
|
// of the dialog
|
|
|
|
|
|
bmptosep = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_NOVERBMP,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
septodlg = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
|
|
// Render the no publisher found statement
|
|
|
|
|
|
deltavpos = RenderACUIStringToEditControl(
|
|
hwnd,
|
|
IDC_NOPUBLISHERFOUND,
|
|
IDC_SEPARATORLINE,
|
|
m_pszNoPublisherFound,
|
|
deltavpos,
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
bmptosep,
|
|
NULL
|
|
);
|
|
|
|
|
|
// Rebase the static line
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_SEPARATORLINE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Rebase the buttons
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDYES);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
hControl = GetDlgItem(hwnd, IDMORE);
|
|
RebaseControlVertical(hwnd, hControl, NULL, FALSE, deltavpos, 0, 0, &deltaheight);
|
|
assert( deltaheight == 0 );
|
|
|
|
|
|
// Resize the bitmap and the dialog rectangle if necessary
|
|
|
|
|
|
if ( deltavpos > 0 )
|
|
{
|
|
int cyupd;
|
|
|
|
hControl = GetDlgItem(hwnd, IDC_NOVERBMP);
|
|
GetWindowRect(hControl, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistance(
|
|
hwnd,
|
|
IDC_NOVERBMP,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd -= bmptosep;
|
|
|
|
SetWindowPos(
|
|
hControl,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
|
|
cyupd = CalculateControlVerticalDistanceFromDlgBottom(
|
|
hwnd,
|
|
IDC_SEPARATORLINE
|
|
);
|
|
|
|
cyupd = septodlg - cyupd;
|
|
|
|
SetWindowPos(
|
|
hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
rect.right - rect.left,
|
|
(rect.bottom - rect.top) + cyupd,
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
}
|
|
|
|
|
|
// check for overridden button texts
|
|
|
|
this->SetupButtons(hwnd);
|
|
|
|
|
|
// Set focus to appropriate control
|
|
|
|
|
|
hControl = GetDlgItem(hwnd, IDNO);
|
|
::PostMessage(hwnd, WM_NEXTDLGCTL, (WPARAM) hControl, (LPARAM) MAKEWORD(TRUE, 0));
|
|
|
|
return( FALSE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::OnYes, public
|
|
|
|
// Synopsis: process IDYES button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CNoSignatureUI::OnYes (HWND hwnd)
|
|
{
|
|
m_hrInvokeResult = S_OK;
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::OnNo, public
|
|
|
|
// Synopsis: process IDNO button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CNoSignatureUI::OnNo (HWND hwnd)
|
|
{
|
|
m_hrInvokeResult = TRUST_E_SUBJECT_NOT_TRUSTED;
|
|
|
|
EndDialog(hwnd, (int)m_hrInvokeResult);
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Member: CNoSignatureUI::OnMore, public
|
|
|
|
// Synopsis: process the IDMORE button click
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
|
|
// Returns: TRUE
|
|
|
|
// Notes:
|
|
|
|
|
|
BOOL
|
|
CNoSignatureUI::OnMore (HWND hwnd)
|
|
{
|
|
WinHelp(hwnd, "SECAUTH.HLP", HELP_CONTEXT, IDH_SECAUTH_UNSIGNED);
|
|
|
|
// ACUIViewHTMLHelpTopic(hwnd, "sec_unsigned.htm");
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
// Function: ACUIMessageProc
|
|
|
|
// Synopsis: message proc to process UI messages
|
|
|
|
// Arguments: [hwnd] -- window
|
|
// [uMsg] -- message id
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if message processing should continue, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
INT_PTR CALLBACK ACUIMessageProc (
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
IACUIControl* pUI = NULL;
|
|
|
|
|
|
// Get the control
|
|
|
|
|
|
if (uMsg == WM_INITDIALOG)
|
|
{
|
|
pUI = (IACUIControl *)lParam;
|
|
SetWindowLongPtr(hwnd, DWLP_USER, (LONG_PTR)lParam);
|
|
}
|
|
else
|
|
{
|
|
pUI = (IACUIControl *)GetWindowLongPtr(hwnd, DWLP_USER);
|
|
}
|
|
|
|
|
|
// If we couldn't find it, we must not have set it yet, so ignore this
|
|
// message
|
|
|
|
|
|
if ( pUI == NULL )
|
|
{
|
|
return( FALSE );
|
|
}
|
|
|
|
|
|
// Pass the message on to the control
|
|
|
|
|
|
return( pUI->OnUIMessage(hwnd, uMsg, wParam, lParam) );
|
|
}
|
|
|
|
|
|
int GetRichEditControlLineHeight(HWND hwnd)
|
|
{
|
|
RECT rect;
|
|
POINT pointInFirstRow;
|
|
POINT pointInSecondRow;
|
|
int secondLineCharIndex;
|
|
int i;
|
|
RECT originalRect;
|
|
|
|
GetWindowRect(hwnd, &originalRect);
|
|
|
|
|
|
// HACK ALERT, believe it or not there is no way to get the height of the current
|
|
// font in the edit control, so get the position a character in the first row and the position
|
|
// of a character in the second row, and do the subtraction to get the
|
|
// height of the font
|
|
|
|
SendMessageA(hwnd, EM_POSFROMCHAR, (WPARAM) &pointInFirstRow, (LPARAM) 0);
|
|
|
|
|
|
// HACK ON TOP OF HACK ALERT,
|
|
// since there may not be a second row in the edit box, keep reducing the width
|
|
// by half until the first row falls over into the second row, then get the position
|
|
// of the first char in the second row and finally reset the edit box size back to
|
|
// it's original size
|
|
|
|
secondLineCharIndex = (int)SendMessageA(hwnd, EM_LINEINDEX, (WPARAM) 1, (LPARAM) 0);
|
|
if (secondLineCharIndex == -1)
|
|
{
|
|
for (i=0; i<20; i++)
|
|
{
|
|
GetWindowRect(hwnd, &rect);
|
|
SetWindowPos( hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
(rect.right-rect.left)/2,
|
|
rect.bottom-rect.top,
|
|
SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOOWNERZORDER);
|
|
secondLineCharIndex = (int)SendMessageA(hwnd, EM_LINEINDEX, (WPARAM) 1, (LPARAM) 0);
|
|
if (secondLineCharIndex != -1)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (secondLineCharIndex == -1)
|
|
{
|
|
// if we failed after twenty tries just reset the control to its original size
|
|
// and get the heck outa here!!
|
|
SetWindowPos(hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
originalRect.right-originalRect.left,
|
|
originalRect.bottom-originalRect.top,
|
|
SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOOWNERZORDER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
SendMessageA(hwnd, EM_POSFROMCHAR, (WPARAM) &pointInSecondRow, (LPARAM) secondLineCharIndex);
|
|
|
|
SetWindowPos(hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
originalRect.right-originalRect.left,
|
|
originalRect.bottom-originalRect.top,
|
|
SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOOWNERZORDER);
|
|
}
|
|
else
|
|
{
|
|
SendMessageA(hwnd, EM_POSFROMCHAR, (WPARAM) &pointInSecondRow, (LPARAM) secondLineCharIndex);
|
|
}
|
|
|
|
return (pointInSecondRow.y - pointInFirstRow.y);
|
|
}
|
|
|
|
|
|
|
|
// Function: RebaseControlVertical
|
|
|
|
// Synopsis: Take the window control, if it has to be resized for text, do
|
|
// so. Reposition it adjusted for delta pos and return any
|
|
// height difference for the text resizing
|
|
|
|
// Arguments: [hwndDlg] -- host dialog
|
|
// [hwnd] -- control
|
|
// [hwndNext] -- next control
|
|
// [fResizeForText] -- resize for text flag
|
|
// [deltavpos] -- delta vertical position
|
|
// [oline] -- original number of lines
|
|
// [minsep] -- minimum separator
|
|
// [pdeltaheight] -- delta in control height
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
VOID RebaseControlVertical (
|
|
HWND hwndDlg,
|
|
HWND hwnd,
|
|
HWND hwndNext,
|
|
BOOL fResizeForText,
|
|
int deltavpos,
|
|
int oline,
|
|
int minsep,
|
|
int* pdeltaheight
|
|
)
|
|
{
|
|
int x = 0;
|
|
int y = 0;
|
|
int odn = 0;
|
|
int orig_w;
|
|
RECT rect;
|
|
RECT rectNext;
|
|
RECT rectDlg;
|
|
TEXTMETRIC tm;
|
|
|
|
|
|
// Set the delta height to zero for now. If we resize the text
|
|
// a new one will be calculated
|
|
|
|
|
|
*pdeltaheight = 0;
|
|
|
|
|
|
// Get the control window rectangle
|
|
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
GetWindowRect(hwndNext, &rectNext);
|
|
|
|
odn = rectNext.top - rect.bottom;
|
|
|
|
orig_w = rect.right - rect.left;
|
|
|
|
MapWindowPoints(NULL, hwndDlg, (LPPOINT) &rect, 2);
|
|
|
|
|
|
// If we have to resize the control due to text, find out what font
|
|
// is being used and the number of lines of text. From that we'll
|
|
// calculate what the new height for the control is and set it up
|
|
|
|
|
|
if ( fResizeForText == TRUE )
|
|
{
|
|
HDC hdc;
|
|
HFONT hfont;
|
|
HFONT hfontOld;
|
|
int cline;
|
|
int h;
|
|
int w;
|
|
int dh;
|
|
int lineHeight;
|
|
|
|
|
|
// Get the metrics of the current control font
|
|
|
|
|
|
hdc = GetDC(hwnd);
|
|
if (hdc == NULL)
|
|
{
|
|
hdc = GetDC(NULL);
|
|
if (hdc == NULL)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
hfont = (HFONT)SendMessage(hwnd, WM_GETFONT, 0, 0);
|
|
if ( hfont == NULL )
|
|
{
|
|
hfont = (HFONT)SendMessage(hwndDlg, WM_GETFONT, 0, 0);
|
|
}
|
|
|
|
hfontOld = (HFONT)SelectObject(hdc, hfont);
|
|
GetTextMetrics(hdc, &tm);
|
|
|
|
lineHeight = GetRichEditControlLineHeight(hwnd);
|
|
if (lineHeight == 0)
|
|
{
|
|
lineHeight = tm.tmHeight;
|
|
}
|
|
|
|
|
|
// Set the minimum separation value
|
|
|
|
|
|
if ( minsep == 0 )
|
|
{
|
|
minsep = lineHeight;
|
|
}
|
|
|
|
|
|
// Calculate the width and the new height needed
|
|
|
|
|
|
cline = (int)SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
|
|
|
|
h = cline * lineHeight;
|
|
|
|
w = GetEditControlMaxLineWidth(hwnd, hdc, cline);
|
|
w += 3; // a little bump to make sure string will fit
|
|
|
|
if (w > orig_w)
|
|
{
|
|
w = orig_w;
|
|
}
|
|
|
|
SelectObject(hdc, hfontOld);
|
|
ReleaseDC(hwnd, hdc);
|
|
|
|
|
|
// Calculate an addition to height by checking how much space was
|
|
// left when there were the original # of lines and making sure that
|
|
// that amount is still left when we do any adjustments
|
|
|
|
|
|
h += ( ( rect.bottom - rect.top ) - ( oline * lineHeight ) );
|
|
dh = h - ( rect.bottom - rect.top );
|
|
|
|
|
|
// If the current height is too small, adjust for it, otherwise
|
|
// leave the current height and just adjust for the width
|
|
|
|
|
|
if ( dh > 0 )
|
|
{
|
|
SetWindowPos(hwnd, NULL, 0, 0, w, h, SWP_NOZORDER | SWP_NOMOVE);
|
|
}
|
|
else
|
|
{
|
|
SetWindowPos(
|
|
hwnd,
|
|
NULL,
|
|
0,
|
|
0,
|
|
w,
|
|
( rect.bottom - rect.top ),
|
|
SWP_NOZORDER | SWP_NOMOVE
|
|
);
|
|
}
|
|
|
|
if ( cline < SendMessage(hwnd, EM_GETLINECOUNT, 0, 0) )
|
|
{
|
|
AdjustEditControlWidthToLineCount(hwnd, cline, &tm);
|
|
}
|
|
}
|
|
|
|
|
|
// If we have to use deltavpos then calculate the X and the new Y
|
|
// and set the window position appropriately
|
|
|
|
|
|
if ( deltavpos != 0 )
|
|
{
|
|
GetWindowRect(hwndDlg, &rectDlg);
|
|
|
|
MapWindowPoints(NULL, hwndDlg, (LPPOINT) &rectDlg, 2);
|
|
|
|
x = rect.left - rectDlg.left - GetSystemMetrics(SM_CXEDGE);
|
|
y = rect.top - rectDlg.top - GetSystemMetrics(SM_CYCAPTION) + deltavpos;
|
|
|
|
SetWindowPos(hwnd, NULL, x, y, 0, 0, SWP_NOZORDER | SWP_NOSIZE);
|
|
}
|
|
|
|
|
|
// Get the window rect for the next control and see what the distance
|
|
// is between the current control and it. With that we must now
|
|
// adjust our deltaheight, if the distance to the next control is less
|
|
// than a line height then make it a line height, otherwise just let it
|
|
// be
|
|
|
|
|
|
if ( hwndNext != NULL )
|
|
{
|
|
int dn;
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
GetWindowRect(hwndNext, &rectNext);
|
|
|
|
dn = rectNext.top - rect.bottom;
|
|
|
|
if ( odn > minsep )
|
|
{
|
|
if ( dn < minsep )
|
|
{
|
|
*pdeltaheight = minsep - dn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( dn < odn )
|
|
{
|
|
*pdeltaheight = odn - dn;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Function: ACUISetArrowCursorSubclass
|
|
|
|
// Synopsis: subclass routine for setting the arrow cursor. This can be
|
|
// set on multiline edit routines used in the dialog UIs for
|
|
// the default Authenticode provider
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
// [uMsg] -- message id
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if message handled, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
LRESULT CALLBACK ACUISetArrowCursorSubclass (
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
HDC hdc;
|
|
WNDPROC wndproc;
|
|
PAINTSTRUCT ps;
|
|
|
|
wndproc = (WNDPROC)GetWindowLongPtr(hwnd, GWLP_USERDATA);
|
|
|
|
switch ( uMsg )
|
|
{
|
|
case WM_SETCURSOR:
|
|
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
return( TRUE );
|
|
|
|
break;
|
|
|
|
case WM_CHAR:
|
|
|
|
if ( wParam != (WPARAM)' ' )
|
|
{
|
|
break;
|
|
}
|
|
|
|
case WM_LBUTTONDOWN:
|
|
|
|
if ( hwnd == GetDlgItem(GetParent(hwnd), IDC_PERSONALTRUST) )
|
|
{
|
|
int check;
|
|
HWND hwndCheck;
|
|
|
|
|
|
// Toggle the check state of the PTCHECK control if the
|
|
// personal trust statement is clicked on
|
|
|
|
|
|
hwndCheck = GetDlgItem(GetParent(hwnd), IDC_PTCHECK);
|
|
check = (int)SendMessage(hwndCheck, BM_GETCHECK, 0, 0);
|
|
|
|
if ( check == BST_CHECKED )
|
|
{
|
|
check = BST_UNCHECKED;
|
|
}
|
|
else if ( check == BST_UNCHECKED )
|
|
{
|
|
check = BST_CHECKED;
|
|
}
|
|
else
|
|
{
|
|
check = BST_UNCHECKED;
|
|
}
|
|
|
|
SendMessage(hwndCheck, BM_SETCHECK, (WPARAM)check, 0);
|
|
SetFocus(hwnd);
|
|
return( TRUE );
|
|
}
|
|
|
|
return(TRUE);
|
|
|
|
case WM_LBUTTONDBLCLK:
|
|
case WM_MBUTTONDBLCLK:
|
|
case WM_RBUTTONDBLCLK:
|
|
case WM_RBUTTONUP:
|
|
case WM_MBUTTONUP:
|
|
case WM_RBUTTONDOWN:
|
|
case WM_MBUTTONDOWN:
|
|
|
|
return( TRUE );
|
|
|
|
break;
|
|
|
|
case EM_SETSEL:
|
|
|
|
return( TRUE );
|
|
|
|
break;
|
|
|
|
case WM_PAINT:
|
|
|
|
CallWindowProc(wndproc, hwnd, uMsg, wParam, lParam);
|
|
if ( hwnd == GetFocus() )
|
|
{
|
|
DrawFocusRectangle(hwnd, NULL);
|
|
}
|
|
return( TRUE );
|
|
|
|
break;
|
|
|
|
case WM_SETFOCUS:
|
|
|
|
if ( hwnd != GetDlgItem(GetParent(hwnd), IDC_PERSONALTRUST) )
|
|
{
|
|
SetFocus(GetNextDlgTabItem(GetParent(hwnd), hwnd, FALSE));
|
|
return( TRUE );
|
|
}
|
|
else
|
|
{
|
|
InvalidateRect(hwnd, NULL, FALSE);
|
|
UpdateWindow(hwnd);
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
return( TRUE );
|
|
}
|
|
|
|
break;
|
|
|
|
case WM_KILLFOCUS:
|
|
|
|
InvalidateRect(hwnd, NULL, TRUE);
|
|
UpdateWindow(hwnd);
|
|
return( TRUE );
|
|
|
|
}
|
|
|
|
return(CallWindowProc(wndproc, hwnd, uMsg, wParam, lParam));
|
|
}
|
|
|
|
|
|
|
|
// Function: SubclassEditControlForArrowCursor
|
|
|
|
// Synopsis: subclasses edit control so that the arrow cursor can replace
|
|
// the edit bar
|
|
|
|
// Arguments: [hwndEdit] -- edit control
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
VOID SubclassEditControlForArrowCursor (HWND hwndEdit)
|
|
{
|
|
LONG_PTR PrevWndProc;
|
|
|
|
PrevWndProc = GetWindowLongPtr(hwndEdit, GWLP_WNDPROC);
|
|
SetWindowLongPtr(hwndEdit, GWLP_USERDATA, (LONG_PTR)PrevWndProc);
|
|
SetWindowLongPtr(hwndEdit, GWLP_WNDPROC, (LONG_PTR)ACUISetArrowCursorSubclass);
|
|
}
|
|
|
|
|
|
|
|
// Function: SubclassEditControlForLink
|
|
// Synopsis: subclasses the edit control for a link using the link subclass
|
|
// data
|
|
// Arguments: [hwndDlg] -- dialog
|
|
// [hwndEdit] -- edit control
|
|
// [wndproc] -- window proc to subclass with
|
|
// [plsd] -- data to pass on to window proc
|
|
// Returns: (none)
|
|
// Notes:
|
|
VOID SubclassEditControlForLink (
|
|
HWND hwndDlg,
|
|
HWND hwndEdit,
|
|
WNDPROC wndproc,
|
|
PTUI_LINK_SUBCLASS_DATA plsd
|
|
)
|
|
{
|
|
HWND hwndTip;
|
|
|
|
plsd->hwndTip = CreateWindowA(
|
|
TOOLTIPS_CLASSA,
|
|
(LPSTR)NULL,
|
|
WS_POPUP | TTS_ALWAYSTIP,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
hwndDlg,
|
|
(HMENU)NULL,
|
|
g_hModule,
|
|
NULL
|
|
);
|
|
|
|
if ( plsd->hwndTip != NULL )
|
|
{
|
|
TOOLINFOA tia;
|
|
DWORD cb;
|
|
LPSTR psz;
|
|
|
|
memset(&tia, 0, sizeof(TOOLINFOA));
|
|
tia.cbSize = sizeof(TOOLINFOA);
|
|
tia.hwnd = hwndEdit;
|
|
tia.uId = 1;
|
|
tia.hinst = g_hModule;
|
|
//GetClientRect(hwndEdit, &tia.rect);
|
|
SendMessage(hwndEdit, EM_GETRECT, 0, (LPARAM)&tia.rect);
|
|
|
|
|
|
// if plsd->uToolTipText is a string then convert it
|
|
|
|
if (plsd->uToolTipText &0xffff0000)
|
|
{
|
|
cb = WideCharToMultiByte(0, 0, (LPWSTR)plsd->uToolTipText, -1, NULL, 0, NULL, NULL);
|
|
if (NULL == (psz = new char[cb]))
|
|
{
|
|
return;
|
|
}
|
|
|
|
WideCharToMultiByte(0, 0, (LPWSTR)plsd->uToolTipText, -1, psz, cb, NULL, NULL);
|
|
tia.lpszText = psz;
|
|
}
|
|
else
|
|
{
|
|
tia.lpszText = (LPSTR)plsd->uToolTipText;
|
|
}
|
|
|
|
SendMessage(plsd->hwndTip, TTM_ADDTOOL, 0, (LPARAM)&tia);
|
|
|
|
if (plsd->uToolTipText &0xffff0000)
|
|
{
|
|
delete[] psz;
|
|
}
|
|
}
|
|
|
|
plsd->fMouseCaptured = FALSE;
|
|
plsd->wpPrev = (WNDPROC)GetWindowLongPtr(hwndEdit, GWLP_WNDPROC);
|
|
SetWindowLongPtr(hwndEdit, GWLP_USERDATA, (LONG_PTR)plsd);
|
|
SetWindowLongPtr(hwndEdit, GWLP_WNDPROC, (LONG_PTR)wndproc);
|
|
}
|
|
|
|
|
|
|
|
// Function: ACUILinkSubclass
|
|
|
|
// Synopsis: subclass for the publisher link
|
|
|
|
// Arguments: [hwnd] -- window handle
|
|
// [uMsg] -- message id
|
|
// [wParam] -- parameter 1
|
|
// [lParam] -- parameter 2
|
|
|
|
// Returns: TRUE if message handled, FALSE otherwise
|
|
|
|
// Notes:
|
|
|
|
|
|
LRESULT CALLBACK ACUILinkSubclass (
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
PTUI_LINK_SUBCLASS_DATA plsd;
|
|
CInvokeInfoHelper* piih;
|
|
|
|
plsd = (PTUI_LINK_SUBCLASS_DATA)GetWindowLongPtr(hwnd, GWLP_USERDATA);
|
|
piih = (CInvokeInfoHelper *)plsd->pvData;
|
|
|
|
switch ( uMsg )
|
|
{
|
|
case WM_SETCURSOR:
|
|
|
|
if (!plsd->fMouseCaptured)
|
|
{
|
|
SetCapture(hwnd);
|
|
plsd->fMouseCaptured = TRUE;
|
|
}
|
|
|
|
SetCursor(LoadCursor((HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE),
|
|
MAKEINTRESOURCE(IDC_TUIHAND)));
|
|
return( TRUE );
|
|
|
|
break;
|
|
|
|
case WM_CHAR:
|
|
|
|
if ( wParam != (WPARAM)' ')
|
|
{
|
|
break;
|
|
}
|
|
|
|
// fall through to wm_lbuttondown....
|
|
|
|
case WM_LBUTTONDOWN:
|
|
|
|
SetFocus(hwnd);
|
|
|
|
switch(plsd->uId)
|
|
{
|
|
case IDC_PUBLISHER:
|
|
piih->CallCertViewProperties(plsd->hwndParent);
|
|
break;
|
|
|
|
case IDC_INSTALLANDRUN:
|
|
piih->CallWebLink(plsd->hwndParent, (WCHAR *)piih->ControlWebPage());
|
|
break;
|
|
|
|
case IDC_AUTHENTICITY:
|
|
piih->CallWebLink(plsd->hwndParent, (WCHAR *)piih->CAWebPage());
|
|
break;
|
|
|
|
|
|
case IDC_ADVANCED:
|
|
piih->CallAdvancedLink(plsd->hwndParent);
|
|
break;
|
|
}
|
|
|
|
return( TRUE );
|
|
|
|
case WM_LBUTTONDBLCLK:
|
|
case WM_MBUTTONDBLCLK:
|
|
case WM_RBUTTONDBLCLK:
|
|
case WM_RBUTTONUP:
|
|
case WM_MBUTTONUP:
|
|
case WM_RBUTTONDOWN:
|
|
case WM_MBUTTONDOWN:
|
|
|
|
return( TRUE );
|
|
|
|
case EM_SETSEL:
|
|
|
|
return( TRUE );
|
|
|
|
case WM_PAINT:
|
|
|
|
CallWindowProc(plsd->wpPrev, hwnd, uMsg, wParam, lParam);
|
|
if ( hwnd == GetFocus() )
|
|
{
|
|
DrawFocusRectangle(hwnd, NULL);
|
|
}
|
|
return( TRUE );
|
|
|
|
case WM_SETFOCUS:
|
|
|
|
if ( hwnd == GetFocus() )
|
|
{
|
|
InvalidateRect(hwnd, NULL, FALSE);
|
|
UpdateWindow(hwnd);
|
|
SetCursor(LoadCursor((HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE),
|
|
MAKEINTRESOURCE(IDC_TUIHAND)));
|
|
return( TRUE );
|
|
}
|
|
break;
|
|
|
|
case WM_KILLFOCUS:
|
|
|
|
InvalidateRect(hwnd, NULL, TRUE);
|
|
UpdateWindow(hwnd);
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
|
|
return( TRUE );
|
|
|
|
case WM_MOUSEMOVE:
|
|
|
|
MSG msg;
|
|
DWORD dwCharLine;
|
|
CHARFORMAT sCharFmt;
|
|
RECT rect;
|
|
int xPos, yPos;
|
|
|
|
memset(&msg, 0, sizeof(MSG));
|
|
msg.hwnd = hwnd;
|
|
msg.message = uMsg;
|
|
msg.wParam = wParam;
|
|
msg.lParam = lParam;
|
|
|
|
SendMessage(plsd->hwndTip, TTM_RELAYEVENT, 0, (LPARAM)&msg);
|
|
|
|
// check to see if the mouse is in this windows rect, if not, then reset
|
|
// the cursor to an arrow and release the mouse
|
|
GetClientRect(hwnd, &rect);
|
|
xPos = LOWORD(lParam);
|
|
yPos = HIWORD(lParam);
|
|
if ((xPos < 0) ||
|
|
(yPos < 0) ||
|
|
(xPos > (rect.right - rect.left)) ||
|
|
(yPos > (rect.bottom - rect.top)))
|
|
{
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
ReleaseCapture();
|
|
plsd->fMouseCaptured = FALSE;
|
|
}
|
|
|
|
/*
|
|
warning!
|
|
EM_CHARFROMPOS gets an access violation!
|
|
|
|
dwCharLine = SendMessage(hwnd, EM_CHARFROMPOS, 0, lParam);
|
|
|
|
if (dwCharLine == (-1))
|
|
{
|
|
return(TRUE);
|
|
}
|
|
|
|
SendMessage(hwnd, EM_SETSEL, (WPARAM)LOWORD(dwCharLine), (LPARAM)(LOWORD(dwCharLine) + 1));
|
|
|
|
memset(&sCharFmt, 0x00, sizeof(CHARFORMAT));
|
|
sCharFmt.cbSize = sizeof(CHARFORMAT);
|
|
|
|
SendMessage(hwnd, EM_GETCHARFORMAT, TRUE, (LPARAM)&sCharFmt);
|
|
|
|
if (sCharFmt.dwEffects & CFE_UNDERLINE)
|
|
{
|
|
SetCursor(LoadCursor((HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE),
|
|
MAKEINTRESOURCE(IDC_TUIHAND)));
|
|
}
|
|
else
|
|
{
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
}
|
|
|
|
*/
|
|
return( TRUE );
|
|
}
|
|
|
|
return(CallWindowProc(plsd->wpPrev, hwnd, uMsg, wParam, lParam));
|
|
}
|
|
|
|
|
|
|
|
// Function: FormatACUIResourceString
|
|
|
|
// Synopsis: formats a string given a resource id and message arguments
|
|
|
|
// Arguments: [StringResourceId] -- resource id
|
|
// [aMessageArgument] -- message arguments
|
|
// [ppszFormatted] -- formatted string goes here
|
|
|
|
// Returns: S_OK if successful, any valid HRESULT otherwise
|
|
|
|
|
|
HRESULT FormatACUIResourceString (
|
|
UINT StringResourceId,
|
|
DWORD_PTR* aMessageArgument,
|
|
LPWSTR* ppszFormatted
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
WCHAR sz[MAX_LOADSTRING_BUFFER];
|
|
LPVOID pvMsg;
|
|
|
|
pvMsg = NULL;
|
|
sz[0] = NULL;
|
|
|
|
|
|
// Load the string resource and format the message with that string and
|
|
// the message arguments
|
|
|
|
|
|
if (StringResourceId != 0)
|
|
{
|
|
if ( LoadStringU(g_hModule, StringResourceId, sz, MAX_LOADSTRING_BUFFER) == 0 )
|
|
{
|
|
return(HRESULT_FROM_WIN32(GetLastError()));
|
|
}
|
|
|
|
if ( FormatMessageU(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_STRING |
|
|
FORMAT_MESSAGE_ARGUMENT_ARRAY, sz, 0, 0, (LPWSTR)&pvMsg, 0,
|
|
(va_list *)aMessageArgument) == 0)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( FormatMessageU(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_STRING |
|
|
FORMAT_MESSAGE_ARGUMENT_ARRAY, (char *)aMessageArgument[0], 0, 0,
|
|
(LPWSTR)&pvMsg, 0, (va_list *)&aMessageArgument[1]) == 0)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
}
|
|
|
|
if (pvMsg)
|
|
{
|
|
*ppszFormatted = new WCHAR[wcslen((WCHAR *)pvMsg) + 1];
|
|
|
|
if (*ppszFormatted)
|
|
{
|
|
wcscpy(*ppszFormatted, (WCHAR *)pvMsg);
|
|
}
|
|
|
|
LocalFree(pvMsg);
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
|
|
// Function: RenderACUIStringToEditControl
|
|
|
|
// Synopsis: renders a string to the control given and if requested, gives
|
|
// it a link look and feel, subclassed to the wndproc and plsd
|
|
// given
|
|
|
|
// Arguments: [hwndDlg] -- dialog window handle
|
|
// [ControlId] -- control id
|
|
// [NextControlId] -- next control id
|
|
// [psz] -- string
|
|
// [deltavpos] -- delta vertical position
|
|
// [fLink] -- a link?
|
|
// [wndproc] -- optional wndproc, valid if fLink == TRUE
|
|
// [plsd] -- optional plsd, valid if fLink === TRUE
|
|
// [minsep] -- minimum separation
|
|
// [pszThisTextOnlyInLink -- only change this text.
|
|
|
|
// Returns: delta in height of the control
|
|
|
|
// Notes:
|
|
|
|
|
|
int RenderACUIStringToEditControl (
|
|
HWND hwndDlg,
|
|
UINT ControlId,
|
|
UINT NextControlId,
|
|
LPCWSTR psz,
|
|
int deltavpos,
|
|
BOOL fLink,
|
|
WNDPROC wndproc,
|
|
PTUI_LINK_SUBCLASS_DATA plsd,
|
|
int minsep,
|
|
LPCWSTR pszThisTextOnlyInLink
|
|
)
|
|
{
|
|
HWND hControl;
|
|
int deltaheight = 0;
|
|
int oline = 0;
|
|
int hkcharpos;
|
|
|
|
|
|
// Get the control and set the text on it, make sure the background
|
|
// is right if it is a rich edit control
|
|
|
|
|
|
hControl = GetDlgItem(hwndDlg, ControlId);
|
|
oline = (int)SendMessage(hControl, EM_GETLINECOUNT, 0, 0);
|
|
CryptUISetRicheditTextW(hwndDlg, ControlId, L"");
|
|
CryptUISetRicheditTextW(hwndDlg, ControlId, psz); //SetWindowTextU(hControl, psz);
|
|
|
|
|
|
// If there is a '&' in the string, then get rid of it
|
|
|
|
hkcharpos = GetHotKeyCharPosition(hControl);
|
|
if (hkcharpos != 0)
|
|
{
|
|
CHARRANGE cr;
|
|
CHARFORMAT cf;
|
|
|
|
cr.cpMin = hkcharpos - 1;
|
|
cr.cpMax = hkcharpos;
|
|
|
|
SendMessage(hControl, EM_EXSETSEL, 0, (LPARAM) &cr);
|
|
SendMessage(hControl, EM_REPLACESEL, FALSE, (LPARAM) "");
|
|
|
|
cr.cpMin = -1;
|
|
cr.cpMax = 0;
|
|
SendMessage(hControl, EM_EXSETSEL, 0, (LPARAM) &cr);
|
|
}
|
|
|
|
SendMessage(
|
|
hControl,
|
|
EM_SETBKGNDCOLOR,
|
|
0,
|
|
(LPARAM)GetSysColor(COLOR_3DFACE)
|
|
);
|
|
|
|
|
|
// If we have a link then update for the link look
|
|
|
|
|
|
if ( fLink == TRUE )
|
|
{
|
|
CHARFORMAT cf;
|
|
|
|
memset(&cf, 0, sizeof(CHARFORMAT));
|
|
cf.cbSize = sizeof(CHARFORMAT);
|
|
cf.dwMask = CFM_COLOR | CFM_UNDERLINE;
|
|
|
|
cf.crTextColor = RGB(0, 0, 255);
|
|
cf.dwEffects |= CFM_UNDERLINE;
|
|
|
|
if (pszThisTextOnlyInLink)
|
|
{
|
|
FINDTEXTEX ft;
|
|
DWORD pos;
|
|
char *pszOnlyThis;
|
|
DWORD cb;
|
|
|
|
cb = WideCharToMultiByte(0, 0, pszThisTextOnlyInLink, -1, NULL, 0, NULL, NULL);
|
|
|
|
if (NULL == (pszOnlyThis = new char[cb]))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
WideCharToMultiByte(0, 0, pszThisTextOnlyInLink, -1, pszOnlyThis, cb, NULL, NULL);
|
|
|
|
memset(&ft, 0x00, sizeof(FINDTEXTEX));
|
|
ft.chrg.cpMin = 0;
|
|
ft.chrg.cpMax = (-1);
|
|
ft.lpstrText = (char *)pszOnlyThis;
|
|
|
|
if ((pos = (DWORD)SendMessage(hControl, EM_FINDTEXTEX, 0, (LPARAM)&ft)) != (-1))
|
|
{
|
|
SendMessage(hControl, EM_EXSETSEL, 0, (LPARAM)&ft.chrgText);
|
|
SendMessage(hControl, EM_SETCHARFORMAT, SCF_WORD | SCF_SELECTION, (LPARAM)&cf);
|
|
ft.chrgText.cpMin = 0;
|
|
ft.chrgText.cpMax = 0;
|
|
SendMessage(hControl, EM_EXSETSEL, 0, (LPARAM)&ft.chrgText);
|
|
}
|
|
|
|
delete[] pszOnlyThis;
|
|
}
|
|
else
|
|
{
|
|
SendMessage(hControl, EM_SETCHARFORMAT, SCF_ALL, (LPARAM)&cf);
|
|
}
|
|
}
|
|
|
|
|
|
// Rebase the control
|
|
|
|
|
|
RebaseControlVertical(
|
|
hwndDlg,
|
|
hControl,
|
|
GetDlgItem(hwndDlg, NextControlId),
|
|
TRUE,
|
|
deltavpos,
|
|
oline,
|
|
minsep,
|
|
&deltaheight
|
|
);
|
|
|
|
|
|
// If we have the link look then we must subclass for the appropriate
|
|
// link feel, otherwise we subclass for a static text control feel
|
|
|
|
|
|
if ( fLink == TRUE )
|
|
{
|
|
SubclassEditControlForLink(hwndDlg, hControl, wndproc, plsd);
|
|
}
|
|
else
|
|
{
|
|
SubclassEditControlForArrowCursor(hControl);
|
|
}
|
|
|
|
return( deltaheight );
|
|
}
|
|
|
|
|
|
|
|
// Function: CalculateControlVerticalDistance
|
|
|
|
// Synopsis: calculates the vertical distance from the bottom of Control1
|
|
// to the top of Control2
|
|
|
|
// Arguments: [hwnd] -- parent dialog
|
|
// [Control1] -- first control
|
|
// [Control2] -- second control
|
|
|
|
// Returns: the distance in pixels
|
|
|
|
// Notes: assumes control1 is above control2
|
|
|
|
|
|
int CalculateControlVerticalDistance (HWND hwnd, UINT Control1, UINT Control2)
|
|
{
|
|
RECT rect1;
|
|
RECT rect2;
|
|
|
|
GetWindowRect(GetDlgItem(hwnd, Control1), &rect1);
|
|
GetWindowRect(GetDlgItem(hwnd, Control2), &rect2);
|
|
|
|
return( rect2.top - rect1.bottom );
|
|
}
|
|
|
|
|
|
|
|
// Function: CalculateControlVerticalDistanceFromDlgBottom
|
|
|
|
// Synopsis: calculates the distance from the bottom of the control to
|
|
// the bottom of the dialog
|
|
|
|
// Arguments: [hwnd] -- dialog
|
|
// [Control] -- control
|
|
|
|
// Returns: the distance in pixels
|
|
|
|
// Notes:
|
|
|
|
|
|
int CalculateControlVerticalDistanceFromDlgBottom (HWND hwnd, UINT Control)
|
|
{
|
|
RECT rect;
|
|
RECT rectControl;
|
|
|
|
GetClientRect(hwnd, &rect);
|
|
GetWindowRect(GetDlgItem(hwnd, Control), &rectControl);
|
|
|
|
return( rect.bottom - rectControl.bottom );
|
|
}
|
|
|
|
|
|
|
|
// Function: ACUICenterWindow
|
|
|
|
// Synopsis: centers the given window
|
|
|
|
// Arguments: [hWndToCenter] -- window handle
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes: This code was stolen from ATL and hacked upon madly :-)
|
|
|
|
|
|
VOID ACUICenterWindow (HWND hWndToCenter)
|
|
{
|
|
HWND hWndCenter;
|
|
|
|
// determine owner window to center against
|
|
DWORD dwStyle = (DWORD)GetWindowLong(hWndToCenter, GWL_STYLE);
|
|
|
|
if(dwStyle & WS_CHILD)
|
|
hWndCenter = ::GetParent(hWndToCenter);
|
|
else
|
|
hWndCenter = ::GetWindow(hWndToCenter, GW_OWNER);
|
|
|
|
if (hWndCenter == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// get coordinates of the window relative to its parent
|
|
RECT rcDlg;
|
|
::GetWindowRect(hWndToCenter, &rcDlg);
|
|
RECT rcArea;
|
|
RECT rcCenter;
|
|
HWND hWndParent;
|
|
if(!(dwStyle & WS_CHILD))
|
|
{
|
|
// don't center against invisible or minimized windows
|
|
if(hWndCenter != NULL)
|
|
{
|
|
DWORD dwStyle = ::GetWindowLong(hWndCenter, GWL_STYLE);
|
|
if(!(dwStyle & WS_VISIBLE) || (dwStyle & WS_MINIMIZE))
|
|
hWndCenter = NULL;
|
|
}
|
|
|
|
// center within screen coordinates
|
|
::SystemParametersInfo(SPI_GETWORKAREA, NULL, &rcArea, NULL);
|
|
|
|
if(hWndCenter == NULL)
|
|
rcCenter = rcArea;
|
|
else
|
|
::GetWindowRect(hWndCenter, &rcCenter);
|
|
}
|
|
else
|
|
{
|
|
// center within parent client coordinates
|
|
hWndParent = ::GetParent(hWndToCenter);
|
|
|
|
::GetClientRect(hWndParent, &rcArea);
|
|
::GetClientRect(hWndCenter, &rcCenter);
|
|
::MapWindowPoints(hWndCenter, hWndParent, (POINT*)&rcCenter, 2);
|
|
}
|
|
|
|
int DlgWidth = rcDlg.right - rcDlg.left;
|
|
int DlgHeight = rcDlg.bottom - rcDlg.top;
|
|
|
|
// find dialog's upper left based on rcCenter
|
|
int xLeft = (rcCenter.left + rcCenter.right) / 2 - DlgWidth / 2;
|
|
int yTop = (rcCenter.top + rcCenter.bottom) / 2 - DlgHeight / 2;
|
|
|
|
// if the dialog is outside the screen, move it inside
|
|
if(xLeft < rcArea.left)
|
|
xLeft = rcArea.left;
|
|
else if(xLeft + DlgWidth > rcArea.right)
|
|
xLeft = rcArea.right - DlgWidth;
|
|
|
|
if(yTop < rcArea.top)
|
|
yTop = rcArea.top;
|
|
else if(yTop + DlgHeight > rcArea.bottom)
|
|
yTop = rcArea.bottom - DlgHeight;
|
|
|
|
// map screen coordinates to child coordinates
|
|
::SetWindowPos(
|
|
hWndToCenter,
|
|
HWND_TOPMOST,
|
|
xLeft,
|
|
yTop,
|
|
-1,
|
|
-1,
|
|
SWP_NOSIZE | SWP_NOACTIVATE
|
|
);
|
|
}
|
|
|
|
|
|
|
|
// Function: ACUIViewHTMLHelpTopic
|
|
|
|
// Synopsis: html help viewer
|
|
|
|
// Arguments: [hwnd] -- caller window
|
|
// [pszTopic] -- topic
|
|
|
|
// Returns: (none)
|
|
|
|
// Notes:
|
|
|
|
|
|
VOID ACUIViewHTMLHelpTopic (HWND hwnd, LPSTR pszTopic)
|
|
{
|
|
// HtmlHelpA(
|
|
// hwnd,
|
|
// "%SYSTEMROOT%\\help\\iexplore.chm>large_context",
|
|
// HH_DISPLAY_TOPIC,
|
|
// (DWORD)pszTopic
|
|
// );
|
|
}
|
|
|
|
|
|
|
|
// Function: GetEditControlMaxLineWidth
|
|
|
|
// Synopsis: gets the maximum line width of the edit control
|
|
|
|
|
|
int GetEditControlMaxLineWidth (HWND hwndEdit, HDC hdc, int cline)
|
|
{
|
|
int index;
|
|
int line;
|
|
int charwidth;
|
|
int maxwidth = 0;
|
|
CHAR szMaxBuffer[1024];
|
|
WCHAR wsz[1024];
|
|
TEXTRANGEA tr;
|
|
SIZE size;
|
|
|
|
tr.lpstrText = szMaxBuffer;
|
|
|
|
for ( line = 0; line < cline; line++ )
|
|
{
|
|
index = (int)SendMessage(hwndEdit, EM_LINEINDEX, (WPARAM)line, 0);
|
|
charwidth = (int)SendMessage(hwndEdit, EM_LINELENGTH, (WPARAM)index, 0);
|
|
|
|
tr.chrg.cpMin = index;
|
|
tr.chrg.cpMax = index + charwidth;
|
|
SendMessage(hwndEdit, EM_GETTEXTRANGE, 0, (LPARAM)&tr);
|
|
|
|
wsz[0] = NULL;
|
|
|
|
MultiByteToWideChar(0, 0, (const char *)tr.lpstrText, -1, &wsz[0], 1024);
|
|
|
|
if (wsz[0])
|
|
{
|
|
GetTextExtentPoint32W(hdc, &wsz[0], charwidth, &size);
|
|
|
|
if ( size.cx > maxwidth )
|
|
{
|
|
maxwidth = size.cx;
|
|
}
|
|
}
|
|
}
|
|
|
|
return( maxwidth );
|
|
}
|
|
|
|
|
|
|
|
// Function: DrawFocusRectangle
|
|
|
|
// Synopsis: draws the focus rectangle for the edit control
|
|
|
|
|
|
void DrawFocusRectangle (HWND hwnd, HDC hdc)
|
|
{
|
|
RECT rect;
|
|
PAINTSTRUCT ps;
|
|
BOOL fReleaseDC = FALSE;
|
|
|
|
if ( hdc == NULL )
|
|
{
|
|
hdc = GetDC(hwnd);
|
|
if ( hdc == NULL )
|
|
{
|
|
return;
|
|
}
|
|
fReleaseDC = TRUE;
|
|
}
|
|
|
|
GetClientRect(hwnd, &rect);
|
|
DrawFocusRect(hdc, &rect);
|
|
|
|
if ( fReleaseDC == TRUE )
|
|
{
|
|
ReleaseDC(hwnd, hdc);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Function: GetHotKeyCharPosition
|
|
|
|
// Synopsis: gets the character position for the hotkey, zero means
|
|
// no-hotkey
|
|
|
|
|
|
int GetHotKeyCharPosition (HWND hwnd)
|
|
{
|
|
WCHAR szText[MAX_LOADSTRING_BUFFER];
|
|
LPWSTR psz;
|
|
|
|
GetWindowTextU(hwnd, szText, MAX_LOADSTRING_BUFFER);
|
|
|
|
psz = szText;
|
|
while ( ( psz = wcschr(psz, L'&') ) != NULL )
|
|
{
|
|
psz++;
|
|
if ( *psz != L'&' )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( psz == NULL )
|
|
{
|
|
return( 0 );
|
|
}
|
|
|
|
return (int)(( psz - szText ) );
|
|
}
|
|
|
|
|
|
|
|
// Function: FormatHotKeyOnEditControl
|
|
|
|
// Synopsis: formats the hot key on an edit control by making it underlined
|
|
|
|
|
|
VOID FormatHotKeyOnEditControl (HWND hwnd, int hkcharpos)
|
|
{
|
|
CHARRANGE cr;
|
|
CHARFORMAT cf;
|
|
|
|
assert( hkcharpos != 0 );
|
|
|
|
cr.cpMin = hkcharpos - 1;
|
|
cr.cpMax = hkcharpos;
|
|
|
|
SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM)&cr);
|
|
|
|
memset(&cf, 0, sizeof(CHARFORMAT));
|
|
cf.cbSize = sizeof(CHARFORMAT);
|
|
cf.dwMask = CFM_UNDERLINE;
|
|
cf.dwEffects |= CFM_UNDERLINE;
|
|
|
|
SendMessage(hwnd, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&cf);
|
|
|
|
cr.cpMin = -1;
|
|
cr.cpMax = 0;
|
|
SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM)&cr);
|
|
}
|
|
|
|
|
|
|
|
// Function: AdjustEditControlWidthToLineCount
|
|
|
|
// Synopsis: adjust edit control width to the given line count
|
|
|
|
|
|
void AdjustEditControlWidthToLineCount(HWND hwnd, int cline, TEXTMETRIC* ptm)
|
|
{
|
|
RECT rect;
|
|
int w;
|
|
int h;
|
|
|
|
GetWindowRect(hwnd, &rect);
|
|
h = rect.bottom - rect.top;
|
|
w = rect.right - rect.left;
|
|
|
|
while ( cline < SendMessage(hwnd, EM_GETLINECOUNT, 0, 0) )
|
|
{
|
|
w += ptm->tmMaxCharWidth;
|
|
SetWindowPos(hwnd, NULL, 0, 0, w, h, SWP_NOZORDER | SWP_NOMOVE);
|
|
printf(
|
|
"Line count adjusted to = %d\n",
|
|
(DWORD) SendMessage(hwnd, EM_GETLINECOUNT, 0, 0)
|
|
);
|
|
}
|
|
}
|