Panda3d 介绍

Panda3d 介绍

文章目录

  • Panda3d 介绍
    • Panda3d 的安装
    • Panda3d 的坐标系统介绍
    • Panda3d 的运行
    • Panda3d 加载一个熊猫
      • 父节点和子节点之间的关系
    • 验证Panda3d 的坐标系统
      • X 轴的平移
      • Y 轴的平移
      • Z 轴的平移
      • X 轴的旋转
      • Y 轴的旋转
      • Z 轴的旋转

Panda3D是一个3D引擎:一个用于3D渲染和游戏开发的子程序库。该库是带有一组Python绑定的c++。使用Panda3D进行游戏开发通常需要编写一个Python或c++程序来控制Panda3D库。 Panda3D是为商业游戏开发而创建的,并且仍然用于开发商业游戏。

特点如下:

  • 提供了Python 和 C++ 两种形式的接口,底层采用C++实现的,因此在保证运行效率的前提下,使用panda3更加容易编写用户想要的界面;
  • 安装体积小,代码开源;
  • 具有良好的防崩溃机制;

Panda3d的官方网站
Panda3d的源代码
Panda3d国外的教程

Panda3d 的安装

首先通过pip指令下载:

pip3 install panda3d

Panda3d 的坐标系统介绍

Panda3d 的坐标系系统如下图所示, X 轴朝右,Y 轴朝后,Z 轴朝上。 姿态描述为 HPR,可以是欧拉角的形式,不过 H 表示的是绕着 Z 轴旋转, P 表示的是绕着 X 轴旋转, R 表示的是绕着 Y 轴旋转。
Panda3d姿态旋转图

Panda3d 的运行

任何一个3D引擎都需要一个窗口来承载其中的场景内容,那么我们的第一步就是了解Panda3d中如何打开一个窗口界面。

在Panda3d中是通过ShowBase类来创建一个游戏窗口,代码如下:

from direct.showbase.ShowBase import ShowBase

class App(ShowBase):
    def __init__(self):
        ShowBase.__init__(self)

app = App()
app.run()

程序解释如下:
上述程序中创建了一个ShowBase类,通过 Run 函数来显示这个窗口,运行上述代码,打开的窗口如下所示,从下图中可以看出窗口的title默认显示的是 Panda3d,默认打开的窗口大小是 800 * 600。

Pan3d 空白窗口
可以调整对应的参数让这个窗口显示的更大,一个方法就是直接点击窗口的最大化按钮。另外一种是通过代码实现,简而言之就是通过创建一个WindowProperties来实现,比如说你可以创建一个 1000 * 800 的窗口,这样如下操作:

from direct.showbase.ShowBase import ShowBase
from panda3d.core import WindowProperties

class App(ShowBase):
    def __init__(self):
        ShowBase.__init__(self)

        properties = WindowProperties()
        properties.setSize(1000, 800)
        self.win.requestProperties(properties)

app = App()
app.run()

Panda3d 加载一个熊猫

一个游戏是需要一个背景的,因此,场景(Scenery)是需要第一个添加的元素。我们可以直接导入预设的场景模型来构建游戏背景。在添加之前,需要说明panda3d的一个基本数据结构:场景图(Scene Graph)。场景图其实是一个树形的数据结构,根节点是一个render对象,在根节点下的都是需要被渲染的元素。所以一般通过需要渲染元素的reparentTo() 方法将括号内的元素设为需要渲染元素的父节点(通过创建节点路径来实现)。比如我们获取了场景模型scene,那么我们可以通过scene.reparentTo(self.render)来将render设为场景模型的父节点(也可以理解为将场景模型挂载在render对象下),那么游戏启动时,由于render有scene这个子节点,所以scene就会被渲染在窗口上,从而让我们看到该场景。

Panda3d场景关系图

此外,给定节点的状态(例如其位置、姿态变化、模型的缩放)相对于其父节点的状态。也就是说,当我们获取一个子节点的位置的时候,如果位置参数为“零”,则表示“与其父节点位于同一位置”; 旋转姿态参数为“零”,则表示“与父节点具有相同的姿态”。因此,每个节点都会影响其子节点及其子节点,依此类推。父节点和子节点的之间的关系是父节点的移动一定会带动子节点的移动,但是子节点移动并不会带动父节点的移动。

你可以把它想象成有点像铰接小雕像的手臂:移动下臂实际上不会改变上臂。但是,移动上臂会导致下臂相应地移动。此外,如果上臂旋转为向上指向,则下臂同样指向上方,并且下臂的任何旋转都是相对于该方向的。
Panda3d场景中子父节点的关系
但是,大多数时候我们不会直接与节点进行交互。相反,我们与称为“NodePath”的进行交互。这实质上描述了从根到相关节点的场景图的路径。甚至可以为单个节点设置多个不同的 NodePath。

