Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improvements to publishing API #1403

Merged
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
69 commits
Select commit Hold shift + click to select a range
660f779
API logic updates and updated testing.
kwokcb Jul 11, 2023
92ac629
Add createdefinition Python utility.
kwokcb Jul 11, 2023
595e8f0
Revert whitespace edit
jstone-lucasfilm Jul 13, 2023
f8d7ad6
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Jul 24, 2023
16f0b53
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Jul 30, 2023
59e8e89
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Aug 16, 2023
f06edc3
Remove sample script.
kwokcb Aug 20, 2023
e0f9d5e
Proposed simplified const interface.
kwokcb Aug 20, 2023
281fbe3
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Sep 1, 2023
18ad877
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Sep 1, 2023
ced6cb7
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Sep 4, 2023
1fa27c1
Update Javascript wrapper.
kwokcb Sep 4, 2023
198fabc
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Sep 4, 2023
025a37a
Minor format edit
jstone-lucasfilm Sep 5, 2023
2b0da25
Review updates.
kwokcb Sep 6, 2023
80354bd
Minor spelling fix
jstone-lucasfilm Sep 7, 2023
8b75c28
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Oct 9, 2023
cc7ad49
Omit file writes in unit tests
jstone-lucasfilm Oct 10, 2023
358d359
Minor whitespace fixes
jstone-lucasfilm Oct 10, 2023
e0917f4
Additional whitespace fixes
jstone-lucasfilm Oct 10, 2023
75a413a
Standardize formatting
jstone-lucasfilm Oct 10, 2023
090d435
Align formatting
jstone-lucasfilm Oct 10, 2023
ae75615
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Oct 19, 2023
1feba6e
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Dec 5, 2023
929614c
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Jan 24, 2024
ca4cbae
Clarify comment
jstone-lucasfilm Jan 24, 2024
65818e0
Clarify comments
jstone-lucasfilm Jan 24, 2024
52eab15
Clarify comment
jstone-lucasfilm Jan 24, 2024
f9516b1
Remove second unit test
jstone-lucasfilm Jan 24, 2024
188541c
Maintain original validation step
jstone-lucasfilm Jan 24, 2024
bfa9270
Minor grammar fix
jstone-lucasfilm Jan 24, 2024
b1e4b7b
Remove extra parentheses
jstone-lucasfilm Jan 24, 2024
9d1733e
Remove assignment that duplicates constructor
jstone-lucasfilm Jan 24, 2024
d1c95a2
Remove extra parentheses
jstone-lucasfilm Jan 25, 2024
c04789b
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Jan 26, 2024
8943ba3
Update test to create node instances from definitions.
kwokcb Jan 30, 2024
c32136d
Remove hard-coded strings.
kwokcb Jan 30, 2024
abe7265
Merge branch 'AcademySoftwareFoundation:main' into nodedef_creation_api
kwokcb Jan 30, 2024
c034d51
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Jan 31, 2024
475a981
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Feb 1, 2024
dcc97f8
Merge branch 'main' into nodedef_creation_api
jstone-lucasfilm Mar 6, 2024
a9cf526
Merge branch 'dev_1.39' into nodedef_creation_api
kwokcb Mar 16, 2024
39ac8a9
Add in method deprecation.
kwokcb Mar 16, 2024
050961b
Add C++ fallback, and proper pybind11 and emscripten deprecation inte…
kwokcb Mar 18, 2024
e7ebe25
Fix string for JS message.
kwokcb Mar 18, 2024
083f2c5
Merge branch 'dev_1.39' of https://github.com/AcademySoftwareFoundati…
kwokcb Apr 9, 2024
d315c05
Remove JS deprecation method.
kwokcb Apr 9, 2024
76402bd
Minor whitespace fixes
jstone-lucasfilm Apr 11, 2024
2855b3c
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm Apr 15, 2024
0f36e4b
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm May 11, 2024
e48f4f9
Remove references to channels in 1.39
jstone-lucasfilm May 11, 2024
56cb334
Remove references to channels in 1.39
jstone-lucasfilm May 11, 2024
bf85dfe
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm May 21, 2024
41099a1
Minor fix to JavaScript binding
jstone-lucasfilm May 24, 2024
616f00e
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm May 24, 2024
a2ccbfd
Simplify validation logic
jstone-lucasfilm May 24, 2024
5aaf451
Minor consistency improvements
jstone-lucasfilm May 24, 2024
779af87
Minor formatting fixes
jstone-lucasfilm May 26, 2024
39f25db
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm May 27, 2024
cdff9e1
Simplify unit tests
jstone-lucasfilm May 27, 2024
3f184df
Fix typo
jstone-lucasfilm May 27, 2024
266d41e
Fix typo
jstone-lucasfilm May 27, 2024
1d8a34a
Minor clarity improvements
jstone-lucasfilm May 28, 2024
d8010a4
Remove unused qualifier
jstone-lucasfilm May 28, 2024
987c32f
Remove unused qualifiers
jstone-lucasfilm May 28, 2024
c342133
Remove unused qualifier
jstone-lucasfilm May 28, 2024
735e13e
Remove unused qualifiers
jstone-lucasfilm May 28, 2024
47f04f4
Clarify variable name
jstone-lucasfilm May 28, 2024
2198f7b
Merge branch 'dev_1.39' into nodedef_creation_api
jstone-lucasfilm May 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
205 changes: 205 additions & 0 deletions python/Scripts/createdefinition.py
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,205 @@
#!/usr/bin/env python
'''
Create definitions based on the nodegraphs in the specified document. Definitions are
saved to new documents with either the definition and functional graph in a single file or
in separated into a file for definitions and a file for functional graphs.

It is assumed that all nodegraphs have the same category, nodegroup, version and namespace
as only one set of these input options can be provided. It is also assumed that each nodegraph
has a different output signature. If the same signature if out then no definition will be created
for that nodegraph.
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved

e.g. python createdefinition.py
resources/Materials/Examples/StandardSurface/standard_surface_marble_solid.mtlx
--category mymarble
--version "2.0.9"
--nodegroup 'texture2d'
--separateDocuments 1
--comment "This is a new custom marble node."
'''

