Skip to content

Commit

Permalink
EMSUSD-947 units conversion for USD import
Browse files Browse the repository at this point in the history
- Add -unit (-unt) option to the import command.
- Document it.
- Add unit option to the job import argument structure.
- Parse the unit option.
- Expose it to Python.
- Add check-box UI to the import UI.
- Add private structure to the import read job class to hold all conversion info.
- Remove old warning about unit conversion not being implemented.
- Refactor the code for up-axis conversion to also handle the units conversion.
- Change prefs for units if requested and needed.
- Add unit tests.
  • Loading branch information
pierrebai-adsk committed Oct 2, 2024
1 parent 78890d7 commit 4174814
Show file tree
Hide file tree
Showing 16 changed files with 329 additions and 74 deletions.
1 change: 1 addition & 0 deletions lib/mayaUsd/commands/Readme.md
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ Each base command class is documented in the following sections.
| `-importUSDZTexturesFilePath` | `-itf` | string | none | Specifies an explicit directory to write imported textures to from a USDZ archive. Has no effect if `-importUSDZTextures` is not specified. |
| `-importRelativeTextures` | `-rtx` | string | none | Selects how textures filenames are generated: absolute, relative, automatic or none. When automatic, the filename is relative if the source filename of the texture being imported is relative. When none, the file path is left alone, for backward compatible behavior. |
| `-upAxis` | `-upa` | bool | true | Enable changing axis on import. |
| `-unit` | `-unt` | bool | true | Enable changing units on import. |
| `-axisAndUnitMethod` | `-aum` | string | rotateScale | Selects how the unit and axis are handled during import. |

