Panda3d 相机控制

news2024/11/30 12:50:15

Panda3d 相机控制

文章目录

  • Panda3d 相机控制
    • Panda3d中的透视镜头和垂直镜头
      • 透视镜头
      • 垂直镜头
    • Panda3d 中用代码控制相机的移动
      • 用键盘控制相机的移动
      • 用鼠标控制相机的移动

Panda3d 把相机也当做是一个 PandaNode,因此可以向操作其他节点对其进行操作。

真正的相机是在ShowBase类中的一个叫做base.camNodePath,在这个上面还有一个更简单的叫做base.cameraNodePath,一般对相机进行控制的话,是在代码中进行控制。

默认情况下,panda运行一个task使我们可以通过鼠标来移动相机。用户自己写的移动相机的代码将和这个task产生冲突。该task根据鼠标当前的每一帧输入来更新相机的位置。这意味着直接控制相机的代码将不能工作,因为它会跟默认的相机控制任务相冲突。为了处理这种冲突,那么用户可以通过调用以下函数进行:

base.disableMouse()

ShowBase类为用户准备了一些控制相机的方法。useDrive()命令打开键盘和鼠标控制,这两种控制系统都只能在X和Y轴上移动,不能在Z轴上移动。

键盘系统使用方向键,“上”向前移动相机,“下”向后移动。“左”“右”键左右移动相机镜头。
鼠标系统对按下的任何一个键都有反应。若光标向屏幕上方移动,相机向前;若光标向屏幕下方移动,相机向后。如果光标在屏幕两边,相机向那个方向进行旋转。相机的移动速度取决于光标距离中心的远近。另外,Panda还提供一个命令允许使用跟踪球(trackball)鼠标:

base.useDrive() 
base.useTrackball()

ShowBase还提供了oobe()方法,当你的代码在移动相机节点(base.camera)时,你可以用鼠标/跟踪球来控制基相机节点(base.cam)。这对debug非常有用。oobe代表“out-of-body experience”(灵魂出窍),即开发时你可以对程序进行全方位的观察(God’s-eye view)。该方法是一个开关,你要打开oobe模式时调用它,然后再调用一次关闭它:

base.oobe()

oobeCull()是oobe()的一个变形,它们的作用相近,oobeCull()的不同在于,当从新的相机位置重绘场景时,场景仍然会以原先的相机位置进行剔除(cull)。因此,你可以从你“灵魂飞出”地方来观察场景,你可以四处游走,看到物体进入和弹出视区,就像你的视棱台(view frustum)也在移动一样。

Panda3d中的透视镜头和垂直镜头

透视镜头

每个相机都有一个镜头,决定它的成像参数。对于简单的应用程序,你无需考虑镜头问题,默认的镜头参数已经很好了。但是,有时候你想调整一些镜头的参数,如视域。根据对镜头的不同需求,我们提供了几种接口来修改参数。
Panda3d 启动时,它自动为你创建了一个默认的相机和镜头。这个默认相机对象保持在base.cam(从方便的角度,我们应该使用base.camera来移动相机),默认镜头是base.camLens。
默认镜头几乎总是一个透视镜头——即 PerspectiveLens 类的一个实例——除非你换成另一种镜头。到目前为止,透视镜头是一种使用最广泛的镜头,它就像一台真实的相机的镜头,功能与人眼晶状体相同。

透视镜头

上图展示一个常规透镜相机的成像。相机只能看到黑线框里面的物体,这个区域被称为镜头棱台(frustum)。
图中可以看到镜头拍摄的图像(图像是颠倒的,跟真实的物理镜头成像一样)。颠倒的图像只是起到说明作用,它并不是Panda3D相机的一部分,它帮助我们理解Panda3D镜头和真实镜头的关系。
PerspectiveLens有很多参数可以设置,这些参数不都是独立的,设置某些参数将改变另外一些参数的值。

垂直镜头

前面介绍了PerspectiveLens类,以及透视镜头,3D渲染常用的另外一种镜头就是垂直镜头,它没有视域的概念, 如下图所示:

垂直镜头

在垂直镜头里没有透视——穿过镜头的平行光不汇聚,而是保持平行。透视镜头模拟了真实的物理镜头,但现实中不存在垂直镜头。它主要用于特殊效果,比如非自然的景观、模拟即时战略游戏的2.5D场景,或绘制不需要透视的2D物体。事实上,对于 render2d scene graph,默认的相机就是一个OrthographicLens,用于绘制屏幕GUI。

既然垂直镜头没有视域角度,lens.setFov()方法就不起作用。为了调节垂直镜头的范围,你需要调整它的胶片规格。与PerspectiveLens不同,对OrthographicLens来说胶片规格的单位不是任意的,应该用空间单位,与场景建模时使用的单位一致。例如,上图OrthographicLens的胶片规格被设为lens.setFilmSize(20, 15), 20 英尺 x 15 英尺 ——因为场景建模以英尺为单位,一只大熊猫大概有 12 英尺 高。

垂直镜头的另一个方便的参数是近距离,它的值不必一定是正数。实际上可以是负数——可以把近平面放在相机平面之后,也就是说相机可以看到它身后的物体。为render2d准备的OrthographicLens被设成setNearFar(-1000, 1000),将绘制所有Z值在-1000到1000之间的物体。(当然,在render2d中,几乎全部物体的Z值都为0,因此不会有什么问题)
如果需要,你可以把默认的相机换成一个垂直镜头:

lens = OrthographicLens() 
lens.setFilmSize(20, 15) # 根据你的场景来选取适当的值
base.cam.node().setLens(lens)

注意,使用垂直镜头可能让人失去空间感——比如,物体不因为你靠近它而变大,也不因为你远离它而变小——因此你可能不知道相机在移动。

Panda3d 中用代码控制相机的移动

下面两个代码就需要禁用Panda3d中默认的鼠标控制相机任务,否则不能达到用户预期的目的。

用键盘控制相机的移动

最终的代码如下所示:

from direct.actor.Actor import Actor
from panda3d.core import loadPrcFileData
from direct.showbase.ShowBase import ShowBase #基本显示模块

# 画面显示配置,设置窗口大小,窗口名称、显示帧率
confVar = """
win-size 1280 720
window-title example
show-frame-rate-meter True
"""

loadPrcFileData("", confVar)

class MyApp(ShowBase):
    def __init__(self):#场景初始化
        super(MyApp, self).__init__()
        base.disableMouse()
        self.person = base.loader.loadModel('smiley')
        self.person.reparentTo(self.render)
        # 循环一个动作
        # self.person.loop('run')
        self.cam.setPos(0, -10, 0)
        self.keyMap = {
            'up':False,
            'down':False,
            'left':False,
            'right':False,
            'go':False,
            'back':False,
            'rotate':False,
        }

        self.speed = 4
        self.angle = 0
        # self.accept(<event-name>,<function name>)
        # self.accept(<event-name>,<function name>, <parameters-list>)
        self.accept('arrow_up', self.updateKeyMap, ['up', True])
        self.accept('arrow_up-up', self.updateKeyMap, ['up', False])
        self.accept('arrow_down', self.updateKeyMap, ['down', True])
        self.accept('arrow_down-up', self.updateKeyMap, ['down', False])
        self.accept('arrow_left', self.updateKeyMap, ['left', True])
        self.accept('arrow_left-up', self.updateKeyMap, ['left', False])
        self.accept('arrow_right', self.updateKeyMap, ['right', True])
        self.accept('arrow_right-up', self.updateKeyMap, ['right', False])
        self.accept('w', self.updateKeyMap, ['go', True])
        self.accept('w-up', self.updateKeyMap, ['go', False])
        self.accept('s', self.updateKeyMap, ['back', True])
        self.accept('s-up', self.updateKeyMap, ['back', False])
        self.accept('space-up', self.updateKeyMap, ['rotate', True])
        self.accept('apace-up-up', self.updateKeyMap, ['rotate', False])

        self.taskMgr.add(self.update,"update")

    def updateKeyMap(self, key, state):
        self.keyMap[key] = state

    def update(self, task):
        # 或者这个函数的运行时间,或者说是更新的帧率
        dt = globalClock.getDt()
        pos = self.person.getPos()
        if self.keyMap['up']:
            pos.z += self.speed * dt
        if self.keyMap['down']:
            pos.z -= self.speed * dt
        if self.keyMap['left']:
            pos.x -= self.speed * dt
        if self.keyMap['right']:
            pos.x += self.speed * dt
        if self.keyMap['go']:
            pos.y -= self.speed * dt
        if self.keyMap['back']:
            pos.y += self.speed * dt
        if self.keyMap['rotate']:
            self.angle += 1
            if self.angle == 360:
                self.angle = 0
            self.person.setH(self.angle)
        self.person.setPos(pos)

        return task.cont