import sys, argparse, os
import MaterialX as mx

def main():
parser = argparse.ArgumentParser(description="Create definition and functional graph from compound graphs.")
parser.add_argument("--category", dest="category", default='', help="Category of the definition. If not specified the nodegraph name will be used")
parser.add_argument("--nodegroup", dest="nodegroup", default='', help="Node group of the definition. Defaults to an empty string")
parser.add_argument("--version", dest="version", default='', help="Version string of the definition. Defaults to empty string")
parser.add_argument("--defaultversion", dest="defaultversion", type=mx.stringToBoolean, default=False, help="Flag to indicate this is the default version. Defaults to false.")
parser.add_argument("--namespace", dest="namespace", default='', help="Namespace of the definition. Defaults to an empty string.")
parser.add_argument("--comment", dest="comment", default='', help="XML comment to embed before the definition or nodegraph")
parser.add_argument("--documentation", dest="documentation", default='', help="Definition documentation")
parser.add_argument('--separateDocuments', dest='separateDocuments', type=mx.stringToBoolean, default=False, help="Writ the definition and functional graph to separate files.")
parser.add_argument(dest="inputFilename", help="Filename of the input document.")
parser.add_argument('--outputPath', dest='outputPath', default="./", help='File path to output documents to. If not specified the files are output to the current directory.')

opts = parser.parse_args()

version_major, version_minor, version_patch = mx.getVersionIntegers()
use_1_38_8 = False
if version_major >=1 and version_minor >= 38 and version_patch >= 8:
use_1_38_8 = True

doc = mx.createDocument()
try:
mx.readFromXmlFile(doc, opts.inputFilename)
except mx.ExceptionFileMissing as err:
print(err)
sys.exit(-1)

valid, msg = doc.validate()
if not valid:
print("Validation warnings for input document:")
print(msg)
sys.exit(-1)

nodeGraphs = doc.getNodeGraphs()
parameter_signatures = set()
for nodeGraph in nodeGraphs:
# Skip nodegraphs which are already functional graphs
if nodeGraph.getNodeDef():
print('Skip functional nodegraph %s' % nodeGraph.getNamePath())
continue

