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

相关文章

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

java常见报错及解决方案总结

《java常见报错及解决方案总结》:本文主要介绍Java编程中常见错误类型及示例,包括语法错误、空指针异常、数组下标越界、类型转换异常、文件未找到异常、除以零异常、非法线程操作异常、方法未定义异常... 目录1. 语法错误 (Syntax Errors)示例 1:解决方案:2. 空指针异常 (NullPoi

基于Python开发批量提取Excel图片的小工具

《基于Python开发批量提取Excel图片的小工具》这篇文章主要为大家详细介绍了如何使用Python中的openpyxl库开发一个小工具,可以实现批量提取Excel图片,有需要的小伙伴可以参考一下... 目前有一个需求,就是批量读取当前目录下所有文件夹里的Excel文件,去获取出Excel文件中的图片,并

Java反转字符串的五种方法总结

《Java反转字符串的五种方法总结》:本文主要介绍五种在Java中反转字符串的方法,包括使用StringBuilder的reverse()方法、字符数组、自定义StringBuilder方法、直接... 目录前言方法一:使用StringBuilder的reverse()方法方法二:使用字符数组方法三:使用自

基于Python开发PDF转PNG的可视化工具

《基于Python开发PDF转PNG的可视化工具》在数字文档处理领域,PDF到图像格式的转换是常见需求,本文介绍如何利用Python的PyMuPDF库和Tkinter框架开发一个带图形界面的PDF转P... 目录一、引言二、功能特性三、技术架构1. 技术栈组成2. 系统架构javascript设计3.效果图

Python依赖库的几种离线安装方法总结

《Python依赖库的几种离线安装方法总结》:本文主要介绍如何在Python中使用pip工具进行依赖库的安装和管理,包括如何导出和导入依赖包列表、如何下载和安装单个或多个库包及其依赖,以及如何指定... 目录前言一、如何copy一个python环境二、如何下载一个包及其依赖并安装三、如何导出requirem

基于Python开发PDF转Doc格式小程序

《基于Python开发PDF转Doc格式小程序》这篇文章主要为大家详细介绍了如何基于Python开发PDF转Doc格式小程序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用python实现PDF转Doc格式小程序以下是一个使用Python实现PDF转DOC格式的GUI程序,采用T