Win32 API CreateWindowW конфликтует с созданием меню

Я пытаюсь добавить некоторые элементы управления приложением в свой код Windows API (Примечание: я использую Visual Studio). У меня возникла проблема, когда я пытаюсь добавить функцию CreateWindowW(), которая генерирует текст и поле («статическое» и "редактировать") и меню одновременно. Меню прекрасно работает само по себе ("Расчеты"):

Без функции CreateWindowW()

Однако добавление функции CreateWindow() «полностью стирает» меню, но дает выходные данные CreateWindowW() (также немного мерцающие):

С функцией CreateWindowW()

Код, который у меня есть сейчас, таков (функция меню и функции CreateWindowW() находятся внизу):

#define MAX_LOADSTRING 100
#define FILE_MENU_DESTROY 1
#define FILE_MENU_ABOUT 2

// Global Variables:
HINSTANCE hInst;                                // current instance
WCHAR szTitle[MAX_LOADSTRING];                  // The title bar text
WCHAR szWindowClass[MAX_LOADSTRING];            // the main window class name

// Forward declarations of functions included in this code module:
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
//INT_PTR CALLBACK    About(HWND, UINT, WPARAM, LPARAM);

//Custom Application Function Forwrd Declarations
void WinControls(HWND);
void WinMenu(HWND);
HMENU hMenu;

int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
                     _In_opt_ HINSTANCE hPrevInstance,
                     _In_ LPWSTR    lpCmdLine,
                     _In_ int       nCmdShow)
{
    UNREFERENCED_PARAMETER(hPrevInstance);
    UNREFERENCED_PARAMETER(lpCmdLine);

    // TODO: Place code here.

    // Initialize global strings
    LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
    LoadStringW(hInstance, IDC_RANKTOOLADVANCED, szWindowClass, MAX_LOADSTRING);
    MyRegisterClass(hInstance); //Generates an instance of a window class

    // Perform application initialization:
    if (!InitInstance (hInstance, nCmdShow))
    {
        return FALSE;
    }
    HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_RANKTOOLADVANCED));

    MSG msg;

    // Main message loop:
    while (GetMessage(&msg, nullptr, 0, 0))
    {
        if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
        {
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
    }

    return (int) msg.wParam;
}



//
//  FUNCTION: MyRegisterClass()
//
//  PURPOSE: Registers the window class.
//
ATOM MyRegisterClass(HINSTANCE hInstance)//Class properties of the window
{
    WNDCLASSEXW wcex;

    wcex.cbSize = sizeof(WNDCLASSEX);

    wcex.style          = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = WndProc; //assigns a window to the instance of the class
    wcex.cbClsExtra     = 0;
    wcex.cbWndExtra     = 0;
    wcex.hInstance      = hInstance; //creates an instance
    wcex.hIcon          = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_RANKTOOLADVANCED));
    wcex.hCursor        = LoadCursor(nullptr, IDC_ARROW); //defines a type of cursor (arrow, help, cross, etc.)
    wcex.hbrBackground  = (HBRUSH)(COLOR_WINDOW+1); 
    wcex.lpszMenuName   = MAKEINTRESOURCEW(IDC_RANKTOOLADVANCED);
    wcex.lpszClassName  = szWindowClass;
    wcex.hIconSm        = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));

    return RegisterClassExW(&wcex);
}

//
//   FUNCTION: InitInstance(HINSTANCE, int)
//
//   PURPOSE: Saves instance handle and creates main window
//
//   COMMENTS:
//
//        In this function, we save the instance handle in a global variable and
//        create and display the main program window.
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{//initializes instance of window class (invocation)
   hInst = hInstance; // Store instance handle in our global variable

   HWND hWnd = CreateWindowW(szWindowClass, L"test", WS_OVERLAPPEDWINDOW,
      CW_USEDEFAULT, 0, 900, 600, nullptr, nullptr, hInstance, nullptr);

   if (!hWnd)
   {
      return FALSE;
   }

   ShowWindow(hWnd, nCmdShow);
   UpdateWindow(hWnd);

   return TRUE;
}

//
//  FUNCTION: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  PURPOSE: Processes messages for the main window.
//
//  WM_COMMAND  - process the application menu
//  WM_PAINT    - Paint the main window
//  WM_DESTROY  - post a quit message and return
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    case WM_COMMAND:
    {
        int wmId = LOWORD(wParam);
        // Parse the menu selections:
        switch (wmId)
        {/*
        case IDM_ABOUT:
            DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
            break;
        case IDM_EXIT:
            DestroyWindow(hWnd);
            break;
        default:
            return DefWindowProc(hWnd, message, wParam, lParam);
        }
        case FILE_MENU_NEW:
        {
            MessageBox(hWnd, L"task failed successfully", L"you done goofed", MB_OK | MB_ICONEXCLAMATION);
            break;
        }*/
        case FILE_MENU_DESTROY:
        {
            DestroyWindow(hWnd);
            break;
        }
        case FILE_MENU_ABOUT:
        {
            MessageBox(hWnd, L"about", L"About", MB_OK);
            break;
        }
        break;
        }
    }
    case WM_CREATE:
        {
            WinControls(hWnd);
            WinMenu(hWnd);
            break;
        }/*
    case WM_PAINT:
        {
            PAINTSTRUCT ps;
            HDC hdc = BeginPaint(hWnd, &ps);
            // TODO: Add any drawing code that uses hdc here...
            TextOut(hdc, 10, 10, rekt, _tcslen(rekt));
            TextOut(hdc, 10, 40, reverse, _tcslen(reverse));
            EndPaint(hWnd, &ps);
            break;
        }*/
    case WM_DESTROY:
    {
        PostQuitMessage(0);
        break;
    }
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