### Return Value
Expand Down
1 change: 1 addition & 0 deletions lib/mayaUsd/commands/baseImportCommand.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@ MSyntax MayaUSDImportCommand::createSyntax()
MSyntax::kString);
syntax.addFlag(
kImportUpAxisFlag, UsdMayaJobImportArgsTokens->upAxis.GetText(), MSyntax::kBoolean);
syntax.addFlag(kImportUnitFlag, UsdMayaJobImportArgsTokens->unit.GetText(), MSyntax::kBoolean);
syntax.addFlag(
kImportAxisAndUnitMethodFlag,
UsdMayaJobImportArgsTokens->axisAndUnitMethod.GetText(),
Expand Down
1 change: 1 addition & 0 deletions lib/mayaUsd/commands/baseImportCommand.h
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ class MAYAUSD_CORE_PUBLIC MayaUSDImportCommand : public MPxCommand
static constexpr auto kImportUSDZTexturesFilePathFlag = "itf";
static constexpr auto kImportRelativeTexturesFlag = "rtx";
static constexpr auto kImportUpAxisFlag = "upa";
static constexpr auto kImportUnitFlag = "unt";
static constexpr auto kImportAxisAndUnitMethodFlag = "aum";
static constexpr auto kMetadataFlag = "md";
static constexpr auto kApiSchemaFlag = "api";
Expand Down
4 changes: 4 additions & 0 deletions lib/mayaUsd/fileio/jobs/jobArgs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1382,6 +1382,7 @@ UsdMayaJobImportArgs::UsdMayaJobImportArgs(
UsdMayaJobImportArgsTokens->addTransform,
UsdMayaJobImportArgsTokens->overwritePrefs }))
, upAxis(extractBoolean(userArgs, UsdMayaJobImportArgsTokens->upAxis))
, unit(extractBoolean(userArgs, UsdMayaJobImportArgsTokens->unit))
, importInstances(extractBoolean(userArgs, UsdMayaJobImportArgsTokens->importInstances))
, useAsAnimationCache(extractBoolean(userArgs, UsdMayaJobImportArgsTokens->useAsAnimationCache))
, importWithProxyShapes(importWithProxyShapes)
Expand Down Expand Up @@ -1473,6 +1474,7 @@ const VtDictionary& UsdMayaJobImportArgs::GetDefaultDictionary()
d[UsdMayaJobImportArgsTokens->axisAndUnitMethod]
= UsdMayaJobImportArgsTokens->rotateScale.GetString();
d[UsdMayaJobImportArgsTokens->upAxis] = true;
d[UsdMayaJobImportArgsTokens->unit] = true;
d[UsdMayaJobImportArgsTokens->pullImportStage] = UsdStageRefPtr();
d[UsdMayaJobImportArgsTokens->useAsAnimationCache] = false;
d[UsdMayaJobImportArgsTokens->preserveTimeline] = false;
Expand Down Expand Up @@ -1557,6 +1559,7 @@ const VtDictionary& UsdMayaJobImportArgs::GetGuideDictionary()
d[UsdMayaJobImportArgsTokens->importRelativeTextures] = _string;
d[UsdMayaJobImportArgsTokens->axisAndUnitMethod] = _string;
d[UsdMayaJobImportArgsTokens->upAxis] = _boolean;
d[UsdMayaJobImportArgsTokens->unit] = _boolean;
d[UsdMayaJobImportArgsTokens->pullImportStage] = _usdStageRefPtr;
d[UsdMayaJobImportArgsTokens->useAsAnimationCache] = _boolean;
d[UsdMayaJobImportArgsTokens->preserveTimeline] = _boolean;
Expand Down Expand Up @@ -1649,6 +1652,7 @@ std::ostream& operator<<(std::ostream& out, const UsdMayaJobImportArgs& importAr
<< "importRelativeTextures: " << TfStringify(importArgs.importRelativeTextures) << std::endl
<< "axisAndUnitMethod: " << TfStringify(importArgs.axisAndUnitMethod) << std::endl
<< "upAxis: " << TfStringify(importArgs.upAxis) << std::endl
<< "unit: " << TfStringify(importArgs.unit) << std::endl
<< "pullImportStage: " << TfStringify(importArgs.pullImportStage) << std::endl
<< std::endl
<< "timeInterval: " << importArgs.timeInterval << std::endl
Expand Down
2 changes: 2 additions & 0 deletions lib/mayaUsd/fileio/jobs/jobArgs.h
Original file line number Diff line number Diff line change
Expand Up @@ -197,6 +197,7 @@ TF_DECLARE_PUBLIC_TOKENS(
(preserveTimeline) \
(remapUVSetsTo) \
(upAxis) \
(unit) \
(axisAndUnitMethod) \
/* values for axis and unit method */ \
(rotateScale) \
Expand Down Expand Up @@ -436,6 +437,7 @@ struct UsdMayaJobImportArgs
const std::string importRelativeTextures;
const std::string axisAndUnitMethod;
const bool upAxis;
const bool unit;
const bool importInstances;
const bool useAsAnimationCache;
const bool importWithProxyShapes;
Expand Down
178 changes: 112 additions & 66 deletions lib/mayaUsd/fileio/jobs/readJob.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -199,7 +199,7 @@ bool UsdMaya_ReadJob::Read(std::vector<MDagPath>* addedDagPaths)
// When we are called from PrimUpdaterManager we should already have
// a computation scope. If we are called from elsewhere don't show any
// progress bar here.
MayaUsd::ProgressBarScope progressBar(17);
MayaUsd::ProgressBarScope progressBar(16);

// Do not use the global undo info recording system.
// The read job Undo() / redo() functions will handle all operations.
Expand Down Expand Up @@ -259,22 +259,6 @@ bool UsdMaya_ReadJob::Read(std::vector<MDagPath>* addedDagPaths)
_setTimeSampleMultiplierFrom(stage->GetTimeCodesPerSecond());
progressBar.advance();

// XXX Currently all distance values are set directly from USD and will be
// interpreted as centimeters (Maya's internal distance unit). Future work
// could include converting distance values based on the specified meters-
// per-unit in the USD stage metadata. For now, simply warn.
if (UsdGeomStageHasAuthoredMetersPerUnit(stage)) {
MDistance::Unit mdistanceUnit = UsdMayaUtil::ConvertUsdGeomLinearUnitToMDistanceUnit(
UsdGeomGetStageMetersPerUnit(stage));

if (mdistanceUnit != MDistance::internalUnit()) {
TF_WARN("Distance unit conversion is not yet supported. "
"All distance values will be imported in Maya's internal "
"distance unit.");
}
}
progressBar.advance();

// If the import time interval isn't empty, we expand the Min/Max time
// sliders to include the stage's range if necessary.
AutoTimelineRestore timelineRestore(mArgs.preserveTimeline);
Expand Down Expand Up @@ -483,7 +467,7 @@ bool UsdMaya_ReadJob::Read(std::vector<MDagPath>* addedDagPaths)
}
progressBar.advance();

_ConvertUpAxis(stage);
_ConvertUpAxisAndUnits(stage);
progressBar.advance();

UsdMayaReadUtil::mapFileHashes.clear();
Expand All @@ -498,37 +482,39 @@ static bool getUSDUpAxisZ(const UsdStageRefPtr& stage)
return UsdGeomGetStageUpAxis(stage) == UsdGeomTokens->z;
}

