From 2dfd75791e87fb2548417b060524c99608f23145 Mon Sep 17 00:00:00 2001
From: Ashwin Bhat <1727158+ashwinbhat@users.noreply.github.com>
Date: Mon, 18 Apr 2022 19:15:28 -0700
Subject: [PATCH] Improvements to support locale and units (#892)
- Add utilities UnitConverterRegistry::convertToUnit and MaterialX::joinStrings.
- Use C Locale for string stream operations (https://github.com/autodesk-forks/MaterialX/issues/1183).
- Add tests for unit and locale content.
---
.../TestSuite/locale/numericformat.mtlx | 13 ++++
.../Materials/TestSuite/locale/utf8.mtlx | 16 +++++
source/MaterialXCore/Unit.cpp | 66 +++++++++++++++++
source/MaterialXCore/Unit.h | 4 ++
source/MaterialXCore/Util.cpp | 16 +++--
source/MaterialXCore/Util.h | 4 ++
source/MaterialXCore/Value.cpp | 2 +
.../MaterialXTest/MaterialXFormat/XmlIo.cpp | 72 +++++++++++++++++++
source/PyMaterialX/PyMaterialXCore/PyUtil.cpp | 1 +
.../PyMaterialX/PyMaterialXFormat/PyXmlIo.cpp | 6 ++
10 files changed, 195 insertions(+), 5 deletions(-)
create mode 100644 resources/Materials/TestSuite/locale/numericformat.mtlx
create mode 100644 resources/Materials/TestSuite/locale/utf8.mtlx
diff --git a/resources/Materials/TestSuite/locale/numericformat.mtlx b/resources/Materials/TestSuite/locale/numericformat.mtlx
new file mode 100644
index 0000000000..b9400e270c
--- /dev/null
+++ b/resources/Materials/TestSuite/locale/numericformat.mtlx
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/resources/Materials/TestSuite/locale/utf8.mtlx b/resources/Materials/TestSuite/locale/utf8.mtlx
new file mode 100644
index 0000000000..a6a30f02d2
--- /dev/null
+++ b/resources/Materials/TestSuite/locale/utf8.mtlx
@@ -0,0 +1,16 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/source/MaterialXCore/Unit.cpp b/source/MaterialXCore/Unit.cpp
index f55cdc108a..959f697b24 100644
--- a/source/MaterialXCore/Unit.cpp
+++ b/source/MaterialXCore/Unit.cpp
@@ -241,4 +241,70 @@ void UnitConverterRegistry::write(DocumentPtr doc) const
}
}
+bool UnitConverterRegistry::convertToUnit(DocumentPtr doc, const string& unitType, const string& targetUnit)
+{
+ UnitTypeDefPtr unitTypeDef = doc->getUnitTypeDef(unitType);
+ UnitConverterPtr unitConverter = getUnitConverter(unitTypeDef);
+
+ if (!unitTypeDef || !unitConverter)
+ {
+ return false;
+ }
+
+ bool convertedUnits = false;
+ for (ElementPtr elem : doc->traverseTree())
+ {
+ NodePtr pNode = elem->asA();
+ if (!pNode || !pNode->getInputCount())
+ {
+ continue;
+ }
+ for (InputPtr input : pNode->getInputs())
+ {
+ const std::string type = input->getType();
+ const ValuePtr value = input->getValue();
+ if (value && input->hasUnit() && (input->getUnitType() == unitType) && value)
+ {
+ if (type == getTypeString())
+ {
+ float originalval = value->asA();
+ float convertedValue = unitConverter->convert(originalval, input->getUnit(), targetUnit);
+ input->setValue(convertedValue);
+ input->removeAttribute(ValueElement::UNIT_ATTRIBUTE);
+ input->removeAttribute(ValueElement::UNITTYPE_ATTRIBUTE);
+ convertedUnits = true;
+ }
+ else if (type == getTypeString())
+ {
+ Vector2 originalval = value->asA();
+ Vector2 convertedValue = unitConverter->convert(originalval, input->getUnit(), targetUnit);
+ input->setValue(convertedValue);
+ input->removeAttribute(ValueElement::UNIT_ATTRIBUTE);
+ input->removeAttribute(ValueElement::UNITTYPE_ATTRIBUTE);
+ convertedUnits = true;
+ }
+ else if (type == getTypeString())
+ {
+ Vector3 originalval = value->asA();
+ Vector3 convertedValue = unitConverter->convert(originalval, input->getUnit(), targetUnit);
+ input->setValue(convertedValue);
+ input->removeAttribute(ValueElement::UNIT_ATTRIBUTE);
+ input->removeAttribute(ValueElement::UNITTYPE_ATTRIBUTE);
+ convertedUnits = true;
+ }
+ else if (type == getTypeString())
+ {
+ Vector4 originalval = value->asA();
+ Vector4 convertedValue = unitConverter->convert(originalval, input->getUnit(), targetUnit);
+ input->setValue(convertedValue);
+ input->removeAttribute(ValueElement::UNIT_ATTRIBUTE);
+ input->removeAttribute(ValueElement::UNITTYPE_ATTRIBUTE);
+ convertedUnits = true;
+ }
+ }
+ }
+ }
+ return convertedUnits;
+}
+
MATERIALX_NAMESPACE_END
diff --git a/source/MaterialXCore/Unit.h b/source/MaterialXCore/Unit.h
index f50101a1bf..26b2b0df58 100644
--- a/source/MaterialXCore/Unit.h
+++ b/source/MaterialXCore/Unit.h
@@ -195,6 +195,10 @@ class MX_CORE_API UnitConverterRegistry
/// Create unit definitions in a document based on registered converters
void write(DocumentPtr doc) const;
+ /// Convert input values which have a source unit to a given target unit.
+ /// Returns if any unit conversion occured.
+ bool convertToUnit(DocumentPtr doc, const string& unitType, const string& targetUnit);
+
private:
UnitConverterRegistry(const UnitConverterRegistry&) = delete;
UnitConverterRegistry() { }
diff --git a/source/MaterialXCore/Util.cpp b/source/MaterialXCore/Util.cpp
index a0fb1ff1f2..9805f4a7c9 100644
--- a/source/MaterialXCore/Util.cpp
+++ b/source/MaterialXCore/Util.cpp
@@ -91,6 +91,16 @@ StringVec splitString(const string& str, const string& sep)
return split;
}
+string joinStrings(const StringVec& stringVec, const string& sep)
+{
+ string res;
+ for (const string& name : stringVec)
+ {
+ res = res.empty() ? name : res + sep + name;
+ }
+ return res;
+}
+
string replaceSubstrings(string str, const StringMap& stringMap)
{
for (const auto& pair : stringMap)
@@ -145,11 +155,7 @@ StringVec splitNamePath(const string& namePath)
string createNamePath(const StringVec& nameVec)
{
- string res;
- for (const string& name : nameVec)
- {
- res = res.empty() ? name : res + NAME_PATH_SEPARATOR + name;
- }
+ string res = joinStrings(nameVec, NAME_PATH_SEPARATOR);
return res;
}
diff --git a/source/MaterialXCore/Util.h b/source/MaterialXCore/Util.h
index 8e201b37a3..a3f2e8a64c 100644
--- a/source/MaterialXCore/Util.h
+++ b/source/MaterialXCore/Util.h
@@ -35,6 +35,10 @@ MX_CORE_API string incrementName(const string& name);
/// separator characters.
MX_CORE_API StringVec splitString(const string& str, const string& sep);
+/// Join a vector of substrings into a single string, placing the given
+/// separator between each substring.
+MX_CORE_API string joinStrings(const StringVec& strVec, const string& sep);
+
/// Apply the given substring substitutions to the input string.
MX_CORE_API string replaceSubstrings(string str, const StringMap& stringMap);
diff --git a/source/MaterialXCore/Value.cpp b/source/MaterialXCore/Value.cpp
index f83fb81e63..ebf1346f8d 100644
--- a/source/MaterialXCore/Value.cpp
+++ b/source/MaterialXCore/Value.cpp
@@ -31,6 +31,7 @@ template using enable_if_std_vector_t =
template void stringToData(const string& str, T& data)
{
std::stringstream ss(str);
+ ss.imbue(std::locale::classic());
if (!(ss >> data))
{
throw ExceptionTypeError("Type mismatch in generic stringToData: " + str);
@@ -94,6 +95,7 @@ template void stringToData(const string& str, enable_if_std_vector_t void dataToString(const T& data, string& str)
{
std::stringstream ss;
+ ss.imbue(std::locale::classic());
// Set float format and precision for the stream
const Value::FloatFormat fmt = Value::getFloatFormat();
diff --git a/source/MaterialXTest/MaterialXFormat/XmlIo.cpp b/source/MaterialXTest/MaterialXFormat/XmlIo.cpp
index d59e8d930d..fb2df02acf 100644
--- a/source/MaterialXTest/MaterialXFormat/XmlIo.cpp
+++ b/source/MaterialXTest/MaterialXFormat/XmlIo.cpp
@@ -227,3 +227,75 @@ TEST_CASE("Load content", "[xmlio]")
mx::DocumentPtr nonExistentDoc = mx::createDocument();
REQUIRE_THROWS_AS(mx::readFromXmlFile(nonExistentDoc, "NonExistent.mtlx", mx::FileSearchPath(), &readOptions), mx::ExceptionFileMissing&);
}
+
+TEST_CASE("Locale region testing", "[xmlio]")
+{
+ /// In the United States, the thousands separator is a comma, while in Germany it is a period.
+ /// Thus, one thousand twenty five is displayed as 1,025 in the United States and 1.025 in Germany.
+ ///
+ /// In a MaterialX document, a vector3 value of "1,1.5,2.0" should be interpreted as (1.0f, 1.5f, 2.0f).
+
+ try
+ {
+ // Set locale to de
+ std::locale deLocale("de_DE");
+ std::locale::global(deLocale);
+ }
+ catch (const std::runtime_error& e)
+ {
+ WARN("Unable to change locale " << e.what());
+ return;
+ }
+
+ mx::FilePath libraryPath("libraries/stdlib");
+ mx::FilePath testPath("resources/Materials/TestSuite/locale");
+ mx::FileSearchPath searchPath = libraryPath.asString() +
+ mx::PATH_LIST_SEPARATOR +
+ testPath.asString();
+
+ // Read the standard library.
+ std::vector libs;
+ for (const mx::FilePath& filename : libraryPath.getFilesInDirectory(mx::MTLX_EXTENSION))
+ {
+ mx::DocumentPtr lib = mx::createDocument();
+ mx::readFromXmlFile(lib, filename, searchPath);
+ libs.push_back(lib);
+ }
+
+ // Read and validate each example document.
+ for (const mx::FilePath& filename : testPath.getFilesInDirectory(mx::MTLX_EXTENSION))
+ {
+ mx::DocumentPtr doc = mx::createDocument();
+ mx::readFromXmlFile(doc, filename, searchPath);
+ for (mx::DocumentPtr lib : libs)
+ {
+ doc->importLibrary(lib);
+ }
+ std::string message;
+
+ bool docValid = doc->validate(&message);
+ if (!docValid)
+ {
+ WARN("[" + filename.asString() + "] " + message);
+ }
+ REQUIRE(docValid);
+
+ // Traverse the document tree
+ int valueElementCount = 0;
+ int uiAttributeCount = 0;
+ for (mx::ElementPtr elem : doc->traverseTree())
+ {
+ if (elem->isA())
+ {
+ valueElementCount++;
+ if (elem->hasAttribute("uiname"))
+ {
+ REQUIRE(!elem->getAttribute("uiname").empty());
+ uiAttributeCount++;
+ }
+ }
+ }
+ REQUIRE(valueElementCount > 0);
+ REQUIRE(uiAttributeCount > 0);
+ }
+}
diff --git a/source/PyMaterialX/PyMaterialXCore/PyUtil.cpp b/source/PyMaterialX/PyMaterialXCore/PyUtil.cpp
index a646a5ecf1..190668a608 100644
--- a/source/PyMaterialX/PyMaterialXCore/PyUtil.cpp
+++ b/source/PyMaterialX/PyMaterialXCore/PyUtil.cpp
@@ -21,6 +21,7 @@ void bindPyUtil(py::module& mod)
mod.def("isValidName", &mx::isValidName);
mod.def("incrementName", &mx::incrementName);
mod.def("splitString", &mx::splitString);
+ mod.def("joinStrings", &mx::joinStrings);
mod.def("replaceSubstrings", &mx::replaceSubstrings);
mod.def("stringEndsWith", &mx::stringEndsWith);
mod.def("splitNamePath", &mx::splitNamePath);
diff --git a/source/PyMaterialX/PyMaterialXFormat/PyXmlIo.cpp b/source/PyMaterialX/PyMaterialXFormat/PyXmlIo.cpp
index 683a487931..1ac0c5ca7c 100644
--- a/source/PyMaterialX/PyMaterialXFormat/PyXmlIo.cpp
+++ b/source/PyMaterialX/PyMaterialXFormat/PyXmlIo.cpp
@@ -35,6 +35,12 @@ void bindPyXmlIo(py::module& mod)
py::arg("doc"), py::arg("writeOptions") = nullptr);
mod.def("prependXInclude", mx::prependXInclude);
+ mod.def("getEnvironmentPath", &mx::getEnvironmentPath,
+ py::arg("sep") = mx::PATH_LIST_SEPARATOR);
+
+ mod.attr("PATH_LIST_SEPARATOR") = mx::PATH_LIST_SEPARATOR;
+ mod.attr("MATERIALX_SEARCH_PATH_ENV_VAR") = mx::MATERIALX_SEARCH_PATH_ENV_VAR;
+
py::register_exception(mod, "ExceptionParseError");
py::register_exception(mod, "ExceptionFileMissing");
}