MoGe / utils3d /io /ply.py
Ruicheng's picture
first commit
ec0c8fa
raw
history blame
3.9 kB
import numpy as np
from typing import *
from pathlib import Path
def read_ply(
file: Union[str, Path],
encoding: Union[str, None] = None,
ignore_unknown: bool = False
) -> Tuple[np.ndarray, np.ndarray]:
"""
Read .ply file, without preprocessing.
Args:
file (Any): filepath
encoding (str, optional):
Returns:
Tuple[np.ndarray, np.ndarray]: vertices, faces
"""
import plyfile
plydata = plyfile.PlyData.read(file)
vertices = np.stack([plydata['vertex'][k] for k in ['x', 'y', 'z']], axis=-1)
if 'face' in plydata:
faces = np.array(plydata['face']['vertex_indices'].tolist())
else:
faces = None
return vertices, faces
def write_ply(
file: Union[str, Path],
vertices: np.ndarray,
faces: np.ndarray = None,
edges: np.ndarray = None,
vertex_colors: np.ndarray = None,
edge_colors: np.ndarray = None,
text: bool = False
):
"""
Write .ply file, without preprocessing.
Args:
file (Any): filepath
vertices (np.ndarray): [N, 3]
faces (np.ndarray): [T, E]
edges (np.ndarray): [E, 2]
vertex_colors (np.ndarray, optional): [N, 3]. Defaults to None.
edge_colors (np.ndarray, optional): [E, 3]. Defaults to None.
text (bool, optional): save data in text format. Defaults to False.
"""
import plyfile
assert vertices.ndim == 2 and vertices.shape[1] == 3
vertices = vertices.astype(np.float32)
if faces is not None:
assert faces.ndim == 2
faces = faces.astype(np.int32)
if edges is not None:
assert edges.ndim == 2 and edges.shape[1] == 2
edges = edges.astype(np.int32)
if vertex_colors is not None:
assert vertex_colors.ndim == 2 and vertex_colors.shape[1] == 3
if vertex_colors.dtype in [np.float32, np.float64]:
vertex_colors = vertex_colors * 255
vertex_colors = np.clip(vertex_colors, 0, 255).astype(np.uint8)
vertices_data = np.zeros(len(vertices), dtype=[('x', 'f4'), ('y', 'f4'), ('z', 'f4'), ('red', 'u1'), ('green', 'u1'), ('blue', 'u1')])
vertices_data['x'] = vertices[:, 0]
vertices_data['y'] = vertices[:, 1]
vertices_data['z'] = vertices[:, 2]
vertices_data['red'] = vertex_colors[:, 0]
vertices_data['green'] = vertex_colors[:, 1]
vertices_data['blue'] = vertex_colors[:, 2]
else:
vertices_data = np.array([tuple(v) for v in vertices], dtype=[('x', 'f4'), ('y', 'f4'), ('z', 'f4')])
if faces is not None:
faces_data = np.zeros(len(faces), dtype=[('vertex_indices', 'i4', (faces.shape[1],))])
faces_data['vertex_indices'] = faces
if edges is not None:
if edge_colors is not None:
assert edge_colors.ndim == 2 and edge_colors.shape[1] == 3
if edge_colors.dtype in [np.float32, np.float64]:
edge_colors = edge_colors * 255
edge_colors = np.clip(edge_colors, 0, 255).astype(np.uint8)
edges_data = np.zeros(len(edges), dtype=[('vertex1', 'i4'), ('vertex2', 'i4'), ('red', 'u1'), ('green', 'u1'), ('blue', 'u1')])
edges_data['vertex1'] = edges[:, 0]
edges_data['vertex2'] = edges[:, 1]
edges_data['red'] = edge_colors[:, 0]
edges_data['green'] = edge_colors[:, 1]
edges_data['blue'] = edge_colors[:, 2]
else:
edges_data = np.array([tuple(e) for e in edges], dtype=[('vertex1', 'i4'), ('vertex2', 'i4')])
ply_data = [plyfile.PlyElement.describe(vertices_data, 'vertex')]
if faces is not None:
ply_data.append(plyfile.PlyElement.describe(faces_data, 'face'))
if edges is not None:
ply_data.append(plyfile.PlyElement.describe(edges_data, 'edge'))
plyfile.PlyData(ply_data, text=text).write(file)