app = MyApp()
app.run()
run()

用鼠标控制相机的移动

Panda3d 中默认的鼠标操作定义如下:

  • 鼠标左键:按住左键再移动光标可以控制画面左右旋摆。
  • 鼠标右键:按住右键键再移动光标可以控制画面的远近。
  • 鼠标滚轮:按住滚轮键键再移动光标可以控制角度上下左右的角度旋转(盘旋)。
  • 鼠标右键+滚轮:按住右键+滚轮键键再移动光标绕垂直电脑屏幕的轴旋转。

下面代码中的鼠标操作定义如下:

  • 鼠标左键:按住左键再移动光标可以控制画面左右上下移动。

  • 鼠标右键:按住右键键再移动光标可以控制画面的左右上下旋转。

  • 鼠标滚轮:按住滚轮键可以控制画面的前后移动,也就是画面的放大和缩小。

鼠标事件定义

  • mouse1 是鼠标左键
  • mouse2 是鼠标中键
  • mouse3 是鼠标右键

代码如下:


from direct.actor.Actor import Actor
from panda3d.core import loadPrcFileData
from direct.showbase.ShowBase import ShowBase #基本显示模块
from direct.showbase.ShowBase import (Filename, LVecBase3f, NodePath, Task)
from direct.interval.IntervalGlobal import LerpPosInterval
from IPython import embed

# 画面显示配置,设置窗口大小,窗口名称、显示帧率
confVar = """
win-size 1280 720
window-title example
show-frame-rate-meter True
"""

loadPrcFileData("", confVar)

