当前位置: 首页 > news >正文

GDI 区域检测与边框宽度的关系

GDI32中用PtInRegion检测坐标是否在图形区域内: 如果画了一个矩形,外边框线条粗细是10像素,检测区域时只能在第6像素开始才能反应,鼠标移动到边框线的1-5像素时,检测鼠标未在矩形区域

您遇到的问题是由于 GDI 中区域 (Region) 的定义与边框 (Stroke) 的渲染方式不同导致的。这是 GDI 绘图系统的一个常见特性,需要特别处理。

问题原因分析

  1. 区域 (Region) vs 边框 (Stroke)

    • CreateRectRgn 创建的区域是填充区域,不包含边框
    • 边框是绘制在区域边缘外部的,默认情况下不被视为区域的一部分
  2. 边框宽度的影响

    • 当您设置边框宽度为 10 像素时,这 10 像素是从区域边缘向外扩展的
    • 区域本身的边界并没有改变,因此PtInRegion只检测填充区域
  3. 坐标系统差异

    • 区域使用整数坐标,而边框渲染可能使用亚像素精度
    • 这种差异可能导致视觉上的边框与实际区域边界不一致

解决方案

方法一:创建包含边框的扩展区域vb
' API声明
Private Declare Function CreateRectRgn Lib "gdi32" (ByVal X1 As Long, ByVal Y1 As Long, ByVal X2 As Long, ByVal Y2 As Long) As Long
Private Declare Function CombineRgn Lib "gdi32" (ByVal hDestRgn As Long, ByVal hSrcRgn1 As Long, ByVal hSrcRgn2 As Long, ByVal nCombineMode As Long) As Long
Private Declare Function PtInRegion Lib "gdi32" (ByVal hRgn As Long, ByVal x As Long, ByVal y As Long) As Long
Private Declare Function DeleteObject Lib "gdi32" (ByVal hObject As Long) As LongPrivate Const RGN_OR = 2  ' 区域合并操作Private Sub CreateExtendedRegion(ByVal left As Long, ByVal top As Long, ByVal right As Long, ByVal bottom As Long, ByVal borderWidth As Long, ByRef hRegion As Long)Dim hFillRgn As LongDim hBorderRgn As Long' 创建填充区域hFillRgn = CreateRectRgn(left, top, right, bottom)' 创建扩展区域(包含边框)hBorderRgn = CreateRectRgn(left - borderWidth, top - borderWidth, _right + borderWidth, bottom + borderWidth)' 合并两个区域hRegion = CreateRectRgn(0, 0, 1, 1)  ' 创建临时区域CombineRgn hRegion, hFillRgn, hBorderRgn, RGN_OR' 释放临时区域DeleteObject hFillRgnDeleteObject hBorderRgn
End Sub' 使用示例
Private Sub Form_Load()Dim hRegion As LongDim borderWidth As LongborderWidth = 10' 创建包含10像素边框的区域CreateExtendedRegion 100, 100, 300, 200, borderWidth, hRegion' 现在可以使用这个区域进行鼠标检测' PtInRegion(hRegion, x, y) 将包含边框区域
End Sub
方法二:分别检测填充区域和边框区域vb
Private Sub MouseInBorderOrFill(ByVal x As Long, ByVal y As Long, ByVal left As Long, ByVal top As Long, ByVal right As Long, ByVal bottom As Long, ByVal borderWidth As Long, ByRef inBorder As Boolean, ByRef inFill As Boolean)' 检测是否在填充区域内inFill = (x >= left And x <= right And y >= top And y <= bottom)' 检测是否在边框区域内If Not inFill TheninBorder = (x >= left - borderWidth And x <= right + borderWidth And _y >= top - borderWidth And y <= bottom + borderWidth) And _Not inFillElseinBorder = FalseEnd If
End Sub' 使用示例
Private Sub Form_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)Dim pixelX As Long, pixelY As LongDim inBorder As Boolean, inFill As BooleanpixelX = X / Screen.TwipsPerPixelXpixelY = Y / Screen.TwipsPerPixelY' 检测鼠标是否在矩形(100,100,300,200)的填充区域或边框(宽度10)内MouseInBorderOrFill pixelX, pixelY, 100, 100, 300, 200, 10, inBorder, inFillIf inFill ThenMe.Caption = "鼠标在填充区域内"ElseIf inBorder ThenMe.Caption = "鼠标在边框区域内"ElseMe.Caption = "鼠标在区域外"End If
End Sub

方法三:使用 GDI + 进行精确检测

GDI + 提供了更精确的区域检测功能,可以直接包含边框:

