Cura/scripts/obj_trimmer.py
2022-01-31 13:35:42 +01:00

133 lines
4.0 KiB
Python

# Copyright (c) 2022 Ultimaker B.V.
# Cura is released under the terms of the LGPLv3 or higher.
import argparse
import os
from typing import Optional, List, TextIO
"""
Used to reduce the size of obj files used for printer platform models.
Trims trailing 0 from coordinates
Removes duplicate vertex texture coordinates
Removes any rows that are not a face, vertex or vertex texture
"""
def process_obj(input_file: str, output_file: str) -> None:
with open(input_file, "r") as in_obj, open("temp", "w") as temp:
trim_lines(in_obj, temp)
with open("temp", "r") as temp, open(output_file, "w") as out_obj:
merge_duplicate_vt(temp, out_obj)
os.remove("temp")
def trim_lines(in_obj: TextIO, out_obj: TextIO) -> None:
for line in in_obj:
line = trim_line(line)
if line:
out_obj.write(line + "\n")
def trim_line(line: str) -> Optional[str]:
# Discards all rows that are not a vertex ("v"), face ("f") or vertex texture ("vt")
values = line.split()
if values[0] == "vt":
return trim_vertex_texture(values)
elif values[0] == "f":
return trim_face(values)
elif values[0] == "v":
return trim_vertex(values)
return
def trim_face(values: List[str]) -> str:
# Removes face normals (vn)
# f 15/15/17 15/15/17 14/14/17 -> f 15/15 15/15 14/14
for i, coordinates in enumerate(values[1:]):
v, vt = coordinates.split("/")[:2]
values[i + 1] = v + "/" + vt
return " ".join(values)
def trim_vertex(values: List[str]) -> str:
# Removes trailing zeros from vertex coordinates
# v 0.044000 0.137000 0.123000 -> v 0.044 0.137 0.123
for i, coordinate in enumerate(values[1:]):
values[i + 1] = str(float(coordinate))
return " ".join(values)
def trim_vertex_texture(values: List[str]) -> str:
# Removes trailing zeros from vertex texture coordinates
# vt 0.137000 0.123000 -> v 0.137 0.123
for i, coordinate in enumerate(values[1:]):
values[i + 1] = str(float(coordinate))
return " ".join(values)
def merge_duplicate_vt(in_obj, out_obj):
# Removes duplicate vertex texture ("vt")
# Points references to all deleted copies in face ("f") to a single vertex texture
# Maps index of all copies of a vt line to the same index
vt_index_mapping = {}
# Maps vt line to index ("vt 0.043 0.137" -> 23)
vt_to_index = {}
# .obj file indexes start at 1
vt_index = 1
skipped_count = 0
# First write everything except faces
for line in in_obj.readlines():
if line[0] == "f":
continue
if line[:2] == "vt":
if line in vt_to_index.keys():
# vt with same value has already been written
# this points the current vt index to the one that has been written
vt_index_mapping[vt_index] = vt_to_index[line]
skipped_count += 1
else:
# vt has not been seen, point vt line to index
vt_to_index[line] = vt_index - skipped_count
vt_index_mapping[vt_index] = vt_index - skipped_count
out_obj.write(line)
vt_index += 1
else:
out_obj.write(line)
# Second pass remaps face vt index
in_obj.seek(0)
for line in in_obj.readlines():
if line[0] != "f":
continue
values = line.split()
for i, coordinates in enumerate(values[1:]):
v, vt = coordinates.split("/")[:2]
vt = int(vt)
if vt in vt_index_mapping.keys():
vt = vt_index_mapping[vt]
values[i + 1] = v + "/" + str(vt)
out_obj.write(" ".join(values) + "\n")
if __name__ == "__main__":
parser = argparse.ArgumentParser(description = "Reduce the size of a .obj file")
parser.add_argument("input_file", type = str, help = "Input .obj file name")
parser.add_argument("--output_file", default = "output.obj", type = str, help = "Output .obj file name")
args = parser.parse_args()
process_obj(args.input_file, args.output_file)