C++windows内核编程笔记day06 代码创建菜单

简介: 创建菜单: HMENU CreateMenu(VOID); 添加菜单项: BOOL AppendMenu(  HMENU hMenu,         // handle to menu   UINT uFlags,         ...

创建菜单:
HMENU CreateMenu(VOID);
添加菜单项:
BOOL AppendMenu(  HMENU hMenu,         // handle to menu
  UINT uFlags,         // menu-item options
  UINT_PTR uIDNewItem, // identifier, menu, or submenu
  LPCTSTR lpNewItem    // menu-item content
  );
 uFlags:
    MF_POPUP  -有子菜单的菜单项目,uIDNewItem要设置为子菜单的句柄(转换为uint)
    MF_SEPARATOR -分隔线
    MF_STRING    -一般菜单项,被点击后会发送消息 WM_COMMAND
    MF_MENUBREAK -换一列

把菜单加到窗口:
BOOL SetMenu(  HWND hWnd,  // handle to window  HMENU hMenu // handle to menu
);
//修改菜单的勾选状态
DWORD CheckMenuItem(  HMENU hmenu,        // handle to menu
  UINT uIDCheckItem,  // menu item to check or uncheck
  UINT uCheck         // menu item options
  );
uCheck:
    MF_BYCOMMAND  -填写菜单id
    MF_BYPOSITION -填写菜单索引
    MF_CHECKED/MF_UNCHECKED //1/0 --是否选中

//触发菜单点击事件
case WM_COMMAND:
    On_Command(hwnd,wparam);
    break;
//点击事件处理
void On_Command(HWND hwnd,WPARAM wparam)
{
    bool IsFromMenu=(HIWORD(wparam)==0?true:false);//点击菜单:0,加速键:1
    unsigned int id=LOWORD(wparam);//被点的控件id
    switch(id)
    {
    case 10011:
        //MessageBox(NULL,"打开","info",MB_OK);
        if(isChecked)
        {
            CheckMenuItem(menu1,id,MF_BYCOMMAND|MF_UNCHECKED);isChecked=false;
        }
        else{  CheckMenuItem(menu1,id,MF_BYCOMMAND|MF_CHECKED);isChecked=true;
        //MessageBox(NULL,"1","info",MB_OK);
        }
        break;
    }
}
//设置可用或不可用,与CheckMenuItem用法一样
BOOL EnableMenuItem(HMENU hMenu,// handle to menu
  UINT uIDEnableItem,  // menu item to update
  UINT uEnable         // options
  );
WM_INITMENUPOPUP  //弹出菜单在被点击还未弹出来之前触发的消息
LRESULT CALLBACK WindowProc(
  HWND hwnd,       // handle to window
  UINT uMsg,       // WM_INITMENUPOPUP
  WPARAM wParam,   // handle to menu (HMENU)
  LPARAM lParam    // item position and indicator
);
lParam:
    LOWORD  被点击菜单的索引
    HIWORD  将要显示的菜单是否为窗口菜单(弹出式菜单不属于窗口菜单)

系统菜单的操作,消息:WM_SYSCOMMAND,不太重要,一般不用。
HMENU GetSystemMenu(  HWND hWnd,    // handle to window
  BOOL bRevert  // 是否返回默认菜单
  );
  AppendMenu //前面用过
 //删除菜单,根据位置删除时,删除前一个,后面一个会改变位置。
 BOOL DeleteMenu(  HMENU hMenu,     // handle to menu
  UINT uPosition,  // menu item identifier or position
  UINT uFlags      // option(MF_BYPOSITION/MF_BYCOMMAND)
  );

右键菜单:弹出式菜单,前面用过,这里主要在右键时显示菜单。
BOOL TrackPopupMenu(
  HMENU hMenu,         // handle to shortcut menu
  UINT uFlags,         // options
  int x,               // horizontal position
  int y,               // vertical position
  int nReserved,       // reserved, must be zero
  HWND hWnd,           // handle to owner window
  CONST RECT *prcRect  // ignored
);
uFlags:
    TPM_RETURNCMD  -点击菜单项,不发出WM_COMMAND消息,但有返回值(被点击菜单的ID)

//右键消息处理函数,也可以在WM_CONTEXTMENU消息中处理(专门处理)
//WM_CONTEXTMENU 在 WM_RBUTTONUP之后触发的
//WM_CONTEXTMENU 中的坐标不需要转换
void OnRButtonUp(HWND hwnd,LPARAM lparam)
{
    HMENU menucontext=CreatePopupMenu();
    AppendMenu(menucontext,MF_STRING,103,"查看");
    AppendMenu(menucontext,MF_STRING,104,"放大");
    AppendMenu(menucontext,MF_STRING,105,"缩小");
    POINT pt={0};
    pt.y=HIWORD(lparam);
    pt.x=LOWORD(lparam);
    ClientToScreen(hwnd,&pt);//将客户区坐标转换为相对于屏幕的坐标
    int cmdid= TrackPopupMenu(menucontext,TPM_LEFTALIGN|TPM_LEFTBUTTON|TPM_RETURNCMD,pt.x,pt.y,0,
        hwnd,NULL);//在鼠标位置显示菜单
    char txt[30]={0};
    sprintf(txt,"被点的菜单ID:%d\n",cmdid);
    MessageBox(hwnd,txt,"info",MB_OK);
}

ScreenToClient();//坐标转换函数


代码示例:

同样创建win32 application

主要cpp代码修改为:

// win32app.cpp : Defines the entry point for the application.
//

#include "stdafx.h"
#include <stdio.h>
HINSTANCE g_hinstance=0;//全局句柄
HMENU menu1=0,menu2=0;
HANDLE g_houtput=0;
void OnCreate(HWND hwnd,LPARAM lparam)
{
	HMENU menu=CreateMenu();
	menu1=CreatePopupMenu();
	AppendMenu(menu1,MF_STRING|MF_CHECKED,10011,"打开");
	AppendMenu(menu1,MF_STRING,10012,"保存");
	AppendMenu(menu1,MF_SEPARATOR,0,NULL);
	AppendMenu(menu1,MF_STRING|MF_MENUBREAK,10013,"退出");
	menu2=CreatePopupMenu();
	AppendMenu(menu2,MF_STRING,10021,"复制");
	AppendMenu(menu2,MF_STRING|MF_GRAYED,10022,"粘贴");

	AppendMenu(menu,MF_POPUP,(UINT)menu1,"文件");
	AppendMenu(menu,MF_SEPARATOR,0,NULL);
	AppendMenu(menu,MF_POPUP,(UINT)menu2,"编辑");
	SetMenu(hwnd,menu);

	
   HMENU menusys=GetSystemMenu(hwnd,FALSE);
   for(int i=0;i<6;i++)
   {
	  // DeleteMenu(menusys,0,MF_BYPOSITION);//删除系统菜单项
   }
   AppendMenu(menusys,MF_STRING,101,"我的菜单");

}
LRESULT CALLBACK OnInitMunuPopUp(
							HWND hwnd,       // handle to window
							UINT uMsg,       // WM_INITMENUPOPUP
							WPARAM wParam,   // handle to menu (HMENU)
							LPARAM lParam    // item position and indicator
)
{
	unsigned int index=LOWORD(lParam);
	int IsWindowMenu=HIWORD(lParam);
	char txt[200]={0};
	sprintf(txt,"OnInitMenuPopUp:menu:%d,Index:%d,IsWindow:%d\n",wParam,index,IsWindowMenu);
	WriteConsole(g_houtput,txt,strlen(txt),NULL,NULL);
	return 0;
}
bool isChecked=true;
void On_Command(HWND hwnd,WPARAM wparam)
{
	bool IsFromMenu=(HIWORD(wparam)==0?true:false);
	unsigned int id=LOWORD(wparam);
	switch(id)
	{
	case 10011:
	
		//MessageBox(NULL,"打开","info",MB_OK);
		if(isChecked)
		{
			CheckMenuItem(menu1,id,MF_BYCOMMAND|MF_UNCHECKED);isChecked=false;
		}
		else{  CheckMenuItem(menu1,id,MF_BYCOMMAND|MF_CHECKED);isChecked=true;
		//MessageBox(NULL,"1","info",MB_OK);
		}
		break;
	}
}
void OnSysCommand(HWND hwnd,WPARAM wparam)
{
	bool IsFromMenu=(HIWORD(wparam)==0?true:false);
	unsigned int id=LOWORD(wparam);
	switch(id)
	{
	case 101:
		MessageBox(hwnd,"我的菜单","info",MB_OK);
		break;
	default:
		//MessageBox(hwnd,"我的菜单1","info",MB_OK);
		break;
	}
}
void OnRButtonUp(HWND hwnd,LPARAM lparam)
{
// 	HMENU menucontext=CreatePopupMenu();
// 	AppendMenu(menucontext,MF_STRING,103,"查看");
// 	AppendMenu(menucontext,MF_STRING,104,"放大");
// 	AppendMenu(menucontext,MF_STRING,105,"缩小");
// 	POINT pt={0};
// 	pt.y=HIWORD(lparam);
// 	pt.x=LOWORD(lparam);
// 	ClientToScreen(hwnd,&pt);//将客户区坐标转换为相对于屏幕的坐标
// 	int cmdid= TrackPopupMenu(menucontext,TPM_LEFTALIGN|TPM_LEFTBUTTON|TPM_RETURNCMD,pt.x,pt.y,0,
// 		hwnd,NULL);//在鼠标位置显示菜单
// 	char txt[30]={0};
// 	sprintf(txt,"被点的菜单ID:%d\n",cmdid);
// 	MessageBox(hwnd,txt,"info",MB_OK);
}
void OnContextMenu(HWND hwnd,LPARAM lparam)
{
	 	HMENU menucontext=CreatePopupMenu();
	 	AppendMenu(menucontext,MF_STRING,103,"查看");
	 	AppendMenu(menucontext,MF_STRING,104,"放大");
	 	AppendMenu(menucontext,MF_STRING,105,"缩小");
	 	POINT pt={0};
	 	pt.y=HIWORD(lparam);
	 	pt.x=LOWORD(lparam);
	 	//ClientToScreen(hwnd,&pt);//将客户区坐标转换为相对于屏幕的坐标
	 	int cmdid= TrackPopupMenu(menucontext,TPM_LEFTALIGN|TPM_LEFTBUTTON|TPM_RETURNCMD,pt.x,pt.y,0,
	 		hwnd,NULL);//在鼠标位置显示菜单
		if(cmdid!=0){
	 	char txt[30]={0};
	 	sprintf(txt,"被点的菜单ID:%d\n",cmdid);
	 	MessageBox(hwnd,txt,"info",MB_OK);
		}
}
//回调函数
LRESULT CALLBACK WinProc(HWND hwnd,UINT msg,WPARAM wparam,LPARAM lparam)
{
	switch(msg)
	{
	case WM_SYSCOMMAND:
		if(wparam==SC_CLOSE){
			int ret=MessageBox(NULL,"是否退出","info",MB_YESNO);
			if(ret==IDYES){
				//下面代码会自动关闭和销毁
				//PostQuitMessage(0);
			}
			else return 0;//不执行下面代码
		}
		OnSysCommand(hwnd,wparam);
		break;
		//在创建窗口之后还未显示的时候
	case WM_CREATE:
		OnCreate(hwnd,lparam);
		break;
	case WM_DESTROY:
		PostQuitMessage(0);
		break;
	case WM_COMMAND:
		On_Command(hwnd,wparam);
		break;
	case WM_INITMENUPOPUP:
		OnInitMunuPopUp(hwnd,msg,wparam,lparam);
		break;
	case WM_RBUTTONUP:
		OnRButtonUp(hwnd,lparam);
		break;
	case WM_CONTEXTMENU:
		OnContextMenu(hwnd,lparam);
		break;
	}
	return DefWindowProc(hwnd,msg,wparam,lparam);
}
//注册窗口类
BOOL Register(LPSTR lpClassName,WNDPROC wndproc)
{
	WNDCLASSEX wce={0};
	wce.cbSize=sizeof(wce);
	wce.cbClsExtra=200;
	wce.cbWndExtra=200;
	wce.hbrBackground=(HBRUSH)(COLOR_WINDOW+1);
	wce.hCursor=NULL;
	wce.hIcon=NULL;
	wce.hIconSm=NULL;
	wce.hInstance=g_hinstance;
	wce.lpfnWndProc=wndproc;
	wce.lpszClassName=lpClassName;
	wce.lpszMenuName=NULL;
	wce.style=CS_HREDRAW|CS_VREDRAW;
	ATOM atom= RegisterClassEx(&wce);
	if(atom==0){
		MessageBox(NULL,"注册失败","info",MB_OK);
		return FALSE;
	}
	return TRUE;
}
//创建窗口
HWND CreateMain(LPSTR lpClassName,LPSTR lpWndName)
{
	HWND hwnd=CreateWindowEx(0,lpClassName,lpWndName,WS_OVERLAPPEDWINDOW,CW_USEDEFAULT,CW_USEDEFAULT,
		CW_USEDEFAULT,CW_USEDEFAULT,NULL,NULL,g_hinstance,"hello create");
	return hwnd;
}
//创建子窗口
HWND CreateChild(HWND phwnd,LPSTR lpClassName,LPSTR lpWndName)
{
	if(Register(lpClassName,DefWindowProc)==0)
	{
		MessageBox(phwnd,"创建子窗口失败","info",MB_OK);
		return NULL;
	}
	//子窗口风格,都要 WS_CHILD|WS_VISIBLE
	HWND hwnd=CreateWindowEx(0,lpClassName,lpWndName,WS_CHILD|WS_VISIBLE|WS_OVERLAPPEDWINDOW,CW_USEDEFAULT,CW_USEDEFAULT,
		200,200,phwnd,NULL,g_hinstance,NULL);
	return hwnd;
}
//显示窗口
void Display(HWND hwnd)
{
	ShowWindow(hwnd,SW_SHOW);
	UpdateWindow(hwnd);
}
//处理消息
void MSGdeal()
{
	MSG msg={0};
	while(GetMessage(&msg,NULL,0,0)){
		TranslateMessage(&msg);//翻译消息 
		DispatchMessage(&msg);//派发给 WinProc 处理消息 
	}
}



int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
	AllocConsole();//调试程序方法
	g_houtput=GetStdHandle(STD_OUTPUT_HANDLE);
	g_hinstance=hInstance;
 	// TODO: Place code here.
	if(Register("main",WinProc)==0)
	{
		MessageBox(NULL,"注册失败","提示",MB_OK);
		return 0;
	}
	HWND hwnd= CreateMain("main","pro1");
	
	Display(hwnd);

	MSGdeal();
	return 0;
}







