Prepar3D开发总结02:仪表与按键

2023-11-04 05:20

本文主要是介绍Prepar3D开发总结02:仪表与按键,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一期的博客总结中,Prepar3D开发总结01:模型与SDK详细介绍了Prepar3D二次开发的过程、以及开发过程中两个重要的配置文件。在先前的基础之上,我们一起来总结下在二次开发过程中,仪表与按键的开发流程。

项目工程

模板目录

整个的二次开发工程,可以选用C++或者C#作为开发语言。在此我选择了C++作为整个项目的开发语言。打开模板项目工程,首先是工程的目录文件结构,如下图所示:
image.png
整个CGaugeSample工程一共包括五大项,分别为引用、外部依赖项、Header Files、Resource Files、Source Files文件夹。对其中比较关键的四项做详细说明

  • 外部依赖项:主要是程序运行所依赖Windows的一些.h文件
  • Header Files:包含了程序中自己编写实现的.h和.cpp文件。因为.cpp文件在主文件中采用#include的形式包含进来,所以文件被放在了Header Files目录中。
  • Resource Files:资源文件包含程序中所有的贴图,例如仪表中的背景图,指针的样式等等,注意这些图片的文件的格式是BMP格式,后面还有详细的介绍如何向工程中添加自己的资源图片
  • Source Files:相当于整个项目的工程主文件,在此文件中,定义了要实现的仪表或者按键,同时将对应的程序文件也包含在内。
资源文件介绍

Resource Files中包含一个.RC的文件,这个文件中存放着项目所需的一些图片或者其他资源。可以选中该文件,右击查看代码,查看该文件对应的代码,从代码中,可以看到以下的内容:
版本定义

BEGINBLOCK "StringFileInfo"BEGINBLOCK "040904b0"BEGINVALUE "CompanyName", "Your Company\0"VALUE "FileDescription", "Prepar3D Gauge\0"VALUE "FileVersion", VERSION_STRINGVALUE "LegalCopyright", "Your Copyright.\0"VALUE "ProductName", "Your Product\0"VALUE "ProductVersion", VERSION_STRINGENDENDBLOCK "VarFileInfo"BEGINVALUE "Translation", 0x409, 1200END
END

该处的代码就是定义输出的dll文件的属性,在程序中修改Value后面的字符串即可,在DLL文件中,相关的属性中,就可以查看到以上程序的定义。
image.png
资源定义

BMP_FUEL_SELECTOR_OFF				BITMAP	DISCARDABLE		"res\\Fuel_Selector.Off.BMP"
BMP_FUEL_SELECTOR_LEFT				BITMAP	DISCARDABLE		"res\\Fuel_Selector.Left.BMP"

上面的代码含义为BMP_FUEL_SELECTOR_OFF代表该图片在程序中的名称,中间BITMAP DISCARDABLE是对该名称的描述,最后是资源文件的路径。上述代码的含义就是将文件目录res下的Fuel_Selector.Off.BMP文件定义为BMP_FUEL_SELECTOR_OFF名称。其他都是重复的代码。如果我们想向项目中添加资源,实现的代码也是上面所示。在上面添加相应的代码后,点击CGaugeSample.rc文件就可以看到刚才添加的资源。如果对应的资源名称没有在CGaugeSample.h文件中定义,会显示双引号,这时我们只需要在CGaugeSample.h文件中对名称进行定义即可。即

#define		BMP_FUEL_SELECTOR_OFF				0x2100
#define		BMP_FUEL_SELECTOR_LEFT				0x2101

注意后面的十六进制表示不可重复,其实也就是文件存放在内存中的地址。对于稍微大的文件,中间要有一定的间隔。

仪表实现

以燃油仪表为例,首先需要在CGaugeSample.cpp中对仪表进行定义,实现代码如下:

#define     GAUGE_NAME          "Fuel"         //panel.cfg中添加仪表的名字要于之对应
#define     GAUGEHDR_VAR_NAME   gaugehdr_fuel  // GAUGE_TABLE_ENTRY会用到此变量
#define     GAUGE_W             100            //默认
#include "CGaugeSample.Fuel.hpp"               //仪表的代码文件
//上述代码定义好仪表之后,还需要在文件最后添加定义的仪表
GAUGE_TABLE_ENTRY(&gaugehdr_fuel)

如果该仪表有处理的逻辑变量也需要在CGaugeSample.cpp文件中定义。下面看下CGaugeSample.Fuel.hpp文件中针对仪表是如何实现的。在该文件中,先是根据定义的变量去声明这个仪表,然后用MAKE_NEEDLE去定义仪表的指针,用MAKE_STATIC定义仪表的背景,最后编写相应的回调函数。
注意在该文件中,MAKE_NEEDLE是核心,下面对该宏做详细说明:

