骑砍战团MOD开发(52)-使用BrfExporterBlender制作BRF文件

一.Blender模型导出为BRF文件

import bpy
import structclass BrfFile:def __init__(self):self.meshes = []self.bodies = []class Mesh:def __init__(self):self.name = ''self.material_name = ''self.vertices = []self.morph_keys = []self.vertices_fvf = []self.faces = []class Vertex:def __init__(self):self.x = 0.000self.y = 0.000self.z = 0.000class VertexFvf:def __init__(self):self.vertex_index = 0self.vertex_color = -1self.normal_x = 0.000self.normal_y = 0.000        self.normal_z = 0.000self.uv_x = 0.000self.uv_y = 0.000class Face:def __init__(self):self.vertex_fvf_indices = []def write_brf_file(brf_file, f):#write meshesf.write(struct.pack('i', len('mesh')))f.write(b'mesh')f.write(struct.pack('i', len(brf_file.meshes)))for mesh in brf_file.meshes:write_mesh(mesh, f)f.write(struct.pack('i', len('end')))f.write(b'end')def write_mesh(mesh, f):f.write(struct.pack('i', len(mesh.name)))f.write(mesh.name.encode())f.write(struct.pack('i', 0))f.write(struct.pack('i', len(mesh.material_name)))f.write(mesh.material_name.encode())#write verticesf.write(struct.pack('i', len(mesh.vertices)))for vertex in mesh.vertices:f.write(struct.pack('f', vertex.x))f.write(struct.pack('f', vertex.y))f.write(struct.pack('f', vertex.z))#write morph keysf.write(struct.pack('i', 0))f.write(struct.pack('i', 0))#write vertices_fvff.write(struct.pack('i', len(mesh.vertices_fvf)))for vertex_fvf in mesh.vertices_fvf:f.write(struct.pack('i', vertex_fvf.vertex_index))f.write(struct.pack('i', vertex_fvf.vertex_color))f.write(struct.pack('f', vertex_fvf.normal_x))f.write(struct.pack('f', vertex_fvf.normal_y))f.write(struct.pack('f', vertex_fvf.normal_z))f.write(struct.pack('f', vertex_fvf.uv_x))f.write(struct.pack('f', vertex_fvf.uv_y))f.write(struct.pack('f', vertex_fvf.uv_x))f.write(struct.pack('f', vertex_fvf.uv_y))#write facesf.write(struct.pack('i', len(mesh.faces)))for face in mesh.faces:for vertex_fvf_index in face.vertex_fvf_indices:f.write(struct.pack('i', vertex_fvf_index))def create_blender_mesh(blender_object_name):blender_mesh = bpy.context.scene.objects[blender_object_name].datablender_uv = blender_mesh.uv_layers.active.datablender_loops = blender_mesh.loopsmesh = Mesh()mesh.name = blender_mesh.namemesh.material_name = blender_mesh.materials[0].name#create vetexfor blender_vertex in blender_mesh.vertices:vertex = Vertex()vertex.x = round(blender_vertex.co.x, 6)vertex.y = round(blender_vertex.co.y, 6)vertex.z = round(blender_vertex.co.z, 6)mesh.vertices.append(vertex)#create fvf and facesfor blender_face in blender_mesh.polygons:vertex_indices = blender_face.verticesloop_indices = blender_face.loop_indicesface = Face()for loop_index, vertex_index in enumerate(vertex_indices):vertex_normal_x = round(blender_loops[loop_indices[loop_index]].normal.x, 4)vertex_normal_y = round(blender_loops[loop_indices[loop_index]].normal.y, 4)vertex_normal_z = round(blender_loops[loop_indices[loop_index]].normal.z, 4)vertex_uv_x = round(blender_uv[loop_indices[loop_index]].uv.x, 4)vertex_uv_y = round(blender_uv[loop_indices[loop_index]].uv.y, 4)vertex_fvf = VertexFvf()vertex_fvf.vertex_index = vertex_indexvertex_fvf.normal_x = vertex_normal_xvertex_fvf.normal_y = vertex_normal_yvertex_fvf.normal_z = vertex_normal_zvertex_fvf.uv_x = vertex_uv_xvertex_fvf.uv_y = vertex_uv_yvertex_fvf_index = get_vertex_fvf_index(vertex_fvf, mesh.vertices_fvf)if vertex_fvf_index == -1:mesh.vertices_fvf.append(vertex_fvf)vertex_fvf_index = len(mesh.vertices_fvf) - 1face.vertex_fvf_indices.append(vertex_fvf_index)     mesh.faces.append(face)return meshdef get_vertex_fvf_index(vertex_fvf, vertices_fvf):vertex_fvf_index = -1for index, element in enumerate(vertices_fvf):if element.vertex_index == vertex_fvf.vertex_index and round(element.normal_x, 3) == round(vertex_fvf.normal_x, 3) and round(element.normal_y, 3) == round(vertex_fvf.normal_y, 3) and round(element.normal_z, 3) == round(vertex_fvf.normal_z, 3) and round(element.uv_x, 3) == round(vertex_fvf.uv_x, 3) and round(element.uv_y, 3) == round(vertex_fvf.uv_y, 3):vertex_fvf_index = indexbreakreturn vertex_fvf_indexbrf_export_path = 'D:/work/Tool/MB_Warband/BrfExporter/building.brf'
brf_file = BrfFile()
mesh = create_blender_mesh('89_hushi_tank_cheti')
brf_file.meshes.append(mesh)print('vertex count: ' + str(len(brf_file.meshes[0].vertices)))
print('vertex fvf count: ' + str(len(brf_file.meshes[0].vertices_fvf)))
print('face count: ' + str(len(brf_file.meshes[0].faces)))with open(brf_export_path, 'wb') as f:write_brf_file(brf_file, f)