然而,出于实际目的,NodePath可以被认为是与节点几乎(但不完全)相同的:你可以用节点做的大部分事情,你可以用NodePath做(而且更方便)。此外,NodePaths包含对其节点的引用,因此我们也可以通过NodePaths 去操作对应的子节点。

最终我们往场景中导入一个熊猫,panda3d 模型下载

代码如下:

    self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda.reparentTo(self.render)
    self.panda.setScale(0.005,  0.005, 0.005)
    self.panda.setPos(0, 10, -1)

Panda3d导入一个熊猫

上述程序中,我们通过loader.loadModel()方法加载预设的场景模型,需要说明的是,该方法返回的是一个NodePath类,这其实是一个指向模型的指针。

上述的setScale(0.005, 0.005, 0.005)方法把载入场景模型的三个维度都缩小到原本的0.005倍再渲染。 setPos(0, 10, -1)方法将场景模型的位置设为(0, 10, -1),因为Panda3d 中默认相机的位置是在(0, 0, 0) 的位置,因此我们需要调整物体的位置来取得更好的显示效果。

当我们我们也可以保持物体的位置不变,通过修改相机的位置也是达到相同的效果, 在代码中控制相机的时候,最好是要调用disableMouse 函数。

    self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda.reparentTo(self.render)
    self.panda.setScale(0.005,  0.005, 0.005)
	self.disableMouse()
    self.camera.setPos(0, -10, 1)
    # self.panda.setPos(0, 10, -1)

Panda3d 中默认是采用鼠标进行控制相机的视角,如果我们不想采用鼠标进行控制,而是想在后续通过在代码中进行控制的话,那么也是可以禁用这种操作的,通过调用如下函数即可实现该功能。

	self.disableMouse()

这个函数的名字很容易使人误解。它只是停止了鼠标对摄影机的控制,并没有禁用鼠标。你仍可以获得鼠标的位置,以及鼠标的点击。

父节点和子节点之间的关系

下面我们通过在场景中导入两个一样的模型文件,但是修改两个模型挂载的父节点,效果和对应的代码如下:

    self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda.reparentTo(self.render)
    self.panda.setScale(0.005,  0.005, 0.005)
    self.panda.setPos(0, 10, -1)


    self.panda1 = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda1.reparentTo(self.render)
    self.panda1.setScale(0.005,  0.005, 0.005)
    self.panda1.setPos(3, 10, -1)

Panda3d导入两个熊猫

    self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda.reparentTo(self.render)
    self.panda.setScale(0.005,  0.005, 0.005)
    self.panda.setPos(0, 10, -1)


    self.panda1 = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda1.reparentTo(self.panda)
    self.panda1.setScale(1.0,  1.0, 1.0)
    self.panda1.setPos(600, 0.0, 0.0)

Panda3d导入两个熊猫父节点不同

最终上述两个代码显示的效果都是一样的,但是我们可以很明显的发现panda1的缩放比例和设置的位置是不一样的。在第一个例子中 panda1 导入的时候需要缩放 0.005 倍,但是在第一个例子中我们发现是不需要的缩放的。而对于位置而言,在第一个例子中 panda1 导入的时候设置的位置为(3, 10, -1),但是在第二个例子中我们发现设置的位置是(600, 0, 0), 通过一个简单的换算关系我们得出如下换算关系(3, 10, -1) = (0, 10, -1) + (600, 0, 0) * (0.005, 0.005, 0.005)。这个换算关系也最终验证上述所述的父节点的位置、姿态、缩放都是会影响子节点的,这个概念关系在后续的编程中是需要注意的。

    self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda.reparentTo(self.render)
    self.panda.setScale(0.001,  0.001, 0.001)
    self.panda.setPos(0, 10, -1)


    self.panda1 = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
    self.panda1.reparentTo(self.panda)
    self.panda1.setScale(1.0,  1.0, 1.0)
    self.panda1.setPos(600, 0.0, 0.0)

Panda3d导入两个熊猫父节点不同父节点缩放

在上述代码中,我们只对父节点进行缩放,在最终的显示效果中我们可以看出子节点也进行了对应的缩放。说明父节点的位置、姿态、缩放都是会影响子节点的,这个概念关系在后续的编程中是需要注意的。

验证Panda3d 的坐标系统

