Приложения для 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, он должен работать после изменения любого другого атрибута или стиля и т. Д.

Другие вопросы по тегам