二.Blender顶点动画导出为BRF文件

import bpy
import structclass BrfFile:def __init__(self):self.meshes = []self.bodies = []class Mesh:def __init__(self):self.name = ''self.material_name = ''self.vertices = []self.morph_keys = []self.vertices_fvf = []self.faces = []class Vertex:def __init__(self):self.x = 0.000self.y = 0.000self.z = 0.000class MorphKey:def __init__(self):self.morph_time = 0self.vertices = []self.vertices_fvf = []class VertexFvf:def __init__(self):self.vertex_index = 0self.vertex_color = -1self.normal_x = 0.000self.normal_y = 0.000        self.normal_z = 0.000self.uv_x = 0.000self.uv_y = 0.000class Face:def __init__(self):self.vertex_fvf_indices = []def write_brf_file(brf_file, f):#write meshesf.write(struct.pack('i', len('mesh')))f.write(b'mesh')f.write(struct.pack('i', len(brf_file.meshes)))for mesh in brf_file.meshes:write_mesh(mesh, f)f.write(struct.pack('i', len('end')))f.write(b'end')def write_mesh(mesh, f):f.write(struct.pack('i', len(mesh.name)))f.write(mesh.name.encode())f.write(struct.pack('i', 0))f.write(struct.pack('i', len(mesh.material_name)))f.write(mesh.material_name.encode())#write verticesf.write(struct.pack('i', len(mesh.vertices)))for vertex in mesh.vertices:f.write(struct.pack('f', vertex.x))f.write(struct.pack('f', vertex.y))f.write(struct.pack('f', vertex.z))#non rigged meshf.write(struct.pack('i', 0))#write morph keysf.write(struct.pack('i', len(mesh.morph_keys)))for morph_key in mesh.morph_keys:write_morph_key(morph_key, f)#write vertices_fvff.write(struct.pack('i', len(mesh.vertices_fvf)))for vertex_fvf in mesh.vertices_fvf:f.write(struct.pack('i', vertex_fvf.vertex_index))f.write(struct.pack('i', vertex_fvf.vertex_color))f.write(struct.pack('f', vertex_fvf.normal_x))f.write(struct.pack('f', vertex_fvf.normal_y))f.write(struct.pack('f', vertex_fvf.normal_z))f.write(struct.pack('f', vertex_fvf.uv_x))f.write(struct.pack('f', vertex_fvf.uv_y))f.write(struct.pack('f', vertex_fvf.uv_x))f.write(struct.pack('f', vertex_fvf.uv_y))#write facesf.write(struct.pack('i', len(mesh.faces)))for face in mesh.faces:for vertex_fvf_index in face.vertex_fvf_indices:f.write(struct.pack('i', vertex_fvf_index))def write_morph_key(morph_key, f):f.write(struct.pack('i', morph_key.morph_time))#write verticesf.write(struct.pack('i', len(morph_key.vertices)))for vertex in morph_key.vertices:f.write(struct.pack('f', vertex.x))f.write(struct.pack('f', vertex.y))f.write(struct.pack('f', vertex.z))#write vertices_fvff.write(struct.pack('i', len(morph_key.vertices_fvf)))for vertex_fvf in morph_key.vertices_fvf:f.write(struct.pack('f', vertex_fvf.normal_x))f.write(struct.pack('f', vertex_fvf.normal_y))f.write(struct.pack('f', vertex_fvf.normal_z))def create_blender_mesh(blender_object_name):blender_mesh = bpy.context.scene.objects[blender_object_name].datablender_uv = blender_mesh.uv_layers.active.datablender_loops = blender_mesh.loopsmesh = Mesh()mesh.name = blender_mesh.namemesh.material_name = blender_mesh.materials[0].name#create vetexfor blender_vertex in blender_mesh.vertices:vertex = Vertex()vertex.x = round(blender_vertex.co.x, 6)vertex.y = round(blender_vertex.co.y, 6)vertex.z = round(blender_vertex.co.z, 6)mesh.vertices.append(vertex)#create fvf and facesfor blender_face in blender_mesh.polygons:vertex_indices = blender_face.verticesloop_indices = blender_face.loop_indicesface = Face()for loop_index, vertex_index in enumerate(vertex_indices):vertex_normal_x = round(blender_loops[loop_indices[loop_index]].normal.x, 3)vertex_normal_y = round(blender_loops[loop_indices[loop_index]].normal.y, 3)vertex_normal_z = round(blender_loops[loop_indices[loop_index]].normal.z, 3)vertex_uv_x = round(blender_uv[loop_indices[loop_index]].uv.x, 3)vertex_uv_y = round(blender_uv[loop_indices[loop_index]].uv.y, 3)vertex_fvf = VertexFvf()vertex_fvf.vertex_index = vertex_indexvertex_fvf.normal_x = vertex_normal_xvertex_fvf.normal_y = vertex_normal_yvertex_fvf.normal_z = vertex_normal_zvertex_fvf.uv_x = vertex_uv_xvertex_fvf.uv_y = vertex_uv_yvertex_fvf_index = get_vertex_fvf_index(vertex_fvf, mesh.vertices_fvf)if vertex_fvf_index == -1:mesh.vertices_fvf.append(vertex_fvf)vertex_fvf_index = len(mesh.vertices_fvf) - 1face.vertex_fvf_indices.append(vertex_fvf_index)     mesh.faces.append(face)return mesh  def add_morph_key_to_blender_mesh(blender_object_name, morph_time, mesh):blender_mesh = bpy.context.scene.objects[blender_object_name].datablender_loops = blender_mesh.loopsmorph_key = MorphKey()#create vetexfor blender_vertex in blender_mesh.vertices:vertex = Vertex()vertex.x = round(blender_vertex.co.x, 6)vertex.y = round(blender_vertex.co.y, 6)vertex.z = round(blender_vertex.co.z, 6)morph_key.vertices.append(vertex)#create fvf normalsfor i in range(0, len(mesh.vertices_fvf)):fvf = VertexFvf()morph_key.vertices_fvf.append(fvf)for blender_face_index, blender_face in enumerate(blender_mesh.polygons):vertex_indices = blender_face.verticesloop_indices = blender_face.loop_indicesfor loop_index, vertex_index in enumerate(vertex_indices):vertex_fvf_index = mesh.faces[blender_face_index].vertex_fvf_indices[loop_index]vertex_normal_x = round(blender_loops[loop_indices[loop_index]].normal.x, 3)vertex_normal_y = round(blender_loops[loop_indices[loop_index]].normal.y, 3)vertex_normal_z = round(blender_loops[loop_indices[loop_index]].normal.z, 3)morph_key.vertices_fvf[vertex_fvf_index].normal_x = vertex_normal_xmorph_key.vertices_fvf[vertex_fvf_index].normal_y = vertex_normal_ymorph_key.vertices_fvf[vertex_fvf_index].normal_z = vertex_normal_z#set morph timemorph_key.morph_time = morph_timemesh.morph_keys.append(morph_key)return meshdef get_vertex_fvf_index(vertex_fvf, vertices_fvf):vertex_fvf_index = -1for index, element in enumerate(vertices_fvf):if element.vertex_index == vertex_fvf.vertex_index and round(element.normal_x, 3) == round(vertex_fvf.normal_x, 3) and round(element.normal_y, 3) == round(vertex_fvf.normal_y, 3) and round(element.normal_z, 3) == round(vertex_fvf.normal_z, 3) and round(element.uv_x, 3) == round(vertex_fvf.uv_x, 3) and round(element.uv_y, 3) == round(vertex_fvf.uv_y, 3):vertex_fvf_index = indexbreakreturn vertex_fvf_indexbrf_export_path = 'D:/work/Tool/MB_Warband/BlenderBrfExporter/building.brf'
brf_file = BrfFile()
mesh = create_blender_mesh('jap_car_wheel_morph_key_base')
for morph_key_index in range(0, 12):add_morph_key_to_blender_mesh('jap_car_wheel_morph_key_' + str(morph_key_index + 1), morph_key_index + 1, mesh)
brf_file.meshes.append(mesh)print('vertex count: ' + str(len(brf_file.meshes[0].vertices)))
print('vertex morph key count: ' + str(len(brf_file.meshes[0].morph_keys)))
print('vertex fvf count: ' + str(len(brf_file.meshes[0].vertices_fvf)))
print('face count: ' + str(len(brf_file.meshes[0].faces)))with open(brf_export_path, 'wb') as f:write_brf_file(brf_file, f)

