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集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式

《Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式》本文详细介绍如何使用Java通过JDBC连接MySQL数据库,包括下载驱动、配置Eclipse环境、检测数据库连接等关键步骤,... 目录一、下载驱动包二、放jar包三、检测数据库连接JavaJava 如何使用 JDBC 连接 mys

JavaSE正则表达式用法总结大全

《JavaSE正则表达式用法总结大全》正则表达式就是由一些特定的字符组成,代表的是一个规则,:本文主要介绍JavaSE正则表达式用法的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录常用的正则表达式匹配符正则表China编程达式常用的类Pattern类Matcher类PatternSynta

SpringBoot开发中十大常见陷阱深度解析与避坑指南

《SpringBoot开发中十大常见陷阱深度解析与避坑指南》在SpringBoot的开发过程中,即使是经验丰富的开发者也难免会遇到各种棘手的问题,本文将针对SpringBoot开发中十大常见的“坑... 目录引言一、配置总出错?是不是同时用了.properties和.yml?二、换个位置配置就失效?搞清楚加

Python中对FFmpeg封装开发库FFmpy详解

《Python中对FFmpeg封装开发库FFmpy详解》:本文主要介绍Python中对FFmpeg封装开发库FFmpy,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、FFmpy简介与安装1.1 FFmpy概述1.2 安装方法二、FFmpy核心类与方法2.1 FF

基于Python开发Windows屏幕控制工具

《基于Python开发Windows屏幕控制工具》在数字化办公时代,屏幕管理已成为提升工作效率和保护眼睛健康的重要环节,本文将分享一个基于Python和PySide6开发的Windows屏幕控制工具,... 目录概述功能亮点界面展示实现步骤详解1. 环境准备2. 亮度控制模块3. 息屏功能实现4. 息屏时间

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部