class MyApp(ShowBase):
	def __init__(self):#场景初始化
		super(MyApp, self).__init__()
		base.disableMouse()
		self.person = base.loader.loadModel('smiley')
		self.person.reparentTo(self.render)
		# 循环一个动作
		# self.person.loop('run')
		self.cam.setPos(0, -10, 0)
		self.keyMap = {
			'up':False,
			'down':False,
			'left':False,
			'right':False,
			'go':False,
			'back':False,
			'rotate':False,
		}

		self.speed = 4
		self.angle = 0
		self.mouse_map = {}
		self.Cursor2D_X = 0.0
		self.Cursor2D_Y = 0.0
		self.Cursor2D_X_pre = 0.0
		self.Cursor2D_Y_pre = 0.0
		self.Cursor2D_X_direction = 0.0
		self.Cursor2D_Y_direction = 0.0

		# 可以取消原来默认的鼠标点击事件,可以用户自定义鼠标控制事件
		self.accept("mouse1", self.SetMouse, ["mouse1_event", True])
		self.accept("mouse1-up", self.SetMouse, ["mouse1_event", False])
		# mouse2 是鼠标中键
		# 鼠标的右键
		self.accept("mouse3", self.SetMouse, ["mouse3_event", True])
		self.accept("mouse3-up", self.SetMouse, ["mouse3_event", False])


		# self.mouseWatcherNode.set_modifier_buttons(ModifierButtons())
		# self.buttonThrowers[0].node().set_modifier_buttons(ModifierButtons())

		self.accept("mouse3-up_mouse1-up", self.SetMouse, ["mouse31_event", False])

		# 鼠标中键向上滚动
		self.accept("wheel_up", self.cameraZoom,[-1])
		# 鼠标中键向下滚动
		self.accept("wheel_down", self.cameraZoom,[1])

		self.taskMgr.add(self.UpdateMouseCameraTask, "UpdateMouseCameraTask")

	def SetMouse(self, mouse, val):
		self.mouse_map[mouse] = val

	def UpdateMouseCameraTask(self, task):

		# time since last frame
		dt = globalClock.getDt()
		step = 90
		if base.mouseWatcherNode.hasMouse():
			# 两条指令值等价的,都是得到当前鼠标的位置
			self.Cursor2D_X = base.mouseWatcherNode.getMouseX()
			self.Cursor2D_Y = base.mouseWatcherNode.getMouseY()

			if(self.Cursor2D_X - self.Cursor2D_X_pre > 1e-4):
				self.Cursor2D_X_direction = 0.1
			elif (self.Cursor2D_X - self.Cursor2D_X_pre < -1e-4):
				self.Cursor2D_X_direction = -0.1
			else:
				self.Cursor2D_X_direction = 0.0

			if(self.Cursor2D_Y - self.Cursor2D_Y_pre > 1e-4):
				self.Cursor2D_Y_direction = 0.1
			elif (self.Cursor2D_Y - self.Cursor2D_Y_pre < -1e-4):
				self.Cursor2D_Y_direction = -0.1
			else:
				self.Cursor2D_Y_direction = 0.0

			if self.mouse_map.get("mouse1_event") == True and (self.mouse_map.get("mouse3_event") == False or  self.mouse_map.get("mouse3_event") == None):
				self.camera.setPos(LVecBase3f(self.camera.getX()-(2*self.Cursor2D_X_direction), self.camera.getY(), self.camera.getZ()-2*self.Cursor2D_Y_direction))

			if self.mouse_map.get("mouse3_event") == True and (self.mouse_map.get("mouse1_event") == False or self.mouse_map.get("mouse1_event") == None):
				self.camera.setHpr(self.camera.getH()+(2*self.Cursor2D_X_direction), self.camera.getP()+(2*self.Cursor2D_Y_direction), self.camera.getR())

			# # 以下两个指令用来获取得到当前窗口的大小
			# print(base.win.getProperties().getXSize())
			# print(base.win.getProperties().getYSize())

			self.Cursor2D_X_pre = self.Cursor2D_X
			self.Cursor2D_Y_pre = self.Cursor2D_Y

		return Task.cont

	# 进行相机视野的放大和缩小
	def cameraZoom(self,dir):
		self.camera.setPos(LVecBase3f(self.camera.getX(), self.camera.getY()-(2*dir), self.camera.getZ()+(2*dir)))
		# self.camZoom = LerpPosInterval(self.camera, self.speed, LVecBase3f(self.camera.getX(), self.camera.getY()-(2*dir), self.camera.getZ()+(2*dir)))
		# self.camZoom.start()


app = MyApp()
app.run()
run()

上述UpdateMouseCameraTask 主要是通过不断的比较当前鼠标光标的位置来调整相机的位置,按下鼠标的左键主要是控制相机的位置移动, 而按下右键主要是控制相机的姿态旋转。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1159505.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

stm32整理(三)ADC

1 ADC简介 1.1 ADC 简介 12 位 ADC 是逐次趋近型模数转换器。它具有多达 19 个复用通道&#xff0c;可测量来自 16 个外部 源、两个内部源和 VBAT 通道的信号。这些通道的 A/D 转换可在单次、连续、扫描或不连续 采样模式下进行。ADC 的结果存储在一个左对齐或右对齐的 16 位…

Azure 机器学习 - 使用 AutoML 和 Python 训练物体检测模型

