0

0

如何在 moderngl_window 中实现鼠标独占模式(锁定光标于窗口内)

霞舞

霞舞

发布时间:2026-02-12 09:06:29

|

663人浏览过

|

来源于php中文网

原创

如何在 moderngl_window 中实现鼠标独占模式(锁定光标于窗口内)

本文介绍如何使用 moderngl_window 的 `mouse_exclusivity` 属性实现真正的鼠标锁定(即光标不可移出窗口、不可见、仅提供相对位移),替代低效的轮询或外部工具方案,并附可运行示例与关键注意事项。

在基于 OpenGL 的交互式图形应用(如 Shader Toy 风格的片段着色器演示或轻量级 3D 查看器)中,实现「鼠标独占模式」(Mouse Exclusivity / Raw Input Mode)是构建沉浸式相机控制的基础——它让光标完全隐藏、不响应系统级边界限制,并持续输出平滑的相对位移(dx, dy),而非绝对屏幕坐标。这正是现代 3D 引擎(如 GLFW 默认行为)所依赖的核心机制。

moderngl_window 基于 GLFW 构建,原生支持该功能,无需切换库或引入 pyautogui 等易出错的模拟方案。关键在于启用 Window.mouse_exclusivity 属性,它会自动调用 GLFW 的 glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_DISABLED),并启用原始输入(raw mouse motion),确保跨平台(包括 Arch Linux + X11/Wayland)行为一致。

✅ 正确启用方式

只需在窗口配置类中动态设置 self.wnd.mouse_exclusivity = True 即可。推荐通过按键事件(如 C 键)实时切换,便于调试:

WHEE
WHEE

WHEE是一款AI绘画与图片生成器,提供一站式AI视觉创作服务。WHEE不仅会画也会修图,各种AI修图功能一应俱全。

下载
import moderngl_window as mgl

class App(mgl.WindowConfig):
    window_size = (1336, 768)
    resource_dir = "src"
    cursor = False  # 初始隐藏光标(非锁定!)
    fullscreen = False

    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        # 初始化你的着色器、几何体等...
        self.quad = mgl.geometry.quad_fs()
        self.prog = self.load_program(
            vertex_shader="vertex.glsl",
            fragment_shader="frag.glsl"
        )
        self.prog["resolution"] = self.window_size

    def key_event(self, key, action, modifiers):
        keys = self.wnd.keys
        if key == keys.C and action == keys.ACTION_PRESS:
            # 切换鼠标独占状态
            self.wnd.mouse_exclusivity = not self.wnd.mouse_exclusivity
            print(f"→ Mouse exclusivity: {self.wnd.mouse_exclusivity}")

    def mouse_position_event(self, x, y, dx, dy):
        # ⚠️ 注意:锁定后 x, y 恒为窗口中心(逻辑值),无实际意义
        # ✅ 唯一可靠的是 dx, dy —— 未经加速/缩放的原始像素位移
        if self.wnd.mouse_exclusivity:
            # 示例:用于第一人称相机 yaw/pitch 更新
            self._rotate_camera(dx * 0.002, dy * 0.002)

    def _rotate_camera(self, delta_yaw, delta_pitch):
        # 此处插入你的相机旋转逻辑(如更新 uniform)
        try:
            self.prog["camera_yaw"] = self.prog["camera_yaw"].value + delta_yaw
            self.prog["camera_pitch"] = max(-1.5, min(1.5, self.prog["camera_pitch"].value + delta_pitch))
        except KeyError:
            pass

    def render(self, time, frame_time):
        self.ctx.clear(0.1, 0.1, 0.15)
        self.prog["time"] = time
        self.quad.render(self.prog)

if __name__ == "__main__":
    print("? Press 'C' to toggle mouse lock/unlock")
    mgl.run_window_config(App)