' GDI+ API声明
Private Declare Function GdipCreateFromHDC Lib "gdiplus" (ByVal hdc As Long, ByRef graphics As Long) As Long
Private Declare Function GdipDeleteGraphics Lib "gdiplus" (ByVal graphics As Long) As Long
Private Declare Function GdipCreatePen1 Lib "gdiplus" (ByVal color As Long, ByVal width As Single, ByVal unit As Long, ByRef pen As Long) As Long
Private Declare Function GdipCreatePath Lib "gdiplus" (ByVal fillMode As Long, ByRef path As Long) As Long
Private Declare Function GdipAddPathRectangle Lib "gdiplus" (ByVal path As Long, ByVal x As Single, ByVal y As Single, ByVal width As Single, ByVal height As Single) As Long
Private Declare Function GdipIsVisiblePathPoint Lib "gdiplus" (ByVal x As Single, ByVal y As Single, ByVal path As Long, ByVal pen As Long, ByVal graphics As Long) As Long
Private Declare Function GdipDisposePath Lib "gdiplus" (ByVal path As Long) As Long
Private Declare Function GdipDisposePen Lib "gdiplus" (ByVal pen As Long) As LongPrivate Const UnitPixel As Long = 2Private Sub Form_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)Dim pixelX As Single, pixelY As SingleDim hGraphics As LongDim hPath As LongDim hPen As LongDim isVisible As LongpixelX = X / Screen.TwipsPerPixelXpixelY = Y / Screen.TwipsPerPixelY' 创建GDI+图形对象GdipCreateFromHDC Me.hdc, hGraphics' 创建路径和笔GdipCreatePath 0, hPath  ' 0 = FillModeAlternateGdipCreatePen1 &HFF0000, 10, UnitPixel, hPen  ' 10像素宽的笔' 添加矩形到路径GdipAddPathRectangle hPath, 100, 100, 200, 100' 检测点是否在路径(包括边框)上isVisible = GdipIsVisiblePathPoint(pixelX, pixelY, hPath, hPen, hGraphics)If isVisible <> 0 ThenMe.Caption = "鼠标在矩形(包括边框)内"ElseMe.Caption = "鼠标在矩形外"End If' 释放资源GdipDisposePath hPathGdipDisposePen hPenGdipDeleteGraphics hGraphics
End Sub

性能考虑

  1. 扩展区域方法:性能较高,但需要额外管理区域对象
  2. 分别检测方法:简单直接,适合小型应用
  3. GDI + 方法:最精确,但性能开销较大,适合对精度要求高的场景

根据您的具体需求选择合适的方法,可以有效解决边框检测问题。

http://www.xdnf.cn/news/1034551.html

相关文章:

  • 实习记录1
  • ImportError: DLL load failed while importing win32api: 找不到指定的模块
  • 18.vue.js的scoped样式隔离?原理和使用?(1)
  • 在线五子棋
  • 【Docker基础】Docker核心概念:命名空间(Namespace)与资源隔离联系
  • 从0开始学习R语言--Day23--稳健回归
  • 电路问题处理:SGMII链路中的AC耦合电容摆放位置
  • 轮廓 裂缝修复 轮廓修复 填补孔洞 源代码
  • 「Flink」Flink项目搭建方法介绍
  • 【飞牛os0.9.9系统使用docker 挂载cgroup2异常问题】
  • 傅里叶级数从三角函数形式到复指数形式的完整推导步骤
  • 位运,模拟,分治,BFS,栈和哈希表
  • Ant Design 版本演进详解:从 1.x 到 5.x 的发展历程
  • 【项目实训#09】智能代码文件助手模式前后端设计与实现
  • 读取配置文件到Settings对象的完整实现
  • synchronized和ReentrantLock的区别
  • gpt3大模型蒸馏后效果会变差么
  • HTTP 请求报文 方法
  • 湖北理元理律师事务所债务优化实务:平衡还款与生活的法律路径
  • 2022mpsPTE岗位笔试题
  • 自动化立体仓库堆垛机控制系统STEP7 FC1功能块 读取位置值SSI接口
  • KJY0047-J1阶段测试
  • 模拟 AJAX 提交 form 表单及请求头设置详解
  • 人工智能学习24-BP激活函数
  • 全球化2.0|云轴科技ZStack联合Teleplex举办技术沙龙 · 吉隆坡站
  • Oracle 单实例双IP配置
  • Qt:Qt桌面程序正常退出注意事项
  • 人工智能学习16-Numpy
  • YOLOv2 训练过程详解:从数据到模型落地的全流程解析
  • 设计模式-创建型模式(详解)