version = mx.createValidName(opts.version)
category = mx.createValidName(opts.category)
nodegroup = mx.createValidName(opts.nodegroup)
namespace = mx.createValidName(opts.namespace)
documentation = opts.documentation
defaultversion = opts.defaultversion

# User nodegraph name if no category provided as a a category name
# must be provided
if not category:
category = nodeGraph.getName()

# Build identifier for new nodedef and functional graph. Includes:
# - version
# - output types
# - namespace
#
identifier = category
if version:
identifier = identifier + '_' + version

outputs = nodeGraph.getOutputs()
parameter_signature = ''
for output in outputs:
outputType = output.getType()
parameter_signature = parameter_signature + '_' + outputType
if parameter_signature in parameter_signatures:
print('Duplicate parameter signature found on on nodegraph %s. Skipping definition createion.' % nodeGraph.getNamePath())
continue
else:
parameter_signatures.add(parameter_signature)
identifier = identifier + parameter_signature

# Prefix with "ND_" or "NG_" for definition and functional graph names
nodedefName = 'ND_' + identifier
nodegraphName = 'NG_' + identifier

# Create the definition
definition = None
definition = doc.addNodeDefFromGraph(nodeGraph, nodedefName,
category, opts.version, defaultversion, nodegroup, nodegraphName)
if definition and documentation:
definition.setDocString(documentation)
if namespace:
definition.setNamespace(namespace)
funcgraph = doc.getNodeGraph(nodegraphName)
if funcgraph and namespace:
funcgraph.setNamespace(namespace)

if not definition or not funcgraph:
print('Failed to create definition for nodegraph %s' % nodeGraph.getNamePath())
continue

if not use_1_38_8:
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved
print('Patch 1.38.8 definition creation...')
definition.setDocString(documentation)
definition.setNamespace(namespace)

funcgraph.setNamespace(namespace)
for graphChild in funcgraph.getChildren():
graphChild.removeAttribute('xpos')
graphChild.removeAttribute('ypos')

filterAttributes = { 'nodegraph', 'nodename', 'channels', 'interfacename', 'xpos', 'ypos' }

# Transfer input interface from the graph to the nodedef
for input in funcgraph.getInputs():
nodeDefInput = definition.addInput(input.getName(), input.getType())
if nodeDefInput:
nodeDefInput.copyContentFrom(input)
for filterAttribute in filterAttributes:
nodeDefInput.removeAttribute(filterAttribute);
nodeDefInput.setSourceUri('')
input.setInterfaceName(nodeDefInput.getName())

# Remove interface from the nodegraph
for input in funcgraph.getInputs():
funcgraph.removeInput(input.getName())

# Copy the output interface from the graph to the nodedef
for output in nodeGraph.getOutputs():
nodeDefOutput = definition.getOutput(output.getName())
if nodeDefOutput:
definition.removeOutput(output.getName())
definition.addOutput(output.getName(), output.getType())
if nodeDefOutput:
nodeDefOutput.copyContentFrom(output)
for filterAttribute in filterAttributes:
nodeDefOutput.removeAttribute(filterAttribute)
nodeDefOutput.setSourceUri('')

# Generate an XML comment string from the given comment.
commentString = ''
if opts.comment:
commentString = '\n\tNode: <' + category + '> '
commentString = commentString + '\n\t' + opts.comment + '\n '

# Separate out the new definition and functional graph and write
# to either 1 or 2 files depending on if the "separateDocuments" flag
# is set.
#
defDoc = mx.createDocument()
if commentString:
comment = defDoc.addChildOfCategory('comment')
comment.setDocString(commentString);
# Note: that addNodeDef() will automatically add an output if a type is specified
# so leave this empty. The outputs will be copied when the content is copied over.
newDef = defDoc.addNodeDef(definition.getName(), '', definition.getCategory())
newDef.copyContentFrom(definition)

graphDoc = defDoc
if opts.separateDocuments:
graphDoc = mx.createDocument()
if commentString:
comment = graphDoc.addChildOfCategory('comment')
comment.setDocString(commentString);
newGraph = graphDoc.addNodeGraph(funcgraph.getName())
newGraph.copyContentFrom(funcgraph);