相关文章
|
8月前
|
安全 生物认证 网络安全
windows10无法设置默认保存位置怎么办?显示错误代码0x80070002怎么解决?
Win10系统下载文件时,默认会保存在特定位置,但用户可自行修改。若更改后仍无效,可通过删除目标磁盘中的特定文件夹、修改注册表权限、“干净启动”排除干扰软件或使用第三方修复工具等方式解决此问题。
1464 0
|
5月前
|
Ubuntu API C++
C++标准库、Windows API及Ubuntu API的综合应用
总之,C++标准库、Windows API和Ubuntu API的综合应用是一项挑战性较大的任务,需要开发者具备跨平台编程的深入知识和丰富经验。通过合理的架构设计和有效的工具选择,可以在不同的操作系统平台上高效地开发和部署应用程序。
235 11
|
11月前
|
存储 算法 C++
Windows共享文件:探秘C++实现的B树索引算法奇境
在数字化时代,Windows共享文件的高效管理至关重要。B树算法以其自平衡多路搜索特性,在文件索引与存储优化中表现出色。本文探讨B树在Windows共享文件中的应用,通过C++实现具体代码,展示其构建文件索引、优化数据存储的能力,提升文件检索效率。B树通过减少磁盘I/O操作,确保查询高效,为企业和个人提供流畅的文件共享体验。
|
6月前
|
C++ Windows
应用程序无法正常启动(0xc0000005)?C++报错0xC0000005如何解决?使命召唤17频频出现闪退,错误代码0xC0000005(0x0)
简介: 本文介绍了Windows应用程序出现错误代码0xc0000005的解决方法,该错误多由C++运行库配置不一致或内存访问越界引起。提供包括统一运行库配置、调试排查及安装Visual C++运行库等解决方案,并附有修复工具下载链接。
1718 1
|
存储 安全 C语言
C++ String揭秘:写高效代码的关键
在C++编程中,字符串操作是不可避免的一部分。从简单的字符串拼接到复杂的文本处理,C++的string类为开发者提供了一种更高效、灵活且安全的方式来管理和操作字符串。本文将从基础操作入手,逐步揭开C++ string类的奥秘,帮助你深入理解其内部机制,并学会如何在实际开发中充分发挥其性能和优势。
|
7月前
|
Windows
office出现0xc0000142错误?windows错误代码为0xc0000142?
office出现0xc0000142错误?windows错误代码为0xc0000142?
335 0
|
8月前
|
API 数据安全/隐私保护 C++
永久修改机器码工具, exe一机一码破解工具,软件机器码一键修改工具【c++代码】
程序实现了完整的机器码修改功能,包含进程查找、内存扫描、模式匹配和修改操作。代码使用
|
9月前
|
C++
爱心代码 C++
这段C++代码使用EasyX图形库生成动态爱心图案。程序通过数学公式绘制爱心形状,并以帧动画形式呈现渐变效果。运行时需安装EasyX库,教程链接:http://【EasyX图形库的安装和使用】https://www.bilibili.com/video/BV1Xv4y1p7z1。代码中定义了屏幕尺寸、颜色数组等参数,利用随机数与数学函数生成动态点位,模拟爱心扩散与收缩动画,最终实现流畅的视觉效果。
1106 0
|
算法 安全 C++
提高C/C++代码的可读性
提高C/C++代码的可读性
332 4
|
Linux C语言 C++
vsCode远程执行c和c++代码并操控linux服务器完整教程
这篇文章提供了一个完整的教程,介绍如何在Visual Studio Code中配置和使用插件来远程执行C和C++代码,并操控Linux服务器,包括安装VSCode、安装插件、配置插件、配置编译工具、升级glibc和编写代码进行调试的步骤。
3462 0
vsCode远程执行c和c++代码并操控linux服务器完整教程