为了更好的说明 Panda3d 的坐标系, X 轴朝右,Y 轴朝后,Z 轴朝上。 姿态描述为 HPR,可以是欧拉角的形式,不过 H 表示的是绕着 Z 轴旋转, P 表示的是绕着 X 轴旋转, R 表示的是绕着 Y 轴旋转。我们在场景中两个不同样子的熊猫,一开始两个熊猫都在相同的位置,然后调整第二个熊猫的位置和姿态来比较不同参数的含义,姿态的旋转以逆时针方向为正, 姿态的单位是 deg, 位置的单位是 m。

      self.panda = self.loader.loadModel('./panda3d-master/models/panda-model.egg')
      self.panda.reparentTo(self.render)
      self.panda.setScale(0.003,  0.003, 0.003)
      self.panda.setPos(0, 10, -1)


      self.panda1 = self.loader.loadModel('./panda3d-master/models/panda.egg')
      self.panda1.reparentTo(self.render)
      self.panda1.setScale(0.2,  0.2, 0.2)
      self.panda1.setPos(0, 10, -1)

      self.disableMouse()
      self.camera.setPos(1.5, 0, 0)

Panda3d导入两个熊猫相同位置

X 轴的平移

    self.panda1.setPos(3, 10, -1)
    self.panda1.setHpr(0, 0, 0)

Panda3d导入两个熊猫X方向的平移

Y 轴的平移

    self.panda1.setPos(0, 20, -1)
    self.panda1.setHpr(0, 0, 0)

Panda3d导入两个熊猫Y方向的平移

Z 轴的平移

    self.panda1.setPos(0, 10, 0)
    self.panda1.setHpr(0, 0, 0)

Panda3d导入两个熊猫Z方向的平移

X 轴的旋转

    self.panda1.setPos(3, 10, -1)
    self.panda1.setHpr(0, 90, 0)

Panda3d导入两个熊猫饶X轴旋转

因为 X 轴是朝向右边的,因此沿着 X 轴旋转90度,原来头是朝上的,现在改成朝前了。

Y 轴的旋转

    self.panda1.setPos(3, 10, -1)
    self.panda1.setHpr(0, 0, 90)

Panda3d导入两个熊猫饶Y轴旋转

因为 Y 轴是朝向后边的,因此沿着 Y 轴旋转90度,原来两只手是水平放置的,现在改成垂直方向的了。

Z 轴的旋转

    self.panda1.setPos(3, 10, -1)
    self.panda1.setHpr(90, 0, 0)

Panda3d导入两个熊猫饶Z轴旋转

因为 Z 轴是朝向上的,因此沿着 Z 轴旋转90度,原来肚子是朝前的,现在改成朝右了。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

为什么看了那么多测试技术帖,感觉自己还是菜?

作为测试新手,最爱莫过于看各大牛发的技术贴,这篇很牛叉,那篇也很有道理,似乎自己看着看着也会成为高手。然而几年后,发现自己对专业知识的理解乱的很,里面更有很多自相矛盾的地方,这到底是哪里…

iTransformer: INVERTED TRANSFORMERS ARE EFFECTIVE FOR TIME SERIES FORECASTING

#论文题目:ITRANSFORMER: INVERTED TRANSFORMERS ARE EFFECTIVE FOR TIME SERIES FORECASTING #论文地址:https://arxiv.org/abs/2310.06625 #论文源码开源地址:https://github.com/thuml/Time-Series-Library #论文所属会议:Mach…

Flask基本教程以及Jinjia2模板引擎简介

flask基本使用 直接看代码吧,非常容易上手: # 创建flask应用 app Flask(__name__)# 路由 app.route("/index", methods[GET]) def index():return "FLASK:欢迎访问主页!"if __name__ "__main__"…

vulnhub靶机Venus

下载地址:The Planets: Venus ~ VulnHub 主机发现 arp-scan -l 端口扫描 nmap --min-rate 1000 -p- 192.168.21.132 端口版本扫描 nmap -sV -sT -O -p22,8080 192.168.21.132 对于http-alt HTTP Alternative Services 介绍 | JerryQu 的小站 (imququ.com) 总结…

[迁移学习]UniDAformer域自适应全景分割网络

原文的标题为:UniDAformer: Unified Domain Adaptive Panoptic Segmentation Transformer via Hierarchical Mask Calibration,发表于CVPR2023。 一、概述 域自适应全景分割是指利用一个或多个相关域中的现成标注数据来缓解语义分割数据标注复杂的问题。…

MyBatis的增删改查