三.Blender绑骨模型导出为BRF文件

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

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

相关文章

《JavaSE》---21.<简单认识Java的集合框架包装类泛型>

目录 前言 一、什么是集合框架 1.1类和接口总览 二、集合框架的重要性 2.1 开发中的使用 2.2 笔试及面试题 三、背后所涉及的数据结构 3.1 什么是数据结构 3.2 容器背后对应的数据结构 四、包装类 4.1 基本数据类型和对应的包装类 4.2 装箱和拆箱 1.最初的写法 2.…

org.springframework.context.ApplicationContext发送消息

1、创建消息的实体类 package com.demo;/*** 监听的实体类**/ public class EventMessage {private String name;public EventMessage(String name) {this.name name;}public String getName() {return name;}public void setName(String name) {this.name name;} }2、创建消…

【Linux】如何使用docker快速部署Stirling-PDF并实现远程处理本地文档

文章目录 前言1. 安装Docker2. 本地安装部署StirlingPDF3. Stirling-PDF功能介绍4. 安装cpolar内网穿透5. 固定Stirling-PDF公网地址 前言 本篇文章我们将在Linux上使用Docker在本地部署一个开源的PDF工具——Stirling PDF,并且结合cpolar的内网穿透实现公网随时随…

lua 实现 函数 判断两个时间戳是否在同一天