# Write to file
outputFilePath = mx.FilePath(opts.outputPath)
pathExists = os.path.exists(outputFilePath.asString())
if not pathExists:
print('Created folder: ', outputFilePath.asString())
os.makedirs(outputFilePath.asString())

fileLocation = identifier + '.mtlx'
if opts.separateDocuments:
fileLocation = 'ND_' + fileLocation
fileLocation = outputFilePath / mx.FilePath(fileLocation)
print('Write definition to file:', fileLocation.asString())
mx.writeToXmlFile(defDoc, fileLocation);

if opts.separateDocuments:
fileLocation = 'NG_' + identifier + '.mtlx'
fileLocation = outputFilePath / mx.FilePath(fileLocation)
print('Write functional graph to file:', fileLocation.asString())
mx.writeToXmlFile(graphDoc, fileLocation);

if __name__ == '__main__':
main()
Original file line number Diff line number Diff line change
@@ -1,76 +1,29 @@
<?xml version="1.0"?>
<materialx version="1.38">
<nodegraph name="test_colorcorrect">
<range name="AlphaClampAndGamma" type="float">
<input name="in" type="float" nodename="AlphaOffset" />
<input name="gamma" type="float" value="3" />
<input name="doclamp" type="boolean" value="true" />
</range>
<multiply name="AlphaGain" type="float">
<input name="in1" type="float" nodename="extractAlphaForGain" />
<input name="in1" type="float" nodename="inputAlpha" />
<input name="in2" type="float" value="0.8" />
</multiply>
<add name="AlphaOffset" type="float">
<input name="in1" type="float" nodename="AlphaGain" />
<input name="in2" type="float" value="1" />
</add>
<range name="ColorClampAndGamma" type="color3">
<input name="in" type="color3" nodename="ColorOffset" />
<input name="gamma" type="color3" value="2, 1, 1" />
</range>
<multiply name="ColorGain" type="color3">
<input name="in1" type="color3" nodename="HSV_adjust" />
<input name="in1" type="color3" nodename="inputColor" />
<input name="in2" type="color3" value="0.9, 0.9, 0.9" />
</multiply>
<add name="ColorOffset" type="color3">
<input name="in1" type="color3" nodename="ColorGain" />
<input name="in2" type="color3" value="0.379147, 0.0341412, 0.0341412" />
</add>
<combine2 name="CombineColorAlpha" type="color4">
<input name="in1" type="color3" nodename="ColorClampAndGamma" />
<input name="in2" type="float" nodename="AlphaClampAndGamma" />
</combine2>
<hsvadjust name="HSV_adjust" type="color3">
<input name="in" type="color3" nodename="extractColorForHsv" />
</hsvadjust>
<premult name="premultiplyColor" type="color4">
<input name="in" type="color4" nodename="CombineColorAlpha" />
</premult>
<ifequal name="if_premultiply_condition" type="color4">
<input name="value2" type="boolean" value="true" />
<input name="in1" type="color4" nodename="premultiplyColor" />
<input name="in2" type="color4" nodename="CombineColorAlpha" />
</ifequal>
<unpremult name="unpremultiply" type="color4">
<input name="in" type="color4" nodename="combineInput" />
</unpremult>
<ifequal name="if_unpremultiply_condition" type="color4">
<input name="value2" type="boolean" value="true" />
<input name="in1" type="color4" nodename="unpremultiply" />
<input name="in2" type="color4" nodename="combineInput" />
</ifequal>
<swizzle name="outputColor" type="color3">
<input name="in" type="color4" nodename="if_premultiply_condition" />
<input name="channels" type="string" value="rgb" />
</swizzle>
<swizzle name="outputAlpha" type="float">
<input name="in" type="color4" nodename="if_premultiply_condition" />
<input name="channels" type="string" value="a" />
</swizzle>
<swizzle name="extractColorForHsv" type="color3">
<input name="in" type="color4" nodename="if_unpremultiply_condition" />
<input name="channels" type="string" value="rgb" />
</swizzle>
<swizzle name="extractAlphaForGain" type="float">
<input name="in" type="color4" nodename="if_unpremultiply_condition" />
</swizzle>
<constant name="inputColor" type="color3">
<input name="value" type="color3" value="0.5, 0.5, 0.5" />
</constant>
<constant name="inputAlpha" type="float">
<input name="value" type="float" value="1" />
</constant>
<combine2 name="combineInput" type="color4">
<input name="in1" type="color3" nodename="inputColor" />
<input name="in2" type="float" nodename="inputAlpha" />
</combine2>
<output name="out" type="color3" nodename="outputColor" />
<output name="out1" type="float" nodename="outputAlpha" />
<output name="out" type="color3" nodename="ColorOffset" />
<output name="out1" type="float" nodename="AlphaOffset" />
</nodegraph>
</materialx>
50 changes: 48 additions & 2 deletions source/MaterialXCore/Document.cpp
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -187,6 +187,12 @@ NodeDefPtr Document::addNodeDefFromGraph(const NodeGraphPtr nodeGraph, const str
}
graph = addNodeGraph(newGraphName);
graph->copyContentFrom(nodeGraph);

