-
Notifications
You must be signed in to change notification settings - Fork 117
/
dnacalib_neutral_mesh_subtract.py
139 lines (109 loc) · 5.15 KB
/
dnacalib_neutral_mesh_subtract.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
"""
This example demonstrates how to subtract values from a neutral mesh and transfer those changes to its lower LOD meshes.
IMPORTANT: You have to setup the environment before running this example. Please refer to the 'Environment setup' section in README.md.
- usage in command line:
python dnacalib_neutral_mesh_subtract.py
mayapy dnacalib_neutral_mesh_subtract.py
- usage in Maya:
1. copy whole content of this file to Maya Script Editor
2. change value of ROOT_DIR to absolute path of dna_calibration, e.g. `c:/dna_calibration` in Windows or `/home/user/dna_calibration`. Important:
Use `/` (forward slash), because Maya uses forward slashes in path.
- customization:
- change CHARACTER_NAME to Taro, or the name of a custom DNA file placed in /data/dna_files
Expected: Script will generate Ada_output.dna in OUTPUT_DIR from original Ada.dna.
NOTE: If OUTPUT_DIR does not exist, it will be created.
"""
from os import makedirs
from os import path as ospath
# if you use Maya, use absolute path
ROOT_DIR = f"{ospath.dirname(ospath.abspath(__file__))}/..".replace("\\", "/")
OUTPUT_DIR = f"{ROOT_DIR}/output"
CHARACTER_NAME = "Ada"
DATA_DIR = f"{ROOT_DIR}/data"
CHARACTER_DNA = f"{DATA_DIR}/dna_files/{CHARACTER_NAME}.dna"
OUTPUT_DNA = f"{OUTPUT_DIR}/{CHARACTER_NAME}_output.dna"
from dna import DataLayer_All, FileStream, Status, BinaryStreamReader, BinaryStreamWriter
from dnacalib import (
DNACalibDNAReader,
SetVertexPositionsCommand,
VectorOperation_Subtract,
CalculateMeshLowerLODsCommand
)
from math import isclose
def load_dna(path):
stream = FileStream(path, FileStream.AccessMode_Read, FileStream.OpenMode_Binary)
reader = BinaryStreamReader(stream, DataLayer_All)
reader.read()
if not Status.isOk():
status = Status.get()
raise RuntimeError(f"Error loading DNA: {status.message}")
return reader
def save_dna(reader, path):
stream = FileStream(path, FileStream.AccessMode_Write, FileStream.OpenMode_Binary)
writer = BinaryStreamWriter(stream)
writer.setFrom(reader)
writer.write()
if not Status.isOk():
status = Status.get()
raise RuntimeError(f"Error saving DNA: {status.message}")
def calibrate_dna(input_path, output_path):
dna = load_dna(input_path)
# Copies DNA contents and will serve as input/output parameter to commands
calibrated = DNACalibDNAReader(dna)
if calibrated.getMeshCount() == 0:
print("No meshes found in DNA.")
return
mesh_index = 0
vtx_count_mesh0 = calibrated.getVertexPositionCount(mesh_index)
xs = calibrated.getVertexPositionXs(mesh_index)
ys = calibrated.getVertexPositionYs(mesh_index)
zs = calibrated.getVertexPositionZs(mesh_index)
# Example values to subtract from original vertex positions for mesh with index 0
ones = [1.0] * vtx_count_mesh0
# Command used to subtract example values from a specified neutral mesh
subtract_command = SetVertexPositionsCommand()
subtract_command.setMeshIndex(mesh_index)
subtract_command.setOperation(VectorOperation_Subtract)
subtract_command.setPositions(ones, ones, ones)
# Alternatively, if you wanted to do the opposite, to subtract neutral mesh values
# from example values, you could do the following:
# xs_2 = [x * 2 for x in xs]
# ys_2 = [y * 2 for y in ys]
# zs_2 = [z * 2 for z in zs]
# subtract_command.setPositions(xs_2, ys_2, zs_2)
# # After running the command, the calibrated DNA will contain -xs, -ys, -zs
# subtract_command.run(calibrated)
#
# # Add the example values, which will result in:
# # ones - xs
# # ones - ys
# # ones - zs
# add_command = SetVertexPositionsCommand()
# add_command.setMeshIndex(mesh_index)
# add_command.setOperation(VectorOperation_Add) # remember to add VectorOperation_Add to the list of imports
# add_command.setPositions(ones, ones, ones)
# add_command.run(calibrated)
print(f"Subtracting values from neutral mesh \'{calibrated.getMeshName(mesh_index)}\'...")
# Modifies calibrated DNA in-place
subtract_command.run(calibrated)
# Command used to recalculate vertex positions of lower LOD neutral meshes of the specified neutral mesh
calculate_lower_lods_command = CalculateMeshLowerLODsCommand()
calculate_lower_lods_command.setMeshIndex(mesh_index)
print("Recalculating values for lower LOD meshes...")
# Modifies calibrated DNA in-place
calculate_lower_lods_command.run(calibrated)
new_xs = calibrated.getVertexPositionXs(mesh_index)
new_ys = calibrated.getVertexPositionYs(mesh_index)
new_zs = calibrated.getVertexPositionZs(mesh_index)
for i in range(vtx_count_mesh0):
if (not isclose(xs[i] - 1, new_xs[i], rel_tol=1e-7) or
not isclose(ys[i] - 1, new_ys[i], rel_tol=1e-7) or
not isclose(zs[i] - 1, new_zs[i], rel_tol=1e-7)):
raise RuntimeError("Vertex positions were not changed successfully!")
print("\nSuccessfully changed vertex positions.")
print("Saving DNA...")
save_dna(calibrated, output_path)
print("Done.")
if __name__ == "__main__":
makedirs(OUTPUT_DIR, exist_ok=True)
calibrate_dna(CHARACTER_DNA, OUTPUT_DNA)