目录 一、Azure环境准备二、计算目标设置三、试验设置四、直观呈现输入数据五、上传数据并创建 MLTable六、配置物体检测试验适用于图像任务的自动超参数扫描 (AutoMode)适用于图像任务的手动超参数扫描作业限制 七、注册和部署模型获取最佳试用版注册模型配置联机终结点创建终…

Flask——接口路由技术

接口路由技术 一、Flask 简介1、环境安装&#xff1a;2、一个最小的应用3、两种运行方式 二、定义路由1、普通路由2、动态路由3、限定类型4、地址尾部的“/” 三、请求与响应-请求方法四、请求与响应-处理请求数据1、request的常用属性/方法2、get 请求参数3、json 请求4、表单…

高效文件整理:按数量划分自动建立文件夹,轻松管理海量文件

在日常生活和工作中&#xff0c;我们经常需要处理大量的文件。然而&#xff0c;如何高效地整理这些文件却是一个棘手的问题。有时候&#xff0c;我们可能需要按照特定的规则来建立文件夹&#xff0c;以便更高效地整理文件。例如&#xff0c;您可以按照日期、时间或者特定的标签…

老杨说运维 | 历时180天,跟复旦大学共研的运维大模型终于来了!

写在前面 Q1&#xff1a;到处都在说的AI大模型到底是什么? ? ? A1&#xff1a;AI大模型是“人工智能预训练大模型"的简称&#xff0c;它包含了"预训练“和”大模型“两层含义&#xff0c;二者结合产生了一种新的人工智能模式即模型在大规模数据集上完成了预训练…

Azure 机器学习 - 使用无代码 AutoML 训练分类模型

了解如何在 Azure 机器学习工作室中使用 Azure 机器学习自动化 ML&#xff0c;通过无代码 AutoML 来训练分类模型。 此分类模型预测某个金融机构的客户是否会认购定期存款产品。 关注TechLead&#xff0c;分享AI全维度知识。作者拥有10年互联网服务架构、AI产品研发经验、团队管…

【C语法学习】5 - fputc()函数

文章目录 1 函数原型2 参数3 返回值4 示例4.1 示例14.2 示例24.3 示例3 1 函数原型 fputc()&#xff1a;将一个字符发送至指定流stream&#xff0c;函数原型如下&#xff1a; int fputc(int c, FILE *stream);2 参数 fputc()函数有两个参数c和stream&#xff1a; 参数c是待…

Proteus仿真--基于51单片机的按键控制LED仿真(仿真文件+程序)

本文主要介绍基于51单片机的按键控制LED仿真&#xff08;完整仿真源文件及代码见文末链接&#xff09; 本仿真文件主要涉及4个按键&#xff0c;其中&#xff1a; K1按键的逻辑是——逐个点亮 K2按键的逻辑是——上四个点亮 K3按键的逻辑是——下四个点亮 K4按键的逻辑是——关…

opencv复习(很乱)

2-高斯与中值滤波_哔哩哔哩_bilibili 1、均值滤波 2、高斯滤波 3、中值滤波 4、腐蚀操作 卷积核不都是255就腐蚀掉 5、膨胀操作 6、开运算 先腐蚀再膨胀 7、闭运算 先膨胀再腐蚀 8、礼帽 原始数据-开运算结果 9、黑帽 闭运算结果-原始数据 10、Sobel算子 左-右&#x…

⾯向对象编程:封装数据和⾏为、定义交互协议、扩展与复⽤ - GO语言从入门到实战

⾯向对象编程&#xff1a;封装数据和⾏为、定义交互协议、扩展与复⽤ - GO语言从入门到实战 一、封装数据和⾏为 结构体定义 定义了一个名为Structural的结构体。结构体是一种用户自定义的数据类型&#xff0c;可以包含不同类型的字段&#xff08;成员变量&#xff09;。 与…

【Unity编辑器扩展】艺术字/自定义图片字体生成工具

