Skip to content

Commit a6e0219

Browse files
Utilize Fusion's Unit Manager [AARD-1773] (#1101)
2 parents 1da19af + ae6e78f commit a6e0219

File tree

9 files changed

+87
-200
lines changed

9 files changed

+87
-200
lines changed

exporter/SynthesisFusionAddin/src/Parser/ExporterOptions.py

+1-5
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@
2121
Joint,
2222
ModelHierarchy,
2323
PhysicalDepth,
24-
PreferredUnits,
2524
Wheel,
2625
encodeNestedObjects,
2726
makeObjectFromJson,
@@ -43,10 +42,7 @@ class ExporterOptions:
4342
wheels: list[Wheel] = field(default_factory=list)
4443
joints: list[Joint] = field(default_factory=list)
4544
gamepieces: list[Gamepiece] = field(default_factory=list)
46-
preferredUnits: PreferredUnits = field(default=PreferredUnits.IMPERIAL)
47-
48-
# Always stored in kg regardless of 'preferredUnits'
49-
robotWeight: KG = field(default=0.0)
45+
robotWeight: KG = field(default=KG(0.0))
5046
autoCalcRobotWeight: bool = field(default=False)
5147
autoCalcGamepieceWeight: bool = field(default=False)
5248

Binary file not shown.
Binary file not shown.

exporter/SynthesisFusionAddin/src/Types.py

+12-12
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,24 @@
55
from enum import Enum, EnumType
66
from typing import Any, TypeAlias, get_args, get_origin
77

8+
import adsk.fusion
9+
810
# Not 100% sure what this is for - Brandon
911
JointParentType = Enum("JointParentType", ["ROOT", "END"])
1012

1113
WheelType = Enum("WheelType", ["STANDARD", "OMNI", "MECANUM"])
1214
SignalType = Enum("SignalType", ["PWM", "CAN", "PASSIVE"])
1315
ExportMode = Enum("ExportMode", ["ROBOT", "FIELD"]) # Dynamic / Static export
14-
PreferredUnits = Enum("PreferredUnits", ["METRIC", "IMPERIAL"])
1516
ExportLocation = Enum("ExportLocation", ["UPLOAD", "DOWNLOAD"])
17+
UnitSystem = Enum("UnitSystem", ["METRIC", "IMPERIAL"])
18+
19+
FUSION_UNIT_SYSTEM: dict[int, UnitSystem] = {
20+
adsk.fusion.DistanceUnits.MillimeterDistanceUnits: UnitSystem.METRIC,
21+
adsk.fusion.DistanceUnits.CentimeterDistanceUnits: UnitSystem.METRIC,
22+
adsk.fusion.DistanceUnits.MeterDistanceUnits: UnitSystem.METRIC,
23+
adsk.fusion.DistanceUnits.InchDistanceUnits: UnitSystem.IMPERIAL,
24+
adsk.fusion.DistanceUnits.FootDistanceUnits: UnitSystem.IMPERIAL,
25+
}
1626

1727

1828
@dataclass
@@ -72,18 +82,8 @@ class ModelHierarchy(Enum):
7282
SingleMesh = 3
7383

7484

75-
LBS: TypeAlias = float
7685
KG: TypeAlias = float
77-
78-
79-
def toLbs(kgs: KG) -> LBS:
80-
return LBS(round(kgs * 2.2062, 2))
81-
82-
83-
def toKg(pounds: LBS) -> KG:
84-
return KG(round(pounds / 2.2062, 2))
85-
86-
86+
LBS: TypeAlias = float
8787
PRIMITIVES = (bool, str, int, float, type(None))
8888

8989

exporter/SynthesisFusionAddin/src/UI/ConfigCommand.py

-7
Original file line numberDiff line numberDiff line change
@@ -331,12 +331,6 @@ def notify(self, args: adsk.core.CommandEventArgs) -> None:
331331
selectedJoints, selectedWheels = jointConfigTab.getSelectedJointsAndWheels()
332332
selectedGamepieces = gamepieceConfigTab.getGamepieces()
333333

