winapi — C ++ Как заменить новые строки при вставке текста в элемент управления Edit?

У меня есть простая программа чата. Я использую функцию «CreateWindow» для поля ввода:

chat_handle_11 = CreateWindow("EDIT", "", WS_BORDER | WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL | WS_EX_CONTROLPARENT, 226, 447, 424, 23, hWnd, NULL, NULL, NULL);
SendMessage(chat_handle_11, EM_LIMITTEXT, ChatTextLimitInBox, 0L);

Когда я вставляю любой текст, содержащий символы новой строки (используя правую кнопку мыши или Ctrl + V), например:

Test line 1 text
Test line 2 text
Test line 3 text

Только первая строка вставляется в окно ввода:

Test line 1 text

Я хотел бы изменить текст при вставке, чтобы игнорировать символы новой строки:

Test line 1 text Test line 2 text Test line 3 text

Я попытался обработать сообщение WM_PASTE, к сожалению, оно не сработало:

switch (message)
{
case WM_PASTE:
{
MessageBox(NULL, "pasting", "pasting", MB_YESNO | MB_ICONQUESTION);
break;
}
...

MessageBox никогда не показывался. Является ли WM_PASTE правильным сообщением в этом случае?

Кроме того, я попытался добавить «ES_MULTILINE» в CreateWindow, но затем, когда я пытаюсь вставить текст, содержащий несколько строк, текст вообще не вставляется, я слышу только звуковой сигнал.

Я знаю, что могу удалить новые строки, обнаружив изменения в буфере обмена, а затем перезаписав их, но это решение «вторгнется» в буфер обмена пользователя, поэтому я не хочу его использовать.

Буду очень признателен за любую помощь.

0

Решение

Спасибо @RbMm за помощь. Я смог решить проблему.

  1. Я не использовал подкласс для элемента управления Edit и пытался обработать сообщение WM_PASTE в родительском окне.

Фиксированный код:

chat_handle_11 = CreateWindow("EDIT", "", WS_BORDER | WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL | WS_EX_CONTROLPARENT, 226, 447, 424, 23, hWnd, NULL, NULL, NULL);
SendMessage(chat_handle_11, EM_LIMITTEXT, ChatTextLimitInBox, 0L);
SetWindowSubclass(chat_handle_11, EditBoxForPasteFixes, 0, 0);

Тогда новый CALLBACK:

LRESULT CALLBACK EditBoxForPasteFixes(HWND handle, UINT uMsg, WPARAM wParam, LPARAM lParam, UINT_PTR, DWORD_PTR) {
switch (uMsg) {
case WM_PASTE:
{
try {
wstring ClipboardText = GetClipboardText();
find_and_replace_ws(ClipboardText, L"\r\n", L" ");
find_and_replace_ws(ClipboardText, L"\r", L" ");
find_and_replace_ws(ClipboardText, L"\n", L" ");
//We don't need to SETSEL, so we keep original position for pasting
//SendMessage(handle, EM_SETSEL, WPARAM(0), LPARAM(-1));
SendMessageW(handle, EM_REPLACESEL, WPARAM(TRUE), LPARAM(ClipboardText.c_str()));
}
catch (...) {
return FALSE;
}
return TRUE;
break;
}

/*case WM_LBUTTONDOWN:
//std::wcout << handle << L" click\n"; //click event works
break;*/
case WM_NCDESTROY:
{
RemoveWindowSubclass(handle, EditBoxForPasteFixes, 0);
// fall through
}
default:
{
return DefSubclassProc(handle, uMsg, wParam, lParam);
}
}
return 0;
}

И функция GetClipboardText:

std::wstring GetClipboardText()
{
bool Failed = false;
std::wstring ReturnText = L"";
// Try opening the clipboard
if (!OpenClipboard(nullptr)) {
Failed = true;
}
// Get handle of clipboard object for ANSI text
if (!Failed) {
//HANDLE hData = GetClipboardData(CF_TEXT);
HANDLE hData = GetClipboardData(CF_UNICODETEXT);
if (hData == nullptr) {
Failed = true;
}

// Lock the handle to get the actual text pointer
if (!Failed) {
wchar_t * pszText = static_cast<wchar_t*>(GlobalLock(hData));
if (pszText == nullptr) {
Failed = true;
}
if (!Failed) {
std::wstring text(pszText);
ReturnText = text;
}
// Release the lock
GlobalUnlock(hData);
}
// Release the clipboard
CloseClipboard();
}
return ReturnText;
}

Для find_and_replace_ws я использую функцию boost, но можно заменить на что-нибудь еще:

void find_and_replace_ws(wstring& source, wstring const& find, wstring const& replace)
{
boost::replace_all(source, find, replace);
/*for (std::string::size_type i = 0; (i = source.find(find, i)) != std::string::npos;)
{
source.replace(i, find.length(), replace);
i += replace.length() - find.length() + 1;
}*/
}

Я знаю, не идеальный код, но достаточно для моих нужд 🙂

0

Другие решения

Других решений пока нет …

По вопросам рекламы ammmcru@yandex.ru
Adblock
detector