艺术字在游戏中很常用&#xff0c;由于普通字体样式过于平淡&#xff0c;制作花里胡哨的文字图片作为游戏字体使用&#xff0c;这就是艺术字。 不依赖第三方工具&#xff0c;仅使用Unity自带的Custom Font 一张艺术字图集就能实现这个功能&#xff0c;但是为了便于使用&#…

多伦多公共图书馆遭遇周末网络攻击,服务中断

多伦多公共图书馆&#xff08;TPL&#xff09;在10月28日星期六遭遇网络攻击后&#xff0c;警告称其许多在线服务已经中断。 作为加拿大最大的公共图书馆系统&#xff0c;TPL通过多伦多市内的100个分馆为人们提供1200万本图书的借阅服务。图书馆拥有120万注册会员&#xff0c;…

[SpringCloud | Linux] CentOS7 部署 SpringCloud 微服务

目录 一、环境准备 1、工具准备 2、虚拟机环境 3、Docker 环境 二、项目准备 1、配置各个模块&#xff08;微服务&#xff09;的 Dockerfile 2、配置 docker-compose.yml 文件 3、Maven 打包 4、文件整合并传输 三、微服务部署 1、部署至 Docker 2、访问微服务 四…

【Verilog】7.2.1 Verilog 并行 FIR 滤波器设计

FIR&#xff08;Finite Impulse Response&#xff09;滤波器是一种有限长单位冲激响应滤波器&#xff0c;又称为非递归型滤波器。 FIR 滤波器具有严格的线性相频特性&#xff0c;同时其单位响应是有限长的&#xff0c;因而是稳定的系统&#xff0c;在数字通信、图像处理等领域…

MacOS安装git

文章目录 通过Xcode Command Lines Tool安装(推荐)终端直接运行git命令根据流程安装先安装Command Lines Tool后再安装git 官网下载二进制文件进行安装官方国外源下载二进制文件(不推荐)国内镜像下载二进制文件(推荐)安装git 通过Xcode Command Lines Tool安装(推荐) 简单来讲C…

性能压力测试主要目标及步骤

性能压力测试是软件开发生命周期中至关重要的一部分&#xff0c;旨在评估应用程序或系统在高负载和极端条件下的性能表现。这种测试有助于发现性能瓶颈、资源耗尽和错误&#xff0c;以确保应用程序在真实使用情况下的可靠性和稳定性。本文将探讨性能压力测试的概念、方法和最佳…

Distribution-Aware Coordinate Representation for Human Pose Estimation阅读笔记

主要研究人体姿态估计中heatmap转坐标的方法&#xff0c;提出一种新的解码方法 &#xff08;其实这人体姿态我毛也不会&#xff0c;过来看看这个heatmap解码方法&#xff09; 代码&#xff1a;https://github.com/ilovepose/DarkPose/blob/master/lib/core/inference.py 方法…

保障效率与可用,分析Kafka的消费者组与Rebalance机制

系列文章目录 上手第一关&#xff0c;手把手教你安装kafka与可视化工具kafka-eagle Kafka是什么&#xff0c;以及如何使用SpringBoot对接Kafka 架构必备能力——kafka的选型对比及应用场景 Kafka存取原理与实现分析&#xff0c;打破面试难关 防止消息丢失与消息重复——Kafka可…

YOLOv5 分类模型的预处理

YOLOv5 分类模型的预处理 flyfish 版本 6.2 将整个代码简化成如下代码 imgsz224 file "/home/a/Pictures/1.jpg" transforms classify_transforms(imgsz) im cv2.cvtColor(cv2.imread(file), cv2.COLOR_BGR2RGB) print(im.shape)im transforms(im) print(im.…

【计算机网络】第二章:应用层

应用层协议原理 客户-服务器体系结构&#xff1a; 特点&#xff1a;客户之间不能直接通信&#xff1b;服务器具有周知的&#xff0c;固定的地址&#xff0c;该地址称为IP地址。 配备大量主机的数据中心常被用于创建强大的虚拟服务器&#xff1b;P2P体系结构&#xff1a; 特点&…