函数用于判断两个时间戳是否在同一天。下面是对代码的详细解释: ### 函数参数 - stampA 和 stampB:两个时间戳,用于比较。- resetInfo:一个可选参数,包含小时、分钟和秒数,用于调整时间戳。 ### 函数实现…

Java 集合框架:Java 中的双端队列 ArrayDeque 的实现

大家好,我是栗筝i,这篇文章是我的 “栗筝i 的 Java 技术栈” 专栏的第 019 篇文章,在 “栗筝i 的 Java 技术栈” 这个专栏中我会持续为大家更新 Java 技术相关全套技术栈内容。专栏的主要目标是已经有一定 Java 开发经验,并希望进…

共享模型之无锁

一、问题提出 1.1 需求描述 有如下的需求,需要保证 account.withdraw() 取款方法的线程安全,代码如下: interface Account {// 获取余额Integer getBalance();// 取款void withdraw(Integer amount);/*** 方法内会启动 1000 个线程&#xf…

[技术总结] C++ 使用经验

const 和 constexpr 有什么区别. const 一般是设置一个只读的属性, 在运行时还有可能通过cast变成一个可修改的. 但是constexpr是告诉编译器这就是一个常亮, 在编译时就可以计算出来然后进行替换.static 修饰的成员函数 & 成员变量 static 修饰的成员函数只能访问 static 修…

GraphPad prism处理cck-8获得ic50

C组为空白对照组,a组为dmso对照组,b组为细胞加药组,八个梯度的药物浓度 一、数据转化 首先,打开软件,选项中选择x的第一项,y的第二项,单一药物浓度设定了几个孔就选几 把自己的药物浓度直接复制…

ubuntu22安装拼音输入法

专栏总目录 一、安装命令: sudo apt update sudo apt install fcitx sudo apt install fcitx-pinyin 二、切换输入法

游戏常用运行库安装包 Game Runtime Libraries Package

游戏常用运行库安装包(Game Runtime Libraries Package)是一个整合了多种游戏所需运行库的安装程序,旨在帮助玩家和开发者解决游戏无法正常运行的问题。该安装包支持从Windows XP到Windows 11的系统,并且具备自动检测系统并推荐合…

代码随想录训练第二十七天|LeetCode56.合并区间、LeetCode738.单调递增的数字、LeetCode968.监控二叉树

文章目录 56.合并区间思路 738.单调递增的数字思路 968.监控二叉树思路确定遍历顺序如何隔两个节点放一个摄像头 56.合并区间 以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间,并返回 一…

数据科学测试 1:为什么要测试?

数据科学测试简介 测试是任何数据科学家都需要学习的重要技能之一。 根据我的经验,为代码添加测试是在整体时间节省方面最有价值的投资。 在这一系列的博客文章中,我们将介绍当今在Python中进行代码测试的最重要理念。 这篇博客文章首先探讨了测试的…

Step-DPO 论文——数学大语言模型理解

论文题目:STEP-DPO: STEP-WISE PREFERENCE OPTIMIZATION FOR LONG-CHAIN REASONING OF LLMS 翻译为中文就是:“LLMs长链推理的逐步偏好优化” 论文由港中文贾佳亚团队推出,基于推理步骤的大模型优化策略,能够像老师教学生一样优…

String 和StringBuilder字符串操作快慢的举例比较

System.currentTimeMillis(); //当前时间与1970年1月1日午夜UTC之间的毫秒差。public class HelloWorld {public static void main(String[] args) {String s1 "";StringBuilder s2 new StringBuilder("");long time System.currentTimeMillis();long s…

git命令学习分享

分布式版本控制系统,本地仓库和远程仓库相互独立。 使用repository仓库进行控制,可以对里面的文件进行跟踪,复原。 git config --global --list:查看git配置列表 cd ** :进入** cd .. :退回上一级 echo…

AI Agent项目探索与实践记录

AI Agent项目探索与实践记录 1. 概述2. 总体结构2.1 记忆模块2.2 模型服务模块2.2.1 LLM服务2.2.2 retrieval服务2.2.3 rerank服务 2.3 Agent系统2.3.1 Planner2.3.2 Code/SQL Generator2.3.3 Code Executor2.3.4 Responser2.3.5 Round Compressor2.3.6 New Turn Discriminator…

基于Llama Index构建RAG应用(Datawhale AI 夏令营)

前言 Hello,大家好,我是GISer Liu😁,一名热爱AI技术的GIS开发者,本文参与活动是2024 DataWhale AI夏令营;😲 在本文中作者将通过: Gradio、Streamlit和LlamaIndex介绍 LlamaIndex 构…

linux文本查看命令

在Linux中,查找文件通常使用几个不同的命令,具体取决于你的需求和上下文。以下是一些最常用的命令: find 命令: find 是最强大和灵活的命令之一,用于在目录树中搜索文件,并执行对找到的文件执行指定的操作…

全局 loading

好久不见! 做项目中一直想用一个统一的 loading 状态控制全部的接口加载,但是一直不知道怎么处理,最近脑子突然灵光了一下想到了一个办法。 首先设置一个全局的 loading 状态,优先想到的就是 Pinia 然后因为页面会有很多接口会…

数据结构——栈(链式结构)

一、栈的链式存储结构 如果一个栈存在频繁进栈和出栈操作,可以考虑链式结构。 栈的链式存储结构是指使用链表来实现栈这种数据结构。在链式存储结构中,栈的每个元素被封装成一个节点,节点之间通过指针相连,形成一个链表。栈顶元…