MFC C ++从CEdit派生并派生GetWindowText

时间:2017-10-26 21:44:19

标签: c++ mfc mfc-feature-pack

我从CEdit派生,制作自定义控件。如果像MFC功能包控件(Mask,Browsable)那样我可以更改GetWindowText以实际报告不是控件上通常显示的内容(例如,在十六进制和十进制之间转换数据,然后返回字符串)。

这是否可以在派生的CEdit中使用?

2 个答案:

答案 0 :(得分:3)

WM_GETTEXTWM_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);
    }
}