2023.10.29 本章学习MyBatis的基本crud操作。 insert java程序如下: ①使用map集合传参 Testpublic void testInsertCar(){SqlSession sqlSession SqlSessionUtil.openSession();//先将数据放到Map集合中,在sql语句中使用 #{map集合的key} 来完成传…

Steger算法实现结构光光条中心提取(python版本)

Steger算法原理 对结构光进行光条中心提取时,Steger算法是以Hessian矩阵为基础的。它的基础步骤如下所示: 从Hessian矩阵中求出线激光条纹的法线方向在光条纹法线方向上将其灰度分布按照泰勒多项式展开,求取的极大值即为光条在该法线方向上…

【Flutter】自定义分段选择器Slider

【Flutter】ZFJ自定义分段选择器Slider 前言 在开发一个APP的时候,需要用到一个分段选择器,系统的不满足就自己自定义了一个; 可以自定义节点的数量、自定义节点的大小、自定义滑竿的粗细,自定义气泡的有无等等… 基本上满足你…

Springboot的Container Images,docker加springboot

Spring Boot应用程序可以使用Dockerfiles容器化,或者使用Cloud Native Buildpacks来创建优化的docker兼容的容器映像,您可以在任何地方运行。 1. Efficient Container Images 很容易将Spring Boot fat jar打包为docker映像。然而,像在docke…

合肥中科深谷嵌入式项目实战——人工智能与机械臂(三)

订阅:新手可以订阅我的其他专栏。免费阶段订阅量1000 python项目实战 Python编程基础教程系列(零基础小白搬砖逆袭) 作者:爱吃饼干的小白鼠。Python领域优质创作者,2022年度博客新星top100入围,荣获多家平台专家称号。…

信号去噪之卡尔曼滤波

在前面的文章 卡尔曼滤波 中曾讲解过卡尔曼滤波在惯性导航和飞行姿态控制中的应用,今天来聊一聊卡尔曼滤波在信号去噪中的应用。 卡尔曼滤波(Kalman Filtering)是一种用于估计系统状态的数学方法,它通过考虑系统的动态模型和传感…

86 最小栈

最小栈 题解1 STL大法好题解2 辅助最小栈(直观,空间换时间)题解3 不需要额外空间(!!!差值!!!) 设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。 实现 MinStack 类: MinStack() 初…

树莓派 qt 调用multimedia、multimediawidgets、serialport、Qchats

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、测试11.命令安装出现错误 二、测试21. 安装 Qt Charts:2. 安装 Qt Multimedia 和 Qt MultimediaWidgets:3. 安装 Qt SerialPort&…

重磅新闻-国内首家八类网线认证分析仪上市了

伴随USA对国内某些敏感企业的非常不友好,设置层层障碍,技术堡垒。使得一些网线基础制造研发、线缆线束厂、汽车生产生产厂、军工用途的线缆品质的认证、以及相关高校的研发受到了不同的程度的阻碍。重磅消息,国内首家八类网线认证测仪-维信仪…

几个常用的nosql数据库的操作方式

dynamoDB 键 partition key:分区键 定义:分区键是用于分布数据存储的主键,每个项(Item)在表中都必须有一个唯一的分区键值。 特点: 唯一性:每个分区键值在表中必须是唯一的,这是因为…

【Java 进阶篇】Java HTTP 请求消息详解

HTTP(Hypertext Transfer Protocol)是一种用于传输超文本的应用层协议,广泛用于构建互联网应用。在Java中,我们经常需要发送HTTP请求来与远程服务器进行通信。本文将详细介绍Java中HTTP请求消息的各个部分,包括请求行、…

基于纵横交叉算法的无人机航迹规划-附代码

基于纵横交叉算法的无人机航迹规划 文章目录 基于纵横交叉算法的无人机航迹规划1.纵横交叉搜索算法2.无人机飞行环境建模3.无人机航迹规划建模4.实验结果4.1地图创建4.2 航迹规划 5.参考文献6.Matlab代码 摘要:本文主要介绍利用纵横交叉算法来优化无人机航迹规划。 …

vscode 通过ssh 连接虚拟机vmware(ubuntu)

1.网络连接是否ping的通(ubuntu虚拟机使用的是net 连接方式) 2.配置环境 ubuntu 需要安装ssh server 服务 (1): 安装(Ubuntu安装ssh server) apt-get install openssh-server 检查是否ssh server 是否启动…

PostMan 之 Mock 接口测试

在测试的时候经常会碰到后端开发工程师的接口还没有开发完成,但是测试任务已经分配过来。没有接口怎么测试呢? 测试人员可以通过 mock server 自己去造一个接口来访问。mock server 可用于模拟真实的接口。收到请求时,它会根据配置返回对应的…

图解Kafka高性能之谜(五)

高性能的多分区、冗余副本集群架构 高性能网络模型NIO 简单架构设计: 详细架构设计: 高性能的磁盘写技术 高性能的消息查找设计 索引文件定位使用跳表的设计 偏移量定位消息时使用稀疏索引: 高响应的磁盘拷贝技术 kafka采用sendFile()的…