diff --git a/README.md b/README.md index 6bb89816..42cc12d7 100644 --- a/README.md +++ b/README.md @@ -20,7 +20,7 @@ If you find value in here and want support its progress, Please consider sponsor ## Visit the site - + ## March, 2024 Update diff --git a/about.html b/about.html new file mode 100644 index 00000000..4641eb92 --- /dev/null +++ b/about.html @@ -0,0 +1,341 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + MaterialX Learn + + + + + + + + + + + + + + +
+
+ + +
+
+
+
+
+

About

+

The aim of this site is target as wide an audience as possible, with the + primary goals to aid in learning, developing and integrating MaterialX with + respect to user workflows. +

+

+ Detailed specification information can be found at the + MaterialX site + which includes additional documents and presentations from the authors. + The aim is not to reproduce but to distill and enhance with complementary content. +

+

+ In the "spirit" of MaterialX, the site design tries to match the related content. + Where applicable, all drawings and renderings (including the site logo) are generated using MaterialX. +

+
+ +
+ +
+
+ +
+
+ + +
+
+
+

+

Sponsor

+

If you find value in here and want support its progress, + we kindly invite you to consider sponsoring the project. It will help to keep it going and evolve. +

+

+ +

+

Development

+

+ + The hope is that this site will grow organically over time based on user feedback. + Naturally any contributions are greatly appreciated, especially in terms of "learning" tutorials, + and example user workflows. More will be added to the GitHub repo + as things progress. Currently all content is structured to be as reusable as possible outside of this + site, + even if not fully discussed yet. +
+ To contribute to the MaterialX open source effort any applicable work done here will be + integrated back into the core repository + . +

+

Authors

+
+
+
+

+ + + Bernard Kwok: Bernard has been working in the graphics industry for 30+ years in the areas + of rendering, games and interactive workflows. His current passion is open standards + and is an active contributor for + + MaterialX + + at the + Academy Software Foundation and the Khronos glTF PBR working group. +

+
+
+ + + Matthew Kwok: Matthew is a UX designer with experience in product design for content creation. + His current interests lie in the area of learning and visualization for web and AR, + which includes workflows using realistic materials. +
+
+ Note: Image filtering written as a MaterialX + graph +
+
+
+
+
+
+ +
+
+ + + + + + \ No newline at end of file diff --git a/about.md b/about.md index 6efb6025..c2db3604 100644 --- a/about.md +++ b/about.md @@ -9,13 +9,13 @@ In the "spirit" of MaterialX, the site design tries to match the related content
- + - + - +
@@ -33,7 +33,7 @@ with open standards as an active contributor for MaterialX at the Academy Softwa ### Development - The hope is that this site will grow organically over time based on user feedback. + The hope is that this site will grow organically over time based on user feedback. Naturally any contributions are greatly appreciated, especially in terms of "learning" tutorials, and example user workflows. More will be added to the GitHub repo as things progress. @@ -47,7 +47,7 @@ Currently all content is structured to be as reusable as possible outside of thi @@ -531,17 +536,17 @@

Learning Notebooks

@@ -558,17 +563,17 @@

Learning Notebooks

@@ -583,17 +588,17 @@

Learning Notebooks

@@ -609,17 +614,17 @@

Learning Notebooks

@@ -634,17 +639,17 @@

Learning Notebooks

@@ -661,17 +666,17 @@

Learning Notebooks

@@ -688,17 +693,17 @@

Learning Notebooks

@@ -715,17 +720,17 @@

Learning Notebooks

@@ -740,17 +745,17 @@

Learning Notebooks

@@ -765,17 +770,17 @@

Learning Notebooks

@@ -786,17 +791,17 @@

Learning Notebooks

@@ -811,17 +816,17 @@

Learning Notebooks

@@ -835,16 +840,16 @@

Learning Notebooks

@@ -858,13 +863,13 @@

Learning Notebooks

Below is an example of the "Basics" notebook after execution inside of VSCode.

- +

Below is a snapshot of Python and Markdown with meta data tags within a Python generated from a notebook.

- +

@@ -897,7 +902,7 @@

Learning Notebooks

- Creative Commons License © 2022-2024 NanMu Consulting.

diff --git a/documents/using_library.html b/documents/using_library.html index bb83bbef..8e4f9e7a 100644 --- a/documents/using_library.html +++ b/documents/using_library.html @@ -3,6 +3,9 @@ + @@ -22,24 +25,26 @@ } - + - + + MaterialX Learn - + @@ -53,7 +58,7 @@
- +
Bernard Kwok

Bernard has been working in the graphics industry for 30+ years in the areas of rendering, games and interactive workflows. His current passion is open standards and is an active contributor for MaterialX @@ -56,7 +56,7 @@ Foundation and the K
-
Matthew Kwok
+
Matthew Kwok
Matthew is a UX designer with experience in graphic design and content creation. His current interests lie in the area of learning and visualization for web and AR, diff --git a/documents/about.html b/documents/about.html index adaa71ad..2a398025 100644 --- a/documents/about.html +++ b/documents/about.html @@ -3,6 +3,9 @@ + @@ -22,24 +25,26 @@ } - + - + + MaterialX Learn - + @@ -53,7 +58,7 @@
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
- + - + - +
+
+ +

+
+ + + @@ -494,23 +504,51 @@

Document Visualization

}); */ function getUniformValues(shaderStage) { - console.log('Get Uniforms for stage:', shaderStage); + //console.log('Get Uniforms for stage:', shaderStage); let foundUniforms = {}; const uniformBlocks = Object.values(shaderStage.getUniformBlocks()); + + let uniformsResult = [] uniformBlocks.forEach(uniforms => { if (!uniforms.empty()) { for (let i = 0; i < uniforms.size(); ++i) { const var1 = uniforms.get(i); const value = var1.getValue()?.getValueString(); const name = var1.getVariable(); - console.log(' Uniform name: ', name, ' type: ', var1.getType().getName(), ' value: ', value, ' path: ', var1.getPath()) + + //console.log('Uniform:', name, var1.getType().getName(), value, var1.getPath()) + uniformEntry = [ name, var1.getType().getName(), value, var1.getPath() ]; + uniformsResult.push(uniformEntry); //foundUniforms[name] = { var1.getType().getName(), value, name, uniforms }; } } }); - return foundUniforms; + return uniformsResult; } + let cachedReflectionData = '' + + function saveReflection() + { + if (cachedReflectionData) + { + console.log('Save Reflection...'); + let outputJSON = {} + let jsonString = JSON.stringify(cachedReflectionData, null, 2); + var blob = new Blob([jsonString], { type: 'application/json' }); + + // Create a link element + var a = document.getElementById('reflectionLink'); + a.href = URL.createObjectURL(blob); + a.download = cachedReflectionData.element + '_' + + cachedReflectionData.generate + '_' + + cachedReflectionData.interface + + '_reflection.json'; + a.click(); + } + } + document.getElementById('saveReflection').addEventListener('click', saveReflection); + // Add event handler for generate shader document.getElementById('generateShaderButton').addEventListener('click', () => { if (renderableItems.length == 0) { @@ -527,8 +565,8 @@

Document Visualization

return; } - let pixelSource = 'None' - let vertexSource = 'None' + let pixelSource = '' + let vertexSource = '' setupGenerators(mdoc) @@ -536,30 +574,22 @@

Document Visualization

let genContext = esslgenContext; let shaderLanguage = document.getElementById('shaderLanguage').value; if (shaderLanguage == 1) { - console.log('-------------- USE OSL ---------------') - generator = oslgenerator; - genContext = oslgenContext; - } - else if (shaderLanguage == 1) { - console.log('-------------- USE OSL ---------------') generator = oslgenerator; genContext = oslgenContext; } else if (shaderLanguage == 2) { - console.log('-------------- USE GLSL ---------------') generator = glslgenerator; genContext = glslgenContext; } else if (shaderLanguage == 3) { - console.log('-------------- USE MSL ---------------') generator = mslgenerator; genContext = mslgenContext; } else if (shaderLanguage == 4) { - console.log('-------------- USE VK ---------------') generator = vkgenerator; genContext = vkgenContext; } + console.log('-------------- Use Generator for: %s ---------------', generator.getTarget()) const isTransparent = mmx.isTransparentSurface(elem, generator.getTarget()); genContext.getOptions().hwTransparency = isTransparent; @@ -601,8 +631,52 @@

Document Visualization

//let uniforms = getUniformValues(shader.getStage('vertex')); //console.log('Vertex Uniforms:', uniforms); - //let puniforms = getUniformValues(shader.getStage('pixel')); - //console.log('Pixel Uniforms:', puniforms); + let pixelUniforms = getUniformValues(shader.getStage('pixel')); + cachedReflectionData = + { + "element": elemPath, + "generate": generator.getTarget(), + "interface": interfaceString.toLowerCase(), + "uniforms": pixelUniforms + }; + if (vertexSource == '') + cachedReflectionData.vertex = 'None'; + else + cachedReflectionData.vertex = vertexSource; + if (pixelSource == '') + cachedReflectionData.pixel = 'None'; + else + cachedReflectionData.pixel = pixelSource; + + let reflectionArea = document.getElementById('materialXShaderReflect'); + // remove any previous children + while (reflectionArea.firstChild) { + reflectionArea.removeChild(reflectionArea.firstChild); + } + + // Add header: + let headers = ['Uniform', 'Type', 'Value', 'Path']; + let headerRow = document.createElement('tr'); + headers.forEach(function(title) { + var headerCell = document.createElement("th"); + headerCell.textContent = title; + headerRow.appendChild(headerCell); + }); + reflectionArea.appendChild(headerRow); + + // Add content + for (let i = 0; i < pixelUniforms.length; i++) { + let uniform = pixelUniforms[i]; + let row = document.createElement('tr'); + for (let j = 0; j < uniform.length; j++) { + let cell = document.createElement('td'); + cell.textContent = uniform[j]; + row.appendChild(cell); + } + reflectionArea.appendChild(row); + } + //let jsonResult = JSON.stringify(cachedReflectionData, null, 2); + //console.log('Reflection Data:', jsonResult); } ); diff --git a/javascript/viewer/dist/index_out.html b/javascript/viewer/dist/index_out.html index 34d4a3e5..6bbd5731 100644 --- a/javascript/viewer/dist/index_out.html +++ b/javascript/viewer/dist/index_out.html @@ -3,6 +3,9 @@ + @@ -22,19 +25,21 @@ } - + - + + @@ -75,7 +80,7 @@ MaterialX Learn - + @@ -327,9 +332,19 @@

Document Visualization


- +
+ +

+
+ +
+
+

+
+ + @@ -687,23 +702,51 @@

Document Visualization

}); */ function getUniformValues(shaderStage) { - console.log('Get Uniforms for stage:', shaderStage); + //console.log('Get Uniforms for stage:', shaderStage); let foundUniforms = {}; const uniformBlocks = Object.values(shaderStage.getUniformBlocks()); + + let uniformsResult = [] uniformBlocks.forEach(uniforms => { if (!uniforms.empty()) { for (let i = 0; i < uniforms.size(); ++i) { const var1 = uniforms.get(i); const value = var1.getValue()?.getValueString(); const name = var1.getVariable(); - console.log(' Uniform name: ', name, ' type: ', var1.getType().getName(), ' value: ', value, ' path: ', var1.getPath()) + + //console.log('Uniform:', name, var1.getType().getName(), value, var1.getPath()) + uniformEntry = [ name, var1.getType().getName(), value, var1.getPath() ]; + uniformsResult.push(uniformEntry); //foundUniforms[name] = { var1.getType().getName(), value, name, uniforms }; } } }); - return foundUniforms; + return uniformsResult; } + let cachedReflectionData = '' + + function saveReflection() + { + if (cachedReflectionData) + { + console.log('Save Reflection...'); + let outputJSON = {} + let jsonString = JSON.stringify(cachedReflectionData, null, 2); + var blob = new Blob([jsonString], { type: 'application/json' }); + + // Create a link element + var a = document.getElementById('reflectionLink'); + a.href = URL.createObjectURL(blob); + a.download = cachedReflectionData.element + '_' + + cachedReflectionData.generate + '_' + + cachedReflectionData.interface + + '_reflection.json'; + a.click(); + } + } + document.getElementById('saveReflection').addEventListener('click', saveReflection); + // Add event handler for generate shader document.getElementById('generateShaderButton').addEventListener('click', () => { if (renderableItems.length == 0) { @@ -720,8 +763,8 @@

Document Visualization

return; } - let pixelSource = 'None' - let vertexSource = 'None' + let pixelSource = '' + let vertexSource = '' setupGenerators(mdoc) @@ -729,30 +772,22 @@

Document Visualization

let genContext = esslgenContext; let shaderLanguage = document.getElementById('shaderLanguage').value; if (shaderLanguage == 1) { - console.log('-------------- USE OSL ---------------') - generator = oslgenerator; - genContext = oslgenContext; - } - else if (shaderLanguage == 1) { - console.log('-------------- USE OSL ---------------') generator = oslgenerator; genContext = oslgenContext; } else if (shaderLanguage == 2) { - console.log('-------------- USE GLSL ---------------') generator = glslgenerator; genContext = glslgenContext; } else if (shaderLanguage == 3) { - console.log('-------------- USE MSL ---------------') generator = mslgenerator; genContext = mslgenContext; } else if (shaderLanguage == 4) { - console.log('-------------- USE VK ---------------') generator = vkgenerator; genContext = vkgenContext; } + console.log('-------------- Use Generator for: %s ---------------', generator.getTarget()) const isTransparent = mmx.isTransparentSurface(elem, generator.getTarget()); genContext.getOptions().hwTransparency = isTransparent; @@ -794,8 +829,52 @@

Document Visualization

//let uniforms = getUniformValues(shader.getStage('vertex')); //console.log('Vertex Uniforms:', uniforms); - //let puniforms = getUniformValues(shader.getStage('pixel')); - //console.log('Pixel Uniforms:', puniforms); + let pixelUniforms = getUniformValues(shader.getStage('pixel')); + cachedReflectionData = + { + "element": elemPath, + "generate": generator.getTarget(), + "interface": interfaceString.toLowerCase(), + "uniforms": pixelUniforms + }; + if (vertexSource == '') + cachedReflectionData.vertex = 'None'; + else + cachedReflectionData.vertex = vertexSource; + if (pixelSource == '') + cachedReflectionData.pixel = 'None'; + else + cachedReflectionData.pixel = pixelSource; + + let reflectionArea = document.getElementById('materialXShaderReflect'); + // remove any previous children + while (reflectionArea.firstChild) { + reflectionArea.removeChild(reflectionArea.firstChild); + } + + // Add header: + let headers = ['Uniform', 'Type', 'Value', 'Path']; + let headerRow = document.createElement('tr'); + headers.forEach(function(title) { + var headerCell = document.createElement("th"); + headerCell.textContent = title; + headerRow.appendChild(headerCell); + }); + reflectionArea.appendChild(headerRow); + + // Add content + for (let i = 0; i < pixelUniforms.length; i++) { + let uniform = pixelUniforms[i]; + let row = document.createElement('tr'); + for (let j = 0; j < uniform.length; j++) { + let cell = document.createElement('td'); + cell.textContent = uniform[j]; + row.appendChild(cell); + } + reflectionArea.appendChild(row); + } + //let jsonResult = JSON.stringify(cachedReflectionData, null, 2); + //console.log('Reflection Data:', jsonResult); } ); diff --git a/pymaterialx/data/IM_mx_acescg_to_lin_rec709_color4.mtlx b/pymaterialx/data/IM_mx_acescg_to_lin_rec709_color4.mtlx index ea8e96db..bbeed47e 100644 --- a/pymaterialx/data/IM_mx_acescg_to_lin_rec709_color4.mtlx +++ b/pymaterialx/data/IM_mx_acescg_to_lin_rec709_color4.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/ND_acescg_to_lin_rec709_color4.mtlx b/pymaterialx/data/ND_acescg_to_lin_rec709_color4.mtlx index afe07d47..75d83bb3 100644 --- a/pymaterialx/data/ND_acescg_to_lin_rec709_color4.mtlx +++ b/pymaterialx/data/ND_acescg_to_lin_rec709_color4.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/ND_acescg_to_lin_rec709_color4_2.mtlx b/pymaterialx/data/ND_acescg_to_lin_rec709_color4_2.mtlx index bd4d346f..a970fdb8 100644 --- a/pymaterialx/data/ND_acescg_to_lin_rec709_color4_2.mtlx +++ b/pymaterialx/data/ND_acescg_to_lin_rec709_color4_2.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/graph_image.md b/pymaterialx/data/graph_image.md index 2a7315ae..081b0b06 100644 --- a/pymaterialx/data/graph_image.md +++ b/pymaterialx/data/graph_image.md @@ -4,6 +4,6 @@ graph LR; a[Input] --> test --> output b[Input2] --> test -test[ Lama Add] +test[ Lama Add] ``` \ No newline at end of file diff --git a/pymaterialx/data/myadd_definition.mtlx b/pymaterialx/data/myadd_definition.mtlx index 8b47416c..f041ddbd 100644 --- a/pymaterialx/data/myadd_definition.mtlx +++ b/pymaterialx/data/myadd_definition.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/myadd_definition_file.mtlx b/pymaterialx/data/myadd_definition_file.mtlx index bbb670e4..e8b8584c 100644 --- a/pymaterialx/data/myadd_definition_file.mtlx +++ b/pymaterialx/data/myadd_definition_file.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/sample_nodegraph.mtlx b/pymaterialx/data/sample_nodegraph.mtlx index cefb32a6..e2a0eeef 100644 --- a/pymaterialx/data/sample_nodegraph.mtlx +++ b/pymaterialx/data/sample_nodegraph.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/data/test_usd_mtlx.mtlx b/pymaterialx/data/test_usd_mtlx.mtlx index 3b479c43..57d01005 100644 --- a/pymaterialx/data/test_usd_mtlx.mtlx +++ b/pymaterialx/data/test_usd_mtlx.mtlx @@ -1,5 +1,5 @@ - + diff --git a/pymaterialx/mtlx_brick.json b/pymaterialx/mtlx_brick.json index 61f610f6..8b974370 100644 --- a/pymaterialx/mtlx_brick.json +++ b/pymaterialx/mtlx_brick.json @@ -1,6 +1,6 @@ { "materialx": { - "version": "1.38", + "version": "1.39", "nodegraph": { "name": "NG_BrickPattern", "input": [ diff --git a/pymaterialx/mtlx_copilot.html b/pymaterialx/mtlx_copilot.html index 267ff98e..16c19aeb 100644 --- a/pymaterialx/mtlx_copilot.html +++ b/pymaterialx/mtlx_copilot.html @@ -7560,7 +7560,7 @@

Correctness and Repeatability +No description has been provided for this image

For the example below, it was able to suggest the correct call node graph creation, but the complete code including the import and doc creation must be visible on screen and within the same notebook cell for this to work. Sometimes it will just not come up with anything so there is @@ -7611,7 +7611,7 @@

Correctness and Repeatability

Use the Co-Pilot action to "fix" or explain code is mostly of no use as it will return generic information about language or environment syntax and not anything specific to the code.

For example querying to "explain" the document doc. returns generic information about the need to call a member function, and the "fix" action actually fairs no better, reusing the same non-existent API calls.

-No description has been provided for this image +No description has been provided for this image @@ -7658,7 +7658,7 @@

Standards¶

For examine move semantics are suggested for performance on a given function, but this may not be available. (For this example it could not parse the function signature correctly and suggested acting on non-existent arguments.)

-No description has been provided for this image +No description has been provided for this image

Relevance¶

Using fixed data is one of the major drawbacks. It appears to be possible to add in additional data but this has not been attempted for Copilot or any other LLM.

Repeatability¶

This is a general issue that tuning can help to alleviate but just from Visual Studio (Code) @@ -7677,20 +7677,20 @@

Appendix: Setup -No description has been provided for this image -No description has been provided for this image +No description has been provided for this image +No description has been provided for this image

The chat interface for Copilot is available by signing up for the beta program at this time of writing. The chat feature can be used in a similar way to ChatGPT but within the context of a working environment. This has the added advantage of avoiding some context switching. (There are ChatGPT and other unofficial add-ons which are available but not considered here.)

The following is a snapshot with and chat add-ons installed: -No description has been provided for this image

+No description has been provided for this image

diff --git a/pymaterialx/mtlx_copilot.ipynb b/pymaterialx/mtlx_copilot.ipynb index 8850e959..9151a8bd 100644 --- a/pymaterialx/mtlx_copilot.ipynb +++ b/pymaterialx/mtlx_copilot.ipynb @@ -110,7 +110,7 @@ "libraries are not documented. After using Co-Pilot for a while it was able to suggest API calls and\n", "argument completions. An example is shown below of additional suggestions added for `Document` calls.`\n", "\n", - "\n", + "\n", "\n", "For the example below, it was able to suggest the correct call node graph creation, but the complete\n", "code including the import and doc creation must be visible on screen and within the same\n", @@ -151,7 +151,7 @@ "\n", "For example querying to \"explain\" the document `doc.` returns generic information about the need to call a member function, and the \"fix\" action actually fairs no better, reusing the same non-existent API calls.\n", "\n", - "" + "" ] }, { @@ -201,7 +201,7 @@ "(For this example it could not parse the function signature correctly and suggested acting on\n", "non-existent arguments.)\n", "\n", - "\n", + "\n", "\n", "### Relevance\n", "\n", @@ -226,8 +226,8 @@ "\n", "\n", "\n", - "\n", "\n", "
\n", - "\n", + "\n", + "\n", "
\n", @@ -236,13 +236,13 @@ "but not considered here.) \n", "\n", "The following is a snapshot with and chat add-ons installed:\n", - "\n", + "\n", "\n", "
\n", "
\n", "
\n", "
\n", - " \"Image\n", + " \"Image\n", "
\n", "
\n", "
\n", @@ -258,7 +258,7 @@ "\n", "The last option (Running from terminal) only makes sense for code which can run as a command. At the current time the response is not checked to see if it's runnable.\n", "\n", - "Copilot can be turned on or off via the icon highlighted on the bottom right." + "Copilot can be turned on or off via the icon highlighted on the bottom right." ] } ], diff --git a/pymaterialx/mtlx_copilot.py b/pymaterialx/mtlx_copilot.py index 57d48037..cf5cc0b4 100644 --- a/pymaterialx/mtlx_copilot.py +++ b/pymaterialx/mtlx_copilot.py @@ -74,7 +74,7 @@ # libraries are not documented. After using Co-Pilot for a while it was able to suggest API calls and # argument completions. An example is shown below of additional suggestions added for `Document` calls.` # -# +# # # For the example below, it was able to suggest the correct call node graph creation, but the complete # code including the import and doc creation must be visible on screen and within the same @@ -97,7 +97,7 @@ # # For example querying to "explain" the document `doc.` returns generic information about the need to call a member function, and the "fix" action actually fairs no better, reusing the same non-existent API calls. # -# +# # %% [markdown] # Using Chat is mostly not recommended. It is unknown how the response is generated but it is pretty well always unusable. As an example the prompt was on how to create a node definition. The example XML returned was: @@ -140,7 +140,7 @@ # (Strangely, for this example it could not parse the function signature correctly and suggested acting on # non-existent arguments.) # -# +# # # ### Relevance # @@ -162,8 +162,8 @@ # # # -# # #
-# +# +# #
@@ -172,13 +172,13 @@ # but not considered here) # # The following is a snapshot with and chat add-ons installed: -# +# # #
@@ -15130,7 +15130,7 @@