for (auto graphChild : graph->getChildren())
{
graphChild->removeAttribute("xpos");
graphChild->removeAttribute("ypos");
}
}
graph->setNodeDefString(nodeDefName);

Expand All @@ -208,9 +214,49 @@ NodeDefPtr Document::addNodeDefFromGraph(const NodeGraphPtr nodeGraph, const str
}
}

for (auto output : graph->getOutputs())
// Expose any existing interface.
// Any connection attributes on the existing interface should be removed from the definition.
// as well as any source URI

// Attributes which should not be copied over
StringSet filterAttributes = { PortElement::NODE_GRAPH_ATTRIBUTE, PortElement::NODE_NAME_ATTRIBUTE,
PortElement::CHANNELS_ATTRIBUTE, PortElement::INTERFACE_NAME_ATTRIBUTE,
"xpos", "ypos" };
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved

// Transfer input interface from the graph to the nodedef
for (InputPtr input : graph->getInputs())
{
InputPtr nodeDefInput = nodeDef->addInput(input->getName(), input->getType());
if (nodeDefInput)
{
nodeDefInput->copyContentFrom(input);
for (const string& filterAttribute : filterAttributes )
{
nodeDefInput->removeAttribute(filterAttribute);
}
nodeDefInput->setSourceUri(EMPTY_STRING);
input->setInterfaceName(nodeDefInput->getName());
}
}
// Remove interface from the nodegraph
for (InputPtr input : graph->getInputs())
{
graph->removeInput(input->getName());
}

// Copy the output interface from the graph to the nodedef
for (OutputPtr output : graph->getOutputs())
{
nodeDef->addOutput(output->getName(), output->getType());
OutputPtr nodeDefOutput = nodeDef->addOutput(output->getName(), output->getType());
if (nodeDefOutput)
{
nodeDefOutput->copyContentFrom(output);
for (const string& filterAttribute : filterAttributes)
{
nodeDefOutput->removeAttribute(filterAttribute);
}
nodeDefOutput->setSourceUri(EMPTY_STRING);
}
}

return nodeDef;
Expand Down
2 changes: 1 addition & 1 deletion source/MaterialXCore/Document.h
Original file line number Diff line number Diff line change
Expand Up @@ -339,9 +339,9 @@ class MX_CORE_API Document : public GraphElement
/// @param node Node type for the new declaration
/// @param version Version for the new declaration
/// @param isDefaultVersion If a version is specified is thie definition the default version
/// @param nodeGroup Optional node group for the new declaration. The Default value is an emptry string.
/// @param newGraphName Make a copy of this NodeGraph with the given name if a non-empty name is provided. Otherwise
/// modify the existing NodeGraph. Default value is an empty string.
jstone-lucasfilm marked this conversation as resolved.
Show resolved Hide resolved
/// @param nodeGroup Optional node group for the new declaration. The Default value is an emptry string.
/// @return New declaration if successful.
NodeDefPtr addNodeDefFromGraph(const NodeGraphPtr nodeGraph, const string& nodeDefName, const string& node, const string& version,
bool isDefaultVersion, const string& nodeGroup, const string& newGraphName);
Expand Down
Loading