void WinMenu(HWND hWnd) {

    hMenu = CreateMenu();
    HMENU hFileMenu = CreateMenu();
    HMENU hSubMenu = CreateMenu();
    HMENU hSubMenu2 = CreateMenu();

    //XP Calculations
    AppendMenuW(hSubMenu, MF_STRING, NULL, L"Rank -> XP");
    AppendMenuW(hSubMenu, MF_STRING, NULL, L"XP -> Rank");

    //Credit Calculations
    AppendMenuW(hSubMenu2, MF_STRING, NULL, L"Rank -> Cred");
    AppendMenuW(hSubMenu2, MF_STRING, NULL, L"Cred -> Rank");



    AppendMenuW(hFileMenu, MF_POPUP, (UINT_PTR)hSubMenu, L"Points"); //option that popups submenu of hSubMenu
    AppendMenuW(hFileMenu, MF_POPUP, (UINT_PTR)hSubMenu2, L"Credits"); //option that popups submenu of hSubMenu
    AppendMenuW(hFileMenu, MF_SEPARATOR, NULL, NULL); // separator
    AppendMenuW(hFileMenu, MF_STRING, FILE_MENU_ABOUT, L"About");
    AppendMenuW(hFileMenu, MF_STRING, FILE_MENU_DESTROY, L"Exit"); // option 

    AppendMenuW(hMenu, MF_POPUP, (UINT_PTR)hFileMenu, L"Calculations"); //popups up submenu of hFileMenu

    SetMenu(hWnd, hMenu);
}

void WinControls(HWND hWnd) {
    CreateWindowW(L"Static", L"Enter text here: ", WS_VISIBLE | WS_CHILD | WS_BORDER | SS_CENTER, 400, 100, 100, 50, hWnd, NULL, NULL, NULL);
    CreateWindowW(L"Edit", L"...", WS_VISIBLE | WS_CHILD, 400, 155, 100, 50, hWnd, NULL, NULL, NULL);
}

Обратите внимание, что установка является базовой установкой Visual Studio для программы. Любая помощь приветствуется!


person Peter    schedule 29.03.2020    source источник
comment
Вы уверены, что проблема в CreateWindow, а не в SetMenu? Подменю должны быть всплывающими меню, а не меню строки меню.   -  person Raymond Chen    schedule 29.03.2020
comment
И есть ли причина создавать это меню в коде, а не как ресурс, назначенный классу окна?   -  person SoronelHaetir    schedule 29.03.2020
comment
@Raymond Chen Я попробую поиграться с SetMenu, а потом посмотрю, куда я могу пойти дальше. Кроме того, не могли бы вы подробнее рассказать о всплывающих меню для подменю?   -  person Peter    schedule 29.03.2020
comment
@SoronalHaetir Я не знаю, как использовать файл .rc, так как я совсем новичок в этом   -  person Peter    schedule 29.03.2020
comment
Итак, я повозился с кодом, и оказалось, что SetMenu() конфликтует с функцией CreateWindowW(). Комментирование SetMenu() создает меню по умолчанию и создает желаемые статические и редактируемые элементы, которые я хочу, но пользовательское меню не позволяет. Это весьма интересно...   -  person Peter    schedule 29.03.2020
comment
У вас нет break; в конце case WM_COMMAND:, поэтому, когда команда поступает из элемента управления редактированием, вы просто попадаете в case WM_CREATE.   -  person Raymond Chen    schedule 30.03.2020
comment
@RaymondChen Спасибо, что нашли это! Я исправил это сейчас   -  person Peter    schedule 31.03.2020


Ответы (1)


Я выполнил некоторую отладку и выяснил, что CreateWindowW(), который я использовал для статических элементов управления и элементов управления редактированием, конфликтует с определением свойств окна.

Поскольку окно было определено WNDCLASSEXW wcex, мне просто нужно было заменить CreateWindowW() на CreateWindowExW().

Единственная новая проблема заключается в том, что статический элемент управления (и другие) работает, но не элемент управления редактирования, который убивает меню.

person Peter    schedule 29.03.2020
comment
CreateWindow аналогичен CreatewindowEx с нулевыми расширенными стилями. И что вы подразумеваете под «определением свойств окна»? - person user2120666; 29.03.2020
comment
В идеальном мире мне хотелось бы думать, что я много знаю, но я все еще новичок в Win32 API, поэтому я действительно не знаю, как это работает, но я знаю, что могу его использовать (каким-то образом). . - person Peter; 29.03.2020