Definition and Instance ValidationIntegration with Node Graph Editors

By including the search path and library names to the MaterialX node editor which comes with the core distribution, graphs can be created and edited with this tool.

The following is a similar nodegraph example but created in the editor.

- +
<?xml version="1.0"?>
 <materialx version="1.38" colorspace="lin_rec709">
   <surfacematerial name="MayaBliinChecker" type="material" xpos="14.442029" ypos="-2.456897">
@@ -15498,7 +15498,7 @@ 

Loading Libraries +

@@ -15518,7 +15518,7 @@

Houdini MaterialX ExportFor now both do no handle these MaterialX nodes.

  • Export to a MaterialX document is directly exposed (unlike in Maya currently). For this, the vop2mtlx Python script can be used directly or via the Houdini user interface.

    - +

In either case, an asset (hda) must be created from the subnet before export. The additional houdini library definitions are recogized by the converter.

@@ -15552,11 +15552,11 @@

Houdini MaterialX Export
  • Top level nodegraph
  • - +
    • Expanded nodegraph contents
    - + diff --git a/pymaterialx/mtlx_definitions_libraries_notebook.ipynb b/pymaterialx/mtlx_definitions_libraries_notebook.ipynb index 4378397c..e09372a5 100644 --- a/pymaterialx/mtlx_definitions_libraries_notebook.ipynb +++ b/pymaterialx/mtlx_definitions_libraries_notebook.ipynb @@ -23,7 +23,7 @@ "\n", "The basic data flow examined is shown below:\n", "\n", - "\n", + "\n", "\n", "Note that regardless of all libraries are in MaterialX form so can be accessed outside of the application integraion.\n", "\n", @@ -435,7 +435,7 @@ "\n", "The following is a similar nodegraph example but created in the editor.\n", "\n", - "\n", + "\n", "\n", "```xml\n", "\n", @@ -744,7 +744,7 @@ "* `nodegroup` is set to be `houdini` which makes it harder to classify for code generation, but Houdini does\n", "not directly use a MaterialX code generator to produce VEX.\n", "\n", - "" + "" ] }, { @@ -764,7 +764,7 @@ "\n", "* Export to a MaterialX document is directly exposed (unlike in Maya currently). For this, the `vop2mtlx` Python script can be used directly or via the Houdini user interface. \n", " \n", - " \n", + " \n", "\n", "In either case, an asset (`hda`) must be created from the subnet before export. The additional `houdini` library definitions are recogized by the converter.\n", "\n", @@ -791,11 +791,11 @@ "\n", "* Top level nodegraph\n", "\n", - "\n", + "\n", "\n", "* Expanded nodegraph contents\n", "\n", - "\n", + "\n", "\n" ] }, diff --git a/pymaterialx/mtlx_definitions_libraries_notebook.py b/pymaterialx/mtlx_definitions_libraries_notebook.py index 326cd8e6..44fedc57 100644 --- a/pymaterialx/mtlx_definitions_libraries_notebook.py +++ b/pymaterialx/mtlx_definitions_libraries_notebook.py @@ -849,7 +849,7 @@ def setImplementationSourceCode(doc, nodedef, targets, sourceCode, inlined): # # Below shows a layout for how the standard libraries are organized on the left. # -# +# # # For example if we consider the grouping on the left to be `stdlib`, then it is composed of: # * A separate file `stdlib_defs.mtlx` containing all definitions (`_defs`) diff --git a/pymaterialx/mtlx_definitions_notebook.html b/pymaterialx/mtlx_definitions_notebook.html index 1e5d0a7e..fcdca1d2 100644 --- a/pymaterialx/mtlx_definitions_notebook.html +++ b/pymaterialx/mtlx_definitions_notebook.html @@ -277,8 +277,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-Widget, /* */ +/* */ +.p-Widget, /* */ .lm-Widget { box-sizing: border-box; position: relative; @@ -286,12 +286,19 @@ cursor: default; } - -/* */ .p-Widget.p-mod-hidden, /* */ +/* */ +.p-Widget.p-mod-hidden, /* */ .lm-Widget.lm-mod-hidden { display: none !important; } +.lm-AccordionPanel[data-orientation='horizontal'] > .lm-AccordionPanel-title { + /* Title is rotated for horizontal accordion panel using CSS */ + display: block; + transform-origin: top left; + transform: rotate(-90deg) translate(-100%); +} + /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Copyright (c) 2014-2017, PhosphorJS Contributors @@ -301,8 +308,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-CommandPalette, /* */ +/* */ +.p-CommandPalette, /* */ .lm-CommandPalette { display: flex; flex-direction: column; @@ -312,14 +319,14 @@ user-select: none; } - -/* */ .p-CommandPalette-search, /* */ +/* */ +.p-CommandPalette-search, /* */ .lm-CommandPalette-search { flex: 0 0 auto; } - -/* */ .p-CommandPalette-content, /* */ +/* */ +.p-CommandPalette-content, /* */ .lm-CommandPalette-content { flex: 1 1 auto; margin: 0; @@ -329,42 +336,42 @@ list-style-type: none; } - -/* */ .p-CommandPalette-header, /* */ +/* */ +.p-CommandPalette-header, /* */ .lm-CommandPalette-header { overflow: hidden; white-space: nowrap; text-overflow: ellipsis; } - -/* */ .p-CommandPalette-item, /* */ +/* */ +.p-CommandPalette-item, /* */ .lm-CommandPalette-item { display: flex; flex-direction: row; } - -/* */ .p-CommandPalette-itemIcon, /* */ +/* */ +.p-CommandPalette-itemIcon, /* */ .lm-CommandPalette-itemIcon { flex: 0 0 auto; } - -/* */ .p-CommandPalette-itemContent, /* */ +/* */ +.p-CommandPalette-itemContent, /* */ .lm-CommandPalette-itemContent { flex: 1 1 auto; overflow: hidden; } - -/* */ .p-CommandPalette-itemShortcut, /* */ +/* */ +.p-CommandPalette-itemShortcut, /* */ .lm-CommandPalette-itemShortcut { flex: 0 0 auto; } - -/* */ .p-CommandPalette-itemLabel, /* */ +/* */ +.p-CommandPalette-itemLabel, /* */ .lm-CommandPalette-itemLabel { overflow: hidden; white-space: nowrap; @@ -372,30 +379,30 @@ } .lm-close-icon { - border:1px solid transparent; + border: 1px solid transparent; background-color: transparent; position: absolute; - z-index:1; - right:3%; - top: 0; - bottom: 0; - margin: auto; - padding: 7px 0; - display: none; - vertical-align: middle; + z-index: 1; + right: 3%; + top: 0; + bottom: 0; + margin: auto; + padding: 7px 0; + display: none; + vertical-align: middle; outline: 0; cursor: pointer; } .lm-close-icon:after { - content: "X"; - display: block; - width: 15px; - height: 15px; - text-align: center; - color:#000; - font-weight: normal; - font-size: 12px; - cursor: pointer; + content: 'X'; + display: block; + width: 15px; + height: 15px; + text-align: center; + color: #000; + font-weight: normal; + font-size: 12px; + cursor: pointer; } /*----------------------------------------------------------------------------- @@ -407,38 +414,38 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-DockPanel, /* */ +/* */ +.p-DockPanel, /* */ .lm-DockPanel { z-index: 0; } - -/* */ .p-DockPanel-widget, /* */ +/* */ +.p-DockPanel-widget, /* */ .lm-DockPanel-widget { z-index: 0; } - -/* */ .p-DockPanel-tabBar, /* */ +/* */ +.p-DockPanel-tabBar, /* */ .lm-DockPanel-tabBar { z-index: 1; } - -/* */ .p-DockPanel-handle, /* */ +/* */ +.p-DockPanel-handle, /* */ .lm-DockPanel-handle { z-index: 2; } - -/* */ .p-DockPanel-handle.p-mod-hidden, /* */ +/* */ +.p-DockPanel-handle.p-mod-hidden, /* */ .lm-DockPanel-handle.lm-mod-hidden { display: none !important; } - -/* */ .p-DockPanel-handle:after, /* */ +/* */ +.p-DockPanel-handle:after, /* */ .lm-DockPanel-handle:after { position: absolute; top: 0; @@ -448,7 +455,6 @@ content: ''; } - /* */ .p-DockPanel-handle[data-orientation='horizontal'], /* */ @@ -456,7 +462,6 @@ cursor: ew-resize; } - /* */ .p-DockPanel-handle[data-orientation='vertical'], /* */ @@ -464,7 +469,6 @@ cursor: ns-resize; } - /* */ .p-DockPanel-handle[data-orientation='horizontal']:after, /* */ @@ -474,7 +478,6 @@ transform: translateX(-50%); } - /* */ .p-DockPanel-handle[data-orientation='vertical']:after, /* */ @@ -484,16 +487,16 @@ transform: translateY(-50%); } - -/* */ .p-DockPanel-overlay, /* */ +/* */ +.p-DockPanel-overlay, /* */ .lm-DockPanel-overlay { z-index: 3; box-sizing: border-box; pointer-events: none; } - -/* */ .p-DockPanel-overlay.p-mod-hidden, /* */ +/* */ +.p-DockPanel-overlay.p-mod-hidden, /* */ .lm-DockPanel-overlay.lm-mod-hidden { display: none !important; } @@ -507,8 +510,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-Menu, /* */ +/* */ +.p-Menu, /* */ .lm-Menu { z-index: 10000; position: absolute; @@ -522,8 +525,8 @@ user-select: none; } - -/* */ .p-Menu-content, /* */ +/* */ +.p-Menu-content, /* */ .lm-Menu-content { margin: 0; padding: 0; @@ -531,13 +534,12 @@ list-style-type: none; } - -/* */ .p-Menu-item, /* */ +/* */ +.p-Menu-item, /* */ .lm-Menu-item { display: table-row; } - /* */ .p-Menu-item.p-mod-hidden, .p-Menu-item.p-mod-collapsed, @@ -547,7 +549,6 @@ display: none !important; } - /* */ .p-Menu-itemIcon, .p-Menu-itemSubmenuIcon, @@ -558,15 +559,15 @@ text-align: center; } - -/* */ .p-Menu-itemLabel, /* */ +/* */ +.p-Menu-itemLabel, /* */ .lm-Menu-itemLabel { display: table-cell; text-align: left; } - -/* */ .p-Menu-itemShortcut, /* */ +/* */ +.p-Menu-itemShortcut, /* */ .lm-Menu-itemShortcut { display: table-cell; text-align: right; @@ -581,8 +582,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-MenuBar, /* */ +/* */ +.p-MenuBar, /* */ .lm-MenuBar { outline: none; -webkit-user-select: none; @@ -591,8 +592,8 @@ user-select: none; } - -/* */ .p-MenuBar-content, /* */ +/* */ +.p-MenuBar-content, /* */ .lm-MenuBar-content { margin: 0; padding: 0; @@ -601,13 +602,12 @@ list-style-type: none; } - -/* */ .p--MenuBar-item, /* */ +/* */ +.p--MenuBar-item, /* */ .lm-MenuBar-item { box-sizing: border-box; } - /* */ .p-MenuBar-itemIcon, .p-MenuBar-itemLabel, @@ -626,8 +626,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-ScrollBar, /* */ +/* */ +.p-ScrollBar, /* */ .lm-ScrollBar { display: flex; -webkit-user-select: none; @@ -636,7 +636,6 @@ user-select: none; } - /* */ .p-ScrollBar[data-orientation='horizontal'], /* */ @@ -644,7 +643,6 @@ flex-direction: row; } - /* */ .p-ScrollBar[data-orientation='vertical'], /* */ @@ -652,15 +650,15 @@ flex-direction: column; } - -/* */ .p-ScrollBar-button, /* */ +/* */ +.p-ScrollBar-button, /* */ .lm-ScrollBar-button { box-sizing: border-box; flex: 0 0 auto; } - -/* */ .p-ScrollBar-track, /* */ +/* */ +.p-ScrollBar-track, /* */ .lm-ScrollBar-track { box-sizing: border-box; position: relative; @@ -668,8 +666,8 @@ flex: 1 1 auto; } - -/* */ .p-ScrollBar-thumb, /* */ +/* */ +.p-ScrollBar-thumb, /* */ .lm-ScrollBar-thumb { box-sizing: border-box; position: absolute; @@ -684,26 +682,26 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-SplitPanel-child, /* */ +/* */ +.p-SplitPanel-child, /* */ .lm-SplitPanel-child { z-index: 0; } - -/* */ .p-SplitPanel-handle, /* */ +/* */ +.p-SplitPanel-handle, /* */ .lm-SplitPanel-handle { z-index: 1; } - -/* */ .p-SplitPanel-handle.p-mod-hidden, /* */ +/* */ +.p-SplitPanel-handle.p-mod-hidden, /* */ .lm-SplitPanel-handle.lm-mod-hidden { display: none !important; } - -/* */ .p-SplitPanel-handle:after, /* */ +/* */ +.p-SplitPanel-handle:after, /* */ .lm-SplitPanel-handle:after { position: absolute; top: 0; @@ -713,7 +711,6 @@ content: ''; } - /* */ .p-SplitPanel[data-orientation='horizontal'] > .p-SplitPanel-handle, /* */ @@ -721,7 +718,6 @@ cursor: ew-resize; } - /* */ .p-SplitPanel[data-orientation='vertical'] > .p-SplitPanel-handle, /* */ @@ -729,7 +725,6 @@ cursor: ns-resize; } - /* */ .p-SplitPanel[data-orientation='horizontal'] > .p-SplitPanel-handle:after, /* */ @@ -739,7 +734,6 @@ transform: translateX(-50%); } - /* */ .p-SplitPanel[data-orientation='vertical'] > .p-SplitPanel-handle:after, /* */ @@ -758,8 +752,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-TabBar, /* */ +/* */ +.p-TabBar, /* */ .lm-TabBar { display: flex; -webkit-user-select: none; @@ -768,22 +762,22 @@ user-select: none; } - -/* */ .p-TabBar[data-orientation='horizontal'], /* */ +/* */ +.p-TabBar[data-orientation='horizontal'], /* */ .lm-TabBar[data-orientation='horizontal'] { flex-direction: row; align-items: flex-end; } - -/* */ .p-TabBar[data-orientation='vertical'], /* */ +/* */ +.p-TabBar[data-orientation='vertical'], /* */ .lm-TabBar[data-orientation='vertical'] { flex-direction: column; align-items: flex-end; } - -/* */ .p-TabBar-content, /* */ +/* */ +.p-TabBar-content, /* */ .lm-TabBar-content { margin: 0; padding: 0; @@ -792,7 +786,6 @@ list-style-type: none; } - /* */ .p-TabBar[data-orientation='horizontal'] > .p-TabBar-content, /* */ @@ -800,7 +793,6 @@ flex-direction: row; } - /* */ .p-TabBar[data-orientation='vertical'] > .p-TabBar-content, /* */ @@ -808,16 +800,16 @@ flex-direction: column; } - -/* */ .p-TabBar-tab, /* */ +/* */ +.p-TabBar-tab, /* */ .lm-TabBar-tab { display: flex; flex-direction: row; box-sizing: border-box; overflow: hidden; + touch-action: none; /* Disable native Drag/Drop */ } - /* */ .p-TabBar-tabIcon, .p-TabBar-tabCloseIcon, @@ -827,39 +819,36 @@ flex: 0 0 auto; } - -/* */ .p-TabBar-tabLabel, /* */ +/* */ +.p-TabBar-tabLabel, /* */ .lm-TabBar-tabLabel { flex: 1 1 auto; overflow: hidden; white-space: nowrap; } - .lm-TabBar-tabInput { user-select: all; width: 100%; - box-sizing : border-box; + box-sizing: border-box; } - -/* */ .p-TabBar-tab.p-mod-hidden, /* */ +/* */ +.p-TabBar-tab.p-mod-hidden, /* */ .lm-TabBar-tab.lm-mod-hidden { display: none !important; } - .lm-TabBar-addButton.lm-mod-hidden { display: none !important; } - -/* */ .p-TabBar.p-mod-dragging .p-TabBar-tab, /* */ +/* */ +.p-TabBar.p-mod-dragging .p-TabBar-tab, /* */ .lm-TabBar.lm-mod-dragging .lm-TabBar-tab { position: relative; } - /* */ .p-TabBar.p-mod-dragging[data-orientation='horizontal'] .p-TabBar-tab, /* */ @@ -868,7 +857,6 @@ transition: left 150ms ease; } - /* */ .p-TabBar.p-mod-dragging[data-orientation='vertical'] .p-TabBar-tab, /* */ @@ -877,7 +865,6 @@ transition: top 150ms ease; } - /* */ .p-TabBar.p-mod-dragging .p-TabBar-tab.p-mod-dragging, /* */ @@ -888,7 +875,7 @@ .lm-TabBar-tabLabel .lm-TabBar-tabInput { user-select: all; width: 100%; - box-sizing : border-box; + box-sizing: border-box; background: inherit; } @@ -901,14 +888,14 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-TabPanel-tabBar, /* */ +/* */ +.p-TabPanel-tabBar, /* */ .lm-TabPanel-tabBar { z-index: 1; } - -/* */ .p-TabPanel-stackedPanel, /* */ +/* */ +.p-TabPanel-stackedPanel, /* */ .lm-TabPanel-stackedPanel { z-index: 0; } @@ -9365,7 +9352,11 @@ /* Icons urls */ :root { + --jp-icon-add-above: url(); + --jp-icon-add-below: url(); --jp-icon-add: url(); + --jp-icon-bell: url(); + --jp-icon-bug-dot: url(); --jp-icon-bug: url(); --jp-icon-build: url(); --jp-icon-caret-down-empty-thin: url(); @@ -9382,11 +9373,13 @@ --jp-icon-clear: url(); --jp-icon-close: url(); --jp-icon-code: url(); - --jp-icon-console: url(); + --jp-icon-console: url(); --jp-icon-copy: url(); --jp-icon-copyright: url(); --jp-icon-cut: url(); + --jp-icon-delete: url(); --jp-icon-download: url(); + --jp-icon-duplicate: url(); --jp-icon-edit: url(); --jp-icon-ellipses: url(); --jp-icon-extension: url(); @@ -9394,32 +9387,37 @@ --jp-icon-file-upload: url(); --jp-icon-file: url(); --jp-icon-filter-list: url(); + --jp-icon-folder-favorite: url(); --jp-icon-folder: url(); + --jp-icon-home: url(); --jp-icon-html5: url(); --jp-icon-image: url(); - --jp-icon-inspector: url(); - --jp-icon-json: url(); + --jp-icon-inspector: url(); + --jp-icon-json: url(); --jp-icon-julia: url(); - --jp-icon-jupyter-favicon: url(); - --jp-icon-jupyter: url(); + --jp-icon-jupyter-favicon: url(); + --jp-icon-jupyter: url(); --jp-icon-jupyterlab-wordmark: url(); --jp-icon-kernel: url(); --jp-icon-keyboard: url(); + --jp-icon-launch: url(); --jp-icon-launcher: url(); --jp-icon-line-form: url(); --jp-icon-link: url(); --jp-icon-list: url(); --jp-icon-listings-info: url(); --jp-icon-markdown: url(); + --jp-icon-move-down: url(); + --jp-icon-move-up: url(); --jp-icon-new-folder: url(); --jp-icon-not-trusted: url(); - --jp-icon-notebook: url(); + --jp-icon-notebook: url(); --jp-icon-numbering: url(); --jp-icon-offline-bolt: url(); --jp-icon-palette: url(); --jp-icon-paste: url(); --jp-icon-pdf: url(); - --jp-icon-python: url(); + --jp-icon-python: url(); --jp-icon-r-kernel: url(); --jp-icon-react: url(); --jp-icon-redo: url(); @@ -9430,26 +9428,41 @@ --jp-icon-save: url(); --jp-icon-search: url(); --jp-icon-settings: url(); + --jp-icon-share: url(); --jp-icon-spreadsheet: url(); --jp-icon-stop: url(); --jp-icon-tab: url(); --jp-icon-table-rows: url(); --jp-icon-tag: url(); - --jp-icon-terminal: url(); - --jp-icon-text-editor: url(); + --jp-icon-terminal: url(); + --jp-icon-text-editor: url(); --jp-icon-toc: url(); --jp-icon-tree-view: url(); --jp-icon-trusted: url(); --jp-icon-undo: url(); + --jp-icon-user: url(); + --jp-icon-users: url(); --jp-icon-vega: url(); --jp-icon-yaml: url(); } /* Icon CSS class declarations */ +.jp-AddAboveIcon { + background-image: var(--jp-icon-add-above); +} +.jp-AddBelowIcon { + background-image: var(--jp-icon-add-below); +} .jp-AddIcon { background-image: var(--jp-icon-add); } +.jp-BellIcon { + background-image: var(--jp-icon-bell); +} +.jp-BugDotIcon { + background-image: var(--jp-icon-bug-dot); +} .jp-BugIcon { background-image: var(--jp-icon-bug); } @@ -9510,9 +9523,15 @@ .jp-CutIcon { background-image: var(--jp-icon-cut); } +.jp-DeleteIcon { + background-image: var(--jp-icon-delete); +} .jp-DownloadIcon { background-image: var(--jp-icon-download); } +.jp-DuplicateIcon { + background-image: var(--jp-icon-duplicate); +} .jp-EditIcon { background-image: var(--jp-icon-edit); } @@ -9534,9 +9553,15 @@ .jp-FilterListIcon { background-image: var(--jp-icon-filter-list); } +.jp-FolderFavoriteIcon { + background-image: var(--jp-icon-folder-favorite); +} .jp-FolderIcon { background-image: var(--jp-icon-folder); } +.jp-HomeIcon { + background-image: var(--jp-icon-home); +} .jp-Html5Icon { background-image: var(--jp-icon-html5); } @@ -9567,6 +9592,9 @@ .jp-KeyboardIcon { background-image: var(--jp-icon-keyboard); } +.jp-LaunchIcon { + background-image: var(--jp-icon-launch); +} .jp-LauncherIcon { background-image: var(--jp-icon-launcher); } @@ -9585,6 +9613,12 @@ .jp-MarkdownIcon { background-image: var(--jp-icon-markdown); } +.jp-MoveDownIcon { + background-image: var(--jp-icon-move-down); +} +.jp-MoveUpIcon { + background-image: var(--jp-icon-move-up); +} .jp-NewFolderIcon { background-image: var(--jp-icon-new-folder); } @@ -9642,6 +9676,9 @@ .jp-SettingsIcon { background-image: var(--jp-icon-settings); } +.jp-ShareIcon { + background-image: var(--jp-icon-share); +} .jp-SpreadsheetIcon { background-image: var(--jp-icon-spreadsheet); } @@ -9675,6 +9712,12 @@ .jp-UndoIcon { background-image: var(--jp-icon-undo); } +.jp-UserIcon { + background-image: var(--jp-icon-user); +} +.jp-UsersIcon { + background-image: var(--jp-icon-users); +} .jp-VegaIcon { background-image: var(--jp-icon-vega); } @@ -9733,6 +9776,36 @@ | Distributed under the terms of the Modified BSD License. |----------------------------------------------------------------------------*/ +.lm-TabBar .lm-TabBar-addButton { + align-items: center; + display: flex; + padding: 4px; + padding-bottom: 5px; + margin-right: 1px; + background-color: var(--jp-layout-color2); +} + +.lm-TabBar .lm-TabBar-addButton:hover { + background-color: var(--jp-layout-color1); +} + +.lm-DockPanel-tabBar .lm-TabBar-tab { + width: var(--jp-private-horizontal-tab-width); +} + +.lm-DockPanel-tabBar .lm-TabBar-content { + flex: unset; +} + +.lm-DockPanel-tabBar[data-orientation='horizontal'] { + flex: 1 1 auto; +} + +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ + /** * Support for icons as inline SVG HTMLElements */ @@ -9894,15 +9967,40 @@ stroke: var(--jp-icon-contrast-color3); } -/* CSS for icons in selected items in the settings editor */ -#setting-editor .jp-PluginList .jp-mod-selected .jp-icon-selectable[fill] { - fill: #fff; +.jp-jupyter-icon-color[fill] { + fill: var(--jp-jupyter-icon-color, var(--jp-warn-color0)); } -#setting-editor - .jp-PluginList - .jp-mod-selected - .jp-icon-selectable-inverse[fill] { - fill: var(--jp-brand-color1); + +.jp-notebook-icon-color[fill] { + fill: var(--jp-notebook-icon-color, var(--jp-warn-color0)); +} + +.jp-json-icon-color[fill] { + fill: var(--jp-json-icon-color, var(--jp-warn-color1)); +} + +.jp-console-icon-color[fill] { + fill: var(--jp-console-icon-color, white); +} + +.jp-console-icon-background-color[fill] { + fill: var(--jp-console-icon-background-color, var(--jp-brand-color1)); +} + +.jp-terminal-icon-color[fill] { + fill: var(--jp-terminal-icon-color, var(--jp-layout-color2)); +} + +.jp-terminal-icon-background-color[fill] { + fill: var(--jp-terminal-icon-background-color, var(--jp-inverse-layout2)); +} + +.jp-text-editor-icon-color[fill] { + fill: var(--jp-text-editor-icon-color, var(--jp-inverse-layout3)); +} + +.jp-inspector-icon-color[fill] { + fill: var(--jp-inspector-icon-color, var(--jp-inverse-layout3)); } /* CSS for icons in selected filebrowser listing items */ @@ -10085,7 +10183,7 @@ | Distributed under the terms of the Modified BSD License. |----------------------------------------------------------------------------*/ -.jp-icon-hoverShow:not(:hover) svg { +.jp-icon-hoverShow:not(:hover) .jp-icon-hoverShow-content { display: none !important; } @@ -10269,7 +10367,7 @@ .jp-switch-track { cursor: pointer; - background-color: var(--jp-border-color1); + background-color: var(--jp-switch-color, var(--jp-border-color1)); -webkit-transition: 0.4s; transition: 0.4s; border-radius: 34px; @@ -10292,7 +10390,7 @@ } .jp-switch[aria-checked='true'] .jp-switch-track { - background-color: var(--jp-warn-color0); + background-color: var(--jp-switch-true-position-color, var(--jp-warn-color0)); } .jp-switch[aria-checked='true'] .jp-switch-track::before { @@ -10742,8 +10840,7 @@ margin-left: auto; margin-right: auto; background: var(--jp-layout-color1); - padding: 24px; - padding-bottom: 12px; + padding: 24px 24px 12px 24px; min-width: 300px; min-height: 150px; max-width: 1000px; @@ -10759,6 +10856,10 @@ resize: both; } +.jp-Dialog-content.jp-Dialog-content-small { + max-width: 500px; +} + .jp-Dialog-button { overflow: visible; } @@ -10773,6 +10874,25 @@ border: 0; } +button.jp-Dialog-button.jp-mod-styled.jp-mod-accept:focus, +button.jp-Dialog-button.jp-mod-styled.jp-mod-warn:focus, +button.jp-Dialog-button.jp-mod-styled.jp-mod-reject:focus { + outline-offset: 4px; + -moz-outline-radius: 0px; +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-accept:focus { + outline: 1px solid var(--md-blue-700); +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-warn:focus { + outline: 1px solid var(--md-red-600); +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-reject:focus { + outline: 1px solid var(--md-grey-700); +} + button.jp-Dialog-close-button { padding: 0; height: 100%; @@ -10803,12 +10923,26 @@ display: flex; flex-direction: row; justify-content: flex-end; + align-items: center; flex: 0 0 auto; margin-left: -12px; margin-right: -12px; padding: 12px; } +.jp-Dialog-checkbox { + padding-right: 5px; +} + +.jp-Dialog-checkbox > input:focus-visible { + outline: 1px solid var(--jp-input-active-border-color); + outline-offset: 1px; +} + +.jp-Dialog-spacer { + flex: 1 1 auto; +} + .jp-Dialog-title { overflow: hidden; white-space: nowrap; @@ -10900,6 +11034,26 @@ outline: none; } +.jp-MainAreaWidget .jp-MainAreaWidget-error { + padding: 6px; +} + +.jp-MainAreaWidget .jp-MainAreaWidget-error > pre { + width: auto; + padding: 10px; + background: var(--jp-error-color3); + border: var(--jp-border-width) solid var(--jp-error-color1); + border-radius: var(--jp-border-radius); + color: var(--jp-ui-font-color1); + font-size: var(--jp-ui-font-size1); + white-space: pre-wrap; + word-wrap: break-word; +} + +.jp-MainAreaWidget { + contain: strict; +} + /** * google-material-color v1.2.6 * https://github.com/danlevan/google-material-color @@ -11383,8 +11537,8 @@ background: var(--jp-toolbar-background); min-height: var(--jp-toolbar-micro-height); padding: 2px; - z-index: 1; - overflow-x: auto; + z-index: 8; + overflow-x: hidden; } /* Toolbar items */ @@ -11492,8 +11646,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ body.p-mod-override-cursor *, /* */ +/* */ +body.p-mod-override-cursor *, /* */ body.lm-mod-override-cursor * { cursor: inherit !important; } @@ -11553,6 +11707,80 @@ box-shadow: var(--jp-input-box-shadow); } +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- +| Variables +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- + +/*----------------------------------------------------------------------------- +| Styles +|----------------------------------------------------------------------------*/ + +.jp-Statusbar-ProgressCircle svg { + display: block; + margin: 0 auto; + width: 16px; + height: 24px; + align-self: normal; +} +.jp-Statusbar-ProgressCircle path { + fill: var(--jp-inverse-layout-color3); +} + +.jp-Statusbar-ProgressBar-progress-bar { + height: 10px; + width: 100px; + border: solid 0.25px var(--jp-brand-color2); + border-radius: 3px; + overflow: hidden; + align-self: center; +} +.jp-Statusbar-ProgressBar-progress-bar > div { + background-color: var(--jp-brand-color2); + background-image: linear-gradient( + -45deg, + rgba(255, 255, 255, 0.2) 25%, + transparent 25%, + transparent 50%, + rgba(255, 255, 255, 0.2) 50%, + rgba(255, 255, 255, 0.2) 75%, + transparent 75%, + transparent + ); + background-size: 40px 40px; + float: left; + width: 0%; + height: 100%; + font-size: 12px; + line-height: 14px; + color: #ffffff; + text-align: center; + animation: jp-Statusbar-ExecutionTime-progress-bar 2s linear infinite; +} + +.jp-Statusbar-ProgressBar-progress-bar p { + color: var(--jp-ui-font-color1); + font-family: var(--jp-ui-font-family); + font-size: var(--jp-ui-font-size1); + line-height: 10px; + width: 100px; +} + +@keyframes jp-Statusbar-ExecutionTime-progress-bar { + 0% { + background-position: 0 0; + } + 100% { + background-position: 40px 40px; + } +} + /* BASICS */ .CodeMirror { @@ -11977,6 +12205,10 @@ padding: 0 var(--jp-code-padding); } +.CodeMirror.cm-fat-cursor .cm-overlay.cm-searching { + opacity: 0.5; +} + .jp-CodeMirrorEditor[data-type='inline'] .CodeMirror-dialog { background-color: var(--jp-layout-color0); color: var(--jp-content-font-color1); @@ -12047,6 +12279,12 @@ color: var(--jp-search-unselected-match-color) !important; } +.cm-trailingspace { + background-image: url(); + background-position: center left; + background-repeat: repeat-x; +} + .CodeMirror-focused .CodeMirror-selected { background-color: var(--jp-editor-selected-focused-background); } @@ -12220,7 +12458,10 @@ opacity: 0; } -.jp-CodeMirrorEditor .remote-caret:hover > div { +/* Use `div[style]` as more specific selector on 3.4.x to reduce the impact of + * Chromium style invalidation strategy on performance when many divs are present. + */ +.jp-CodeMirrorEditor .remote-caret:hover > div[style] { opacity: 1; transition-delay: 0s; } @@ -12607,7 +12848,7 @@ border-spacing: 0; border: none; color: var(--jp-ui-font-color1); - font-size: 12px; + font-size: var(--jp-ui-font-size1); table-layout: fixed; margin-left: auto; margin-right: auto; @@ -12796,7 +13037,7 @@ border-radius: 3px; box-shadow: inset 0 -1px 0 rgba(0, 0, 0, 0.25); display: inline-block; - font-size: 0.8em; + font-size: var(--jp-ui-font-size0); line-height: 1em; padding: 0.2em 0.5em; } @@ -12851,8 +13092,10 @@ .jp-FileBrowser-toolbar.jp-Toolbar { border-bottom: none; height: auto; - margin: var(--jp-toolbar-header-margin); + margin: 8px 12px 0px 12px; + padding: 0px; box-shadow: none; + justify-content: flex-start; } .jp-BreadCrumbs { @@ -12884,43 +13127,33 @@ | Buttons |----------------------------------------------------------------------------*/ -.jp-FileBrowser-toolbar.jp-Toolbar { - padding: 0px; - margin: 8px 12px 0px 12px; -} - -.jp-FileBrowser-toolbar.jp-Toolbar { - justify-content: flex-start; -} - -.jp-FileBrowser-toolbar.jp-Toolbar .jp-Toolbar-item { +.jp-FileBrowser-toolbar > .jp-Toolbar-item { flex: 0 0 auto; padding-left: 0px; padding-right: 2px; } -.jp-FileBrowser-toolbar.jp-Toolbar .jp-ToolbarButtonComponent { +.jp-FileBrowser-toolbar > .jp-Toolbar-item .jp-ToolbarButtonComponent { width: 40px; } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent { +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher'] { width: 72px; background: var(--jp-brand-color1); } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent:focus-visible { - background-color: var(--jp-brand-color0); +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher']:hover, +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher']:focus-visible { + background-color: var(--jp-brand-color0) !important; } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher'] .jp-icon3 { - fill: white; + fill: var(--jp-layout-color1); } /*----------------------------------------------------------------------------- @@ -12957,7 +13190,8 @@ } .jp-DirListing:focus-visible { - border: 1px solid var(--jp-brand-color1); + outline: 1px solid var(--jp-brand-color1); + outline-offset: -2px; } .jp-DirListing-header { @@ -13088,6 +13322,8 @@ flex: 1 0 64px; outline: none; border: none; + color: var(--jp-ui-font-color1); + background-color: var(--jp-layout-color1); } .jp-DirListing-item.jp-mod-running .jp-DirListing-itemIcon:before { @@ -13116,15 +13352,6 @@ transform: translateX(-40%) translateY(-58%); } -.jp-DirListing-deadSpace { - flex: 1 1 auto; - margin: 0; - padding: 0; - list-style-type: none; - overflow: auto; - background-color: var(--jp-layout-color1); -} - .jp-Document { min-width: 120px; min-height: 120px; @@ -13284,6 +13511,12 @@ margin: 0; } +.jp-TrimmedOutputs a { + margin: 10px; + text-decoration: none; + cursor: pointer; +} + /* Hide the gutter in case of * - nested output areas (e.g. in the case of output widgets) * - mirrored output areas @@ -13292,6 +13525,12 @@ display: none; } +/* Hide empty lines in the output area, for instance due to cleared widgets */ +.jp-OutputArea-prompt:empty { + padding: 0; + border: 0; +} + /*----------------------------------------------------------------------------- | executeResult is added to any Output-result for the display of the object | returned by a cell @@ -13314,12 +13553,6 @@ | The Stdin output |----------------------------------------------------------------------------*/ -.jp-OutputArea-stdin { - line-height: var(--jp-code-line-height); - padding-top: var(--jp-code-padding); - display: flex; -} - .jp-Stdin-prompt { color: var(--jp-content-font-color0); padding-right: var(--jp-code-padding); @@ -13342,10 +13575,18 @@ flex: 0 0 70%; } +.jp-Stdin-input::placeholder { + opacity: 0; +} + .jp-Stdin-input:focus { box-shadow: none; } +.jp-Stdin-input:focus::placeholder { + opacity: 1; +} + /*----------------------------------------------------------------------------- | Output Area View |----------------------------------------------------------------------------*/ @@ -13582,11 +13823,28 @@ .jp-CodeCell.jp-mod-outputsScrolled .jp-Cell-outputArea { overflow-y: auto; - max-height: 200px; - box-shadow: inset 0 0 6px 2px rgba(0, 0, 0, 0.3); + max-height: 24em; margin-left: var(--jp-private-cell-scrolling-output-offset); } +.jp-CodeCell.jp-mod-outputsScrolled .jp-Cell-outputArea::after { + content: ' '; + box-shadow: inset 0 0 6px 2px rgb(0 0 0 / 30%); + width: 100%; + height: 100%; + position: sticky; + bottom: 0; + top: 0; + margin-top: -50%; + float: left; + display: block; + pointer-events: none; +} + +.jp-CodeCell.jp-mod-outputsScrolled .jp-OutputArea-child { + padding-top: 6px; +} + .jp-CodeCell.jp-mod-outputsScrolled .jp-OutputArea-prompt { flex: 0 0 calc( @@ -13614,6 +13872,79 @@ overflow: auto; } +/* collapseHeadingButton (show always if hiddenCellsButton is _not_ shown) */ +.jp-collapseHeadingButton { + display: none; + min-height: var(--jp-cell-collapser-min-height); + font-size: var(--jp-code-font-size); + position: absolute; + right: 0; + top: 0; + bottom: 0; + background-color: transparent; + background-size: 25px; + background-repeat: no-repeat; + background-position-x: center; + background-position-y: top; + background-image: var(--jp-icon-caret-down); + border: none; + cursor: pointer; +} + +.jp-collapseHeadingButton:hover { + background-color: var(--jp-layout-color2); +} + +.jp-collapseHeadingButton.jp-mod-collapsed { + background-image: var(--jp-icon-caret-right); +} + +:is(.jp-MarkdownCell:hover, .jp-mod-active) .jp-collapseHeadingButton { + display: flex; +} + +/* + set the container font size to match that of content + so that the nested collapse buttons have the right size +*/ +.jp-MarkdownCell .jp-InputPrompt { + font-size: var(--jp-content-font-size1); +} + +/* + Align collapseHeadingButton with cell top header + The font sizes are identical to the ones in packages/rendermime/style/base.css +*/ +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='1'] { + font-size: var(--jp-content-font-size5); + background-position-y: calc(0.3 * var(--jp-content-font-size5)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='2'] { + font-size: var(--jp-content-font-size4); + background-position-y: calc(0.3 * var(--jp-content-font-size4)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='3'] { + font-size: var(--jp-content-font-size3); + background-position-y: calc(0.3 * var(--jp-content-font-size3)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='4'] { + font-size: var(--jp-content-font-size2); + background-position-y: calc(0.3 * var(--jp-content-font-size2)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='5'] { + font-size: var(--jp-content-font-size1); + background-position-y: top; +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='6'] { + font-size: var(--jp-content-font-size0); + background-position-y: top; +} + .jp-showHiddenCellsButton { margin-left: calc(var(--jp-cell-prompt-width) + 2 * var(--jp-code-padding)); margin-top: var(--jp-code-padding); @@ -13626,19 +13957,6 @@ background-color: var(--jp-border-color2) !important; } -.jp-collapseHeadingButton { - display: none; -} - -.jp-MarkdownCell:hover .jp-collapseHeadingButton { - display: flex; - min-height: var(--jp-cell-collapser-min-height); - position: absolute; - right: 0; - top: 0; - bottom: 0; -} - /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Distributed under the terms of the Modified BSD License. @@ -13653,6 +13971,10 @@ | Variables |----------------------------------------------------------------------------*/ +:root { + --jp-notebook-toolbar-padding: 2px 5px 2px 2px; +} + /*----------------------------------------------------------------------------- /*----------------------------------------------------------------------------- @@ -13660,7 +13982,7 @@ |----------------------------------------------------------------------------*/ .jp-NotebookPanel-toolbar { - padding: 2px; + padding: var(--jp-notebook-toolbar-padding); } .jp-Toolbar-item.jp-Notebook-toolbarCellType .jp-select-wrapper.jp-mod-focused { @@ -13680,6 +14002,93 @@ top: 5px !important; } +.jp-Toolbar-responsive-popup { + position: absolute; + height: fit-content; + display: flex; + flex-direction: row; + flex-wrap: wrap; + justify-content: flex-end; + border-bottom: var(--jp-border-width) solid var(--jp-toolbar-border-color); + box-shadow: var(--jp-toolbar-box-shadow); + background: var(--jp-toolbar-background); + min-height: var(--jp-toolbar-micro-height); + padding: var(--jp-notebook-toolbar-padding); + z-index: 1; + right: 0px; + top: 0px; +} + +.jp-Toolbar > .jp-Toolbar-responsive-opener { + margin-left: auto; +} + +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- +| Variables +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- + +/*----------------------------------------------------------------------------- +| Styles +|----------------------------------------------------------------------------*/ + +.jp-Notebook-ExecutionIndicator { + position: relative; + display: inline-block; + height: 100%; + z-index: 9997; +} + +.jp-Notebook-ExecutionIndicator-tooltip { + visibility: hidden; + height: auto; + width: max-content; + width: -moz-max-content; + background-color: var(--jp-layout-color2); + color: var(--jp-ui-font-color1); + text-align: justify; + border-radius: 6px; + padding: 0 5px; + position: fixed; + display: table; +} + +.jp-Notebook-ExecutionIndicator-tooltip.up { + transform: translateX(-50%) translateY(-100%) translateY(-32px); +} + +.jp-Notebook-ExecutionIndicator-tooltip.down { + transform: translateX(calc(-100% + 16px)) translateY(5px); +} + +.jp-Notebook-ExecutionIndicator-tooltip.hidden { + display: none; +} + +.jp-Notebook-ExecutionIndicator:hover .jp-Notebook-ExecutionIndicator-tooltip { + visibility: visible; +} + +.jp-Notebook-ExecutionIndicator span { + font-size: var(--jp-ui-font-size1); + font-family: var(--jp-ui-font-family); + color: var(--jp-ui-font-color1); + line-height: 24px; + display: block; +} + +.jp-Notebook-ExecutionIndicator-progress-bar { + display: flex; + justify-content: center; + height: 100%; +} + /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Distributed under the terms of the Modified BSD License. @@ -13731,10 +14140,6 @@ contain: strict; } -.jp-Notebook-render * { - contain: none !important; -} - .jp-Notebook .jp-Cell { overflow: visible; } @@ -13974,6 +14379,86 @@ flex: 0 0 110px; } +/*----------------------------------------------------------------------------- +| Side-by-side Mode (.jp-mod-sideBySide) +|----------------------------------------------------------------------------*/ +:root { + --jp-side-by-side-output-size: 1fr; + --jp-side-by-side-resized-cell: var(--jp-side-by-side-output-size); +} + +.jp-mod-sideBySide.jp-Notebook .jp-Notebook-cell { + margin-top: 3em; + margin-bottom: 3em; + margin-left: 5%; + margin-right: 5%; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell { + display: grid; + grid-template-columns: minmax(0, 1fr) min-content minmax( + 0, + var(--jp-side-by-side-output-size) + ); + grid-template-rows: auto minmax(0, 1fr) auto; + grid-template-areas: + 'header header header' + 'input handle output' + 'footer footer footer'; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell.jp-mod-resizedCell { + grid-template-columns: minmax(0, 1fr) min-content minmax( + 0, + var(--jp-side-by-side-resized-cell) + ); +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellHeader { + grid-area: header; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-Cell-inputWrapper { + grid-area: input; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-Cell-outputWrapper { + /* overwrite the default margin (no vertical separation needed in side by side move */ + margin-top: 0; + grid-area: output; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellFooter { + grid-area: footer; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellResizeHandle { + grid-area: handle; + user-select: none; + display: block; + height: 100%; + cursor: ew-resize; + padding: 0 var(--jp-cell-padding); +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellResizeHandle::after { + content: ''; + display: block; + background: var(--jp-border-color2); + height: 100%; + width: 5px; +} + +.jp-mod-sideBySide.jp-Notebook + .jp-CodeCell.jp-mod-resizedCell + .jp-CellResizeHandle::after { + background: var(--jp-border-color0); +} + +.jp-CellResizeHandle { + display: none; +} + /*----------------------------------------------------------------------------- | Placeholder |----------------------------------------------------------------------------*/ @@ -14153,6 +14638,7 @@ --jp-border-color1: var(--md-grey-400); --jp-border-color2: var(--md-grey-300); --jp-border-color3: var(--md-grey-200); + --jp-inverse-border-color: var(--md-grey-600); --jp-border-radius: 2px; /* UI Fonts @@ -14389,7 +14875,7 @@ --jp-input-active-background: var(--jp-layout-color1); --jp-input-hover-background: var(--jp-layout-color1); --jp-input-background: var(--md-grey-100); - --jp-input-border-color: var(--jp-border-color1); + --jp-input-border-color: var(--jp-inverse-border-color); --jp-input-active-border-color: var(--jp-brand-color1); --jp-input-active-box-shadow-color: rgba(19, 124, 189, 0.3); @@ -14426,6 +14912,20 @@ --jp-mirror-editor-error-color: #f00; --jp-mirror-editor-hr-color: #999; + /* + RTC user specific colors. + These colors are used for the cursor, username in the editor, + and the icon of the user. + */ + + --jp-collaborator-color1: #ffad8e; + --jp-collaborator-color2: #dac83d; + --jp-collaborator-color3: #72dd76; + --jp-collaborator-color4: #00e4d0; + --jp-collaborator-color5: #45d4ff; + --jp-collaborator-color6: #e2b1ff; + --jp-collaborator-color7: #ff9de6; + /* Vega extension styles */ --jp-vega-background: white; @@ -14451,6 +14951,19 @@ --jp-icon-contrast-color1: var(--md-green-600); --jp-icon-contrast-color2: var(--md-pink-600); --jp-icon-contrast-color3: var(--md-blue-600); + + /* File or activity icons and switch semantic variables */ + --jp-jupyter-icon-color: #f37626; + --jp-notebook-icon-color: #f37626; + --jp-json-icon-color: var(--md-orange-700); + --jp-console-icon-background-color: var(--md-blue-700); + --jp-console-icon-color: white; + --jp-terminal-icon-background-color: var(--md-grey-800); + --jp-terminal-icon-color: var(--md-grey-200); + --jp-text-editor-icon-color: var(--md-grey-700); + --jp-inspector-icon-color: var(--md-grey-700); + --jp-switch-color: var(--md-grey-400); + --jp-switch-true-position-color: var(--md-orange-900); } @@ -14667,7 +15180,7 @@

    Core Library Setup -
    Loaded 730 standard library definitions
    +
    Loaded 750 standard library definitions
     
    @@ -14856,6 +15369,7 @@

    Helpers

    Example

    <?xml version="1.0"?>
    -<materialx version="1.38">
    +<materialx version="1.39">
       <!-- Node: <myprocedural> -->
       <nodedef name="ND_myprocedural_1_0_color3" node="myprocedural" nodegroup="texture2d" version="1.0" isdefaultversion="true">
         <output name="output_color3" type="color3" />
    @@ -15496,7 +16010,7 @@ 

    1.38.7 Definition Patching
    <?xml version="1.0"?>
    -<materialx version="1.38">
    +<materialx version="1.39">
       <!-- Definition: nodeGraph.getName() -->
       <nodedef name="ND_myprocedural_1_0_color3" node="myprocedural" nodegroup="texture2d" version="1.0" isdefaultversion="true" doc="Documentation for new definition: myProcedural" namespace="mynamespace">
         <input name="modulo" type="vector2" value="2.1, 1.1" />
    @@ -15667,7 +16181,7 @@ 

    2.1 Creating the Interface
    <?xml version="1.0"?>
    -<materialx version="1.38">
    +<materialx version="1.39">
       <!-- Definition of a simple node <myadd>, adding two colors. -->
       <nodedef name="ND_myadd_explicit_1_0_color3" version="1.0" nodegroup="math" node="myadd_explicit">
         <input name="in1" type="color3" value="1.0, 0.0, 0.0" />
    @@ -15818,7 +16332,7 @@ 

    2.1 Creating the Interface
    <?xml version="1.0"?>
    -<materialx version="1.38">
    +<materialx version="1.39">
       <!-- Definition of a simple node <myadd>, adding two colors. -->
       <nodedef name="ND_myadd_1_0_color3" version="1.0" nodegroup="math" node="myadd">
         <input name="in1" type="color3" value="1, 0, 0" />
    @@ -16029,7 +16543,7 @@ 

    2.3 Adding Source Code
    <?xml version="1.0"?>
    -<materialx version="1.38">
    +<materialx version="1.39">
       <!-- Definition of a simple node <myadd>, adding two colors. -->
       <nodedef name="ND_myadd_1_0_color3" version="1.0" nodegroup="math" node="myadd">
         <input name="in1" type="color3" value="1, 0, 0" />
    @@ -16154,7 +16668,7 @@ 

    2.3.2 Adding Source diff --git a/pymaterialx/mtlx_render_notebook.ipynb b/pymaterialx/mtlx_render_notebook.ipynb index d30eadf4..914a18ce 100644 --- a/pymaterialx/mtlx_render_notebook.ipynb +++ b/pymaterialx/mtlx_render_notebook.ipynb @@ -23,14 +23,14 @@ "Some example results are shown below to show: lit vs unlit, texture resource usage. (more to come)\n", "\n", "\n", - "\n", "\n", - "\n", "
    \n", - " \n", - " \n", + " \n", + " \n", + " \n", "
    \n", - " \n", - " \n", + " \n", + " \n", + " \n", "
    \n", "The top row of images are renders from different files. Left shows the sample Marble from the MaterialX distribution,\n", @@ -62,7 +62,7 @@ "name": "stdout", "output_type": "stream", "text": [ - "Loaded 746 standard library definitions\n" + "Loaded 750 standard library definitions\n" ] } ], @@ -808,8 +808,8 @@ "output_type": "stream", "text": [ "Setup lighting:\n", - "- Loaded radiance map: 1 x 1\n", - "- Loaded irradiance map: 1 x 1\n" + "- Loaded radiance map: 2048 x 1024\n", + "- Loaded irradiance map: 256 x 128\n" ] } ], @@ -974,7 +974,6 @@ " - Variable: displacementshader1. Value: (). Type: displacementshader, Path: \"\"\n", " - Variable: texcoord_vector2_index. Value: (0). Type: integer, Path: \"nodegraph1/texcoord_vector2/index\"\n", " - Variable: image_color3_file. Value: (checker.png). Type: filename, Path: \"nodegraph1/filename_port\"\n", - " - Unit:, ColorSpace:lin_rec709\n", " - Variable: image_color3_layer. Value: (). Type: string, Path: \"nodegraph1/image_color3/layer\"\n", " - Variable: image_color3_default. Value: (0.574572, 0.0112386, 0.0112386). Type: color3, Path: \"nodegraph1/image_color3/default\"\n", " - Variable: image_color3_uaddressmode. Value: (2). Type: integer, Path: \"nodegraph1/image_color3/uaddressmode\"\n", @@ -1064,8 +1063,8 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "\n", - "\n" + "\n", + "\n" ] }, { @@ -1227,7 +1226,7 @@ "data": { "text/markdown": [ "### .\\data\\unlit_image.png\n", - "" + "" ] }, "metadata": {}, @@ -1252,7 +1251,7 @@ " fileName.addExtension('png') \n", " glslRenderer.saveCapture(fileName, flipImage) \n", "\n", - " imageMD = '### %s\\n' % (fileName.asString(), fileName.asString())\n", + " imageMD = '### %s\\n' % (fileName.asString(), fileName.asString())\n", " display_markdown(imageMD, raw=True)" ] }, @@ -1280,7 +1279,6 @@ "text": [ "- Loaded image: \"checker.png\". Size: 1920 x 1920. Channel count: 3\n", " - Base Type: BaseType.UINT8 . Base Stride: 1\n", - " - Source color space: lin_rec709\n", "- Bind to program / shader port: image_color3_file\n" ] } diff --git a/pymaterialx/mtlx_render_notebook.py b/pymaterialx/mtlx_render_notebook.py index 40ceb4e3..ea424d9e 100644 --- a/pymaterialx/mtlx_render_notebook.py +++ b/pymaterialx/mtlx_render_notebook.py @@ -1,32 +1,48 @@ # %% [markdown] -# # Rendering +# # Rendering # -# This book will examine how to set up for rendering with MaterialX. It is not about how to write a renderer. +# This book will examine how to set up for rendering with MaterialX. It is not about how to write a renderer. # -# The topics covered include: +# The topics covered include: # -# 1. What is `renderable` and how to find `renderable` -# 2. Setting up graphs which are renderable. -# 2. Semantic differences between `roots` and handling "transparency". -# 3. Accessing inputs and binding resources. +# 1. Setting up renderers and using rendering utilities for geometry, images, and lighting. +# 1. Using graphs which are have `renderable` items. +# 2. Semantic differences between `roots` and handling "transparency". +# 3. Accessing inputs and binding resources. # -# This book is **WIP**. +# This book requires a MaterialX build or release which contains the following fixes for +# code generation.
    PR +# which is part of the 1.38.9 release. +# +# Some example results are shown below to show: lit vs unlit, texture resource usage. (more to come) +# +# +# +# +# +#
    +# +# +#
    +# +# +#
    +# The top row of images are renders from different files. Left shows the sample Marble from the MaterialX distribution, +# the middle is a modified version which uses an `unlit surface` shader, and the last is a graph which uses an external image resources modulated by an input color. Both the image and input color have a input color space specified. +# The bottom row is the sample nodegraph created in a nodegraph book, a stained glass shader, and a shader generated based on the logic in the Blender notebook. +# +# +# > Execution Note: The notebook can cause some loss to the current context for the renderer resulting in bad state. If this occurs then the notebook can be restarted, or the Python file can be run from the command line. In general, a renderer would not inject Python queries and Markdown code intermixed with rendering as is the case for this book. # %% [markdown] -# ## Setup +# ## 1. Setup # -# The following code will be used to perform basic setup, which includes creating a working document -# and loading in standard libraries. +# ### 1.1 Core Setup +# The following code will be used to perform basic setup, which includes creating a working document +# and loading in standard libraries. # %% import MaterialX as mx -import MaterialX.PyMaterialXGenShader as mx_gen_shader -import MaterialX.PyMaterialXGenGlsl as mx_gen_glsl -import MaterialX.PyMaterialXRender as mx_render -import MaterialX.PyMaterialXRenderGlsl as mx_render_glsl -import os, inspect, sys - -from mtlxutils import mxshadergen stdlib = mx.createDocument() searchPath = mx.getDefaultDataSearchPath() @@ -42,49 +58,158 @@ doc.importLibrary(stdlib) print('Loaded %s standard library definitions' % len(doc.getNodeDefs())) +# %% [markdown] +# ### 1.2 Code Generation and Rendering Modules +# For code generation `MaterialxGenShader` and any per target generation modules are loaded. +# In this example we load in the GLSL code generator. Note that we use the `mtlxutils` utility +# logic found in `mxshadergen` to handle code generation so that the GLSL generation module +# is not directly used. +# +# For rendering an example GLSL renderer (PyMaterialXRenderGlsl) is loaded. This is used for MaterialX Viewer and Node Editor +# as well as render test suite unit testing. This makes use of the base rendering module (`MaterialXRender`) which provides access to utilities such as geometry and image loaders as well as higher level utilities such as texture baking. +# +# Some additional utilities are added for display (`IPython`) and module discovery (`inspect`) # %% +import MaterialX.PyMaterialXGenShader as mx_gen_shader +import MaterialX.PyMaterialXGenGlsl as mx_gen_glsl +import MaterialX.PyMaterialXRender as mx_render +import MaterialX.PyMaterialXRenderGlsl as mx_render_glsl +from mtlxutils import mxshadergen -#inputFilename = './data/standard_surface_marble_solid.mtlx' -inputFilename = './data/unlit_marble_solid.mtlx' -#inputFilename = './data/unlit_image.mtlx' -try: - mx.readFromXmlFile(doc, inputFilename) - valid, msg = doc.validate() - if not valid: - raise mx.Exception('Document is invalid') +import inspect, sys +from IPython.display import display_markdown - print('Read in valid file "'"%s"'" for rendering.' % inputFilename) +# %% [markdown] +# ## 2. Sample Renderer Logic +# +# A class called `GlslRenderer` is added to encapsulate the logic required to set up the GLSL example renderer, +# set up resource handlers and a source code generator, create executable shader programs, and run the render pipeline. +# +# > A "TODO" has been added as a comment for C++ apis which are missing Python API wrappers. +# +# The main methods of interest are: +# +# 1. `initialize()` which calls the GLSL example renderer to initialize a device and framebuffer. Image and geometry handlers are also initialized. +# +# 2. `initializeImageHandler` which initialize image handlers and +# loaders such as the built in STB image loader. If built the `Open Image IO (OIIO)` loader (OiioImageLoader) can also be instantiated and used. Note that as this is a hardware renderer a specific GLSL handler is instantiated which allows for hardware texture resource management. The handler used by the renderer is set using `setImageHandler()`. +# +# 3. `initializeGeometryHandler` which uses a geometry handler to setup geometry loaders. By default an 'obj' file loader is created. +# A GLTF loader is available in C++, but at time of writing has no Python wrapper. The loader will be used to +# load in geometry for rendering. Note that the loaders will automatically create tangent and bitangents. This +# is important to note as all shading models (except for `unlit surface`) require these geometric streams. +# +# `loadGeometry()` calls into actual geometry loaders to load geometry files. +# +# 4. `initializeLights` which is used to set up a light handler which handles setting up directional lights specified in a MaterialX file as well as set up indirect lighting by specifying environment lighting files. These files are loaded in using the `ImageLoader.acquireImage()` interface. +# +# 5. `setupGenerator` which sets up the shader code generator for the desired target (`genglsl`). Note that the +# shader generation utility keeps GenContext for reuse. It is important to "register" where source code stored in files can be found by calling registerSourceCodeSearchPath() on the context. Generally this would be to the root of where +# the definition libraries are found but could also be elsewhere. The interface appends additional paths to search. +# +# 6. `generateShader` sets up some basic generation options such as whether to use lighting or not and a hint if the shader is transparent. The utility methods: elementRequiresShading() and isTransparentSurface() perform this introspection respectively. They are both found as utilities within the `MaterialXGenShader` module. +# +# > Note that an application integration should provide the following additional information for the renderer's working color space as as well as the geometry scene real-world units. This is because only an integration can provide this information. +# See the reference Glossary for units and color space transform information. +# +# 7. `createProgram` is used to create a GLSL program from a Shader which is created via code generation. This is just one example of source code usage. +# +# 8. `render` is used to render a frame. As the example renderer's pipline is a limited one used for unit testing, it +# will perform all of the required program setup and input bindings based on inspecting the program itself, making +# used of the specified image, geometry and light handlers to (in this case) set up hardware resources for binding. -except mx.ExceptionFileMissing as err: - print('File %s could not be loaded: "' % inputFilename, err, '"') -except mx.Exception as err: - print('File %s fail to load properly: "' % inputFilename, err, '"') +# %% [markdown] +# ### 2.1 Handling Real-World Units and Color Management +# +# Thought not strictly necessary, it is useful to check for available unit and color management support. +# +# 1. Units: The `buildUnitDict()` will scan for available unit types and unit identifiers. For example `distance` units are supported with unit identifiers such as `meter`, `inch`, and `foot` conversions being supported. +# 2. Color Transforms: The `buildColorTransformDict()` will scan for available `colorspace` transforms. Note that only transforms to a (target) linear color space (`lin_rec709`) is currently supported. + +# %% +def buildUnitDict(doc): + ''' + Sample code to examine unit types and unit name information + ''' + unitdict = {} + + for ud in doc.getUnitDefs(): + unittype = ud.getAttribute('unittype') + unitinfo = {} + for unit in ud.getChildren(): + unitinfo[unit.getName()] = unit.getAttribute('scale') + + unitdict[unittype] = unitinfo + return unitdict + +def buildColorTransformDict(doc): + colordict = {} + targetdict = {} + for cmnode in doc.getNodeDefs(): + if cmnode.getNodeGroup() == 'colortransform': + name = cmnode.getName() + name = name.removeprefix('ND_') + namesplit = name.split('_to_') + type = 'color3' + if 'color4' in namesplit[1]: + continue + else: + namesplit[1] = namesplit[1].removesuffix('_color3') + + sourceSpace = namesplit[0] + targetSpace = namesplit[1] + + if sourceSpace in colordict: + sourceItem = colordict[sourceSpace] + sourceItem.append(targetSpace) + else: + colordict[sourceSpace] = [targetSpace] + + if targetSpace in targetdict: + taregetItem = targetdict[targetSpace] + taregetItem.append(sourceSpace) + else: + targetdict[targetSpace] = [sourceSpace] + + + return colordict, targetdict + +# Build unit dictionary +unitdict = buildUnitDict(doc) +for unittype in unitdict: + print('Unit Type: %s' % unittype) + units = unitdict[unittype] + for unit in units: + print(' Unit: %s. Scale Factor: %s' % (unit, units[unit])) + +print('') + +# Build colorspace dictionary +stdict, tsdict = buildColorTransformDict(doc) +print('Supported Source to Target Transforms:') +for sourceSpace in stdict: + print(' %s --> %s supported' % (sourceSpace, ', '.join(stdict[sourceSpace]))) +print('Supported Target From Source Transforms:') +for targetSpace in tsdict: + print(' %s <-- %s supported' % (targetSpace, ', '.join(tsdict[targetSpace]))) # %% -#import PIL -#help(Image) - -def imageLoaderTest(): - return False - #if imageLoader: - # print('created image loader') - # newImage = imageLoader.loadImage('./images/Houdini_Node_Editor.png') - # print('load image:', newImage.getWidth(), ',', newImage.getHeight(), ',', newImage.getChannelCount()) - # print('- bt:', newImage.getBaseType(), '. bs', newImage.getBaseStride() ) - # imageLoader.saveImage('./test.png', newImage, False) - - #self.image = mx_render.Image.create(512, 512, 4, mx_render.BaseType.UINT8 ) - #if self.image: - # self.image.createResourceBuffer() - # color = mx.Color4(1, 1, 0, 1) - # self.image.setUniformColor(color) class GlslRenderer(): + ''' + Wrapper for GLSL sample renderer. + + Handles setup of image, geometry and light handlers as well as GLSL code and + program generation. + + Calls into sample renderer to render and capture images as desired. + ''' def __init__(self): # Renderer + self.renderSize = [512, 512] self.renderer = None # Code Generator @@ -93,12 +218,19 @@ def __init__(self): self.activeShaderErrors = '' self.sourceCode = {} - # Image capture + # Image Handling self.capturedImage = None + self.haveOIIOImageHandler = False + mxrenderMembers = inspect.getmembers(sys.modules['MaterialX.PyMaterialXRender']) + for className, classObject in mxrenderMembers: + if className == 'OiioImageLoader' and inspect.isclass(classObject): + self.haveOIIOImageHandler = True + break # Geometry loading self.haveCGLTFLoader = False - for className, classObject in inspect.getmembers(sys.modules['MaterialX.PyMaterialXRender']): + # Note: TODO: Test for existence of GLTF loader in Python module. This does not exist in a release currently. + for className, classObject in mxrenderMembers: if className == 'CgltfLoader' and inspect.isclass(classObject): self.haveCGLTFLoader = True break @@ -106,9 +238,19 @@ def __init__(self): # Light setup self.lightHandler = None + # Units dictionary + self.unitDict = None + + # Colorspace dictionaries + self.sourceColorDict = None + self.targetColorDict = None + def getRenderer(self): return self.renderer + def getDefaultRenderSize(self): + return self.renderSize + def getCodeGenerator(self): return self.mxgen @@ -123,11 +265,25 @@ def getSourceCode(self): def haveGLTFLoader(self): return self.haveCGLTFLoader + + def haveOIIOLoader(self): + return self.haveOIIOImageHandler def getLightHandler(self): return self.lightHandler - def initialize(self, w=512, h=512, bufferFormat=mx_render.BaseType.UINT8): + def initialize(self, w=0, h=0, bufferFormat=mx_render.BaseType.UINT8): + ''' + Setup sample renderer with a given frame buffer size. + Initialize image and geometry handlers. + ''' + if w == 0 and h == 0: + w = self.renderSize[0] + h = self.renderSize[1] + if w < 4: + w = 4 + if h < 4: + h = 4 self.renderer = mx_render_glsl.GlslRenderer.create(w, h, bufferFormat) if self.renderer: self.renderer.initialize() @@ -135,23 +291,38 @@ def initialize(self, w=512, h=512, bufferFormat=mx_render.BaseType.UINT8): self.initializeGeometryHandler() def resize(self, w, h): + ''' + Resize frame buffer. + Clears any cached captured image. + ''' if not self.renderer: return False self.renderer.setSize(w, h) self.capturedImage = None - def initializeImageHandler(self): + def initializeImageHandler(self): + ''' + Initialize image handler. + ''' + if self.renderer.getImageHandler(): + return + + # TODO: Missing fom the Python API for createImageHandler() + #imageHandler = renderer.createImageHandler() imageLoader = mx_render.StbImageLoader.create() imageHandler = mx_render_glsl.GLTextureHandler.create(imageLoader) - # Missing fom the Python API ! - #imageHandler = renderer.createImageHandler() + # Add OIIO handler if it exists + if self.haveOIIOImageHandler: + imageHandler.addLoader(mx_render.OIIOHandler.create()) + if imageHandler: - imageHandler.setSearchPath(searchPath) - print('Initialize image handler on renderer') + imageSearchPath = mx.FileSearchPath() + imageSearchPath.append(mx.FilePath('./data')) + imageHandler.setSearchPath(imageSearchPath) self.renderer.setImageHandler(imageHandler) - def initializeGeometryHandler(self): + def initializeGeometryHandler(self): # renderer has a geometry handler created by # default so not need to call: mx_render.GeometryHandler.create() geometryHandler = self.renderer.getGeometryHandler() @@ -171,6 +342,13 @@ def getGeometyHandler(self): return self.renderer.getGeometryHandler() def initializeLights(self, doc, enableDirectLighting, radianceIBLPath, irradianceIBLPath, enableReferenceQuality): + if self.lightHandler: + return + + # Ensure image handler is initialized + self.initializeImageHandler() + + # Create a light handler self.lightHandler = mx_render.LightHandler.create() # Scan for lights @@ -196,10 +374,22 @@ def initializeLights(self, doc, enableDirectLighting, radianceIBLPath, irradianc #self.lightHandler.setRefractionTwoSided(True) def captureImage(self): - + ''' + Capture the framebuffer contents to an image + ''' self.capturedImage = self.renderer.captureImage(self.capturedImage) + def clearCaptureImage(self): + ''' + Clear out any captured image + ''' + self.captureImage = None + def saveCapture(self, filePath, verticalFlip=True): + ''' + Save captured image to a file. + Vertical flip image as needed. + ''' if not self.capturedImage: self.captureImage() @@ -214,23 +404,99 @@ def getCapturedImage(self): return self.capturedImage def setupGenerator(self, doc, stdlib, searchPath): - # Setup generation + ''' + Setup code generation. Returns the generator instantiated. + Note: It is important to set up the source code path so that + file implementations can be found. + ''' self.mxgen = mxshadergen.MtlxShaderGen(stdlib) self.mxgen.setup() - # Set generator and generator options + # Check generator and generator options + mxgenerator = None mxcontext = self.mxgen.setGeneratorForTarget('genglsl') - mxgenerator = mxcontext.getShaderGenerator() + if mxcontext: + mxgenerator = mxcontext.getShaderGenerator() # Set source code path self.mxgen.registerSourceCodeSearchPath(searchPath) + return mxgenerator + def findRenderableElements(self, doc): # Generate shader for a given node self.nodes = self.mxgen.findRenderableElements(doc) return self.nodes - def generateShader(self, node): + def buildUnitDict(self, doc): + ''' + Create real-world units dictionary for target unit checking + ''' + if self.unitDict: + return + + self.unitDict = {} + + for ud in doc.getUnitDefs(): + unittype = ud.getAttribute('unittype') + unitinfo = {} + for unit in ud.getChildren(): + unitinfo[unit.getName()] = unit.getAttribute('scale') + + self.unitDict[unittype] = unitinfo + return self.unitDict + + def buildColorTransformDict(self,doc): + ''' + Build a pair of dictionaries to test for supported colorspace transforms. + One is from source color space to target, and the other is to a target from source. + ''' + if self.sourceColorDict: + return + + colordict = {} + targetdict = {} + for cmnode in doc.getNodeDefs(): + if cmnode.getNodeGroup() == 'colortransform': + name = cmnode.getName() + name = name.removeprefix('ND_') + namesplit = name.split('_to_') + type = 'color3' + if 'color4' in namesplit[1]: + continue + else: + namesplit[1] = namesplit[1].removesuffix('_color3') + + sourceSpace = namesplit[0] + targetSpace = namesplit[1] + + if sourceSpace in colordict: + sourceItem = colordict[sourceSpace] + sourceItem.append(targetSpace) + else: + colordict[sourceSpace] = [targetSpace] + + if targetSpace in targetdict: + taregetItem = targetdict[targetSpace] + taregetItem.append(sourceSpace) + else: + targetdict[targetSpace] = [sourceSpace] + + self.sourceColorDict = colordict + self.targetColorDict = targetdict + return colordict, targetdict + + def getColorTransformDict(self): + return self.sourceColorDict, self.targetColorDict + + def generateShader(self, node, targetColorSpaceOverride='lin_rec709', targetDistanceUnit='meter'): + ''' + Generate new GLSL shader. + - Inspects node to check if it requires lighting and / or is transparent. + - Sets target colorspace and real-world units + - Generates code and caches it + - Caches the "active" Shader node + ''' self.activeShader = None if not node: return None @@ -245,7 +511,25 @@ def generateShader(self, node): else: mxoptions.hwMaxActiveLightSources = 0 mxoptions.hwTransparency = mx_gen_shader.isTransparentSurface(node, mxgenerator.getTarget()) - mxoptions.targetColorSpaceOverride = 'lin_rec709' + + # Check support of units and working color space + doc = node.getDocument() + if doc: + self.buildUnitDict(doc) + units = self.unitDict['distance'] + if targetDistanceUnit not in units: + targetDistanceUnit = 'meter' + + sdict, tdict = self.buildColorTransformDict(doc) + if tdict: + if targetColorSpaceOverride not in tdict: + targetColorSpaceOverride = 'lin_rec709' + else: + targetDistanceUnit = 'meter' + targetColorSpaceOverride = 'lin_rec709' + + mxoptions.targetDistanceUnit = targetDistanceUnit + mxoptions.targetColorSpaceOverride = targetColorSpaceOverride self.activeShader, self.activeShaderErrors = self.mxgen.generateShader(node) if self.activeShader: @@ -255,16 +539,13 @@ def generateShader(self, node): return self.activeShader def createProgram(self): + ''' + Create a GLSL program from the active shader node and validates it's inputs. + Note: A light handler **must** be set to for validation to work properly. + ''' if not self.activeShader: return False - #psStage = shader.getStage('pixel') #mx_gen_shader.Stage.PIXEL - #block = psStage.getUniformBlock('PublicUniforms') #mx_gen_shader.HW.PUBLIC_UNIFORMS) - # getVariableOrder() is not defined in the Python API - #for uniform in block: #.getVariableOrder(): - # uniformVariable = uniform.getVariable() - # print(uniformVariable) - self.renderer.setLightHandler(self.lightHandler) self.renderer.createProgram(self.activeShader) #self.renderer.validateInputs() @@ -280,31 +561,88 @@ def getProgram(self): return self.renderer.getProgram() def render(self): + ''' + Render a frame. + - Note: LookupError's are returned if any failure occurs. + - Status and and any errors are returned. + ''' if not self.renderer: return False, 'No renderer' # Render try: - print('Render') self.renderer.render() - print('Render done') except LookupError as err: return False, err return True, '' - # %% [markdown] -# ## Rendering Setup +# ## 3. Rendering Setup +# +# This utility class can now be used for rendering with specified output frame parameters. +# # %% glslRenderer = GlslRenderer() -glslRenderer.initialize(512, 512, mx_render.BaseType.UINT8) +renderSize = glslRenderer.getDefaultRenderSize() +glslRenderer.initialize(renderSize[0], renderSize[1], mx_render.BaseType.UINT8) +print('Initialized renderer') +print('- Have OIIO loader support: %s' % glslRenderer.haveOIIOLoader()) +print('- Have GLTF loader support: %s' % glslRenderer.haveGLTFLoader()) + # This is not exposed #clearColor = mx.Color3(1.0, 1.0, 1.0) #glslRenderer.setScreenColor(clearColor) -glslRenderer.initializeLights(doc, False, +# %% [markdown] +# In the sample code we set up to: +# +# 1. Use a sphere as the scene geometry + +# %% + +geometryHandler = glslRenderer.getGeometyHandler() +if geometryHandler: + print('- Initialized geometry loader:') + + desiredGeometry = 'sphere' + geometryFile = './data/sphere.obj' + if desiredGeometry == 'shaderball': + if glslRenderer.haveGLTFLoader(): + geometryFile = './data/shaderball.glb' + + glslRenderer.loadGeometry(geometryFile) + for mesh in geometryHandler.getMeshes(): + print(' - Loaded Mesh: "%s"' % mesh.getName()) + +# %% [markdown] +# 2. Set up the input file to render + +# %% +inputFilename = './data/standard_surface_marble_solid.mtlx' +inputFilename = './data/unlit_marble_solid.mtlx' +inputFilename = './data/unlit_image.mtlx' +try: + mx.readFromXmlFile(doc, inputFilename) + valid, msg = doc.validate() + if not valid: + raise mx.Exception('Document is invalid') + + print('Read in valid file "'"%s"'" for rendering.' % inputFilename) + +except mx.ExceptionFileMissing as err: + print('File %s could not be loaded: "' % inputFilename, err, '"') +except mx.Exception as err: + print('File %s fail to load properly: "' % inputFilename, err, '"') + +# %% [markdown] +# 3. Set up the lighting. A document which specifies +# the lighting is required. This could be in the working document or a separately loaded in document. +# Here only indirect lighting is setup. + +# %% +glslRenderer.initializeLights(None, False, './data/lights/san_giuseppe_bridge.hdr', './data/lights/irradiance/san_giuseppe_bridge.hdr', False) @@ -316,72 +654,88 @@ def render(self): print('- Loaded radiance map: %d x %d' % (radMap.getWidth(), radMap.getHeight())) print('- Loaded irradiance map: %d x %d' % (irradMap.getWidth(), irradMap.getHeight())) -geometryFile = './data/sphere.obj' -#if glslRenderer.haveGLTFLoader(): -# geometryFile = './data/shaderball.glb' -glslRenderer.loadGeometry(geometryFile) -geometryHandler = glslRenderer.getGeometyHandler() -if geometryHandler: - print('Setup geometry:') - for mesh in geometryHandler.getMeshes(): - print('- Initialze geometry mesh: "%s"' % mesh.getName()) +# %% [markdown] +# 4. Set up source code generation for `GLSL`. This requires a working document to initialize +# based on the working document and a definition document (which may be the same). Additionally +# a source code search path needs to specified. The default `libraries` path is used as the +# search path. If source code resides elsewhere then the search path can be extended as needed +# but at a minimum the `libraries` path must be included to use the standard definition library. -glslRenderer.setupGenerator(doc, stdlib, searchPath) +# %% +sourceCodeSearchPath = searchPath +glslRenderer.setupGenerator(doc, stdlib, sourceCodeSearchPath) context = glslRenderer.getCodeGenerator().getContext() if context: generator = context.getShaderGenerator() if generator: - print('- Iniitialize generator for target:', generator.getTarget()) + print('- Iniitialize generator for target: %s.\n - Source path: %s' % + (generator.getTarget(), sourceCodeSearchPath.asString())) +# %% [markdown] +# 5. Chose a node to render with and generate the `Shader` # %% # Set up additional options for generation context = glslRenderer.getCodeGenerator().getContext() genOptions = context.getOptions() -# TODO: This has not been exposed in Python +genOptions.emitColorTransforms = True # This is True by default +genOptions.fileTextureVerticalFlip = True +# TODO: This and a number of other options are not been exposed in the Python API #genOptions.addUpstreamDependencies = True # Find a renderable and generate the shader for it nodes = glslRenderer.findRenderableElements(doc) shader = None printSource = True + +# Set up overrides for color space and units. Color space may come from the document, +# but units are a property of the application. +targetColorSpaceOverride = 'lin_rec709' +docColorSpace = doc.getColorSpace() +targetDistanceUnit = 'centimeter' if nodes: - shader = glslRenderer.generateShader(nodes[0]) + shader = glslRenderer.generateShader(nodes[0], targetColorSpaceOverride, targetDistanceUnit) if shader: - print('Generate shader for node: %s' % nodes[0].getNamePath()) - + print('Generate shader for node: "%s"\n- Is Transparent: %s. V-Flip textures: %d.\n- Emit Color Xforms: %d. Default input colorspace: "%s".\n- Target Color space: "%s". Scene Units: "%s"' % + (nodes[0].getNamePath(), + genOptions.hwTransparency, + genOptions.fileTextureVerticalFlip, + genOptions.emitColorTransforms, + docColorSpace, + genOptions.targetColorSpaceOverride, + genOptions.targetDistanceUnit)) # %% [markdown] -# ## Shader Stages / Uniform Blocks / Shader Ports -# -# * For languages like OSL and MSL there is only one shader which is the `pixel` shader -- and thus one stage. -# * For hardware shading languages like GLSL, MSL, Vulkan there can be more than 1 stage. Currently the -# defaults code generators only emit a `vertex` and `pixel` stage. -# * Within each stage the list of uniforms can be extracted. These are organized into "blocks". User facing uniforms will be organiz "public" blocks, and internal ones in "private" blocks. -# * Lighting uniforms are exposed as a "lighting" block. For example environment lighting can be bound there. -# * Within each block the each uniform is represented as a ShaderPort -# -# ### Shader Ports -# * shader ports will provide the exact name of the uniform in the shader via `getVariable()` interface -# * they will also provide the value after all "resolves" have been performed. Note that this can differ -# from the original value stored on a node Input. For example tokens may be resolved on geometric attribute and -# filenames. -# * It is possible to "pre-resolve" values as needed. For example MDL has a special resolver to handle file names. -# It makes use of the flattenFilenames() utility before performing additional resolves -# for `Omniverse` compatibility -# -# * To find correspondence back to the original MaterialX input the path may be found using `getPath()`, and then -# calling `Document.getDescendent()` with the path as the interface argument. An `Input` will be returned if found. -# * Note that an input to an graph's interior node may be returned as the port path. In this case, -# the interface input should be found to provide the correct upstream corresponding path. The method -# `getPortPath()` shows this logic. -# * Note that a Shader may be generated at a given time, and if the MaterialX graph changes then the Shader paths -# may reference inputs which may no longer exist. It is up the integration to regenerate shaders on any "topological" -# changes. -# -# In the sample funciton `debugStages()`, each stage is iterated over. For each stage the list of uniform blocks is extracted. -# Then for each block the list of shader ports is printed out. Note that "private" vertex stage uniforms involve things like model / view transforms, there are private and pixel stage uniforms as well as "light data" uniforms for environment map binding. +# ## 4. Shader Stages / Uniform Blocks / Shader Ports +# +# * For languages like OSL and MSL there is only one shader which is the `pixel` shader -- and thus one stage. +# * For hardware shading languages like GLSL, MSL, Vulkan there can be more than 1 stage. Currently the +# defaults code generators only emit a `vertex` and `pixel` stage. +# * Within each stage the list of uniforms can be extracted. These are organized into "blocks". User facing uniforms will be organiz "public" blocks, and internal ones in "private" blocks. +# * Lighting uniforms are exposed as a "lighting" block. For example environment lighting can be bound there. +# * Within each block the each uniform is represented as a ShaderPort +# +# ### 4.1 Shader Ports +# * shader ports will provide the exact name of the uniform in the shader via `getVariable()` interface +# * they will also provide the value after all "resolves" have been performed. Note that this can differ +# from the original value stored on a node Input. For example tokens may be resolved on geometric attribute and +# filenames. +# * It is possible to "pre-resolve" values as needed. For example MDL has a special resolver to handle file names. +# It makes use of the flattenFilenames() utility before performing additional resolves +# for `Omniverse` compatibility +# +# * To find correspondence back to the original MaterialX input the path may be found using `getPath()`, and then +# calling `Document.getDescendent()` with the path as the interface argument. An `Input` will be returned if found. +# * Note that an input to an graph's interior node may be returned as the port path. In this case, +# the interface input should be found to provide the correct upstream corresponding path. The method +# `getPortPath()` shows this logic. +# * Note that a Shader may be generated at a given time, and if the MaterialX graph changes then the Shader paths +# may reference inputs which may no longer exist. It is up the integration to regenerate shaders on any "topological" +# changes. +# +# In the sample function `debugStages()`, each stage is iterated over. For each stage the list of uniform blocks is extracted. +# Then for each block the list of shader ports is printed out. Note that "private" vertex stage uniforms involve things like model / view transforms, there are private and pixel stage uniforms as well as "light data" uniforms for environment map binding. # %% def getPortPath(inputPath, doc): @@ -448,35 +802,34 @@ def debugStages(shader, doc, filter='Public'): # Examine public uniforms first debugStages(shader, doc, 'Public') - # %% [markdown] -# -# +# +# # # %% [markdown] # -# In the output, you will note that: -# * the shader variable `multiply_color3_in1` corresponds to an input: `nodegraph1/multiply_color3/in1` maps to the interface input `nodegraph1/color3_port`. -# * the shader variable `image_color3_file` corresponds to an interior input: `nodegraph1/image_color3/file` is maps to the interface input `nodegraph1/filename_port`. +# In the output, you will note that: +# * the shader variable `multiply_color3_in1` corresponds to an input: `nodegraph1/multiply_color3/in1` maps to the interface input `nodegraph1/color3_port`. +# * the shader variable `image_color3_file` corresponds to an interior input: `nodegraph1/image_color3/file` is maps to the interface input `nodegraph1/filename_port`. # -# Then updating the interface ports, the appropriate shader uniform needs ot be used. +# Then updating the interface ports, the appropriate shader uniform needs ot be used. # -# The file image input 'nodegraph1/filename_port' is an interface input which has a `colorspace` transform specified. +# The file image input 'nodegraph1/filename_port' is an interface input which has a `colorspace` transform specified. # %% [markdown] -# ### Building UI +# ### 3.2 Building UI # -# MaterialXRender has the utility `createUIPropertyGroups()` which performs parsing on a block to build -# UI for the MaterialX Viewer and Graph Editor. +# MaterialXRender has the utility `createUIPropertyGroups()` which performs parsing on a block to build +# UI for the MaterialX Viewer and Graph Editor. # -# It goes through the interface mapping step as well as extracting desired information from the MaterialX `Inputs` and `ShaderPort` inputs. +# It goes through the interface mapping step as well as extracting desired information from the MaterialX `Inputs` and `ShaderPort` inputs. # %% [markdown] -# ### Source Code +# ### 3.3 Examining Source Code # -# The uniform information can be compared against the produced source code. -# In the sample code below we scan the source for "uniforms" and prints them out. +# The uniform information can be compared against the produced source code. +# In the sample code below we scan the source for "uniforms" and prints them out. # %% if printSource: @@ -490,28 +843,6 @@ def debugStages(shader, doc, filter='Public'): print(' ', l) - -# %% [markdown] -# -# ## Note on Topological Changes -# * In previous version there was a dirty/notification system which could be hooked into when a document changed. As this no longer exists, it is up the integration to keep track of relevant changes. -# * Value changes can require rebinding of resources such as geometry and images as well as scalar values. -# * Topological changes can include: -# * changes between node port connections, -# * changes in value on conditional nodes, -# * changes to attributes which extract channels from a tuple, -# * changes to values which affect transparency -# * changes which affect "uniform blocks", if the blocks organization / layout changes. (e.g. `Vulkan` creates uniform blocks) -# For this it would be **very useful if there was a way to specify a hint that a value change means a topological change.** -# * Value changes only require rebinding to an existing shader while topological changes require a shader to be rebuilt. -# -# - -# %% [markdown] -# ## Binding Inputs -# -# **WIP** - # %% createdProgram = False if shader: @@ -540,12 +871,10 @@ def debugStages(shader, doc, filter='Public'): - # %% [markdown] -# ## Rendering and Capturing Images +# ## 4. Rendering and Capturing Images # %% - runRender = True if createdProgram and runRender: rendered, renderErrors = glslRenderer.render() @@ -558,14 +887,99 @@ def debugStages(shader, doc, filter='Public'): capturedImage = glslRenderer.getCapturedImage() if capturedImage: flipImage = True - fileName = './data/render_notebook_capture.png' - print('Saved rendering to: %s' % fileName) + fileName = mx.FilePath(inputFilename) + fileName.removeExtension() + fileName.addExtension('png') glslRenderer.saveCapture(fileName, flipImage) + imageMD = '### %s\n' % (fileName.asString(), fileName.asString()) + display_markdown(imageMD, raw=True) + +# %% [markdown] +# ## 5. Binding Inputs +# +# This section goes over binding of scalars and images. +# +# Note that currently we don't do the actual process of binding but just find the shader uniform, load in an image +# and then find the target shader variable to update. + +# %% +imagesToBind = [] +variablesToBind = [] +nodePathsToBind = [] + +# Loading in images +imageHandler = glslRenderer.getImageHandler() + +# Get the program +program = glslRenderer.getProgram() + +# Scan for input filenames, create the image and bind it to the program +stage = shader.getStage('pixel') if shader else None +if stage: + block = stage.getUniformBlock('PublicUniforms') + + for shaderPort in block: + value = shaderPort.getValue() + if not value: + continue + + type = shaderPort.getType().getName() + if type != 'filename': + continue + + variable = shaderPort.getVariable() + value = shaderPort.getValue().getValueString() if shaderPort.getValue() else '' + + origPath = shaderPort.getPath() + path, interfaceInput = getPortPath(shaderPort.getPath(), doc) + + unit = shaderPort.getUnit() + if interfaceInput: + colorspace = interfaceInput.getColorSpace() + else: + colorspace = shaderPort.getColorSpace() + + imagesToBind.append(value) + variablesToBind.append(variable) + nodePathsToBind.append(path) + + newImage = imageHandler.acquireImage(value) + if newImage: + print('- Loaded image: "%s". Size: %d x %d. Channel count: %d' % + (value, newImage.getWidth(), newImage.getHeight(), newImage.getChannelCount())) + print(' - Base Type:', newImage.getBaseType(), '. Base Stride:', newImage.getBaseStride() ) + if colorspace: + print(' - Source color space: %s' % colorspace) + elif unit: + print(' - Source unit: %s' % unit) + + # Find the appropriate port on the program + if program: + uniforms = program.getUniformsList() + if variable in uniforms: + print('- Bind to program / shader port:', variable) + # %% [markdown] -# Rendering Images +# ## 6. Handling Topological Changes +# +# In earlier versions of MaterialX there was a "dirty/notification" system which could be hooked into when a document changed. As this no longer exists, it is up the integration to keep track of relevant changes. +# +# Value changes can require rebinding of resources such as geometry and images as well as scalar values. +# +# Topological changes can occur due to: +# * changes between node port connections, +# * changes in value on conditional nodes, +# * changes in enumerations which result in conditional branching, +# * changes to attributes which extract channels from a tuple, +# * changes to values which affect transparency +# * changes which affect "uniform blocks", if the blocks organization / layout changes. (e.g. `Vulkan` creates uniform blocks) +# For this it would be **very useful if there was a way to specify a hint that a value change means a topological change.** +# +# Value changes only require rebinding to an existing shader while topological changes require a shader to be rebuilt. +# +# # -# diff --git a/pymaterialx/mtlx_render_notebook2.py b/pymaterialx/mtlx_render_notebook2.py index f0d6f7c3..c8fd5136 100644 --- a/pymaterialx/mtlx_render_notebook2.py +++ b/pymaterialx/mtlx_render_notebook2.py @@ -465,8 +465,8 @@ def debugStages(shader, doc, filter='Public'): debugStages(shader, doc, 'Public') # %% [markdown] -# -# +# +# # # %% [markdown] @@ -575,7 +575,7 @@ def debugStages(shader, doc, filter='Public'): fileName.addExtension('.png') glslRenderer.saveCapture(fileName, flipImage) - imageMD = '### %s\n' % (fileName.asString(), fileName.asString()) + imageMD = '### %s\n' % (fileName.asString(), fileName.asString()) display_markdown(imageMD, raw=True) diff --git a/pymaterialx/mtlx_render_notebook_working.html b/pymaterialx/mtlx_render_notebook_working.html index f3a51007..656a1249 100644 --- a/pymaterialx/mtlx_render_notebook_working.html +++ b/pymaterialx/mtlx_render_notebook_working.html @@ -15909,8 +15909,8 @@

    Shader Ports&#

    -

    - -

    @@ -15683,11 +15769,11 @@

    Implementation Targets -
    Found target identifier: essl with 581 source implementations.
    -Found target identifier: genglsl with 581 source implementations.
    -Found target identifier: genmdl with 589 source implementations.
    -Found target identifier: genmsl with 580 source implementations.
    -Found target identifier: genosl with 578 source implementations.
    +
    Found target identifier: essl with 540 source implementations.
    +Found target identifier: genglsl with 540 source implementations.
    +Found target identifier: genmdl with 549 source implementations.
    +Found target identifier: genmsl with 539 source implementations.
    +Found target identifier: genosl with 537 source implementations.
     
    @@ -15712,7 +15798,7 @@

    Code GeneratorsShaderGenerator

    -

    +

    Note that Vulkan has the same target as genglsl, but has it's own generator. Also that the Metal generator will only show up in the Mac build of documentation.

    Integrations are free to create custom generators. Some notable existing generators include those used to support USD HDStorm, VEX, and Arnold OSL.

    @@ -15745,8 +15831,8 @@

    Code Generatorsgeneratordict[gen.getTarget()] = gen # Choose generator to use based on target identifier -language = 'osl' -target = 'genosl' +language = 'glsl' +target = 'genglsl' if language == 'osl': target = 'genosl' elif language == 'mdl': @@ -15784,7 +15870,7 @@

    Code Generators
    Find code generator for target: essl  for language: essl . Version: 300 es
    -Use code generator for target: genosl  for language:  osl
    +Use code generator for target: genglsl  for language:  glsl
     
    @@ -15892,7 +15978,7 @@

    Color Management -
    Set up CMS: default_cms for target: genosl
    +
    Set up CMS: default_cms for target: genglsl
     
    @@ -16152,6 +16238,8 @@

    Generating Codeif nodeName: shaderName = mx.createValidName(nodeName) try: + genoptions = context.getOptions() + genoptions.shaderInterfaceType = mx_gen_shader.ShaderInterfaceType.SHADER_INTERFACE_COMPLETE shader = shadergen.generate(shaderName, node, context) except mx.Exception as err: print('Shader generation errors:', err) @@ -16248,82 +16336,175 @@

    Generated Code -
    Pixel Shader For: "Marble_3D" -
    #include "mx_funcs.h"
    -
    -#define true 1
    -#define false 0
    -struct textureresource { string filename; string colorspace; };
    -struct BSDF { closure color response; color throughput; float thickness; float ior; };
    -#define EDF closure color
    -#define VDF closure color
    -struct surfaceshader { closure color bsdf; closure color edf; float opacity; };
    -#define volumeshader closure color
    -#define displacementshader vector
    -#define lightshader closure color
    -#define MATERIAL closure color
    +
    Vertex Shader For: "Marble_3D" +
    #version 450
     
    -#define M_FLOAT_EPS 1e-8
    +#pragma shader_stage(vertex)
     
    -void mx_fractal3d_float(float amplitude, int octaves, float lacunarity, float diminish, vector position, output float result)
    +// Uniform block: PrivateUniforms
    +layout (std140, binding=0) uniform PrivateUniforms_vertex
    +{
    +    mat4 u_worldMatrix;
    +    mat4 u_viewProjectionMatrix;
    +    mat4 u_worldInverseTransposeMatrix;
    +};
    +
    +// Inputs block: VertexInputs
    +layout (location = 0) in vec3 i_position;
    +layout (location = 1) in vec3 i_normal;
    +layout (location = 2) in vec3 i_tangent;
    +
    +layout (location = 0) out vec3 normalWorld;
    +layout (location = 1) out vec3 tangentWorld;
    +layout (location = 2) out vec3 positionObject;
    +layout (location = 3) out vec3 positionWorld;
    +
    +void main()
     {
    -    float f = mx_fbm(position, octaves, lacunarity, diminish, "snoise");
    -    result = f * amplitude;
    +    vec4 hPositionWorld = u_worldMatrix * vec4(i_position, 1.0);
    +    gl_Position = u_viewProjectionMatrix * hPositionWorld;
    +    normalWorld = normalize((u_worldInverseTransposeMatrix * vec4(i_normal, 0.0)).xyz);
    +    tangentWorld = normalize((u_worldMatrix * vec4(i_tangent, 0.0)).xyz);
    +    positionObject = i_position;
    +    positionWorld = hPositionWorld.xyz;
     }
    +
    +
    + +
    + + +
    + + + + + + + + + + + + - - - - - @@ -15226,34 +15735,34 @@

    Viewing a Graph
      graph TB;
    -    NG_marble1_add_xyz[NG_marble1/add_xyz] --".in1"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]
    -    NG_marble1_sin[NG_marble1/sin] --".in1"--> NG_marble1_scale[NG_marble1/scale]
    +    NG_marble1_sum[NG_marble1/sum] --".in"--> NG_marble1_sin[NG_marble1/sin]
    +    NG_marble1_power[NG_marble1/power] --".mix"--> NG_marble1_color_mix[NG_marble1/color_mix]
    +    SR_marble1[SR_marble1] --".surfaceshader"--> Marble_3D[Marble_3D]
    +    NG_marble1_scale_xyz[NG_marble1/scale_xyz] --".in1"--> NG_marble1_sum[NG_marble1/sum]
    +    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_add_xyz[NG_marble1/add_xyz]
         NG_marble1_scale_noise[NG_marble1/scale_noise] --".in2"--> NG_marble1_sum[NG_marble1/sum]
    -    NG_marble1_color_mix[NG_marble1/color_mix] --".subsurface_color"--> SR_marble1[SR_marble1]
         NG_marble1_noise[NG_marble1/noise] --".in1"--> NG_marble1_scale_noise[NG_marble1/scale_noise]
    -    NG_marble1_color_mix[NG_marble1/color_mix] --".base_color"--> SR_marble1[SR_marble1]
    +    NG_marble1_add_xyz[NG_marble1/add_xyz] --".in1"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]
         NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_scale_pos[NG_marble1/scale_pos]
    -    NG_marble1_sum[NG_marble1/sum] --".in"--> NG_marble1_sin[NG_marble1/sin]
    -    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_add_xyz[NG_marble1/add_xyz]
    -    NG_marble1_power[NG_marble1/power] --".mix"--> NG_marble1_color_mix[NG_marble1/color_mix]
    +    NG_marble1_scale[NG_marble1/scale] --".in1"--> NG_marble1_bias[NG_marble1/bias]
         NG_marble1_scale_pos[NG_marble1/scale_pos] --".position"--> NG_marble1_noise[NG_marble1/noise]
         NG_marble1_bias[NG_marble1/bias] --".in1"--> NG_marble1_power[NG_marble1/power]
    -    NG_marble1_scale_xyz[NG_marble1/scale_xyz] --".in1"--> NG_marble1_sum[NG_marble1/sum]
    -    NG_marble1_scale[NG_marble1/scale] --".in1"--> NG_marble1_bias[NG_marble1/bias]
    -    SR_marble1[SR_marble1] --".surfaceshader"--> Marble_3D[Marble_3D]
    +    NG_marble1_color_mix[NG_marble1/color_mix] --".base_color"--> SR_marble1[SR_marble1]
    +    NG_marble1_sin[NG_marble1/sin] --".in1"--> NG_marble1_scale[NG_marble1/scale]
    +    NG_marble1_color_mix[NG_marble1/color_mix] --".subsurface_color"--> SR_marble1[SR_marble1]
     subgraph NG_marble1:
    +   NG_marble1_scale
        NG_marble1_obj_pos
    +   NG_marble1_noise
    +   NG_marble1_power
        NG_marble1_scale_noise
    +   NG_marble1_sum
        NG_marble1_bias
    -   NG_marble1_add_xyz
    +   NG_marble1_scale_pos
        NG_marble1_sin
    -   NG_marble1_power
    -   NG_marble1_sum
    +   NG_marble1_add_xyz
        NG_marble1_scale_xyz
    -   NG_marble1_scale_pos
    -   NG_marble1_noise
        NG_marble1_color_mix
    -   NG_marble1_scale
     end
     
    @@ -15271,7 +15780,7 @@

    Viewing a Graph @@ -15515,48 +16024,48 @@

    Handling Graph Interfaces
    mermaid
       graph TB;
    -    NG_marble1_noise_power([noise_power]) --".in2"--> NG_marble1_power
    +    NG_marble1_color_mix[NG_marble1/color_mix] --> NG_marble1_out[out]
         NG_marble1_sin[NG_marble1/sin] --".in1"--> NG_marble1_scale[NG_marble1/scale]
    -    NG_marble1_scale_noise[NG_marble1/scale_noise] --".in2"--> NG_marble1_sum[NG_marble1/sum]
    -    NG_marble1_noise_scale_2([noise_scale_2]) --".in2"--> NG_marble1_scale_pos
    -    NG_marble1_scale_xyz[NG_marble1/scale_xyz] --".in1"--> NG_marble1_sum[NG_marble1/sum]
         SR_marble1[SR_marble1] --".surfaceshader"--> Marble_3D[Marble_3D]
    -    NG_marble1_base_color_1([base_color_1]) --".bg"--> NG_marble1_color_mix
    +    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_scale_pos[NG_marble1/scale_pos]
    +    NG_marble1_scale[NG_marble1/scale] --".in1"--> NG_marble1_bias[NG_marble1/bias]
    +    NG_marble1_noise_power([noise_power]) --".in2"--> NG_marble1_power
    +    NG_marble1_bias[NG_marble1/bias] --".in1"--> NG_marble1_power[NG_marble1/power]
         NG_marble1_noise_octaves([noise_octaves]) --".octaves"--> NG_marble1_noise
    -    NG_marble1_out[NG_marble1/out] --".base_color"--> SR_marble1[SR_marble1]
    -    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_add_xyz[NG_marble1/add_xyz]
    -    NG_marble1_color_mix[NG_marble1/color_mix] --> NG_marble1_out[out]
    +    NG_marble1_sum[NG_marble1/sum] --".in"--> NG_marble1_sin[NG_marble1/sin]
         NG_marble1_base_color_2([base_color_2]) --".fg"--> NG_marble1_color_mix
    -    NG_marble1_bias[NG_marble1/bias] --".in1"--> NG_marble1_power[NG_marble1/power]
    -    NG_marble1_scale[NG_marble1/scale] --".in1"--> NG_marble1_bias[NG_marble1/bias]
    -    NG_marble1_noise_scale_1([noise_scale_1]) --".in2"--> NG_marble1_scale_xyz
    -    NG_marble1_add_xyz[NG_marble1/add_xyz] --".in1"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]
    +    NG_marble1_scale_xyz[NG_marble1/scale_xyz] --".in1"--> NG_marble1_sum[NG_marble1/sum]
         NG_marble1_out[NG_marble1/out] --".subsurface_color"--> SR_marble1[SR_marble1]
    +    NG_marble1_add_xyz[NG_marble1/add_xyz] --".in1"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]
    +    NG_marble1_noise_scale_2([noise_scale_2]) --".in2"--> NG_marble1_scale_pos
    +    NG_marble1_noise_scale_1([noise_scale_1]) --".in2"--> NG_marble1_scale_xyz
    +    NG_marble1_power[NG_marble1/power] --".mix"--> NG_marble1_color_mix[NG_marble1/color_mix]
    +    NG_marble1_out[NG_marble1/out] --".base_color"--> SR_marble1[SR_marble1]
    +    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_add_xyz[NG_marble1/add_xyz]
    +    NG_marble1_scale_noise[NG_marble1/scale_noise] --".in2"--> NG_marble1_sum[NG_marble1/sum]
         NG_marble1_noise[NG_marble1/noise] --".in1"--> NG_marble1_scale_noise[NG_marble1/scale_noise]
    -    NG_marble1_obj_pos[NG_marble1/obj_pos] --".in1"--> NG_marble1_scale_pos[NG_marble1/scale_pos]
    -    NG_marble1_sum[NG_marble1/sum] --".in"--> NG_marble1_sin[NG_marble1/sin]
    +    NG_marble1_base_color_1([base_color_1]) --".bg"--> NG_marble1_color_mix
         NG_marble1_scale_pos[NG_marble1/scale_pos] --".position"--> NG_marble1_noise[NG_marble1/noise]
    -    NG_marble1_power[NG_marble1/power] --".mix"--> NG_marble1_color_mix[NG_marble1/color_mix]
     subgraph NG_marble1:
    -   NG_marble1_obj_pos
        NG_marble1_noise_octaves
    -   NG_marble1_bias
    +   NG_marble1_scale
    +   NG_marble1_power
    +   NG_marble1_scale_noise
    +   NG_marble1_out
        NG_marble1_noise_scale_2
    +   NG_marble1_color_mix
    +   NG_marble1_base_color_1
    +   NG_marble1_noise
        NG_marble1_base_color_2
    -   NG_marble1_scale
    -   NG_marble1_sin
        NG_marble1_noise_scale_1
        NG_marble1_noise_power
    +   NG_marble1_sin
        NG_marble1_add_xyz
    -   NG_marble1_power
    -   NG_marble1_noise
    -   NG_marble1_out
    -   NG_marble1_scale_noise
    -   NG_marble1_scale_xyz
    +   NG_marble1_obj_pos
        NG_marble1_sum
    -   NG_marble1_base_color_1
    +   NG_marble1_bias
        NG_marble1_scale_pos
    -   NG_marble1_color_mix
    +   NG_marble1_scale_xyz
     end
     
    @@ -15576,9 +16085,9 @@

    Handling Graph Interfaces diff --git a/pymaterialx/mtlx_traversal_notebook.ipynb b/pymaterialx/mtlx_traversal_notebook.ipynb index b04dc3bc..8f6dac6c 100644 --- a/pymaterialx/mtlx_traversal_notebook.ipynb +++ b/pymaterialx/mtlx_traversal_notebook.ipynb @@ -21,10 +21,7 @@ "2. Port getConnectedNode() for finding the node connected to an input. This uses the previous interface.\n", "3. Input getInterfaceInput()() for finding if an input is connected to an input interface on a node graph. \n", "\n", - "> Note that for diagramming there is a PR underway to add in a new `GraphIO` interface which would encapsulate the supported required for Mermaid and GraphViz Dot support.\n", - "For this site all Mermaid diagrams are generated using this interface.\n", - "\n", - "> The utilities (including Mermaid generation) in this tutorial are collected in the `mtlxutils` file: mtlxutils/mxtraversal.\n" + "> For this site all Mermaid diagrams are generated using a custom utility. The Javascript version of this utility is used for the interactive graph generation page: Graph Generator. The utilities (including Mermaid generation) in this tutorial are collected in the `mtlxutils` file: mtlxutils/mxtraversal.\n" ] }, { @@ -148,21 +145,21 @@ "name": "stdout", "output_type": "stream", "text": [ - "Edge: NG_marble1/sum --> NG_marble1/sin/in\n", - "Edge: NG_marble1/scale --> NG_marble1/bias/in1\n", + "Edge: SR_marble1 --> Marble_3D/surfaceshader\n", "Edge: NG_marble1/scale_xyz --> NG_marble1/sum/in1\n", - "Edge: NG_marble1/sin --> NG_marble1/scale/in1\n", + "Edge: NG_marble1/obj_pos --> NG_marble1/add_xyz/in1\n", "Edge: NG_marble1/noise --> NG_marble1/scale_noise/in1\n", + "Edge: NG_marble1/add_xyz --> NG_marble1/scale_xyz/in1\n", + "Edge: NG_marble1/color_mix --> SR_marble1/subsurface_color\n", + "Edge: NG_marble1/power --> NG_marble1/color_mix/mix\n", + "Edge: NG_marble1/color_mix --> SR_marble1/base_color\n", "Edge: NG_marble1/obj_pos --> NG_marble1/scale_pos/in1\n", + "Edge: NG_marble1/sum --> NG_marble1/sin/in\n", "Edge: NG_marble1/scale_noise --> NG_marble1/sum/in2\n", + "Edge: NG_marble1/sin --> NG_marble1/scale/in1\n", + "Edge: NG_marble1/scale --> NG_marble1/bias/in1\n", "Edge: NG_marble1/scale_pos --> NG_marble1/noise/position\n", - "Edge: NG_marble1/obj_pos --> NG_marble1/add_xyz/in1\n", - "Edge: NG_marble1/color_mix --> SR_marble1/subsurface_color\n", - "Edge: NG_marble1/bias --> NG_marble1/power/in1\n", - "Edge: SR_marble1 --> Marble_3D/surfaceshader\n", - "Edge: NG_marble1/add_xyz --> NG_marble1/scale_xyz/in1\n", - "Edge: NG_marble1/color_mix --> SR_marble1/base_color\n", - "Edge: NG_marble1/power --> NG_marble1/color_mix/mix\n" + "Edge: NG_marble1/bias --> NG_marble1/power/in1\n" ] } ], @@ -208,18 +205,18 @@ "- Marble_3D\n", "- SR_marble1\n", "NG_marble1:\n", - "- NG_marble1/sum\n", - "- NG_marble1/add_xyz\n", - "- NG_marble1/color_mix\n", - "- NG_marble1/obj_pos\n", - "- NG_marble1/sin\n", - "- NG_marble1/bias\n", "- NG_marble1/scale\n", - "- NG_marble1/power\n", + "- NG_marble1/obj_pos\n", "- NG_marble1/noise\n", + "- NG_marble1/power\n", "- NG_marble1/scale_noise\n", + "- NG_marble1/sum\n", + "- NG_marble1/bias\n", "- NG_marble1/scale_pos\n", - "- NG_marble1/scale_xyz\n" + "- NG_marble1/sin\n", + "- NG_marble1/add_xyz\n", + "- NG_marble1/scale_xyz\n", + "- NG_marble1/color_mix\n" ] } ], @@ -453,34 +450,34 @@ "output_type": "stream", "text": [ " graph TB;\n", - " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_scale_pos[NG_marble1/scale_pos]\n", " NG_marble1_sum[NG_marble1/sum] --\".in\"--> NG_marble1_sin[NG_marble1/sin]\n", " NG_marble1_power[NG_marble1/power] --\".mix\"--> NG_marble1_color_mix[NG_marble1/color_mix]\n", + " SR_marble1[SR_marble1] --\".surfaceshader\"--> Marble_3D[Marble_3D]\n", " NG_marble1_scale_xyz[NG_marble1/scale_xyz] --\".in1\"--> NG_marble1_sum[NG_marble1/sum]\n", - " NG_marble1_color_mix[NG_marble1/color_mix] --\".subsurface_color\"--> SR_marble1[SR_marble1]\n", - " NG_marble1_color_mix[NG_marble1/color_mix] --\".base_color\"--> SR_marble1[SR_marble1]\n", + " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_add_xyz[NG_marble1/add_xyz]\n", + " NG_marble1_scale_noise[NG_marble1/scale_noise] --\".in2\"--> NG_marble1_sum[NG_marble1/sum]\n", + " NG_marble1_noise[NG_marble1/noise] --\".in1\"--> NG_marble1_scale_noise[NG_marble1/scale_noise]\n", " NG_marble1_add_xyz[NG_marble1/add_xyz] --\".in1\"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]\n", + " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_scale_pos[NG_marble1/scale_pos]\n", " NG_marble1_scale[NG_marble1/scale] --\".in1\"--> NG_marble1_bias[NG_marble1/bias]\n", " NG_marble1_scale_pos[NG_marble1/scale_pos] --\".position\"--> NG_marble1_noise[NG_marble1/noise]\n", - " NG_marble1_scale_noise[NG_marble1/scale_noise] --\".in2\"--> NG_marble1_sum[NG_marble1/sum]\n", - " NG_marble1_sin[NG_marble1/sin] --\".in1\"--> NG_marble1_scale[NG_marble1/scale]\n", - " SR_marble1[SR_marble1] --\".surfaceshader\"--> Marble_3D[Marble_3D]\n", - " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_add_xyz[NG_marble1/add_xyz]\n", - " NG_marble1_noise[NG_marble1/noise] --\".in1\"--> NG_marble1_scale_noise[NG_marble1/scale_noise]\n", " NG_marble1_bias[NG_marble1/bias] --\".in1\"--> NG_marble1_power[NG_marble1/power]\n", + " NG_marble1_color_mix[NG_marble1/color_mix] --\".base_color\"--> SR_marble1[SR_marble1]\n", + " NG_marble1_sin[NG_marble1/sin] --\".in1\"--> NG_marble1_scale[NG_marble1/scale]\n", + " NG_marble1_color_mix[NG_marble1/color_mix] --\".subsurface_color\"--> SR_marble1[SR_marble1]\n", "subgraph NG_marble1:\n", - " NG_marble1_sum\n", - " NG_marble1_add_xyz\n", - " NG_marble1_color_mix\n", - " NG_marble1_obj_pos\n", - " NG_marble1_sin\n", - " NG_marble1_bias\n", " NG_marble1_scale\n", - " NG_marble1_power\n", + " NG_marble1_obj_pos\n", " NG_marble1_noise\n", + " NG_marble1_power\n", " NG_marble1_scale_noise\n", + " NG_marble1_sum\n", + " NG_marble1_bias\n", " NG_marble1_scale_pos\n", + " NG_marble1_sin\n", + " NG_marble1_add_xyz\n", " NG_marble1_scale_xyz\n", + " NG_marble1_color_mix\n", "end\n" ] } @@ -530,7 +527,7 @@ "source": [ "The resulting diagram looks like this:\n", "\n", - "" + "" ] }, { @@ -684,48 +681,48 @@ "text/markdown": [ "```mermaid\n", " graph TB;\n", - " NG_marble1_sum[NG_marble1/sum] --\".in\"--> NG_marble1_sin[NG_marble1/sin]\n", - " NG_marble1_out[NG_marble1/out] --\".subsurface_color\"--> SR_marble1[SR_marble1]\n", - " NG_marble1_scale_noise[NG_marble1/scale_noise] --\".in2\"--> NG_marble1_sum[NG_marble1/sum]\n", + " NG_marble1_color_mix[NG_marble1/color_mix] --> NG_marble1_out[out]\n", " NG_marble1_sin[NG_marble1/sin] --\".in1\"--> NG_marble1_scale[NG_marble1/scale]\n", + " SR_marble1[SR_marble1] --\".surfaceshader\"--> Marble_3D[Marble_3D]\n", + " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_scale_pos[NG_marble1/scale_pos]\n", + " NG_marble1_scale[NG_marble1/scale] --\".in1\"--> NG_marble1_bias[NG_marble1/bias]\n", " NG_marble1_noise_power([noise_power]) --\".in2\"--> NG_marble1_power\n", - " NG_marble1_scale_pos[NG_marble1/scale_pos] --\".position\"--> NG_marble1_noise[NG_marble1/noise]\n", + " NG_marble1_bias[NG_marble1/bias] --\".in1\"--> NG_marble1_power[NG_marble1/power]\n", " NG_marble1_noise_octaves([noise_octaves]) --\".octaves\"--> NG_marble1_noise\n", - " SR_marble1[SR_marble1] --\".surfaceshader\"--> Marble_3D[Marble_3D]\n", - " NG_marble1_noise[NG_marble1/noise] --\".in1\"--> NG_marble1_scale_noise[NG_marble1/scale_noise]\n", - " NG_marble1_power[NG_marble1/power] --\".mix\"--> NG_marble1_color_mix[NG_marble1/color_mix]\n", - " NG_marble1_scale_xyz[NG_marble1/scale_xyz] --\".in1\"--> NG_marble1_sum[NG_marble1/sum]\n", + " NG_marble1_sum[NG_marble1/sum] --\".in\"--> NG_marble1_sin[NG_marble1/sin]\n", " NG_marble1_base_color_2([base_color_2]) --\".fg\"--> NG_marble1_color_mix\n", - " NG_marble1_out[NG_marble1/out] --\".base_color\"--> SR_marble1[SR_marble1]\n", - " NG_marble1_scale[NG_marble1/scale] --\".in1\"--> NG_marble1_bias[NG_marble1/bias]\n", - " NG_marble1_base_color_1([base_color_1]) --\".bg\"--> NG_marble1_color_mix\n", + " NG_marble1_scale_xyz[NG_marble1/scale_xyz] --\".in1\"--> NG_marble1_sum[NG_marble1/sum]\n", + " NG_marble1_out[NG_marble1/out] --\".subsurface_color\"--> SR_marble1[SR_marble1]\n", + " NG_marble1_add_xyz[NG_marble1/add_xyz] --\".in1\"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]\n", + " NG_marble1_noise_scale_2([noise_scale_2]) --\".in2\"--> NG_marble1_scale_pos\n", " NG_marble1_noise_scale_1([noise_scale_1]) --\".in2\"--> NG_marble1_scale_xyz\n", + " NG_marble1_power[NG_marble1/power] --\".mix\"--> NG_marble1_color_mix[NG_marble1/color_mix]\n", + " NG_marble1_out[NG_marble1/out] --\".base_color\"--> SR_marble1[SR_marble1]\n", " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_add_xyz[NG_marble1/add_xyz]\n", - " NG_marble1_obj_pos[NG_marble1/obj_pos] --\".in1\"--> NG_marble1_scale_pos[NG_marble1/scale_pos]\n", - " NG_marble1_noise_scale_2([noise_scale_2]) --\".in2\"--> NG_marble1_scale_pos\n", - " NG_marble1_add_xyz[NG_marble1/add_xyz] --\".in1\"--> NG_marble1_scale_xyz[NG_marble1/scale_xyz]\n", - " NG_marble1_color_mix[NG_marble1/color_mix] --> NG_marble1_out[out]\n", - " NG_marble1_bias[NG_marble1/bias] --\".in1\"--> NG_marble1_power[NG_marble1/power]\n", + " NG_marble1_scale_noise[NG_marble1/scale_noise] --\".in2\"--> NG_marble1_sum[NG_marble1/sum]\n", + " NG_marble1_noise[NG_marble1/noise] --\".in1\"--> NG_marble1_scale_noise[NG_marble1/scale_noise]\n", + " NG_marble1_base_color_1([base_color_1]) --\".bg\"--> NG_marble1_color_mix\n", + " NG_marble1_scale_pos[NG_marble1/scale_pos] --\".position\"--> NG_marble1_noise[NG_marble1/noise]\n", "subgraph NG_marble1:\n", - " NG_marble1_bias\n", + " NG_marble1_noise_octaves\n", " NG_marble1_scale\n", - " NG_marble1_base_color_1\n", - " NG_marble1_scale_pos\n", + " NG_marble1_power\n", + " NG_marble1_scale_noise\n", " NG_marble1_out\n", - " NG_marble1_sum\n", + " NG_marble1_noise_scale_2\n", " NG_marble1_color_mix\n", - " NG_marble1_obj_pos\n", + " NG_marble1_base_color_1\n", " NG_marble1_noise\n", - " NG_marble1_scale_noise\n", - " NG_marble1_noise_octaves\n", - " NG_marble1_noise_scale_2\n", - " NG_marble1_add_xyz\n", + " NG_marble1_base_color_2\n", + " NG_marble1_noise_scale_1\n", " NG_marble1_noise_power\n", " NG_marble1_sin\n", - " NG_marble1_power\n", + " NG_marble1_add_xyz\n", + " NG_marble1_obj_pos\n", + " NG_marble1_sum\n", + " NG_marble1_bias\n", + " NG_marble1_scale_pos\n", " NG_marble1_scale_xyz\n", - " NG_marble1_base_color_2\n", - " NG_marble1_noise_scale_1\n", "end\n", "```\n" ] @@ -787,11 +784,11 @@ "source": [ "The resulting diagram looks like this:\n", "\n", - "\n", + "\n", "\n", "with the same graph as seen in the graph editor:\n", "\n", - "\n" + "\n" ] } ], @@ -811,7 +808,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.10.9" + "version": "3.10.10" }, "orig_nbformat": 4, "vscode": { diff --git a/pymaterialx/mtlx_traversal_notebook.py b/pymaterialx/mtlx_traversal_notebook.py index 65029246..e9acf45b 100644 --- a/pymaterialx/mtlx_traversal_notebook.py +++ b/pymaterialx/mtlx_traversal_notebook.py @@ -15,10 +15,7 @@ # 2. Port getConnectedNode() for finding the node connected to an input. This uses the previous interface. # 3. Input getInterfaceInput()() for finding if an input is connected to an input interface on a node graph. # -# > Note that for diagramming there is a PR underway to add in a new `GraphIO` interface which would encapsulate the supported required for Mermaid and GraphViz Dot support. -# For this site all Mermaid diagrams are generated using this interface. -# -# > The utilities (including Mermaid generation) in this tutorial are collected in the `mtlxutils` file: mtlxutils/mxtraversal. +# > For this site all Mermaid diagrams are generated using a custom utility. The Javascript version of this utility is used for the interactive graph generation page: Graph Generator. The utilities (including Mermaid generation) in this tutorial are collected in the `mtlxutils` file: mtlxutils/mxtraversal. # # %% [markdown] @@ -311,7 +308,7 @@ def generateMermaidGraph_nointerfaces(roots, orientation): # %% [markdown] # The resulting diagram looks like this: # -# +# # %% [markdown] # ## Handling Graph Interfaces @@ -475,11 +472,11 @@ def generateMermaidGraph(roots, orientation): # %% [markdown] # The resulting diagram looks like this: # -# +# # # with the same graph as seen in the graph editor: # -# +# # diff --git a/pymaterialx/mtlx_usd_notebook.html b/pymaterialx/mtlx_usd_notebook.html index e16c9c49..589ba8dd 100644 --- a/pymaterialx/mtlx_usd_notebook.html +++ b/pymaterialx/mtlx_usd_notebook.html @@ -277,8 +277,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-Widget, /* */ +/* */ +.p-Widget, /* */ .lm-Widget { box-sizing: border-box; position: relative; @@ -286,12 +286,19 @@ cursor: default; } - -/* */ .p-Widget.p-mod-hidden, /* */ +/* */ +.p-Widget.p-mod-hidden, /* */ .lm-Widget.lm-mod-hidden { display: none !important; } +.lm-AccordionPanel[data-orientation='horizontal'] > .lm-AccordionPanel-title { + /* Title is rotated for horizontal accordion panel using CSS */ + display: block; + transform-origin: top left; + transform: rotate(-90deg) translate(-100%); +} + /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Copyright (c) 2014-2017, PhosphorJS Contributors @@ -301,8 +308,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-CommandPalette, /* */ +/* */ +.p-CommandPalette, /* */ .lm-CommandPalette { display: flex; flex-direction: column; @@ -312,14 +319,14 @@ user-select: none; } - -/* */ .p-CommandPalette-search, /* */ +/* */ +.p-CommandPalette-search, /* */ .lm-CommandPalette-search { flex: 0 0 auto; } - -/* */ .p-CommandPalette-content, /* */ +/* */ +.p-CommandPalette-content, /* */ .lm-CommandPalette-content { flex: 1 1 auto; margin: 0; @@ -329,42 +336,42 @@ list-style-type: none; } - -/* */ .p-CommandPalette-header, /* */ +/* */ +.p-CommandPalette-header, /* */ .lm-CommandPalette-header { overflow: hidden; white-space: nowrap; text-overflow: ellipsis; } - -/* */ .p-CommandPalette-item, /* */ +/* */ +.p-CommandPalette-item, /* */ .lm-CommandPalette-item { display: flex; flex-direction: row; } - -/* */ .p-CommandPalette-itemIcon, /* */ +/* */ +.p-CommandPalette-itemIcon, /* */ .lm-CommandPalette-itemIcon { flex: 0 0 auto; } - -/* */ .p-CommandPalette-itemContent, /* */ +/* */ +.p-CommandPalette-itemContent, /* */ .lm-CommandPalette-itemContent { flex: 1 1 auto; overflow: hidden; } - -/* */ .p-CommandPalette-itemShortcut, /* */ +/* */ +.p-CommandPalette-itemShortcut, /* */ .lm-CommandPalette-itemShortcut { flex: 0 0 auto; } - -/* */ .p-CommandPalette-itemLabel, /* */ +/* */ +.p-CommandPalette-itemLabel, /* */ .lm-CommandPalette-itemLabel { overflow: hidden; white-space: nowrap; @@ -372,30 +379,30 @@ } .lm-close-icon { - border:1px solid transparent; + border: 1px solid transparent; background-color: transparent; position: absolute; - z-index:1; - right:3%; - top: 0; - bottom: 0; - margin: auto; - padding: 7px 0; - display: none; - vertical-align: middle; + z-index: 1; + right: 3%; + top: 0; + bottom: 0; + margin: auto; + padding: 7px 0; + display: none; + vertical-align: middle; outline: 0; cursor: pointer; } .lm-close-icon:after { - content: "X"; - display: block; - width: 15px; - height: 15px; - text-align: center; - color:#000; - font-weight: normal; - font-size: 12px; - cursor: pointer; + content: 'X'; + display: block; + width: 15px; + height: 15px; + text-align: center; + color: #000; + font-weight: normal; + font-size: 12px; + cursor: pointer; } /*----------------------------------------------------------------------------- @@ -407,38 +414,38 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-DockPanel, /* */ +/* */ +.p-DockPanel, /* */ .lm-DockPanel { z-index: 0; } - -/* */ .p-DockPanel-widget, /* */ +/* */ +.p-DockPanel-widget, /* */ .lm-DockPanel-widget { z-index: 0; } - -/* */ .p-DockPanel-tabBar, /* */ +/* */ +.p-DockPanel-tabBar, /* */ .lm-DockPanel-tabBar { z-index: 1; } - -/* */ .p-DockPanel-handle, /* */ +/* */ +.p-DockPanel-handle, /* */ .lm-DockPanel-handle { z-index: 2; } - -/* */ .p-DockPanel-handle.p-mod-hidden, /* */ +/* */ +.p-DockPanel-handle.p-mod-hidden, /* */ .lm-DockPanel-handle.lm-mod-hidden { display: none !important; } - -/* */ .p-DockPanel-handle:after, /* */ +/* */ +.p-DockPanel-handle:after, /* */ .lm-DockPanel-handle:after { position: absolute; top: 0; @@ -448,7 +455,6 @@ content: ''; } - /* */ .p-DockPanel-handle[data-orientation='horizontal'], /* */ @@ -456,7 +462,6 @@ cursor: ew-resize; } - /* */ .p-DockPanel-handle[data-orientation='vertical'], /* */ @@ -464,7 +469,6 @@ cursor: ns-resize; } - /* */ .p-DockPanel-handle[data-orientation='horizontal']:after, /* */ @@ -474,7 +478,6 @@ transform: translateX(-50%); } - /* */ .p-DockPanel-handle[data-orientation='vertical']:after, /* */ @@ -484,16 +487,16 @@ transform: translateY(-50%); } - -/* */ .p-DockPanel-overlay, /* */ +/* */ +.p-DockPanel-overlay, /* */ .lm-DockPanel-overlay { z-index: 3; box-sizing: border-box; pointer-events: none; } - -/* */ .p-DockPanel-overlay.p-mod-hidden, /* */ +/* */ +.p-DockPanel-overlay.p-mod-hidden, /* */ .lm-DockPanel-overlay.lm-mod-hidden { display: none !important; } @@ -507,8 +510,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-Menu, /* */ +/* */ +.p-Menu, /* */ .lm-Menu { z-index: 10000; position: absolute; @@ -522,8 +525,8 @@ user-select: none; } - -/* */ .p-Menu-content, /* */ +/* */ +.p-Menu-content, /* */ .lm-Menu-content { margin: 0; padding: 0; @@ -531,13 +534,12 @@ list-style-type: none; } - -/* */ .p-Menu-item, /* */ +/* */ +.p-Menu-item, /* */ .lm-Menu-item { display: table-row; } - /* */ .p-Menu-item.p-mod-hidden, .p-Menu-item.p-mod-collapsed, @@ -547,7 +549,6 @@ display: none !important; } - /* */ .p-Menu-itemIcon, .p-Menu-itemSubmenuIcon, @@ -558,15 +559,15 @@ text-align: center; } - -/* */ .p-Menu-itemLabel, /* */ +/* */ +.p-Menu-itemLabel, /* */ .lm-Menu-itemLabel { display: table-cell; text-align: left; } - -/* */ .p-Menu-itemShortcut, /* */ +/* */ +.p-Menu-itemShortcut, /* */ .lm-Menu-itemShortcut { display: table-cell; text-align: right; @@ -581,8 +582,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-MenuBar, /* */ +/* */ +.p-MenuBar, /* */ .lm-MenuBar { outline: none; -webkit-user-select: none; @@ -591,8 +592,8 @@ user-select: none; } - -/* */ .p-MenuBar-content, /* */ +/* */ +.p-MenuBar-content, /* */ .lm-MenuBar-content { margin: 0; padding: 0; @@ -601,13 +602,12 @@ list-style-type: none; } - -/* */ .p--MenuBar-item, /* */ +/* */ +.p--MenuBar-item, /* */ .lm-MenuBar-item { box-sizing: border-box; } - /* */ .p-MenuBar-itemIcon, .p-MenuBar-itemLabel, @@ -626,8 +626,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-ScrollBar, /* */ +/* */ +.p-ScrollBar, /* */ .lm-ScrollBar { display: flex; -webkit-user-select: none; @@ -636,7 +636,6 @@ user-select: none; } - /* */ .p-ScrollBar[data-orientation='horizontal'], /* */ @@ -644,7 +643,6 @@ flex-direction: row; } - /* */ .p-ScrollBar[data-orientation='vertical'], /* */ @@ -652,15 +650,15 @@ flex-direction: column; } - -/* */ .p-ScrollBar-button, /* */ +/* */ +.p-ScrollBar-button, /* */ .lm-ScrollBar-button { box-sizing: border-box; flex: 0 0 auto; } - -/* */ .p-ScrollBar-track, /* */ +/* */ +.p-ScrollBar-track, /* */ .lm-ScrollBar-track { box-sizing: border-box; position: relative; @@ -668,8 +666,8 @@ flex: 1 1 auto; } - -/* */ .p-ScrollBar-thumb, /* */ +/* */ +.p-ScrollBar-thumb, /* */ .lm-ScrollBar-thumb { box-sizing: border-box; position: absolute; @@ -684,26 +682,26 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-SplitPanel-child, /* */ +/* */ +.p-SplitPanel-child, /* */ .lm-SplitPanel-child { z-index: 0; } - -/* */ .p-SplitPanel-handle, /* */ +/* */ +.p-SplitPanel-handle, /* */ .lm-SplitPanel-handle { z-index: 1; } - -/* */ .p-SplitPanel-handle.p-mod-hidden, /* */ +/* */ +.p-SplitPanel-handle.p-mod-hidden, /* */ .lm-SplitPanel-handle.lm-mod-hidden { display: none !important; } - -/* */ .p-SplitPanel-handle:after, /* */ +/* */ +.p-SplitPanel-handle:after, /* */ .lm-SplitPanel-handle:after { position: absolute; top: 0; @@ -713,7 +711,6 @@ content: ''; } - /* */ .p-SplitPanel[data-orientation='horizontal'] > .p-SplitPanel-handle, /* */ @@ -721,7 +718,6 @@ cursor: ew-resize; } - /* */ .p-SplitPanel[data-orientation='vertical'] > .p-SplitPanel-handle, /* */ @@ -729,7 +725,6 @@ cursor: ns-resize; } - /* */ .p-SplitPanel[data-orientation='horizontal'] > .p-SplitPanel-handle:after, /* */ @@ -739,7 +734,6 @@ transform: translateX(-50%); } - /* */ .p-SplitPanel[data-orientation='vertical'] > .p-SplitPanel-handle:after, /* */ @@ -758,8 +752,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-TabBar, /* */ +/* */ +.p-TabBar, /* */ .lm-TabBar { display: flex; -webkit-user-select: none; @@ -768,22 +762,22 @@ user-select: none; } - -/* */ .p-TabBar[data-orientation='horizontal'], /* */ +/* */ +.p-TabBar[data-orientation='horizontal'], /* */ .lm-TabBar[data-orientation='horizontal'] { flex-direction: row; align-items: flex-end; } - -/* */ .p-TabBar[data-orientation='vertical'], /* */ +/* */ +.p-TabBar[data-orientation='vertical'], /* */ .lm-TabBar[data-orientation='vertical'] { flex-direction: column; align-items: flex-end; } - -/* */ .p-TabBar-content, /* */ +/* */ +.p-TabBar-content, /* */ .lm-TabBar-content { margin: 0; padding: 0; @@ -792,7 +786,6 @@ list-style-type: none; } - /* */ .p-TabBar[data-orientation='horizontal'] > .p-TabBar-content, /* */ @@ -800,7 +793,6 @@ flex-direction: row; } - /* */ .p-TabBar[data-orientation='vertical'] > .p-TabBar-content, /* */ @@ -808,16 +800,16 @@ flex-direction: column; } - -/* */ .p-TabBar-tab, /* */ +/* */ +.p-TabBar-tab, /* */ .lm-TabBar-tab { display: flex; flex-direction: row; box-sizing: border-box; overflow: hidden; + touch-action: none; /* Disable native Drag/Drop */ } - /* */ .p-TabBar-tabIcon, .p-TabBar-tabCloseIcon, @@ -827,39 +819,36 @@ flex: 0 0 auto; } - -/* */ .p-TabBar-tabLabel, /* */ +/* */ +.p-TabBar-tabLabel, /* */ .lm-TabBar-tabLabel { flex: 1 1 auto; overflow: hidden; white-space: nowrap; } - .lm-TabBar-tabInput { user-select: all; width: 100%; - box-sizing : border-box; + box-sizing: border-box; } - -/* */ .p-TabBar-tab.p-mod-hidden, /* */ +/* */ +.p-TabBar-tab.p-mod-hidden, /* */ .lm-TabBar-tab.lm-mod-hidden { display: none !important; } - .lm-TabBar-addButton.lm-mod-hidden { display: none !important; } - -/* */ .p-TabBar.p-mod-dragging .p-TabBar-tab, /* */ +/* */ +.p-TabBar.p-mod-dragging .p-TabBar-tab, /* */ .lm-TabBar.lm-mod-dragging .lm-TabBar-tab { position: relative; } - /* */ .p-TabBar.p-mod-dragging[data-orientation='horizontal'] .p-TabBar-tab, /* */ @@ -868,7 +857,6 @@ transition: left 150ms ease; } - /* */ .p-TabBar.p-mod-dragging[data-orientation='vertical'] .p-TabBar-tab, /* */ @@ -877,7 +865,6 @@ transition: top 150ms ease; } - /* */ .p-TabBar.p-mod-dragging .p-TabBar-tab.p-mod-dragging, /* */ @@ -888,7 +875,7 @@ .lm-TabBar-tabLabel .lm-TabBar-tabInput { user-select: all; width: 100%; - box-sizing : border-box; + box-sizing: border-box; background: inherit; } @@ -901,14 +888,14 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ .p-TabPanel-tabBar, /* */ +/* */ +.p-TabPanel-tabBar, /* */ .lm-TabPanel-tabBar { z-index: 1; } - -/* */ .p-TabPanel-stackedPanel, /* */ +/* */ +.p-TabPanel-stackedPanel, /* */ .lm-TabPanel-stackedPanel { z-index: 0; } @@ -9365,7 +9352,11 @@ /* Icons urls */ :root { + --jp-icon-add-above: url(); + --jp-icon-add-below: url(); --jp-icon-add: url(); + --jp-icon-bell: url(); + --jp-icon-bug-dot: url(); --jp-icon-bug: url(); --jp-icon-build: url(); --jp-icon-caret-down-empty-thin: url(); @@ -9382,11 +9373,13 @@ --jp-icon-clear: url(); --jp-icon-close: url(); --jp-icon-code: url(); - --jp-icon-console: url(); + --jp-icon-console: url(); --jp-icon-copy: url(); --jp-icon-copyright: url(); --jp-icon-cut: url(); + --jp-icon-delete: url(); --jp-icon-download: url(); + --jp-icon-duplicate: url(); --jp-icon-edit: url(); --jp-icon-ellipses: url(); --jp-icon-extension: url(); @@ -9394,32 +9387,37 @@ --jp-icon-file-upload: url(); --jp-icon-file: url(); --jp-icon-filter-list: url(); + --jp-icon-folder-favorite: url(); --jp-icon-folder: url(); + --jp-icon-home: url(); --jp-icon-html5: url(); --jp-icon-image: url(); - --jp-icon-inspector: url(); - --jp-icon-json: url(); + --jp-icon-inspector: url(); + --jp-icon-json: url(); --jp-icon-julia: url(); - --jp-icon-jupyter-favicon: url(); - --jp-icon-jupyter: url(); + --jp-icon-jupyter-favicon: url(); + --jp-icon-jupyter: url(); --jp-icon-jupyterlab-wordmark: url(); --jp-icon-kernel: url(); --jp-icon-keyboard: url(); + --jp-icon-launch: url(); --jp-icon-launcher: url(); --jp-icon-line-form: url(); --jp-icon-link: url(); --jp-icon-list: url(); --jp-icon-listings-info: url(); --jp-icon-markdown: url(); + --jp-icon-move-down: url(); + --jp-icon-move-up: url(); --jp-icon-new-folder: url(); --jp-icon-not-trusted: url(); - --jp-icon-notebook: url(); + --jp-icon-notebook: url(); --jp-icon-numbering: url(); --jp-icon-offline-bolt: url(); --jp-icon-palette: url(); --jp-icon-paste: url(); --jp-icon-pdf: url(); - --jp-icon-python: url(); + --jp-icon-python: url(); --jp-icon-r-kernel: url(); --jp-icon-react: url(); --jp-icon-redo: url(); @@ -9430,26 +9428,41 @@ --jp-icon-save: url(); --jp-icon-search: url(); --jp-icon-settings: url(); + --jp-icon-share: url(); --jp-icon-spreadsheet: url(); --jp-icon-stop: url(); --jp-icon-tab: url(); --jp-icon-table-rows: url(); --jp-icon-tag: url(); - --jp-icon-terminal: url(); - --jp-icon-text-editor: url(); + --jp-icon-terminal: url(); + --jp-icon-text-editor: url(); --jp-icon-toc: url(); --jp-icon-tree-view: url(); --jp-icon-trusted: url(); --jp-icon-undo: url(); + --jp-icon-user: url(); + --jp-icon-users: url(); --jp-icon-vega: url(); --jp-icon-yaml: url(); } /* Icon CSS class declarations */ +.jp-AddAboveIcon { + background-image: var(--jp-icon-add-above); +} +.jp-AddBelowIcon { + background-image: var(--jp-icon-add-below); +} .jp-AddIcon { background-image: var(--jp-icon-add); } +.jp-BellIcon { + background-image: var(--jp-icon-bell); +} +.jp-BugDotIcon { + background-image: var(--jp-icon-bug-dot); +} .jp-BugIcon { background-image: var(--jp-icon-bug); } @@ -9510,9 +9523,15 @@ .jp-CutIcon { background-image: var(--jp-icon-cut); } +.jp-DeleteIcon { + background-image: var(--jp-icon-delete); +} .jp-DownloadIcon { background-image: var(--jp-icon-download); } +.jp-DuplicateIcon { + background-image: var(--jp-icon-duplicate); +} .jp-EditIcon { background-image: var(--jp-icon-edit); } @@ -9534,9 +9553,15 @@ .jp-FilterListIcon { background-image: var(--jp-icon-filter-list); } +.jp-FolderFavoriteIcon { + background-image: var(--jp-icon-folder-favorite); +} .jp-FolderIcon { background-image: var(--jp-icon-folder); } +.jp-HomeIcon { + background-image: var(--jp-icon-home); +} .jp-Html5Icon { background-image: var(--jp-icon-html5); } @@ -9567,6 +9592,9 @@ .jp-KeyboardIcon { background-image: var(--jp-icon-keyboard); } +.jp-LaunchIcon { + background-image: var(--jp-icon-launch); +} .jp-LauncherIcon { background-image: var(--jp-icon-launcher); } @@ -9585,6 +9613,12 @@ .jp-MarkdownIcon { background-image: var(--jp-icon-markdown); } +.jp-MoveDownIcon { + background-image: var(--jp-icon-move-down); +} +.jp-MoveUpIcon { + background-image: var(--jp-icon-move-up); +} .jp-NewFolderIcon { background-image: var(--jp-icon-new-folder); } @@ -9642,6 +9676,9 @@ .jp-SettingsIcon { background-image: var(--jp-icon-settings); } +.jp-ShareIcon { + background-image: var(--jp-icon-share); +} .jp-SpreadsheetIcon { background-image: var(--jp-icon-spreadsheet); } @@ -9675,6 +9712,12 @@ .jp-UndoIcon { background-image: var(--jp-icon-undo); } +.jp-UserIcon { + background-image: var(--jp-icon-user); +} +.jp-UsersIcon { + background-image: var(--jp-icon-users); +} .jp-VegaIcon { background-image: var(--jp-icon-vega); } @@ -9733,6 +9776,36 @@ | Distributed under the terms of the Modified BSD License. |----------------------------------------------------------------------------*/ +.lm-TabBar .lm-TabBar-addButton { + align-items: center; + display: flex; + padding: 4px; + padding-bottom: 5px; + margin-right: 1px; + background-color: var(--jp-layout-color2); +} + +.lm-TabBar .lm-TabBar-addButton:hover { + background-color: var(--jp-layout-color1); +} + +.lm-DockPanel-tabBar .lm-TabBar-tab { + width: var(--jp-private-horizontal-tab-width); +} + +.lm-DockPanel-tabBar .lm-TabBar-content { + flex: unset; +} + +.lm-DockPanel-tabBar[data-orientation='horizontal'] { + flex: 1 1 auto; +} + +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ + /** * Support for icons as inline SVG HTMLElements */ @@ -9894,15 +9967,40 @@ stroke: var(--jp-icon-contrast-color3); } -/* CSS for icons in selected items in the settings editor */ -#setting-editor .jp-PluginList .jp-mod-selected .jp-icon-selectable[fill] { - fill: #fff; +.jp-jupyter-icon-color[fill] { + fill: var(--jp-jupyter-icon-color, var(--jp-warn-color0)); } -#setting-editor - .jp-PluginList - .jp-mod-selected - .jp-icon-selectable-inverse[fill] { - fill: var(--jp-brand-color1); + +.jp-notebook-icon-color[fill] { + fill: var(--jp-notebook-icon-color, var(--jp-warn-color0)); +} + +.jp-json-icon-color[fill] { + fill: var(--jp-json-icon-color, var(--jp-warn-color1)); +} + +.jp-console-icon-color[fill] { + fill: var(--jp-console-icon-color, white); +} + +.jp-console-icon-background-color[fill] { + fill: var(--jp-console-icon-background-color, var(--jp-brand-color1)); +} + +.jp-terminal-icon-color[fill] { + fill: var(--jp-terminal-icon-color, var(--jp-layout-color2)); +} + +.jp-terminal-icon-background-color[fill] { + fill: var(--jp-terminal-icon-background-color, var(--jp-inverse-layout2)); +} + +.jp-text-editor-icon-color[fill] { + fill: var(--jp-text-editor-icon-color, var(--jp-inverse-layout3)); +} + +.jp-inspector-icon-color[fill] { + fill: var(--jp-inspector-icon-color, var(--jp-inverse-layout3)); } /* CSS for icons in selected filebrowser listing items */ @@ -10085,7 +10183,7 @@ | Distributed under the terms of the Modified BSD License. |----------------------------------------------------------------------------*/ -.jp-icon-hoverShow:not(:hover) svg { +.jp-icon-hoverShow:not(:hover) .jp-icon-hoverShow-content { display: none !important; } @@ -10269,7 +10367,7 @@ .jp-switch-track { cursor: pointer; - background-color: var(--jp-border-color1); + background-color: var(--jp-switch-color, var(--jp-border-color1)); -webkit-transition: 0.4s; transition: 0.4s; border-radius: 34px; @@ -10292,7 +10390,7 @@ } .jp-switch[aria-checked='true'] .jp-switch-track { - background-color: var(--jp-warn-color0); + background-color: var(--jp-switch-true-position-color, var(--jp-warn-color0)); } .jp-switch[aria-checked='true'] .jp-switch-track::before { @@ -10742,8 +10840,7 @@ margin-left: auto; margin-right: auto; background: var(--jp-layout-color1); - padding: 24px; - padding-bottom: 12px; + padding: 24px 24px 12px 24px; min-width: 300px; min-height: 150px; max-width: 1000px; @@ -10759,6 +10856,10 @@ resize: both; } +.jp-Dialog-content.jp-Dialog-content-small { + max-width: 500px; +} + .jp-Dialog-button { overflow: visible; } @@ -10773,6 +10874,25 @@ border: 0; } +button.jp-Dialog-button.jp-mod-styled.jp-mod-accept:focus, +button.jp-Dialog-button.jp-mod-styled.jp-mod-warn:focus, +button.jp-Dialog-button.jp-mod-styled.jp-mod-reject:focus { + outline-offset: 4px; + -moz-outline-radius: 0px; +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-accept:focus { + outline: 1px solid var(--md-blue-700); +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-warn:focus { + outline: 1px solid var(--md-red-600); +} + +button.jp-Dialog-button.jp-mod-styled.jp-mod-reject:focus { + outline: 1px solid var(--md-grey-700); +} + button.jp-Dialog-close-button { padding: 0; height: 100%; @@ -10803,12 +10923,26 @@ display: flex; flex-direction: row; justify-content: flex-end; + align-items: center; flex: 0 0 auto; margin-left: -12px; margin-right: -12px; padding: 12px; } +.jp-Dialog-checkbox { + padding-right: 5px; +} + +.jp-Dialog-checkbox > input:focus-visible { + outline: 1px solid var(--jp-input-active-border-color); + outline-offset: 1px; +} + +.jp-Dialog-spacer { + flex: 1 1 auto; +} + .jp-Dialog-title { overflow: hidden; white-space: nowrap; @@ -10900,6 +11034,26 @@ outline: none; } +.jp-MainAreaWidget .jp-MainAreaWidget-error { + padding: 6px; +} + +.jp-MainAreaWidget .jp-MainAreaWidget-error > pre { + width: auto; + padding: 10px; + background: var(--jp-error-color3); + border: var(--jp-border-width) solid var(--jp-error-color1); + border-radius: var(--jp-border-radius); + color: var(--jp-ui-font-color1); + font-size: var(--jp-ui-font-size1); + white-space: pre-wrap; + word-wrap: break-word; +} + +.jp-MainAreaWidget { + contain: strict; +} + /** * google-material-color v1.2.6 * https://github.com/danlevan/google-material-color @@ -11383,8 +11537,8 @@ background: var(--jp-toolbar-background); min-height: var(--jp-toolbar-micro-height); padding: 2px; - z-index: 1; - overflow-x: auto; + z-index: 8; + overflow-x: hidden; } /* Toolbar items */ @@ -11492,8 +11646,8 @@ | The full license is in the file LICENSE, distributed with this software. |----------------------------------------------------------------------------*/ - -/* */ body.p-mod-override-cursor *, /* */ +/* */ +body.p-mod-override-cursor *, /* */ body.lm-mod-override-cursor * { cursor: inherit !important; } @@ -11553,31 +11707,105 @@ box-shadow: var(--jp-input-box-shadow); } -/* BASICS */ +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ -.CodeMirror { - /* Set height, width, borders, and global font properties here */ - font-family: monospace; - height: 300px; - color: black; - direction: ltr; -} +/*----------------------------------------------------------------------------- +| Variables +|----------------------------------------------------------------------------*/ -/* PADDING */ +/*----------------------------------------------------------------------------- -.CodeMirror-lines { - padding: 4px 0; /* Vertical padding around content */ -} -.CodeMirror pre.CodeMirror-line, -.CodeMirror pre.CodeMirror-line-like { - padding: 0 4px; /* Horizontal padding of content */ -} +/*----------------------------------------------------------------------------- +| Styles +|----------------------------------------------------------------------------*/ -.CodeMirror-scrollbar-filler, .CodeMirror-gutter-filler { - background-color: white; /* The little square between H and V scrollbars */ +.jp-Statusbar-ProgressCircle svg { + display: block; + margin: 0 auto; + width: 16px; + height: 24px; + align-self: normal; +} +.jp-Statusbar-ProgressCircle path { + fill: var(--jp-inverse-layout-color3); } -/* GUTTER */ +.jp-Statusbar-ProgressBar-progress-bar { + height: 10px; + width: 100px; + border: solid 0.25px var(--jp-brand-color2); + border-radius: 3px; + overflow: hidden; + align-self: center; +} +.jp-Statusbar-ProgressBar-progress-bar > div { + background-color: var(--jp-brand-color2); + background-image: linear-gradient( + -45deg, + rgba(255, 255, 255, 0.2) 25%, + transparent 25%, + transparent 50%, + rgba(255, 255, 255, 0.2) 50%, + rgba(255, 255, 255, 0.2) 75%, + transparent 75%, + transparent + ); + background-size: 40px 40px; + float: left; + width: 0%; + height: 100%; + font-size: 12px; + line-height: 14px; + color: #ffffff; + text-align: center; + animation: jp-Statusbar-ExecutionTime-progress-bar 2s linear infinite; +} + +.jp-Statusbar-ProgressBar-progress-bar p { + color: var(--jp-ui-font-color1); + font-family: var(--jp-ui-font-family); + font-size: var(--jp-ui-font-size1); + line-height: 10px; + width: 100px; +} + +@keyframes jp-Statusbar-ExecutionTime-progress-bar { + 0% { + background-position: 0 0; + } + 100% { + background-position: 40px 40px; + } +} + +/* BASICS */ + +.CodeMirror { + /* Set height, width, borders, and global font properties here */ + font-family: monospace; + height: 300px; + color: black; + direction: ltr; +} + +/* PADDING */ + +.CodeMirror-lines { + padding: 4px 0; /* Vertical padding around content */ +} +.CodeMirror pre.CodeMirror-line, +.CodeMirror pre.CodeMirror-line-like { + padding: 0 4px; /* Horizontal padding of content */ +} + +.CodeMirror-scrollbar-filler, .CodeMirror-gutter-filler { + background-color: white; /* The little square between H and V scrollbars */ +} + +/* GUTTER */ .CodeMirror-gutters { border-right: 1px solid #ddd; @@ -11977,6 +12205,10 @@ padding: 0 var(--jp-code-padding); } +.CodeMirror.cm-fat-cursor .cm-overlay.cm-searching { + opacity: 0.5; +} + .jp-CodeMirrorEditor[data-type='inline'] .CodeMirror-dialog { background-color: var(--jp-layout-color0); color: var(--jp-content-font-color1); @@ -12047,6 +12279,12 @@ color: var(--jp-search-unselected-match-color) !important; } +.cm-trailingspace { + background-image: url(); + background-position: center left; + background-repeat: repeat-x; +} + .CodeMirror-focused .CodeMirror-selected { background-color: var(--jp-editor-selected-focused-background); } @@ -12220,7 +12458,10 @@ opacity: 0; } -.jp-CodeMirrorEditor .remote-caret:hover > div { +/* Use `div[style]` as more specific selector on 3.4.x to reduce the impact of + * Chromium style invalidation strategy on performance when many divs are present. + */ +.jp-CodeMirrorEditor .remote-caret:hover > div[style] { opacity: 1; transition-delay: 0s; } @@ -12607,7 +12848,7 @@ border-spacing: 0; border: none; color: var(--jp-ui-font-color1); - font-size: 12px; + font-size: var(--jp-ui-font-size1); table-layout: fixed; margin-left: auto; margin-right: auto; @@ -12796,7 +13037,7 @@ border-radius: 3px; box-shadow: inset 0 -1px 0 rgba(0, 0, 0, 0.25); display: inline-block; - font-size: 0.8em; + font-size: var(--jp-ui-font-size0); line-height: 1em; padding: 0.2em 0.5em; } @@ -12851,8 +13092,10 @@ .jp-FileBrowser-toolbar.jp-Toolbar { border-bottom: none; height: auto; - margin: var(--jp-toolbar-header-margin); + margin: 8px 12px 0px 12px; + padding: 0px; box-shadow: none; + justify-content: flex-start; } .jp-BreadCrumbs { @@ -12884,43 +13127,33 @@ | Buttons |----------------------------------------------------------------------------*/ -.jp-FileBrowser-toolbar.jp-Toolbar { - padding: 0px; - margin: 8px 12px 0px 12px; -} - -.jp-FileBrowser-toolbar.jp-Toolbar { - justify-content: flex-start; -} - -.jp-FileBrowser-toolbar.jp-Toolbar .jp-Toolbar-item { +.jp-FileBrowser-toolbar > .jp-Toolbar-item { flex: 0 0 auto; padding-left: 0px; padding-right: 2px; } -.jp-FileBrowser-toolbar.jp-Toolbar .jp-ToolbarButtonComponent { +.jp-FileBrowser-toolbar > .jp-Toolbar-item .jp-ToolbarButtonComponent { width: 40px; } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent { +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher'] { width: 72px; background: var(--jp-brand-color1); } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent:focus-visible { - background-color: var(--jp-brand-color0); +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher']:hover, +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher']:focus-visible { + background-color: var(--jp-brand-color0) !important; } -.jp-FileBrowser-toolbar.jp-Toolbar - .jp-Toolbar-item:first-child - .jp-ToolbarButtonComponent +.jp-FileBrowser-toolbar + .jp-ToolbarButtonComponent[data-command='filebrowser:create-main-launcher'] .jp-icon3 { - fill: white; + fill: var(--jp-layout-color1); } /*----------------------------------------------------------------------------- @@ -12957,7 +13190,8 @@ } .jp-DirListing:focus-visible { - border: 1px solid var(--jp-brand-color1); + outline: 1px solid var(--jp-brand-color1); + outline-offset: -2px; } .jp-DirListing-header { @@ -13088,6 +13322,8 @@ flex: 1 0 64px; outline: none; border: none; + color: var(--jp-ui-font-color1); + background-color: var(--jp-layout-color1); } .jp-DirListing-item.jp-mod-running .jp-DirListing-itemIcon:before { @@ -13116,15 +13352,6 @@ transform: translateX(-40%) translateY(-58%); } -.jp-DirListing-deadSpace { - flex: 1 1 auto; - margin: 0; - padding: 0; - list-style-type: none; - overflow: auto; - background-color: var(--jp-layout-color1); -} - .jp-Document { min-width: 120px; min-height: 120px; @@ -13284,6 +13511,12 @@ margin: 0; } +.jp-TrimmedOutputs a { + margin: 10px; + text-decoration: none; + cursor: pointer; +} + /* Hide the gutter in case of * - nested output areas (e.g. in the case of output widgets) * - mirrored output areas @@ -13292,6 +13525,12 @@ display: none; } +/* Hide empty lines in the output area, for instance due to cleared widgets */ +.jp-OutputArea-prompt:empty { + padding: 0; + border: 0; +} + /*----------------------------------------------------------------------------- | executeResult is added to any Output-result for the display of the object | returned by a cell @@ -13314,12 +13553,6 @@ | The Stdin output |----------------------------------------------------------------------------*/ -.jp-OutputArea-stdin { - line-height: var(--jp-code-line-height); - padding-top: var(--jp-code-padding); - display: flex; -} - .jp-Stdin-prompt { color: var(--jp-content-font-color0); padding-right: var(--jp-code-padding); @@ -13342,10 +13575,18 @@ flex: 0 0 70%; } +.jp-Stdin-input::placeholder { + opacity: 0; +} + .jp-Stdin-input:focus { box-shadow: none; } +.jp-Stdin-input:focus::placeholder { + opacity: 1; +} + /*----------------------------------------------------------------------------- | Output Area View |----------------------------------------------------------------------------*/ @@ -13582,11 +13823,28 @@ .jp-CodeCell.jp-mod-outputsScrolled .jp-Cell-outputArea { overflow-y: auto; - max-height: 200px; - box-shadow: inset 0 0 6px 2px rgba(0, 0, 0, 0.3); + max-height: 24em; margin-left: var(--jp-private-cell-scrolling-output-offset); } +.jp-CodeCell.jp-mod-outputsScrolled .jp-Cell-outputArea::after { + content: ' '; + box-shadow: inset 0 0 6px 2px rgb(0 0 0 / 30%); + width: 100%; + height: 100%; + position: sticky; + bottom: 0; + top: 0; + margin-top: -50%; + float: left; + display: block; + pointer-events: none; +} + +.jp-CodeCell.jp-mod-outputsScrolled .jp-OutputArea-child { + padding-top: 6px; +} + .jp-CodeCell.jp-mod-outputsScrolled .jp-OutputArea-prompt { flex: 0 0 calc( @@ -13614,6 +13872,79 @@ overflow: auto; } +/* collapseHeadingButton (show always if hiddenCellsButton is _not_ shown) */ +.jp-collapseHeadingButton { + display: none; + min-height: var(--jp-cell-collapser-min-height); + font-size: var(--jp-code-font-size); + position: absolute; + right: 0; + top: 0; + bottom: 0; + background-color: transparent; + background-size: 25px; + background-repeat: no-repeat; + background-position-x: center; + background-position-y: top; + background-image: var(--jp-icon-caret-down); + border: none; + cursor: pointer; +} + +.jp-collapseHeadingButton:hover { + background-color: var(--jp-layout-color2); +} + +.jp-collapseHeadingButton.jp-mod-collapsed { + background-image: var(--jp-icon-caret-right); +} + +:is(.jp-MarkdownCell:hover, .jp-mod-active) .jp-collapseHeadingButton { + display: flex; +} + +/* + set the container font size to match that of content + so that the nested collapse buttons have the right size +*/ +.jp-MarkdownCell .jp-InputPrompt { + font-size: var(--jp-content-font-size1); +} + +/* + Align collapseHeadingButton with cell top header + The font sizes are identical to the ones in packages/rendermime/style/base.css +*/ +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='1'] { + font-size: var(--jp-content-font-size5); + background-position-y: calc(0.3 * var(--jp-content-font-size5)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='2'] { + font-size: var(--jp-content-font-size4); + background-position-y: calc(0.3 * var(--jp-content-font-size4)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='3'] { + font-size: var(--jp-content-font-size3); + background-position-y: calc(0.3 * var(--jp-content-font-size3)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='4'] { + font-size: var(--jp-content-font-size2); + background-position-y: calc(0.3 * var(--jp-content-font-size2)); +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='5'] { + font-size: var(--jp-content-font-size1); + background-position-y: top; +} + +.jp-mod-rendered .jp-collapseHeadingButton[data-heading-level='6'] { + font-size: var(--jp-content-font-size0); + background-position-y: top; +} + .jp-showHiddenCellsButton { margin-left: calc(var(--jp-cell-prompt-width) + 2 * var(--jp-code-padding)); margin-top: var(--jp-code-padding); @@ -13626,19 +13957,6 @@ background-color: var(--jp-border-color2) !important; } -.jp-collapseHeadingButton { - display: none; -} - -.jp-MarkdownCell:hover .jp-collapseHeadingButton { - display: flex; - min-height: var(--jp-cell-collapser-min-height); - position: absolute; - right: 0; - top: 0; - bottom: 0; -} - /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Distributed under the terms of the Modified BSD License. @@ -13653,6 +13971,10 @@ | Variables |----------------------------------------------------------------------------*/ +:root { + --jp-notebook-toolbar-padding: 2px 5px 2px 2px; +} + /*----------------------------------------------------------------------------- /*----------------------------------------------------------------------------- @@ -13660,7 +13982,7 @@ |----------------------------------------------------------------------------*/ .jp-NotebookPanel-toolbar { - padding: 2px; + padding: var(--jp-notebook-toolbar-padding); } .jp-Toolbar-item.jp-Notebook-toolbarCellType .jp-select-wrapper.jp-mod-focused { @@ -13680,6 +14002,93 @@ top: 5px !important; } +.jp-Toolbar-responsive-popup { + position: absolute; + height: fit-content; + display: flex; + flex-direction: row; + flex-wrap: wrap; + justify-content: flex-end; + border-bottom: var(--jp-border-width) solid var(--jp-toolbar-border-color); + box-shadow: var(--jp-toolbar-box-shadow); + background: var(--jp-toolbar-background); + min-height: var(--jp-toolbar-micro-height); + padding: var(--jp-notebook-toolbar-padding); + z-index: 1; + right: 0px; + top: 0px; +} + +.jp-Toolbar > .jp-Toolbar-responsive-opener { + margin-left: auto; +} + +/*----------------------------------------------------------------------------- +| Copyright (c) Jupyter Development Team. +| Distributed under the terms of the Modified BSD License. +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- +| Variables +|----------------------------------------------------------------------------*/ + +/*----------------------------------------------------------------------------- + +/*----------------------------------------------------------------------------- +| Styles +|----------------------------------------------------------------------------*/ + +.jp-Notebook-ExecutionIndicator { + position: relative; + display: inline-block; + height: 100%; + z-index: 9997; +} + +.jp-Notebook-ExecutionIndicator-tooltip { + visibility: hidden; + height: auto; + width: max-content; + width: -moz-max-content; + background-color: var(--jp-layout-color2); + color: var(--jp-ui-font-color1); + text-align: justify; + border-radius: 6px; + padding: 0 5px; + position: fixed; + display: table; +} + +.jp-Notebook-ExecutionIndicator-tooltip.up { + transform: translateX(-50%) translateY(-100%) translateY(-32px); +} + +.jp-Notebook-ExecutionIndicator-tooltip.down { + transform: translateX(calc(-100% + 16px)) translateY(5px); +} + +.jp-Notebook-ExecutionIndicator-tooltip.hidden { + display: none; +} + +.jp-Notebook-ExecutionIndicator:hover .jp-Notebook-ExecutionIndicator-tooltip { + visibility: visible; +} + +.jp-Notebook-ExecutionIndicator span { + font-size: var(--jp-ui-font-size1); + font-family: var(--jp-ui-font-family); + color: var(--jp-ui-font-color1); + line-height: 24px; + display: block; +} + +.jp-Notebook-ExecutionIndicator-progress-bar { + display: flex; + justify-content: center; + height: 100%; +} + /*----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Distributed under the terms of the Modified BSD License. @@ -13731,10 +14140,6 @@ contain: strict; } -.jp-Notebook-render * { - contain: none !important; -} - .jp-Notebook .jp-Cell { overflow: visible; } @@ -13974,6 +14379,86 @@ flex: 0 0 110px; } +/*----------------------------------------------------------------------------- +| Side-by-side Mode (.jp-mod-sideBySide) +|----------------------------------------------------------------------------*/ +:root { + --jp-side-by-side-output-size: 1fr; + --jp-side-by-side-resized-cell: var(--jp-side-by-side-output-size); +} + +.jp-mod-sideBySide.jp-Notebook .jp-Notebook-cell { + margin-top: 3em; + margin-bottom: 3em; + margin-left: 5%; + margin-right: 5%; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell { + display: grid; + grid-template-columns: minmax(0, 1fr) min-content minmax( + 0, + var(--jp-side-by-side-output-size) + ); + grid-template-rows: auto minmax(0, 1fr) auto; + grid-template-areas: + 'header header header' + 'input handle output' + 'footer footer footer'; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell.jp-mod-resizedCell { + grid-template-columns: minmax(0, 1fr) min-content minmax( + 0, + var(--jp-side-by-side-resized-cell) + ); +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellHeader { + grid-area: header; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-Cell-inputWrapper { + grid-area: input; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-Cell-outputWrapper { + /* overwrite the default margin (no vertical separation needed in side by side move */ + margin-top: 0; + grid-area: output; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellFooter { + grid-area: footer; +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellResizeHandle { + grid-area: handle; + user-select: none; + display: block; + height: 100%; + cursor: ew-resize; + padding: 0 var(--jp-cell-padding); +} + +.jp-mod-sideBySide.jp-Notebook .jp-CodeCell .jp-CellResizeHandle::after { + content: ''; + display: block; + background: var(--jp-border-color2); + height: 100%; + width: 5px; +} + +.jp-mod-sideBySide.jp-Notebook + .jp-CodeCell.jp-mod-resizedCell + .jp-CellResizeHandle::after { + background: var(--jp-border-color0); +} + +.jp-CellResizeHandle { + display: none; +} + /*----------------------------------------------------------------------------- | Placeholder |----------------------------------------------------------------------------*/ @@ -14153,6 +14638,7 @@ --jp-border-color1: var(--md-grey-400); --jp-border-color2: var(--md-grey-300); --jp-border-color3: var(--md-grey-200); + --jp-inverse-border-color: var(--md-grey-600); --jp-border-radius: 2px; /* UI Fonts @@ -14389,7 +14875,7 @@ --jp-input-active-background: var(--jp-layout-color1); --jp-input-hover-background: var(--jp-layout-color1); --jp-input-background: var(--md-grey-100); - --jp-input-border-color: var(--jp-border-color1); + --jp-input-border-color: var(--jp-inverse-border-color); --jp-input-active-border-color: var(--jp-brand-color1); --jp-input-active-box-shadow-color: rgba(19, 124, 189, 0.3); @@ -14426,6 +14912,20 @@ --jp-mirror-editor-error-color: #f00; --jp-mirror-editor-hr-color: #999; + /* + RTC user specific colors. + These colors are used for the cursor, username in the editor, + and the icon of the user. + */ + + --jp-collaborator-color1: #ffad8e; + --jp-collaborator-color2: #dac83d; + --jp-collaborator-color3: #72dd76; + --jp-collaborator-color4: #00e4d0; + --jp-collaborator-color5: #45d4ff; + --jp-collaborator-color6: #e2b1ff; + --jp-collaborator-color7: #ff9de6; + /* Vega extension styles */ --jp-vega-background: white; @@ -14451,6 +14951,19 @@ --jp-icon-contrast-color1: var(--md-green-600); --jp-icon-contrast-color2: var(--md-pink-600); --jp-icon-contrast-color3: var(--md-blue-600); + + /* File or activity icons and switch semantic variables */ + --jp-jupyter-icon-color: #f37626; + --jp-notebook-icon-color: #f37626; + --jp-json-icon-color: var(--md-orange-700); + --jp-console-icon-background-color: var(--md-blue-700); + --jp-console-icon-color: white; + --jp-terminal-icon-background-color: var(--md-grey-800); + --jp-terminal-icon-color: var(--md-grey-200); + --jp-text-editor-icon-color: var(--md-grey-700); + --jp-inspector-icon-color: var(--md-grey-700); + --jp-switch-color: var(--md-grey-400); + --jp-switch-true-position-color: var(--md-orange-900); } @@ -14621,7 +15134,7 @@

    USD and MaterialX NodeGraphs @@ -14699,9 +15212,9 @@

    1. Usd Setup&# @@ -14765,7 +15278,7 @@

    1. Usd Setup&#
    Flattened Usd File