Приложения для Win32, пытающиеся обновить текст метки
Этот код отображает окно с текстовой меткой "Пожалуйста, введите номер" и кнопку.
Когда вы нажимаете кнопку, он должен заменить текст на "ТЕКСТ". Это работает, но он пишет / печатает новый текст поверх первого текста. Так что это перекрытие.
Я хочу, чтобы строка текста изменялась вместо записи поверх первой, но я не знаю как, поскольку я новичок в разработке приложений для Windows.
Пожалуйста, помогите мне, ребята.
Весь источник:
#include <windows.h>
#include <iostream>
using namespace std;
enum { ID_LABEL = 1,ID_BUTTON0};
static HWND static_label, button0;
HDC hdc;
HBRUSH NewBrush;
HINSTANCE g_hInst;
LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam);
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
{
LPCTSTR className = TEXT("myClass");
WNDCLASSEX wc;
wc.cbSize = sizeof(wc);
wc.style = CS_HREDRAW | CS_VREDRAW;
wc.cbWndExtra = 0;
wc.cbClsExtra = 0;
wc.lpfnWndProc = WndProc;
wc.hInstance = hInstance;
wc.lpszMenuName = NULL;
wc.lpszClassName = TEXT("myClass");
wc.hbrBackground = (HBRUSH)(CreateSolidBrush(RGB(48, 38, 88)));
wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wc.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
if(!RegisterClassEx(&wc))
{
MessageBox(NULL, TEXT("ERROR! FAILED TO REGISTER CLASS!"), TEXT("FATAL ERROR!"), MB_IConerror | MB_OK);
return 1;
}
HWND hwnd = CreateWindowEx(0, TEXT("myClass"), TEXT("WINDOW TITLE"), WS_OVERLAPPEDWINDOW, 450, 100, 500 + 7, 500 + 33 , NULL, NULL, hInstance, NULL);
if(!hwnd)
{
MessageBox(NULL, TEXT("ERROR! FAILED TO CREATE WINDOW!"), TEXT("FATAL ERROR!"), MB_IConerror | MB_OK);
return true;
}
ShowWindow(hwnd, nShowCmd);
UpdateWindow(hwnd);
MSG msg;
while(GetMessage(&msg, NULL, 0, 0) > 0)
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return (int)msg.wParam;
}
LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
case WM_PAINT:
{
}
case WM_CTLCOLORSTATIC:
{
SetBkMode((HDC) wParam, TRANSPARENT);
return (LONG) GetStockObject(NULL_BRUSH);
}
break;
case WM_CREATE:
{
static_label = CreateWindow(L"Static",L"Please Enter A Number",WS_CHILD | WS_VISIBLE,35,15,175,25,hwnd,0, g_hInst,0);
button0 = CreateWindow(L"Button",L"OK",BS_PUSHBUTTON | WS_CHILD | WS_VISIBLE ,80,220,35,35,hwnd,(HMENU)ID_BUTTON0,g_hInst,0);
}
break;
case WM_COMMAND: //Command from Child windows and menus are under this message
switch(wParam) //the ID is wParam
{
case ID_BUTTON0: //check for our button ID
{
SetWindowText(static_label,L"TEXT");
break;
}
}//switch.
break;
case WM_DESTROY:
PostQuitMessage(0);
break; // pass to DefWindowProc(...) as well
case WM_CLOSE:
DestroyWindow(hwnd);
break;
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}
4 ответа
Проблема здесь:
case WM_CTLCOLORSTATIC:
{
SetBkMode((HDC) wParam, TRANSPARENT);
return (LONG) GetStockObject(NULL_BRUSH);
}
Этот код говорит статическому элементу управления рисовать текст без цвета фона и не перекрашивать фон. Таким образом, новый текст рисуется поверх старого текста, а не на новом фоне.
Если вам нужен какой-то пользовательский фон для отображения, вам придется сделать недействительной эту часть основного родительского окна и, возможно, использовать что-то вроде WS_EX_TRANSPARENT
чтобы гарантировать, что дочерний статический элемент управления нарисован последним. Таким образом, к тому времени, когда он пытается нарисовать новый текст, должен быть нарисован новый фон.
Обратите внимание, что это означает, что вы не можете использовать WS_CLIPCHILDREN
в родительском окне, которое может усилить мерцание при перерисовке.
Используйте этот код для обновления метки после функции SetWindowText:
SetWindowText(static_label,L"TEXT");
ShowWindow(static_label, SW_HIDE);
ShowWindow(static_label, SW_SHOW);
Ваш текст отображается в "статическом" окне, и они не ожидают, что текст изменится, поэтому они не обрабатывают его изящно. Вы должны заставить элемент управления стереть и перерисовать себя.
RedrawWindow(static_label, NULL, NULL, RDW_ERASE);
RedrawWindow(H_frame,NULL,NULL,RDW_INVALIDATE) работал для меня, когда я хотел спровоцировать изменения, вызванные WM_CTLCOLORSTATIC, чтобы получить немедленный видимый эффект.
Ранее я получил частичный результат от SendMessage(H_frame,WM_CTLCOLORSTATIC,0,(int)SpecificControlHandle), но это сработало даже частично, потому что вызываемая функция выполняла GetWindowDC и рисовал материал....
Метод RedrawWindow(H_frame,NULL,NULL,RDW_INVALIDATE) намного лучше. Я изменял цвет текста и фона для текста (а также идентификатор элемента управления с помощью SetWindowLong(SpecificControlHandle,GWL_ID,SomeNewID), чтобы сообщить программе, какой маршрут выбрать после отправки WM_CTLCOLORSTATIC. RedrawWindow(H_frame,NULL,NULL,RDW_INVALIDATE), поэтому мой пост. Если он работает после изменения идентификатора STATIC, он должен работать после изменения любого другого атрибута или стиля и т. Д.