void UsdMaya_ReadJob::_ConvertUpAxis(const UsdStageRefPtr& stage)
void UsdMaya_ReadJob::_ConvertUpAxisAndUnits(const UsdStageRefPtr& stage)
{
// If up-axis fixing is turned off, do nothing.
if (!mArgs.upAxis)
return;

// If up axis are the same in Maya and USD, do nothing.
const bool isMayaUpAxisZ = getMayaUpAxisZ();
const bool isUSDUpAxisUZ = getUSDUpAxisZ(stage);

if (isMayaUpAxisZ == isUSDUpAxisUZ)
ConversionInfo conversion;

// Convert up-axis based if required and different between Maya and USD.
const bool convertUpAxis = mArgs.upAxis;
conversion.isMayaUpAxisZ = getMayaUpAxisZ();
conversion.isUSDUpAxisUZ = getUSDUpAxisZ(stage);
conversion.needUpAxisConversion
= (convertUpAxis && (conversion.isMayaUpAxisZ != conversion.isUSDUpAxisUZ));

// Convert units if required and different between Maya and USD.
const bool convertUnits = mArgs.unit;
conversion.mayaMetersPerUnit
= UsdMayaUtil::ConvertMDistanceUnitToUsdGeomLinearUnit(MDistance::internalUnit());
conversion.usdMetersPerUnit = UsdGeomGetStageMetersPerUnit(stage);
conversion.needUnitsConversion
= (convertUnits && (conversion.mayaMetersPerUnit != conversion.usdMetersPerUnit));

// If neither up-axis nor units need to change, do nothing.
if (!conversion.needUpAxisConversion && !conversion.needUnitsConversion)
return;

// Convert axis based on desired method.
const bool convertYtoZ = isMayaUpAxisZ;

bool success = false;

if (mArgs.axisAndUnitMethod == UsdMayaJobImportArgsTokens->rotateScale)
success = _ConvertUpAxisWithRotation(stage, convertYtoZ, false);
_ConvertUpAxisAndUnitsByModifyingData(stage, conversion, false);
else if (mArgs.axisAndUnitMethod == UsdMayaJobImportArgsTokens->addTransform)
success = _ConvertUpAxisWithRotation(stage, convertYtoZ, true);
_ConvertUpAxisAndUnitsByModifyingData(stage, conversion, true);
else if (mArgs.axisAndUnitMethod == UsdMayaJobImportArgsTokens->overwritePrefs)
success = _ConvertUpAxisByChangingMayPrefs(convertYtoZ);
_ConvertUpAxisAndUnitsByChangingMayaPrefs(stage, conversion);
else
TF_WARN(
"Unknown method of converting the USD up axis to Maya: %s",
"Unknown method of converting the USD up axis and units to Maya: %s",
mArgs.axisAndUnitMethod.c_str());

if (success)
MGlobal::displayInfo("Mismatching axis have been converted for accurate orientation.");
}

// Construct list of top level DAG nodes.
Expand Down Expand Up @@ -583,27 +569,37 @@ static std::string _cleanMayaNodeName(const std::string& name)
return cleaned;
}

static void
_addOrignalUpAxisAttribute(const std::vector<MDagPath>& dagNodePaths, bool convertUsdYtoMayaZ)
static void _addOrignalUpAxisAttribute(const std::vector<MDagPath>& dagNodePaths, bool isUSDUpAxisZ)
{
// Note: if we're converting from Y to Z, then the original up axis was Y, otherwise Z.
const MString originalUpAxis = convertUsdYtoMayaZ ? "Y" : "Z";
const MString originalUpAxis = isUSDUpAxisZ ? "Z" : "Y";
const MString attrName = "OriginalUSDUpAxis";
for (const MDagPath& dagPath : dagNodePaths) {
MFnDependencyNode depNode(dagPath.node());
MayaUsd::setDynamicAttribute(depNode, attrName, originalUpAxis);
}
}