// 绘制一个可以绕另外一个图像特定点旋转的图像
MAKE_NEEDLE
(fuel_needle,					 //NAMEBMP_FUEL_SMALL_NEEDLE,			 // RES_ID  the name of resource fileNULL,							 // NEXT_LIST fuel_fail,						 // FAILURE   Probably failure definationIMAGE_USE_TRANSPARENCY|IMAGE_USE_ERASE|IMAGE_BILINEAR_COLOR,     //DRAW_FLAGS0,								 //Aircraft Special Instrument   only with autopilot150, 150,						 // the position of rotate point6, 12,							 // NDL_POSITIONTOTAL_FUEL_QUANTITY_GALLONS,fuel_needle_cb,     // SOURECE_VAR:Token Variable used to drive the needle  CALLBACK: fuel_needle_cbfuel_nonlinearity,               // name of the non-linearity table 6								 //MAX_DEG_PER_SEC
)

MAKE_NEEDLE可以实现让一个图层绕着一个点实现旋转,这也就实现了仪表指针的动画。在定义中,150, 150为围定点选装的坐标,注意这个坐标是相对于背景图片来讲的。图层实现旋转可以还需要数据的驱动,这个数据就是TOTAL_FUEL_QUANTITY_GALLONS,该变量在GaugeDefiners.h文件中定义,代表飞机的燃油总重。如果要实现其他仪表,只需要在该文件中找相关变量即可。有了数据源,接下来就是对数据源进行处理,与之相关的有非线性表fuel_nonlinearity和回调函数fuel_needle_cb。其中回调函数是对数据进行逻辑的判断和单位的转换,而转换后的数据再根据非线性表在图层上进行变化。而这个指针图层如何在背景图层上进行变化,则通过这几个宏IMAGE_USE_TRANSPARENCY|IMAGE_USE_ERASE|IMAGE_BILINEAR_COLOR实现。除此之外,BMP_FUEL_SMALL_NEEDLE代表在资源在程序中的定义。其他参数默认。定义好指针之后,要在lsit数组中进行实现,(PELEMENT_HEADER)&fuel_needle。
对于非线程的定义,相关实现如下:需要注意的一点就是这个范围的定义每次的定义都是起始点开始。

// 非线性区间表 格式:像素点{x,y} 范围a~b
NONLINEARITY  fuel_nonlinearity[] =
{{{30,	182},	 0, 0},{{119,	 47},	25, 0},{{246,	 93},	50, 0},{{241,	221},	75, 0}
};
效果实现

根据以上的代码编译之后,就可以在P3D中看到与之对应的效果图了,如下图所示,此时的仪表为2D仪表,后续还需3DMAX软件绘制3D仪表,通过XML文件传递动画实现3D的效果。
image.png

总结

这篇主要总结了下工程文件的结构以及仪表的具体实现,其实仪表的实现会一个,其他都会了。模板都是一样的。只是变量不同。理解里面的几个关键的宏定义。但是现在有一个问题,仪表的数据源的那个宏的数据是从哪来的,肯定是P3D输出的,具体P3D哪个文件或者项目产生,后面有机会再详细分析吧。这篇就这样,作者文笔有限,如果有错误的地方,欢迎大家指正。

这篇关于Prepar3D开发总结02:仪表与按键的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/343513

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

基于Python开发PPTX压缩工具

《基于Python开发PPTX压缩工具》在日常办公中,PPT文件往往因为图片过大而导致文件体积过大,不便于传输和存储,所以本文将使用Python开发一个PPTX压缩工具,需要的可以了解下... 目录引言全部代码环境准备代码结构代码实现运行结果引言在日常办公中,PPT文件往往因为图片过大而导致文件体积过大,

Python中连接不同数据库的方法总结

《Python中连接不同数据库的方法总结》在数据驱动的现代应用开发中,Python凭借其丰富的库和强大的生态系统,成为连接各种数据库的理想编程语言,下面我们就来看看如何使用Python实现连接常用的几... 目录一、连接mysql数据库二、连接PostgreSQL数据库三、连接SQLite数据库四、连接Mo

Git提交代码详细流程及问题总结

《Git提交代码详细流程及问题总结》:本文主要介绍Git的三大分区,分别是工作区、暂存区和版本库,并详细描述了提交、推送、拉取代码和合并分支的流程,文中通过代码介绍的非常详解,需要的朋友可以参考下... 目录1.git 三大分区2.Git提交、推送、拉取代码、合并分支详细流程3.问题总结4.git push

使用DeepSeek API 结合VSCode提升开发效率

《使用DeepSeekAPI结合VSCode提升开发效率》:本文主要介绍DeepSeekAPI与VisualStudioCode(VSCode)结合使用,以提升软件开发效率,具有一定的参考价值... 目录引言准备工作安装必要的 VSCode 扩展配置 DeepSeek API1. 创建 API 请求文件2.

Kubernetes常用命令大全近期总结

《Kubernetes常用命令大全近期总结》Kubernetes是用于大规模部署和管理这些容器的开源软件-在希腊语中,这个词还有“舵手”或“飞行员”的意思,使用Kubernetes(有时被称为“... 目录前言Kubernetes 的工作原理为什么要使用 Kubernetes?Kubernetes常用命令总

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

Python中实现进度条的多种方法总结

《Python中实现进度条的多种方法总结》在Python编程中,进度条是一个非常有用的功能,它能让用户直观地了解任务的进度,提升用户体验,本文将介绍几种在Python中实现进度条的常用方法,并通过代码... 目录一、简单的打印方式二、使用tqdm库三、使用alive-progress库四、使用progres

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要