334-
if generalConfigTab.exportMode == ExportMode.ROBOT:
335-
units = generalConfigTab.selectedUnits
336-
else:
337-
assert generalConfigTab.exportMode == ExportMode.FIELD
338-
units = gamepieceConfigTab.selectedUnits
339-
340334
exporterOptions = ExporterOptions(
341335
str(savepath),
342336
name,
@@ -345,7 +339,6 @@ def notify(self, args: adsk.core.CommandEventArgs) -> None:
345339
joints=selectedJoints,
346340
wheels=selectedWheels,
347341
gamepieces=selectedGamepieces,
348-
preferredUnits=units,
349342
robotWeight=generalConfigTab.robotWeight,
350343
autoCalcRobotWeight=generalConfigTab.autoCalculateWeight,
351344
autoCalcGamepieceWeight=gamepieceConfigTab.autoCalculateWeight,

exporter/SynthesisFusionAddin/src/UI/GamepieceConfigTab.py

+15-63
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,13 @@
33

44
from src.Logging import logFailure
55
from src.Parser.ExporterOptions import ExporterOptions
6-
from src.Types import Gamepiece, PreferredUnits, toKg, toLbs
7-
from src.UI import IconPaths
6+
from src.Types import Gamepiece, UnitSystem
87
from src.UI.CreateCommandInputsHelper import (
98
createBooleanInput,
109
createTableInput,
1110
createTextBoxInput,
1211
)
12+
from src.Util import convertMassUnitsFrom, convertMassUnitsTo, getFusionUnitSystem
1313

1414

1515
class GamepieceConfigTab:
@@ -19,7 +19,6 @@ class GamepieceConfigTab:
1919
gamepieceTable: adsk.core.TableCommandInput
2020
previousAutoCalcWeightCheckboxState: bool
2121
previousSelectedUnitDropdownIndex: int
22-
currentUnits: PreferredUnits
2322

2423
@logFailure
2524
def __init__(self, args: adsk.core.CommandCreatedEventArgs, exporterOptions: ExporterOptions) -> None:
@@ -37,20 +36,6 @@ def __init__(self, args: adsk.core.CommandCreatedEventArgs, exporterOptions: Exp
3736
)
3837
self.previousAutoCalcWeightCheckboxState = exporterOptions.autoCalcGamepieceWeight
3938

40-
self.currentUnits = exporterOptions.preferredUnits
41-
imperialUnits = self.currentUnits == PreferredUnits.IMPERIAL
42-
weightUnitTable = gamepieceTabInputs.addDropDownCommandInput(
43-
"gamepieceWeightUnit", "Unit of Mass", adsk.core.DropDownStyles.LabeledIconDropDownStyle
44-
)
45-
46-
# Invisible white space characters are required in the list item name field to make this work.
47-
# I have no idea why, Fusion API needs some special education help - Brandon
48-
weightUnitTable.listItems.add("‎", imperialUnits, IconPaths.massIcons["LBS"])
49-
weightUnitTable.listItems.add("‎", not imperialUnits, IconPaths.massIcons["KG"])
50-
weightUnitTable.tooltip = "Unit of mass"
51-
weightUnitTable.tooltipDescription = "<hr>Configure the unit of mass for for the weight calculation."
52-
self.previousSelectedUnitDropdownIndex = int(not imperialUnits)
53-
5439
self.gamepieceTable = createTableInput(
5540
"gamepieceTable",
5641
"Gamepiece",
@@ -62,8 +47,17 @@ def __init__(self, args: adsk.core.CommandCreatedEventArgs, exporterOptions: Exp
6247
self.gamepieceTable.addCommandInput(
6348
createTextBoxInput("gamepieceNameHeader", "Name", gamepieceTabInputs, "Name", bold=False), 0, 0
6449
)
50+
fusUnitSystem = getFusionUnitSystem()
6551
self.gamepieceTable.addCommandInput(
66-
createTextBoxInput("gamepieceWeightHeader", "Weight", gamepieceTabInputs, "Weight", bold=False), 0, 1
52+
createTextBoxInput(
53+
"gamepieceWeightHeader",
54+
"Weight",
55+
gamepieceTabInputs,
56+
f"Weight {'(lbs)' if fusUnitSystem is UnitSystem.IMPERIAL else '(kg)'}",
57+
bold=False,
58+
),
59+
0,
60+
1,
6761
)
6862
self.gamepieceTable.addCommandInput(
6963
createTextBoxInput(
@@ -112,10 +106,6 @@ def isVisible(self, value: bool) -> None:
112106
def isActive(self) -> bool:
113107
return self.gamepieceConfigTab.isActive or False
114108

115-
@property
116-
def selectedUnits(self) -> PreferredUnits:
117-
return self.currentUnits
118-
119109
@property
120110
def autoCalculateWeight(self) -> bool:
121111
autoCalcWeightButton: adsk.core.BoolValueCommandInput = self.gamepieceConfigTab.children.itemById(
@@ -168,26 +158,13 @@ def addChildOccurrences(childOccurrences: adsk.fusion.OccurrenceList) -> None:
168158
frictionCoefficient.valueOne = 0.5
169159

170160
physical = gamepiece.component.getPhysicalProperties(adsk.fusion.CalculationAccuracy.LowCalculationAccuracy)
171-
if self.currentUnits == PreferredUnits.IMPERIAL:
172-
gamepieceMass = toLbs(physical.mass)
173-
else:
174-
gamepieceMass = round(physical.mass, 2)
175-
161+
gamepieceMass = round(convertMassUnitsFrom(physical.mass), 2)
176162
weight = commandInputs.addValueInput(
177163
"gamepieceWeight", "Weight Input", "", adsk.core.ValueInput.createByString(str(gamepieceMass))
178164
)
179165
weight.tooltip = "Weight of field element"
180166
weight.isEnabled = not self.previousAutoCalcWeightCheckboxState
181167

182-
weightUnitDropdown: adsk.core.DropDownCommandInput = self.gamepieceConfigTab.children.itemById(
183-
"gamepieceWeightUnit"
184-
)
185-
if weightUnitDropdown.selectedItem.index == 0:
186-
weight.tooltipDescription = "<tt>(in pounds)</tt>"
187-
else:
188-
assert weightUnitDropdown.selectedItem.index == 1
189-
weight.tooltipDescription = "<tt>(in kilograms)</tt>"
190-
191168
row = self.gamepieceTable.rowCount
192169
self.gamepieceTable.addCommandInput(gamepieceName, row, 0)
193170
self.gamepieceTable.addCommandInput(weight, row, 1)
@@ -222,7 +199,7 @@ def getGamepieces(self) -> list[Gamepiece]:
222199
gamepieces: list[Gamepiece] = []
223200
for row in range(1, self.gamepieceTable.rowCount): # Row is 1 indexed
224201
gamepieceEntityToken = self.selectedGamepieceList[row - 1].entityToken
225-
gamepieceWeight = self.gamepieceTable.getInputAtPosition(row, 1).value
202+
gamepieceWeight = convertMassUnitsTo(self.gamepieceTable.getInputAtPosition(row, 1).value)
226203
gamepieceFrictionCoefficient = self.gamepieceTable.getInputAtPosition(row, 2).valueOne
227204
gamepieces.append(Gamepiece(gamepieceEntityToken, gamepieceWeight, gamepieceFrictionCoefficient))
228205

@@ -232,25 +209,14 @@ def reset(self) -> None:
232209
self.selectedGamepieceEntityIDs.clear()
233210
self.selectedGamepieceList.clear()
234211

235-
@logFailure
236-
def updateWeightTableToUnits(self, units: PreferredUnits) -> None:
237-
assert units in {PreferredUnits.METRIC, PreferredUnits.IMPERIAL}
238-
conversionFunc = toKg if units == PreferredUnits.METRIC else toLbs
239-
for row in range(1, self.gamepieceTable.rowCount): # Row is 1 indexed
240-
weightInput: adsk.core.ValueCommandInput = self.gamepieceTable.getInputAtPosition(row, 1)
241-
weightInput.value = conversionFunc(weightInput.value)
242-
243212
@logFailure
244213
def calcGamepieceWeights(self) -> None:
245214
for row in range(1, self.gamepieceTable.rowCount): # Row is 1 indexed
246215
weightInput: adsk.core.ValueCommandInput = self.gamepieceTable.getInputAtPosition(row, 1)
247216
physical = self.selectedGamepieceList[row - 1].component.getPhysicalProperties(
248217
adsk.fusion.CalculationAccuracy.LowCalculationAccuracy
249218
)
250-
if self.currentUnits == PreferredUnits.IMPERIAL:
251-
weightInput.value = toLbs(physical.mass)
252-
else:
253-
weightInput.value = round(physical.mass, 2)
219+
weightInput.value = round(convertMassUnitsFrom(physical.mass), 2)
254220

255221
@logFailure
256222
def handleInputChanged(
@@ -283,20 +249,6 @@ def handleInputChanged(
283249

284250
self.previousAutoCalcWeightCheckboxState = autoCalcWeightButton.value
285251

286-
elif commandInput.id == "gamepieceWeightUnit":
287-
weightUnitDropdown = adsk.core.DropDownCommandInput.cast(commandInput)
288-
if weightUnitDropdown.selectedItem.index == self.previousSelectedUnitDropdownIndex:
289-
return
290-
291-
if weightUnitDropdown.selectedItem.index == 0:
292-
self.currentUnits = PreferredUnits.IMPERIAL
293-
else:
294-
assert weightUnitDropdown.selectedItem.index == 1
295-
self.currentUnits = PreferredUnits.METRIC
296-
297-
self.updateWeightTableToUnits(self.currentUnits)
298-
self.previousSelectedUnitDropdownIndex = weightUnitDropdown.selectedItem.index
299-
300252
elif commandInput.id == "gamepieceAddButton":
301253
gamepieceSelection.isVisible = gamepieceSelection.isEnabled = True
302254
gamepieceSelection.clearSelection()

0 commit comments

Comments
 (0)