PowerBASIC之系统菜单的自绘方法

2023-10-19 07:59

本文主要是介绍PowerBASIC之系统菜单的自绘方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、前言:

当你在单击窗口标题栏图标或点击鼠标右键时,系统会弹出一个默认菜单,如果在我们自己的程序中需要统一界面风格,则需要对这个弹出菜单进行些美化处理。

先上效果图,下面再演示实现方法:

二、实现方法:

1、首先初始化这个菜单,修改背景及添加MFT_OWNERDRAW类型风格。

初始化步骤可以是在%WM_INITDIALOG或者%WM_INITMENU消息中完成,我这里选择是在%WM_INITMENU时,对系统菜单进行初始化工作。

GLOBAL oldID AS LONG
GLOBAL oldText AS RECT
GLOBAL oldRSelect AS RECT
GLOBAL oldLSelect AS RECT
GLOBAL oldIcon AS DWORD CASE %WM_INITMENULOCAL lpMenuInfo AS MENUINFO LOCAL mii AS MENUITEMINFOLOCAL hMenu AS DWORDhMenu = GetSystemMenu ( CB.HNDL, %FALSE ) '取得系统菜单句柄'设置默认背景色lpMenuInfo.cbSize = SIZEOF(MENUINFO)lpMenuInfo.fMask = %MIM_BACKGROUNDlpMenuInfo.hbrBack = CreateSolidBrush(RGB(46,46,46)) '背景色SetMenuInfo ( hMenu, lpMenuInfo )DrawMenuBar ( CB.HNDL )'设置菜单栏项目为自绘模式ZeroMemory BYVAL VARPTR(lpMenuInfo), SIZEOF(MENUINFO)    mii.cbSize = SIZEOF(mii)  mii.fMask = %MIIM_FTYPE OR %MIIM_BITMAP OR %MIIM_DATA   mii.fType = %MFT_OWNERDRAWmii.hbmpItem = %HBMMENU_CALLBACK FOR i = 0 TO GetMenuItemCount ( hMenu ) - 1 IF i=0 THENmii.dwItemData = LoadImage(%NULL, "icon\menu_restore_a.ico", %IMAGE_ICON, 16, 16, %LR_DEFAULTCOLOR OR %LR_CREATEDIBSECTION OR %LR_LOADFROMFILE )END IFIF i=3 THENmii.dwItemData = LoadImage(%NULL, "icon\menu_min_a.ico", %IMAGE_ICON, 16, 16, %LR_DEFAULTCOLOR OR %LR_CREATEDIBSECTION OR %LR_LOADFROMFILE )END IFIF i=4 THENmii.dwItemData = LoadImage(%NULL, "icon\menu_max_a.ico", %IMAGE_ICON, 16, 16, %LR_DEFAULTCOLOR OR %LR_CREATEDIBSECTION OR %LR_LOADFROMFILE )END IFIF i=6 THENmii.dwItemData = LoadImage(%NULL, "icon\menu_exit_a.ico", %IMAGE_ICON, 16, 16, %LR_DEFAULTCOLOR OR %LR_CREATEDIBSECTION OR %LR_LOADFROMFILE )END IF  SetMenuItemInfo ( hMenu, i, %true, mii )NEXT iDrawMenuBar ( CB.HNDL ) oldID = 0 oldIcon = 0ZeroMemory BYVAL VARPTR(oldText), SIZEOF(oldText)ZeroMemory BYVAL VARPTR(oldRSelect), SIZEOF(oldRSelect)ZeroMemory BYVAL VARPTR(oldLSelect), SIZEOF(oldLSelect)FUNCTION = %TRUEEXIT FUNCTION

2、初始化菜单项规格。在菜单收到MFT_OWNERDRAW标识通知后,便可以接收到 %WM_DRAWITEM 及 %WM_MEASUREITEM 消息,其中%WM_MEASUREITEM 消息下完成菜单项宽、高的定义。

CASE %WM_MEASUREITEMLOCAL pdis AS MEASUREITEMSTRUCT PTRpdis = CB.LPARAMIF @pdis.CtlType = %ODT_MENU THEN@pdis.itemWidth  = 161@pdis.itemHeight = 20END IFFUNCTION = %TRUEEXIT FUNCTION 

3、完成上述两个步骤后,我既可以开始在 %WM_DRAWITEM消息体下,去绘制自己设定的样式风格了(直接在HDC上绘制即可)。

