Прикладом у цьому розділі використовує елементи меню власник звертається в меню. Елементи меню виберіть шрифт атрибути, і програма відображає кожного пункту меню за допомогою шрифту, який має відповідний атрибут. Наприклад, курсив меню відображається в курсив шрифту. Ім'я персонажа меню в рядку меню відкриває меню.
Рядок меню та розкривного меню визначаються спочатку розширений меню шаблон-ресурсу. Тому, що меню шаблон неможливо вказати звертається власник елементів, спочатку меню містить чотири пункти меню тексту з наступних рядків: "Звичайний", "напівжирний," "Курсив" та "Підкреслення". Вікно процедура зміни їх до власника звертається елементи під час обробки повідомлення WM_CREATE . Коли він отримує WM_CREATE повідомлення, вікно процедура називає визначений у застосунку OnCreate функцію, яка виконує такі дії для кожного пункту меню
Тому що вказівник на кожен ТОВАР визначений у застосунку структури зберігається як елемент даних, передається до вікна порядку разом з WM_MEASUREITEM та WM_DRAWITEM повідомлення для відповідного пункту меню. Курсор міститься в itemData членів як MEASUREITEMSTRUCT і DRAWITEMSTRUCT структур.
WM_MEASUREITEM надсилається повідомлення для кожного пункту меню власник звертається під час першого його відображення. Процеси в застосування цього повідомлення, вибравши шрифт для пункту меню в контексті пристрій і потім визначає простір, потрібний для відображення тексту пункт меню в цьому шрифті. Шрифт і меню елемент текст як визначаються пункт меню ТОВАР структури (структура, визначене застосунком). Програма визначає розмір тексту за допомогою в GetTextExtentPoint32 функції.
Процедура вікно процесів WM_DRAWITEM повідомлення на відображення тексту пункту меню у відповідний шрифт. Шрифт і меню елемент тексту як визначаються пункт меню ТОВАР структури. Застосування вибирає кольори тексту й фону відповідний пункт меню стан.
Процедура вікно обробляє WM_DESTROY повідомлення, щоб знищити шрифти та вільної пам'яті. Застосування видаляє шрифт і звільняє визначений у застосунку ТОВАР структури для кожного пункту меню.
Нижче наводяться відповідні частини програми заголовка файлу.
/ / Ідентифікатори пункт меню для "Символ" # визначити IDM_CHARACTER 10 # визначити IDM_REGULAR 11 # визначити IDM_BOLD 12 # визначити IDM_ITALIC 13 # визначити IDM_UNDERLINE 14 / / структура пов'язаних з меню елементів typedef типу struct tagMYITEM {HFONT hfont;
Int cchItemText;
CHAR szItemText [1];
} ТОВАР;
# визначити CCH_MAXITEMTEXT 256
Нижче наводяться відповідні частини вікна процедура та її пов'язані завдання.
LRESULT CALLBACK MainWindowProc(
HWND hwnd,
UINT uMsg,
WPARAM wParam,
LPARAM lParam
)
{
switch (uMsg)
{
case WM_CREATE:
if (!OnCreate(hwnd))
return -1;
break;
case WM_DESTROY:
OnDestroy(hwnd);
PostQuitMessage(0);
break;
case WM_MEASUREITEM:
OnMeasureItem(hwnd, (LPMEASUREITEMSTRUCT) lParam);
return TRUE;
case WM_DRAWITEM:
OnDrawItem(hwnd, (LPDRAWITEMSTRUCT) lParam);
return TRUE;
// Additional message processing goes here.
default:
return DefWindowProc(hwnd, uMsg, wParam, lParam);
}
return 0;
}
BOOL WINAPI OnCreate(HWND hwnd)
{
HMENU hmenuBar = GetMenu(hwnd);
HMENU hmenuPopup;
MENUITEMINFO mii;
UINT uID;
MYITEM *pMyItem;
// Get the handle to the pop-up menu.
mii.fMask = MIIM_SUBMENU; // information to get
GetMenuItemInfo(hmenuBar, IDM_CHARACTER, FALSE, &mii);
hmenuPopup = mii.hSubMenu;
// Modify each menu item. Assume that the IDs IDM_REGULAR
// through IDM_UNDERLINE are consecutive numbers.
for (uID = IDM_REGULAR; uID <= IDM_UNDERLINE; uID++)
{
// Allocate an item structure, leaving space for a
// string of up to CCH_MAXITEMTEXT characters.
pMyItem = (MYITEM *) LocalAlloc(LMEM_FIXED,
sizeof(MYITEM) + CCH_MAXITEMTEXT);
// Save the item text in the item structure.
mii.fMask = MIIM_TYPE;
mii.dwTypeData = pMyItem->szItemText;
mii.cch = CCH_MAXITEMTEXT;
GetMenuItemInfo(hmenuPopup, uID, FALSE, &mii);
pMyItem->cchItemText = mii.cch;
// Reallocate the structure to the minimum required size.
pMyItem = (MYITEM *) LocalReAlloc(pMyItem,
sizeof(MYITEM) + mii.cch, LMEM_MOVEABLE);
// Create the font used to draw the item.
pMyItem->hfont = CreateMenuItemFont(uID);
// Change the item to an owner-drawn item, and save
// the address of the item structure as item data.
mii.fMask = MIIM_TYPE | MIIM_DATA;
mii.fType = MFT_OWNERDRAW;
mii.dwItemData = (DWORD) pMyItem;
SetMenuItemInfo(hmenuPopup, uID, FALSE, &mii);
}
return TRUE;
}
HFONT CreateMenuItemFont(UINT uID)
{
LOGFONT lf;
ZeroMemory(&lf, sizeof(lf));
lf.lfHeight = 20;
lstrcpy(lf.lfFaceName, "Times New Roman");
switch (uID)
{
case IDM_BOLD:
lf.lfWeight = FW_HEAVY;
break;
case IDM_ITALIC:
lf.lfItalic = TRUE;
break;
case IDM_UNDERLINE:
lf.lfUnderline = TRUE;
break;
}
return CreateFontIndirect(&lf);
}
VOID WINAPI OnDestroy(HWND hwnd)
{
HMENU hmenuBar = GetMenu(hwnd);
HMENU hmenuPopup;
MENUITEMINFO mii;
UINT uID;
MYITEM *pMyItem;
// Get the handle to the menu.
mii.fMask = MIIM_SUBMENU; // information to get
GetMenuItemInfo(hmenuBar, IDM_CHARACTER, FALSE, &mii);
hmenuPopup = mii.hSubMenu;
// Free resources associated with each menu item.
for (uID = IDM_REGULAR; uID <= IDM_UNDERLINE; uID++)
{
// Get the item data.
mii.fMask = MIIM_DATA;
GetMenuItemInfo(hmenuPopup, uID, FALSE, &mii);
pMyItem = (MYITEM *) mii.dwItemData;
// Destroy the font and free the item structure.
DeleteObject(pMyItem->hfont);
LocalFree(pMyItem);
}
}
VOID WINAPI OnMeasureItem(HWND hwnd, LPMEASUREITEMSTRUCT lpmis)
{
MYITEM *pMyItem = (MYITEM *) lpmis->itemData;
HDC hdc = GetDC(hwnd);
HFONT hfntOld = SelectObject(hdc, pMyItem->hfont);
SIZE size;
GetTextExtentPoint32(hdc, pMyItem->szItemText,
pMyItem->cchItemText, &size);
lpmis->itemWidth = size.cx;
lpmis->itemHeight = size.cy;
SelectObject(hdc, hfntOld);
ReleaseDC(hwnd, hdc);
}
VOID WINAPI OnDrawItem(HWND hwnd, LPDRAWITEMSTRUCT lpdis)
{
MYITEM *pMyItem = (MYITEM *) lpdis->itemData;
COLORREF clrPrevText, clrPrevBkgnd;
HFONT hfntPrev;
int x, y;
// Set the appropriate foreground and background colors.
if (lpdis->itemState & ODS_SELECTED)
{
clrPrevText = SetTextColor(lpdis->hDC,
GetSysColor(COLOR_HIGHLIGHTTEXT));
clrPrevBkgnd = SetBkColor(lpdis->hDC,
GetSysColor(COLOR_HIGHLIGHT));
}
else
{
clrPrevText = SetTextColor(lpdis->hDC,
GetSysColor(COLOR_MENUTEXT));
clrPrevBkgnd = SetBkColor(lpdis->hDC,
GetSysColor(COLOR_MENU));
}
// Determine where to draw and leave space for a check-mark.
x = lpdis->rcItem.left;
y = lpdis->rcItem.top;
x += GetSystemMetrics(SM_CXMENUCHECK);
// Select the font and draw the text.
hfntPrev = SelectObject(lpdis->hDC, pMyItem->hfont);
ExtTextOut(lpdis->hDC, x, y, ETO_OPAQUE,
&lpdis->rcItem, pMyItem->szItemText,
pMyItem->cchItemText, NULL);
// Restore the original font and colors.
SelectObject(lpdis->hDC, hfntPrev);
SetTextColor(lpdis->hDC, clrPrevText);
SetBkColor(lpdis->hDC, clrPrevBkgnd);
}