我从CEdit派生,制作自定义控件。如果像MFC功能包控件(Mask,Browsable)那样我可以更改GetWindowText以实际报告不是控件上通常显示的内容(例如,在十六进制和十进制之间转换数据,然后返回字符串)。
这是否可以在派生的CEdit中使用?
答案 0 :(得分:3)
将WM_GETTEXT
和WM_GETTEXTLENGTH
的消息映射条目添加到派生的CEdit
类:
BEGIN_MESSAGE_MAP( CMyEdit, CEdit )
ON_WM_GETTEXT()
ON_WM_GETTEXTLENGTH()
END_MESSAGE_MAP()
当我们覆盖这些消息时,我们需要一种获取编辑控件的原始文本而无需进行无限递归的方法。为此,我们可以直接调用名为DefWindowProc
的默认窗口过程:
CStringW CMyEdit::GetTextInternal()
{
CStringW text;
LRESULT len = DefWindowProcW( WM_GETTEXTLENGTH, 0, 0 );
if( len > 0 )
{
// WPARAM = len + 1 because the length must include the null terminator.
len = DefWindowProcW( WM_GETTEXT, len + 1, reinterpret_cast<LPARAM>( text.GetBuffer( len ) ) );
text.ReleaseBuffer( len );
}
return text;
}
以下方法获取原始窗口文本并对其进行转换。这里可以做任何事情,包括在十六进制和十进制之间转换的例子。为简单起见,我只是用破折号将文本括起来。
CStringW CMyEdit::GetTransformedText()
{
CStringW text = GetTextInternal();
return L"--" + text + L"--";
}
现在出现了WM_GETTEXT
的实际处理程序,它将转换后的文本复制到输出缓冲区。
int CMyEdit::OnGetText( int cchDest, LPWSTR pDest )
{
// Sanity checks
if( cchDest <= 0 || ! pDest )
return 0;
CStringW text = GetTransformedText();
// Using StringCchCopyExW() to make sure that we don't write outside of the bounds of the pDest buffer.
// cchDest defines the maximum number of characters to be copied, including the terminating null character.
LPWSTR pDestEnd = nullptr;
HRESULT hr = StringCchCopyExW( pDest, cchDest, text.GetString(), &pDestEnd, nullptr, 0 );
// If our text is greater in length than cchDest - 1, the function will truncate the text and
// return STRSAFE_E_INSUFFICIENT_BUFFER.
if( SUCCEEDED( hr ) || hr == STRSAFE_E_INSUFFICIENT_BUFFER )
{
// The return value is the number of characters copied, not including the terminating null character.
return pDestEnd - pDest;
}
return 0;
}
WM_GETTEXTLENGTH
的处理程序不言自明:
UINT CMyEdit::OnGetTextLength()
{
return GetTransformedText().GetLength();
}
答案 1 :(得分:0)
感谢大家指点我正确的方向。我尝试了OnGetText
,但问题似乎是我无法获取底层字符串,或者在调用GetWindowText
时(或者只是再次调用OnGetText
时会崩溃...而且不能找到底层字符串。)
在看到他们在蒙面控制上做了什么之后,我做了一个更简单的答案。有什么缺点吗?它似乎没有引起任何问题或副作用......
直接从GetWindowText
void CConvertibleEdit::GetWindowText(CString& strString) const
{
CEdit::GetWindowText(strString);
ConvertibleDataType targetDataType;
if (currentDataType == inputType)
{
}
else
{
strString = ConvertEditType(strString, currentDataType, inputType);
}
}