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

相关文章

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

504 Gateway Timeout网关超时的根源及完美解决方法

《504GatewayTimeout网关超时的根源及完美解决方法》在日常开发和运维过程中,504GatewayTimeout错误是常见的网络问题之一,尤其是在使用反向代理(如Nginx)或... 目录引言为什么会出现 504 错误?1. 探索 504 Gateway Timeout 错误的根源 1.1 后端

基于Python实现自动化邮件发送系统的完整指南

《基于Python实现自动化邮件发送系统的完整指南》在现代软件开发和自动化流程中,邮件通知是一个常见且实用的功能,无论是用于发送报告、告警信息还是用户提醒,通过Python实现自动化的邮件发送功能都能... 目录一、前言:二、项目概述三、配置文件 `.env` 解析四、代码结构解析1. 导入模块2. 加载环

linux系统上安装JDK8全过程

《linux系统上安装JDK8全过程》文章介绍安装JDK的必要性及Linux下JDK8的安装步骤,包括卸载旧版本、下载解压、配置环境变量等,强调开发需JDK,运行可选JRE,现JDK已集成JRE... 目录为什么要安装jdk?1.查看linux系统是否有自带的jdk:2.下载jdk压缩包2.解压3.配置环境

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消