⚠️ 关键注意事项

  • cursor = False ≠ mouse_exclusivity = True
    cursor = False 仅隐藏光标,但光标仍可移出窗口、触发系统焦点丢失;而 mouse_exclusivity = True 才真正捕获并锁定光标,同时隐式隐藏。

  • x, y 在锁定状态下失效
    GLFW 在独占模式下不提供绝对坐标(因光标被“固定”在窗口中心),因此 mouse_position_event 中的 x, y 是恒定的伪值(通常为窗口中心)。务必只使用 dx, dy 进行运动计算,它们代表未处理的原始输入增量,精度高且无系统指针加速干扰。

  • 退出锁定需显式恢复
    程序异常退出时,GLFW 通常能自动恢复光标,但为保险起见,建议在 close_event() 中添加 self.wnd.mouse_exclusivity = False 清理。

  • Linux 兼容性提示(Arch/X11/Wayland)
    moderngl_window 0.14+ 对 Wayland 支持已显著改善。若遇锁定失败,请确认:

    • 使用 glfw-x11 或 glfw-wayland 后端(通过环境变量 MODERNGL_WINDOW=glfw + export GLFW_PLATFORM=wayland 控制);
    • 用户属于 input 组(sudo usermod -aG input $USER);
    • 避免在远程桌面(如 VNC)中运行——独占模式依赖本地输入设备访问。

✅ 总结

Window.mouse_exclusivity 是 moderngl_window 提供的、符合现代图形应用标准的鼠标锁定方案:零依赖、跨平台、低延迟、语义明确。它完美替代了手动坐标钳制、pyautogui 强制归位等反模式。结合 mouse_position_event 的 dx/dy 数据流,即可快速构建稳定的第一人称视角、轨道相机或任意基于鼠标的交互逻辑。记住核心原则:锁光标用 mouse_exclusivity,读位移只取 dx/dy,一切尽在 GLFW 的优雅封装之中。

相关文章

Windows激活工具
Windows激活工具

Windows激活工具是正版认证的激活工具,永久激活,一键解决windows许可证即将过期。可激活win7系统、win8.1系统、win10系统、win11系统。下载后先看完视频激活教程,再进行操作,100%激活成功。

下载

相关标签:

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
点击input框没有光标怎么办
点击input框没有光标怎么办

点击input框没有光标的解决办法:1、确认输入框焦点;2、清除浏览器缓存;3、更新浏览器;4、使用JavaScript;5、检查硬件设备;6、检查输入框属性;7、调试JavaScript代码;8、检查页面其他元素;9、考虑浏览器兼容性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

192

2023.11.24

磁盘配额是什么
磁盘配额是什么

磁盘配额是计算机中指定磁盘的储存限制,就是管理员可以为用户所能使用的磁盘空间进行配额限制,每一用户只能使用最大配额范围内的磁盘空间。php中文网为大家提供各种磁盘配额相关的内容,教程,供大家免费下载安装。

1480

2023.06.21

如何安装LINUX
如何安装LINUX

本站专题提供如何安装LINUX的相关教程文章,还有相关的下载、课程,大家可以免费体验。

712

2023.06.29

linux find
linux find

find是linux命令,它将档案系统内符合 expression 的档案列出来。可以指要档案的名称、类别、时间、大小、权限等不同资讯的组合,只有完全相符的才会被列出来。find根据下列规则判断 path 和 expression,在命令列上第一个 - ( ) , ! 之前的部分为 path,之后的是 expression。还有指DOS 命令 find,Excel 函数 find等。本站专题提供linux find相关教程文章,还有相关

298

2023.06.30

linux修改文件名
linux修改文件名

本专题为大家提供linux修改文件名相关的文章,这些文章可以帮助用户快速轻松地完成文件名的修改工作,大家可以免费体验。

790

2023.07.05

linux系统安装教程
linux系统安装教程

linux系统是一种可以免费使用,自由传播,多用户、多任务、多线程、多CPU的操作系统。本专题提供linux系统安装教程相关的文章,大家可以免费体验。

582

2023.07.06

linux查看文件夹大小
linux查看文件夹大小

Linux是一种自由和开放源码的类Unix操作系统,存在着许多不同的Linux版本,但它们都使用了Linux内核。Linux可安装在各种计算机硬件设备中,比如手机、平板电脑、路由器、视频游戏控制台、台式计算机、大型机和超级计算机。linux怎么查看文件夹大小呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

560

2023.07.20

linux查看ip命令
linux查看ip命令

本专题为大家提供linux查看ip命令相关文章内容,感兴趣的朋友可以免费下载体验试试。

302

2023.07.20

2026春节习俗大全
2026春节习俗大全

本专题整合了2026春节习俗大全,阅读专题下面的文章了解更多详细内容。

189

2026.02.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号