CASE %WM_DRAWITEM LOCAL hvBrush AS DWORDLOCAL lpdis AS DRAWITEMSTRUCT PTRlpdis = CB.LPARAMIF @lpdis.CtlType = %ODT_MENU THENLOCAL hFont AS DWORDLOCAL ItemCount AS LONG  LOCAL MenuWidth AS LONGLOCAL MenuHigh AS LONG     LOCAL rcRight AS RECTLOCAL rcLeft AS RECTLOCAL rcRSelect AS RECTLOCAL rcLSelect AS RECT hMenu = @lpdis.hwndItemMenuWidth = @lpdis.rcItem.right-@lpdis.rcItem.leftMenuHigh = @lpdis.rcItem.bottom-@lpdis.rcItem.top  ItemCount = GetMenuItemCount ( hMenu )hFont = GetStockObject( %DEFAULT_GUI_FONT ) '菜单字体SelectObject ( @lpdis.hdc, hFont )SetBkMode @lpdis.hdc, %TRANSPARENT SetTextColor @lpdis.hdc, RGB(112,112,112) '-------------------------------'绘制左MenuBar'-------------------------------SetRect(rcLeft, 0, @lpdis.rcItem.top, 28, @lpdis.rcItem.bottom)hvBrush = CreateSolidBrush(RGB(113,96,232))FillRect(@lpdis.hdc,rcLeft,hvBrush)DeleteObject hvBrushSELECT CASE @lpdis.itemIDCASE %SC_RESTOREDrawIconEx(@lpdis.hdc, 7, 2, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_MINIMIZEDrawIconEx(@lpdis.hdc, 7, 62, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_MAXIMIZEDrawIconEx(@lpdis.hdc, 7, 82, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_CLOSEDrawIconEx(@lpdis.hdc, 7, 122, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )END SELECT   '-------------------------------'绘制右文本区'-------------------------------SetRect(rcRSelect, 30, @lpdis.rcItem.top, MenuWidth, @lpdis.rcItem.bottom)SetRect(rcRight, 32, @lpdis.rcItem.top, MenuWidth, @lpdis.rcItem.bottom)hvBrush = CreateSolidBrush(RGB(46,46,46)) '背景色FillRect(@lpdis.hdc,rcRight,hvBrush)DeleteObject hvBrushGetMenuString( hMenu, @lpdis.itemID, czText, 256, %MF_BYCOMMAND )ReplaceHotkeys(VARPTR(czText),LEN(czText))DrawText(@lpdis.hdc, czText, -1, rcRight, %DT_SINGLELINE OR %DT_LEFT OR %DT_VCENTER )    '-------------------------------'绘制分割线'-------------------------------IF @lpdis.itemID = 0 THENhvBrush = CreateSolidBrush(RGB(112,112,112))                    SetRect(rc,30,@lpdis.rcItem.top+10,@lpdis.rcItem.right,@lpdis.rcItem.top+11)FillRect(@lpdis.hdc,rc,hvBrush)DeleteObject hvBrushEND IF '-------------------------------'绘制选择菜单项时的鼠标移动效果'-------------------------------IF @lpdis.itemAction = %ODA_SELECT AND @lpdis.itemID > 0 THEN'--绘制左半部选项条--hvBrush = CreateSolidBrush(RGB(255,0,0))FillRect(@lpdis.hdc,rcRSelect,hvBrush)DeleteObject hvBrush'--绘制右半部选项条--hvBrush = CreateSolidBrush(BGR(220,240,120))FillRect(@lpdis.hdc,rcLeft,hvBrush)DeleteObject hvBrush'--绘制文本--SetTextColor @lpdis.hdc, RGB(255,255,255)GetMenuString( hMenu, @lpdis.itemID, czText, 256, %MF_BYCOMMAND )DrawText( @lpdis.hdc, czText, -1, rcRight, %DT_SINGLELINE OR %DT_LEFT OR %DT_VCENTER ) '--绘制选项条左部图标-- SELECT CASE @lpdis.itemIDCASE %SC_RESTOREDrawIconEx(@lpdis.hdc, 7, 2, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_MINIMIZEDrawIconEx(@lpdis.hdc, 7, 62, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_MAXIMIZEDrawIconEx(@lpdis.hdc, 7, 82, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE %SC_CLOSEDrawIconEx(@lpdis.hdc, 7, 122, @lpdis.itemData, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )END SELECT '--恢复鼠标移动后,上一个选项的背景及文本--hvBrush = CreateSolidBrush(RGB(46,46,46)) '背景色FillRect(@lpdis.hdc,oldRSelect,hvBrush)DeleteObject hvBrushhvBrush = CreateSolidBrush(RGB(113,96,232)) 'FillRect(@lpdis.hdc,oldLSelect,hvBrush)DeleteObject hvBrush'--绘制文本--SetTextColor @lpdis.hdc, RGB(112,112,112)GetMenuString( hMenu, oldID, czText, 256, %MF_BYPOSITION )DrawText( @lpdis.hdc, czText, -1, oldText, %DT_SINGLELINE OR %DT_LEFT OR %DT_VCENTER )'--绘制选项条左部移动后上一个图标-- SELECT CASE oldIDCASE 0DrawIconEx(@lpdis.hdc, 7, 2, oldIcon, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE 3DrawIconEx(@lpdis.hdc, 7, 62, oldIcon, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE 4DrawIconEx(@lpdis.hdc, 7, 82, oldIcon, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )CASE 6DrawIconEx(@lpdis.hdc, 7, 122, oldIcon, 0, 0, 0, %NULL, %DI_NORMAL OR %DI_COMPAT )END SELECT  oldID = GetMenuItemPos(hMenu,@lpdis.itemID)oldIcon = @lpdis.itemData CopyRect(oldText,rcRight)CopyRect(oldRSelect,rcRSelect)CopyRect(oldLSelect,rcLeft)                   END IFDeleteObject hFont      END IF  FUNCTION = %TRUEEXIT FUNCTION
'-------------------------------
'取消快捷键定义字母下划线
'-------------------------------
SUB ReplaceHotkeys(BYVAL s AS BYTE PTR, BYVAL k AS LONG)DIM i AS INTEGERFOR i = 0 TO kIF @s[i] = 38 THEN@s[i] = 9END IFNEXT iEND SUB'-------------------------------
'根据ID返回序号
'-------------------------------
FUNCTION GetMenuItemPos(BYVAL hMenu AS DWORD, ItemID AS DWORD) AS LONGLOCAL i AS LONGLOCAL ItemCount AS LONGItemCount = GetMenuItemCount ( hMenu ) - 1FOR i=0 TO ItemCountIF GetMenuItemID(hMenu,i)=ItemID THENFUNCTION = iEXIT FUNCTIONEND IFNEXT i
END FUNCTION 

 

 三、自此一个具有个性化的标题菜单就绘制完成了。由于水平有限代码未必严谨,仅供大家交流学习。

这篇关于PowerBASIC之系统菜单的自绘方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/238482

相关文章

检查 Nginx 是否启动的几种方法

《检查Nginx是否启动的几种方法》本文主要介绍了检查Nginx是否启动的几种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1. 使用 systemctl 命令(推荐)2. 使用 service 命令3. 检查进程是否存在4

Java方法重载与重写之同名方法的双面魔法(最新整理)

《Java方法重载与重写之同名方法的双面魔法(最新整理)》文章介绍了Java中的方法重载Overloading和方法重写Overriding的区别联系,方法重载是指在同一个类中,允许存在多个方法名相同... 目录Java方法重载与重写:同名方法的双面魔法方法重载(Overloading):同门师兄弟的不同绝

MySQL字符串转数值的方法全解析

《MySQL字符串转数值的方法全解析》在MySQL开发中,字符串与数值的转换是高频操作,本文从隐式转换原理、显式转换方法、典型场景案例、风险防控四个维度系统梳理,助您精准掌握这一核心技能,需要的朋友可... 目录一、隐式转换:自动但需警惕的&ld编程quo;双刃剑”二、显式转换:三大核心方法详解三、典型场景

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

springboot中配置logback-spring.xml的方法

《springboot中配置logback-spring.xml的方法》文章介绍了如何在SpringBoot项目中配置logback-spring.xml文件来进行日志管理,包括如何定义日志输出方式、... 目录一、在src/main/resources目录下,也就是在classpath路径下创建logba

SQL Server中行转列方法详细讲解

《SQLServer中行转列方法详细讲解》SQL行转列、列转行可以帮助我们更方便地处理数据,生成需要的报表和结果集,:本文主要介绍SQLServer中行转列方法的相关资料,需要的朋友可以参考下... 目录前言一、为什么需要行转列二、行转列的基本概念三、使用PIVOT运算符进行行转列1.创建示例数据表并插入数

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias

python项目打包成docker容器镜像的两种方法实现

《python项目打包成docker容器镜像的两种方法实现》本文介绍两种将Python项目打包为Docker镜像的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目录简单版:(一次成功,后续下载对应的软件依赖)第一步:肯定是构建dockerfile,如下:第二步

C# GC回收的方法实现

《C#GC回收的方法实现》本文主要介绍了C#GC回收的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、什么是 GC? 二、GC 管理的是哪部分内存? 三、GC 什么时候触发?️ 四、GC 如何判断一个对象是“垃圾