2023-11-04 21:58:58 +08:00
|
|
|
import ctypes, typing, atexit
|
|
|
|
from . import bmap, virtools_types
|
|
|
|
|
|
|
|
#region Basic Class Defines
|
|
|
|
|
2023-11-05 10:44:11 +08:00
|
|
|
g_InvalidCKID: int = 0
|
2023-11-04 21:58:58 +08:00
|
|
|
g_BMapEncoding: str = "utf-8"
|
|
|
|
|
|
|
|
class _AbstractPointer():
|
|
|
|
__mRawPointer: int
|
|
|
|
|
|
|
|
def __init__(self, raw_pointer: bmap.bm_void_p):
|
|
|
|
if raw_pointer.value is None:
|
|
|
|
self.__mRawPointer = 0
|
|
|
|
else:
|
|
|
|
self.__mRawPointer = raw_pointer.value
|
|
|
|
|
|
|
|
def _is_valid(self) -> bool:
|
|
|
|
return self.__mRawPointer != 0
|
|
|
|
|
|
|
|
def _get_pointer(self) -> bmap.bm_void_p:
|
|
|
|
return bmap.bm_void_p(self.__mRawPointer)
|
|
|
|
|
|
|
|
def __eq__(self, obj: object) -> bool:
|
|
|
|
if isinstance(obj, self.__class__):
|
|
|
|
return obj.__mRawPointer == self.__mRawPointer
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def __hash__(self) -> int:
|
|
|
|
return hash(self.__mRawPointer)
|
|
|
|
|
|
|
|
class _AbstractCKObject(_AbstractPointer):
|
|
|
|
__mCKID: int
|
|
|
|
|
|
|
|
def __init__(self, raw_pointer: bmap.bm_void_p, ckid: bmap.bm_CKID):
|
|
|
|
_AbstractPointer.__init__(self, raw_pointer)
|
|
|
|
self.__mCKID = ckid.value
|
|
|
|
|
|
|
|
def _is_valid(self) -> bool:
|
|
|
|
return _AbstractPointer._is_valid(self) and self.__mCKID != 0
|
|
|
|
|
|
|
|
def _get_ckid(self) -> bmap.bm_CKID:
|
|
|
|
return bmap.bm_CKID(self.__mCKID)
|
|
|
|
|
|
|
|
def __eq__(self, obj: object) -> bool:
|
|
|
|
if isinstance(obj, self.__class__):
|
|
|
|
return obj.__mRawPointer == self.__mRawPointer and obj.__mCKID == self.__mCKID
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def __hash__(self) -> int:
|
|
|
|
return hash((self.__mRawPointer, self.__mCKID))
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region Valid Check, Init and Dispose
|
|
|
|
|
|
|
|
def is_bmap_available() -> bool:
|
|
|
|
return bmap.is_bmap_available()
|
|
|
|
|
|
|
|
# init module self and register exit function
|
|
|
|
if is_bmap_available():
|
|
|
|
bmap.BMInit()
|
|
|
|
|
|
|
|
def _auto_exit():
|
|
|
|
bmap.BMDispose()
|
|
|
|
atexit.register(_auto_exit)
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region Real Type Defines
|
|
|
|
|
2023-11-05 10:44:11 +08:00
|
|
|
"""!
|
|
|
|
BMFileReader, BMFileWriter, and BMMeshTrans can be create by given constructor.
|
|
|
|
But they must be destroyed by calling dispose(). Otherwise it may cause memory leak.
|
|
|
|
You also can use python `with` statement to achieve this automatically.
|
|
|
|
|
|
|
|
BMObject, BMTexture, BMMaterial, BMMesh, and BM3dObject should NOT be constructed from given constructor.
|
|
|
|
They must be obtained from BMFileReader, BMFileWriter, and BMMeshTrans.
|
|
|
|
Thus BMObject, BMTexture, BMMaterial, BMMesh, and BM3dObject also do not need to free
|
|
|
|
because these resources are sotred in BMFileReader, BMFileWriter, and BMMeshTrans.
|
|
|
|
We just provide them as a visitor.
|
|
|
|
"""
|
|
|
|
|
2023-11-04 21:58:58 +08:00
|
|
|
class BMFileReader(_AbstractPointer):
|
2023-11-05 10:44:11 +08:00
|
|
|
def __init__(self, file_name: str, temp_folder: str, texture_folder: str, encodings: tuple[str]):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_value, traceback):
|
|
|
|
self.dispose()
|
|
|
|
|
|
|
|
def dispose(self) -> None:
|
|
|
|
if self.is_valid():
|
|
|
|
bmap.BMFile_Free(self._get_pointer())
|
2023-11-04 21:58:58 +08:00
|
|
|
|
|
|
|
class BMFileWriter(_AbstractPointer):
|
2023-11-05 10:44:11 +08:00
|
|
|
def __init__(self, temp_folder: str, texture_folder: str, encodings: tuple[str]):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_value, traceback):
|
|
|
|
self.dispose()
|
|
|
|
|
|
|
|
def save(self, file_name: str, compress_level: int) -> None:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def dispose(self) -> None:
|
|
|
|
if self.is_valid():
|
|
|
|
bmap.BMFile_Free(self._get_pointer())
|
2023-11-04 21:58:58 +08:00
|
|
|
|
|
|
|
class BMMeshTrans(_AbstractPointer):
|
2023-11-05 10:44:11 +08:00
|
|
|
def __init__(self):
|
|
|
|
ptr: bmap.bm_void_p = bmap.bm_void_p()
|
|
|
|
bmap.BMMeshTrans_New(ctypes.byref(ptr))
|
|
|
|
_AbstractPointer.__init__(ptr)
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_value, traceback):
|
|
|
|
self.dispose()
|
|
|
|
|
|
|
|
def dispose(self) -> None:
|
|
|
|
if self.is_valid():
|
|
|
|
bmap.BMMeshTrans_Delete(self._get_pointer())
|
|
|
|
|
2023-11-04 21:58:58 +08:00
|
|
|
|
|
|
|
class BMObject(_AbstractCKObject):
|
2023-11-05 10:44:11 +08:00
|
|
|
def get_name(self) -> str | None:
|
2023-11-04 21:58:58 +08:00
|
|
|
name: bmap.bm_CKSTRING = bmap.bm_CKSTRING()
|
|
|
|
bmap.BMObject_GetName(self._get_pointer(), self._get_ckid(), ctypes.byref(name))
|
2023-11-05 10:44:11 +08:00
|
|
|
if name.value is None:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return name.value.decode(g_BMapEncoding)
|
2023-11-04 21:58:58 +08:00
|
|
|
|
2023-11-05 10:44:11 +08:00
|
|
|
def set_name(self, name: str | None) -> None:
|
|
|
|
name: bmap.bm_CKSTRING
|
|
|
|
if name is None:
|
|
|
|
name = bmap.bm_CKSTRING(0)
|
|
|
|
else:
|
|
|
|
name = bmap.bm_CKSTRING(name.encode(g_BMapEncoding))
|
2023-11-04 21:58:58 +08:00
|
|
|
bmap.BMObject_SetName(self._get_pointer(), self._get_ckid(), name)
|
|
|
|
|
|
|
|
class BMGroup(BMObject):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class BMTexture(BMObject):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class BMMaterial(BMObject):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class BMMesh(BMObject):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class BM3dObject(BMObject):
|
2023-11-05 10:44:11 +08:00
|
|
|
def get_world_matrix(self) -> virtools_types.ConstVxMatrix:
|
|
|
|
mat: bmap.bm_VxMatrix = bmap.bm_VxMatrix()
|
|
|
|
bmap.BM3dObject_GetWorldMatrix(self._get_pointer(), self._get_ckid(), ctypes.byref(mat))
|
|
|
|
# use cast & pointer to get matrix data conveniently
|
|
|
|
flat: bmap.bm_CKFLOAT_p = ctypes.cast(ctypes.byref(mat), bmap.bm_CKFLOAT_p)
|
|
|
|
return tuple(flat[i] for i in range(16))
|
|
|
|
|
|
|
|
def set_world_matrix(self, mat: virtools_types.ConstVxMatrix) -> None:
|
|
|
|
# star syntax expand the tuple as the argument.
|
|
|
|
mat: bmap.bm_VxMatrix = bmap.bm_VxMatrix(*mat)
|
|
|
|
bmap.BM3dObject_SetWorldMatrix(self._get_pointer(), self._get_ckid(), mat)
|
|
|
|
|
|
|
|
def get_current_mesh(self) -> BMMesh | None:
|
|
|
|
ckid: bmap.bm_CKID = bmap.bm_CKID()
|
|
|
|
bmap.BM3dObject_GetCurrentMesh(self._get_pointer(), self._get_ckid(), ctypes.byref(ckid))
|
|
|
|
if ckid.value == g_InvalidCKID:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return BMMesh(self._get_pointer(), ckid)
|
|
|
|
|
|
|
|
def set_current_mesh(self, mesh: BMMesh | None) -> None:
|
|
|
|
ckid: bmap.bm_CKID
|
|
|
|
if mesh is None:
|
|
|
|
ckid = bmap.bm_CKID(g_InvalidCKID)
|
|
|
|
else:
|
|
|
|
ckid = bmap.bm_CKID(mesh._get_ckid())
|
|
|
|
bmap.BM3dObject_SetCurrentMesh(self._get_pointer(), self._get_ckid(), ckid)
|
|
|
|
|
|
|
|
def get_visibility(self) -> bool:
|
|
|
|
visb: bmap.bm_bool = bmap.bm_bool()
|
|
|
|
bmap.BM3dObject_GetVisibility(self._get_pointer(), self._get_ckid(), ctypes.byref(visb))
|
|
|
|
return visb.value
|
|
|
|
|
|
|
|
def set_visibility(self, visb_: bool) -> None:
|
|
|
|
visb: bmap.bm_bool = bmap.bm_bool(visb_)
|
|
|
|
bmap.BM3dObject_SetVisibility(self._get_pointer(), self._get_ckid(), visb)
|
2023-11-04 21:58:58 +08:00
|
|
|
|
|
|
|
#endregion
|