import bpy from bpy_extras.wm_utils.progress_report import ProgressReport import tempfile, os, typing from . import PROP_preferences, UTIL_ioport_shared from . import UTIL_virtools_types, UTIL_functions, UTIL_file_browser, UTIL_blender_mesh from . import PROP_virtools_group, PROP_virtools_material, PROP_virtools_texture, PROP_virtools_mesh from .PyBMap import bmap_wrapper as bmap class BBP_OT_import_virtools(bpy.types.Operator, UTIL_file_browser.ImportVirtoolsFile, UTIL_ioport_shared.ImportParams, UTIL_ioport_shared.VirtoolsParams): """Import Virtools File""" bl_idname = "bbp.import_virtools" bl_label = "Import Virtools File" bl_options = {'PRESET', 'UNDO'} @classmethod def poll(self, context): return ( PROP_preferences.get_raw_preferences().has_valid_blc_tex_folder() and bmap.is_bmap_available()) def execute(self, context): _import_virtools( self.general_get_filename(), self.general_get_vt_encodings() ) self.report({'INFO'}, "Virtools File Importing Finished.") return {'FINISHED'} def draw(self, context): layout = self.layout layout.label(text = 'Conflict Options') self.draw_import_params(layout) layout.separator() layout.label(text = 'Virtools Params') self.draw_virtools_params(layout) def _import_virtools(file_name_: str, encodings_: tuple[str]) -> None: # create temp folder with tempfile.TemporaryDirectory() as vt_temp_folder: print(f'Virtools Engine Temp: {vt_temp_folder}') # create virtools reader context with bmap.BMFileReader( file_name_, vt_temp_folder, PROP_preferences.get_raw_preferences().mBallanceTextureFolder, encodings_) as reader: # prepare progress reporter with ProgressReport(wm = bpy.context.window_manager) as progress: # import textures texture_cret_map: dict[bmap.BMTexture, bpy.types.Image] = _import_virtools_textures( reader, progress) # import materials material_cret_map: dict[bmap.BMMaterial, bpy.types.Material] = _import_virtools_materials( reader, progress, texture_cret_map) # import meshes mesh_cret_map: dict[bmap.BMMesh, bpy.types.Mesh] = _import_virtools_meshes( reader, progress, material_cret_map) # import 3dobjects obj3d_cret_map: dict[bmap.BM3dObject, bpy.types.Object] = _import_virtools_3dobjects( reader, progress, mesh_cret_map) # import groups _import_virtools_groups(reader, progress, obj3d_cret_map) def _import_virtools_textures( reader: bmap.BMFileReader, progress:ProgressReport ) -> dict[bmap.BMTexture, bpy.types.Image]: # create map texture_cret_map: dict[bmap.BMTexture, bpy.types.Image] = {} progress.enter_substeps(reader.get_texture_count(), "Loading Textures") # create another temp folder for raw data virtools texture importing with tempfile.TemporaryDirectory() as rawdata_temp: print(f'Texture Raw Data Temp: {rawdata_temp}') for vttexture in reader.get_textures(): tex: bpy.types.Image texpath_to_load: str | None = vttexture.get_file_name() # if no assoc file path (what? but it is real happended) # this is invalid image, create a blank image instead if texpath_to_load is None: tex = bpy.data.images.new("", 1, 1) else: # if this image is raw data, save it in external folder before loading # the attribute of raw data saving is the file path is not absolute path if not os.path.isabs(texpath_to_load): texpath_to_load = os.path.join( rawdata_temp, os.path.basename(texpath_to_load) ) vttexture.save_image(texpath_to_load) # detect whether it is ballance texture and load try_blc_tex: str | None = PROP_virtools_texture.get_ballance_texture_filename(texpath_to_load) if try_blc_tex: # load as ballance texture tex = PROP_virtools_texture.load_ballance_texture(try_blc_tex) else: # load as other textures tex = PROP_virtools_texture.load_other_texture(texpath_to_load) # rename and insert it to map tex.name = UTIL_functions.virtools_name_regulator(vttexture.get_name()) texture_cret_map[vttexture] = tex # inc steps progress.step() # leave progress and return map progress.leave_substeps() return texture_cret_map def _import_virtools_materials( reader: bmap.BMFileReader, progress: ProgressReport, texture_cret_map: dict[bmap.BMTexture, bpy.types.Image] ) -> dict[bmap.BMMaterial, bpy.types.Material]: # create map and prepare progress material_cret_map: dict[bmap.BMMaterial, bpy.types.Material] = {} progress.enter_substeps(reader.get_material_count(), "Loading Materials") for vtmaterial in reader.get_materials(): # create new raw material rawmtl: PROP_virtools_material.RawVirtoolsMaterial = PROP_virtools_material.RawVirtoolsMaterial() rawmtl.mDiffuse = vtmaterial.get_diffuse() rawmtl.mAmbient = vtmaterial.get_ambient() rawmtl.mSpecular = vtmaterial.get_specular() rawmtl.mEmissive = vtmaterial.get_emissive() rawmtl.mSpecularPower = vtmaterial.get_specular_power() mtltex: bmap.BMTexture | None = vtmaterial.get_texture() if mtltex: rawmtl.mTexture = texture_cret_map.get(mtltex, None) else: rawmtl.mTexture = None rawmtl.mTextureBorderColor = vtmaterial.get_texture_border_color() rawmtl.mTextureBlendMode = vtmaterial.get_texture_blend_mode() rawmtl.mTextureMinMode = vtmaterial.get_texture_min_mode() rawmtl.mTextureMagMode = vtmaterial.get_texture_mag_mode() rawmtl.mTextureAddressMode = vtmaterial.get_texture_address_mode() rawmtl.mSourceBlend = vtmaterial.get_source_blend() rawmtl.mDestBlend = vtmaterial.get_dest_blend() rawmtl.mFillMode = vtmaterial.get_fill_mode() rawmtl.mShadeMode = vtmaterial.get_shade_mode() rawmtl.mEnableAlphaTest = vtmaterial.get_alpha_test_enabled() rawmtl.mEnableAlphaBlend = vtmaterial.get_alpha_blend_enabled() rawmtl.mEnablePerspectiveCorrection = vtmaterial.get_perspective_correction_enabled() rawmtl.mEnableZWrite = vtmaterial.get_z_write_enabled() rawmtl.mEnableTwoSided = vtmaterial.get_two_sided_enabled() rawmtl.mAlphaRef = vtmaterial.get_alpha_ref() rawmtl.mAlphaFunc = vtmaterial.get_alpha_func() rawmtl.mZFunc = vtmaterial.get_z_func() # create mtl and apply it mtl: bpy.types.Material = bpy.data.materials.new( UTIL_functions.virtools_name_regulator(vtmaterial.get_name()) ) PROP_virtools_material.set_raw_virtools_material(mtl, rawmtl) PROP_virtools_material.apply_to_blender_material(mtl) # add into map and step material_cret_map[vtmaterial] = mtl progress.step() # leave progress and return progress.leave_substeps() return material_cret_map def _import_virtools_meshes( reader: bmap.BMFileReader, progress: ProgressReport, material_cret_map: dict[bmap.BMMaterial, bpy.types.Material] ) -> dict[bmap.BMMesh, bpy.types.Mesh]: # create map and prepare progress mesh_cret_map: dict[bmap.BMMesh, bpy.types.Mesh] = {} progress.enter_substeps(reader.get_material_count(), "Loading Meshes") for vtmesh in reader.get_meshs(): # create mesh mesh: bpy.types.Mesh = bpy.data.meshes.new( UTIL_functions.virtools_name_regulator(vtmesh.get_name()) ) # open mesh writer with UTIL_blender_mesh.MeshWriter(mesh) as meshoper: # construct data provider data_prov: UTIL_blender_mesh.MeshWriterIngredient = UTIL_blender_mesh.MeshWriterIngredient() # constructor data itor def pos_iterator() -> typing.Iterator[UTIL_virtools_types.VxVector3]: for v in vtmesh.get_vertex_positions(): UTIL_virtools_types.vxvector3_conv_co(v) yield v def nml_iterator() -> typing.Iterator[UTIL_virtools_types.VxVector3]: for v in vtmesh.get_vertex_normals(): UTIL_virtools_types.vxvector3_conv_co(v) yield v def uv_iterator() -> typing.Iterator[UTIL_virtools_types.VxVector2]: for v in vtmesh.get_vertex_uvs(): UTIL_virtools_types.vxvector2_conv_co(v) yield v def face_iterator() -> typing.Iterator[UTIL_blender_mesh.FaceData]: face: UTIL_blender_mesh.FaceData = UTIL_blender_mesh.FaceData( [UTIL_blender_mesh.FaceVertexData() for i in range(3)] ) findices_itor = vtmesh.get_face_indices() fmtl_itor = vtmesh.get_face_material_slot_indexs() for _ in range(vtmesh.get_face_count()): # set indices data vtindices = next(findices_itor) face.mIndices[0].mPosIdx = vtindices.i1 face.mIndices[0].mNmlIdx = vtindices.i1 face.mIndices[0].mUvIdx = vtindices.i1 face.mIndices[1].mPosIdx = vtindices.i2 face.mIndices[1].mNmlIdx = vtindices.i2 face.mIndices[1].mUvIdx = vtindices.i2 face.mIndices[2].mPosIdx = vtindices.i3 face.mIndices[2].mNmlIdx = vtindices.i3 face.mIndices[2].mUvIdx = vtindices.i3 # swap indices face.conv_co() # set mtl data vtmtl = next(fmtl_itor) face.mMtlIdx = vtmtl # return yield face def mtl_iterator() -> typing.Iterator[bpy.types.Material]: for vtmtl in vtmesh.get_material_slots(): if vtmtl: yield material_cret_map.get(vtmtl, None) else: yield None # assign to data provider data_prov.mVertexPosition = pos_iterator() data_prov.mVertexNormal = nml_iterator() data_prov.mVertexUV = uv_iterator() data_prov.mFace = face_iterator() data_prov.mMaterial = mtl_iterator() # add part meshoper.add_ingredient(data_prov) # end of mesh writer # set other mesh settings mesh_settings: PROP_virtools_mesh.RawVirtoolsMesh = PROP_virtools_mesh.RawVirtoolsMesh() mesh_settings.mLitMode = vtmesh.get_lit_mode() PROP_virtools_mesh.set_raw_virtools_mesh(mesh, mesh_settings) # add into map and step mesh_cret_map[vtmesh] = mesh progress.step() # leave progress and return progress.leave_substeps() return mesh_cret_map def _import_virtools_3dobjects( reader: bmap.BMFileReader, progress: ProgressReport, mesh_cret_map: dict[bmap.BMMesh, bpy.types.Mesh] ) -> dict[bmap.BM3dObject, bpy.types.Object]: # create map and prepare progress obj3d_cret_map: dict[bmap.BM3dObject, bpy.types.Object] = {} progress.enter_substeps(reader.get_material_count(), "Loading 3dObjects") # get some essential blender data blender_view_layer = bpy.context.view_layer blender_collection = blender_view_layer.active_layer_collection.collection for vt3dobj in reader.get_3dobjects(): # create 3d object with mesh obj3d: bpy.types.Object = bpy.data.objects.new( UTIL_functions.virtools_name_regulator(vt3dobj.get_name()), mesh_cret_map.get(vt3dobj.get_current_mesh(), None) ) # link to collection blender_collection.objects.link(obj3d) # set world matrix vtmat = vt3dobj.get_world_matrix() UTIL_virtools_types.vxmatrix_conv_co(vtmat) obj3d.matrix_world = UTIL_virtools_types.vxmatrix_to_blender(vtmat) # set visibility obj3d.hide_set(not vt3dobj.get_visibility()) # add into map and step obj3d_cret_map[vt3dobj] = obj3d progress.step() # leave progress and return progress.leave_substeps() return obj3d_cret_map def _import_virtools_groups( reader: bmap.BMFileReader, progress: ProgressReport, obj3d_cret_map: dict[bmap.BM3dObject, bpy.types.Object] ) -> dict[bmap.BM3dObject, bpy.types.Object]: # we need iterate all groups to construct a reversed map # to indicate which groups should this 3dobject be grouped into. reverse_map: dict[bmap.BM3dObject, set[str]] = {} # prepare progress progress.enter_substeps(reader.get_material_count(), "Loading Groups") for vtgroup in reader.get_groups(): # if this group do not have name, skip it group_name: str | None = vtgroup.get_name() if group_name is None: continue for item in vtgroup.get_objects(): # get or create set objgroups: set[str] = reverse_map.get(item, None) if objgroups is None: objgroups = set() reverse_map[item] = objgroups # add into list objgroups.add(group_name) # step progress.step() # leave progress progress.leave_substeps() # now we can assign 3dobject group data by reverse map progress.enter_substeps(reader.get_material_count(), "Applying Groups") for mapk, mapv in reverse_map.items(): # check object assoc_obj = obj3d_cret_map.get(mapk, None) if assoc_obj is None: continue # assign group with PROP_virtools_group.VirtoolsGroupsHelper(assoc_obj) as gpoper: gpoper.clear_all_groups() gpoper.add_groups(mapv) progress.leave_substeps() def register() -> None: bpy.utils.register_class(BBP_OT_import_virtools) def unregister() -> None: bpy.utils.unregister_class(BBP_OT_import_virtools)