bool UsdMaya_ReadJob::_ConvertUpAxisWithRotation(
static void
_addOrignalUnitsAttribute(const std::vector<MDagPath>& dagNodePaths, double usdMetersPerUnit)
{
MString originalUnits;
originalUnits.set(usdMetersPerUnit);
const MString attrName = "OriginalUSDMetersPerUnit";
for (const MDagPath& dagPath : dagNodePaths) {
MFnDependencyNode depNode(dagPath.node());
MayaUsd::setDynamicAttribute(depNode, attrName, originalUnits);
}
}

void UsdMaya_ReadJob::_ConvertUpAxisAndUnitsByModifyingData(
const UsdStageRefPtr& stage,
bool convertUsdYtoMayaZ,
const ConversionInfo& conversion,
bool keepParentGroup)
{
std::vector<MDagPath> dagNodePaths
= _findAllRootDagNodePaths(mNewNodeRegistry, mMayaRootDagPath);
if (dagNodePaths.size() <= 0)
return true;
return;

std::vector<std::string> dagNodeNames = _convertDagPathToNames(dagNodePaths);

Expand Down Expand Up @@ -648,7 +644,7 @@ bool UsdMaya_ReadJob::_ConvertUpAxisWithRotation(

if (mMayaRootDagPath.node() != MObject::kNullObj) {
static const char groupUnderParentCmdFormat[]
= "string $groupName = `group -name \"%s\" -absolute -parent \"%s\" \"%s\"`;";
= "string $groupName = `group -name \"%s\" -absolute -parent \"%s\" \"%s\"`;\n";
std::string rootName = mMayaRootDagPath.fullPathName().asChar();
groupCmd = TfStringPrintf(
groupUnderParentCmdFormat,
Expand All @@ -657,7 +653,7 @@ bool UsdMaya_ReadJob::_ConvertUpAxisWithRotation(
rootName.c_str());
} else {
static const char groupUnderWorldCmdFormat[]
= "string $groupName = `group -name \"%s\" -absolute -world \"%s\"`;";
= "string $groupName = `group -name \"%s\" -absolute -world \"%s\"`;\n";
groupCmd = TfStringPrintf(
groupUnderWorldCmdFormat, groupName.c_str(), joinedDAGNodeNames.c_str());
}
Expand All @@ -673,26 +669,35 @@ bool UsdMaya_ReadJob::_ConvertUpAxisWithRotation(
// - Use -pivot to make sure we are rotating relative to the origin
// (The group is positioned at the center of all sub-object, so we need to specify the
// pivot)
{
if (conversion.needUpAxisConversion) {
static const char rotationCmdFormat[]
= "rotate -relative -euler -pivot 0 0 0 -forceOrderXYZ %d 0 0 $groupName;";
= "rotate -relative -euler -pivot 0 0 0 -forceOrderXYZ %d 0 0 $groupName;\n";
const int angleYtoZ = 90;
const int angleZtoY = -90;
std::string rotationCmd
= TfStringPrintf(rotationCmdFormat, convertUsdYtoMayaZ ? angleYtoZ : angleZtoY);
= TfStringPrintf(rotationCmdFormat, conversion.isMayaUpAxisZ ? angleYtoZ : angleZtoY);
fullScript += rotationCmd;
}

if (conversion.needUnitsConversion) {
static const char scalingCmdFormat[]
= "scale -relative -pivot 0 0 0 -scaleXYZ %f %f %f $groupName;\n";
const double usdToMayaScaling = conversion.usdMetersPerUnit / conversion.mayaMetersPerUnit;
std::string scalingCmd = TfStringPrintf(
scalingCmdFormat, usdToMayaScaling, usdToMayaScaling, usdToMayaScaling);
fullScript += scalingCmd;
}

if (!keepParentGroup) {
static const char ungroupCmdFormat[] = "ungroup -absolute \"%s\";";
static const char ungroupCmdFormat[] = "ungroup -absolute \"%s\";\n";
std::string ungroupCmd = TfStringPrintf(ungroupCmdFormat, groupName.c_str());
fullScript += ungroupCmd;
}

if (!MGlobal::executeCommand(fullScript.c_str())) {
MGlobal::displayWarning("Failed to add a transform to convert the up-axis to align "
"the USD data with Maya up-axis.");
return false;
return;
}

if (keepParentGroup) {
Expand All @@ -702,26 +707,67 @@ bool UsdMaya_ReadJob::_ConvertUpAxisWithRotation(
sel.add(groupName.c_str());
sel.getDagPath(0, groupDagPath);
}
_addOrignalUpAxisAttribute({ groupDagPath }, convertUsdYtoMayaZ);
if (conversion.needUpAxisConversion)
_addOrignalUpAxisAttribute({ groupDagPath }, conversion.isUSDUpAxisUZ);
if (conversion.needUnitsConversion)
_addOrignalUnitsAttribute({ groupDagPath }, conversion.usdMetersPerUnit);
} else {
_addOrignalUpAxisAttribute(dagNodePaths, convertUsdYtoMayaZ);
if (conversion.needUpAxisConversion)
_addOrignalUpAxisAttribute(dagNodePaths, conversion.isUSDUpAxisUZ);
if (conversion.needUnitsConversion)
_addOrignalUnitsAttribute(dagNodePaths, conversion.usdMetersPerUnit);
}

return true;
MGlobal::displayInfo(
"Mismatching axis and units have been converted for accurate orientation and scale.");
}

bool UsdMaya_ReadJob::_ConvertUpAxisByChangingMayPrefs(const bool convertUsdYtoMayaZ)
void UsdMaya_ReadJob::_ConvertUpAxisAndUnitsByChangingMayaPrefs(
const UsdStageRefPtr& stage,
const ConversionInfo& conversion)
{
const bool rotateView = true;
const MStatus status
= convertUsdYtoMayaZ ? MGlobal::setYAxisUp(rotateView) : MGlobal::setZAxisUp(rotateView);
if (!status) {
MGlobal::displayWarning(
"Failed to change the Maya up-axis preferences to match USD data up-axis.");
return false;
bool success = true;

// Set up-axis preferences if needed.
if (conversion.needUpAxisConversion) {
const bool rotateView = true;
const MStatus status = conversion.isUSDUpAxisUZ ? MGlobal::setZAxisUp(rotateView)
: MGlobal::setYAxisUp(rotateView);
if (!status) {
MGlobal::displayWarning(
"Failed to change the Maya up-axis preferences to match USD data up-axis.");
success = false;
}
}

return true;
// Set units preferences if needed.
if (conversion.needUnitsConversion) {
const MDistance::Unit mayaUnit
= UsdMayaUtil::ConvertUsdGeomLinearUnitToMDistanceUnit(conversion.usdMetersPerUnit);
if (mayaUnit == MDistance::kInvalid) {
MGlobal::displayWarning(
"Unable to convert <unit> to a Maya unit. Supported units include millimeters, "
"centimeters, meters, kilometers, inches, feet, yards and miles.");
success = false;
} else {
const MString mayaUnitText = UsdMayaUtil::ConvertMDistanceUnitToText(mayaUnit);
MString changeUnitsCmd;
changeUnitsCmd.format("currentUnit -linear ^1s;", mayaUnitText);

// Note: we *must* execute the units change on-idle because the import process
// saves and restores all units! If we change it now, the change would be lost.
if (!MGlobal::executeCommandOnIdle(changeUnitsCmd)) {
MGlobal::displayWarning(
"Failed to change the Maya units preferences to match USD data "
"because the units are not supported by Maya.");
success = false;
}
}
}

if (success)
MGlobal::displayInfo(
"Changed Maya preferences to match up-axis and units from the imported USD scene.");
}

bool UsdMaya_ReadJob::DoImport(UsdPrimRange& rootRange, const UsdPrim& usdRootPrim)
Expand Down
21 changes: 17 additions & 4 deletions lib/mayaUsd/fileio/jobs/readJob.h
Original file line number Diff line number Diff line change
Expand Up @@ -141,12 +141,25 @@ class UsdMaya_ReadJob

double _setTimeSampleMultiplierFrom(const double layerFPS);

void _ConvertUpAxis(const UsdStageRefPtr& stage);
bool _ConvertUpAxisWithRotation(
struct ConversionInfo
{
bool isMayaUpAxisZ { false };
bool isUSDUpAxisUZ { false };
bool needUpAxisConversion { false };

double mayaMetersPerUnit = { 0.01 };
double usdMetersPerUnit = { 0.01 };
bool needUnitsConversion = { false };
};

void _ConvertUpAxisAndUnits(const UsdStageRefPtr& stage);
void _ConvertUpAxisAndUnitsByModifyingData(
const UsdStageRefPtr& stage,
bool convertUsdYtoMayaZ,
const ConversionInfo& conversion,
bool keepParentGroup);
bool _ConvertUpAxisByChangingMayPrefs(const bool convertUsdYtoMayaZ);
void _ConvertUpAxisAndUnitsByChangingMayaPrefs(
const UsdStageRefPtr& stage,
const ConversionInfo& conversion);

// Data
MDagModifier mDagModifierUndo;
Expand Down
2 changes: 0 additions & 2 deletions lib/mayaUsd/fileio/jobs/writeJob.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -493,8 +493,6 @@ bool UsdMaya_WriteJob::_BeginWriting(const std::string& fileName, bool append)
_autoAxisAndUnitsChanger = std::make_unique<AutoUpAxisAndUnitsChanger>(
mJobCtx.mStage, mJobCtx.mArgs.upAxis, mJobCtx.mArgs.unit);

// TODO: handle mJobCtx.mArgs.unit

// Set the customLayerData on the layer
if (!mJobCtx.mArgs.customLayerData.empty()) {
mJobCtx.mStage->GetRootLayer()->SetCustomLayerData(mJobCtx.mArgs.customLayerData);
Expand Down
Loading

0 comments on commit 4174814

Please sign in to comment.